nadesiko3 3.3.63 → 3.3.64

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (44) hide show
  1. package/batch/command.txt +15 -15
  2. package/core/package.json +4 -1
  3. package/core/src/nako3.mjs +5 -2
  4. package/core/src/nako3.mts +5 -4
  5. package/core/src/nako_core_version.mjs +2 -2
  6. package/core/src/nako_core_version.mts +2 -2
  7. package/core/src/nako_from_dncl.mjs +0 -254
  8. package/core/src/nako_from_dncl.mts +0 -247
  9. package/core/src/nako_from_dncl2.mjs +313 -0
  10. package/core/src/nako_from_dncl2.mts +299 -0
  11. package/core/src/nako_gen.mjs +1 -1
  12. package/core/src/nako_gen.mts +1 -1
  13. package/core/src/nako_indent_inline.mjs +152 -34
  14. package/core/src/nako_indent_inline.mts +132 -34
  15. package/core/src/nako_lex_rules.mjs +3 -2
  16. package/core/src/nako_lex_rules.mts +3 -2
  17. package/core/src/nako_lexer.mjs +27 -3
  18. package/core/src/nako_lexer.mts +27 -3
  19. package/core/src/nako_parser3.mjs +45 -14
  20. package/core/src/nako_parser3.mts +40 -18
  21. package/core/src/nako_prepare.mjs +13 -0
  22. package/core/src/nako_prepare.mts +11 -0
  23. package/core/src/nako_tools.mjs +53 -0
  24. package/core/src/nako_tools.mts +46 -0
  25. package/core/test/basic_test.mjs +16 -8
  26. package/core/test/dncl2_test.mjs +207 -0
  27. package/core/test/flow_test.mjs +42 -0
  28. package/core/test/indent_test.mjs +74 -50
  29. package/core/test/inline_indent_test.mjs +47 -0
  30. package/core/test/lex_test.mjs +1 -0
  31. package/demo/js/a.js +30 -0
  32. package/package.json +1 -1
  33. package/release/_hash.txt +16 -16
  34. package/release/_script-tags.txt +14 -14
  35. package/release/nako_gen_async.js +1 -1
  36. package/release/stats.json +1 -1
  37. package/release/version.js +1 -1
  38. package/release/wnako3.js +1 -1
  39. package/release/wnako3webworker.js +1 -1
  40. package/src/cnako3mod.mjs +43 -33
  41. package/src/cnako3mod.mts +45 -35
  42. package/src/nako_version.mjs +2 -2
  43. package/src/nako_version.mts +2 -2
  44. package/src/plugin_browser_canvas.mjs +5 -0
@@ -2,7 +2,6 @@
2
2
  * DNCLに対応する構文
3
3
  */
4
4
  // import { NakoIndentError } from './nako_errors.mjs'
5
- import { NakoPrepare, checkNakoMode } from './nako_prepare.mjs'
6
5
  import { Token, NewEmptyToken } from './nako_types.mjs'
7
6
  import { joinTokenLines, splitTokens } from './nako_indent_inline.mjs'
8
7
 
@@ -177,7 +176,6 @@ export function convertDNCL (tokens: Token[]): Token[] {
177
176
  }
178
177
  j++
179
178
  }
180
- //console.log('@@@', line)
181
179
  }
182
180
 
183
181
  // 最後に単純な置換を行う
@@ -188,15 +186,7 @@ export function convertDNCL (tokens: Token[]): Token[] {
188
186
  t.type = a[0]
189
187
  t.value = a[1]
190
188
  }
191
- // console.log(t)
192
189
  }
193
- /*
194
- // 表示
195
- lines.map(line => {
196
- console.log(line.map(t => t.type + '_' + ('' + t.value).replace('\n', '') + t.josi).join(' | '))
197
- })
198
- console.log('===')
199
- */
200
190
  tokens = joinTokenLines(lines)
201
191
  return tokens
202
192
  }
@@ -268,243 +258,6 @@ function useDNCLmode (tokens: Token[]): boolean {
268
258
  return false
269
259
  }
270
260
 
