nadesiko3 3.3.48 → 3.3.49

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 (105) hide show
  1. package/core/.editorconfig +6 -0
  2. package/core/.eslintrc.cjs +33 -0
  3. package/core/.github/dependabot.yml +7 -0
  4. package/core/.github/workflows/nodejs.yml +37 -0
  5. package/core/.github/workflows/super-linter.yml +61 -0
  6. package/core/.github/workflows/textlint.yml +199 -0
  7. package/core/LICENSE +21 -0
  8. package/core/README.md +66 -0
  9. package/core/batch/build_nako_version.nako3 +42 -0
  10. package/core/command/snako.mjs +105 -0
  11. package/core/command/snako.mts +116 -0
  12. package/core/index.mjs +21 -0
  13. package/core/index.mts +21 -0
  14. package/core/package.json +47 -0
  15. package/core/sample/hello.nako3 +7 -0
  16. package/core/sample/hoge.mjs +4 -0
  17. package/core/sample/hoge.mts +6 -0
  18. package/core/src/nako3.mjs +858 -0
  19. package/core/src/nako3.mts +967 -0
  20. package/core/src/nako_colors.mjs +78 -0
  21. package/core/src/nako_colors.mts +86 -0
  22. package/core/src/nako_core_version.mjs +8 -0
  23. package/core/src/nako_core_version.mts +19 -0
  24. package/core/src/nako_csv.mjs +185 -0
  25. package/core/src/nako_csv.mts +188 -0
  26. package/core/src/nako_errors.mjs +173 -0
  27. package/core/src/nako_errors.mts +197 -0
  28. package/core/src/nako_from_dncl.mjs +255 -0
  29. package/core/src/nako_from_dncl.mts +250 -0
  30. package/core/src/nako_gen.mjs +1648 -0
  31. package/core/src/nako_gen.mts +1719 -0
  32. package/core/src/nako_gen_async.mjs +1659 -0
  33. package/core/src/nako_gen_async.mts +1732 -0
  34. package/core/src/nako_global.mjs +107 -0
  35. package/core/src/nako_global.mts +138 -0
  36. package/core/src/nako_indent.mjs +445 -0
  37. package/core/src/nako_indent.mts +492 -0
  38. package/core/src/nako_josi_list.mjs +38 -0
  39. package/core/src/nako_josi_list.mts +45 -0
  40. package/core/src/nako_lex_rules.mjs +253 -0
  41. package/core/src/nako_lex_rules.mts +260 -0
  42. package/core/src/nako_lexer.mjs +609 -0
  43. package/core/src/nako_lexer.mts +612 -0
  44. package/core/src/nako_logger.mjs +199 -0
  45. package/core/src/nako_logger.mts +232 -0
  46. package/core/src/nako_parser3.mjs +2439 -0
  47. package/core/src/nako_parser3.mts +2195 -0
  48. package/core/src/nako_parser_base.mjs +370 -0
  49. package/core/src/nako_parser_base.mts +370 -0
  50. package/core/src/nako_parser_const.mjs +37 -0
  51. package/core/src/nako_parser_const.mts +37 -0
  52. package/core/src/nako_prepare.mjs +304 -0
  53. package/core/src/nako_prepare.mts +315 -0
  54. package/core/src/nako_reserved_words.mjs +38 -0
  55. package/core/src/nako_reserved_words.mts +38 -0
  56. package/core/src/nako_source_mapping.mjs +207 -0
  57. package/core/src/nako_source_mapping.mts +262 -0
  58. package/core/src/nako_test.mjs +37 -0
  59. package/core/src/nako_types.mjs +25 -0
  60. package/core/src/nako_types.mts +151 -0
  61. package/core/src/plugin_csv.mjs +49 -0
  62. package/core/src/plugin_csv.mts +50 -0
  63. package/core/src/plugin_math.mjs +328 -0
  64. package/core/src/plugin_math.mts +326 -0
  65. package/core/src/plugin_promise.mjs +91 -0
  66. package/core/src/plugin_promise.mts +91 -0
  67. package/core/src/plugin_system.mjs +2832 -0
  68. package/core/src/plugin_system.mts +2690 -0
  69. package/core/src/plugin_test.mjs +34 -0
  70. package/core/src/plugin_test.mts +34 -0
  71. package/core/test/array_test.mjs +34 -0
  72. package/core/test/basic_test.mjs +344 -0
  73. package/core/test/calc_test.mjs +140 -0
  74. package/core/test/core_module_test.mjs +23 -0
  75. package/core/test/debug_test.mjs +16 -0
  76. package/core/test/dncl_test.mjs +94 -0
  77. package/core/test/error_message_test.mjs +210 -0
  78. package/core/test/error_test.mjs +16 -0
  79. package/core/test/flow_test.mjs +373 -0
  80. package/core/test/func_call.mjs +160 -0
  81. package/core/test/func_test.mjs +149 -0
  82. package/core/test/indent_test.mjs +364 -0
  83. package/core/test/lex_test.mjs +168 -0
  84. package/core/test/literal_test.mjs +73 -0
  85. package/core/test/nako_lexer_test.mjs +35 -0
  86. package/core/test/nako_logger_test.mjs +76 -0
  87. package/core/test/nako_logger_test.mts +78 -0
  88. package/core/test/plugin_csv_test.mjs +38 -0
  89. package/core/test/plugin_promise_test.mjs +18 -0
  90. package/core/test/plugin_system_test.mjs +630 -0
  91. package/core/test/prepare_test.mjs +96 -0
  92. package/core/test/re_test.mjs +22 -0
  93. package/core/test/side_effects_test.mjs +92 -0
  94. package/core/test/variable_scope_test.mjs +149 -0
  95. package/core/tsconfig.json +101 -0
  96. package/package.json +4 -2
  97. package/release/_hash.txt +12 -12
  98. package/release/_script-tags.txt +14 -14
  99. package/release/editor.js +1 -1
  100. package/release/stats.json +1 -1
  101. package/release/version.js +1 -1
  102. package/release/wnako3.js +1 -1
  103. package/src/nako_version.mjs +2 -2
  104. package/src/nako_version.mts +2 -2
  105. package/test/async/async_basic_test.mjs +3 -3
