scancscode 1.0.29 → 1.0.30

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.
@@ -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;
@@ -0,0 +1,220 @@
1
+ "use strict";
2
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
+ if (k2 === undefined) k2 = k;
4
+ var desc = Object.getOwnPropertyDescriptor(m, k);
5
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
+ desc = { enumerable: true, get: function() { return m[k]; } };
7
+ }
8
+ Object.defineProperty(o, k2, desc);
9
+ }) : (function(o, m, k, k2) {
10
+ if (k2 === undefined) k2 = k;
11
+ o[k2] = m[k];
12
+ }));
13
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
15
+ }) : function(o, v) {
16
+ o["default"] = v;
17
+ });
18
+ var __importStar = (this && this.__importStar) || (function () {
19
+ var ownKeys = function(o) {
20
+ ownKeys = Object.getOwnPropertyNames || function (o) {
21
+ var ar = [];
22
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
+ return ar;
24
+ };
25
+ return ownKeys(o);
26
+ };
27
+ return function (mod) {
28
+ if (mod && mod.__esModule) return mod;
29
+ var result = {};
30
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
+ __setModuleDefault(result, mod);
32
+ return result;
33
+ };
34
+ })();
35
+ Object.defineProperty(exports, "__esModule", { value: true });
36
+ exports.CSVUtils = void 0;
37
+ const csv = __importStar(require("fast-csv"));
38
+ const fs = __importStar(require("fs"));
39
+ class CSVUtils {
40
+ filePath;
41
+ constructor(filePath) {
42
+ this.filePath = filePath;
43
+ }
44
+ async parseCsv() {
45
+ if (fs.existsSync(this.filePath)) {
46
+ let stream = csv.parseFile(this.filePath, {
47
+ delimiter: ",",
48
+ });
49
+ let rows = await stream.toArray();
50
+ stream.destroy();
51
+ return rows;
52
+ }
53
+ else {
54
+ console.error(`文件不存在: ${this.filePath}`);
55
+ return [];
56
+ }
57
+ }
58
+ getShortKey(content) {
59
+ // let max = 16
60
+ // if (content.length < max) {
61
+ // return content
62
+ // } else {
63
+ // return content.substring(0, max)
64
+ // }
65
+ return content;
66
+ }
67
+ tryGetEntry(content, map01, map02) {
68
+ if (map02.has(content)) {
69
+ let row2 = map02.get(content);
70
+ return [row2 != null, content, row2];
71
+ }
72
+ let shortKey = this.getShortKey(content);
73
+ if (map01.has(shortKey)) {
74
+ let row1 = map01.get(shortKey);
75
+ if (row1 != null && (row1[0] == shortKey || row1[1] == content)) {
76
+ return [true, shortKey, row1];
77
+ }
78
+ else {
79
+ return [false, shortKey, undefined];
80
+ }
81
+ }
82
+ return [false, shortKey, undefined];
83
+ }
84
+ merge(rows0, rows1, langs) {
85
+ // 短key
86
+ let map01 = new Map();
87
+ // 全文key
88
+ let map02 = new Map();
89
+ if (rows0 != null) {
90
+ for (let index = 1; index < rows0.length; index++) {
91
+ const row = rows0[index];
92
+ let key = row[0];
93
+ map01.set(key, row);
94
+ }
95
+ for (let index = 1; index < rows0.length; index++) {
96
+ const row = rows0[index];
97
+ let key = row[1];
98
+ if (key != null && key != "") {
99
+ map02.set(key, row);
100
+ }
101
+ }
102
+ }
103
+ let rows2 = [];
104
+ if (rows0 != null && rows0.length > 0) {
105
+ // rows2.push(rows0[0]);
106
+ // 合并 langs 中没有的语言列
107
+ let headers = [...rows0[0]];
108
+ for (const lang of langs) {
109
+ if (!headers.includes(lang)) {
110
+ headers.push(lang);
111
+ }
112
+ }
113
+ rows2.push(headers);
114
+ }
115
+ else {
116
+ rows2.push(["key", ...langs]);
117
+ }
118
+ for (let index = 0; index < rows1.length; index++) {
119
+ const content = rows1[index];
120
+ let [exist, key, row] = this.tryGetEntry(content, map01, map02);
121
+ if (exist) {
122
+ if (row != null) {
123
+ rows2.push(row);
124
+ }
125
+ else {
126
+ if (key != content) {
127
+ rows2.push([key, content]);
128
+ }
129
+ else {
130
+ rows2.push([content, '']);
131
+ }
132
+ }
133
+ }
134
+ else {
135
+ if (key != content) {
136
+ rows2.push([key, content]);
137
+ }
138
+ else {
139
+ rows2.push([content, '']);
140
+ }
141
+ }
142
+ }
143
+ return rows2;
144
+ }
145
+ static async writeCsv(filePath, rows) {
146
+ let str = await csv.writeToString(rows);
147
+ fs.writeFileSync(filePath, str, "utf-8");
148
+ }
149
+ static async updateToFile(filePath, literals, langs) {
150
+ let csvUtils = new CSVUtils(filePath);
151
+ let rows0 = await csvUtils.parseCsv();
152
+ // 优化:使用Set去重,减少重复处理
153
+ const uniqueLiterals = [...new Set(literals)];
154
+ let rows2 = csvUtils.merge(rows0, uniqueLiterals, langs);
155
+ await CSVUtils.writeCsv(filePath, rows2);
156
+ console.log(`已经更新多语言表: ${filePath} , 共有 ${rows2.length - 1} 条目`);
157
+ }
158
+ /**
159
+ * 表格第一列为 key + 语言列表, 按照 langs 精简表格列, 只保留指定key列和指定语言列
160
+ * @param filePath csv文件路径
161
+ * @param outFilePath 输出csv文件路径
162
+ * @param langs 指定要保留的语言列表
163
+ */
164
+ static async slimCsvWithLangs(filePaths, outFilePath, langs) {
165
+ let rows2 = [];
166
+ let keySpace = new Set();
167
+ let conflictKeySpace = new Set();
168
+ let existAnyInput = false;
169
+ for (const filePath of filePaths) {
170
+ if (fs.existsSync(filePath) == false) {
171
+ console.warn(`文件不存在: ${filePath}`);
172
+ continue;
173
+ }
174
+ existAnyInput = true;
175
+ let csvUtils = new CSVUtils(filePath);
176
+ let rows0 = await csvUtils.parseCsv();
177
+ let header = rows0[0];
178
+ if (rows2.length == 0) {
179
+ rows2.push([header[0], ...langs]);
180
+ }
181
+ let langIndexes = [];
182
+ for (const lang of langs) {
183
+ let langIndex = header.indexOf(lang);
184
+ if (langIndex != -1) {
185
+ langIndexes.push(langIndex);
186
+ }
187
+ }
188
+ for (let index = 1; index < rows0.length; index++) {
189
+ const row = rows0[index];
190
+ let newRow = [];
191
+ let key = row[0];
192
+ if (keySpace.has(key)) {
193
+ conflictKeySpace.add(key);
194
+ }
195
+ else {
196
+ keySpace.add(key);
197
+ }
198
+ newRow.push(key);
199
+ for (const langIndex of langIndexes) {
200
+ newRow.push(row[langIndex]);
201
+ }
202
+ rows2.push(newRow);
203
+ }
204
+ }
205
+ if (existAnyInput == false) {
206
+ console.warn(`不存在有效文件, 未输出文件:`, filePaths);
207
+ return;
208
+ }
209
+ if (rows2.length > 0) {
210
+ await CSVUtils.writeCsv(outFilePath, rows2);
211
+ if (conflictKeySpace.size > 0) {
212
+ console.error(`冲突的key有 ${conflictKeySpace.size} 个:`, conflictKeySpace);
213
+ }
214
+ }
215
+ else {
216
+ console.warn(`没有生成任何数据,未输出文件。`);
217
+ }
218
+ }
219
+ }
220
+ exports.CSVUtils = CSVUtils;