1
0
mirror of https://github.com/robertkrimen/otto synced 2025-10-19 19:55:30 +08:00
otto/type_function.go
Robert Krimen a879744c20 Add Go <=> JavaScript type interaction
Via reflection for struct, map, and slice/array
Fix #10
2013-02-04 10:31:44 -08:00

197 lines
5.1 KiB
Go

package otto
import ()
type _functionObject struct {
Call _callFunction
Construct _constructFunction
}
func (runtime *_runtime) newNativeFunctionObject(native _nativeFunction, length int, name string) *_object {
self := runtime.newClassObject("Function")
self._Function = &_functionObject{
Call: newNativeCallFunction(native, name),
Construct: defaultConstructFunction,
}
self.stash.set("length", toValue(length), _propertyMode(0))
return self
}
func (runtime *_runtime) newNodeFunctionObject(node *_functionNode, scopeEnvironment _environment) *_object {
self := runtime.newClassObject("Function")
self._Function = &_functionObject{
Call: newNodeCallFunction(node, scopeEnvironment),
Construct: defaultConstructFunction,
}
self.stash.set("length", toValue(len(node.ParameterList)), _propertyMode(0))
return self
}
func (self *_object) Call(this Value, argumentList ...interface{}) Value {
if self._Function == nil {
panic(newTypeError("%v is not a function", toValue(self)))
}
return self.runtime.Call(self, this, self.runtime.toValueArray(argumentList...))
// ... -> runtime -> self.Function.Call.Dispatch -> ...
}
func (self *_object) Construct(this Value, argumentList ...interface{}) Value {
if self._Function == nil {
panic(newTypeError("%v is not a function", toValue(self)))
}
return self._Function.Construct(self, this, self.runtime.toValueArray(argumentList...))
}
func defaultConstructFunction(self *_object, this Value, argumentList []Value) Value {
newObject := self.runtime.newObject()
newObject.class = "Object"
prototypeValue := self.get("prototype")
if !prototypeValue.IsObject() {
prototypeValue = toValue(self.runtime.Global.ObjectPrototype)
}
newObject.prototype = prototypeValue._object()
newObjectValue := toValue(newObject)
result := self.Call(newObjectValue, argumentList)
if result.IsObject() {
return result
}
return newObjectValue
}
func (self *_object) CallGet(name string) Value {
return self.runtime.Call(self, toValue(self), []Value{toValue(name)})
}
func (self *_object) CallSet(name string, value Value) {
self.runtime.Call(self, toValue(self), []Value{toValue(name), value})
}
// 15.3.5.3
func (self *_object) HasInstance(of Value) bool {
if !of.IsObject() {
panic(newTypeError())
}
prototype := self.get("prototype")
if !prototype.IsObject() {
panic(newTypeError())
}
ofPrototype := of._object().prototype
if ofPrototype == nil {
return false
}
return ofPrototype == prototype._object()
}
type _functionSignature string
type _nativeFunction func(FunctionCall) Value
// _constructFunction
type _constructFunction func(*_object, Value, []Value) Value
// _callFunction
type _callFunction interface {
Dispatch(*_functionEnvironment, *_runtime, Value, []Value) Value
Source() string
ScopeEnvironment() _environment
name() string
}
type _callFunction_ struct {
scopeEnvironment _environment // Can be either Lexical or Variable
_name string
}
func (self _callFunction_) ScopeEnvironment() _environment {
return self.scopeEnvironment
}
func (self _callFunction_) name() string {
return self._name
}
// _nativeCallFunction
type _nativeCallFunction struct {
_callFunction_
Native _nativeFunction
}
func newNativeCallFunction(native _nativeFunction, name string) *_nativeCallFunction {
self := &_nativeCallFunction{
Native: native,
}
self._callFunction_._name = name
return self
}
func (self _nativeCallFunction) Dispatch(_ *_functionEnvironment, runtime *_runtime, this Value, argumentList []Value) Value {
return self.Native(FunctionCall{
runtime: runtime,
This: this,
ArgumentList: argumentList,
})
}
func (self _nativeCallFunction) Source() string {
return ""
}
// _nodeCallFunction
type _nodeCallFunction struct {
_callFunction_
node *_functionNode
}
func newNodeCallFunction(node *_functionNode, scopeEnvironment _environment) *_nodeCallFunction {
self := &_nodeCallFunction{
node: node,
}
self.scopeEnvironment = scopeEnvironment
return self
}
func (self _nodeCallFunction) Dispatch(environment *_functionEnvironment, runtime *_runtime, this Value, argumentList []Value) Value {
return runtime._callNode(environment, self.node, this, argumentList)
}
func (self _nodeCallFunction) Source() string {
return ""
}
// FunctionCall{}
// FunctionCall is an enscapulation of a JavaScript function call.
type FunctionCall struct {
runtime *_runtime
This Value
_thisObject *_object
ArgumentList []Value
}
// Argument will return the value of the argument at the given index.
//
// If no such argument exists, undefined is returned.
func (self FunctionCall) Argument(index int) Value {
return valueOfArrayIndex(self.ArgumentList, index)
}
func (self *FunctionCall) thisObject() *_object {
if self._thisObject == nil {
this := self.runtime.GetValue(self.This) // FIXME Is this right?
self._thisObject = self.runtime.toObject(this)
}
return self._thisObject
}
func (self *FunctionCall) thisClassObject(class string) *_object {
thisObject := self.thisObject()
if thisObject.class != class {
panic(newTypeError())
}
return self._thisObject
}
func (self FunctionCall) toObject(value Value) *_object {
return self.runtime.toObject(value)
}