@@ -0,0 +1,304 @@
1
+ /**
2
+ * nako_prepare.js
3
+ * 字句解析の前の前処理。全角文字を半角文字に変換するのが主な処理。
4
+ * ただし、コメントや文字列の中は変換しないように考慮して変換する。
5
+ */
6
+ class ReplaceHistory {
7
+ constructor(from, to, index) {
8
+ this.from = from;
9
+ this.to = to;
10
+ this.index = index;
11
+ }
12
+ }
13
+ class ConvertResult {
14
+ constructor(text, sourcePosition) {
15
+ this.text = text;
16
+ this.sourcePosition = sourcePosition;
17
+ }
18
+ }
19
+ /**
20
+ * 置換後の位置から置換前の位置へマッピングできる文字列
21
+ */
22
+ export class Replace {
23
+ constructor(code) {
24
+ this.history = [];
25
+ this.code = code;
26
+ }
27
+ getText() {
28
+ return this.code;
29
+ }
30
+ replaceAll(from, to) {
31
+ while (true) {
32
+ const index = this.getText().indexOf(from);
33
+ if (index === -1) {
34
+ break;
35
+ }
36
+ if (from.length !== to.length) {
37
+ this.history.unshift(new ReplaceHistory(from.length, to.length, index));
38
+ }
39
+ this.code = this.code.replace(from, to);
40
+ }
41
+ }
42
+ getSourcePosition(i) {
43
+ // 少し遅い。パース時間1.4秒に対して0.15秒かかる。iが単調増加することを利用して高速化できるはず。
44
+ for (const item of this.history) {
45
+ if (i >= item.index + item.to) { // 置換範囲より後ろ
46
+ i += item.from - item.to;
47
+ }
48
+ else if (item.index <= i && i < item.index + item.to) { // 置換範囲
49
+ // 置換文字列が2文字以上のとき、最後の文字は最後の文字へマップする。それ以外は最初の文字へマップする。
50
+ if (item.to >= 2 && i === item.index + item.to - 1) {
51
+ i = item.index + item.from - 1;
52
+ }
53
+ else {
54
+ i = item.index;
55
+ }
56
+ }
57
+ }
58
+ return i;
59
+ }
60
+ }
61
+ /**
62
+ * 字句解析を行う前に全角文字を半角に揃える
63
+ * [memo]
64
+ * ただし、文字列部分だけは、そのまま全角で出力するようにする
65
+ * for https://github.com/kujirahand/nadesiko3/issues/94
66
+ */
67
+ export class NakoPrepare {
68
+ constructor() {
69
+ // 単純な変換テーブル
70
+ this.convertTable = new Map([
71
+ // ハイフンへの変換
72
+ // 参考) https://hydrocul.github.io/wiki/blog/2014/1101-hyphen-minus-wave-tilde
73
+ // 0x2d: true, // ASCIIのハイフン
74
+ [0x2010, '-'],
75
+ [0x2011, '-'],
76
+ [0x2013, '-'],
77
+ [0x2014, '-'],
78
+ [0x2015, '-'],
79
+ [0x2212, '-'],
80
+ // チルダの変換
81
+ // 0x7e: true,
82
+ [0x02dc, '~'],
83
+ [0x02F7, '~'],
84
+ [0x2053, '~'],
85
+ [0x223c, '~'],
86
+ [0x301c, '~'],
87
+ [0xFF5E, '~'],
88
+ // スペースの変換
89
+ // 参考) http://anti.rosx.net/etc/memo/002_space.html
90
+ // 0x20: true,
91
+ [0x2000, ' '],
92
+ [0x2002, ' '],
93
+ [0x2003, ' '],
94
+ [0x2004, ' '],
95
+ [0x2005, ' '],
96
+ [0x2006, ' '],
97
+ [0x2007, ' '],
98
+ [0x2009, ' '],
99
+ [0x200A, ' '],
100
+ [0x200B, ' '],
101
+ [0x202F, ' '],
102
+ [0x205F, ' '],
103
+ [0x3000, ' '],
104
+ [0x3164, ' '],
105
+ // その他の変換
106
+ [0x09, ' '],
107
+ [0x203B, '#'],
108
+ [0x3002, ';'],
109
+ [0x3010, '['],
110
+ [0x3011, ']'],
111
+ // 読点は「,」に変換する (#877)
112
+ [0x3001, ','],
113
+ [0xFF0C, ','],
114
+ [0x2716, '*'],
115
+ [0x2795, '+'],
116
+ [0x2796, '-'],
117
+ [0x2797, '÷'] // ÷の絵文字 (#1183)
118
+ ]);
119
+ }
120
+ /** 唯一のインスタンスを返す */
121
+ static getInstance() {
122
+ if (!NakoPrepare._instance) {
123
+ NakoPrepare._instance = new NakoPrepare();
124
+ }
125
+ return NakoPrepare._instance;
126
+ }
127
+ // 一文字だけ変換
128
+ /**
129
+ * @param {string} ch
130
+ */
131
+ convert1ch(ch) {
132
+ if (!ch) {
133
+ return '';
134
+ }
135
+ const c = ch.codePointAt(0) || 0;
136
+ // テーブルによる変換
137
+ const c2 = this.convertTable.get(c) || '';
138
+ if (c2) {
139
+ return c2;
140
+ }
141
+ // ASCIIエリア
142
+ if (c < 0x7F) {
143
+ return ch;
144
+ }
145
+ // 全角半角単純変換可能 --- '!' - '~'
146
+ if (c >= 0xFF01 && c <= 0xFF5E) {
147
+ const c2 = c - 0xFEE0;
148
+ return String.fromCodePoint(c2);
149
+ }
150
+ return ch;
151
+ }
152
+ /** convert code */
153
+ convert(code) {
154
+ if (!code) {
155
+ return [];
156
+ }
157
+ const src = new Replace(code);
158
+ // 改行コードを統一
159
+ src.replaceAll('\r\n', '\n');
160
+ src.replaceAll('\r', '\n');
161
+ let flagStr = false; // 文字列リテラル内かどうか
162
+ let flagStr2 = false; // 絵文字による文字列リテラル内かどうか
163
+ let endOfStr = ''; // 文字列リテラルを終了させる記号
164
+ const res = [];
165
+ let left = 0; // 現在処理中の部分文字列の左端の位置
166
+ let str = ''; // 文字列リテラルの値
167
+ // 一文字ずつ全角を半角に置換する
168
+ let i = 0;
169
+ while (i < src.getText().length) {
170
+ const c = src.getText().charAt(i);
171
+ const ch2 = src.getText().substr(i, 2);
172
+ // 文字列のとき
173
+ if (flagStr) {
174
+ if (c === endOfStr) {
175
+ flagStr = false;
176
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)));
177
+ i++;
178
+ left = i;
179
+ continue;
180
+ }
181
+ str += c;
182
+ i++;
183
+ continue;
184
+ }
185
+ // 絵文字制御による文字列のとき
186
+ if (flagStr2) {
187
+ if (ch2 === endOfStr) {
188
+ flagStr2 = false;
189
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)));
190
+ i += 2;
191
+ left = i;
192
+ continue;
193
+ }
194
+ str += c;
195
+ i++;
196
+ continue;
197
+ }
198
+ // 文字列判定
199
+ if (c === '「') {
200
+ res.push(new ConvertResult(c, src.getSourcePosition(left)));
201
+ i++;
202
+ left = i;
203
+ flagStr = true;
204
+ endOfStr = '」';
205
+ str = '';
206
+ continue;
207
+ }
208
+ if (c === '『') {
209
+ res.push(new ConvertResult(c, src.getSourcePosition(left)));
210
+ i++;
211
+ left = i;
212
+ flagStr = true;
213
+ endOfStr = '』';
214
+ str = '';
215
+ continue;
216
+ }
217
+ if (c === '“') {
218
+ res.push(new ConvertResult(c, src.getSourcePosition(left)));
219
+ i++;
220
+ left = i;
221
+ flagStr = true;
222
+ endOfStr = '”';
223
+ str = '';
224
+ continue;
225
+ }
226
+ // JavaScriptの内部的には文字列はUTF-16で扱われてるので charAt を使う場合 絵文字が2文字扱いになる --- #726
227
+ if (ch2 === '🌴' || ch2 === '🌿') {
228
+ res.push(new ConvertResult(ch2, src.getSourcePosition(left)));
229
+ i += 2;
230
+ left = i;
231
+ flagStr2 = true;
232
+ endOfStr = ch2;
233
+ str = '';
234
+ continue;
235
+ }
236
+ const c1 = this.convert1ch(c);
237
+ if (c1 === '"' || c1 === '\'') {
238
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)));
239
+ i++;
240
+ left = i;
241
+ flagStr = true;
242
+ endOfStr = c;
243
+ str = '';
244
+ continue;
245
+ }
246
+ // ラインコメントを飛ばす (#725)
247
+ if (c1 === '#') {
248
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)));
249
+ i++;
250
+ left = i;
251
+ flagStr = true; // 本当はコメントだけど便宜上
252
+ endOfStr = '\n';
253
+ str = '';
254
+ continue;
255
+ }
256
+ // ラインコメントを飛ばす
257
+ if (ch2 === '//' || ch2 === '//') {
258
+ res.push(new ConvertResult('//', src.getSourcePosition(left))); // 強制的に'//'とする
259
+ i += 2;
260
+ left = i;
261
+ flagStr = true;
262
+ endOfStr = '\n';
263
+ str = '';
264
+ continue;
265
+ }
266
+ // 複数行コメント内を飛ばす (#731)
267
+ if (ch2 === '/*') {
268
+ res.push(new ConvertResult(ch2, src.getSourcePosition(left)));
269
+ i += 2;
270
+ left = i;
271
+ flagStr2 = true;
272
+ endOfStr = '*/';
273
+ str = '';
274
+ continue;
275
+ }
276
+ // 変換したものを追加
277
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)));
278
+ i++;
279
+ left = i;
280
+ }
281
+ if (flagStr || flagStr2) {
282
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)));
283
+ }
284
+ return res;
285
+ }
286
+ }
287
+ /** なでしこのソースコードのモード(!インデント構文など)が設定されているか調べる */
288
+ export function checkNakoMode(code, modeNames) {
289
+ // 先頭の256文字について調べる
290
+ code = code.substring(0, 256);
291
+ // 全角半角の揺れを吸収
292
+ code = code.replace(/(!|💡)/, '!');
293
+ // 範囲コメントを削除
294
+ code = code.replace(/\/\*.*?\*\//g, '');
295
+ // 毎文調べる
296
+ const lines = code.split(/[;。\n]/, 30);
297
+ for (let line of lines) {
298
+ line = line.replace(/^\s+/, '').replace(/\s+$/, ''); // trim
299
+ if (modeNames.indexOf(line) >= 0) {
300
+ return true;
301
+ }
302
+ }
303
+ return false;
304
+ }
@@ -0,0 +1,315 @@
1
+ /**
2
+ * nako_prepare.js
3
+ * 字句解析の前の前処理。全角文字を半角文字に変換するのが主な処理。
4
+ * ただし、コメントや文字列の中は変換しないように考慮して変換する。
5
+ */
6
+ class ReplaceHistory {
7
+ from: number;
8
+ to: number;
9
+ index: number;
10
+ constructor (from: number, to: number, index: number) {
11
+ this.from = from
12
+ this.to = to
13
+ this.index = index
14
+ }
15
+ }
16
+
17
+ class ConvertResult {
18
+ public text: string;
19
+ public sourcePosition: number;
20
+ constructor (text: string, sourcePosition: number) {
21
+ this.text = text
22
+ this.sourcePosition = sourcePosition
23
+ }
24
+ }
25
+
26
+ /**
27
+ * 置換後の位置から置換前の位置へマッピングできる文字列
28
+ */
29
+ export class Replace {
30
+ history: ReplaceHistory[];
31
+ private code: string;
32
+ constructor (code: string) {
33
+ this.history = []
34
+ this.code = code
35
+ }
36
+
37
+ getText (): string {
38
+ return this.code
39
+ }
40
+
41
+ replaceAll (from: string, to: string) {
42
+ while (true) {
43
+ const index = this.getText().indexOf(from)
44
+ if (index === -1) {
45
+ break
46
+ }
47
+ if (from.length !== to.length) {
48
+ this.history.unshift(new ReplaceHistory(from.length, to.length, index))
49
+ }
50
+ this.code = this.code.replace(from, to)
51
+ }
52
+ }
53
+
54
+ getSourcePosition (i: number): number {
55
+ // 少し遅い。パース時間1.4秒に対して0.15秒かかる。iが単調増加することを利用して高速化できるはず。
56
+ for (const item of this.history) {
57
+ if (i >= item.index + item.to) { // 置換範囲より後ろ
58
+ i += item.from - item.to
59
+ } else if (item.index <= i && i < item.index + item.to) { // 置換範囲
60
+ // 置換文字列が2文字以上のとき、最後の文字は最後の文字へマップする。それ以外は最初の文字へマップする。
61
+ if (item.to >= 2 && i === item.index + item.to - 1) {
62
+ i = item.index + item.from - 1
63
+ } else {
64
+ i = item.index
65
+ }
66
+ }
67
+ }
68
+ return i
69
+ }
70
+ }
71
+
72
+ /**
73
+ * 字句解析を行う前に全角文字を半角に揃える
74
+ * [memo]
75
+ * ただし、文字列部分だけは、そのまま全角で出力するようにする
76
+ * for https://github.com/kujirahand/nadesiko3/issues/94
77
+ */
78
+ export class NakoPrepare {
79
+ // 唯一のインスタンス
80
+ private static _instance: NakoPrepare;
81
+ /** 唯一のインスタンスを返す */
82
+ public static getInstance (): NakoPrepare {
83
+ if (!NakoPrepare._instance) {
84
+ NakoPrepare._instance = new NakoPrepare()
85
+ }
86
+ return NakoPrepare._instance
87
+ }
88
+
89
+ private convertTable: Map<number, string>;
90
+ private constructor () {
91
+ // 単純な変換テーブル
92
+ this.convertTable = new Map([
93
+ // ハイフンへの変換
94
+ // 参考) https://hydrocul.github.io/wiki/blog/2014/1101-hyphen-minus-wave-tilde
95
+ // 0x2d: true, // ASCIIのハイフン
96
+ [0x2010, '-'], // 別のハイフン
97
+ [0x2011, '-'], // 改行しないハイフン
98
+ [0x2013, '-'], // ENダッシュ
99
+ [0x2014, '-'], // EMダッシュ
100
+ [0x2015, '-'], // 全角のダッシュ
101
+ [0x2212, '-'], // 全角のマイナス
102
+ // チルダの変換
103
+ // 0x7e: true,
104
+ [0x02dc, '~'], // 小さなチルダ
105
+ [0x02F7, '~'], // Modifier Letter Low Tilde
106
+ [0x2053, '~'], // Swung Dash - 辞書のみだし
107
+ [0x223c, '~'], // Tilde Operator: 数学で Similar to
108
+ [0x301c, '~'], // Wave Dash(一般的な波ダッシュ)
109
+ [0xFF5E, '~'], // 全角チルダ
110
+ // スペースの変換
111
+ // 参考) http://anti.rosx.net/etc/memo/002_space.html
112
+ // 0x20: true,
113
+ [0x2000, ' '], // EN QUAD
114
+ [0x2002, ' '], // EN SPACE
115
+ [0x2003, ' '], // EM SPACE
116
+ [0x2004, ' '], // THREE-PER-EM SPACE
117
+ [0x2005, ' '], // FOUR-PER-EM SPACE
118
+ [0x2006, ' '], // SIX-PER-EM SPACE
119
+ [0x2007, ' '], // FIGURE SPACE
120
+ [0x2009, ' '], // THIN SPACE
121
+ [0x200A, ' '], // HAIR SPACE
122
+ [0x200B, ' '], // ZERO WIDTH SPACE
123
+ [0x202F, ' '], // NARROW NO-BREAK SPACE
124
+ [0x205F, ' '], // MEDIUM MATHEMATICAL SPACE
125
+ [0x3000, ' '], // 全角スペース
126
+ [0x3164, ' '], // HANGUL FILLER
127
+ // その他の変換
128
+ [0x09, ' '], // TAB --> SPC
129
+ [0x203B, '#'], // '※' --- コメント
130
+ [0x3002, ';'], // 句点
131
+ [0x3010, '['], // '【'
132
+ [0x3011, ']'], // '】'
133
+ // 読点は「,」に変換する (#877)
134
+ [0x3001, ','], // 読点 --- JSON記法で「,」と「、」を区別したいので読点は変換しないことに。(#276)
135
+ [0xFF0C, ','], // 読点 ',' 論文などで利用、ただし句点はドットと被るので変換しない (#735)
136
+ [0x2716, '*'], // ×の絵文字 (#1183)
137
+ [0x2795, '+'], // +の絵文字 (#1183)
138
+ [0x2796, '-'], // -の絵文字 (#1183)
139
+ [0x2797, '÷'] // ÷の絵文字 (#1183)
140
+ ])
141
+ }
142
+
143
+ // 一文字だけ変換
144
+ /**
145
+ * @param {string} ch
146
+ */
147
+ public convert1ch (ch: string): string {
148
+ if (!ch) { return '' }
149
+ const c: number = ch.codePointAt(0) || 0
150
+ // テーブルによる変換
151
+ const c2: string = this.convertTable.get(c) || ''
152
+ if (c2) { return c2 }
153
+ // ASCIIエリア
154
+ if (c < 0x7F) { return ch }
155
+ // 全角半角単純変換可能 --- '!' - '~'
156
+ if (c >= 0xFF01 && c <= 0xFF5E) {
157
+ const c2 = c - 0xFEE0
158
+ return String.fromCodePoint(c2)
159
+ }
160
+ return ch
161
+ }
162
+
163
+ /** convert code */
164
+ convert (code: string): ConvertResult[] {
165
+ if (!code) { return [] }
166
+ const src = new Replace(code)
167
+
168
+ // 改行コードを統一
169
+ src.replaceAll('\r\n', '\n')
170
+ src.replaceAll('\r', '\n')
171
+
172
+ let flagStr = false // 文字列リテラル内かどうか
173
+ let flagStr2 = false // 絵文字による文字列リテラル内かどうか
174
+ let endOfStr = '' // 文字列リテラルを終了させる記号
175
+ const res: ConvertResult[] = []
176
+ let left = 0 // 現在処理中の部分文字列の左端の位置
177
+ let str = '' // 文字列リテラルの値
178
+
179
+ // 一文字ずつ全角を半角に置換する
180
+ let i = 0
181
+ while (i < src.getText().length) {
182
+ const c = src.getText().charAt(i)
183
+ const ch2 = src.getText().substr(i, 2)
184
+ // 文字列のとき
185
+ if (flagStr) {
186
+ if (c === endOfStr) {
187
+ flagStr = false
188
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)))
189
+ i++
190
+ left = i
191
+ continue
192
+ }
193
+ str += c
194
+ i++
195
+ continue
196
+ }
197
+ // 絵文字制御による文字列のとき
198
+ if (flagStr2) {
199
+ if (ch2 === endOfStr) {
200
+ flagStr2 = false
201
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)))
202
+ i += 2
203
+ left = i
204
+ continue
205
+ }
206
+ str += c
207
+ i++
208
+ continue
209
+ }
210
+ // 文字列判定
211
+ if (c === '「') {
212
+ res.push(new ConvertResult(c, src.getSourcePosition(left)))
213
+ i++
214
+ left = i
215
+ flagStr = true
216
+ endOfStr = '」'
217
+ str = ''
218
+ continue
219
+ }
220
+ if (c === '『') {
221
+ res.push(new ConvertResult(c, src.getSourcePosition(left)))
222
+ i++
223
+ left = i
224
+ flagStr = true
225
+ endOfStr = '』'
226
+ str = ''
227
+ continue
228
+ }
229
+ if (c === '“') {
230
+ res.push(new ConvertResult(c, src.getSourcePosition(left)))
231
+ i++
232
+ left = i
233
+ flagStr = true
234
+ endOfStr = '”'
235
+ str = ''
236
+ continue
237
+ }
238
+ // JavaScriptの内部的には文字列はUTF-16で扱われてるので charAt を使う場合 絵文字が2文字扱いになる --- #726
239
+ if (ch2 === '🌴' || ch2 === '🌿') {
240
+ res.push(new ConvertResult(ch2, src.getSourcePosition(left)))
241
+ i += 2
242
+ left = i
243
+ flagStr2 = true
244
+ endOfStr = ch2
245
+ str = ''
246
+ continue
247
+ }
248
+ const c1 = this.convert1ch(c)
249
+ if (c1 === '"' || c1 === '\'') {
250
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)))
251
+ i++
252
+ left = i
253
+ flagStr = true
254
+ endOfStr = c
255
+ str = ''
256
+ continue
257
+ }
258
+ // ラインコメントを飛ばす (#725)
259
+ if (c1 === '#') {
260
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)))
261
+ i++
262
+ left = i
263
+ flagStr = true // 本当はコメントだけど便宜上
264
+ endOfStr = '\n'
265
+ str = ''
266
+ continue
267
+ }
268
+ // ラインコメントを飛ばす
269
+ if (ch2 === '//' || ch2 === '//') {
270
+ res.push(new ConvertResult('//', src.getSourcePosition(left))) // 強制的に'//'とする
271
+ i += 2
272
+ left = i
273
+ flagStr = true
274
+ endOfStr = '\n'
275
+ str = ''
276
+ continue
277
+ }
278
+ // 複数行コメント内を飛ばす (#731)
279
+ if (ch2 === '/*') {
280
+ res.push(new ConvertResult(ch2, src.getSourcePosition(left)))
281
+ i += 2
282
+ left = i
283
+ flagStr2 = true
284
+ endOfStr = '*/'
285
+ str = ''
286
+ continue
287
+ }
288
+ // 変換したものを追加
289
+ res.push(new ConvertResult(c1, src.getSourcePosition(left)))
290
+ i++
291
+ left = i
292
+ }
293
+ if (flagStr || flagStr2) {
294
+ res.push(new ConvertResult(str + endOfStr, src.getSourcePosition(left)))
295
+ }
296
+ return res
297
+ }
298
+ }
299
+
300
+ /** なでしこのソースコードのモード(!インデント構文など)が設定されているか調べる */
301
+ export function checkNakoMode (code: string, modeNames: string[]): boolean {
302
+ // 先頭の256文字について調べる
303
+ code = code.substring(0, 256)
304
+ // 全角半角の揺れを吸収
305
+ code = code.replace(/(!|💡)/, '!')
306
+ // 範囲コメントを削除
307
+ code = code.replace(/\/\*.*?\*\//g, '')
308
+ // 毎文調べる
309
+ const lines = code.split(/[;。\n]/, 30)
310
+ for (let line of lines) {
311
+ line = line.replace(/^\s+/, '').replace(/\s+$/, '') // trim
312
+ if (modeNames.indexOf(line) >= 0) { return true }
313
+ }
314
+ return false
315
+ }
@@ -0,0 +1,38 @@
1
+ /** 予約語 */
2
+ const reserved = {
3
+ '回': '回',
4
+ '回繰返': '回',
5
+ '間': '間',
6
+ '間繰返': '間',
7
+ '繰返': '繰返',
8
+ '増繰返': '増繰返',
9
+ '減繰返': '減繰返',
10
+ '後判定': '後判定',
11
+ '反復': '反復',
12
+ '抜': '抜ける',
13
+ '続': '続ける',
14
+ '戻': '戻る',
15
+ '先': '先に',
16
+ '次': '次に',
17
+ '代入': '代入',
18
+ '実行速度優先': '実行速度優先',
19
+ 'パフォーマンスモニタ適用': 'パフォーマンスモニタ適用',
20
+ '定': '定める',
21
+ '逐次実行': '逐次実行',
22
+ '条件分岐': '条件分岐',
23
+ '増': '増',
24
+ '減': '減',
25
+ '変数': '変数',
26
+ '定数': '定数',
27
+ 'エラー監視': 'エラー監視',
28
+ 'エラー': 'エラー',
29
+ 'それ': 'word',
30
+ 'そう': 'word',
31
+ '関数': 'def_func',
32
+ 'インデント構文': 'インデント構文',
33
+ '非同期モード': '非同期モード',
34
+ 'DNCLモード': 'DNCLモード',
35
+ 'モード設定': 'モード設定',
36
+ '取込': '取込'
37
+ };
38
+ export default reserved;
@@ -0,0 +1,38 @@
1
+ /** 予約語 */
2
+ const reserved: {[key: string]: string} = {
3
+ '回': '回',
4
+ '回繰返': '回', // (#924)
5
+ '間': '間',
6
+ '間繰返': '間', // (#927)
7
+ '繰返': '繰返',
8
+ '増繰返': '増繰返', // (#1140)
9
+ '減繰返': '減繰返',
10
+ '後判定': '後判定', // (#1147)
11
+ '反復': '反復',
12
+ '抜': '抜ける',
13
+ '続': '続ける',
14
+ '戻': '戻る',
15
+ '先': '先に',
16
+ '次': '次に',
17
+ '代入': '代入',
18
+ '実行速度優先': '実行速度優先',
19
+ 'パフォーマンスモニタ適用': 'パフォーマンスモニタ適用', // (#986)
20
+ '定': '定める',
21
+ '逐次実行': '逐次実行',
22
+ '条件分岐': '条件分岐',
23
+ '増': '増',
24
+ '減': '減',
25
+ '変数': '変数',
26
+ '定数': '定数',
27
+ 'エラー監視': 'エラー監視', // 例外処理:エラーならばと対
28
+ 'エラー': 'エラー',
29
+ 'それ': 'word',
30
+ 'そう': 'word', // 「それ」のエイリアス
31
+ '関数': 'def_func', // 無名関数の定義用
32
+ 'インデント構文': 'インデント構文', // https://nadesi.com/v3/doc/go.php?949
33
+ '非同期モード': '非同期モード', // (#637)
34
+ 'DNCLモード': 'DNCLモード', // (#1140)
35
+ 'モード設定': 'モード設定', // (#1020)
36
+ '取込': '取込'
37
+ }
38
+ export default reserved