@cloudpss/expression 0.6.0-alpha.1

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 (165) hide show
  1. package/README.md +3 -0
  2. package/dist/analyze.d.ts +7 -0
  3. package/dist/analyze.d.ts.map +1 -0
  4. package/dist/analyze.js +38 -0
  5. package/dist/analyze.js.map +1 -0
  6. package/dist/context.d.ts +41 -0
  7. package/dist/context.d.ts.map +1 -0
  8. package/dist/context.js +18 -0
  9. package/dist/context.js.map +1 -0
  10. package/dist/definitions/argument.d.ts +33 -0
  11. package/dist/definitions/argument.d.ts.map +1 -0
  12. package/dist/definitions/argument.js +2 -0
  13. package/dist/definitions/argument.js.map +1 -0
  14. package/dist/definitions/constraint.d.ts +32 -0
  15. package/dist/definitions/constraint.d.ts.map +1 -0
  16. package/dist/definitions/constraint.js +20 -0
  17. package/dist/definitions/constraint.js.map +1 -0
  18. package/dist/definitions/parameter-decoration.d.ts +27 -0
  19. package/dist/definitions/parameter-decoration.d.ts.map +1 -0
  20. package/dist/definitions/parameter-decoration.js +17 -0
  21. package/dist/definitions/parameter-decoration.js.map +1 -0
  22. package/dist/definitions/parameter-group.d.ts +38 -0
  23. package/dist/definitions/parameter-group.d.ts.map +1 -0
  24. package/dist/definitions/parameter-group.js +31 -0
  25. package/dist/definitions/parameter-group.js.map +1 -0
  26. package/dist/definitions/parameter.d.ts +214 -0
  27. package/dist/definitions/parameter.d.ts.map +1 -0
  28. package/dist/definitions/parameter.js +44 -0
  29. package/dist/definitions/parameter.js.map +1 -0
  30. package/dist/definitions/variable.d.ts +14 -0
  31. package/dist/definitions/variable.d.ts.map +1 -0
  32. package/dist/definitions/variable.js +14 -0
  33. package/dist/definitions/variable.js.map +1 -0
  34. package/dist/definitions.d.ts +7 -0
  35. package/dist/definitions.d.ts.map +1 -0
  36. package/dist/definitions.js +7 -0
  37. package/dist/definitions.js.map +1 -0
  38. package/dist/eval.d.ts +15 -0
  39. package/dist/eval.d.ts.map +1 -0
  40. package/dist/eval.js +54 -0
  41. package/dist/eval.js.map +1 -0
  42. package/dist/expression.d.ts +37 -0
  43. package/dist/expression.d.ts.map +1 -0
  44. package/dist/expression.js +18 -0
  45. package/dist/expression.js.map +1 -0
  46. package/dist/index.d.ts +6 -0
  47. package/dist/index.d.ts.map +1 -0
  48. package/dist/index.js +5 -0
  49. package/dist/index.js.map +1 -0
  50. package/dist/main.d.ts +49 -0
  51. package/dist/main.d.ts.map +1 -0
  52. package/dist/main.js +238 -0
  53. package/dist/main.js.map +1 -0
  54. package/dist/migrate.d.ts +5 -0
  55. package/dist/migrate.d.ts.map +1 -0
  56. package/dist/migrate.js +23 -0
  57. package/dist/migrate.js.map +1 -0
  58. package/dist/migrator/access.d.ts +6 -0
  59. package/dist/migrator/access.d.ts.map +1 -0
  60. package/dist/migrator/access.js +176 -0
  61. package/dist/migrator/access.js.map +1 -0
  62. package/dist/migrator/call.d.ts +6 -0
  63. package/dist/migrator/call.d.ts.map +1 -0
  64. package/dist/migrator/call.js +405 -0
  65. package/dist/migrator/call.js.map +1 -0
  66. package/dist/migrator/concat.d.ts +6 -0
  67. package/dist/migrator/concat.d.ts.map +1 -0
  68. package/dist/migrator/concat.js +32 -0
  69. package/dist/migrator/concat.js.map +1 -0
  70. package/dist/migrator/condition.d.ts +6 -0
  71. package/dist/migrator/condition.d.ts.map +1 -0
  72. package/dist/migrator/condition.js +110 -0
  73. package/dist/migrator/condition.js.map +1 -0
  74. package/dist/migrator/interface.d.ts +24 -0
  75. package/dist/migrator/interface.d.ts.map +1 -0
  76. package/dist/migrator/interface.js +8 -0
  77. package/dist/migrator/interface.js.map +1 -0
  78. package/dist/migrator/node.d.ts +12 -0
  79. package/dist/migrator/node.d.ts.map +1 -0
  80. package/dist/migrator/node.js +119 -0
  81. package/dist/migrator/node.js.map +1 -0
  82. package/dist/migrator/operator.d.ts +6 -0
  83. package/dist/migrator/operator.d.ts.map +1 -0
  84. package/dist/migrator/operator.js +299 -0
  85. package/dist/migrator/operator.js.map +1 -0
  86. package/dist/migrator/parser.d.ts +3 -0
  87. package/dist/migrator/parser.d.ts.map +1 -0
  88. package/dist/migrator/parser.js +76 -0
  89. package/dist/migrator/parser.js.map +1 -0
  90. package/dist/migrator/special.d.ts +4 -0
  91. package/dist/migrator/special.d.ts.map +1 -0
  92. package/dist/migrator/special.js +26 -0
  93. package/dist/migrator/special.js.map +1 -0
  94. package/dist/migrator/state.d.ts +64 -0
  95. package/dist/migrator/state.d.ts.map +1 -0
  96. package/dist/migrator/state.js +251 -0
  97. package/dist/migrator/state.js.map +1 -0
  98. package/dist/migrator/symbol.d.ts +6 -0
  99. package/dist/migrator/symbol.d.ts.map +1 -0
  100. package/dist/migrator/symbol.js +50 -0
  101. package/dist/migrator/symbol.js.map +1 -0
  102. package/dist/migrator/to-type.d.ts +10 -0
  103. package/dist/migrator/to-type.d.ts.map +1 -0
  104. package/dist/migrator/to-type.js +89 -0
  105. package/dist/migrator/to-type.js.map +1 -0
  106. package/dist/migrator/utils.d.ts +14 -0
  107. package/dist/migrator/utils.d.ts.map +1 -0
  108. package/dist/migrator/utils.js +64 -0
  109. package/dist/migrator/utils.js.map +1 -0
  110. package/dist/parser.d.ts +41 -0
  111. package/dist/parser.d.ts.map +1 -0
  112. package/dist/parser.js +36 -0
  113. package/dist/parser.js.map +1 -0
  114. package/dist/scope.d.ts +43 -0
  115. package/dist/scope.d.ts.map +1 -0
  116. package/dist/scope.js +204 -0
  117. package/dist/scope.js.map +1 -0
  118. package/dist/type.d.ts +21 -0
  119. package/dist/type.d.ts.map +1 -0
  120. package/dist/type.js +50 -0
  121. package/dist/type.js.map +1 -0
  122. package/jest.config.js +3 -0
  123. package/package.json +26 -0
  124. package/src/analyze.ts +44 -0
  125. package/src/context.ts +54 -0
  126. package/src/definitions/argument.ts +40 -0
  127. package/src/definitions/constraint.ts +49 -0
  128. package/src/definitions/parameter-decoration.ts +42 -0
  129. package/src/definitions/parameter-group.ts +62 -0
  130. package/src/definitions/parameter.ts +276 -0
  131. package/src/definitions/variable.ts +23 -0
  132. package/src/definitions.ts +6 -0
  133. package/src/eval.ts +64 -0
  134. package/src/expression.ts +62 -0
  135. package/src/index.ts +5 -0
  136. package/src/main.ts +295 -0
  137. package/src/migrate.ts +25 -0
  138. package/src/migrator/access.ts +201 -0
  139. package/src/migrator/call.ts +397 -0
  140. package/src/migrator/concat.ts +28 -0
  141. package/src/migrator/condition.ts +120 -0
  142. package/src/migrator/interface.ts +30 -0
  143. package/src/migrator/node.ts +131 -0
  144. package/src/migrator/operator.ts +339 -0
  145. package/src/migrator/parser.ts +106 -0
  146. package/src/migrator/special.ts +30 -0
  147. package/src/migrator/state.ts +262 -0
  148. package/src/migrator/symbol.ts +53 -0
  149. package/src/migrator/to-type.ts +93 -0
  150. package/src/migrator/utils.ts +67 -0
  151. package/src/parser.ts +69 -0
  152. package/src/scope.ts +208 -0
  153. package/src/type.ts +66 -0
  154. package/tests/analyze.ts +15 -0
  155. package/tests/condition.ts +49 -0
  156. package/tests/definition.ts +42 -0
  157. package/tests/eval-complex.ts +150 -0
  158. package/tests/eval.ts +42 -0
  159. package/tests/functions.ts +22 -0
  160. package/tests/import.ts +31 -0
  161. package/tests/scope.ts +33 -0
  162. package/tests/setup.ts +41 -0
  163. package/tests/tsconfig.json +3 -0
  164. package/tests/unwrap.ts +46 -0
  165. package/tsconfig.json +3 -0