271
- /**
272
- * DNCLのソースコードをなでしこに変換する
273
- * @param src
274
- * @param filename
275
- * @returns converted soruce
276
- */
277
- export function convertDNCLfromCode (src: string, filename: string): string {
278
- // 改行を合わせる
279
- src = src.replace(/(\r\n|\r)/g, '\n')
280
- // 「!DNCLモード」を使うかチェック
281
- if (!checkNakoMode(src, DNCL_KEYWORDS)) { return src }
282
- const result = dncl2nako(src, filename)
283
- // console.log("=====\n" + result)
284
- // process.exit()
285
- return result
286
- }
287
-
288
- /**
289
- * make space string
290
- * @param {number} n
291
- */
292
- function makeSpaces (n: number): string {
293
- let s = ''
294
- for (let i = 0; i < n; i++) {
295
- s += ' '
296
- }
297
- return s
298
- }
299
-
300
- /**
301
- * DNCLからなでしこに変換する(判定なし)
302
- * @param {string} src
303
- * @param {string} filename
304
- * @returns {string} converted source
305
- */
306
- function dncl2nako (src: string, filename: string): string {
307
- // 全角半角を統一
308
- src = conv2half(src)
309
- // 行頭の「|」はインデントを表す記号なので無視する
310
- // 後判定の「繰り返し,」を「後判定で繰り返す」に置換する
311
- const a = src.split('\n')
312
- for (let i = 0; i < a.length; i++) {
313
- // インデントを消す
314
- let line = a[i]
315
- a[i] = line.replace(/^(\s*[|\s]+)(.*$)/, (m0, m1, m2) => {
316
- return makeSpaces(m1.length) + m2
317
- })
318
- line = a[i]
319
- // 後判定の繰り返しの実装のため
320
- const line2 = line.replace(/^\s+/, '').replace(/\s+$/, '')
321
- if (line2 === '繰り返し,' || line2 === '繰り返し') {
322
- a[i] = '後判定で繰り返し'
323
- }
324
- const r = line.match(/^\s*を,?(.+)になるまで(繰り返す|実行する)/)
325
- if (r) {
326
- a[i] = `ここまで、(${r[1]})になるまでの間`
327
- continue
328
- }
329
- // 『もしj>hakosuならばhakosu←jを実行する』のような単文のもし文
330
- const rif = line.match(/^もし(.+)を実行する(。|.)*/)
331
- if (rif) {
332
- const sent = dncl2nako(rif[1], filename)
333
- a[i] = `もし、${sent};`
334
- continue
335
- }
336
- // 'のすべての値を0にする'
337
- // 'のすべての要素を0にする'
338
- // 'のすべての要素に0を代入する'
339
- const rall = line.match(/^(.+?)のすべての(要素|値)(を|に)(.+?)(にする|を代入)/)
340
- if (rall) {
341
- const varname = rall[1]
342
- const v = rall[4]
343
- a[i] = `${varname} = [${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v},${v}]`
344
- continue
345
- }
346
- }
347
- src = a.join('\n')
348
- // ---------------------------------
349
- // 置換開始
350
- // ---------------------------------
351
- // 単純置換リスト
352
- const simpleConvList: {[key: string]: string} = {
353
- 'を実行する': 'ここまで',
354
- 'を実行し,そうでなくもし': '違えば、もし',
355
- 'を実行し,そうでなくもし': '違えば、もし',
356
- 'を実行し、そうでなくもし': '違えば、もし',
357
- 'を実行し,そうでなければ': '違えば',
358
- 'を実行し,そうでなければ': '違えば',
359
- 'を実行し、そうでなければ': '違えば',
360
- 'を繰り返す': 'ここまで',
361
- '改行なしで表示': '連続無改行表示',
362
- 'ずつ増やしながら': 'ずつ増やし繰り返す',
363
- 'ずつ減らしながら': 'ずつ減らし繰り返す',
364
- '二進で表示': '二進表示',
365
- 'でないならば': 'でなければ'
366
- }
367
- const nextChar = () => {
368
- const ch = src.charAt(0)
369
- src = src.substring(1)
370
- return ch
371
- }
372
- // 文字列を判定するフラグ
373
- let flagStr = false
374
- let poolStr = ''
375
- let endStr = ''
376
- // 結果
377
- let result = ''
378
- while (src !== '') {
379
- // 代入記号を変更
380
- const ch = src.charAt(0)
381
- if (flagStr) {
382
- if (ch === endStr) {
383
- result += poolStr + endStr
384
- poolStr = ''
385
- flagStr = false
386
- nextChar()
387
- continue
388
- }
389
- poolStr += nextChar()
390
- continue
391
- }
392
- // 文字列?
393
- if (ch === '"') {
394
- flagStr = true
395
- endStr = '"'
396
- poolStr = nextChar()
397
- continue
398
- }
399
- if (ch === '「') {
400
- flagStr = true
401
- endStr = '」'
402
- poolStr = nextChar()
403
- continue
404
- }
405
- if (ch === '『') {
406
- flagStr = true
407
- endStr = '』'
408
- poolStr = nextChar()
409
- continue
410
- }
411
- // 空白を飛ばす
412
- if (ch === ' ' || ch === ' ' || ch === '\t') {
413
- result += nextChar()
414
- continue
415
- }
416
- // 表示を連続表示に置き換える
417
- const ch3 = src.substring(0, 3)
418
- if (ch3 === 'を表示') {
419
- result += 'を連続表示'
420
- src = src.substring(3)
421
- continue
422
- }
423
- if (src.substring(0, 4) === 'を 表示') {
424
- result += 'を連続表示'
425
- src = src.substring(4)
426
- continue
427
- }
428
- // 乱数を乱数範囲に置き換える
429
- if (src.substring(0, 2) === '乱数' && src.substring(0, 4) !== '乱数範囲') {
430
- result += '乱数範囲'
431
- src = src.substring(2)
432
- continue
433
- }
434
- // 増やす・減らすの前に「だけ」を追加する #1149
435
- if (ch3 === '増やす' || ch3 === '減らす') {
436
- if (result.substring(result.length - 2) !== 'だけ') {
437
- result += 'だけ'
438
- }
439
- result += ch3
440
- src = src.substring(3)
441
- }
442
- // 一覧から単純な変換
443
- let flag = false
444
- for (const key in simpleConvList) {
445
- const srcKey = src.substring(0, key.length)
446
- if (srcKey === key) {
447
- result += simpleConvList[key]
448
- src = src.substring(key.length)
449
- flag = true
450
- break
451
- }
452
- }
453
- if (flag) { continue }
454
-
455
- // 1文字削る
456
- result += nextChar()
457
- }
458
- return result
459
- }
460
-
461
- /**
462
- * 半角に変換
463
- * @param {String} src
464
- * @returns {string} converted source
465
- */
466
- function conv2half (src: string): string {
467
- const prepare = NakoPrepare.getInstance() // `※`, `//`, `/*` といったパターン全てに対応するために必要
468
- // 全角半角の統一
469
- let result = ''
470
- let flagStr = false
471
- let flagStrClose = ''
472
- for (let i = 0; i < src.length; i++) {
473
- const c = src.charAt(i)
474
- let cHalf = prepare.convert1ch(c)
475
- if (flagStr) {
476
- if (cHalf === flagStrClose) {
477
- flagStr = false
478
- flagStrClose = ''
479
- result += cHalf
480
- continue
481
- }
482
- result += c
483
- continue
484
- }
485
- if (cHalf === '「') {
486
- flagStr = true
487
- flagStrClose = '」'
488
- result += cHalf
489
- continue
490
- }
491
- if (cHalf === '"') {
492
- flagStr = true
493
- flagStrClose = '"'
494
- result += cHalf
495
- continue
496
- }
497
- // 単純な置き換えはここでやってしまう
498
- // 配列記号の { ... } を [ ... ] に置換
499
- if (cHalf === '{') { cHalf = '[' }
500
- if (cHalf === '}') { cHalf = ']' }
501
- if (cHalf === '←') { cHalf = '=' }
502
- if (cHalf === '÷') { cHalf = '÷÷' } // #1152
503
- result += cHalf
504
- }
505
- return result
506
- }
507
-
508
261
  export const NakoDncl = {
509
262
  convert: convertDNCL
510
263
  }
