mirror of
https://gitlab.com/freepascal.org/fpc/pas2js.git
synced 2025-04-05 21:47:47 +02:00
1141 lines
31 KiB
ObjectPascal
1141 lines
31 KiB
ObjectPascal
{
|
|
JOB - JS Object Bridge for Webassembly
|
|
|
|
Webassembly unit giving access to the browser DOM.
|
|
|
|
see https://wiki.freepascal.org/WebAssembly/DOM
|
|
}
|
|
unit JOB_WAsm;
|
|
|
|
{$mode ObjFPC}{$H+}
|
|
|
|
{$define VerboseJOB}
|
|
|
|
interface
|
|
|
|
uses
|
|
SysUtils, Types, Math, Classes, JOB_Shared;
|
|
|
|
const
|
|
MinSafeIntDouble = -$1fffffffffffff; // -9007199254740991 54 bits (52 plus signed bit plus implicit highest bit)
|
|
MaxSafeIntDouble = $1fffffffffffff; // 9007199254740991
|
|
|
|
Type
|
|
PJOBObjectID = ^TJOBObjectID;
|
|
|
|
EJSObject = class(Exception);
|
|
EJSInvoke = class(EJSObject)
|
|
public
|
|
ObjectID: TJOBObjectID;
|
|
FuncName: string;
|
|
end;
|
|
|
|
TJOB_JSValueKind = (
|
|
jjvkUndefined,
|
|
jjvkBoolean,
|
|
jjvkDouble,
|
|
jjvkString,
|
|
jjvkObject,
|
|
jivkMethod
|
|
);
|
|
TJOB_JSValueKinds = set of TJOB_JSValueKind;
|
|
|
|
const
|
|
JOB_JSValueKindNames: array[TJOB_JSValueKind] of string = (
|
|
'Undefined',
|
|
'Boolean',
|
|
'Double',
|
|
'String',
|
|
'Object',
|
|
'Callback'
|
|
);
|
|
|
|
JOB_Undefined = Pointer(1);
|
|
|
|
type
|
|
|
|
{ TJOB_JSValue }
|
|
|
|
TJOB_JSValue = class
|
|
public
|
|
Kind: TJOB_JSValueKind;
|
|
constructor Create(aKind: TJOB_JSValueKind);
|
|
function AsString: string; virtual;
|
|
end;
|
|
|
|
{ TJOB_JSValueBoolean }
|
|
|
|
TJOB_JSValueBoolean = class(TJOB_JSValue)
|
|
public
|
|
Value: Boolean;
|
|
constructor Create(aValue: Boolean);
|
|
function AsString: string; override;
|
|
end;
|
|
|
|
{ TJOB_JSValueDouble }
|
|
|
|
TJOB_JSValueDouble = class(TJOB_JSValue)
|
|
public
|
|
Value: Double;
|
|
constructor Create(const aValue: Double);
|
|
function AsString: string; override;
|
|
end;
|
|
|
|
{ TJOB_JSValueString }
|
|
|
|
TJOB_JSValueString = class(TJOB_JSValue)
|
|
public
|
|
Value: UnicodeString;
|
|
constructor Create(const aValue: UnicodeString);
|
|
function AsString: string; override;
|
|
end;
|
|
|
|
IJSObject = interface;
|
|
|
|
{ TJOB_JSValueObject }
|
|
|
|
TJOB_JSValueObject = class(TJOB_JSValue)
|
|
public
|
|
Value: IJSObject;
|
|
constructor Create(aValue: IJSObject);
|
|
function AsString: string; override;
|
|
end;
|
|
|
|
TJOBCallback = function(const aMethod: TMethod; Args: NativeInt): TJOB_JSValue;
|
|
|
|
{ TJOB_JSValueMethod }
|
|
|
|
TJOB_JSValueMethod = class(TJOB_JSValue)
|
|
public
|
|
Value: TMethod;
|
|
Invoke: TJOBCallback;
|
|
constructor Create(const aMethod: TMethod; const AnInvoke: TJOBCallback);
|
|
function AsString: string; override;
|
|
end;
|
|
|
|
TJOBInvokeGetType = (
|
|
jigCall, // call function
|
|
jigGetter, // read property
|
|
jigNew // new operator
|
|
);
|
|
TJOBInvokeSetType = (
|
|
jisCall, // call function
|
|
jisSetter // write property
|
|
);
|
|
|
|
TJSObject = class;
|
|
TJSObjectClass = class of TJSObject;
|
|
|
|
{ IJSObject }
|
|
|
|
IJSObject = interface
|
|
['{BE5CDE03-D471-4AB3-8F27-A5EA637416F7}']
|
|
function GetJSObjectID: TJOBObjectID;
|
|
function GetJSObjectCasted: IJSObject;
|
|
function GetPascalClassName: string;
|
|
procedure InvokeJSNoResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeSetType = jisCall); virtual;
|
|
function InvokeJSBooleanResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): Boolean; virtual;
|
|
function InvokeJSDoubleResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): Double; virtual;
|
|
function InvokeJSUnicodeStringResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): UnicodeString; virtual;
|
|
function InvokeJSObjectResult(const aName: string; Const Args: Array of const; aResultClass: TJSObjectClass; Invoke: TJOBInvokeGetType = jigCall): TJSObject; virtual;
|
|
function InvokeJSValueResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): TJOB_JSValue; virtual;
|
|
function InvokeJSUtf8StringResult(const aName: string; Const args: Array of const; Invoke: TJOBInvokeGetType = jigCall): String; virtual;
|
|
function InvokeJSLongIntResult(const aName: string; Const args: Array of const; Invoke: TJOBInvokeGetType = jigCall): LongInt; virtual;
|
|
function ReadJSPropertyBoolean(const aName: string): boolean; virtual;
|
|
function ReadJSPropertyDouble(const aName: string): double; virtual;
|
|
function ReadJSPropertyUnicodeString(const aName: string): UnicodeString; virtual;
|
|
function ReadJSPropertyObject(const aName: string; aResultClass: TJSObjectClass): TJSObject; virtual;
|
|
function ReadJSPropertyUtf8String(const aName: string): string; virtual;
|
|
function ReadJSPropertyLongInt(const aName: string): LongInt; virtual;
|
|
function ReadJSPropertyValue(const aName: string): TJOB_JSValue; virtual;
|
|
procedure WriteJSPropertyBoolean(const aName: string; Value: Boolean); virtual;
|
|
procedure WriteJSPropertyDouble(const aName: string; Value: Double); virtual;
|
|
procedure WriteJSPropertyUnicodeString(const aName: string; const Value: UnicodeString); virtual;
|
|
procedure WriteJSPropertyUtf8String(const aName: string; const Value: String); virtual;
|
|
procedure WriteJSPropertyObject(const aName: string; Value: TJSObject); virtual;
|
|
procedure WriteJSPropertyLongInt(const aName: string; Value: LongInt); virtual;
|
|
function NewJSObject(Const Args: Array of const; aResultClass: TJSObjectClass): TJSObject; virtual;
|
|
end;
|
|
|
|
{ TJSObject }
|
|
|
|
TJSObject = class(TInterfacedObject,IJSObject)
|
|
private
|
|
FObjectID: TJOBObjectID;
|
|
FCasted: IJSObject;
|
|
protected
|
|
type
|
|
TJOBInvokeOneResultFunc = function(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultP: PByte
|
|
): TJOBResult;
|
|
function GetJSObjectID: TJOBObjectID;
|
|
function GetJSObjectCasted: IJSObject;
|
|
function GetPascalClassName: string;
|
|
function FetchString(Len: NativeInt): UnicodeString;
|
|
function InvokeJSOneResult(const aName: string; Const Args: Array of const;
|
|
const InvokeFunc: TJOBInvokeOneResultFunc; ResultP: PByte; Invoke: TJOBInvokeGetType): TJOBResult;
|
|
procedure InvokeJS_Raise(const aName, Msg: string); virtual;
|
|
procedure InvokeJS_RaiseResultMismatch(const aName: string; Expected, Actual: TJOBResult); virtual;
|
|
procedure InvokeJS_RaiseResultMismatchStr(const aName: string; const Expected, Actual: string); virtual;
|
|
function CreateInvokeJSArgs(const Args: array of const): PByte; virtual;
|
|
public
|
|
constructor Cast(Intf: IJSObject);
|
|
constructor CreateFromID(aID: TJOBObjectID); virtual;
|
|
destructor Destroy; override;
|
|
property ObjectID: TJOBObjectID read FObjectID;
|
|
// call a function
|
|
procedure InvokeJSNoResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeSetType = jisCall); virtual;
|
|
function InvokeJSBooleanResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): Boolean; virtual;
|
|
function InvokeJSDoubleResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): Double; virtual;
|
|
function InvokeJSUnicodeStringResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): UnicodeString; virtual;
|
|
function InvokeJSObjectResult(const aName: string; Const Args: Array of const; aResultClass: TJSObjectClass; Invoke: TJOBInvokeGetType = jigCall): TJSObject; virtual;
|
|
function InvokeJSValueResult(const aName: string; Const Args: Array of const; Invoke: TJOBInvokeGetType = jigCall): TJOB_JSValue; virtual;
|
|
function InvokeJSUtf8StringResult(const aName: string; Const args: Array of const; Invoke: TJOBInvokeGetType = jigCall): String; virtual;
|
|
function InvokeJSLongIntResult(const aName: string; Const args: Array of const; Invoke: TJOBInvokeGetType = jigCall): LongInt; virtual;
|
|
// read a property
|
|
function ReadJSPropertyBoolean(const aName: string): boolean; virtual;
|
|
function ReadJSPropertyDouble(const aName: string): double; virtual;
|
|
function ReadJSPropertyUnicodeString(const aName: string): UnicodeString; virtual;
|
|
function ReadJSPropertyObject(const aName: string; aResultClass: TJSObjectClass): TJSObject; virtual;
|
|
function ReadJSPropertyUtf8String(const aName: string): string; virtual;
|
|
function ReadJSPropertyLongInt(const aName: string): LongInt; virtual;
|
|
function ReadJSPropertyValue(const aName: string): TJOB_JSValue; virtual;
|
|
// write a property
|
|
procedure WriteJSPropertyBoolean(const aName: string; Value: Boolean); virtual;
|
|
procedure WriteJSPropertyDouble(const aName: string; Value: Double); virtual;
|
|
procedure WriteJSPropertyUnicodeString(const aName: string; const Value: UnicodeString); virtual;
|
|
procedure WriteJSPropertyUtf8String(const aName: string; const Value: String); virtual;
|
|
procedure WriteJSPropertyObject(const aName: string; Value: TJSObject); virtual;
|
|
procedure WriteJSPropertyLongInt(const aName: string; Value: LongInt); virtual;
|
|
procedure WriteJSPropertyValue(const aName: string; Value: TJOB_JSValue); virtual;
|
|
// create a new object using the new-operator
|
|
function NewJSObject(Const Args: Array of const; aResultClass: TJSObjectClass): TJSObject; virtual;
|
|
end;
|
|
|
|
var
|
|
JSObject: TJSObject;
|
|
|
|
// imported functions from browser
|
|
function __job_invoke_noresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeNoResult;
|
|
|
|
function __job_invoke_boolresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultByteBoolP: PByte
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeBooleanResult;
|
|
|
|
function __job_invoke_doubleresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultDoubleP: PByte
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeDoubleResult;
|
|
|
|
function __job_invoke_stringresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultLenP: PByte // nativeint
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeStringResult;
|
|
|
|
function __job_getstringresult(
|
|
ResultP: PByte
|
|
): TJOBResult; external JOBExportName name JOBFn_GetStringResult;
|
|
|
|
function __job_releasestringresult(
|
|
): TJOBResult; external JOBExportName name JOBFn_ReleaseStringResult;
|
|
|
|
function __job_invoke_objectresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultObjIDP: PByte // nativeint
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeObjectResult;
|
|
|
|
function __job_release_object(
|
|
ObjID: TJOBObjectID
|
|
): TJOBResult; external JOBExportName name JOBFn_ReleaseObject;
|
|
|
|
function __job_invoke_jsvalueresult(
|
|
ObjID: TJOBObjectID;
|
|
NameP: PChar;
|
|
NameLen: longint;
|
|
Invoke: longint;
|
|
ArgP: PByte;
|
|
ResultP: PByte // various
|
|
): TJOBResult; external JOBExportName name JOBFn_InvokeJSValueResult;
|
|
|
|
function MyCallBack(ObjID: TJOBObjectID): boolean;
|
|
|
|
implementation
|
|
|
|
const
|
|
InvokeGetToInt: array[TJOBInvokeGetType] of integer = (
|
|
JOBInvokeCall,
|
|
JOBInvokeGet,
|
|
JOBInvokeNew
|
|
);
|
|
InvokeSetToInt: array[TJOBInvokeSetType] of integer = (
|
|
JOBInvokeCall,
|
|
JOBInvokeSet
|
|
);
|
|
|
|
{$IFDEF VerboseJOB}
|
|
function GetVarRecName(vt: word): string;
|
|
begin
|
|
case vt of
|
|
vtInteger: Result:='vtInteger';
|
|
vtBoolean: Result:='vtBoolean';
|
|
vtChar: Result:='vtChar';
|
|
{$ifndef FPUNONE}
|
|
vtExtended: Result:='vtExtended';
|
|
{$endif}
|
|
vtString: Result:='vtString';
|
|
vtPointer: Result:='vtPointer';
|
|
vtPChar: Result:='vtPChar';
|
|
vtObject: Result:='vtObject';
|
|
vtClass: Result:='vtClass';
|
|
vtWideChar: Result:='vtWideChar';
|
|
vtPWideChar: Result:='vtPWideChar';
|
|
vtAnsiString: Result:='vtAnsiString';
|
|
vtCurrency: Result:='vtCurrency';
|
|
vtVariant: Result:='vtVariant';
|
|
vtInterface: Result:='vtInterface';
|
|
vtWideString: Result:='vtWideString';
|
|
vtInt64: Result:='vtInt64';
|
|
vtQWord: Result:='vtQWord';
|
|
vtUnicodeString: Result:='vtUnicodeString';
|
|
else
|
|
Result:='vt?';
|
|
end;
|
|
end;
|
|
|
|
function __job_callback(w: NativeInt): boolean;
|
|
begin
|
|
writeln('__job_callback w=',w);
|
|
Result:=true;
|
|
end;
|
|
|
|
// exported function
|
|
function MyCallBack(ObjID: TJOBObjectID): boolean; //public; alias: JOBFn_CallbackHandler;
|
|
begin
|
|
Result:=ObjID>0;
|
|
end;
|
|
|
|
{ TJOB_JSValueMethod }
|
|
|
|
constructor TJOB_JSValueMethod.Create(const aMethod: TMethod;
|
|
const AnInvoke: TJOBCallback);
|
|
begin
|
|
Kind:=jivkMethod;
|
|
Value:=aMethod;
|
|
Invoke:=AnInvoke;
|
|
end;
|
|
|
|
function TJOB_JSValueMethod.AsString: string;
|
|
begin
|
|
Result:='Callback';
|
|
end;
|
|
|
|
{$ENDIF}
|
|
|
|
{ TJOB_JSValue }
|
|
|
|
constructor TJOB_JSValue.Create(aKind: TJOB_JSValueKind);
|
|
begin
|
|
Kind:=aKind;
|
|
end;
|
|
|
|
function TJOB_JSValue.AsString: string;
|
|
begin
|
|
if Kind=jjvkUndefined then
|
|
Result:='undefined'
|
|
else begin
|
|
Result:='';
|
|
str(Kind,Result);
|
|
end;
|
|
end;
|
|
|
|
{ TJOB_JSValueBoolean }
|
|
|
|
constructor TJOB_JSValueBoolean.Create(aValue: Boolean);
|
|
begin
|
|
Kind:=jjvkBoolean;
|
|
Value:=aValue;
|
|
end;
|
|
|
|
function TJOB_JSValueBoolean.AsString: string;
|
|
begin
|
|
str(Value,Result);
|
|
end;
|
|
|
|
{ TJOB_JSValueDouble }
|
|
|
|
constructor TJOB_JSValueDouble.Create(const aValue: Double);
|
|
begin
|
|
Kind:=jjvkDouble;
|
|
Value:=aValue;
|
|
end;
|
|
|
|
function TJOB_JSValueDouble.AsString: string;
|
|
begin
|
|
str(Value,Result);
|
|
end;
|
|
|
|
{ TJOB_JSValueString }
|
|
|
|
constructor TJOB_JSValueString.Create(const aValue: UnicodeString);
|
|
begin
|
|
Kind:=jjvkString;
|
|
Value:=aValue;
|
|
end;
|
|
|
|
function TJOB_JSValueString.AsString: string;
|
|
begin
|
|
Result:=AnsiQuotedStr(String(Value),'"');
|
|
end;
|
|
|
|
{ TJOB_JSValueObject }
|
|
|
|
constructor TJOB_JSValueObject.Create(aValue: IJSObject);
|
|
begin
|
|
Kind:=jjvkObject;
|
|
Value:=aValue;
|
|
end;
|
|
|
|
function TJOB_JSValueObject.AsString: string;
|
|
begin
|
|
if Value=nil then
|
|
Result:='nil'
|
|
else
|
|
Result:='['+IntToStr(Value.GetJSObjectID)+']:'+Value.GetPascalClassName;
|
|
end;
|
|
|
|
{ TJSObject }
|
|
|
|
function TJSObject.GetJSObjectID: TJOBObjectID;
|
|
begin
|
|
Result:=FObjectID;
|
|
end;
|
|
|
|
function TJSObject.GetJSObjectCasted: IJSObject;
|
|
begin
|
|
Result:=FCasted;
|
|
end;
|
|
|
|
function TJSObject.GetPascalClassName: string;
|
|
begin
|
|
Result:=ClassName;
|
|
end;
|
|
|
|
function TJSObject.FetchString(Len: NativeInt): UnicodeString;
|
|
var
|
|
ok: Boolean;
|
|
begin
|
|
if Len=0 then exit('');
|
|
ok:=false;
|
|
try
|
|
// try to allocate the memory
|
|
SetLength(Result,Len);
|
|
ok:=true;
|
|
finally
|
|
if not ok then
|
|
__job_releasestringresult();
|
|
end;
|
|
__job_getstringresult(PByte(Result));
|
|
end;
|
|
|
|
function TJSObject.InvokeJSOneResult(const aName: string;
|
|
const Args: array of const; const InvokeFunc: TJOBInvokeOneResultFunc;
|
|
ResultP: PByte; Invoke: TJOBInvokeGetType): TJOBResult;
|
|
var
|
|
InvokeArgs: PByte;
|
|
begin
|
|
if length(Args)=0 then
|
|
Result:=InvokeFunc(ObjectID,PChar(aName),length(aName),InvokeGetToInt[Invoke],nil,ResultP)
|
|
else begin
|
|
InvokeArgs:=CreateInvokeJSArgs(Args);
|
|
try
|
|
Result:=InvokeFunc(ObjectID,PChar(aName),length(aName),InvokeGetToInt[Invoke],InvokeArgs,ResultP);
|
|
finally
|
|
if InvokeArgs<>nil then
|
|
FreeMem(InvokeArgs);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
procedure TJSObject.InvokeJS_Raise(const aName, Msg: string);
|
|
var
|
|
E: EJSInvoke;
|
|
begin
|
|
E:=EJSInvoke.Create(Msg);
|
|
E.ObjectID:=ObjectID;
|
|
E.FuncName:=aName;
|
|
raise E;
|
|
end;
|
|
|
|
procedure TJSObject.InvokeJS_RaiseResultMismatch(const aName: string;
|
|
Expected, Actual: TJOBResult);
|
|
begin
|
|
case Actual of
|
|
JOBResult_UnknownObjId: InvokeJS_Raise(aName,'unknown object id '+IntToStr(ObjectID));
|
|
JOBResult_NotAFunction: InvokeJS_Raise(aName,'object '+IntToStr(ObjectID)+' does not have a function "'+aName+'"');
|
|
else
|
|
InvokeJS_RaiseResultMismatchStr(aName,JOBResult_Names[Expected],JOBResult_Names[Actual]);
|
|
end;
|
|
end;
|
|
|
|
procedure TJSObject.InvokeJS_RaiseResultMismatchStr(const aName: string;
|
|
const Expected, Actual: string);
|
|
begin
|
|
InvokeJS_Raise(aName,'expected '+Expected+', but got '+Actual+' from object '+IntToStr(ObjectID)+' function "'+aName+'"');
|
|
end;
|
|
|
|
function TJSObject.CreateInvokeJSArgs(const Args: array of const): PByte;
|
|
|
|
procedure RaiseNotSupported(const Msg: string);
|
|
begin
|
|
raise EJSInvoke.Create('Invoke js: type not supported '+Msg);
|
|
end;
|
|
|
|
procedure RaiseRange;
|
|
begin
|
|
raise ERangeError.Create('Invoke js: number out of bounds');
|
|
end;
|
|
|
|
var
|
|
p: PByte;
|
|
|
|
procedure AddBoolean(b: boolean);
|
|
begin
|
|
if b then
|
|
p^:=JOBArgTrue
|
|
else
|
|
p^:=JOBArgFalse;
|
|
inc(p);
|
|
end;
|
|
|
|
procedure AddDouble(const d: double);
|
|
begin
|
|
p^:=JOBArgDouble;
|
|
inc(p);
|
|
PDouble(p)^:=d;
|
|
inc(p,8);
|
|
end;
|
|
|
|
procedure AddChar(c: word);
|
|
begin
|
|
p^:=JOBArgChar;
|
|
inc(p);
|
|
PWord(p)^:=c;
|
|
inc(p,2);
|
|
end;
|
|
|
|
procedure AddObjectID(const ObjId: TJOBObjectID);
|
|
begin
|
|
p^:=JOBArgObject;
|
|
inc(p);
|
|
PNativeInt(p)^:=ObjId;
|
|
inc(p,sizeof(NativeInt));
|
|
end;
|
|
|
|
procedure AddIJSObject(const Intf: IJSObject);
|
|
begin
|
|
if Intf=nil then
|
|
begin
|
|
p^:=JOBArgNil;
|
|
inc(p);
|
|
end else
|
|
AddObjectID(Intf.GetJSObjectID);
|
|
end;
|
|
|
|
procedure AddUTF8String(s: PByte; Len: NativeInt);
|
|
begin
|
|
p^:=JOBArgUTF8String;
|
|
inc(p);
|
|
PNativeInt(p)^:=Len;
|
|
inc(p,sizeof(NativeInt));
|
|
PPointer(p)^:=s;
|
|
inc(p,sizeof(Pointer));
|
|
end;
|
|
|
|
procedure AddUnicodeString(s: PByte; Len: NativeInt);
|
|
begin
|
|
p^:=JOBArgUnicodeString;
|
|
inc(p);
|
|
PNativeInt(p)^:=Len;
|
|
inc(p,sizeof(NativeInt));
|
|
PPointer(p)^:=s;
|
|
inc(p,sizeof(Pointer));
|
|
end;
|
|
|
|
var
|
|
i, Len: Integer;
|
|
qw: QWord;
|
|
i64: Int64;
|
|
h: PByte;
|
|
s: String;
|
|
ws: WideString;
|
|
us: UnicodeString;
|
|
d: Double;
|
|
Obj: TObject;
|
|
JSValue: TJOB_JSValue;
|
|
aMethod: TJOB_JSValueMethod;
|
|
begin
|
|
Result:=nil;
|
|
if length(Args)>255 then
|
|
raise EJSInvoke.Create('Invoke js: too many args');
|
|
|
|
Len:=1;
|
|
for i:=0 to high(Args) do
|
|
begin
|
|
writeln('TJSObject.CreateInvokeJSArgs ',i,' VType=',Args[i].VType);
|
|
case Args[i].VType of
|
|
vtInteger : inc(Len,5);
|
|
vtBoolean : inc(Len);
|
|
vtChar,
|
|
vtWideChar : inc(Len,3);
|
|
{$ifndef FPUNONE}
|
|
vtExtended:
|
|
begin
|
|
d:=double(Args[i].VExtended^);
|
|
if d=0 then ;
|
|
inc(Len,9);
|
|
end;
|
|
{$endif}
|
|
vtString : inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
vtPointer:
|
|
begin
|
|
p:=Args[i].VPointer;
|
|
if p=JOB_Undefined then
|
|
inc(Len)
|
|
else
|
|
inc(Len,1+SizeOf(PByte));
|
|
end;
|
|
vtPChar:
|
|
begin
|
|
// check length
|
|
strlen(Args[i].VPChar);
|
|
inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
end;
|
|
vtObject:
|
|
begin
|
|
Obj:=Args[i].VObject;
|
|
if Obj=nil then
|
|
inc(Len,1)
|
|
else if Obj is TJSObject then
|
|
inc(Len,1+sizeof(TJOBObjectID))
|
|
else if Obj is TJOB_JSValue then
|
|
begin
|
|
JSValue:=TJOB_JSValue(Obj);
|
|
case JSValue.Kind of
|
|
jjvkUndefined: inc(Len);
|
|
jjvkBoolean: inc(Len);
|
|
jjvkDouble: inc(Len,9);
|
|
jjvkString: inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
jjvkObject:
|
|
if TJOB_JSValueObject(JSValue).Value=nil then
|
|
inc(Len)
|
|
else
|
|
inc(Len,1+sizeof(TJOBObjectID));
|
|
jivkMethod: inc(Len,1+3*SizeOf(PByte));
|
|
end;
|
|
end else
|
|
RaiseNotSupported('object');
|
|
end;
|
|
vtClass : RaiseNotSupported('class');
|
|
vtPWideChar:
|
|
begin
|
|
// check length
|
|
strlen(Args[i].VPWideChar);
|
|
inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
end;
|
|
vtAnsiString:
|
|
inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
vtCurrency : RaiseNotSupported('currency');
|
|
{$ifdef FPC_HAS_FEATURE_VARIANTS}
|
|
vtVariant : RaiseNotSupported('variant');
|
|
{$endif FPC_HAS_FEATURE_VARIANTS}
|
|
vtInterface:
|
|
begin
|
|
p:=Args[i].VInterface;
|
|
if p=nil then
|
|
inc(Len,1)
|
|
else if IInterface(p) is IJSObject then
|
|
inc(Len,1+sizeof(TJOBObjectID))
|
|
else
|
|
RaiseNotSupported('interface');
|
|
end;
|
|
vtWideString:
|
|
inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
vtInt64:
|
|
begin
|
|
i64:=Args[i].VInt64^;
|
|
if (i64<MinSafeIntDouble) or (i64>MaxSafeIntDouble) then
|
|
RaiseRange;
|
|
if (i64>=low(longint)) and (i64<=high(longint)) then
|
|
inc(Len,5)
|
|
else
|
|
inc(Len,9);
|
|
end;
|
|
vtUnicodeString:
|
|
inc(Len,1+SizeOf(NativeInt)+SizeOf(PByte));
|
|
vtQWord:
|
|
begin
|
|
qw:=Args[i].VQWord^;
|
|
if (qw>MaxSafeIntDouble) then
|
|
RaiseRange;
|
|
if (qw<=high(longint)) then
|
|
inc(Len,5)
|
|
else
|
|
inc(Len,9);
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
Result:=GetMem(Len);
|
|
p:=Result;
|
|
p^:=length(Args);
|
|
inc(p);
|
|
for i:=0 to high(Args) do
|
|
begin
|
|
case Args[i].VType of
|
|
vtInteger:
|
|
begin
|
|
p^:=JOBArgLongint;
|
|
inc(p);
|
|
PLongint(p)^:=Args[i].VInteger;
|
|
inc(p,4);
|
|
end;
|
|
vtBoolean:
|
|
AddBoolean(Args[i].VBoolean);
|
|
{$ifndef FPUNONE}
|
|
vtExtended:
|
|
AddDouble(double(Args[i].VExtended^));
|
|
{$endif}
|
|
vtChar:
|
|
AddChar(ord(Args[i].VChar));
|
|
vtWideChar:
|
|
AddChar(ord(Args[i].VWideChar));
|
|
vtString:
|
|
begin
|
|
// shortstring
|
|
h:=PByte(Args[i].VString);
|
|
AddUTF8String(h+1,h^);
|
|
end;
|
|
vtPointer:
|
|
begin
|
|
h:=Args[i].VPointer;
|
|
if h=nil then
|
|
begin
|
|
p^:=JOBArgNil;
|
|
inc(p);
|
|
end else if h=JOB_Undefined then
|
|
begin
|
|
p^:=JOBArgUndefined;
|
|
inc(p);
|
|
end
|
|
else begin
|
|
p^:=JOBArgPointer;
|
|
inc(p);
|
|
PPointer(p)^:=h;
|
|
inc(p,sizeof(Pointer));
|
|
end;
|
|
end;
|
|
vtPChar:
|
|
begin
|
|
h:=PByte(Args[i].VPChar);
|
|
AddUTF8String(h,strlen(PChar(h)));
|
|
end;
|
|
vtObject:
|
|
begin
|
|
Obj:=Args[i].VObject;
|
|
if Obj=nil then
|
|
begin
|
|
p^:=JOBArgNil;
|
|
inc(p);
|
|
end else if Obj is TJSObject then
|
|
AddObjectID(TJSObject(Obj).ObjectID)
|
|
else if Obj is TJOB_JSValue then
|
|
begin
|
|
JSValue:=TJOB_JSValue(Obj);
|
|
case JSValue.Kind of
|
|
jjvkUndefined:
|
|
begin
|
|
p^:=JOBArgUndefined;
|
|
inc(Len);
|
|
end;
|
|
jjvkBoolean:
|
|
AddBoolean(TJOB_JSValueBoolean(Obj).Value);
|
|
jjvkDouble:
|
|
AddDouble(TJOB_JSValueDouble(Obj).Value);
|
|
jjvkString:
|
|
begin
|
|
us:=TJOB_JSValueString(Obj).Value;
|
|
h:=PByte(PWideChar(us));
|
|
AddUnicodeString(h,length(us));
|
|
end;
|
|
jjvkObject:
|
|
AddIJSObject(TJOB_JSValueObject(Obj).Value);
|
|
jivkMethod:
|
|
begin
|
|
aMethod:=TJOB_JSValueMethod(Obj);
|
|
p^:=JOBArgMethod;
|
|
inc(p);
|
|
PPointer(p)^:=Pointer(aMethod.Invoke);
|
|
inc(p,sizeof(Pointer));
|
|
PPointer(p)^:=aMethod.Value.Data;
|
|
inc(p,sizeof(Pointer));
|
|
PPointer(p)^:=aMethod.Value.Code;
|
|
inc(p,sizeof(Pointer));
|
|
end;
|
|
end;
|
|
end else
|
|
RaiseNotSupported(Obj.ClassName);
|
|
end;
|
|
vtClass: ;
|
|
vtPWideChar:
|
|
begin
|
|
h:=PByte(Args[i].VPWideChar);
|
|
AddUnicodeString(h,strlen(PWideChar(h)));
|
|
end;
|
|
vtAnsiString:
|
|
begin
|
|
h:=Args[i].VAnsiString;
|
|
s:=AnsiString(h);
|
|
AddUTF8String(h,length(s));
|
|
end;
|
|
vtCurrency : ;
|
|
{$ifdef FPC_HAS_FEATURE_VARIANTS}
|
|
vtVariant : ;
|
|
{$endif FPC_HAS_FEATURE_VARIANTS}
|
|
vtInterface:
|
|
begin
|
|
h:=Args[i].VInterface;
|
|
if h=nil then
|
|
begin
|
|
p^:=JOBArgNil;
|
|
inc(p);
|
|
end else begin
|
|
AddIJSObject(IJSObject(h));
|
|
end;
|
|
end;
|
|
vtWideString:
|
|
begin
|
|
h:=Args[i].VWideString;
|
|
ws:=WideString(h);
|
|
AddUnicodeString(h,length(ws));
|
|
end;
|
|
vtInt64:
|
|
begin
|
|
i64:=Args[i].VInt64^;
|
|
if (i64>=low(longint)) and (i64<=high(longint)) then
|
|
begin
|
|
p^:=JOBArgLongint;
|
|
inc(p);
|
|
PLongint(p)^:=i64;
|
|
inc(p,4);
|
|
end else begin
|
|
p^:=JOBArgDouble;
|
|
inc(p);
|
|
PDouble(p)^:=i64;
|
|
inc(p,8);
|
|
end;
|
|
end;
|
|
vtUnicodeString:
|
|
begin
|
|
h:=Args[i].VUnicodeString;
|
|
us:=UnicodeString(h);
|
|
AddUnicodeString(h,length(us));
|
|
end;
|
|
vtQWord:
|
|
begin
|
|
qw:=Args[i].VQWord^;
|
|
if (qw<=high(longint)) then
|
|
begin
|
|
p^:=JOBArgLongint;
|
|
inc(p);
|
|
PLongint(p)^:=qw;
|
|
inc(p,4);
|
|
end else begin
|
|
p^:=JOBArgDouble;
|
|
inc(p);
|
|
PDouble(p)^:=qw;
|
|
inc(p,8);
|
|
end;
|
|
end;
|
|
end;
|
|
end;
|
|
|
|
{$IFDEF VerboseInvokeJSArgs}
|
|
s:='TJSObject.CreateInvokeJSArgs ArgCnt='+IntToStr(length(Args));
|
|
for i:=0 to high(Args) do
|
|
s:=s+' '+GetVarRecName(Args[i].VType);
|
|
s:=s+' Len='+IntToStr(Len);
|
|
s:=s+' Bytes=';
|
|
for i:=0 to Len-1 do
|
|
s:=s+HexStr(Result[i],2);
|
|
writeln(s);
|
|
{$ENDIF}
|
|
end;
|
|
|
|
constructor TJSObject.Cast(Intf: IJSObject);
|
|
begin
|
|
FObjectID:=Intf.GetJSObjectID;
|
|
FCasted:=Intf.GetJSObjectCasted;
|
|
if FCasted=nil then
|
|
FCasted:=Intf;
|
|
end;
|
|
|
|
constructor TJSObject.CreateFromID(aID: TJOBObjectID);
|
|
begin
|
|
FObjectID:=aID;
|
|
end;
|
|
|
|
destructor TJSObject.Destroy;
|
|
begin
|
|
if FCasted<>nil then
|
|
FCasted:=nil
|
|
else if ObjectID>=0 then
|
|
__job_release_object(ObjectID);
|
|
FObjectID:=0;
|
|
inherited Destroy;
|
|
end;
|
|
|
|
procedure TJSObject.InvokeJSNoResult(const aName: string;
|
|
const Args: array of const; Invoke: TJOBInvokeSetType);
|
|
var
|
|
aError: TJOBResult;
|
|
InvokeArgs: PByte;
|
|
begin
|
|
if length(Args)=0 then
|
|
aError:=__job_invoke_noresult(ObjectID,PChar(aName),length(aName),InvokeSetToInt[Invoke],nil)
|
|
else begin
|
|
InvokeArgs:=CreateInvokeJSArgs(Args);
|
|
try
|
|
aError:=__job_invoke_noresult(ObjectID,PChar(aName),length(aName),InvokeSetToInt[Invoke],InvokeArgs);
|
|
finally
|
|
if InvokeArgs<>nil then
|
|
FreeMem(InvokeArgs);
|
|
end;
|
|
end;
|
|
if aError<>JOBResult_Success then
|
|
InvokeJS_RaiseResultMismatch(aName,JOBResult_Success,aError);
|
|
end;
|
|
|
|
function TJSObject.InvokeJSBooleanResult(const aName: string;
|
|
const Args: array of const; Invoke: TJOBInvokeGetType): Boolean;
|
|
var
|
|
aError: TJOBResult;
|
|
b: bytebool;
|
|
begin
|
|
b:=false;
|
|
aError:=InvokeJSOneResult(aName,Args,@__job_invoke_boolresult,@b,Invoke);
|
|
if aError<>JOBResult_Boolean then
|
|
InvokeJS_RaiseResultMismatch(aName,JOBResult_Boolean,aError);
|
|
Result:=b;
|
|
end;
|
|
|
|
function TJSObject.InvokeJSDoubleResult(const aName: string;
|
|
const Args: array of const; Invoke: TJOBInvokeGetType): Double;
|
|
var
|
|
aError: TJOBResult;
|
|
begin
|
|
Result:=NaN;
|
|
aError:=InvokeJSOneResult(aName,Args,@__job_invoke_doubleresult,@Result,Invoke);
|
|
if aError<>JOBResult_Double then
|
|
InvokeJS_RaiseResultMismatch(aName,JOBResult_Double,aError);
|
|
end;
|
|
|
|
function TJSObject.InvokeJSUnicodeStringResult(const aName: string;
|
|
const Args: array of const; Invoke: TJOBInvokeGetType): UnicodeString;
|
|
var
|
|
ResultLen: NativeInt;
|
|
aError: TJOBResult;
|
|
begin
|
|
ResultLen:=0;
|
|
aError:=InvokeJSOneResult(aName,Args,@__job_invoke_stringresult,@ResultLen,Invoke);
|
|
if aError<>JOBResult_String then
|
|
InvokeJS_RaiseResultMismatch(aName,JOBResult_String,aError);
|
|
Result:=FetchString(ResultLen);
|
|
//writeln('TJSObject.InvokeJSUnicodeStringResult Result="',Result,'"');
|
|
end;
|
|
|
|
function TJSObject.InvokeJSObjectResult(const aName: string;
|
|
const Args: array of const; aResultClass: TJSObjectClass;
|
|
Invoke: TJOBInvokeGetType): TJSObject;
|
|
var
|
|
aError: TJOBResult;
|
|
NewObjId: TJOBObjectID;
|
|
begin
|
|
Result:=nil;
|
|
NewObjId:=-1;
|
|
aError:=InvokeJSOneResult(aName,Args,@__job_invoke_objectresult,@NewObjId,Invoke);
|
|
if aError=JOBResult_Null then
|
|
exit;
|
|
if aError<>JOBResult_Object then
|
|
InvokeJS_RaiseResultMismatch(aName,JOBResult_Object,aError);
|
|
|
|
Result:=aResultClass.CreateFromID(NewObjId);
|
|
end;
|
|
|
|
function TJSObject.InvokeJSValueResult(const aName: string;
|
|
const Args: array of const; Invoke: TJOBInvokeGetType): TJOB_JSValue;
|
|
var
|
|
Buf: array[0..7] of byte;
|
|
p: PByte;
|
|
aError: TJOBResult;
|
|
Obj: TJSObject;
|
|
begin
|
|
Result:=nil;
|
|
FillByte(Buf[0],length(Buf),0);
|
|
p:=@Buf[0];
|
|
aError:=InvokeJSOneResult(aName,Args,@__job_invoke_jsvalueresult,p,Invoke);
|
|
case aError of
|
|
JOBResult_Undefined:
|
|
Result:=TJOB_JSValue.Create(jjvkUndefined);
|
|
JOBResult_Null:
|
|
Result:=TJOB_JSValueObject.Create(nil);
|
|
JOBResult_Boolean:
|
|
Result:=TJOB_JSValueBoolean.Create(p^<>0);
|
|
JOBResult_Double:
|
|
Result:=TJOB_JSValueDouble.Create(PDouble(p)^);
|
|
JOBResult_String:
|
|
Result:=TJOB_JSValueString.Create(FetchString(PNativeInt(p)^));
|
|
JOBResult_Function,
|
|
JOBResult_Object:
|
|
begin
|
|
Obj:=TJSObject.CreateFromID(PJOBObjectID(p)^);
|
|
Result:=TJOB_JSValueObject.Create(Obj);
|
|
end;
|
|
else
|
|
InvokeJS_RaiseResultMismatchStr(aName,'jsvalue',JOBResult_Names[aError]);
|
|
end;
|
|
end;
|
|
|
|
function TJSObject.InvokeJSUtf8StringResult(const aName: string;
|
|
const args: array of const; Invoke: TJOBInvokeGetType): String;
|
|
begin
|
|
Result:=String(InvokeJSUnicodeStringResult(aName,Args,Invoke));
|
|
end;
|
|
|
|
function TJSObject.InvokeJSLongIntResult(const aName: string;
|
|
const args: array of const; Invoke: TJOBInvokeGetType): LongInt;
|
|
var
|
|
d: Double;
|
|
begin
|
|
d:=InvokeJSDoubleResult(aName,Args,Invoke);
|
|
if (Frac(d)<>0) or (d<low(longint)) or (d>high(longint)) then
|
|
InvokeJS_RaiseResultMismatchStr(aName,'longint','double')
|
|
else
|
|
Result:=Trunc(d);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyBoolean(const aName: string): boolean;
|
|
begin
|
|
Result:=InvokeJSBooleanResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyDouble(const aName: string): double;
|
|
begin
|
|
Result:=InvokeJSDoubleResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyUnicodeString(const aName: string
|
|
): UnicodeString;
|
|
begin
|
|
Result:=InvokeJSUnicodeStringResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyObject(const aName: string;
|
|
aResultClass: TJSObjectClass): TJSObject;
|
|
begin
|
|
Result:=InvokeJSObjectResult(aName,[],aResultClass,jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyUtf8String(const aName: string): string;
|
|
begin
|
|
Result:=InvokeJSUtf8StringResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyLongInt(const aName: string): LongInt;
|
|
begin
|
|
Result:=InvokeJSLongIntResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
function TJSObject.ReadJSPropertyValue(const aName: string): TJOB_JSValue;
|
|
begin
|
|
Result:=InvokeJSValueResult(aName,[],jigGetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyBoolean(const aName: string; Value: Boolean);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyDouble(const aName: string; Value: Double);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyUnicodeString(const aName: string;
|
|
const Value: UnicodeString);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyUtf8String(const aName: string;
|
|
const Value: String);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyObject(const aName: string; Value: TJSObject
|
|
);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyLongInt(const aName: string; Value: LongInt);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
procedure TJSObject.WriteJSPropertyValue(const aName: string;
|
|
Value: TJOB_JSValue);
|
|
begin
|
|
InvokeJSNoResult(aName,[Value],jisSetter);
|
|
end;
|
|
|
|
function TJSObject.NewJSObject(const Args: array of const;
|
|
aResultClass: TJSObjectClass): TJSObject;
|
|
begin
|
|
Result:=InvokeJSObjectResult('',Args,aResultClass,jigNew);
|
|
end;
|
|
|
|
initialization
|
|
JSObject:=TJSObject.CreateFromID(JOBObjIdObject);
|
|
|
|
end.
|
|
|