2024-10-04 23:15:50 +08:00
|
|
|
import type { ASTCtx, LuaFunctionBody } from "./ast.ts";
|
2024-09-27 23:09:25 +08:00
|
|
|
import { evalStatement } from "$common/space_lua/eval.ts";
|
2024-10-09 01:53:09 +08:00
|
|
|
import { asyncQuickSort } from "$common/space_lua/util.ts";
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
export type LuaType =
|
|
|
|
| "nil"
|
|
|
|
| "boolean"
|
|
|
|
| "number"
|
|
|
|
| "string"
|
|
|
|
| "table"
|
|
|
|
| "function"
|
|
|
|
| "userdata"
|
|
|
|
| "thread";
|
|
|
|
|
|
|
|
// These types are for documentation only
|
|
|
|
export type LuaValue = any;
|
|
|
|
export type JSValue = any;
|
|
|
|
|
|
|
|
export interface ILuaFunction {
|
|
|
|
call(...args: LuaValue[]): Promise<LuaValue> | LuaValue;
|
2024-10-09 01:53:09 +08:00
|
|
|
toString(): string;
|
2024-10-03 23:55:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export interface ILuaSettable {
|
|
|
|
set(key: LuaValue, value: LuaValue): void;
|
|
|
|
}
|
|
|
|
|
|
|
|
export interface ILuaGettable {
|
|
|
|
get(key: LuaValue): LuaValue | undefined;
|
|
|
|
}
|
|
|
|
|
2024-09-27 23:09:25 +08:00
|
|
|
export class LuaEnv implements ILuaSettable, ILuaGettable {
|
2024-09-30 18:50:54 +08:00
|
|
|
variables = new Map<string, LuaValue>();
|
2024-09-27 15:11:03 +08:00
|
|
|
|
2024-09-30 18:50:54 +08:00
|
|
|
constructor(readonly parent?: LuaEnv) {
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-09-30 18:50:54 +08:00
|
|
|
setLocal(name: string, value: LuaValue) {
|
|
|
|
this.variables.set(name, value);
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-09-30 18:50:54 +08:00
|
|
|
set(key: string, value: LuaValue): void {
|
|
|
|
if (this.variables.has(key) || !this.parent) {
|
|
|
|
this.variables.set(key, value);
|
|
|
|
} else {
|
|
|
|
this.parent.set(key, value);
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
|
2024-09-30 18:50:54 +08:00
|
|
|
get(name: string): LuaValue | undefined {
|
|
|
|
if (this.variables.has(name)) {
|
|
|
|
return this.variables.get(name);
|
|
|
|
}
|
|
|
|
if (this.parent) {
|
|
|
|
return this.parent.get(name);
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
return undefined;
|
|
|
|
}
|
2024-10-05 21:37:36 +08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Lists all keys in the environment including its parents
|
|
|
|
*/
|
|
|
|
keys(): string[] {
|
|
|
|
const keys = Array.from(this.variables.keys());
|
|
|
|
if (this.parent) {
|
|
|
|
return keys.concat(this.parent.keys());
|
|
|
|
}
|
|
|
|
return keys;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export class LuaMultiRes {
|
2024-10-03 23:55:51 +08:00
|
|
|
values: any[];
|
|
|
|
|
|
|
|
constructor(values: LuaValue[] | LuaValue) {
|
|
|
|
if (values instanceof LuaMultiRes) {
|
|
|
|
this.values = values.values;
|
|
|
|
} else {
|
|
|
|
this.values = Array.isArray(values) ? values : [values];
|
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-09-30 18:50:54 +08:00
|
|
|
unwrap(): any {
|
2024-10-09 01:53:09 +08:00
|
|
|
if (this.values.length === 0) {
|
|
|
|
return null;
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
return this.values[0];
|
|
|
|
}
|
2024-10-03 23:55:51 +08:00
|
|
|
|
|
|
|
// Takes an array of either LuaMultiRes or LuaValue and flattens them into a single LuaMultiRes
|
|
|
|
flatten(): LuaMultiRes {
|
|
|
|
const result: any[] = [];
|
|
|
|
for (const value of this.values) {
|
|
|
|
if (value instanceof LuaMultiRes) {
|
|
|
|
result.push(...value.values);
|
|
|
|
} else {
|
|
|
|
result.push(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return new LuaMultiRes(result);
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function singleResult(value: any): any {
|
2024-09-30 18:50:54 +08:00
|
|
|
if (value instanceof LuaMultiRes) {
|
|
|
|
return value.unwrap();
|
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export class LuaFunction implements ILuaFunction {
|
2024-09-30 18:50:54 +08:00
|
|
|
constructor(private body: LuaFunctionBody, private closure: LuaEnv) {
|
|
|
|
}
|
|
|
|
|
|
|
|
call(...args: LuaValue[]): Promise<LuaValue> | LuaValue {
|
|
|
|
// Create a new environment for this function call
|
|
|
|
const env = new LuaEnv(this.closure);
|
|
|
|
// Assign the passed arguments to the parameters
|
|
|
|
for (let i = 0; i < this.body.parameters.length; i++) {
|
|
|
|
let arg = args[i];
|
|
|
|
if (arg === undefined) {
|
|
|
|
arg = null;
|
|
|
|
}
|
2024-10-04 23:15:50 +08:00
|
|
|
env.setLocal(this.body.parameters[i], arg);
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
return evalStatement(this.body.block, env).catch((e: any) => {
|
|
|
|
if (e instanceof LuaReturn) {
|
|
|
|
if (e.values.length === 0) {
|
|
|
|
return;
|
|
|
|
} else if (e.values.length === 1) {
|
|
|
|
return e.values[0];
|
|
|
|
} else {
|
|
|
|
return new LuaMultiRes(e.values);
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
} else {
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2024-10-09 01:53:09 +08:00
|
|
|
|
|
|
|
toString(): string {
|
|
|
|
return `<lua function(${this.body.parameters.join(", ")})>`;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export class LuaNativeJSFunction implements ILuaFunction {
|
2024-09-30 18:50:54 +08:00
|
|
|
constructor(readonly fn: (...args: JSValue[]) => JSValue) {
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
// Performs automatic conversion between Lua and JS values
|
2024-09-30 18:50:54 +08:00
|
|
|
call(...args: LuaValue[]): Promise<LuaValue> | LuaValue {
|
|
|
|
const result = this.fn(...args.map(luaValueToJS));
|
|
|
|
if (result instanceof Promise) {
|
|
|
|
return result.then(jsToLuaValue);
|
|
|
|
} else {
|
|
|
|
return jsToLuaValue(result);
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-10-09 01:53:09 +08:00
|
|
|
|
|
|
|
toString(): string {
|
|
|
|
return `<native js function: ${this.fn.name}>`;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
export class LuaBuiltinFunction implements ILuaFunction {
|
|
|
|
constructor(readonly fn: (...args: LuaValue[]) => LuaValue) {
|
|
|
|
}
|
|
|
|
|
|
|
|
call(...args: LuaValue[]): Promise<LuaValue> | LuaValue {
|
|
|
|
return this.fn(...args);
|
|
|
|
}
|
2024-10-09 01:53:09 +08:00
|
|
|
|
|
|
|
toString(): string {
|
|
|
|
return `<builtin lua function>`;
|
|
|
|
}
|
2024-10-03 23:55:51 +08:00
|
|
|
}
|
|
|
|
|
2024-09-27 23:09:25 +08:00
|
|
|
export class LuaTable implements ILuaSettable, ILuaGettable {
|
2024-09-30 18:50:54 +08:00
|
|
|
// To optimize the table implementation we use a combination of different data structures
|
|
|
|
// When tables are used as maps, the common case is that they are string keys, so we use a simple object for that
|
|
|
|
private stringKeys: Record<string, any>;
|
|
|
|
// Other keys we can support using a Map as a fallback
|
|
|
|
private otherKeys: Map<any, any> | null;
|
|
|
|
// When tables are used as arrays, we use a native JavaScript array for that
|
|
|
|
private arrayPart: any[];
|
|
|
|
|
2024-10-04 23:15:50 +08:00
|
|
|
public metatable: LuaTable | null;
|
2024-09-30 18:50:54 +08:00
|
|
|
|
2024-10-09 01:53:09 +08:00
|
|
|
constructor(init?: any[] | Record<string, any>) {
|
2024-09-30 18:50:54 +08:00
|
|
|
// For efficiency and performance reasons we pre-allocate these (modern JS engines are very good at optimizing this)
|
2024-10-09 01:53:09 +08:00
|
|
|
this.arrayPart = Array.isArray(init) ? init : [];
|
|
|
|
this.stringKeys = init && !Array.isArray(init) ? init : {};
|
2024-09-30 18:50:54 +08:00
|
|
|
this.otherKeys = null; // Only create this when needed
|
|
|
|
this.metatable = null;
|
|
|
|
}
|
|
|
|
|
|
|
|
get length(): number {
|
|
|
|
return this.arrayPart.length;
|
|
|
|
}
|
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
keys(): any[] {
|
|
|
|
const keys: any[] = Object.keys(this.stringKeys);
|
|
|
|
for (let i = 0; i < this.arrayPart.length; i++) {
|
|
|
|
keys.push(i + 1);
|
|
|
|
}
|
|
|
|
if (this.otherKeys) {
|
|
|
|
for (const key of this.otherKeys.keys()) {
|
|
|
|
keys.push(key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return keys;
|
|
|
|
}
|
|
|
|
|
2024-10-05 21:37:36 +08:00
|
|
|
has(key: LuaValue) {
|
|
|
|
if (typeof key === "string") {
|
|
|
|
return this.stringKeys[key] !== undefined;
|
|
|
|
} else if (Number.isInteger(key) && key >= 1) {
|
|
|
|
return this.arrayPart[key - 1] !== undefined;
|
|
|
|
} else if (this.otherKeys) {
|
|
|
|
return this.otherKeys.has(key);
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2024-10-04 23:15:50 +08:00
|
|
|
rawSet(key: LuaValue, value: LuaValue) {
|
2024-09-30 18:50:54 +08:00
|
|
|
if (typeof key === "string") {
|
|
|
|
this.stringKeys[key] = value;
|
|
|
|
} else if (Number.isInteger(key) && key >= 1) {
|
|
|
|
this.arrayPart[key - 1] = value;
|
|
|
|
} else {
|
|
|
|
if (!this.otherKeys) {
|
|
|
|
this.otherKeys = new Map();
|
|
|
|
}
|
|
|
|
this.otherKeys.set(key, value);
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
|
|
|
|
2024-10-04 23:15:50 +08:00
|
|
|
set(key: LuaValue, value: LuaValue): void {
|
|
|
|
// New index handling for metatables
|
|
|
|
if (this.metatable && this.metatable.has("__newindex") && !this.has(key)) {
|
|
|
|
const metaValue = this.metatable.get("__newindex");
|
|
|
|
if (metaValue.call) {
|
|
|
|
metaValue.call(this, key, value);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
this.rawSet(key, value);
|
|
|
|
}
|
|
|
|
|
2024-10-05 21:37:36 +08:00
|
|
|
rawGet(key: LuaValue): LuaValue | null {
|
2024-09-30 18:50:54 +08:00
|
|
|
if (typeof key === "string") {
|
2024-10-05 21:37:36 +08:00
|
|
|
return this.stringKeys[key];
|
2024-09-30 18:50:54 +08:00
|
|
|
} else if (Number.isInteger(key) && key >= 1) {
|
2024-10-05 21:37:36 +08:00
|
|
|
return this.arrayPart[key - 1];
|
2024-09-30 18:50:54 +08:00
|
|
|
} else if (this.otherKeys) {
|
2024-10-05 21:37:36 +08:00
|
|
|
return this.otherKeys.get(key);
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
2024-10-05 21:37:36 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
get(key: LuaValue): LuaValue | null {
|
|
|
|
const value = this.rawGet(key);
|
2024-10-04 23:15:50 +08:00
|
|
|
if (value === undefined || value === null) {
|
|
|
|
// Invoke the meta table
|
|
|
|
if (this.metatable) {
|
|
|
|
const metaValue = this.metatable.get("__index");
|
|
|
|
if (metaValue.call) {
|
2024-10-05 21:37:36 +08:00
|
|
|
return metaValue.call(this, key);
|
2024-10-04 23:15:50 +08:00
|
|
|
} else if (metaValue instanceof LuaTable) {
|
2024-10-05 21:37:36 +08:00
|
|
|
return metaValue.get(key);
|
2024-10-04 23:15:50 +08:00
|
|
|
} else {
|
|
|
|
throw new Error("Meta table __index must be a function or table");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return value;
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
|
2024-10-09 01:53:09 +08:00
|
|
|
insert(value: LuaValue, pos: number) {
|
|
|
|
this.arrayPart.splice(pos - 1, 0, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
remove(pos: number) {
|
|
|
|
this.arrayPart.splice(pos - 1, 1);
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
|
2024-10-09 01:53:09 +08:00
|
|
|
async sort(fn?: ILuaFunction) {
|
|
|
|
if (fn) {
|
|
|
|
this.arrayPart = await asyncQuickSort(this.arrayPart, async (a, b) => {
|
|
|
|
return (await fn.call(a, b)) ? -1 : 1;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
this.arrayPart.sort();
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
|
2024-10-11 21:34:27 +08:00
|
|
|
asJSObject(): Record<string, any> {
|
2024-10-10 02:35:07 +08:00
|
|
|
const result: Record<string, any> = {};
|
|
|
|
for (const key of this.keys()) {
|
|
|
|
result[key] = luaValueToJS(this.get(key));
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2024-10-11 21:34:27 +08:00
|
|
|
asJSArray(): any[] {
|
|
|
|
return this.arrayPart.map(luaValueToJS);
|
|
|
|
}
|
|
|
|
|
2024-10-05 21:37:36 +08:00
|
|
|
toString(): string {
|
|
|
|
if (this.metatable?.has("__tostring")) {
|
|
|
|
const metaValue = this.metatable.get("__tostring");
|
|
|
|
if (metaValue.call) {
|
|
|
|
return metaValue.call(this);
|
|
|
|
} else {
|
|
|
|
throw new Error("Meta table __tostring must be a function");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let result = "{";
|
|
|
|
let first = true;
|
|
|
|
for (const key of this.keys()) {
|
|
|
|
if (first) {
|
|
|
|
first = false;
|
|
|
|
} else {
|
|
|
|
result += ", ";
|
|
|
|
}
|
|
|
|
if (typeof key === "number") {
|
|
|
|
result += luaToString(this.get(key));
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (typeof key === "string") {
|
|
|
|
result += key;
|
|
|
|
} else {
|
|
|
|
result += "[" + key + "]";
|
|
|
|
}
|
|
|
|
result += " = " + luaToString(this.get(key));
|
|
|
|
}
|
|
|
|
result += "}";
|
|
|
|
return result;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
2024-09-27 15:11:03 +08:00
|
|
|
export type LuaLValueContainer = { env: ILuaSettable; key: LuaValue };
|
|
|
|
|
2024-10-10 02:35:07 +08:00
|
|
|
export function luaSet(obj: any, key: any, value: any, ctx: ASTCtx) {
|
|
|
|
if (!obj) {
|
|
|
|
throw new LuaRuntimeError(
|
|
|
|
`Not a settable object: nil`,
|
|
|
|
ctx,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj instanceof LuaTable || obj instanceof LuaEnv) {
|
2024-09-30 18:50:54 +08:00
|
|
|
obj.set(key, value);
|
|
|
|
} else {
|
|
|
|
obj[key] = value;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
2024-10-10 02:35:07 +08:00
|
|
|
export function luaGet(obj: any, key: any, ctx: ASTCtx): any {
|
|
|
|
if (!obj) {
|
|
|
|
throw new LuaRuntimeError(
|
|
|
|
`Attempting to index a nil value`,
|
|
|
|
ctx,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (key === null || key === undefined) {
|
|
|
|
throw new LuaRuntimeError(
|
|
|
|
`Attempting to index with a nil key`,
|
|
|
|
ctx,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (obj instanceof LuaTable || obj instanceof LuaEnv) {
|
2024-09-30 18:50:54 +08:00
|
|
|
return obj.get(key);
|
2024-10-10 02:35:07 +08:00
|
|
|
} else if (typeof key === "number") {
|
|
|
|
return obj[key - 1];
|
2024-09-30 18:50:54 +08:00
|
|
|
} else {
|
|
|
|
return obj[key];
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
export function luaLen(obj: any): number {
|
2024-09-30 18:50:54 +08:00
|
|
|
if (obj instanceof LuaTable) {
|
2024-10-09 01:53:09 +08:00
|
|
|
return obj.length;
|
2024-09-30 18:50:54 +08:00
|
|
|
} else if (Array.isArray(obj)) {
|
|
|
|
return obj.length;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2024-09-24 16:15:22 +08:00
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
|
2024-10-10 02:35:07 +08:00
|
|
|
export function luaCall(fn: any, args: any[], ctx: ASTCtx): any {
|
|
|
|
if (!fn) {
|
|
|
|
throw new LuaRuntimeError(
|
|
|
|
`Attempting to call a nil value`,
|
|
|
|
ctx,
|
|
|
|
);
|
|
|
|
}
|
|
|
|
if (typeof fn === "function") {
|
|
|
|
const jsArgs = args.map(luaValueToJS);
|
|
|
|
// Native JS function
|
|
|
|
return fn(...jsArgs);
|
|
|
|
}
|
|
|
|
return fn.call(...args);
|
|
|
|
}
|
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
export function luaTypeOf(val: any): LuaType {
|
|
|
|
if (val === null || val === undefined) {
|
|
|
|
return "nil";
|
|
|
|
} else if (typeof val === "boolean") {
|
|
|
|
return "boolean";
|
|
|
|
} else if (typeof val === "number") {
|
|
|
|
return "number";
|
|
|
|
} else if (typeof val === "string") {
|
|
|
|
return "string";
|
|
|
|
} else if (val instanceof LuaTable) {
|
|
|
|
return "table";
|
|
|
|
} else if (Array.isArray(val)) {
|
|
|
|
return "table";
|
2024-10-09 01:53:09 +08:00
|
|
|
} else if (typeof val === "function" || val.call) {
|
2024-10-03 23:55:51 +08:00
|
|
|
return "function";
|
|
|
|
} else {
|
|
|
|
return "userdata";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Both `break` and `return` are implemented by exception throwing
|
2024-09-27 15:11:03 +08:00
|
|
|
export class LuaBreak extends Error {
|
|
|
|
}
|
|
|
|
|
2024-09-27 23:09:25 +08:00
|
|
|
export class LuaReturn extends Error {
|
2024-09-30 18:50:54 +08:00
|
|
|
constructor(readonly values: LuaValue[]) {
|
|
|
|
super();
|
|
|
|
}
|
2024-09-27 23:09:25 +08:00
|
|
|
}
|
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
export class LuaRuntimeError extends Error {
|
|
|
|
constructor(
|
2024-10-10 18:52:28 +08:00
|
|
|
override readonly message: string,
|
2024-10-04 23:15:50 +08:00
|
|
|
public context: ASTCtx,
|
|
|
|
cause?: Error,
|
2024-10-03 23:55:51 +08:00
|
|
|
) {
|
2024-10-04 23:15:50 +08:00
|
|
|
super(message, cause);
|
2024-10-03 23:55:51 +08:00
|
|
|
}
|
|
|
|
|
2024-10-10 18:52:28 +08:00
|
|
|
override toString() {
|
2024-10-04 23:15:50 +08:00
|
|
|
return `LuaRuntimeError: ${this.message} at ${this.context.from}, ${this.context.to}`;
|
2024-10-03 23:55:51 +08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2024-09-27 15:11:03 +08:00
|
|
|
export function luaTruthy(value: any): boolean {
|
2024-09-30 18:50:54 +08:00
|
|
|
if (value === undefined || value === null || value === false) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
if (value instanceof LuaTable) {
|
|
|
|
return value.length > 0;
|
|
|
|
}
|
|
|
|
return true;
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
|
|
|
|
2024-10-03 23:55:51 +08:00
|
|
|
export function luaToString(value: any): string {
|
2024-10-05 21:37:36 +08:00
|
|
|
if (value === null || value === undefined) {
|
|
|
|
return "nil";
|
|
|
|
}
|
|
|
|
if (value.toString) {
|
|
|
|
return value.toString();
|
|
|
|
}
|
2024-10-03 23:55:51 +08:00
|
|
|
return String(value);
|
|
|
|
}
|
|
|
|
|
2024-09-27 15:11:03 +08:00
|
|
|
export function jsToLuaValue(value: any): any {
|
2024-10-05 21:37:36 +08:00
|
|
|
if (value instanceof Promise) {
|
|
|
|
return value.then(luaValueToJS);
|
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
if (value instanceof LuaTable) {
|
|
|
|
return value;
|
|
|
|
} else if (Array.isArray(value)) {
|
2024-10-09 01:53:09 +08:00
|
|
|
const table = new LuaTable();
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
|
|
table.set(i + 1, jsToLuaValue(value[i]));
|
|
|
|
}
|
|
|
|
return table;
|
2024-09-30 18:50:54 +08:00
|
|
|
} else if (typeof value === "object") {
|
2024-10-09 01:53:09 +08:00
|
|
|
const table = new LuaTable();
|
|
|
|
for (const key in value) {
|
|
|
|
table.set(key, jsToLuaValue(value[key]));
|
|
|
|
}
|
|
|
|
return table;
|
|
|
|
} else if (typeof value === "function") {
|
|
|
|
return new LuaNativeJSFunction(value);
|
2024-09-30 18:50:54 +08:00
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
|
|
|
|
2024-10-09 01:53:09 +08:00
|
|
|
// Inverse of jsToLuaValue
|
2024-09-27 15:11:03 +08:00
|
|
|
export function luaValueToJS(value: any): any {
|
2024-10-05 21:37:36 +08:00
|
|
|
if (value instanceof Promise) {
|
|
|
|
return value.then(luaValueToJS);
|
|
|
|
}
|
2024-09-30 18:50:54 +08:00
|
|
|
if (value instanceof LuaTable) {
|
2024-10-09 01:53:09 +08:00
|
|
|
// We'll go a bit on heuristics here
|
|
|
|
// If the table has a length > 0 we'll assume it's a pure array
|
|
|
|
// Otherwise we'll assume it's a pure object
|
2024-09-30 18:50:54 +08:00
|
|
|
if (value.length > 0) {
|
2024-10-09 01:53:09 +08:00
|
|
|
const result = [];
|
|
|
|
for (let i = 0; i < value.length; i++) {
|
|
|
|
result.push(luaValueToJS(value.get(i + 1)));
|
|
|
|
}
|
|
|
|
return result;
|
2024-09-27 15:11:03 +08:00
|
|
|
} else {
|
2024-10-09 01:53:09 +08:00
|
|
|
const result: Record<string, any> = {};
|
|
|
|
for (const key of value.keys()) {
|
|
|
|
result[key] = luaValueToJS(value.get(key));
|
|
|
|
}
|
|
|
|
return result;
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|
2024-10-09 01:53:09 +08:00
|
|
|
} else if (value instanceof LuaNativeJSFunction) {
|
|
|
|
return (...args: any[]) => {
|
|
|
|
return jsToLuaValue(value.fn(...args.map(luaValueToJS)));
|
|
|
|
};
|
2024-09-30 18:50:54 +08:00
|
|
|
} else {
|
|
|
|
return value;
|
|
|
|
}
|
2024-09-27 15:11:03 +08:00
|
|
|
}
|