@@ -0,0 +1,397 @@
1
+ import {
2
+ type ConstantNode,
3
+ isAccessorNode,
4
+ isArrayNode,
5
+ isConstantNode,
6
+ isObjectNode,
7
+ type FunctionNode,
8
+ type MathNode,
9
+ isOperatorNode,
10
+ } from 'mathjs';
11
+ import type { Options, Result } from './interface.js';
12
+ import type { State } from './state.js';
13
+ import { constantValue, equalText, symbolName } from './utils.js';
14
+ import { migrateAtomic, migrateExpr, migrateNode } from './node.js';
15
+ import { concat } from './concat.js';
16
+ import { toNumber, toString } from './to-type.js';
17
+
18
+ const EXACT_UNARY_FUNCTION = new Map<string, Result['type']>([
19
+ // 只支持标量的 mathjs 函数
20
+ ['sin', 'number'],
21
+ ['cos', 'number'],
22
+ ['tan', 'number'],
23
+ ['sinh', 'number'],
24
+ ['cosh', 'number'],
25
+ ['tanh', 'number'],
26
+ ['asin', 'number'],
27
+ ['acos', 'number'],
28
+ ['atan', 'number'],
29
+ ['asinh', 'number'],
30
+ ['acosh', 'number'],
31
+ ['atanh', 'number'],
32
+ ['sqrt', 'number'],
33
+ ['cbrt', 'number'],
34
+ ['log', 'number'],
35
+
36
+ // 自定义函数,在 mirascript 也提供
37
+ ['keys', 'array'],
38
+ ['values', 'array'],
39
+ ['entries', 'array'],
40
+
41
+ // XStudio 提供的函数
42
+ ['service', 'extern'],
43
+ ]);
44
+
45
+ /** 兜底 */
46
+ function call(state: State, node: FunctionNode<MathNode>, done: (fn: Result, args: readonly Result[]) => void): Result {
47
+ const { fn, args } = node;
48
+ const fnRet = migrateExpr(state, fn);
49
+ const argList = args.map((a) => migrateAtomic(state, a));
50
+ done(fnRet, argList);
51
+ return {
52
+ code: `${fnRet.code}(${argList.map((a) => a.code).join(', ')})`,
53
+ };
54
+ }
55
+
56
+ /** 转换 AST */
57
+ export function migrateCall(state: State, node: FunctionNode<MathNode>, options: Options): Result {
58
+ const { fn, args } = node;
59
+
60
+ const fnName = symbolName(fn);
61
+ if (fnName && !state.locals.has(fnName)) {
62
+ const openE = options.format !== 'no-paren' ? '(' : '';
63
+ const closeE = options.format !== 'no-paren' ? ')' : '';
64
+
65
+ if (EXACT_UNARY_FUNCTION.has(fnName) && args.length === 1) {
66
+ const arg = migrateAtomic(state, args[0]!);
67
+ return {
68
+ type: EXACT_UNARY_FUNCTION.get(fnName)!,
69
+ code: `${fnName}(${arg.code})`,
70
+ };
71
+ } else if (fnName === 're' || fnName === 'im') {
72
+ const m = migrateAtomic(state, args[0]!);
73
+ state.err(`不支持复数`);
74
+ return {
75
+ type: 'number',
76
+ code: `/* ${fnName} */(${m.code})`,
77
+ };
78
+ } else if (fnName === 'date' && args.length === 1) {
79
+ return {
80
+ type: 'number',
81
+ code: `to_timestamp(${migrateAtomic(state, args[0]!).code})`,
82
+ };
83
+ } else if (fnName === 'inv' && args.length === 1) {
84
+ const arg = migrateAtomic(state, args[0]!);
85
+ return {
86
+ type: arg.type,
87
+ code: `matrix.invert(${arg.code})`,
88
+ };
89
+ } else if (fnName === 'transpose' && args.length === 1) {
90
+ const arg = migrateAtomic(state, args[0]!);
91
+ return {
92
+ type: arg.type,
93
+ code: `matrix.transpose(${arg.code})`,
94
+ };
95
+ } else if (fnName === 'diag' && (args.length === 1 || args.length === 2)) {
96
+ const a = args.map((a) => migrateAtomic(state, a).code).join(', ');
97
+ return {
98
+ type: 'array',
99
+ code: `matrix.diagonal(${a})`,
100
+ };
101
+ } else if (fnName === 'zeros' || fnName === 'ones' || fnName === 'identity') {
102
+ return {
103
+ type: 'array',
104
+ code: `matrix.${fnName}(${args.map((a) => migrateAtomic(state, a).code).join(', ')})`,
105
+ };
106
+ } else if (fnName === 'concat') {
107
+ const results = args.map((a) => migrateAtomic(state, a));
108
+ if (results.some((r) => r.type === 'string')) return concat(state, results);
109
+ } else if (fnName === 'numeric' && args.length === 1) {
110
+ return toNumber(state, args[0]!);
111
+ } else if (fnName === 'flatten' && args.length === 1) {
112
+ return {
113
+ type: 'array',
114
+ code: `flatten(${migrateAtomic(state, args[0]!).code})`,
115
+ };
116
+ } else if (fnName === 'norm' && args.length === 1) {
117
+ if (isArrayNode(args[0])) {
118
+ return {
119
+ type: 'number',
120
+ code: `hypot(${args[0].items.map((a) => migrateAtomic(state, a).code).join(', ')})`,
121
+ };
122
+ }
123
+ return {
124
+ type: 'number',
125
+ code: `hypot(${migrateAtomic(state, args[0]!).code})`,
126
+ };
127
+ } else if (fnName === 'equalText' && args.length === 2) {
128
+ const p = equalText(state, '==', args[0]!, args[1]!);
129
+ return {
130
+ type: p.type,
131
+ code: `${openE}${p.code}${closeE}`,
132
+ };
133
+ } else if (fnName === 'toJson' && args.length === 1) {
134
+ return {
135
+ type: 'string',
136
+ code: `to_json(${migrateAtomic(state, args[0]!).code})`,
137
+ };
138
+ } else if (fnName === 'fromJson' && args.length === 1) {
139
+ return {
140
+ code: `from_json(${migrateAtomic(state, args[0]!).code})`,
141
+ };
142
+ } else if (fnName === 'is' && args.length === 2 && isConstantNode(args[1])) {
143
+ const t = String((args[1] as ConstantNode<string | number>).value);
144
+ switch (t) {
145
+ case 'number':
146
+ case 'string':
147
+ case 'boolean':
148
+ return {
149
+ type: 'boolean',
150
+ code: `${openE}type(${migrateAtomic(state, args[0]!).code}) == '${t}'${closeE}`,
151
+ };
152
+ case 'null':
153
+ case 'undefined':
154
+ return {
155
+ type: 'boolean',
156
+ code: `${openE}type(${migrateAtomic(state, args[0]!).code}) == 'nil'${closeE}`,
157
+ };
158
+ case 'Array':
159
+ case 'array':
160
+ state.warn(`'type' 与 'is' 结果可能不同`);
161
+ return {
162
+ type: 'boolean',
163
+ code: `${openE}type(${migrateAtomic(state, args[0]!).code}) == 'array'${closeE}`,
164
+ };
165
+ case 'Object':
166
+ case 'object':
167
+ state.warn(`'type' 与 'is' 结果可能不同`);
168
+ return {
169
+ type: 'boolean',
170
+ code: `${openE}type(${migrateAtomic(state, args[0]!).code}) == 'record'${closeE}`,
171
+ };
172
+ case 'Function':
173
+ case 'function': {
174
+ if (symbolName(args[0]!) === 'service' && !state.locals.has('service')) {
175
+ return {
176
+ type: 'boolean',
177
+ code: `${openE}type(service) == 'extern'${closeE}`,
178
+ };
179
+ }
180
+ state.warn(`'type' 与 'is' 结果可能不同`);
181
+ return {
182
+ type: 'boolean',
183
+ code: `${openE}type(${migrateAtomic(state, args[0]!).code}) == 'function'${closeE}`,
184
+ };
185
+ }
186
+ }
187
+ } else if (fnName === 'typeOf' && args.length === 1) {
188
+ state.warn(`'type' 与 'typeOf' 结果可能不同`);
189
+ return {
190
+ type: 'string',
191
+ code: `type(${migrateAtomic(state, args[0]!).code})`,
192
+ };
193
+ } else if (fnName === 'format' && args.length === 1) {
194
+ return toString(state, args[0]!);
195
+ } else if (fnName === 'print' && args.length === 1) {
196
+ state.loose();
197
+ return toString(state, args[0]!);
198
+ } else if (
199
+ fnName === 'print' &&
200
+ args.length === 2 &&
201
+ isConstantNode(args[0]) &&
202
+ (isObjectNode(args[1]) || isArrayNode(args[1]))
203
+ ) {
204
+ const template = String(args[0].value);
205
+ const values = isObjectNode(args[1])
206
+ ? args[1].properties
207
+ : (args[1].items as unknown as Record<string, MathNode>);
208
+ return {
209
+ type: 'string',
210
+ code:
211
+ '`' +
212
+ template.replaceAll(/`|\$([\w.]+)/g, (original, key: string) => {
213
+ if (original === '`') return '\\`';
214
+ const keys = key.split('.').map((part) => {
215
+ const nPart = Number.parseInt(part);
216
+ if (!Number.isNaN(nPart) && part.length > 0 && nPart >= 1) {
217
+ return nPart - 1;
218
+ } else {
219
+ return part;
220
+ }
221
+ });
222
+ const value = values[keys.shift()!];
223
+ if (value == null) return original.replaceAll('$', String.raw`\$`);
224
+ if (!keys.length) {
225
+ return `$(${migrateAtomic(state, value).code})`;
226
+ }
227
+ return `$(${migrateNode(state, value, { format: 'paren' }).code}.${keys.join('.')})`;
228
+ }) +
229
+ '`',
230
+ };
231
+ } else if (fnName === 'string' && args.length === 1) {
232
+ const inner = migrateAtomic(state, args[0]!);
233
+ return {
234
+ type: 'string',
235
+ code: `to_string(${inner.code})`,
236
+ };
237
+ } else if (
238
+ fnName === 'sum' &&
239
+ args.length === 1 &&
240
+ isArrayNode(args[0]) &&
241
+ args[0].items.length === 1 &&
242
+ isOperatorNode(args[0].items[0]) &&
243
+ ['>=', '>', '<=', '<', '==', '!='].includes(args[0].items[0].op) &&
244
+ typeof constantValue(args[0].items[0].args[1]!) === 'number'
245
+ ) {
246
+ // sum([a > v])
247
+ const a = migrateAtomic(state, args[0].items[0].args[0]!);
248
+ const v = migrateAtomic(state, args[0].items[0].args[1]!);
249
+ return {
250
+ type: 'number',
251
+ code: `sum(map(${a.code}, fn { it ${args[0].items[0].op} ${v.code} }))`,
252
+ };
253
+ }
254
+
255
+ return call(state, node, (fn, args) => {
256
+ const fun = state.globals.get(fnName);
257
+ if (typeof fun === 'function') {
258
+ if (args.some((a) => a.type !== 'number' && a.type !== 'boolean' && a.type !== 'string')) {
259
+ state.warn(`函数行为可能不一致: ${fnName}`);
260
+ }
261
+ } else {
262
+ state.err(`不支持的函数: ${fnName}`);
263
+ }
264
+ });
265
+ }
266
+
267
+ if (
268
+ isAccessorNode(fn) &&
269
+ fn.index.dimensions.length === 1 &&
270
+ isConstantNode(fn.index.dimensions[0]) &&
271
+ typeof fn.index.dimensions[0].value == 'string'
272
+ ) {
273
+ const thisArg = migrateExpr(state, fn.object);
274
+ const fnName = String(fn.index.dimensions[0].value);
275
+ if (fnName === 'toString' && args.length === 0) {
276
+ return {
277
+ type: 'string',
278
+ code: `${thisArg.code}::to_string()`,
279
+ };
280
+ } else if (fnName === 'includes' && args.length === 1) {
281
+ if (thisArg.type === 'array') {
282
+ return {
283
+ type: 'boolean',
284
+ code: `(${migrateExpr(state, args[0]!).code} in ${thisArg.code})`,
285
+ };
286
+ }
287
+ if (thisArg.type === 'string') {
288
+ return {
289
+ type: 'boolean',
290
+ code: `${thisArg.code}::contains(${migrateAtomic(state, args[0]!).code})`,
291
+ };
292
+ }
293
+ state.helper(
294
+ `fn @@includes(it, el) { match it::type() { case 'string' { it::contains(el) } case 'array' { el in it } case _ { false } } }`,
295
+ );
296
+ return {
297
+ type: 'boolean',
298
+ code: `${thisArg.code}::@@includes(${migrateAtomic(state, args[0]!).code})`,
299
+ };
300
+ } else if ((fnName === 'map' || fnName === 'filter') && args.length === 1) {
301
+ return {
302
+ type: 'array',
303
+ code: `${thisArg.code}::${fnName}(${migrateAtomic(state, args[0]!).code})`,
304
+ };
305
+ } else if (fnName === 'find' && args.length === 1) {
306
+ return {
307
+ code: `${thisArg.code}::find(${migrateAtomic(state, args[0]!).code}).1`,
308
+ };
309
+ } else if (fnName === 'findIndex' && args.length === 1) {
310
+ return {
311
+ type: 'number',
312
+ code: `(${thisArg.code}::find(${migrateAtomic(state, args[0]!).code}).0 ?? -1)`,
313
+ };
314
+ } else if (fnName === 'flatMap' && args.length === 1) {
315
+ return {
316
+ type: 'array',
317
+ code: `${thisArg.code}::map(${migrateAtomic(state, args[0]!).code})::flatten()`,
318
+ };
319
+ } else if (fnName === 'reverse' && args.length === 0) {
320
+ return {
321
+ type: 'array',
322
+ code: `${thisArg.code}::reverse()`,
323
+ };
324
+ } else if ((fnName === 'replaceAll' || fnName === 'replace') && args.length === 2) {
325
+ if (fnName === 'replace') {
326
+ state.warn(`MiraScript 的 'replace' 函数会替换全部结果`);
327
+ }
328
+ return {
329
+ type: 'string',
330
+ code: `${thisArg.code}::replace(${migrateAtomic(state, args[0]!).code}, ${migrateAtomic(state, args[1]!).code})`,
331
+ };
332
+ } else if (fnName === 'split' && args.length === 1) {
333
+ return {
334
+ type: 'array',
335
+ code: `${thisArg.code}::split(${migrateAtomic(state, args[0]!).code})`,
336
+ };
337
+ } else if (fnName === 'concat') {
338
+ if (thisArg.type === 'string') {
339
+ return concat(state, [thisArg, ...args]);
340
+ }
341
+ if (thisArg.type !== 'array') {
342
+ const c = concat(state, [thisArg, ...args]);
343
+ state.warn(`字符串连接时应使用插值: ${c.code}`);
344
+ }
345
+ return {
346
+ type: 'array',
347
+ code: `flatten([` + [fn.object, ...args].map((a) => migrateAtomic(state, a).code).join(', ') + `])`,
348
+ };
349
+ } else if (
350
+ fnName === 'toFixed' &&
351
+ (args.length === 0 || (args.length === 1 && isConstantNode(args[0]) && typeof args[0].value == 'number'))
352
+ ) {
353
+ const digits = constantValue(args[0]!) ?? 0;
354
+ return {
355
+ type: 'string',
356
+ code: `${thisArg.code}::format('.${digits}')`,
357
+ };
358
+ } else if (isArrayNode(fn.object) && fnName === 'join' && args.length <= 1) {
359
+ let sep = ',';
360
+ if (args[0]) {
361
+ if (isConstantNode(args[0])) {
362
+ sep = String(args[0].value).replaceAll('`', '\\`');
363
+ } else {
364
+ sep = `$(${migrateAtomic(state, args[0]).code})`;
365
+ }
366
+ }
367
+ let ret = '';
368
+ for (let i = 0; i < fn.object.items.length; i++) {
369
+ const item = fn.object.items[i]!;
370
+ if (i > 0) ret += sep;
371
+ if (isConstantNode(item)) {
372
+ const value = String(item.value).replaceAll('`', '\\`');
373
+ ret += value;
374
+ } else {
375
+ ret += `$(${migrateAtomic(state, item).code})`;
376
+ }
377
+ }
378
+ return {
379
+ type: 'string',
380
+ code: '`' + ret + '`',
381
+ };
382
+ } else if (thisArg.type === 'extern') {
383
+ const argList = args.map((a) => migrateAtomic(state, a).code);
384
+ return {
385
+ code: `${thisArg.code}(${argList.join(', ')})`,
386
+ };
387
+ }
388
+
389
+ state.err(`不支持的方法: ${fnName}`);
390
+ const argList = args.map((a) => migrateAtomic(state, a).code);
391
+ return { code: `${thisArg.code}.${fnName}(${argList.join(', ')})` };
392
+ }
393
+
394
+ return call(state, node, () => {
395
+ state.loose();
396
+ });
397
+ }
@@ -0,0 +1,28 @@
1
+ import { isConstantNode, isFunctionNode, isSymbolNode, type MathNode } from 'mathjs';
2
+ import type { State } from './state.js';
3
+ import { migrateNode } from './node.js';
4
+ import { isResult, type Result } from './interface.js';
5
+
6
+ /** 连接字符串 */
7
+ export function concat(state: State, args: ReadonlyArray<MathNode | string | number | Result>): Result {
8
+ const parts: string[] = [];
9
+ for (const arg of args) {
10
+ if (typeof arg != 'object') {
11
+ parts.push(String(arg).replaceAll('`', '\\`'));
12
+ } else if (isConstantNode(arg)) {
13
+ const value = String(arg.value).replaceAll('`', '\\`');
14
+ parts.push(value);
15
+ } else if (isResult(arg)) {
16
+ if (arg.literal && typeof arg.literal != 'object') {
17
+ parts.push(String(arg.literal).replaceAll('`', '\\`'));
18
+ } else {
19
+ parts.push(`$(${arg.code})`);
20
+ }
21
+ } else if (isFunctionNode(arg) && isSymbolNode(arg.fn) && arg.fn.name === 'string') {
22
+ parts.push(`$(${migrateNode(state, arg.args[0]!, { format: 'no-paren' }).code})`);
23
+ } else {
24
+ parts.push(`$(${migrateNode(state, arg, { format: 'no-paren' }).code})`);
25
+ }
26
+ }
27
+ return { code: '`' + parts.join('') + '`', type: 'string' };
28
+ }
@@ -0,0 +1,120 @@
1
+ import { isFunctionNode, isOperatorNode, OperatorNode, type ConditionalNode } from 'mathjs';
2
+ import type { State } from './state.js';
3
+ import type { Options, Result } from './interface.js';
4
+ import { toBoolean } from './to-type.js';
5
+ import { migrateAtomic } from './node.js';
6
+ import { constantValue } from './utils.js';
7
+
8
+ /** 三目运算 */
9
+ export function migrateCondition(state: State, node: ConditionalNode, options: Options): Result {
10
+ const { condition, trueExpr, falseExpr } = node;
11
+
12
+ if (constantValue(trueExpr) === true && constantValue(falseExpr) === false) {
13
+ // x ? true : false
14
+ const condRet = toBoolean(state, condition);
15
+ let { code } = condRet;
16
+ if (options.format === 'paren') {
17
+ code = `(${code})`;
18
+ }
19
+ return {
20
+ type: 'boolean',
21
+ code,
22
+ };
23
+ }
24
+
25
+ if (constantValue(trueExpr) === false && constantValue(falseExpr) === true) {
26
+ // x ? false : true
27
+ const c2 = new OperatorNode('not', 'not', [condition]);
28
+ const condRet = toBoolean(state, c2);
29
+ let { code } = condRet;
30
+ if (options.format === 'paren') {
31
+ code = `(${code})`;
32
+ }
33
+ return {
34
+ type: 'boolean',
35
+ code,
36
+ };
37
+ }
38
+
39
+ if (
40
+ // is(x, 'string')
41
+ isFunctionNode(condition) &&
42
+ condition.fn.isSymbolNode &&
43
+ condition.fn.name === 'is' &&
44
+ condition.args.length === 2 &&
45
+ constantValue(condition.args[1]!) === 'string' &&
46
+ // ? count(x) != 0
47
+ isOperatorNode(trueExpr) &&
48
+ trueExpr.op === '!=' &&
49
+ isFunctionNode(trueExpr.args[0]!) &&
50
+ trueExpr.args[0].fn.isSymbolNode &&
51
+ trueExpr.args[0].fn.name === 'count' &&
52
+ trueExpr.args[0].args.length === 1 &&
53
+ constantValue(trueExpr.args[1]!) === 0 &&
54
+ // : false
55
+ constantValue(falseExpr) === false &&
56
+ // same arg
57
+ condition.args[0]!.equals(trueExpr.args[0].args[0]!)
58
+ ) {
59
+ // is(x, 'string') ? count(x) != 0 : false
60
+ const x = migrateAtomic(state, condition.args[0]!);
61
+ let code = `type(${x.code}) == 'string' && ${x.code} != ''`;
62
+ if (options.format === 'paren') {
63
+ code = `(${code})`;
64
+ }
65
+ return {
66
+ type: 'boolean',
67
+ code,
68
+ };
69
+ }
70
+
71
+ if (
72
+ // is(x, 'string')
73
+ isFunctionNode(condition) &&
74
+ condition.fn.isSymbolNode &&
75
+ condition.fn.name === 'is' &&
76
+ condition.args.length === 2 &&
77
+ constantValue(condition.args[1]!) === 'string' &&
78
+ // ? equalText(x, "xxx")
79
+ isFunctionNode(trueExpr) &&
80
+ trueExpr.fn.isSymbolNode &&
81
+ trueExpr.fn.name === 'equalText' &&
82
+ trueExpr.args.length === 2 &&
83
+ typeof constantValue(trueExpr.args[1]!) === 'string' &&
84
+ // : false
85
+ constantValue(falseExpr) === false &&
86
+ // same arg
87
+ condition.args[0]!.equals(trueExpr.args[0]!)
88
+ ) {
89
+ // is(x, 'string') ? equalText(x, "xxx") : false
90
+ const x = migrateAtomic(state, condition.args[0]!);
91
+ const xxx = migrateAtomic(state, trueExpr.args[1]!);
92
+ let code = `${x.code} == ${xxx.code}`;
93
+ if (options.format === 'paren') {
94
+ code = `(${code})`;
95
+ }
96
+ return {
97
+ type: 'boolean',
98
+ code,
99
+ };
100
+ }
101
+
102
+ const condRet = toBoolean(state, condition);
103
+ let trueRet = migrateAtomic(state, trueExpr);
104
+ let falseRet = migrateAtomic(state, falseExpr);
105
+ if (!(trueRet.code.startsWith('{') && trueRet.code.endsWith('}'))) {
106
+ trueRet = { ...trueRet, code: `{ ${trueRet.code} }` };
107
+ }
108
+ if (!((falseRet.code.startsWith('{') || falseRet.code.startsWith('if ')) && falseRet.code.endsWith('}'))) {
109
+ falseRet = { ...falseRet, code: `{ ${falseRet.code} }` };
110
+ }
111
+
112
+ let code = `if ${condRet.code} ${trueRet.code} else ${falseRet.code}`;
113
+ if (options.format === 'paren') {
114
+ code = `(${code})`;
115
+ }
116
+ return {
117
+ type: trueRet.type === falseRet.type ? trueRet.type : undefined,
118
+ code,
119
+ };
120
+ }
@@ -0,0 +1,30 @@
1
+ import type { TypeName, VmConst } from '@mirascript/mirascript';
2
+
3
+ /** 转换选项 */
4
+ export interface Options {
5
+ /** 格式 */
6
+ readonly format: 'paren' | 'no-paren' | 'expr';
7
+ }
8
+
9
+ /** 转换结果 */
10
+ export interface Result {
11
+ /** 节点类型 */
12
+ readonly type?: TypeName;
13
+ /** 为字面量 */
14
+ readonly literal?: VmConst;
15
+ /** 转换后的代码 */
16
+ readonly code: string;
17
+ /** 转换后的代码 */
18
+ readonly as_boolean?: string;
19
+ /** 转换后的代码 */
20
+ readonly as_number?: string;
21
+ /** 转换后的代码 */
22
+ readonly as_string?: string;
23
+ }
24
+
25
+ /** 判断是否为转换结果 */
26
+ export function isResult(value: unknown): value is Result {
27
+ if (!value || typeof value != 'object') return false;
28
+ const obj = value as Result;
29
+ return typeof obj.code === 'string';
30
+ }