scancscode 1.0.31 → 1.0.34

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