0
|
1 /* system_config.cc
|
|
2 * gameexe.ini ファイルの読み込み
|
|
3 */
|
|
4
|
|
5 /*
|
|
6 *
|
|
7 * Copyright (C) 2000- Kazunori Ueno(JAGARL) <jagarl@creator.club.ne.jp>
|
|
8 *
|
|
9 * This program is free software; you can redistribute it and/or modify
|
|
10 * it under the terms of the GNU General Public License as published by
|
|
11 * the Free Software Foundation; either version 2 of the License, or
|
|
12 * (at your option) any later version.
|
|
13 *
|
|
14 * This program is distributed in the hope that it will be useful,
|
|
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
17 * GNU General Public License for more details.
|
|
18 *
|
|
19 * You should have received a copy of the GNU General Public License
|
|
20 * along with this program; if not, write to the Free Software
|
|
21 * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
22 *
|
|
23 */
|
|
24
|
|
25 #include <stdio.h>
|
|
26 #include <string.h>
|
|
27 #include <stdlib.h>
|
|
28 #include <stdarg.h>
|
|
29 #include <ctype.h>
|
|
30 #include <map>
|
|
31 #include <string>
|
|
32 #include "system_config.h"
|
|
33 #include "../system/file.h"
|
|
34
|
|
35 using namespace std;
|
|
36
|
|
37 // #define DEBUG_CONFIG
|
|
38 #ifdef DEBUG_CONFIG
|
|
39 # define dprintf(X) printf X
|
|
40 #else
|
|
41 # define dprintf(X)
|
|
42 #endif /* DEBUG_CONFIG */
|
|
43
|
|
44 #define MAXTOKEN 10 /* = で区切られた領域の最大数 */
|
|
45 #define MAXVARS 32 /* , で区切られた数値の最大数 */
|
|
46
|
|
47 // 初期化ファイルの読み込み
|
|
48 /* config は 文字列、数列、その複合など、いろいろな形式がありうる */
|
|
49 /* 文字列と数列は一般に AyuSys_Config クラスに含める */
|
|
50
|
|
51
|
|
52 /**********************************************************/
|
|
53 /* とりあえずハッシュ比較付き文字列 */
|
|
54 class HashStr {
|
|
55 const char* str;
|
|
56 unsigned int hash;
|
|
57 public:
|
|
58 HashStr(const char*);
|
|
59 HashStr(const HashStr& orig);
|
|
60 ~HashStr() {
|
|
61 if (str) delete[] str;
|
|
62 }
|
|
63 const char* c_str(void) const { return str; }
|
|
64 friend inline int operator<(const HashStr& a, const HashStr& b) {
|
|
65 if (a.hash == b.hash) {
|
|
66 if (a.str == 0) return 1;
|
|
67 else if (b.str == 0) return 0;
|
|
68 else return strcmp(a.str, b.str);
|
|
69 }
|
|
70 else return a.hash < b.hash;
|
|
71 }
|
|
72 };
|
|
73 HashStr::HashStr(const char* s ) {
|
|
74 if (s == 0 || s[0] == '\0') {
|
|
75 str = 0; hash = 0; return; /* invalid string */
|
|
76 }
|
|
77 char* new_str = new char[strlen(s)+1];
|
|
78 strcpy(new_str, s);
|
|
79 str = new_str;
|
|
80 /* calc hash... 適当 */
|
|
81 int h = strlen(s);
|
|
82 while(*s != 0) {
|
|
83 h = *s + ((h * (0x9449+*s))>>7);
|
|
84 s++;
|
|
85 }
|
|
86 hash = (unsigned int)h;
|
|
87 }
|
|
88 HashStr::HashStr(const HashStr& orig) {
|
|
89 if (orig.str == 0 || orig.str[0] == '\0') {
|
|
90 str = 0; hash = 0; return; /* invalid */
|
|
91 }
|
|
92 char* new_str = new char[strlen(orig.str)+1];
|
|
93 strcpy(new_str, orig.str);
|
|
94 str = new_str;
|
|
95 hash = orig.hash;
|
|
96 }
|
|
97
|
|
98 /**********************************************************
|
|
99 **AyuSys_Config_[String | Intlist] :
|
|
100 ** 設定の本体
|
|
101 ** original : 元設定
|
|
102 ** old_data : 前回 ClearDiff() したときの設定
|
|
103 ** new_data : ClearDiff() 以降に設定した内容を保存
|
|
104 ** データ設定:
|
|
105 ** Init() : 元設定を作成
|
|
106 ** Set() : 設定を変更
|
|
107 ** Get() : 最も新しい設定を得る
|
|
108 **
|
|
109 ** 変更の記録:
|
|
110 ** Diff() : 前回のClearDiff() から変更した内容を得る
|
|
111 ** DiffLen() : Diff() で必要な文字列長を得る
|
|
112 ** ClearDiff() : 変更記録を消す
|
|
113 ** PatchOld() : Diff() で得た記録に基づき、変更前の状態に戻す
|
|
114 ** PatchNew() : Diff() で得た記録に基づき、変更後の状態に戻す
|
|
115 **
|
|
116 ** 元設定からの変更の記録:
|
|
117 ** SetOriginal() : 元設定に戻す
|
|
118 ** DiffOriginal() : 元設定から現在の設定の変更を得る
|
|
119 ** DiffOriginalLen() : DiffOriginal() で必要な文字列長を得る
|
|
120 ** PatchOriginal() : DiffOriginal() で得た記録に基づき、設定を復旧する
|
|
121 */
|
|
122
|
|
123 /************************************************
|
|
124 ** AyuSysConfigStringItem
|
|
125 ** 文字列をデータとしてもつ設定項目
|
|
126 */
|
|
127 class AyuSysConfigStringItem {
|
|
128 char* original_data;
|
|
129 char* old_data;
|
|
130 char* new_data;
|
|
131 public:
|
|
132 AyuSysConfigStringItem(void) {
|
|
133 original_data = 0;
|
|
134 old_data = 0;
|
|
135 new_data = 0;
|
|
136 }
|
|
137 AyuSysConfigStringItem(const AyuSysConfigStringItem& o) {
|
|
138 original_data = 0; old_data = 0; new_data = 0;
|
|
139 if (o.original_data) {
|
|
140 original_data = new char[strlen(o.original_data)+1];
|
|
141 strcpy(original_data, o.original_data);
|
|
142 }
|
|
143 if (o.old_data) {
|
|
144 old_data = new char[strlen(o.old_data)+1];
|
|
145 strcpy(old_data, o.old_data);
|
|
146 }
|
|
147 if (o.new_data) {
|
|
148 new_data = new char[strlen(o.new_data)+1];
|
|
149 strcpy(new_data, o.new_data);
|
|
150 }
|
|
151 }
|
|
152 /* 設定:Init で初期化、Set で変更、Get で変更を優先して取り出す */
|
|
153 void Init(int deal, const char* str) { /* deal は無視 */
|
|
154 if (original_data) delete[] original_data;
|
|
155 int len = strlen(str);
|
|
156 original_data = new char[len+1];
|
|
157 strcpy(original_data, str);
|
|
158 original_data[len] = '\0';
|
|
159 }
|
|
160 void Set(int deal, const char* str) { /* deal は無視 */
|
|
161 if (new_data) delete[] new_data;
|
|
162 int len = strlen(str);
|
|
163 new_data = new char[len+1];
|
|
164 strcpy(new_data, str);
|
|
165 new_data[len] = '\0';
|
|
166 }
|
|
167 const char* Get(int deal) const {/* deal は無視 */
|
|
168 if (new_data) return new_data;
|
|
169 else if (old_data) return old_data;
|
|
170 return original_data;
|
|
171 }
|
|
172 const char* GetOriginal(int deal) const {
|
|
173 return original_data;
|
|
174 }
|
|
175 int Deal(void) const {
|
|
176 return 1;
|
|
177 }
|
|
178 /* オリジナルからの変化の調査 :
|
|
179 ** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
|
|
180 ** 変化を反映
|
|
181 */
|
|
182 int DiffOriginalLen(void) {
|
|
183 if (new_data == 0) return 0;
|
|
184 return strlen(new_data)+1;
|
|
185 }
|
|
186 void DiffOriginal(string& data) {
|
|
187 if (new_data == 0) { /* あり得ない */
|
|
188 fprintf(stderr,"AyuSysConfigStringItem::DiffOriginal : this method must not called if not required!\n");
|
|
189 return;
|
|
190 }
|
|
191 char* out_data = new char[strlen(new_data)*2+1];
|
|
192 char* buf = out_data;
|
|
193 int i;
|
|
194 for (i=0; new_data[i]!=0; i++) {
|
|
195 switch(new_data[i]) {
|
|
196 case '?': *buf++ = '?'; *buf++ = '0'; break;
|
|
197 case '"': *buf++ = '?'; *buf++ = '1'; break;
|
|
198 case '\'': *buf++ = '?'; *buf++ = '2'; break;
|
|
199 case ',': *buf++ = '?'; *buf++ = '3'; break;
|
|
200 case '.': *buf++ = '?'; *buf++ = '4'; break;
|
|
201 case ':': *buf++ = '?'; *buf++ = '5'; break;
|
|
202 case ';': *buf++ = '?'; *buf++ = '6'; break;
|
|
203 case '=': *buf++ = '?'; *buf++ = '7'; break;
|
|
204 case '<': *buf++ = '?'; *buf++ = '8'; break;
|
|
205 case '>': *buf++ = '?'; *buf++ = '9'; break;
|
|
206 default: *buf++ = new_data[i]; break;
|
|
207 }
|
|
208 }
|
|
209 *buf++ = 0;
|
|
210 data += out_data;
|
|
211 delete[] out_data;
|
|
212 return;
|
|
213 }
|
|
214 const char* PatchOriginal(const char* data) {
|
|
215 static char* table = "?\"',.:;=<>";
|
|
216 if (new_data) delete[] new_data;
|
|
217 if (old_data) delete[] old_data;
|
|
218 new_data = 0; old_data = 0;
|
|
219 new_data = new char[1024];
|
|
220 int i,j = 0;
|
|
221 for (i=0; i<1020; i++) {
|
|
222 switch(data[j]) {
|
|
223 case '?':
|
|
224 if (data[j+1] >= '0' && data[j+1] <= '9') {
|
|
225 new_data[i] = table[ data[j+1] - '0'];
|
|
226 j += 2;
|
|
227 break;
|
|
228 }
|
|
229 case '"': case '\'': case ',': case '.': case ':':
|
|
230 case ';': case '=': case '<': case '>':
|
|
231 goto for_end;
|
|
232 default: new_data[i] = data[j++]; break;
|
|
233 }
|
|
234 }
|
|
235 for_end:
|
|
236 new_data[i] = 0;
|
|
237 return data;
|
|
238 }
|
|
239 void SetOriginal(void) {
|
|
240 if (new_data) delete[] new_data;
|
|
241 if (old_data) delete[] old_data;
|
|
242 new_data = 0; old_data = 0;
|
|
243 }
|
|
244 void Dump(FILE* f) const {
|
|
245 if (original_data) fprintf(f, "original %s ",original_data);
|
|
246 if (old_data) fprintf(f, "old_data %s ",old_data);
|
|
247 if (new_data) fprintf(f, "new_data %s ",new_data);
|
|
248 fprintf(f, "\n");
|
|
249 }
|
|
250 };
|
|
251
|
|
252 /************************************************
|
|
253 ** AyuSysConfigIntlistItem
|
|
254 ** 数値列をデータとしてもつ設定項目
|
|
255 */
|
|
256 class AyuSysConfigIntlistItem {
|
|
257 int item_deal;
|
|
258 int* original_data;
|
|
259 int* old_data;
|
|
260 int* new_data;
|
|
261 public:
|
|
262 AyuSysConfigIntlistItem(void) {
|
|
263 item_deal = 0;
|
|
264 original_data = 0;
|
|
265 old_data = 0;
|
|
266 new_data = 0;
|
|
267 }
|
|
268 AyuSysConfigIntlistItem(const AyuSysConfigIntlistItem& o) {
|
|
269 item_deal = o.item_deal;
|
|
270 original_data = 0; old_data = 0; new_data = 0;
|
|
271 if (o.original_data) {
|
|
272 original_data = new int[item_deal];
|
|
273 memcpy(original_data, o.original_data, sizeof(int)*item_deal);
|
|
274 }
|
|
275 if (o.old_data) {
|
|
276 old_data = new int[item_deal];
|
|
277 memcpy(old_data, o.old_data, sizeof(int)*item_deal);
|
|
278 }
|
|
279 if (o.new_data) {
|
|
280 new_data = new int[item_deal];
|
|
281 memcpy(new_data, o.new_data, sizeof(int)*item_deal);
|
|
282 }
|
|
283 }
|
|
284 /* 設定:Init で初期化、Set で変更、Get で変更を優先して取り出す */
|
|
285 void Init(int deal, const int* list) { /* deal は無視 */
|
|
286 if (original_data) delete[] original_data;
|
|
287 original_data = 0;
|
|
288 if (deal <= 0) {
|
|
289 item_deal = 0; return;
|
|
290 }
|
|
291 item_deal = deal;
|
|
292 original_data = new int[item_deal];
|
|
293 memcpy(original_data, list, sizeof(int)*deal);
|
|
294 }
|
|
295 void Set(int deal, const int* list) { /* deal は無視 */
|
|
296 item_deal = deal;
|
|
297 if (new_data) delete[] new_data;
|
|
298 new_data = new int[item_deal];
|
|
299 memcpy(new_data, list, sizeof(int)*item_deal);
|
|
300 }
|
|
301 const int* Get(int deal) const {/* deal は無視 */
|
|
302 if (item_deal == 0) return 0;
|
|
303 if (deal > item_deal) {
|
|
304 fprintf(stderr,"AyuSysConfigIntlistItem::Get : invalid items deal %d (correct: %d)\n",deal,item_deal);
|
|
305 return 0;
|
|
306 }
|
|
307 if (new_data) return new_data;
|
|
308 else if (old_data) return old_data;
|
|
309 return original_data;
|
|
310 }
|
|
311 const int* GetOriginal(int deal) const {/* deal は無視 */
|
|
312 if (item_deal == 0) return 0;
|
|
313 if (deal > item_deal) {
|
|
314 fprintf(stderr,"AyuSysConfigIntlistItem::Get : invalid items deal %d (correct: %d)\n",deal,item_deal);
|
|
315 return 0;
|
|
316 }
|
|
317 return original_data;
|
|
318 }
|
|
319 int Deal(void) const {
|
|
320 return item_deal;
|
|
321 }
|
|
322 /* オリジナルからの変化の調査 :
|
|
323 ** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
|
|
324 ** 変化を反映
|
|
325 */
|
|
326 int DiffOriginalLen(void) {
|
|
327 if (new_data == 0) return 0;
|
|
328 return 12 * item_deal + 1;
|
|
329 }
|
|
330 void DiffOriginal(string& data) {
|
|
331 if (new_data == 0) { /* あり得ない */
|
|
332 fprintf(stderr,"AyuSysConfigStringItem::DiffOriginal : this method must not called if not required!\n");
|
|
333 return;
|
|
334 }
|
|
335 int i; char buf[1024];
|
|
336 for (i=0; i<item_deal; i++) {
|
|
337 sprintf(buf, "%d,",new_data[i]);
|
|
338 data += buf;
|
|
339 }
|
|
340 return;
|
|
341 }
|
|
342 const char* PatchOriginal(const char* data) {
|
|
343 if (old_data) delete[] old_data;
|
|
344 if (new_data) delete[] new_data;
|
|
345 old_data = 0; new_data = 0;
|
|
346 new_data = new int[item_deal];
|
|
347 int i;
|
|
348 for (i=0; i<item_deal; i++) {
|
|
349 new_data[i] = atoi(data);
|
|
350 if (strchr(data, ',') == 0) break;
|
|
351 data = strchr(data, ',') + 1;
|
|
352 }
|
|
353 return data;
|
|
354 }
|
|
355 void SetOriginal(void) {
|
|
356 if (new_data) delete[] new_data;
|
|
357 if (old_data) delete[] old_data;
|
|
358 new_data = 0; old_data = 0;
|
|
359 }
|
|
360 void Dump(FILE* f) const {
|
|
361 fprintf(f, "item deal %d, ",item_deal);
|
|
362 if (original_data) {
|
|
363 fprintf(f, "(%d", original_data[0]);
|
|
364 int i;for (i=1; i<item_deal; i++) {
|
|
365 fprintf(f, ",%d",original_data[i]);
|
|
366 }
|
|
367 fprintf(f, ") ");
|
|
368 }
|
|
369 if (old_data) {
|
|
370 fprintf(f, "old %08x(%d", (unsigned int)(old_data), old_data[0]);
|
|
371 int i;for (i=1; i<item_deal; i++) {
|
|
372 fprintf(f, ",%d",old_data[i]);
|
|
373 }
|
|
374 fprintf(f, ") ");
|
|
375 }
|
|
376 if (new_data) {
|
|
377 fprintf(f, "new %08x(%d", (unsigned int)(new_data), new_data[0]);
|
|
378 int i;for (i=1; i<item_deal; i++) {
|
|
379 fprintf(f, ",%d",new_data[i]);
|
|
380 }
|
|
381 fprintf(f, ") ");
|
|
382 }
|
|
383 fprintf(f, "\n");
|
|
384 }
|
|
385 };
|
|
386
|
|
387 // template map<HashStr, AyuSysConfigStringItem>;
|
|
388 // template map<HashStr, AyuSysConfigIntlistItem>;
|
|
389
|
|
390 /************************************************
|
|
391 ** AyuSysConfigItem
|
|
392 ** データ名 -> データ本体の map と、map 全体に
|
|
393 ** 様々な操作を行うためのメソッド
|
|
394 */
|
|
395
|
|
396 template<class ItemType, class DataType> class AyuSysConfigItem {
|
|
397 typedef map<HashStr,ItemType> maptype;
|
|
398 typedef typename maptype::iterator mapiterator;
|
|
399 typedef typename maptype::const_iterator const_mapiterator;
|
|
400 maptype data;
|
|
401 public:
|
|
402 void SetOrig(HashStr& name, int deal, const DataType* str) {
|
|
403 if (str == 0) return; /* 無効 */
|
|
404 data[name].Init(deal, str);
|
|
405 }
|
|
406 void Set(HashStr& name, int deal, const DataType* new_data) {
|
|
407 if (new_data == 0) return; /* 無効 */
|
|
408 /* 設定を検索 */
|
|
409 mapiterator it = data.find(name);
|
|
410 /* 設定が元設定に見つからないなら失敗 */
|
|
411 if (it == data.end()) {
|
|
412 fprintf(stderr,"AyuSysConfigItem::Set : there is no '%s' parameter\n",name.c_str());
|
|
413 return;
|
|
414 }
|
|
415 /* 設定を変更 */
|
|
416 it->second.Set(deal, new_data);
|
|
417 }
|
|
418 /* 新しい設定を優先して返す */
|
|
419 const DataType* Get(int deal, HashStr& name) const {
|
|
420 const_mapiterator it = data.find(name);
|
|
421 if (it == data.end()) return 0;
|
|
422 return it->second.Get(deal);
|
|
423 }
|
|
424 const DataType* GetOriginal(int deal, HashStr& name) const {
|
|
425 const_mapiterator it = data.find(name);
|
|
426 if (it == data.end()) return 0;
|
|
427 return it->second.GetOriginal(deal);
|
|
428 }
|
|
429 int Deal(HashStr& name) const {
|
|
430 const_mapiterator it = data.find(name);
|
|
431 if (it == data.end()) return 0;
|
|
432 return it->second.Deal();
|
|
433 }
|
|
434 /* オリジナルからの変化の調査 :
|
|
435 ** DiffOriginal で変化を文字列で取り出し、PatchOriginal で
|
|
436 ** 変化を反映
|
|
437 */
|
|
438 void DiffOriginal(string& ret_str) {
|
|
439 mapiterator it = data.begin();
|
|
440 for (; it != data.end(); it++) {
|
|
441 int len = it->second.DiffOriginalLen();
|
|
442 if (len) {
|
|
443 ret_str += it->first.c_str();
|
|
444 ret_str += "=";
|
|
445 it->second.DiffOriginal(ret_str);
|
|
446 ret_str += ";";
|
|
447 }
|
|
448 }
|
|
449 ret_str += ";";
|
|
450 return;
|
|
451 }
|
|
452 const char* PatchOriginal(const char* diff_data) {
|
|
453 while(*diff_data != ';') {
|
|
454 char name[1024];
|
|
455 const char* data_start = strchr(diff_data, '=');
|
|
456 if (data_start == 0) break;
|
|
457 strncpy(name, diff_data, data_start-diff_data);
|
|
458 name[data_start-diff_data] = 0;
|
|
459 data_start++;
|
|
460 mapiterator it = data.find(name);
|
|
461 if (it != data.end()) {
|
|
462 diff_data = data_start;
|
|
463 it->second.PatchOriginal(diff_data);
|
|
464 }
|
|
465 diff_data = strchr(diff_data, ';');
|
|
466 if (diff_data) diff_data++;
|
|
467 }
|
|
468 if (*diff_data == ';') {
|
|
469 diff_data++;
|
|
470 } else {
|
|
471 fprintf(stderr,"AyusysConfigItem::PatchOriginal: invalid data %s\n",diff_data);
|
|
472 }
|
|
473 return diff_data;
|
|
474 }
|
|
475 void SetOriginal(void) {
|
|
476 mapiterator it = data.begin();
|
|
477 for (; it != data.end(); it++) {
|
|
478 it->second.SetOriginal();
|
|
479 }
|
|
480 }
|
|
481 void Dump(FILE* f) const {
|
|
482 const_mapiterator it = data.begin();
|
|
483 for (; it != data.end(); it++) {
|
|
484 fprintf(f, "name %s: ",it->first.c_str());
|
|
485 it->second.Dump(f);
|
|
486 }
|
|
487 }
|
|
488 };
|
|
489 // template AyuSysConfigItem<AyuSysConfigStringItem, char>;
|
|
490 // template AyuSysConfigItem<AyuSysConfigIntlistItem, int>;
|
|
491
|
|
492 /************************************************/
|
|
493 /* ラッパ */
|
|
494 struct AyuSysConfigString {
|
|
495 AyuSysConfigItem<AyuSysConfigStringItem,char> orig;
|
|
496 void Dump(FILE* f) const {
|
|
497 fprintf(f, "string config:\n");
|
|
498 orig.Dump(f);
|
|
499 }
|
|
500 };
|
|
501 struct AyuSysConfigIntlist {
|
|
502 AyuSysConfigItem<AyuSysConfigIntlistItem, int> orig;
|
|
503 void Dump(FILE* f) const {
|
|
504 fprintf(f, "integer array config:\n");
|
|
505 orig.Dump(f);
|
|
506 }
|
|
507 };
|
|
508
|
|
509 /************************************************/
|
|
510 /* AyuSysConfig クラス */
|
|
511 int AyuSysConfig::SearchParam(const char* name) const{
|
|
512 HashStr str(name);
|
|
513 if (str_config->orig.Get(1, str)) return 1; /* char* のパラメータ */
|
|
514 else if (int_config->orig.Get(1, str)) return 2; /* int のパラメータ */
|
|
515 /* XXX.015.XXX の類のキー名を XXX.000.XXX の形に規格化して再検索 */
|
|
516 char name_copy[1024];
|
|
517 strncpy(name_copy, name, 1000);
|
|
518 name_copy[1000] = 0;
|
|
519 char* s;
|
|
520 for (s=name_copy; s != 0; s = strchr(s,'.')) {
|
|
521 if (isdigit(s[1]) && isdigit(s[2]) && isdigit(s[3])) {
|
|
522 s[1] = '0'; s[2] = '0'; s[3] = '0';
|
|
523 }
|
|
524 s++;
|
|
525 }
|
|
526 HashStr str2(name_copy);
|
|
527 if (str_config->orig.Get(1, str2)) return 1; /* char* のパラメータ */
|
|
528 else if (int_config->orig.Get(1, str2)) return 2; /* int のパラメータ */
|
|
529 else return 0;
|
|
530 }
|
|
531 const char* AyuSysConfig::GetParaStr(const char* name) const{
|
|
532 HashStr str(name);
|
|
533 const char* ret = str_config->orig.Get(1,str);
|
|
534 if (ret == 0) {
|
|
535 // fprintf(stderr,"Cannot find config name '%s'\n",name);
|
|
536 }
|
|
537 return ret;
|
|
538 }
|
|
539 int AyuSysConfig::GetParam(const char* name, int deal, ...) const{
|
|
540 HashStr str(name);
|
|
541 va_list va; int i;
|
|
542 const int* vars = int_config->orig.Get(deal, str);
|
|
543 if (vars == 0) {
|
|
544 // fprintf(stderr,"Cannot find config name '%s'\n",name);
|
|
545 va_start(va, deal);
|
|
546 for (i=0; i<deal; i++) {
|
|
547 int* var = va_arg(va, int*);
|
|
548 if (var) *var = 0;
|
|
549 }
|
|
550 return -1;
|
|
551 } else {
|
|
552 va_start(va, deal);
|
|
553 for (i=0; i<deal; i++) {
|
|
554 int* var = va_arg(va, int*);
|
|
555 if (var) *var = vars[i];
|
|
556 }
|
|
557 }
|
|
558 return 0;
|
|
559 }
|
|
560 int AyuSysConfig::GetOriginalParam(const char* name, int deal, ...) const{
|
|
561 HashStr str(name);
|
|
562 va_list va; int i;
|
|
563 const int* vars = int_config->orig.GetOriginal(deal, str);
|
|
564 if (vars == 0) {
|
|
565 // fprintf(stderr,"Cannot find config name '%s'\n",name);
|
|
566 va_start(va, deal);
|
|
567 for (i=0; i<deal; i++) {
|
|
568 int* var = va_arg(va, int*);
|
|
569 if (var) *var = 0;
|
|
570 }
|
|
571 return -1;
|
|
572 } else {
|
|
573 va_start(va, deal);
|
|
574 for (i=0; i<deal; i++) {
|
|
575 int* var = va_arg(va, int*);
|
|
576 if (var) *var = vars[i];
|
|
577 }
|
|
578 }
|
|
579 return 0;
|
|
580 }
|
|
581 const int* AyuSysConfig::GetParamArray(const char* name, int& deal) const{
|
|
582 HashStr str(name);
|
|
583 if (int_config->orig.Deal(str) == 0) { deal = 0; return 0; }
|
|
584 deal = int_config->orig.Deal(str);
|
|
585 const int* vars = int_config->orig.Get(deal, str);
|
|
586 if (vars == 0) { deal = 0; return 0; }
|
|
587 return vars;
|
|
588 }
|
|
589 void AyuSysConfig::SetParaStr(const char* name, const char* var) {
|
|
590 HashStr str(name);
|
|
591 dirty_flag = 1; change_flag = 1;
|
|
592 str_config->orig.Set(str, 1, var);
|
|
593 }
|
|
594 void AyuSysConfig::SetParam(const char* name, int deal, ...) {
|
|
595 if (deal >= MAXVARS) return ;
|
|
596 HashStr str(name);
|
|
597 int vars[MAXVARS]; va_list va; int i;
|
|
598 va_start(va, deal);
|
|
599 for (i=0; i<deal; i++) vars[i] = va_arg(va, int);
|
|
600 int_config->orig.Set(str, deal, vars);
|
|
601 dirty_flag = 1; change_flag = 1;
|
|
602 return;
|
|
603 }
|
|
604 void AyuSysConfig::SetOrigParaStr(const char* name, const char* var) {
|
|
605 HashStr str(name);
|
|
606 str_config->orig.SetOrig(str, 1, var);
|
|
607 change_flag = 1;
|
|
608 }
|
|
609 void AyuSysConfig::SetOrigParam(const char* name, int deal, ...) {
|
|
610 if (deal >= MAXVARS) return;
|
|
611 HashStr str(name);
|
|
612 int vars[MAXVARS]; va_list va; int i;
|
|
613 va_start(va, deal);
|
|
614 for(i=0; i<deal; i++) vars[i] = va_arg(va, int);
|
|
615 int_config->orig.SetOrig(str, deal, vars);
|
|
616 change_flag = 1;
|
|
617 }
|
|
618 void AyuSysConfig::SetOrigParamArray(const char* name, int deal, int* array) {
|
|
619 HashStr str(name);
|
|
620 int_config->orig.SetOrig(str, deal, array);
|
|
621 }
|
|
622 void AyuSysConfig::SetOriginal(void) {
|
|
623 /* 全ての設定を元に戻す */
|
|
624 str_config->orig.SetOriginal();
|
|
625 int_config->orig.SetOriginal();
|
|
626 change_flag = 1;
|
|
627 }
|
|
628 void AyuSysConfig::DiffOriginal(string& data) {
|
|
629 str_config->orig.DiffOriginal(data);
|
|
630 int_config->orig.DiffOriginal(data);
|
|
631 return;
|
|
632 }
|
|
633 const char* AyuSysConfig::PatchOriginal(const char* data) {
|
|
634 data = str_config->orig.PatchOriginal(data);
|
|
635 data = int_config->orig.PatchOriginal(data);
|
|
636 return data;
|
|
637 }
|
|
638
|
|
639 void AyuSysConfig::Dump(FILE* f) const {
|
|
640 str_config->Dump(f);
|
|
641 int_config->Dump(f);
|
|
642 }
|
|
643
|
|
644 /************************************************
|
|
645 ** AyuSysConfig のコンストラクタ:
|
|
646 ** 全ての config 項目を初期化する
|
|
647 */
|
|
648 AyuSysConfig::AyuSysConfig(void) {
|
|
649 int i;
|
|
650
|
|
651 change_flag = 1; dirty_flag = 0;
|
|
652 str_config = new AyuSysConfigString;
|
|
653 int_config = new AyuSysConfigIntlist;
|
|
654
|
|
655 /****** 文字列 *******/
|
|
656 SetOrigParaStr("#WAKUPDT", "GRDAT"); /* 枠、マウスカーソルなどの画像ファイル */
|
|
657 SetOrigParaStr("#REGNAME", "xclannad"); /* レジストリ名。セーブファイルの作成に使う */
|
|
658 SetOrigParaStr("#CAPTION", "xclannad"); /* ウィンドウのタイトル */
|
|
659 SetOrigParaStr("#SAVENAME","SAVE.INI"); /* セーブファイルの名前 */
|
|
660 SetOrigParaStr("#SAVETITLE", "This is save file"); /* セーブファイルの先頭の文字列 */
|
|
661 SetOrigParaStr("#SAVENOTITLE", "-----------------"); /* 使われてないセーブデータの名前 */
|
|
662 SetOrigParaStr("#CGM_FILE", "MODE.CGM");/* CG mode の設定が保存されたファイル名 */
|
|
663 SetOrigParaStr("#CGTABLE_FILE", "MODE.CGM");/* CG mode の設定が保存されたファイル名 */
|
|
664
|
|
665 SetOrigParaStr("#WAKU.000.000.NAME", ""); // テキストウィンドウの窓飾り画像名
|
|
666 SetOrigParaStr("#WAKU.000.000.BACK", ""); // テキストウィンドウのテキスト背景画像名
|
|
667 SetOrigParaStr("#WAKU.000.000.BTN", ""); // テキストウィンドウのボタン画像名
|
|
668
|
|
669 SetOrigParaStr("#MOUSE_CURSOR.000.NAME", ""); // マウスカーソルのファイル名
|
|
670 SetOrigParaStr("#CURSOR.000.NAME", ""); // リターンカーソルのファイル名
|
|
671 SetOrigParaStr("#SELBTN.000.NAME", ""); // 選択肢背景
|
|
672 SetOrigParaStr("#SELBTN.000.BACK", ""); // 選択肢背景
|
|
673
|
|
674 char name_str[8] = "#NAME.A";
|
|
675 for (i='A'; i<='Z'; i++) {
|
|
676 name_str[6] = i;
|
|
677 SetOrigParaStr(name_str, "");
|
|
678 }
|
|
679
|
|
680 /****** 数値列 *******/
|
|
681 SetOrigParam("#CANCELCALL", 2, 0,0); /* キャンセルボタン(右クリック)したときに呼び出されるサブルーチン番号(メニュー) */
|
|
682 SetOrigParam("#COM2_TITLE", 1, 1); /* ? */
|
|
683 SetOrigParam("#COM2_TITLE_COLOR", 1, 2); /* 選択肢タイトルの色 */
|
|
684 SetOrigParam("#COM2_TITLE_INDENT", 1, 2); /* ? */
|
|
685 SetOrigParam("#SAVEFILETIME", 1, 24); /* セーブする場所の数 */
|
|
686 SetOrigParam("#SEEN_START", 1, 0); /* ゲームを開始するシナリオ番号 */
|
|
687 SetOrigParam("#SEEN_SRT", 1, 0); /* ゲームを開始するシナリオ番号(好き好き大好き) */
|
|
688 SetOrigParam("#SEEN_MENU", 1, 0); /* メニューのシナリオ番号 */
|
|
689 SetOrigParam("#SEEN_TEXT_CURRENT", 1, 0); /* seen.txt を root directory に置くか */
|
|
690 SetOrigParam("#FADE_TIME", 1, 40); /* 画面のフェード・アウトの速度 */
|
|
691 SetOrigParam("#NVL_SYSTEM",1, 0); /* テキストウィンドウが全画面か否か */
|
|
692 SetOrigParam("#WINDOW_ATTR", 5, -1, 128,128, 190, 0); /* テキストウィンドウの色 */
|
|
693 SetOrigParam("#WINDOW_ATTR_AREA", 4, 4,4,4,4); /* テキストウィンドウの範囲 */
|
|
694 SetOrigParam("#WINDOW_ATTR_TYPE", 1, 0); /* テキストウィンドウを半透明にするか */
|
|
695 SetOrigParam("#WINDOW_MSG_POS", 2, 22, 350); /* テキストウィンドウの位置 */
|
|
696 SetOrigParam("#WINDOW_COM_POS", 2,450, 250); /* 選択ウィンドウの位置 */
|
|
697 SetOrigParam("#WINDOW_GRP_POS", 2, 16, 100); /* なにかのウィンドウの位置 */
|
|
698 SetOrigParam("#WINDOW_SUB_POS", 2, 48, 100); /* なにかのウィンドウの位置 */
|
|
699 SetOrigParam("#WINDOW_SYS_POS", 2, 32, 100); /* なにかのウィンドウの位置 */
|
|
700 SetOrigParam("#WINDOW_WAKU_TYPE", 1, 0); /* テキストウィンドウの枠の種類。xkanon 独自設定 */
|
|
701 SetOrigParam("#RETN_CONT", 1, 16); /* リターンカーソルの数 */
|
|
702 SetOrigParam("#RETN_SPEED",1,100); /* リターンカーソルの動く速度 */
|
|
703 SetOrigParam("#RETN_XSIZE", 1, 16); /* リターンカーソルの大きさ */
|
|
704 SetOrigParam("#RETN_YSIZE", 1, 16); /* リターンカーソルの大きさ */
|
|
705 SetOrigParam("#FONT_SIZE", 1, 26); /* フォントの大きさ */
|
|
706 SetOrigParam("#FONT_WEIGHT", 1, 100); /* フォントの weight */
|
|
707 SetOrigParam("#MSG_MOJI_SIZE", 2, 12, 29); /* 文字の大きさ(半角) */
|
|
708 SetOrigParam("#MESSAGE_SIZE", 2, 23, 3); /* メッセージウィンドウの文字数 */
|
|
709 SetOrigParam("#COM_MESSAGE_SIZE", 2, 23, 3); /* メッセージウィンドウの文字数 */
|
|
710 SetOrigParam("#INIT_MESSAGE_SPEED", 1, 30); /* テキスト表示速度 */
|
|
711 SetOrigParam("#INIT_MESSAGE_SPEED_MOD", 1, 0); /* テキスト表示 no wait */
|
|
712 SetOrigParam("#MESSAGE_KEY_WAIT_USE", 1, 0); /* テキスト進行オートモード */
|
|
713 SetOrigParam("#MESSAGE_KEY_WAIT_TIME", 1, 1500); /* オートモードでのキー待ち時間 */
|
|
714
|
|
715 SetOrigParam("#GRP_DC_TIMES", 1, 4); /* 裏画面の数 */
|
|
716 SetOrigParam("#MUSIC_LINEAR_PAC",1,0); /* PCM データの 8bit -> 16bit 変換を行うか */
|
|
717 SetOrigParam("#MUSIC_TYPE",1,0); /* PCM データの種類 */
|
|
718 SetOrigParam("#WINDOW_MSGBK_BOX",1,0); /* バックログ用のボタン */
|
|
719 SetOrigParam("#WINDOW_MSGBK_LBOX_POS",4,15,7,8,0); /* バックログ用のボタン(左)の位置 */
|
|
720 SetOrigParam("#WINDOW_MSGBK_RBOX_POS",4,7,7,0,0); /* バックログ用のボタン(左)の位置 */
|
|
721 SetOrigParam("#MSGBK_MOD",1,0); /* バックログ用のボタンを使用するか */
|
|
722
|
|
723 SetOrigParam("#WAKU.000.000.TYPE", 1, 5);
|
|
724 SetOrigParam("#WAKU.000.000.MOVE_BOX", 5, 0, 0, 0, 0, 0); // テキストウィンドウの移動用ボタン位置
|
|
725 SetOrigParam("#WAKU.000.000.CLEAR_BOX", 5, 0, 0, 0, 0, 0); // 一時消去用ボタン位置
|
|
726 SetOrigParam("#WAKU.000.000.READJUMP_BOX", 5, 0, 0, 0, 0, 0); // スキップ用ボタン位置
|
|
727 SetOrigParam("#WAKU.000.000.AUTOMODE_BOX", 5, 0, 0, 0, 0, 0); // オート用ボタン位置
|
|
728 SetOrigParam("#WAKU.000.000.MSGBK_BOX", 5, 0, 0, 0, 0, 0); // バックログボタン位置
|
|
729 SetOrigParam("#WAKU.000.000.MSGBKLEFT_BOX", 5, 0, 0, 0, 0, 0); // バックログ(進める)ボタン位置
|
|
730 SetOrigParam("#WAKU.000.000.MSGBKRIGHT_BOX", 5, 0, 0, 0, 0, 0); // バックログ(戻る)ボタン位置
|
|
731 SetOrigParam("#WAKU.000.000.EXBTN_000_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン0位置
|
|
732 SetOrigParam("#WAKU.000.000.EXBTN_001_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン1位置
|
|
733 SetOrigParam("#WAKU.000.000.EXBTN_002_BOX", 5, 0, 0, 0, 0, 0); // その他ボタン2位置
|
|
734
|
|
735 SetOrigParam("#WINDOW.000.MOJI_SIZE", 1, 21); // 文字サイズ
|
|
736 SetOrigParam("#WINDOW.000.MOJI_REP", 2, -1, 2); // 文字の余裕
|
|
737 SetOrigParam("#WINDOW.000.MOJI_CNT", 2, 20, 3); // ウィンドウ内の文字数
|
|
738 SetOrigParam("#WINDOW.000.MOJI_POS", 4, 100, 0, 180, 40); // テキスト位置。3つ目がx で1つ目がyらしい
|
|
739 SetOrigParam("#WINDOW.000.MOJI_SHADOW", 1, 0); // 文字に影を付けるか
|
|
740 SetOrigParam("#WINDOW.000.LUBY_SIZE", 1, 8); // ルビの文字サイズ
|
|
741 SetOrigParam("#WINDOW.000.MOJI_MIN", 2, 8, 1); // 文字同士の隙間?
|
|
742 SetOrigParam("#WINDOW.000.SELCOM_USE", 1, 0); // 選択肢の実装方法
|
|
743 SetOrigParam("#WINDOW.000.POS", 4, 100, 0, 0, 260); // ウィンドウ位置
|
|
744 SetOrigParam("#WINDOW.000.ATTR_MOD", 1, 0); // ウィンドウ色
|
|
745 SetOrigParam("#WINDOW.000.ATTR", 5, -1, -1, -1, -1, -1); // ウィンドウ色
|
|
746 /* SELCOM はよくわからんので無視 */
|
|
747 SetOrigParam("#WINDOW.000.OPEN_ANM_MOD", 1, 0); // ウィンドウを開くときの効果らしい
|
|
748 SetOrigParam("#WINDOW.000.OPEN_ANM_TIME", 1, 500);
|
|
749 SetOrigParam("#WINDOW.000.CLOSE_ANM_MOD", 1, 0); // ウィンドウを閉じるときの効果らしい
|
|
750 SetOrigParam("#WINDOW.000.CLOSE_ANM_TIME", 1, 500);
|
|
751 SetOrigParam("#WINDOW.000.WAKU_SETNO", 1, 0); // 枠の種類
|
|
752 SetOrigParam("#WINDOW.000.MOVE_USE", 1, 0); // ウィンドウ枠移動ボタン使用の可否
|
|
753 SetOrigParam("#WINDOW.000.CLEAR_USE", 1, 0); // ウィンドウ枠消去ボタン使用の可否
|
|
754 SetOrigParam("#WINDOW.000.READJUMP_USE", 1, 0); // スキップボタン使用の可否
|
|
755 SetOrigParam("#WINDOW.000.AUTOMODE_USE", 1, 0); // スキップボタン使用の可否
|
|
756 SetOrigParam("#WINDOW.000.MSGBK_USE", 1, 0); // バックログボタン使用の可否
|
|
757 SetOrigParam("#WINDOW.000.MSGBKLEFT_USE", 1, 0); // バックログ(進む)ボタン使用の可否
|
|
758 SetOrigParam("#WINDOW.000.MSGBKRIGHT_USE", 1, 0); // バックログ(戻る)ボタン使用の可否
|
|
759 SetOrigParam("#WINDOW.000.EXBTN_000_USE", 1, 0); // その他ボタン0使用の可否
|
|
760 SetOrigParam("#WINDOW.000.EXBTN_001_USE", 1, 0); // その他ボタン1使用の可否
|
|
761 SetOrigParam("#WINDOW.000.EXBTN_002_USE", 1, 0); // その他ボタン2使用の可否
|
|
762 SetOrigParam("#WINDOW.000.NAME_MOD", 1, 0); // 名前ウィンドウを別途使用するか
|
|
763 SetOrigParam("#WINDOW.000.NAME_MOJI_SIZE", 1, 20); // 名前フォントのサイズ
|
|
764 SetOrigParam("#WINDOW.000.NAME_MOJI_POS", 2, 0, 0); // 名前ウィンドウの文字の位置
|
|
765 SetOrigParam("#WINDOW.000.NAME_MOJI_MIN", 1, 0); // 名前ウィンドウの幅
|
|
766 SetOrigParam("#WINDOW.000.NAME_CENTERING", 1, 1); // 名前のセンタリングの有無
|
|
767 SetOrigParam("#WINDOW.000.NAME_POS", 2, 159, 78); // 名前ウィンドウ位置(左下位置らしい)
|
|
768 SetOrigParam("#WINDOW.000.NAME_WAKU_SETNO", 1, -1); // 名前ウィンドウ位置(左下位置らしい)
|
|
769 SetOrigParam("#WINDOW.000.FACE.000", 5, 0, 0, 1, 1, 1); // 顔ウィンドウ位置(始め2つがx,y、MOJI_POSからの相対位置なのに注意)
|
|
770 SetOrigParam("#WINDOW.000.KEYCUR_MOD", 3, 0, 0, 0); // リターンカーソルの位置
|
|
771
|
|
772
|
|
773 SetOrigParam("#CURSOR.000.SIZE", 2, 0, 0); // リターンカーソルの大きさ
|
|
774 SetOrigParam("#CURSOR.000.CONT", 1, 50); // リターンカーソルの繰り返し数
|
|
775 SetOrigParam("#CURSOR.000.SPEED", 1, 1000); // ブリンクする速さ
|
|
776
|
|
777 SetOrigParam("#SELBTN.000.BASEPOS", 2, 0, 0); // 選択肢ウィンドウの位置
|
|
778 SetOrigParam("#SELBTN.000.REPPOS", 2, 0, 50); // 選択肢ウィンドウの次の位置(相対)
|
|
779 SetOrigParam("#SELBTN.000.MOJISIZE", 4, 26, 0,0,0); // 文字の大きさ
|
|
780 SetOrigParam("#SELBTN.000.MOJIDEFAULTCOL", 1, 0); // 非選択時の文字色
|
|
781 SetOrigParam("#SELBTN.000.MOJISELECTCOL", 1, 0); // 選択時の文字色
|
|
782
|
|
783 SetOrigParam("#COLOR_TABLE.000", 3, 255,255,255);
|
|
784 SetOrigParam("#SHAKE.000", 3, 0,0,0);
|
|
785
|
|
786 SetOrigParam("#SELR.000",16,0,0,640,480,0,0,500,50,0,0,0,0,0,0,255,0);
|
|
787 SetOrigParam("#SEL.000", 15,0,0,639,479,0,0, 32, 4,0,0,0,0,0,0,0);
|
|
788
|
|
789 SetOrigParam("#SCREENSIZE_MOD", 1, 0); /* 0 = 640x480; 1 = 800x600 */
|
|
790 }
|
|
791
|
|
792 static int SplitVar(const char* str, int* ret_var, int ret_size) {
|
|
793 /* , あるいは ),:( をセパレータとして、-?[0-9]+ の
|
|
794 ** フォーマットの数値列を読み込む。先頭に (、末尾に ) が付きうる。
|
|
795 ** (),-[0-9] 以外の文字があったらそこで終了
|
|
796 ** 得られたデータ数を返す
|
|
797 */
|
|
798 if (*str == '(') str++;
|
|
799 int i; for (i=0; i<ret_size; i++) {
|
|
800 int c; int is_positive = 1;
|
|
801 /* セパレータの読み飛ばし */
|
|
802 c = *str;
|
|
803 if (c == ',' || c == ':') {
|
|
804 str++;
|
|
805 } else if (c == ')' && str[1] == '(') {
|
|
806 str += 2;
|
|
807 }
|
|
808 /* - を parse */
|
|
809 c = *str;
|
|
810 if (c == '-' && isdigit(str[1])) {
|
|
811 is_positive = -1; str++;
|
|
812 } else if (! isdigit(c)) {
|
|
813 return i; /* 異常な文字を見つけた:終了 */
|
|
814 }
|
|
815 int number = 0;
|
|
816 /* 数字読み込み */
|
|
817 while(isdigit( (c=*str) )) {
|
|
818 number *= 10;
|
|
819 number += c-'0';
|
|
820 str++;
|
|
821 }
|
|
822 ret_var[i] = is_positive * number;
|
|
823 }
|
|
824 return i;
|
|
825 }
|
|
826 /* 決められた数の引数を得る。-1 ならエラーが生じた */
|
|
827 static inline int SplitVar(const char* str, int& var1) {
|
|
828 if (SplitVar(str, &var1, 1) != 1) return -1;
|
|
829 return 0;
|
|
830 }
|
|
831 static inline int SplitVar(const char* str, int& var1, int& var2) {
|
|
832 int vars[2];
|
|
833 if (SplitVar(str, vars, 2) != 2) return -1;
|
|
834 var1 = vars[0]; var2 = vars[1];
|
|
835 return 0;
|
|
836 }
|
|
837 static inline int SplitVar(const char* str, int& var1, int& var2, int& var3) {
|
|
838 int vars[3];
|
|
839 if (SplitVar(str, vars, 3) != 3) return -1;
|
|
840 var1 = vars[0]; var2 = vars[1]; var3 = vars[2];
|
|
841 return 0;
|
|
842 }
|
|
843 static inline int SplitVar(const char* str, int& var1, int& var2, int& var3, int& var4) {
|
|
844 int vars[4];
|
|
845 if (SplitVar(str, vars, 4) != 4) return -1;
|
|
846 var1 = vars[0]; var2 = vars[1]; var3 = vars[2]; var4 = vars[3];
|
|
847 return 0;
|
|
848 }
|
|
849
|
|
850 bool AyuSysConfig::LoadInitFile(void)
|
|
851 {
|
|
852 char buf[1024]; int i;
|
|
853 char* tokens[MAXTOKEN]; int token_deal; int buf_ptr;
|
|
854 int numbers[MAXVARS];
|
|
855
|
|
856 ARCINFO* info = file_searcher.Find(FILESEARCH::ROOT, "gameexe.ini");
|
|
857 if (info == NULL) return false;
|
|
858 int size = info->Size();
|
|
859 unsigned char* buf_orig = (unsigned char*)info->Read();
|
|
860 if (size <= 0 || buf_orig == 0) {
|
|
861 delete info; return false;
|
|
862 }
|
|
863 unsigned char* buf_end = buf_orig + size;
|
|
864 int line_count = 0;
|
|
865 while(buf_orig < buf_end) {
|
|
866 /* buf_orig から一行読み込む */
|
|
867 /* その際に、
|
|
868 ** ・頭が # 以外なら次の行までとばす
|
|
869 ** ・"" 外のスペース、TABを初めとする制御文字 (0x20 以下のASCIIコード)を削除
|
|
870 ** ・= で区切る。区切りは最大で10個で、tokens に代入される
|
|
871 ** などの操作を行う
|
|
872 */
|
|
873
|
|
874 /* # チェック */
|
|
875 if (*buf_orig != '#') {
|
|
876 /* 次の '\n' まで読み飛ばし */
|
|
877 while(buf_orig < buf_end &&
|
|
878 *buf_orig != '\n' && *buf_orig != '\r') buf_orig++;
|
|
879 if (buf_orig < buf_end-1 && *buf_orig == '\r' && buf_orig[1] == '\n') buf_orig += 2;
|
|
880 else if (*buf_orig == '\r' || *buf_orig == '\n') buf_orig++;
|
|
881 line_count++;
|
|
882 continue;
|
|
883 }
|
|
884 /* 初期化 */
|
|
885 token_deal = 1; tokens[0] = buf; buf_ptr = 0;
|
|
886 int in_quote = 0;
|
|
887
|
|
888 while(buf_orig < buf_end && buf_ptr < 1023) {
|
|
889 if (in_quote) {
|
|
890 /* "" の中 */
|
|
891 int c = *buf_orig;
|
|
892 if (c == '\n' || c == '\r') {
|
|
893 break;
|
|
894 } else if (c == '\"') {
|
|
895 in_quote = 0;
|
|
896 } else {
|
|
897 buf[buf_ptr++] = c;
|
|
898 }
|
|
899 buf_orig++;
|
|
900 } else { /* quote されてない */
|
|
901 /* 制御文字を読み飛ばす */
|
|
902 while(*buf_orig <= 0x20 && buf_orig < buf_end &&
|
|
903 *buf_orig != '\n' && *buf_orig != '\r') buf_orig++;
|
|
904 int c = *buf_orig;
|
|
905 if (c == '\n' || c == '\r') break;
|
|
906 /* = なら次の token */
|
|
907 if (c == '=') {
|
|
908 c = 0; tokens[token_deal++] = buf+buf_ptr+1;
|
|
909 if (token_deal >= MAXTOKEN) break;
|
|
910 } else if (c == '\"') {
|
|
911 in_quote = 1; buf_orig++; continue;
|
|
912 }
|
|
913 buf[buf_ptr++] = c;
|
|
914 buf_orig++;
|
|
915 }
|
|
916 }
|
|
917 buf[buf_ptr] = '\0';
|
|
918 /* 末尾の \r\n を消去 */
|
|
919 if (buf_orig < buf_end-1 && buf_orig[0] == '\r' && buf_orig[1] == '\n') buf_orig += 2;
|
|
920 else if (buf_orig < buf_end && (buf_orig[0] == '\r' || buf_orig[0] == '\n')) buf_orig++;
|
|
921 /* 必要なら parse 内容を出力 */
|
|
922 dprintf(("line %3d ",line_count));
|
|
923 for (i=0; i<token_deal; i++) {
|
|
924 dprintf(("%d:\"%s\", ",i,tokens[i]));
|
|
925 }
|
|
926 dprintf(("\n"));
|
|
927 if (in_quote) {
|
|
928 fprintf(stderr, "Warning : open quote is found while parsing gameexe.ini, line %d\n",line_count);
|
|
929 }
|
|
930
|
|
931
|
|
932 /* 得られた内容を parse */
|
|
933
|
|
934 /* #NAME=<文字列> */
|
|
935 int type = SearchParam(tokens[0]);
|
|
936 if (type == 1) { /* #NAME=<文字列> */
|
|
937 if (token_deal != 2) {
|
|
938 dprintf(("Parse error, line %d, %s\n",line_count, tokens[0]));
|
|
939 goto parse_error;
|
|
940 }
|
|
941 SetOrigParaStr(tokens[0], tokens[1]);
|
|
942 goto parse_end;
|
|
943 } else if (type == 2) { /* #NAME=<数値列> */
|
|
944 if (token_deal != 2) {
|
|
945 dprintf(("Parse error, line %d, %s\n",line_count, tokens[0]));
|
|
946 goto parse_error;
|
|
947 }
|
|
948 int number_deal = SplitVar(tokens[1], numbers, MAXVARS);
|
|
949 SetOrigParamArray(tokens[0], number_deal, numbers);
|
|
950 goto parse_end;
|
|
951 }
|
|
952 /* 一般的な設定以外 : cdrom track など */
|
|
953 if (strncmp(tokens[0],"#NAME.", 6) == 0) {
|
|
954 if (token_deal != 2) goto parse_error;
|
|
955 SetOrigParaStr(tokens[0], tokens[1]);
|
|
956 goto parse_end;
|
|
957 } else if (strncmp(tokens[0],"#DIRC.",6) == 0) {
|
|
958 if (token_deal != 3) goto parse_error;
|
|
959 /* ファイル形式の指定 */
|
|
960 FILESEARCH::FILETYPE type;
|
|
961 char* name = tokens[0]+6;
|
|
962 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
|
|
963 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
|
|
964 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
|
|
965 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
|
|
966 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM;
|
|
967 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
|
|
968 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
|
|
969 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
|
|
970 else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE;
|
|
971 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
|
|
972 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
|
|
973 if (tokens[2][0] == 'N') { /* directory */
|
|
974 file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
|
|
975 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
|
|
976 } else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */
|
|
977 file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
|
|
978 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
|
|
979 } else goto parse_error;
|
|
980 goto parse_end;
|
|
981 }
|
|
982 if (strncmp(tokens[0],"#ADRC.",6) == 0) {
|
|
983 if (token_deal != 3) goto parse_error;
|
|
984 /* ファイル形式の指定 */
|
|
985 FILESEARCH::FILETYPE type;
|
|
986 char* name = tokens[0]+6;
|
|
987 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
|
|
988 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
|
|
989 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
|
|
990 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
|
|
991 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM;
|
|
992 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
|
|
993 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
|
|
994 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
|
|
995 else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE;
|
|
996 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
|
|
997 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
|
|
998 if (tokens[2][0] == 'N') { /* directory */
|
|
999 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
|
|
1000 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
|
|
1001 } else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */
|
|
1002 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
|
|
1003 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
|
|
1004 } else if (tokens[2][0] == 'R' && tokens[2][1] == ':') { /* それ散るアーカイブ */
|
|
1005 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2);
|
|
1006 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
|
|
1007 } else goto parse_error;
|
|
1008 goto parse_end;
|
|
1009 }
|
|
1010 if (strncmp(tokens[0],"#FOLDNAME.",10) == 0) {
|
|
1011 if (token_deal != 3) goto parse_error;
|
|
1012 /* ファイル形式の指定 */
|
|
1013 FILESEARCH::FILETYPE type;
|
|
1014 char* name = tokens[0]+10;
|
|
1015 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT;
|
|
1016 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT;
|
|
1017 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT;
|
|
1018 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN;
|
|
1019 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM;
|
|
1020 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD;
|
|
1021 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR;
|
|
1022 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV;
|
|
1023 else if (strcmp(name, "BGM") == 0) type = FILESEARCH::BGM;
|
|
1024 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN;
|
|
1025 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */
|
|
1026 if (tokens[2][0] == '0') { /* directory */
|
|
1027 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]);
|
|
1028 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1]));
|
|
1029 } else if (tokens[2][0] == '1' && tokens[2][1] == ':') { /* アーカイブ */
|
|
1030 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_SCN2k, tokens[2]+2);
|
|
1031 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2));
|
|
1032 } else goto parse_error;
|
|
1033 goto parse_end;
|
|
1034 }
|
|
1035 if (strcmp(tokens[0], "#CDTRACK") == 0) {
|
|
1036 if (token_deal != 3) goto parse_error;
|
|
1037 track_name.AddCDROM(tokens[2], atoi(tokens[1]));
|
|
1038 dprintf(("Set CDTRACK, name %s, track %d\n",tokens[2], atoi(tokens[1])));
|
|
1039 goto parse_end;
|
|
1040 }
|
|
1041 if (strcmp(tokens[0], "#DSTRACK") == 0) {
|
|
1042 /* #DSTRACK=00000000-99999000-00782556="filename" ="name" */
|
|
1043 /* #DSTRACK=00000000-99999000-00782556="name" */
|
|
1044 /* 第二トークンの3つめのパラメータを得る(繰り返しの時の再生開始位置) */
|
|
1045 int start_pt = 0;
|
|
1046 const char* tk1 = strchr(tokens[1], '-');
|
|
1047 const char* tk2 = 0;
|
|
1048 if (tk1 && *tk1) tk2 = strchr(tk1+1, '-');
|
|
1049 if (tk2 && *tk2) start_pt = atoi(tk2+1);
|
|
1050 if (token_deal == 3) {
|
|
1051 track_name.AddWave(tokens[2], tokens[2], start_pt);
|
|
1052 dprintf(("Set Wave track, name %s\n",tokens[2]));
|
|
1053 } else if (token_deal == 4) {
|
|
1054 track_name.AddWave(tokens[3], tokens[2], start_pt);
|
|
1055 dprintf(("Set Wave track, name %s, file %s\n",tokens[3], tokens[2]));
|
|
1056 } else goto parse_error;
|
|
1057 goto parse_end;
|
|
1058 }
|
|
1059 if (strncmp(tokens[0], "#SE.", 4) == 0) {
|
|
1060 /* SE.XXX="XXX"=X */
|
|
1061 if (token_deal == 2) {
|
|
1062 track_name.AddSE(atoi(tokens[0]+4), tokens[1]);
|
|
1063 } else if (token_deal == 3) {
|
|
1064 if (atoi(tokens[2]) != 0) {
|
|
1065 track_name.AddSE(atoi(tokens[0]+4), tokens[1]);
|
|
1066 }
|
|
1067 }
|
|
1068 dprintf(("Set SE %d, name %s\n",atoi(tokens[0]+4), tokens[1]));
|
|
1069 goto parse_end;
|
|
1070 }
|
|
1071 /* 設定項目が見つからなかった */
|
|
1072 dprintf(("Cannot find configuration name: %s\n",tokens[0]));
|
|
1073 parse_error:
|
|
1074 parse_end:
|
|
1075 line_count++;
|
|
1076 }
|
|
1077 delete info;
|
|
1078 /* デフォルトのオプションを指定する */
|
|
1079 // set_game(GetParaStr("#REGNAME"), *this);
|
|
1080 return true;
|
|
1081 }
|
|
1082
|
|
1083 TrackName::TrackName(void) {
|
|
1084 deal = 1;
|
|
1085 track = new char*[deal];
|
|
1086 track_wave = new char*[deal];
|
|
1087 track_num = new int[deal];
|
|
1088 track_start = new int[deal];
|
|
1089 int i; for (i=0; i<deal; i++) track[i] = 0;
|
|
1090 for (i=0; i<deal; i++) track_wave[i] = 0;
|
|
1091 se_deal = 10;
|
|
1092 se_track = new char*[se_deal];
|
|
1093 for (i=0; i<se_deal; i++) se_track[i] = 0;
|
|
1094 }
|
|
1095
|
|
1096 TrackName::~TrackName() {
|
|
1097 int i; for (i=0; i<deal; i++) {
|
|
1098 if (track[i] != 0) delete[] track[i];
|
|
1099 if (track_wave[i] != 0) delete[] track_wave[i];
|
|
1100 }
|
|
1101 for (i=0; i<se_deal; i++) {
|
|
1102 if (se_track[i]) delete[] se_track[i];
|
|
1103 }
|
|
1104 delete[] track;
|
|
1105 delete[] track_wave;
|
|
1106 delete[] track_num;
|
|
1107 delete[] track_start;
|
|
1108 delete[] se_track;
|
|
1109 }
|
|
1110 void TrackName::Expand(void) {
|
|
1111 int new_deal = deal * 2;
|
|
1112 int* new_track_num = new int[new_deal];
|
|
1113 int* new_track_start = new int[new_deal];
|
|
1114 char** new_track = new char*[new_deal];
|
|
1115 char** new_track_wave = new char*[new_deal];
|
|
1116 int i; for (i=0; i<deal; i++) {
|
|
1117 new_track_num[i] = track_num[i];
|
|
1118 new_track_start[i] = track_start[i];
|
|
1119 new_track[i] = track[i];
|
|
1120 new_track_wave[i] = track_wave[i];
|
|
1121 }
|
|
1122 for (; i<new_deal; i++) {
|
|
1123 new_track_num[i] = 0;
|
|
1124 new_track_start[i] = 0;
|
|
1125 new_track[i] = 0;
|
|
1126 new_track_wave[i] = 0;
|
|
1127 }
|
|
1128 deal = new_deal;
|
|
1129 delete[] track; track = new_track;
|
|
1130 delete[] track_num; track_num= new_track_num;
|
|
1131 delete[] track_start; track_start= new_track_start;
|
|
1132 delete[] track_wave; track_wave = new_track_wave;
|
|
1133 }
|
|
1134 void TrackName::ExpandSE(int n) {
|
|
1135 if (n < 0) return;
|
|
1136 n += 10;
|
|
1137 if (se_deal >= n) return;
|
|
1138 char** new_se = new char*[n];
|
|
1139 int i; for (i=0; i<se_deal; i++) new_se[i] = se_track[i];
|
|
1140 for (; i<n; i++) new_se[i] = 0;
|
|
1141 delete[] se_track;
|
|
1142 se_deal = n; se_track = new_se;
|
|
1143 }
|
|
1144 void TrackName::AddCDROM(char* name, int tk) {
|
|
1145 if (CDTrack(name) != -1) return;
|
|
1146 int i; for (i=0; i<deal; i++) {
|
|
1147 if (track[i] == 0) break;
|
|
1148 }
|
|
1149 int num = i;
|
|
1150 if (i == deal) Expand();
|
|
1151 track[num] = new char[strlen(name)+1];
|
|
1152 for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]);
|
|
1153 track[num][i] = 0;
|
|
1154 track_num[num] = tk;
|
|
1155 }
|
|
1156 void TrackName::AddWave(char* name, char* file, int pt) {
|
|
1157 if (CDTrack(name) != -1) return;
|
|
1158 int i; for (i=0; i<deal; i++) {
|
|
1159 if (track[i] == 0) break;
|
|
1160 }
|
|
1161 int num = i;
|
|
1162 if (i == deal) Expand();
|
|
1163 track_num[num] = 0;
|
|
1164 track_start[num] = pt;
|
|
1165 track[num] = new char[strlen(name)+1];
|
|
1166 for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]);
|
|
1167 track[num][i] = 0;
|
|
1168 track_wave[num] = new char[strlen(file)+1]; strcpy(track_wave[num], file);
|
|
1169 }
|
|
1170 int TrackName::CDTrack(char* name) {
|
|
1171 char buf[1024];
|
|
1172 int i;
|
|
1173 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
|
|
1174 buf[i]=0;
|
|
1175 for (i=0; i<deal; i++) {
|
|
1176 if (track[i] == 0) return -1;
|
|
1177 if (strcmp(track[i], buf) == 0) {
|
|
1178 return track_num[i];
|
|
1179 }
|
|
1180 }
|
|
1181 return -1;
|
|
1182 }
|
|
1183 int TrackName::TrackStart(char* name) {
|
|
1184 char buf[1024];
|
|
1185 int i;
|
|
1186 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
|
|
1187 buf[i]=0;
|
|
1188 for (i=0; i<deal; i++) {
|
|
1189 if (track[i] == 0) return -1;
|
|
1190 if (strcmp(track[i], buf) == 0) {
|
|
1191 return track_start[i];
|
|
1192 }
|
|
1193 }
|
|
1194 return 0;
|
|
1195 }
|
|
1196 const char* TrackName::WaveTrack(char* name) {
|
|
1197 char buf[1024];
|
|
1198 int i;
|
|
1199 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]);
|
|
1200 buf[i]=0;
|
|
1201 for (i=0; i<deal; i++) {
|
|
1202 if (track[i] == 0) return 0;
|
|
1203 if (strcmp(track[i], buf) == 0) {
|
|
1204 return track_wave[i];
|
|
1205 }
|
|
1206 }
|
|
1207 return 0;
|
|
1208 }
|
|
1209 const char* TrackName::SETrack(int n) {
|
|
1210 if (n < 0 || n >= se_deal) return 0;
|
|
1211 return se_track[n];
|
|
1212 }
|
|
1213 void TrackName::AddSE(int n, char* file) {
|
|
1214 if (se_deal <= n) ExpandSE(n);
|
|
1215 if (se_track[n]) delete[] se_track[n];
|
|
1216 se_track[n] = new char[strlen(file)+1];
|
|
1217 strcpy(se_track[n], file);
|
|
1218 }
|
|
1219
|