scancscode 1.0.29 → 1.0.31

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.
@@ -1,3 +1,3 @@
1
1
  #!/usr/bin/env node
2
2
 
3
- require("../dist/RunConvert.js")
3
+ require("../dist/src/RunConvert.js")
package/bin/slimlangs.js CHANGED
@@ -1,3 +1,3 @@
1
1
  #!/usr/bin/env node
2
2
 
3
- require("../dist/RunSlimLangs.js")
3
+ require("../dist/src/RunSlimLangs.js")
@@ -0,0 +1,288 @@
1
+ "use strict";
2
+ Object.defineProperty(exports, "__esModule", { value: true });
3
+ exports.CSCodeScanner = void 0;
4
+ const CSharpStringExtractor_1 = require("./CSharpStringExtractor");
5
+ const LiteralCollector_1 = require("./LiteralCollector");
6
+ /**
7
+ * 扫描C#代码, 扫描出其中所有形如 uictrl.text = 其他内容; 的代码片段, 其他内容可以是: $"<font color={colorStr}>{itemConfig.Name}</ font>"; 等等, 其他内容中形如 {变量} 的部分需要提取出来, 转换成 string.Format("<font color={0}>{1}</ font>", colorStr, itemConfig.Name); 的格式
8
+ */
9
+ class CSCodeScanner {
10
+ // 正则表达式匹配 uictrl.text = $"..."; 的模式
11
+ static assignmentPattern = /([\w\.\u4e00-\u9fff]+\.text\s*\+?=(?![=>])\s*|return\s+)([^;]+)\s*;/mg;
12
+ // 正则表达式匹配 xxx = "..."; 的模式
13
+ static assignmentPattern2 = /(^\s*[\.a-zA-Z0-9_\u4e00-\u9fff]+\s*\+?=)\s*;/mg;
14
+ // 正则表达式匹配 匹配每个字符串
15
+ static stringPattern = /\$?"([^"]*)"/mg;
16
+ // 正则表达式匹配 匹配获取成员值的表达式
17
+ static getMemberValuePattern(trMethodName) {
18
+ return new RegExp(`^(\\s*)([\\w\\.\\u4e00-\\u9fff\\[\\]]+)(\\.${trMethodName}\\(\\))?(\\s*)$`, "m");
19
+ }
20
+ static memberSplit = /^[\s\+;]/;
21
+ // 正则表达式匹配插值表达式 {...}
22
+ static interpolationPattern = /\{([^}]+?)(\:[a-zA-Z\d-\.\#(?:\\\\:)\u4e00-\u9fff]+)?\}/mg;
23
+ static stringFormatPattern = /(?<![a-zA-Z_\u4e00-\u9fff])([sS]tring\.Format)(?=\(.*?\))/mg;
24
+ static interpolationHeadPattern = /^[\$\@]*(.+)$/m;
25
+ static isNativeString2(s) {
26
+ return s == '""';
27
+ }
28
+ static isNativeString(s) {
29
+ let m = s.match(this.interpolationHeadPattern);
30
+ if (m != null) {
31
+ let stringContent = m[1];
32
+ return this.isNativeString2(stringContent);
33
+ }
34
+ return this.isNativeString2(s);
35
+ }
36
+ static getLineIndexFromIndex(content, index) {
37
+ let lineIndex = 0;
38
+ // 从第0个字符开始遍历到index位置, 计算换行符数量
39
+ for (let i = 0; i < index && i < content.length; i++) {
40
+ if (content[i] === '\n') {
41
+ lineIndex++;
42
+ }
43
+ }
44
+ let lineColumn = content.lastIndexOf('\n', index);
45
+ return [lineIndex + 1, index - lineColumn];
46
+ }
47
+ static scanFile(filePath, content, trmethod) {
48
+ let trmethodParts = trmethod.split(".");
49
+ let trclassname = "Tr";
50
+ let trmethodname = "TR";
51
+ if (trmethodParts.length == 2) {
52
+ trclassname = trmethodParts[0];
53
+ trmethodname = trmethodParts[1];
54
+ }
55
+ const snippets = [];
56
+ // const assignMatches = [...content.matchAll(CSCodeScanner.assignmentPattern)];
57
+ // for (let j = 0; j < assignMatches.length; j++) {
58
+ // const assignMatch = assignMatches[j];
59
+ // // let assignLine = assignMatch[0];
60
+ // // let assignHead = assignMatch[1];
61
+ // // let bodyLine = assignMatch[2];
62
+ // let [assignLine, assignHead, bodyLine] = assignMatch;
63
+ // let convertedAssignLine: string;
64
+ // /**
65
+ // * 需要翻译的文本
66
+ // */
67
+ // let literals: string[] = [];
68
+ // let unexpects: string[] = [];
69
+ // const stringMatchs = [...bodyLine.matchAll(CSCodeScanner.stringPattern)];
70
+ // if (stringMatchs.length > 0) {
71
+ // // 通过是否单行字符串并以 ".TR(); 结尾和判断是否内插字符串来判断是否需要在结尾附加 .TR()
72
+ // let isSingleString = false;
73
+ // if (stringMatchs.length == 1 && stringMatchs[0][0] == bodyLine) {
74
+ // let stringMatchFirst = stringMatchs[0];
75
+ // let stringLine = stringMatchFirst[0];
76
+ // if (stringLine.startsWith("$") || stringLine.startsWith("@$")) {
77
+ // let stringContent = stringMatchFirst[1];
78
+ // const variableMatches0 = [...stringContent.matchAll(CSCodeScanner.interpolationPattern)];
79
+ // isSingleString = variableMatches0.length == 0;
80
+ // } else {
81
+ // isSingleString = true;
82
+ // }
83
+ // }
84
+ // if (isSingleString) {
85
+ // let stringMatchFirst = stringMatchs[0];
86
+ // let stringLineIndex = stringMatchFirst.index;
87
+ // let prefix = bodyLine.substring(0, stringLineIndex);
88
+ // let stringLineFirst = stringMatchFirst[0];
89
+ // if (this.isNativeString(stringLineFirst)) {
90
+ // convertedAssignLine = assignHead + prefix + stringMatchFirst[0] + ";";
91
+ // } else {
92
+ // convertedAssignLine = assignHead + prefix + stringMatchFirst[0] + `.${trmethodname}();`;
93
+ // }
94
+ // } else {
95
+ // let stringLines: string[] = [];
96
+ // stringLines.push(assignHead);
97
+ // let lastIndex = 0;
98
+ // for (let i = 0; i < stringMatchs.length; i++) {
99
+ // let stringMatch = stringMatchs[i];
100
+ // let stringLine = stringMatch[0];
101
+ // let stringContent = stringMatch[1];
102
+ // let stringLineIndex = stringMatch.index;
103
+ // let stringStartIndexInContent = assignMatch.index + assignHead.length + stringMatch.index
104
+ // /**
105
+ // * 转换后的字符串
106
+ // */
107
+ // let convertedString: string;
108
+ // if (stringLine.startsWith("$") || stringLine.startsWith("@$")) {
109
+ // // 开始转换插值
110
+ // // 提取插值表达式中的变量
111
+ // const variableMatches0 = [...stringContent.matchAll(CSCodeScanner.interpolationPattern)];
112
+ // if (variableMatches0.length > 0) {
113
+ // let variableMatches: RegExpExecArray[] = [];
114
+ // let map = new Map<string, RegExpExecArray>();
115
+ // for (const variableMatche of variableMatches0) {
116
+ // if (!map.has(variableMatche[1])) {
117
+ // map.set(variableMatche[1], variableMatche);
118
+ // variableMatches.push(variableMatche);
119
+ // }
120
+ // }
121
+ // // 构建格式化字符串
122
+ // let formatString = stringContent;
123
+ // variableMatches.forEach((variableMatche, index) => {
124
+ // let varName = variableMatche[1];
125
+ // let suffix = variableMatche[2] ?? "";
126
+ // let replaced = formatString.replaceAll(`{${varName}${suffix}}`, `{${index}${suffix}}`);
127
+ // if (replaced == formatString) {
128
+ // let [lineNum, colNum] = CSCodeScanner.getLineIndexFromIndex(content, stringStartIndexInContent);
129
+ // let tip = `可能无法处理的复杂情形1(${filePath}:${lineNum}:${colNum}): ${stringContent}`;
130
+ // console.error(tip);
131
+ // unexpects.push(tip);
132
+ // } else {
133
+ // formatString = replaced;
134
+ // }
135
+ // });
136
+ // // 构建string.Format调用
137
+ // let formatCall = `string.Format("${formatString}"`;
138
+ // for (const variableMatche of variableMatches) {
139
+ // let varName = variableMatche[1];
140
+ // formatCall += `, ${varName}`;
141
+ // }
142
+ // formatCall += ")";
143
+ // convertedString = formatCall;
144
+ // literals.push(formatString);
145
+ // } else {
146
+ // // throw new Error("暂不支持插值字符串转换为 string.Format, 请手动处理");
147
+ // let isValidFormat = true;
148
+ // if (stringContent.indexOf("{") != -1 || stringContent.indexOf("}") != -1) {
149
+ // if (bodyLine.indexOf("{") != -1 && bodyLine.indexOf("}") != -1) {
150
+ // let [lineNum, colNum] = CSCodeScanner.getLineIndexFromIndex(content, stringStartIndexInContent);
151
+ // let tip = `可能无法处理的复杂情形2(${filePath}:${lineNum}:${colNum}): ${stringContent}`;
152
+ // console.error(tip);
153
+ // unexpects.push(tip);
154
+ // isValidFormat = false;
155
+ // }
156
+ // }
157
+ // let formattedStringLine = stringLine;
158
+ // if (isValidFormat) {
159
+ // // let stringStartIndex = stringMatch.index
160
+ // // let stringEndIndex = stringStartIndex + stringLine.length
161
+ // // let prefix = bodyLine.substring(stringStartIndex - 2, stringStartIndex)
162
+ // // let suffix = bodyLine.substring(stringEndIndex, stringEndIndex + ".TR()".length)
163
+ // // if ((stringStartIndex == 0 || prefix == "+ ") && suffix != ".TR()") {
164
+ // // formattedStringLine = formattedStringLine + ".TR()"
165
+ // // }
166
+ // formattedStringLine = this.replaceStringsTR(stringMatch, stringLine, bodyLine, trmethodname);
167
+ // literals.push(stringContent);
168
+ // }
169
+ // convertedString = formattedStringLine;
170
+ // }
171
+ // } else {
172
+ // literals.push(stringContent);
173
+ // convertedString = this.replaceStringsTR(stringMatch, stringLine, bodyLine, trmethodname);
174
+ // }
175
+ // // 串联字符串列表
176
+ // let prefix = bodyLine.substring(lastIndex, stringLineIndex);
177
+ // lastIndex = stringLineIndex + stringLine.length;
178
+ // stringLines.push(prefix);
179
+ // stringLines.push(convertedString);
180
+ // }
181
+ // let matchEnd = stringMatchs[stringMatchs.length - 1];
182
+ // let endIndex = matchEnd.index + matchEnd[0].length;
183
+ // stringLines.push(bodyLine.substring(endIndex));
184
+ // stringLines.push(";");
185
+ // convertedAssignLine = stringLines.join("");
186
+ // }
187
+ // } else {
188
+ // let convertBodyLine = CSCodeScanner.handleMembersConvert(bodyLine, assignHead, trmethodname)
189
+ // if (convertBodyLine != null) {
190
+ // convertedAssignLine = (assignHead ?? "") + convertBodyLine;
191
+ // } else {
192
+ // convertedAssignLine = assignLine
193
+ // }
194
+ // }
195
+ // let convertedAssignLine2 = convertedAssignLine.replaceAll(this.stringFormatPattern, `${trclassname}.Format`);
196
+ // snippets.push({
197
+ // originalIndex: assignMatch.index,
198
+ // originalCode: assignLine,
199
+ // convertedCode: convertedAssignLine2,
200
+ // literals,
201
+ // unexpects,
202
+ // });
203
+ // }
204
+ let capturer = new CSharpStringExtractor_1.CSharpStringExtractor();
205
+ capturer.extractStrings(content, snippets, trclassname, trmethodname);
206
+ return snippets;
207
+ }
208
+ static filterSnippets(snippets) {
209
+ let len = snippets.length;
210
+ for (let i = len - 1; i >= 0; i--) {
211
+ let snippet = snippets[i];
212
+ let needReplace = snippet.literals.some(literal => {
213
+ const needTr = LiteralCollector_1.LiteralCollector.needTranslate(literal);
214
+ return needTr;
215
+ });
216
+ if (!needReplace) {
217
+ snippets.splice(i, 1);
218
+ }
219
+ }
220
+ }
221
+ static handleMembersConvert(bodyLine, assignHead, trmethodname) {
222
+ let convertedAssignLine;
223
+ let memberLines = bodyLine.split("+");
224
+ if (assignHead != 'return ' && memberLines.length > 0) {
225
+ for (let i = 0; i < memberLines.length; i++) {
226
+ let memberLine = memberLines[i];
227
+ let regex = this.getMemberValuePattern(trmethodname);
228
+ let match = memberLine.match(regex);
229
+ if (match != null) {
230
+ let value = match[2];
231
+ if (value != 'null' && value != 'true' && value != 'false' && !this.isNumericString(value)) {
232
+ memberLines[i] = match[1] + match[2] + `.${trmethodname}()` + match[4];
233
+ }
234
+ }
235
+ }
236
+ convertedAssignLine = memberLines.join("+") + ";";
237
+ }
238
+ else {
239
+ convertedAssignLine = null;
240
+ }
241
+ return convertedAssignLine;
242
+ }
243
+ static isNumericString(str) {
244
+ if (typeof str !== 'string')
245
+ return false;
246
+ const num = Number(str);
247
+ return Number.isFinite(num) && String(num) === str.trim(); // 防止 ' 123 ' → 误判(可选严格匹配)
248
+ }
249
+ static replaceStringsTR(stringMatch, stringLine, bodyLine, trmethodname) {
250
+ if (this.isNativeString(stringLine)) {
251
+ return stringLine;
252
+ }
253
+ let trmethodcall = `.${trmethodname}()`;
254
+ let formattedStringLine = stringLine;
255
+ let stringStartIndex = stringMatch.index;
256
+ let stringEndIndex = stringStartIndex + stringLine.length;
257
+ let prefix = bodyLine.substring(stringStartIndex - 2, stringStartIndex);
258
+ let suffix = bodyLine.substring(stringEndIndex, stringEndIndex + trmethodcall.length);
259
+ if ((stringStartIndex == 0 || prefix == "+ " || prefix == ": ") && suffix != trmethodcall) {
260
+ formattedStringLine = formattedStringLine + trmethodcall;
261
+ }
262
+ return formattedStringLine;
263
+ }
264
+ static replaceInFile(content, snippets) {
265
+ let parts = [];
266
+ if (snippets.length > 0) {
267
+ let lastIndex = 0;
268
+ for (const snippet of snippets) {
269
+ let part = content.substring(lastIndex, snippet.originalIndex);
270
+ lastIndex = snippet.originalIndex + snippet.originalCode.length;
271
+ parts.push(part);
272
+ parts.push(snippet.convertedCode);
273
+ }
274
+ let endSnippet = snippets[snippets.length - 1];
275
+ let partEnd = content.substring(endSnippet.originalIndex + endSnippet.originalCode.length);
276
+ parts.push(partEnd);
277
+ }
278
+ try {
279
+ let convertedContent = parts.join("");
280
+ return convertedContent;
281
+ }
282
+ catch (err) {
283
+ console.error(`替换字符串失败: ${err}`);
284
+ return content;
285
+ }
286
+ }
287
+ }
288
+ exports.CSCodeScanner = CSCodeScanner;
@@ -149,7 +149,9 @@ class CSVUtils {
149
149
  static async updateToFile(filePath, literals, langs) {
150
150
  let csvUtils = new CSVUtils(filePath);
151
151
  let rows0 = await csvUtils.parseCsv();
152
- let rows2 = csvUtils.merge(rows0, literals, langs);
152
+ // 优化:使用Set去重,减少重复处理
153
+ const uniqueLiterals = [...new Set(literals)];
154
+ let rows2 = csvUtils.merge(rows0, uniqueLiterals, langs);
153
155
  await CSVUtils.writeCsv(filePath, rows2);
154
156
  console.log(`已经更新多语言表: ${filePath} , 共有 ${rows2.length - 1} 条目`);
155
157
  }