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.
- package/batch/command.txt +15 -15
- package/core/package.json +4 -1
- package/core/src/nako3.mjs +5 -2
- package/core/src/nako3.mts +5 -4
- package/core/src/nako_core_version.mjs +2 -2
- package/core/src/nako_core_version.mts +2 -2
- package/core/src/nako_from_dncl.mjs +0 -254
- package/core/src/nako_from_dncl.mts +0 -247
- package/core/src/nako_from_dncl2.mjs +313 -0
- package/core/src/nako_from_dncl2.mts +299 -0
- package/core/src/nako_gen.mjs +1 -1
- package/core/src/nako_gen.mts +1 -1
- package/core/src/nako_indent_inline.mjs +152 -34
- package/core/src/nako_indent_inline.mts +132 -34
- package/core/src/nako_lex_rules.mjs +3 -2
- package/core/src/nako_lex_rules.mts +3 -2
- package/core/src/nako_lexer.mjs +27 -3
- package/core/src/nako_lexer.mts +27 -3
- package/core/src/nako_parser3.mjs +45 -14
- package/core/src/nako_parser3.mts +40 -18
- package/core/src/nako_prepare.mjs +13 -0
- package/core/src/nako_prepare.mts +11 -0
- package/core/src/nako_tools.mjs +53 -0
- package/core/src/nako_tools.mts +46 -0
- package/core/test/basic_test.mjs +16 -8
- package/core/test/dncl2_test.mjs +207 -0
- package/core/test/flow_test.mjs +42 -0
- package/core/test/indent_test.mjs +74 -50
- package/core/test/inline_indent_test.mjs +47 -0
- package/core/test/lex_test.mjs +1 -0
- package/demo/js/a.js +30 -0
- package/package.json +1 -1
- package/release/_hash.txt +16 -16
- package/release/_script-tags.txt +14 -14
- package/release/nako_gen_async.js +1 -1
- package/release/stats.json +1 -1
- package/release/version.js +1 -1
- package/release/wnako3.js +1 -1
- package/release/wnako3webworker.js +1 -1
- package/src/cnako3mod.mjs +43 -33
- package/src/cnako3mod.mts +45 -35
- package/src/nako_version.mjs +2 -2
- package/src/nako_version.mts +2 -2
- 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
|
+
}
|