src.dualinventive.com/go/devsim/vendor/github.com/dop251/goja/builtin_function.go

166 lines
4.3 KiB
Go

package goja
import (
"fmt"
)
func (r *Runtime) builtin_Function(args []Value, proto *Object) *Object {
src := "(function anonymous("
if len(args) > 1 {
for _, arg := range args[:len(args)-1] {
src += arg.String() + ","
}
src = src[:len(src)-1]
}
body := ""
if len(args) > 0 {
body = args[len(args)-1].String()
}
src += "){" + body + "})"
return r.toObject(r.eval(src, false, false, _undefined))
}
func (r *Runtime) functionproto_toString(call FunctionCall) Value {
obj := r.toObject(call.This)
repeat:
switch f := obj.self.(type) {
case *funcObject:
return newStringValue(f.src)
case *nativeFuncObject:
return newStringValue(fmt.Sprintf("function %s() { [native code] }", f.nameProp.get(call.This).ToString()))
case *boundFuncObject:
return newStringValue(fmt.Sprintf("function %s() { [native code] }", f.nameProp.get(call.This).ToString()))
case *lazyObject:
obj.self = f.create(obj)
goto repeat
}
r.typeErrorResult(true, "Object is not a function")
return nil
}
func (r *Runtime) toValueArray(a Value) []Value {
obj := r.toObject(a)
l := toUInt32(obj.self.getStr("length"))
ret := make([]Value, l)
for i := uint32(0); i < l; i++ {
ret[i] = obj.self.get(valueInt(i))
}
return ret
}
func (r *Runtime) functionproto_apply(call FunctionCall) Value {
f := r.toCallable(call.This)
var args []Value
if len(call.Arguments) >= 2 {
args = r.toValueArray(call.Arguments[1])
}
return f(FunctionCall{
This: call.Argument(0),
Arguments: args,
})
}
func (r *Runtime) functionproto_call(call FunctionCall) Value {
f := r.toCallable(call.This)
var args []Value
if len(call.Arguments) > 0 {
args = call.Arguments[1:]
}
return f(FunctionCall{
This: call.Argument(0),
Arguments: args,
})
}
func (r *Runtime) boundCallable(target func(FunctionCall) Value, boundArgs []Value) func(FunctionCall) Value {
var this Value
var args []Value
if len(boundArgs) > 0 {
this = boundArgs[0]
args = make([]Value, len(boundArgs)-1)
copy(args, boundArgs[1:])
} else {
this = _undefined
}
return func(call FunctionCall) Value {
a := append(args, call.Arguments...)
return target(FunctionCall{
This: this,
Arguments: a,
})
}
}
func (r *Runtime) boundConstruct(target func([]Value) *Object, boundArgs []Value) func([]Value) *Object {
if target == nil {
return nil
}
var args []Value
if len(boundArgs) > 1 {
args = make([]Value, len(boundArgs)-1)
copy(args, boundArgs[1:])
}
return func(fargs []Value) *Object {
a := append(args, fargs...)
copy(a, args)
return target(a)
}
}
func (r *Runtime) functionproto_bind(call FunctionCall) Value {
obj := r.toObject(call.This)
f := obj.self
var fcall func(FunctionCall) Value
var construct func([]Value) *Object
repeat:
switch ff := f.(type) {
case *funcObject:
fcall = ff.Call
construct = ff.construct
case *nativeFuncObject:
fcall = ff.f
construct = ff.construct
case *boundFuncObject:
f = &ff.nativeFuncObject
goto repeat
case *lazyObject:
f = ff.create(obj)
goto repeat
default:
r.typeErrorResult(true, "Value is not callable: %s", obj.ToString())
}
l := int(toUInt32(obj.self.getStr("length")))
l -= len(call.Arguments) - 1
if l < 0 {
l = 0
}
v := &Object{runtime: r}
ff := r.newNativeFuncObj(v, r.boundCallable(fcall, call.Arguments), r.boundConstruct(construct, call.Arguments), "", nil, l)
v.self = &boundFuncObject{
nativeFuncObject: *ff,
}
//ret := r.newNativeFunc(r.boundCallable(f, call.Arguments), nil, "", nil, l)
//o := ret.self
//o.putStr("caller", r.global.throwerProperty, false)
//o.putStr("arguments", r.global.throwerProperty, false)
return v
}
func (r *Runtime) initFunction() {
o := r.global.FunctionPrototype.self
o.(*nativeFuncObject).prototype = r.global.ObjectPrototype
o._putProp("toString", r.newNativeFunc(r.functionproto_toString, nil, "toString", nil, 0), true, false, true)
o._putProp("apply", r.newNativeFunc(r.functionproto_apply, nil, "apply", nil, 2), true, false, true)
o._putProp("call", r.newNativeFunc(r.functionproto_call, nil, "call", nil, 1), true, false, true)
o._putProp("bind", r.newNativeFunc(r.functionproto_bind, nil, "bind", nil, 1), true, false, true)
r.global.Function = r.newNativeFuncConstruct(r.builtin_Function, "Function", r.global.FunctionPrototype, 1)
r.addToGlobal("Function", r.global.Function)
}