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.
- package/core/.editorconfig +6 -0
- package/core/.eslintrc.cjs +33 -0
- package/core/.github/dependabot.yml +7 -0
- package/core/.github/workflows/nodejs.yml +37 -0
- package/core/.github/workflows/super-linter.yml +61 -0
- package/core/.github/workflows/textlint.yml +199 -0
- package/core/LICENSE +21 -0
- package/core/README.md +66 -0
- package/core/batch/build_nako_version.nako3 +42 -0
- package/core/command/snako.mjs +105 -0
- package/core/command/snako.mts +116 -0
- package/core/index.mjs +21 -0
- package/core/index.mts +21 -0
- package/core/package.json +47 -0
- package/core/sample/hello.nako3 +7 -0
- package/core/sample/hoge.mjs +4 -0
- package/core/sample/hoge.mts +6 -0
- package/core/src/nako3.mjs +858 -0
- package/core/src/nako3.mts +967 -0
- package/core/src/nako_colors.mjs +78 -0
- package/core/src/nako_colors.mts +86 -0
- package/core/src/nako_core_version.mjs +8 -0
- package/core/src/nako_core_version.mts +19 -0
- package/core/src/nako_csv.mjs +185 -0
- package/core/src/nako_csv.mts +188 -0
- package/core/src/nako_errors.mjs +173 -0
- package/core/src/nako_errors.mts +197 -0
- package/core/src/nako_from_dncl.mjs +255 -0
- package/core/src/nako_from_dncl.mts +250 -0
- package/core/src/nako_gen.mjs +1648 -0
- package/core/src/nako_gen.mts +1719 -0
- package/core/src/nako_gen_async.mjs +1659 -0
- package/core/src/nako_gen_async.mts +1732 -0
- package/core/src/nako_global.mjs +107 -0
- package/core/src/nako_global.mts +138 -0
- package/core/src/nako_indent.mjs +445 -0
- package/core/src/nako_indent.mts +492 -0
- package/core/src/nako_josi_list.mjs +38 -0
- package/core/src/nako_josi_list.mts +45 -0
- package/core/src/nako_lex_rules.mjs +253 -0
- package/core/src/nako_lex_rules.mts +260 -0
- package/core/src/nako_lexer.mjs +609 -0
- package/core/src/nako_lexer.mts +612 -0
- package/core/src/nako_logger.mjs +199 -0
- package/core/src/nako_logger.mts +232 -0
- package/core/src/nako_parser3.mjs +2439 -0
- package/core/src/nako_parser3.mts +2195 -0
- package/core/src/nako_parser_base.mjs +370 -0
- package/core/src/nako_parser_base.mts +370 -0
- package/core/src/nako_parser_const.mjs +37 -0
- package/core/src/nako_parser_const.mts +37 -0
- package/core/src/nako_prepare.mjs +304 -0
- package/core/src/nako_prepare.mts +315 -0
- package/core/src/nako_reserved_words.mjs +38 -0
- package/core/src/nako_reserved_words.mts +38 -0
- package/core/src/nako_source_mapping.mjs +207 -0
- package/core/src/nako_source_mapping.mts +262 -0
- package/core/src/nako_test.mjs +37 -0
- package/core/src/nako_types.mjs +25 -0
- package/core/src/nako_types.mts +151 -0
- package/core/src/plugin_csv.mjs +49 -0
- package/core/src/plugin_csv.mts +50 -0
- package/core/src/plugin_math.mjs +328 -0
- package/core/src/plugin_math.mts +326 -0
- package/core/src/plugin_promise.mjs +91 -0
- package/core/src/plugin_promise.mts +91 -0
- package/core/src/plugin_system.mjs +2832 -0
- package/core/src/plugin_system.mts +2690 -0
- package/core/src/plugin_test.mjs +34 -0
- package/core/src/plugin_test.mts +34 -0
- package/core/test/array_test.mjs +34 -0
- package/core/test/basic_test.mjs +344 -0
- package/core/test/calc_test.mjs +140 -0
- package/core/test/core_module_test.mjs +23 -0
- package/core/test/debug_test.mjs +16 -0
- package/core/test/dncl_test.mjs +94 -0
- package/core/test/error_message_test.mjs +210 -0
- package/core/test/error_test.mjs +16 -0
- package/core/test/flow_test.mjs +373 -0
- package/core/test/func_call.mjs +160 -0
- package/core/test/func_test.mjs +149 -0
- package/core/test/indent_test.mjs +364 -0
- package/core/test/lex_test.mjs +168 -0
- package/core/test/literal_test.mjs +73 -0
- package/core/test/nako_lexer_test.mjs +35 -0
- package/core/test/nako_logger_test.mjs +76 -0
- package/core/test/nako_logger_test.mts +78 -0
- package/core/test/plugin_csv_test.mjs +38 -0
- package/core/test/plugin_promise_test.mjs +18 -0
- package/core/test/plugin_system_test.mjs +630 -0
- package/core/test/prepare_test.mjs +96 -0
- package/core/test/re_test.mjs +22 -0
- package/core/test/side_effects_test.mjs +92 -0
- package/core/test/variable_scope_test.mjs +149 -0
- package/core/tsconfig.json +101 -0
- package/package.json +4 -2
- package/release/_hash.txt +12 -12
- package/release/_script-tags.txt +14 -14
- package/release/editor.js +1 -1
- package/release/stats.json +1 -1
- package/release/version.js +1 -1
- package/release/wnako3.js +1 -1
- package/src/nako_version.mjs +2 -2
- package/src/nako_version.mts +2 -2
- package/test/async/async_basic_test.mjs +3 -3
|
@@ -0,0 +1,107 @@
|
|
|
1
|
+
import { NakoColors } from './nako_colors.mjs';
|
|
2
|
+
/**
|
|
3
|
+
* コンパイルされたなでしこのプログラムで、グローバル空間のthisが指すオブジェクト
|
|
4
|
+
*/
|
|
5
|
+
export class NakoGlobal {
|
|
6
|
+
/**
|
|
7
|
+
* @param compiler
|
|
8
|
+
* @param gen
|
|
9
|
+
*/
|
|
10
|
+
constructor(compiler, gen) {
|
|
11
|
+
// ユーザーのプログラムから編集される変数
|
|
12
|
+
this.__locals = {};
|
|
13
|
+
this.__varslist = [
|
|
14
|
+
{ ...compiler.__varslist[0] },
|
|
15
|
+
{ ...compiler.__varslist[1] },
|
|
16
|
+
{ ...compiler.__varslist[2] } // local [2][3][4][5] ...
|
|
17
|
+
];
|
|
18
|
+
this.numFailures = 0;
|
|
19
|
+
this.index = 0;
|
|
20
|
+
this.nextIndex = -1;
|
|
21
|
+
this.__code = [];
|
|
22
|
+
this.__callstack = [];
|
|
23
|
+
this.__stack = [];
|
|
24
|
+
this.__labels = [];
|
|
25
|
+
this.__genMode = gen.genMode;
|
|
26
|
+
// バージョン情報の引き継ぎ
|
|
27
|
+
this.version = compiler.version;
|
|
28
|
+
this.coreVersion = compiler.coreVersion;
|
|
29
|
+
// PluginSystemとdestroy()から参照するため
|
|
30
|
+
this.__module = { ...compiler.__module }; // shallow copy
|
|
31
|
+
this.pluginfiles = { ...compiler.getPluginfiles() };
|
|
32
|
+
// PluginWorkerでユーザー定義関数のJavaScriptコードをworkerのコンパイラのインスタンスへコピーするため
|
|
33
|
+
this.gen = gen;
|
|
34
|
+
// 以下のメソッドで使うため
|
|
35
|
+
this.logger = compiler.getLogger();
|
|
36
|
+
this.compiler = compiler;
|
|
37
|
+
}
|
|
38
|
+
clearLog() {
|
|
39
|
+
this.__varslist[0]['表示ログ'] = '';
|
|
40
|
+
}
|
|
41
|
+
get log() {
|
|
42
|
+
let s = this.__varslist[0]['表示ログ'];
|
|
43
|
+
s = s.replace(/\s+$/, '');
|
|
44
|
+
return s;
|
|
45
|
+
}
|
|
46
|
+
/**
|
|
47
|
+
* 「ナデシコ」命令のためのメソッド
|
|
48
|
+
* @param {string} code
|
|
49
|
+
* @param {string} fname
|
|
50
|
+
* @param {CompilerOptions} opts
|
|
51
|
+
* @param {string} [preCode]
|
|
52
|
+
*/
|
|
53
|
+
runEx(code, fname, opts, preCode = '') {
|
|
54
|
+
// スコープを共有して実行
|
|
55
|
+
opts.preCode = preCode;
|
|
56
|
+
return this.compiler.runSync(code, fname, opts);
|
|
57
|
+
}
|
|
58
|
+
/**
|
|
59
|
+
* テスト実行のためのメソッド
|
|
60
|
+
* @param {{ name: string, f: () => void }[]} tests
|
|
61
|
+
*/
|
|
62
|
+
_runTests(tests) {
|
|
63
|
+
let text = `${NakoColors.color.bold}テストの実行結果${NakoColors.color.reset}\n`;
|
|
64
|
+
let pass = 0;
|
|
65
|
+
let numFailures = 0;
|
|
66
|
+
for (const t of tests) {
|
|
67
|
+
try {
|
|
68
|
+
t.f();
|
|
69
|
+
text += `${NakoColors.color.green}✔${NakoColors.color.reset} ${t.name}\n`;
|
|
70
|
+
pass++;
|
|
71
|
+
}
|
|
72
|
+
catch (err) {
|
|
73
|
+
text += `${NakoColors.color.red}☓${NakoColors.color.reset} ${t.name}: ${err.message}\n`;
|
|
74
|
+
numFailures++;
|
|
75
|
+
}
|
|
76
|
+
}
|
|
77
|
+
if (numFailures > 0) {
|
|
78
|
+
text += `${NakoColors.color.green}成功 ${pass}件 ${NakoColors.color.red}失敗 ${numFailures}件`;
|
|
79
|
+
}
|
|
80
|
+
else {
|
|
81
|
+
text += `${NakoColors.color.green}成功 ${pass}件`;
|
|
82
|
+
}
|
|
83
|
+
this.numFailures = numFailures;
|
|
84
|
+
this.logger.stdout(text);
|
|
85
|
+
}
|
|
86
|
+
/**
|
|
87
|
+
* 毎プラグインの「!クリア」関数を実行
|
|
88
|
+
*/
|
|
89
|
+
clearPlugins() {
|
|
90
|
+
const clearName = '!クリア';
|
|
91
|
+
for (const pname in this.pluginfiles) {
|
|
92
|
+
const po = this.__module[pname];
|
|
93
|
+
if (po[clearName] && po[clearName].fn) {
|
|
94
|
+
po[clearName].fn(this);
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
/**
|
|
99
|
+
* 各種リセット処理
|
|
100
|
+
*/
|
|
101
|
+
reset() {
|
|
102
|
+
this.clearPlugins();
|
|
103
|
+
}
|
|
104
|
+
destroy() {
|
|
105
|
+
this.reset();
|
|
106
|
+
}
|
|
107
|
+
}
|
|
@@ -0,0 +1,138 @@
|
|
|
1
|
+
import { NakoCompiler } from './nako3.mjs'
|
|
2
|
+
import { NakoColors } from './nako_colors.mjs'
|
|
3
|
+
import { NakoGen } from './nako_gen.mjs'
|
|
4
|
+
import { NakoLogger } from './nako_logger.mjs'
|
|
5
|
+
import { CompilerOptions, FuncList } from './nako_types.mjs'
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* コンパイルされたなでしこのプログラムで、グローバル空間のthisが指すオブジェクト
|
|
9
|
+
*/
|
|
10
|
+
export class NakoGlobal {
|
|
11
|
+
version: string;
|
|
12
|
+
coreVersion: string;
|
|
13
|
+
__locals: {[key: string]: any};
|
|
14
|
+
__varslist: {[key: string]: any}[];
|
|
15
|
+
__code: string[];
|
|
16
|
+
__callstack: any[];
|
|
17
|
+
__stack: any[];
|
|
18
|
+
__labels: {[key: string]: any};
|
|
19
|
+
__genMode: string;
|
|
20
|
+
__module: {[key: string]: any};
|
|
21
|
+
pluginfiles: {[key: string]: FuncList};
|
|
22
|
+
index: number;
|
|
23
|
+
nextIndex: number;
|
|
24
|
+
numFailures: number;
|
|
25
|
+
gen: NakoGen;
|
|
26
|
+
logger: NakoLogger;
|
|
27
|
+
compiler: NakoCompiler;
|
|
28
|
+
/**
|
|
29
|
+
* @param compiler
|
|
30
|
+
* @param gen
|
|
31
|
+
*/
|
|
32
|
+
constructor (compiler: NakoCompiler, gen: NakoGen) {
|
|
33
|
+
// ユーザーのプログラムから編集される変数
|
|
34
|
+
this.__locals = {}
|
|
35
|
+
this.__varslist = [
|
|
36
|
+
{ ...compiler.__varslist[0] }, // system
|
|
37
|
+
{ ...compiler.__varslist[1] }, // global
|
|
38
|
+
{ ...compiler.__varslist[2] } // local [2][3][4][5] ...
|
|
39
|
+
]
|
|
40
|
+
this.numFailures = 0
|
|
41
|
+
this.index = 0
|
|
42
|
+
this.nextIndex = -1
|
|
43
|
+
this.__code = []
|
|
44
|
+
this.__callstack = []
|
|
45
|
+
this.__stack = []
|
|
46
|
+
this.__labels = []
|
|
47
|
+
this.__genMode = gen.genMode
|
|
48
|
+
|
|
49
|
+
// バージョン情報の引き継ぎ
|
|
50
|
+
this.version = compiler.version
|
|
51
|
+
this.coreVersion = compiler.coreVersion
|
|
52
|
+
|
|
53
|
+
// PluginSystemとdestroy()から参照するため
|
|
54
|
+
this.__module = { ...compiler.__module } // shallow copy
|
|
55
|
+
this.pluginfiles = { ...compiler.getPluginfiles() }
|
|
56
|
+
|
|
57
|
+
// PluginWorkerでユーザー定義関数のJavaScriptコードをworkerのコンパイラのインスタンスへコピーするため
|
|
58
|
+
this.gen = gen
|
|
59
|
+
|
|
60
|
+
// 以下のメソッドで使うため
|
|
61
|
+
this.logger = compiler.getLogger()
|
|
62
|
+
this.compiler = compiler
|
|
63
|
+
}
|
|
64
|
+
|
|
65
|
+
clearLog () {
|
|
66
|
+
this.__varslist[0]['表示ログ'] = ''
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
get log () {
|
|
70
|
+
let s = this.__varslist[0]['表示ログ']
|
|
71
|
+
s = s.replace(/\s+$/, '')
|
|
72
|
+
return s
|
|
73
|
+
}
|
|
74
|
+
|
|
75
|
+
/**
|
|
76
|
+
* 「ナデシコ」命令のためのメソッド
|
|
77
|
+
* @param {string} code
|
|
78
|
+
* @param {string} fname
|
|
79
|
+
* @param {CompilerOptions} opts
|
|
80
|
+
* @param {string} [preCode]
|
|
81
|
+
*/
|
|
82
|
+
runEx (code: string, fname: string, opts: CompilerOptions, preCode = ''): NakoGlobal {
|
|
83
|
+
// スコープを共有して実行
|
|
84
|
+
opts.preCode = preCode
|
|
85
|
+
return this.compiler.runSync(code, fname, opts)
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* テスト実行のためのメソッド
|
|
90
|
+
* @param {{ name: string, f: () => void }[]} tests
|
|
91
|
+
*/
|
|
92
|
+
_runTests (tests: {name: string, f: () => void }[]): void {
|
|
93
|
+
let text = `${NakoColors.color.bold}テストの実行結果${NakoColors.color.reset}\n`
|
|
94
|
+
let pass = 0
|
|
95
|
+
let numFailures = 0
|
|
96
|
+
for (const t of tests) {
|
|
97
|
+
try {
|
|
98
|
+
t.f()
|
|
99
|
+
text += `${NakoColors.color.green}✔${NakoColors.color.reset} ${t.name}\n`
|
|
100
|
+
pass++
|
|
101
|
+
} catch (err: any) {
|
|
102
|
+
text += `${NakoColors.color.red}☓${NakoColors.color.reset} ${t.name}: ${err.message}\n`
|
|
103
|
+
numFailures++
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
if (numFailures > 0) {
|
|
107
|
+
text += `${NakoColors.color.green}成功 ${pass}件 ${NakoColors.color.red}失敗 ${numFailures}件`
|
|
108
|
+
} else {
|
|
109
|
+
text += `${NakoColors.color.green}成功 ${pass}件`
|
|
110
|
+
}
|
|
111
|
+
this.numFailures = numFailures
|
|
112
|
+
this.logger.stdout(text)
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
/**
|
|
116
|
+
* 毎プラグインの「!クリア」関数を実行
|
|
117
|
+
*/
|
|
118
|
+
clearPlugins () {
|
|
119
|
+
const clearName = '!クリア'
|
|
120
|
+
for (const pname in this.pluginfiles) {
|
|
121
|
+
const po = this.__module[pname]
|
|
122
|
+
if (po[clearName] && po[clearName].fn) {
|
|
123
|
+
po[clearName].fn(this)
|
|
124
|
+
}
|
|
125
|
+
}
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
/**
|
|
129
|
+
* 各種リセット処理
|
|
130
|
+
*/
|
|
131
|
+
reset () {
|
|
132
|
+
this.clearPlugins()
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
destroy () {
|
|
136
|
+
this.reset()
|
|
137
|
+
}
|
|
138
|
+
}
|
|
@@ -0,0 +1,445 @@
|
|
|
1
|
+
import { NakoIndentError } from './nako_errors.mjs';
|
|
2
|
+
import { NakoPrepare, checkNakoMode } from './nako_prepare.mjs';
|
|
3
|
+
// インデント構文のキーワード
|
|
4
|
+
const INDENT_MODE_KEYWORDS = ['!インデント構文', '!ここまでだるい'];
|
|
5
|
+
/**
|
|
6
|
+
* インデント構文指定があればコードを変換する
|
|
7
|
+
* @param {string} code
|
|
8
|
+
* @param {string} filename
|
|
9
|
+
*/
|
|
10
|
+
function convert(code, filename = 'main.nako3') {
|
|
11
|
+
// インデント構文の適用が必要か?
|
|
12
|
+
if (checkNakoMode(code, INDENT_MODE_KEYWORDS)) {
|
|
13
|
+
return convertGo(code, filename);
|
|
14
|
+
}
|
|
15
|
+
return { code, insertedLines: [], deletedLines: [] };
|
|
16
|
+
}
|
|
17
|
+
/**
|
|
18
|
+
* インデント構文指定があるかチェックする
|
|
19
|
+
* @param {string} code
|
|
20
|
+
* @returns {boolean}
|
|
21
|
+
*/
|
|
22
|
+
function isIndentSyntaxEnabled(code) {
|
|
23
|
+
return checkNakoMode(code, INDENT_MODE_KEYWORDS);
|
|
24
|
+
}
|
|
25
|
+
// ありえない改行マークを定義
|
|
26
|
+
const SpecialRetMark = '🌟🌟改行🌟🌟s4j#WjcSb😀/FcX3🌟🌟';
|
|
27
|
+
/**
|
|
28
|
+
* ソースコードのある1行の中のコメントを全て取り除く。
|
|
29
|
+
* 事前にreplaceRetMarkによって文字列や範囲コメント内の改行文字が置換されている必要がある。
|
|
30
|
+
* @param {string} src
|
|
31
|
+
* @return {string}
|
|
32
|
+
*/
|
|
33
|
+
function removeCommentsFromLine(src) {
|
|
34
|
+
const prepare = NakoPrepare.getInstance(); // `※`, `//`, `/*` といったパターン全てに対応するために必要
|
|
35
|
+
const len = src.length;
|
|
36
|
+
let result = '';
|
|
37
|
+
let eos = '';
|
|
38
|
+
let i = 0;
|
|
39
|
+
let isComment = false;
|
|
40
|
+
while (i < len) {
|
|
41
|
+
const c = src.charAt(i);
|
|
42
|
+
const ch2 = src.substring(i, 2);
|
|
43
|
+
const cPrepared = prepare.convert1ch(c);
|
|
44
|
+
const ch2Prepared = ch2.split('').map((c) => prepare.convert1ch(c)).join('');
|
|
45
|
+
// eosか?
|
|
46
|
+
if (eos !== '') {
|
|
47
|
+
// srcのi文字目以降がeosで始まるなら文字列を終了、そうでなければ1文字進める
|
|
48
|
+
if (eos === (eos.length === 1 ? cPrepared : ch2Prepared)) {
|
|
49
|
+
if (!isComment) {
|
|
50
|
+
result += src.substr(i, eos.length);
|
|
51
|
+
}
|
|
52
|
+
i += eos.length;
|
|
53
|
+
isComment = false;
|
|
54
|
+
eos = '';
|
|
55
|
+
}
|
|
56
|
+
else {
|
|
57
|
+
if (!isComment) {
|
|
58
|
+
result += c;
|
|
59
|
+
}
|
|
60
|
+
i++;
|
|
61
|
+
}
|
|
62
|
+
continue;
|
|
63
|
+
}
|
|
64
|
+
// 文字列の改行も無視する
|
|
65
|
+
switch (cPrepared) {
|
|
66
|
+
case '"':
|
|
67
|
+
case '\'':
|
|
68
|
+
eos = c;
|
|
69
|
+
result += c;
|
|
70
|
+
i++;
|
|
71
|
+
continue;
|
|
72
|
+
case '「':
|
|
73
|
+
eos = '」';
|
|
74
|
+
result += c;
|
|
75
|
+
i++;
|
|
76
|
+
continue;
|
|
77
|
+
case '『':
|
|
78
|
+
eos = '』';
|
|
79
|
+
result += c;
|
|
80
|
+
i++;
|
|
81
|
+
continue;
|
|
82
|
+
case '“':
|
|
83
|
+
eos = '”';
|
|
84
|
+
result += c;
|
|
85
|
+
i++;
|
|
86
|
+
continue;
|
|
87
|
+
case '{':
|
|
88
|
+
eos = '}';
|
|
89
|
+
result += c;
|
|
90
|
+
i++;
|
|
91
|
+
continue;
|
|
92
|
+
case '[':
|
|
93
|
+
eos = ']';
|
|
94
|
+
result += c;
|
|
95
|
+
i++;
|
|
96
|
+
continue;
|
|
97
|
+
}
|
|
98
|
+
switch (ch2) {
|
|
99
|
+
case '🌴':
|
|
100
|
+
eos = '🌴';
|
|
101
|
+
result += ch2;
|
|
102
|
+
i += 2;
|
|
103
|
+
continue;
|
|
104
|
+
case '🌿':
|
|
105
|
+
eos = '🌿';
|
|
106
|
+
result += ch2;
|
|
107
|
+
i += 2;
|
|
108
|
+
continue;
|
|
109
|
+
}
|
|
110
|
+
// 行コメント
|
|
111
|
+
if (cPrepared === '#') {
|
|
112
|
+
eos = '\n';
|
|
113
|
+
isComment = true;
|
|
114
|
+
i++;
|
|
115
|
+
continue;
|
|
116
|
+
}
|
|
117
|
+
if (ch2Prepared === '//') {
|
|
118
|
+
eos = '\n';
|
|
119
|
+
isComment = true;
|
|
120
|
+
i += 2;
|
|
121
|
+
continue;
|
|
122
|
+
}
|
|
123
|
+
// 範囲コメント
|
|
124
|
+
if (ch2Prepared === '/*') {
|
|
125
|
+
eos = '*/';
|
|
126
|
+
isComment = true;
|
|
127
|
+
i += 2;
|
|
128
|
+
continue;
|
|
129
|
+
}
|
|
130
|
+
result += c;
|
|
131
|
+
i++;
|
|
132
|
+
}
|
|
133
|
+
return result;
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* @param {string} code
|
|
137
|
+
* @param {string} filename
|
|
138
|
+
* @returns {{ code: string, insertedLines: number[], deletedLines: { lineNumber: number, len: number }[] }}
|
|
139
|
+
*/
|
|
140
|
+
function convertGo(code, filename) {
|
|
141
|
+
const insertedLines = [];
|
|
142
|
+
const deletedLines = [];
|
|
143
|
+
const END = 'ここまで‰';
|
|
144
|
+
const code2 = replaceRetMark(code); // 文字列の中などの改行を置換
|
|
145
|
+
const lines = code2.split('\n');
|
|
146
|
+
const lines2 = [];
|
|
147
|
+
const indentStack = [];
|
|
148
|
+
let lastIndent = 0;
|
|
149
|
+
let lineCount = -1;
|
|
150
|
+
lines.forEach((line) => {
|
|
151
|
+
lineCount += line.split(SpecialRetMark).length;
|
|
152
|
+
// trim line
|
|
153
|
+
// eslint-disable-next-line no-irregular-whitespace
|
|
154
|
+
if (/^[ ・\t]*$/.test(line)) {
|
|
155
|
+
deletedLines.push({ lineNumber: lines2.length, len: line.length });
|
|
156
|
+
return;
|
|
157
|
+
}
|
|
158
|
+
// eslint-disable-next-line no-irregular-whitespace
|
|
159
|
+
const lineTrimed = removeCommentsFromLine(line).replace(/^[ ・\t]+/, '').replace(/\s+$/, '');
|
|
160
|
+
if (lineTrimed === '') {
|
|
161
|
+
lines2.push(line);
|
|
162
|
+
return;
|
|
163
|
+
}
|
|
164
|
+
if (lineTrimed === 'ここまで') {
|
|
165
|
+
throw new NakoIndentError('インデント構文が有効化されているときに『ここまで』を使うことはできません。', lineCount, filename);
|
|
166
|
+
}
|
|
167
|
+
// check indent
|
|
168
|
+
const indent = countIndent(line);
|
|
169
|
+
if (lastIndent === indent) {
|
|
170
|
+
lines2.push(line);
|
|
171
|
+
return;
|
|
172
|
+
}
|
|
173
|
+
// indent
|
|
174
|
+
if (lastIndent < indent) {
|
|
175
|
+
indentStack.push(lastIndent);
|
|
176
|
+
lastIndent = indent;
|
|
177
|
+
lines2.push(line);
|
|
178
|
+
return;
|
|
179
|
+
}
|
|
180
|
+
// unindent
|
|
181
|
+
if (lastIndent > indent) {
|
|
182
|
+
// 5回
|
|
183
|
+
// 3回
|
|
184
|
+
// 1を表示
|
|
185
|
+
// |
|
|
186
|
+
// |
|
|
187
|
+
lastIndent = indent;
|
|
188
|
+
while (indentStack.length > 0) {
|
|
189
|
+
const n = indentStack.pop() || 0;
|
|
190
|
+
if (n === indent) {
|
|
191
|
+
if (lineTrimed.substring(0, 3) !== '違えば') {
|
|
192
|
+
insertedLines.push(lines2.length);
|
|
193
|
+
lines2.push(makeIndent(n) + END);
|
|
194
|
+
}
|
|
195
|
+
lines2.push(line);
|
|
196
|
+
return;
|
|
197
|
+
}
|
|
198
|
+
if (indent < n) {
|
|
199
|
+
insertedLines.push(lines2.length);
|
|
200
|
+
lines2.push(makeIndent(n) + END);
|
|
201
|
+
continue;
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
}
|
|
205
|
+
});
|
|
206
|
+
// 残りのインデントを処理
|
|
207
|
+
while (indentStack.length > 0) {
|
|
208
|
+
const n = indentStack.pop() || 0;
|
|
209
|
+
insertedLines.push(lines2.length);
|
|
210
|
+
lines2.push(makeIndent(n) + END);
|
|
211
|
+
}
|
|
212
|
+
// 特別マーカーを改行に置換
|
|
213
|
+
const lines3 = [];
|
|
214
|
+
for (let i = 0; i < lines2.length; i++) {
|
|
215
|
+
if (lines2[i].includes(SpecialRetMark)) {
|
|
216
|
+
const lines4 = lines2[i].split(SpecialRetMark);
|
|
217
|
+
// 置換されたマーカーの数だけ、それ以降の行数をずらす。
|
|
218
|
+
// unindentによって挿入された行がSpecialRetMarkを含むことはない。
|
|
219
|
+
for (let j = 0; j < insertedLines.length; j++) {
|
|
220
|
+
if (lines3.length < insertedLines[j]) {
|
|
221
|
+
insertedLines[j] += lines4.length - 1;
|
|
222
|
+
}
|
|
223
|
+
}
|
|
224
|
+
for (let j = 0; j < deletedLines.length; j++) {
|
|
225
|
+
if (lines3.length < deletedLines[j].lineNumber) {
|
|
226
|
+
deletedLines[j].lineNumber += lines4.length - 1;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
lines3.push(...lines4);
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
lines3.push(lines2[i]);
|
|
233
|
+
}
|
|
234
|
+
}
|
|
235
|
+
return { code: lines3.join('\n'), insertedLines, deletedLines };
|
|
236
|
+
}
|
|
237
|
+
/**
|
|
238
|
+
* count分だけ字下げする
|
|
239
|
+
* @param {number} count
|
|
240
|
+
*/
|
|
241
|
+
function makeIndent(count) {
|
|
242
|
+
let s = '';
|
|
243
|
+
for (let i = 0; i < count; i++) {
|
|
244
|
+
s += ' ';
|
|
245
|
+
}
|
|
246
|
+
return s;
|
|
247
|
+
}
|
|
248
|
+
/**
|
|
249
|
+
* インデント部分を取り出す
|
|
250
|
+
* @param {string} line
|
|
251
|
+
*/
|
|
252
|
+
function getIndent(line) {
|
|
253
|
+
// eslint-disable-next-line no-irregular-whitespace
|
|
254
|
+
const m = /^([ ・\t]*)/.exec(removeCommentsFromLine(line));
|
|
255
|
+
if (!m) {
|
|
256
|
+
return '';
|
|
257
|
+
}
|
|
258
|
+
return m[1];
|
|
259
|
+
}
|
|
260
|
+
/**
|
|
261
|
+
* インデントの個数を数える
|
|
262
|
+
* @param {string} line
|
|
263
|
+
*/
|
|
264
|
+
function countIndent(line) {
|
|
265
|
+
let cnt = 0;
|
|
266
|
+
for (let i = 0; i < line.length; i++) {
|
|
267
|
+
const ch = line.charAt(i);
|
|
268
|
+
if (ch === ' ') {
|
|
269
|
+
cnt++;
|
|
270
|
+
continue;
|
|
271
|
+
}
|
|
272
|
+
if (ch === ' ') {
|
|
273
|
+
cnt += 2;
|
|
274
|
+
continue;
|
|
275
|
+
}
|
|
276
|
+
if (ch === '・') {
|
|
277
|
+
cnt += 2;
|
|
278
|
+
continue;
|
|
279
|
+
}
|
|
280
|
+
if (ch === '\t') {
|
|
281
|
+
cnt += 4;
|
|
282
|
+
continue;
|
|
283
|
+
}
|
|
284
|
+
break;
|
|
285
|
+
}
|
|
286
|
+
return cnt;
|
|
287
|
+
}
|
|
288
|
+
/**
|
|
289
|
+
* @param {string} src
|
|
290
|
+
* @returns {string}
|
|
291
|
+
*/
|
|
292
|
+
function replaceRetMark(src) {
|
|
293
|
+
const prepare = NakoPrepare.getInstance(); // `※`, `//`, `/*` といったパターン全てに対応するために必要
|
|
294
|
+
const len = src.length;
|
|
295
|
+
let result = '';
|
|
296
|
+
let eos = '';
|
|
297
|
+
let i = 0;
|
|
298
|
+
while (i < len) {
|
|
299
|
+
const c = src.charAt(i);
|
|
300
|
+
const ch2 = src.substr(i, 2);
|
|
301
|
+
const cPrepared = prepare.convert1ch(c);
|
|
302
|
+
const ch2Prepared = ch2.split('').map((c) => prepare.convert1ch(c)).join('');
|
|
303
|
+
// eosか?
|
|
304
|
+
if (eos !== '') {
|
|
305
|
+
// srcのi文字目以降がeosで始まるなら文字列を終了、そうでなければ1文字進める
|
|
306
|
+
if (eos === (eos.length === 1 ? cPrepared : ch2Prepared)) {
|
|
307
|
+
result += src.substr(i, eos.length);
|
|
308
|
+
i += eos.length;
|
|
309
|
+
eos = '';
|
|
310
|
+
}
|
|
311
|
+
else {
|
|
312
|
+
if (c === '\n') {
|
|
313
|
+
result += SpecialRetMark;
|
|
314
|
+
}
|
|
315
|
+
else {
|
|
316
|
+
result += c;
|
|
317
|
+
}
|
|
318
|
+
i++;
|
|
319
|
+
}
|
|
320
|
+
continue;
|
|
321
|
+
}
|
|
322
|
+
// 文字列の改行も無視する
|
|
323
|
+
switch (cPrepared) {
|
|
324
|
+
case '"':
|
|
325
|
+
case '\'':
|
|
326
|
+
eos = c;
|
|
327
|
+
result += c;
|
|
328
|
+
i++;
|
|
329
|
+
continue;
|
|
330
|
+
case '「':
|
|
331
|
+
eos = '」';
|
|
332
|
+
result += c;
|
|
333
|
+
i++;
|
|
334
|
+
continue;
|
|
335
|
+
case '『':
|
|
336
|
+
eos = '』';
|
|
337
|
+
result += c;
|
|
338
|
+
i++;
|
|
339
|
+
continue;
|
|
340
|
+
case '“':
|
|
341
|
+
eos = '”';
|
|
342
|
+
result += c;
|
|
343
|
+
i++;
|
|
344
|
+
continue;
|
|
345
|
+
case '{':
|
|
346
|
+
eos = '}';
|
|
347
|
+
result += c;
|
|
348
|
+
i++;
|
|
349
|
+
continue;
|
|
350
|
+
case '[':
|
|
351
|
+
eos = ']';
|
|
352
|
+
result += c;
|
|
353
|
+
i++;
|
|
354
|
+
continue;
|
|
355
|
+
}
|
|
356
|
+
switch (ch2) {
|
|
357
|
+
case '🌴':
|
|
358
|
+
eos = '🌴';
|
|
359
|
+
result += ch2;
|
|
360
|
+
i += 2;
|
|
361
|
+
continue;
|
|
362
|
+
case '🌿':
|
|
363
|
+
eos = '🌿';
|
|
364
|
+
result += ch2;
|
|
365
|
+
i += 2;
|
|
366
|
+
continue;
|
|
367
|
+
}
|
|
368
|
+
// 行コメント
|
|
369
|
+
if (cPrepared === '#') {
|
|
370
|
+
eos = '\n';
|
|
371
|
+
result += c;
|
|
372
|
+
i++;
|
|
373
|
+
continue;
|
|
374
|
+
}
|
|
375
|
+
if (ch2Prepared === '//') {
|
|
376
|
+
eos = '\n';
|
|
377
|
+
result += ch2;
|
|
378
|
+
i += 2;
|
|
379
|
+
continue;
|
|
380
|
+
}
|
|
381
|
+
// 範囲コメント
|
|
382
|
+
if (ch2Prepared === '/*') {
|
|
383
|
+
eos = '*/';
|
|
384
|
+
result += ch2;
|
|
385
|
+
i += 2;
|
|
386
|
+
continue;
|
|
387
|
+
}
|
|
388
|
+
result += c;
|
|
389
|
+
i++;
|
|
390
|
+
}
|
|
391
|
+
return result;
|
|
392
|
+
}
|
|
393
|
+
/**
|
|
394
|
+
* コードのインデントの構造を取得する。
|
|
395
|
+
* 空白行や複数行にまたがる構文を考慮する。
|
|
396
|
+
* インデント構文が有効化されていない場合にも使われる。
|
|
397
|
+
* @param {string} code
|
|
398
|
+
*/
|
|
399
|
+
export function getBlockStructure(code) {
|
|
400
|
+
const result = {
|
|
401
|
+
lines: [],
|
|
402
|
+
pairs: [],
|
|
403
|
+
parents: [],
|
|
404
|
+
spaces: [] // 各行のインデントの文字列
|
|
405
|
+
};
|
|
406
|
+
const lines = replaceRetMark(code).split('\n');
|
|
407
|
+
const stack = [];
|
|
408
|
+
let lineCount = 0;
|
|
409
|
+
let prev = countIndent(lines[0]);
|
|
410
|
+
for (const line of lines) {
|
|
411
|
+
const numLines = line.split(SpecialRetMark).length;
|
|
412
|
+
const line2 = removeCommentsFromLine(line);
|
|
413
|
+
// eslint-disable-next-line no-irregular-whitespace
|
|
414
|
+
const current = (line2.replace(/^[ ・\t]+/, '') === '')
|
|
415
|
+
? prev
|
|
416
|
+
: countIndent(line2);
|
|
417
|
+
result.lines.push(...Array(numLines).fill(current));
|
|
418
|
+
result.spaces.push(...Array(numLines).fill(getIndent(line2)));
|
|
419
|
+
if (prev < current) {
|
|
420
|
+
stack.push(lineCount - 1);
|
|
421
|
+
}
|
|
422
|
+
else if (prev > current) {
|
|
423
|
+
const last = stack.pop();
|
|
424
|
+
if (last !== undefined) {
|
|
425
|
+
result.pairs.push([last, lineCount]);
|
|
426
|
+
}
|
|
427
|
+
}
|
|
428
|
+
const parent = stack[stack.length - 1] !== undefined ? stack[stack.length - 1] : null;
|
|
429
|
+
result.parents.push(...Array(numLines).fill(parent));
|
|
430
|
+
prev = current;
|
|
431
|
+
lineCount += numLines;
|
|
432
|
+
}
|
|
433
|
+
// スタックが余ったらコードの末尾とペアにする。
|
|
434
|
+
for (const item of stack) {
|
|
435
|
+
result.pairs.push([item, lineCount]);
|
|
436
|
+
}
|
|
437
|
+
return result;
|
|
438
|
+
}
|
|
439
|
+
export default {
|
|
440
|
+
convert,
|
|
441
|
+
getBlockStructure,
|
|
442
|
+
getIndent,
|
|
443
|
+
countIndent,
|
|
444
|
+
isIndentSyntaxEnabled
|
|
445
|
+
};
|