@@ -0,0 +1,313 @@
1
+ /**
2
+ * DNCL ver2 に対応する構文
3
+ */
4
+ // import { NakoIndentError } from './nako_errors.mjs'
5
+ import { NewEmptyToken } from './nako_types.mjs';
6
+ import { joinTokenLines, splitTokens } from './nako_indent_inline.mjs';
7
+ import { newToken, debugTokens } from './nako_tools.mjs';
8
+ const IS_DEBUG = false;
9
+ const DNCL_ARRAY_INIT_COUNT = 30;
10
+ // DNCL2モードのキーワード
11
+ const DNCL2_KEYWORDS = ['!DNCL2モード', '💡DNCL2モード', '!DNCL2', '💡DNCL2'];
12
+ // 単純な置換チェック
13
+ const DNCL_SIMPLES = {
14
+ '←:←': ['eq', '='],
15
+ '÷:÷': ['÷÷', '÷÷'],
16
+ '{:{': ['[', '['],
17
+ '}:}': [']', ']'],
18
+ 'word:and': ['and', 'かつ'],
19
+ 'word:or': ['or', 'または'],
20
+ 'word:乱数': ['word', '乱数範囲'],
21
+ 'word:表示': ['word', '連続表示']
22
+ };
23
+ /**
24
+ * DNCLのソースコードをなでしこに変換する
25
+ */
26
+ export function convertDNCL2(tokens) {
27
+ if (!useDNCL2mode(tokens)) {
28
+ return tokens;
29
+ }
30
+ // 一行ずつに分ける
31
+ const lines = splitTokens(tokens, 'eol');
32
+ for (let i = 0; i < lines.length; i++) {
33
+ const line = lines[i];
34
+ if (line.length <= 1) {
35
+ continue;
36
+ } // 空行は飛ばす
37
+ // --- 制御構文の変換 ---
38
+ // もし(条件)でないならば → もし(条件)でなければ
39
+ const nai = findTokens(line, ['word:ない']);
40
+ if (nai >= 1) {
41
+ const tt = line[nai];
42
+ if (tt.josi === 'ならば') {
43
+ line[nai - 1].josi = 'でなければ';
44
+ line.splice(nai, 1);
45
+ }
46
+ }
47
+ // そうでなければ(そう|でなければ) → 違えば
48
+ for (let ni = 0; ni < line.length; ni++) {
49
+ const t = line[ni];
50
+ if (t.value === 'そう' && t.josi === 'でなければ') {
51
+ t.type = '違えば';
52
+ t.value = '違えば';
53
+ t.josi = '';
54
+ }
55
+ }
56
+ // 'を実行し,そうでなければ': '違えば',
57
+ for (;;) {
58
+ const ni = findTokens(line, ['word:を実行', 'comma:,', 'word:そう']);
59
+ if (ni < 0) {
60
+ break;
61
+ }
62
+ const sou = line[ni + 2];
63
+ if (sou.josi === 'でなければ') {
64
+ sou.type = '違えば';
65
+ sou.value = '違えば';
66
+ sou.josi = '';
67
+ line.splice(ni, 3, sou);
68
+ continue;
69
+ }
70
+ else if (sou.josi === 'で') {
71
+ const nakumosi = line[ni + 3];
72
+ if (nakumosi.value.substring(0, 4) === 'なくもし') {
73
+ sou.type = '違えば';
74
+ sou.value = '違えば';
75
+ sou.josi = '';
76
+ line.splice(ni, 3, sou);
77
+ if (nakumosi.value.length > 4) {
78
+ const nakumosiTudukiStr = nakumosi.value.substring(4);
79
+ const nakumosiToken = NewEmptyToken('word', nakumosiTudukiStr, nakumosi.indent, nakumosi.line, nakumosi.file);
80
+ if (nakumosiTudukiStr.match(/^\d/)) {
81
+ nakumosiToken.type = 'number';
82
+ }
83
+ line.splice(ni + 2, 0, nakumosiToken);
84
+ nakumosi.value = nakumosi.value.substring(0, 4);
85
+ }
86
+ nakumosi.type = 'もし';
87
+ nakumosi.value = 'もし';
88
+ nakumosi.josi = '';
89
+ continue;
90
+ }
91
+ }
92
+ break;
93
+ }
94
+ // Iを1から100まで1(ずつ)|増やしな(が)|ら
95
+ for (;;) {
96
+ const ni = findTokens(line, ['word:増', 'word:ら']);
97
+ if (ni < 0) {
98
+ break;
99
+ }
100
+ const fu = line[ni];
101
+ fu.type = 'word';
102
+ fu.value = '増繰返';
103
+ fu.josi = '';
104
+ line.splice(ni, 2, fu);
105
+ }
106
+ // Iを1から100まで1(ずつ)|増やしな(が)|ら
107
+ for (;;) {
108
+ const ni = findTokens(line, ['word:減', 'word:ら']);
109
+ if (ni < 0) {
110
+ break;
111
+ }
112
+ const fu = line[ni];
113
+ fu.type = 'word';
114
+ fu.value = '減繰返';
115
+ fu.josi = '';
116
+ line.splice(ni, 2, fu);
117
+ }
118
+ // Iを1から100まで1(ずつ)|増やしな(が)|ら繰り返(す)
119
+ for (;;) {
120
+ const ni = findTokens(line, ['word:増', 'word:ら繰り返']);
121
+ if (ni < 0) {
122
+ break;
123
+ }
124
+ const fu = line[ni];
125
+ fu.type = 'word';
126
+ fu.value = '増繰返';
127
+ fu.josi = '';
128
+ line.splice(ni, 2, fu);
129
+ }
130
+ // Iを1から100まで1(ずつ)|増やしな(が)|ら繰り返す
131
+ for (;;) {
132
+ const ni = findTokens(line, ['word:減', 'word:ら繰り返']);
133
+ if (ni < 0) {
134
+ break;
135
+ }
136
+ const fu = line[ni];
137
+ fu.type = 'word';
138
+ fu.value = '減繰返';
139
+ fu.josi = '';
140
+ line.splice(ni, 2, fu);
141
+ }
142
+ // --- 配列変数周りの変換 ---
143
+ for (let i = 0; i < line.length; i++) {
144
+ // 配列|Hindoの|すべての|(要素に|値に)|10を|代入する
145
+ if (tokenEq([['word:配列', 'word:配列変数'], 'word', 'word:すべて', ['word:要素', 'word:値'], '*', 'word:代入'], line, i)) {
146
+ const varToken = line[i + 1];
147
+ varToken.josi = '';
148
+ const valToken = line[i + 4];
149
+ valToken.josi = '';
150
+ line.splice(i, 6, varToken, newToken('eq', '=', varToken), newToken('word', '掛'), newToken('(', '('), newToken('[', '['), valToken, newToken(']', ']'), newToken('comma', ','), newToken('number', DNCL_ARRAY_INIT_COUNT), newToken(')', ')'));
151
+ i += 6; // skip
152
+ }
153
+ // Hensuの|すべての|(要素を|値を)|0に|する
154
+ if (tokenEq(['word', 'word:すべて', ['word:要素', 'word:値'], ['number', 'string', 'word'], 'word:する'], line, i)) {
155
+ const varToken = line[i];
156
+ varToken.josi = '';
157
+ const valToken = line[i + 3];
158
+ valToken.josi = '';
159
+ line.splice(i, 5, varToken, newToken('eq', '=', varToken), newToken('word', '掛'), newToken('(', '('), newToken('[', '['), valToken, newToken(']', ']'), newToken('comma', ','), newToken('number', DNCL_ARRAY_INIT_COUNT), newToken(')', ')'));
160
+ }
161
+ // 配列変数 | xxを | 初期化する
162
+ if (tokenEq([['word:配列変数', 'word:配列'], 'word', 'word:初期化'], line, i)) {
163
+ const varToken = line[i + 1];
164
+ varToken.josi = '';
165
+ line.splice(i, 3, varToken, newToken('eq', '=', varToken), newToken('word', '掛'), newToken('(', '('), newToken('[', '['), newToken('number', 0), newToken(']', ']'), newToken('comma', ','), newToken('number', DNCL_ARRAY_INIT_COUNT), newToken(')', ')'));
166
+ }
167
+ }
168
+ // --- その他の変換 ---
169
+ // 二進で表示 (255) → 二進表示(255)
170
+ for (;;) {
171
+ const ni = findTokens(line, ['word:二進', 'word:表示']);
172
+ if (ni < 0) {
173
+ break;
174
+ }
175
+ line[ni].value = '二進表示';
176
+ line[ni].josi = '';
177
+ line.splice(ni + 1, 1);
178
+ }
179
+ // '改行なしで表示' → '連続無改行表示'
180
+ for (;;) {
181
+ const ni = findTokens(line, ['word:改行', 'word:表示']);
182
+ if (ni < 0) {
183
+ break;
184
+ }
185
+ // ここ「改行なしで表示」でも「改行ありで表示」でも同じになってしまう
186
+ // なでしこの制限のため仕方なし
187
+ // 「改行ありで表示」は今のところDNCLに存在しないので無視する
188
+ // もし将来的に区別が必要なら、プリプロセス処理でマクロ的に置換処理を行うことで対応できると思う
189
+ const t = line[ni];
190
+ t.value = '連続無改行表示';
191
+ t.josi = '';
192
+ line.splice(ni + 1, 1);
193
+ }
194
+ // 一つずつチェック
195
+ let j = 0;
196
+ while (j < line.length) {
197
+ const t = line[j];
198
+ // 減と増の分割
199
+ if (t.type === 'word' && t.value.length >= 2) {
200
+ const c = t.value.charAt(t.value.length - 1);
201
+ if (c === '減' || c === '増') {
202
+ t.value = t.value.substring(0, t.value.length - 1);
203
+ t.josi = 'だけ';
204
+ line.splice(j + 1, 0, NewEmptyToken('word', c, t.indent, t.line, t.file));
205
+ }
206
+ j++;
207
+ continue;
208
+ }
209
+ j++;
210
+ }
211
+ }
212
+ // 最後に単純な置換を行う
213
+ for (let i = 0; i < tokens.length; i++) {
214
+ const t = tokens[i];
215
+ const a = DNCL_SIMPLES[t.type + ':' + t.value];
216
+ if (a !== undefined) {
217
+ t.type = a[0];
218
+ t.value = a[1];
219
+ }
220
+ }
221
+ tokens = joinTokenLines(lines);
222
+ if (IS_DEBUG) {
223
+ console.log('@@@---DNCL:tokens---');
224
+ console.log(debugTokens(tokens));
225
+ console.log('@@@/---DNCL:tokens---');
226
+ }
227
+ return tokens;
228
+ }
229
+ /**
230
+ * トークンが合致するかを確認する
231
+ * @param typeValues ['word:それ']のようなタイプ名と値の配列/'*'でワイルドカードが使える/":"がなればタイプだけ確認/配列で選択
232
+ * @param lines 差し替え
233
+ * @param fromIndex 検索場所
234
+ * @returns 合致したかどうか
235
+ */
236
+ function tokenEq(typeValues, lines, fromIndex) {
237
+ const check = (pattern, t) => {
238
+ if (pattern instanceof Array) {
239
+ for (let i = 0; i < pattern.length; i++) {
240
+ if (check(pattern[i], t)) {
241
+ return true;
242
+ }
243
+ }
244
+ return false;
245
+ }
246
+ if (pattern === '*') {
247
+ return true;
248
+ }
249
+ if (pattern.indexOf(':') < 0) {
250
+ if (pattern === t.type) {
251
+ return true;
252
+ }
253
+ else {
254
+ return false;
255
+ }
256
+ }
257
+ const tv = `${t.type}:${t.value}`;
258
+ if (pattern === tv) {
259
+ return true;
260
+ }
261
+ return false;
262
+ };
263
+ for (let i = 0; i < typeValues.length; i++) {
264
+ const idx = i + fromIndex;
265
+ if (idx >= lines.length) {
266
+ return false;
267
+ }
268
+ const pat = typeValues[i];
269
+ const t = lines[idx];
270
+ if (!check(pat, t)) {
271
+ return false;
272
+ }
273
+ }
274
+ return true;
275
+ }
276
+ function findTokens(tokens, findTypeValue) {
277
+ const findA = findTypeValue.map(s => s.split(':'));
278
+ for (let i = 0; i < tokens.length; i++) {
279
+ let flag = true;
280
+ for (let j = 0; j < findA.length; j++) {
281
+ const f = findA[j];
282
+ const idx = i + j;
283
+ if (idx >= tokens.length) {
284
+ return -1;
285
+ }
286
+ if (tokens[idx].type === f[0] && tokens[idx].value === f[1]) {
287
+ continue;
288
+ }
289
+ else {
290
+ flag = false;
291
+ break;
292
+ }
293
+ }
294
+ if (flag) {
295
+ return i;
296
+ }
297
+ }
298
+ return -1;
299
+ }
300
+ function useDNCL2mode(tokens) {
301
+ // 先頭の100語調べる
302
+ for (let i = 0; i < tokens.length; i++) {
303
+ if (i > 100) {
304
+ break;
305
+ }
306
+ const t = tokens[i];
307
+ if (t.type === 'line_comment' && DNCL2_KEYWORDS.indexOf(t.value) >= 0) {
308
+ t.type = 'DNCL2モード';
309
+ return true;
310
+ }
311
+ }
312
+ return false;
313
+ }