Mercurial > otakunoraifu
annotate system/system_config.cc @ 26:f45da03ca631
* Corrected --disable-vorbis in configure.ac
* Added sound configuration
* Added misc functions
author | thib |
---|---|
date | Sun, 01 Mar 2009 19:08:02 +0000 |
parents | 92765a5661f7 |
children | 3a6aaeab7b4e |
rev | line source |
---|---|
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 */ | |
16
92765a5661f7
added "new" label on last savegame and corrected StatSaveFile
thib
parents:
9
diff
changeset
|
790 |
92765a5661f7
added "new" label on last savegame and corrected StatSaveFile
thib
parents:
9
diff
changeset
|
791 SetOrigParam("#LASTSAVE", 1, 0); |
26 | 792 |
793 SetOrigParam("#VOLMOD", 4, 128, 128, 128, 128); | |
0 | 794 } |
9 | 795 AyuSysConfig::~AyuSysConfig(void) { |
796 delete str_config; | |
797 delete int_config; | |
798 } | |
0 | 799 |
800 static int SplitVar(const char* str, int* ret_var, int ret_size) { | |
801 /* , あるいは ),:( をセパレータとして、-?[0-9]+ の | |
802 ** フォーマットの数値列を読み込む。先頭に (、末尾に ) が付きうる。 | |
803 ** (),-[0-9] 以外の文字があったらそこで終了 | |
804 ** 得られたデータ数を返す | |
805 */ | |
806 if (*str == '(') str++; | |
807 int i; for (i=0; i<ret_size; i++) { | |
808 int c; int is_positive = 1; | |
809 /* セパレータの読み飛ばし */ | |
810 c = *str; | |
811 if (c == ',' || c == ':') { | |
812 str++; | |
813 } else if (c == ')' && str[1] == '(') { | |
814 str += 2; | |
815 } | |
816 /* - を parse */ | |
817 c = *str; | |
818 if (c == '-' && isdigit(str[1])) { | |
819 is_positive = -1; str++; | |
820 } else if (! isdigit(c)) { | |
821 return i; /* 異常な文字を見つけた:終了 */ | |
822 } | |
823 int number = 0; | |
824 /* 数字読み込み */ | |
825 while(isdigit( (c=*str) )) { | |
826 number *= 10; | |
827 number += c-'0'; | |
828 str++; | |
829 } | |
830 ret_var[i] = is_positive * number; | |
831 } | |
832 return i; | |
833 } | |
834 /* 決められた数の引数を得る。-1 ならエラーが生じた */ | |
835 static inline int SplitVar(const char* str, int& var1) { | |
836 if (SplitVar(str, &var1, 1) != 1) return -1; | |
837 return 0; | |
838 } | |
839 static inline int SplitVar(const char* str, int& var1, int& var2) { | |
840 int vars[2]; | |
841 if (SplitVar(str, vars, 2) != 2) return -1; | |
842 var1 = vars[0]; var2 = vars[1]; | |
843 return 0; | |
844 } | |
845 static inline int SplitVar(const char* str, int& var1, int& var2, int& var3) { | |
846 int vars[3]; | |
847 if (SplitVar(str, vars, 3) != 3) return -1; | |
848 var1 = vars[0]; var2 = vars[1]; var3 = vars[2]; | |
849 return 0; | |
850 } | |
851 static inline int SplitVar(const char* str, int& var1, int& var2, int& var3, int& var4) { | |
852 int vars[4]; | |
853 if (SplitVar(str, vars, 4) != 4) return -1; | |
854 var1 = vars[0]; var2 = vars[1]; var3 = vars[2]; var4 = vars[3]; | |
855 return 0; | |
856 } | |
857 | |
858 bool AyuSysConfig::LoadInitFile(void) | |
859 { | |
860 char buf[1024]; int i; | |
861 char* tokens[MAXTOKEN]; int token_deal; int buf_ptr; | |
862 int numbers[MAXVARS]; | |
863 | |
864 ARCINFO* info = file_searcher.Find(FILESEARCH::ROOT, "gameexe.ini"); | |
865 if (info == NULL) return false; | |
866 int size = info->Size(); | |
867 unsigned char* buf_orig = (unsigned char*)info->Read(); | |
868 if (size <= 0 || buf_orig == 0) { | |
869 delete info; return false; | |
870 } | |
871 unsigned char* buf_end = buf_orig + size; | |
872 int line_count = 0; | |
873 while(buf_orig < buf_end) { | |
874 /* buf_orig から一行読み込む */ | |
875 /* その際に、 | |
876 ** ・頭が # 以外なら次の行までとばす | |
877 ** ・"" 外のスペース、TABを初めとする制御文字 (0x20 以下のASCIIコード)を削除 | |
878 ** ・= で区切る。区切りは最大で10個で、tokens に代入される | |
879 ** などの操作を行う | |
880 */ | |
881 | |
882 /* # チェック */ | |
883 if (*buf_orig != '#') { | |
884 /* 次の '\n' まで読み飛ばし */ | |
885 while(buf_orig < buf_end && | |
886 *buf_orig != '\n' && *buf_orig != '\r') buf_orig++; | |
887 if (buf_orig < buf_end-1 && *buf_orig == '\r' && buf_orig[1] == '\n') buf_orig += 2; | |
888 else if (*buf_orig == '\r' || *buf_orig == '\n') buf_orig++; | |
889 line_count++; | |
890 continue; | |
891 } | |
892 /* 初期化 */ | |
893 token_deal = 1; tokens[0] = buf; buf_ptr = 0; | |
894 int in_quote = 0; | |
895 | |
896 while(buf_orig < buf_end && buf_ptr < 1023) { | |
897 if (in_quote) { | |
898 /* "" の中 */ | |
899 int c = *buf_orig; | |
900 if (c == '\n' || c == '\r') { | |
901 break; | |
902 } else if (c == '\"') { | |
903 in_quote = 0; | |
904 } else { | |
905 buf[buf_ptr++] = c; | |
906 } | |
907 buf_orig++; | |
908 } else { /* quote されてない */ | |
909 /* 制御文字を読み飛ばす */ | |
910 while(*buf_orig <= 0x20 && buf_orig < buf_end && | |
911 *buf_orig != '\n' && *buf_orig != '\r') buf_orig++; | |
912 int c = *buf_orig; | |
913 if (c == '\n' || c == '\r') break; | |
914 /* = なら次の token */ | |
915 if (c == '=') { | |
916 c = 0; tokens[token_deal++] = buf+buf_ptr+1; | |
917 if (token_deal >= MAXTOKEN) break; | |
918 } else if (c == '\"') { | |
919 in_quote = 1; buf_orig++; continue; | |
920 } | |
921 buf[buf_ptr++] = c; | |
922 buf_orig++; | |
923 } | |
924 } | |
925 buf[buf_ptr] = '\0'; | |
926 /* 末尾の \r\n を消去 */ | |
927 if (buf_orig < buf_end-1 && buf_orig[0] == '\r' && buf_orig[1] == '\n') buf_orig += 2; | |
928 else if (buf_orig < buf_end && (buf_orig[0] == '\r' || buf_orig[0] == '\n')) buf_orig++; | |
929 /* 必要なら parse 内容を出力 */ | |
930 dprintf(("line %3d ",line_count)); | |
931 for (i=0; i<token_deal; i++) { | |
932 dprintf(("%d:\"%s\", ",i,tokens[i])); | |
933 } | |
934 dprintf(("\n")); | |
935 if (in_quote) { | |
936 fprintf(stderr, "Warning : open quote is found while parsing gameexe.ini, line %d\n",line_count); | |
937 } | |
938 | |
939 | |
940 /* 得られた内容を parse */ | |
941 | |
942 /* #NAME=<文字列> */ | |
943 int type = SearchParam(tokens[0]); | |
944 if (type == 1) { /* #NAME=<文字列> */ | |
945 if (token_deal != 2) { | |
946 dprintf(("Parse error, line %d, %s\n",line_count, tokens[0])); | |
947 goto parse_error; | |
948 } | |
949 SetOrigParaStr(tokens[0], tokens[1]); | |
950 goto parse_end; | |
951 } else if (type == 2) { /* #NAME=<数値列> */ | |
952 if (token_deal != 2) { | |
953 dprintf(("Parse error, line %d, %s\n",line_count, tokens[0])); | |
954 goto parse_error; | |
955 } | |
956 int number_deal = SplitVar(tokens[1], numbers, MAXVARS); | |
957 SetOrigParamArray(tokens[0], number_deal, numbers); | |
958 goto parse_end; | |
959 } | |
960 /* 一般的な設定以外 : cdrom track など */ | |
961 if (strncmp(tokens[0],"#NAME.", 6) == 0) { | |
962 if (token_deal != 2) goto parse_error; | |
963 SetOrigParaStr(tokens[0], tokens[1]); | |
964 goto parse_end; | |
965 } else if (strncmp(tokens[0],"#DIRC.",6) == 0) { | |
966 if (token_deal != 3) goto parse_error; | |
967 /* ファイル形式の指定 */ | |
968 FILESEARCH::FILETYPE type; | |
969 char* name = tokens[0]+6; | |
970 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT; | |
971 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT; | |
972 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT; | |
973 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN; | |
974 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; | |
975 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD; | |
976 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR; | |
977 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV; | |
978 else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE; | |
979 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN; | |
980 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */ | |
981 if (tokens[2][0] == 'N') { /* directory */ | |
982 file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]); | |
983 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1])); | |
984 } else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */ | |
985 file_searcher.SetFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2); | |
986 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2)); | |
987 } else goto parse_error; | |
988 goto parse_end; | |
989 } | |
990 if (strncmp(tokens[0],"#ADRC.",6) == 0) { | |
991 if (token_deal != 3) goto parse_error; | |
992 /* ファイル形式の指定 */ | |
993 FILESEARCH::FILETYPE type; | |
994 char* name = tokens[0]+6; | |
995 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT; | |
996 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT; | |
997 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT; | |
998 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN; | |
999 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; | |
1000 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD; | |
1001 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR; | |
1002 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV; | |
1003 else if (strcmp(name, "KOE") == 0) type = FILESEARCH::KOE; | |
1004 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN; | |
1005 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */ | |
1006 if (tokens[2][0] == 'N') { /* directory */ | |
1007 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]); | |
1008 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1])); | |
1009 } else if (tokens[2][0] == 'P' && tokens[2][1] == ':') { /* アーカイブ */ | |
1010 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2); | |
1011 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2)); | |
1012 } else if (tokens[2][0] == 'R' && tokens[2][1] == ':') { /* それ散るアーカイブ */ | |
1013 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_ARC, tokens[2]+2); | |
1014 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2)); | |
1015 } else goto parse_error; | |
1016 goto parse_end; | |
1017 } | |
1018 if (strncmp(tokens[0],"#FOLDNAME.",10) == 0) { | |
1019 if (token_deal != 3) goto parse_error; | |
1020 /* ファイル形式の指定 */ | |
1021 FILESEARCH::FILETYPE type; | |
1022 char* name = tokens[0]+10; | |
1023 if (strcmp(name, "PDT") == 0) type = FILESEARCH::PDT; | |
1024 else if (strcmp(name, "G00") == 0) type = FILESEARCH::PDT; | |
1025 else if (strcmp(name, "GRP") == 0) type = FILESEARCH::PDT; | |
1026 else if (strcmp(name, "TXT") == 0) type = FILESEARCH::SCN; | |
1027 else if (strcmp(name, "ANM") == 0) type = FILESEARCH::ANM; | |
1028 else if (strcmp(name, "ARD") == 0) type = FILESEARCH::ARD; | |
1029 else if (strcmp(name, "CUR") == 0) type = FILESEARCH::CUR; | |
1030 else if (strcmp(name, "WAV") == 0) type = FILESEARCH::WAV; | |
1031 else if (strcmp(name, "BGM") == 0) type = FILESEARCH::BGM; | |
1032 else if (strcmp(name, "GAN") == 0) type = FILESEARCH::GAN; | |
1033 else goto parse_error; /* 他に ALL,ROOT,MID,KOE,BGM。たぶん、存在しない */ | |
1034 if (tokens[2][0] == '0') { /* directory */ | |
1035 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_DIR, tokens[1]); | |
1036 dprintf(("set file directory; type %s, directory %s\n",name,tokens[1])); | |
1037 } else if (tokens[2][0] == '1' && tokens[2][1] == ':') { /* アーカイブ */ | |
1038 file_searcher.AppendFileInformation(type, FILESEARCH::ATYPE_SCN2k, tokens[2]+2); | |
1039 dprintf(("set file archive; type %s, file %s\n",name,tokens[2]+2)); | |
1040 } else goto parse_error; | |
1041 goto parse_end; | |
1042 } | |
1043 if (strcmp(tokens[0], "#CDTRACK") == 0) { | |
1044 if (token_deal != 3) goto parse_error; | |
1045 track_name.AddCDROM(tokens[2], atoi(tokens[1])); | |
1046 dprintf(("Set CDTRACK, name %s, track %d\n",tokens[2], atoi(tokens[1]))); | |
1047 goto parse_end; | |
1048 } | |
1049 if (strcmp(tokens[0], "#DSTRACK") == 0) { | |
1050 /* #DSTRACK=00000000-99999000-00782556="filename" ="name" */ | |
1051 /* #DSTRACK=00000000-99999000-00782556="name" */ | |
1052 /* 第二トークンの3つめのパラメータを得る(繰り返しの時の再生開始位置) */ | |
1053 int start_pt = 0; | |
1054 const char* tk1 = strchr(tokens[1], '-'); | |
1055 const char* tk2 = 0; | |
1056 if (tk1 && *tk1) tk2 = strchr(tk1+1, '-'); | |
1057 if (tk2 && *tk2) start_pt = atoi(tk2+1); | |
1058 if (token_deal == 3) { | |
1059 track_name.AddWave(tokens[2], tokens[2], start_pt); | |
1060 dprintf(("Set Wave track, name %s\n",tokens[2])); | |
1061 } else if (token_deal == 4) { | |
1062 track_name.AddWave(tokens[3], tokens[2], start_pt); | |
1063 dprintf(("Set Wave track, name %s, file %s\n",tokens[3], tokens[2])); | |
1064 } else goto parse_error; | |
1065 goto parse_end; | |
1066 } | |
1067 if (strncmp(tokens[0], "#SE.", 4) == 0) { | |
1068 /* SE.XXX="XXX"=X */ | |
1069 if (token_deal == 2) { | |
1070 track_name.AddSE(atoi(tokens[0]+4), tokens[1]); | |
1071 } else if (token_deal == 3) { | |
1072 if (atoi(tokens[2]) != 0) { | |
1073 track_name.AddSE(atoi(tokens[0]+4), tokens[1]); | |
1074 } | |
1075 } | |
1076 dprintf(("Set SE %d, name %s\n",atoi(tokens[0]+4), tokens[1])); | |
1077 goto parse_end; | |
1078 } | |
1079 /* 設定項目が見つからなかった */ | |
1080 dprintf(("Cannot find configuration name: %s\n",tokens[0])); | |
1081 parse_error: | |
1082 parse_end: | |
1083 line_count++; | |
1084 } | |
1085 delete info; | |
1086 /* デフォルトのオプションを指定する */ | |
1087 // set_game(GetParaStr("#REGNAME"), *this); | |
1088 return true; | |
1089 } | |
1090 | |
1091 TrackName::TrackName(void) { | |
1092 deal = 1; | |
1093 track = new char*[deal]; | |
1094 track_wave = new char*[deal]; | |
1095 track_num = new int[deal]; | |
1096 track_start = new int[deal]; | |
1097 int i; for (i=0; i<deal; i++) track[i] = 0; | |
1098 for (i=0; i<deal; i++) track_wave[i] = 0; | |
1099 se_deal = 10; | |
1100 se_track = new char*[se_deal]; | |
1101 for (i=0; i<se_deal; i++) se_track[i] = 0; | |
1102 } | |
1103 | |
1104 TrackName::~TrackName() { | |
1105 int i; for (i=0; i<deal; i++) { | |
1106 if (track[i] != 0) delete[] track[i]; | |
1107 if (track_wave[i] != 0) delete[] track_wave[i]; | |
1108 } | |
1109 for (i=0; i<se_deal; i++) { | |
1110 if (se_track[i]) delete[] se_track[i]; | |
1111 } | |
1112 delete[] track; | |
1113 delete[] track_wave; | |
1114 delete[] track_num; | |
1115 delete[] track_start; | |
1116 delete[] se_track; | |
1117 } | |
1118 void TrackName::Expand(void) { | |
1119 int new_deal = deal * 2; | |
1120 int* new_track_num = new int[new_deal]; | |
1121 int* new_track_start = new int[new_deal]; | |
1122 char** new_track = new char*[new_deal]; | |
1123 char** new_track_wave = new char*[new_deal]; | |
1124 int i; for (i=0; i<deal; i++) { | |
1125 new_track_num[i] = track_num[i]; | |
1126 new_track_start[i] = track_start[i]; | |
1127 new_track[i] = track[i]; | |
1128 new_track_wave[i] = track_wave[i]; | |
1129 } | |
1130 for (; i<new_deal; i++) { | |
1131 new_track_num[i] = 0; | |
1132 new_track_start[i] = 0; | |
1133 new_track[i] = 0; | |
1134 new_track_wave[i] = 0; | |
1135 } | |
1136 deal = new_deal; | |
1137 delete[] track; track = new_track; | |
1138 delete[] track_num; track_num= new_track_num; | |
1139 delete[] track_start; track_start= new_track_start; | |
1140 delete[] track_wave; track_wave = new_track_wave; | |
1141 } | |
1142 void TrackName::ExpandSE(int n) { | |
1143 if (n < 0) return; | |
1144 n += 10; | |
1145 if (se_deal >= n) return; | |
1146 char** new_se = new char*[n]; | |
1147 int i; for (i=0; i<se_deal; i++) new_se[i] = se_track[i]; | |
1148 for (; i<n; i++) new_se[i] = 0; | |
1149 delete[] se_track; | |
1150 se_deal = n; se_track = new_se; | |
1151 } | |
1152 void TrackName::AddCDROM(char* name, int tk) { | |
1153 if (CDTrack(name) != -1) return; | |
1154 int i; for (i=0; i<deal; i++) { | |
1155 if (track[i] == 0) break; | |
1156 } | |
1157 int num = i; | |
1158 if (i == deal) Expand(); | |
1159 track[num] = new char[strlen(name)+1]; | |
1160 for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]); | |
1161 track[num][i] = 0; | |
1162 track_num[num] = tk; | |
1163 } | |
1164 void TrackName::AddWave(char* name, char* file, int pt) { | |
1165 if (CDTrack(name) != -1) return; | |
1166 int i; for (i=0; i<deal; i++) { | |
1167 if (track[i] == 0) break; | |
1168 } | |
1169 int num = i; | |
1170 if (i == deal) Expand(); | |
1171 track_num[num] = 0; | |
1172 track_start[num] = pt; | |
1173 track[num] = new char[strlen(name)+1]; | |
1174 for (i=0; name[i] != 0; i++) track[num][i] = tolower(name[i]); | |
1175 track[num][i] = 0; | |
1176 track_wave[num] = new char[strlen(file)+1]; strcpy(track_wave[num], file); | |
1177 } | |
1178 int TrackName::CDTrack(char* name) { | |
1179 char buf[1024]; | |
1180 int i; | |
1181 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]); | |
1182 buf[i]=0; | |
1183 for (i=0; i<deal; i++) { | |
1184 if (track[i] == 0) return -1; | |
1185 if (strcmp(track[i], buf) == 0) { | |
1186 return track_num[i]; | |
1187 } | |
1188 } | |
1189 return -1; | |
1190 } | |
1191 int TrackName::TrackStart(char* name) { | |
1192 char buf[1024]; | |
1193 int i; | |
1194 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]); | |
1195 buf[i]=0; | |
1196 for (i=0; i<deal; i++) { | |
1197 if (track[i] == 0) return -1; | |
1198 if (strcmp(track[i], buf) == 0) { | |
1199 return track_start[i]; | |
1200 } | |
1201 } | |
1202 return 0; | |
1203 } | |
1204 const char* TrackName::WaveTrack(char* name) { | |
1205 char buf[1024]; | |
1206 int i; | |
1207 for (i=0; name[i]!=0; i++) buf[i]=tolower(name[i]); | |
1208 buf[i]=0; | |
1209 for (i=0; i<deal; i++) { | |
1210 if (track[i] == 0) return 0; | |
1211 if (strcmp(track[i], buf) == 0) { | |
1212 return track_wave[i]; | |
1213 } | |
1214 } | |
1215 return 0; | |
1216 } | |
1217 const char* TrackName::SETrack(int n) { | |
1218 if (n < 0 || n >= se_deal) return 0; | |
1219 return se_track[n]; | |
1220 } | |
1221 void TrackName::AddSE(int n, char* file) { | |
1222 if (se_deal <= n) ExpandSE(n); | |
1223 if (se_track[n]) delete[] se_track[n]; | |
1224 se_track[n] = new char[strlen(file)+1]; | |
1225 strcpy(se_track[n], file); | |
1226 } | |
1227 |