1
0
mirror of https://github.com/robertkrimen/otto synced 2025-10-26 20:28:49 +08:00
otto/builtin_object.go
Harrison Ho 085142ee6c
feat: object assign (#547)
Implement object assign
2025-03-31 08:42:01 +01:00

335 lines
8.7 KiB
Go

package otto
import (
"fmt"
"strconv"
)
// Object
func builtinObject(call FunctionCall) Value {
value := call.Argument(0)
switch value.kind {
case valueUndefined, valueNull:
return objectValue(call.runtime.newObject())
}
return objectValue(call.runtime.toObject(value))
}
func builtinNewObject(obj *object, argumentList []Value) Value {
value := valueOfArrayIndex(argumentList, 0)
switch value.kind {
case valueNull, valueUndefined:
case valueNumber, valueString, valueBoolean:
return objectValue(obj.runtime.toObject(value))
case valueObject:
return value
default:
}
return objectValue(obj.runtime.newObject())
}
func builtinObjectValueOf(call FunctionCall) Value {
return objectValue(call.thisObject())
}
func builtinObjectHasOwnProperty(call FunctionCall) Value {
propertyName := call.Argument(0).string()
thisObject := call.thisObject()
return boolValue(thisObject.hasOwnProperty(propertyName))
}
func builtinObjectIsPrototypeOf(call FunctionCall) Value {
value := call.Argument(0)
if !value.IsObject() {
return falseValue
}
prototype := call.toObject(value).prototype
thisObject := call.thisObject()
for prototype != nil {
if thisObject == prototype {
return trueValue
}
prototype = prototype.prototype
}
return falseValue
}
func builtinObjectPropertyIsEnumerable(call FunctionCall) Value {
propertyName := call.Argument(0).string()
thisObject := call.thisObject()
prop := thisObject.getOwnProperty(propertyName)
if prop != nil && prop.enumerable() {
return trueValue
}
return falseValue
}
func builtinObjectToString(call FunctionCall) Value {
var result string
switch {
case call.This.IsUndefined():
result = "[object Undefined]"
case call.This.IsNull():
result = "[object Null]"
default:
result = fmt.Sprintf("[object %s]", call.thisObject().class)
}
return stringValue(result)
}
func builtinObjectToLocaleString(call FunctionCall) Value {
toString := call.thisObject().get("toString")
if !toString.isCallable() {
panic(call.runtime.panicTypeError("Object.toLocaleString %q is not callable", toString))
}
return toString.call(call.runtime, call.This)
}
func builtinObjectGetPrototypeOf(call FunctionCall) Value {
val := call.Argument(0)
obj := val.object()
if obj == nil {
panic(call.runtime.panicTypeError("Object.GetPrototypeOf is nil"))
}
if obj.prototype == nil {
return nullValue
}
return objectValue(obj.prototype)
}
func builtinObjectGetOwnPropertyDescriptor(call FunctionCall) Value {
val := call.Argument(0)
obj := val.object()
if obj == nil {
panic(call.runtime.panicTypeError("Object.GetOwnPropertyDescriptor is nil"))
}
name := call.Argument(1).string()
descriptor := obj.getOwnProperty(name)
if descriptor == nil {
return Value{}
}
return objectValue(call.runtime.fromPropertyDescriptor(*descriptor))
}
func builtinObjectDefineProperty(call FunctionCall) Value {
val := call.Argument(0)
obj := val.object()
if obj == nil {
panic(call.runtime.panicTypeError("Object.DefineProperty is nil"))
}
name := call.Argument(1).string()
descriptor := toPropertyDescriptor(call.runtime, call.Argument(2))
obj.defineOwnProperty(name, descriptor, true)
return val
}
func builtinObjectDefineProperties(call FunctionCall) Value {
val := call.Argument(0)
obj := val.object()
if obj == nil {
panic(call.runtime.panicTypeError("Object.DefineProperties is nil"))
}
properties := call.runtime.toObject(call.Argument(1))
properties.enumerate(false, func(name string) bool {
descriptor := toPropertyDescriptor(call.runtime, properties.get(name))
obj.defineOwnProperty(name, descriptor, true)
return true
})
return val
}
func builtinObjectCreate(call FunctionCall) Value {
prototypeValue := call.Argument(0)
if !prototypeValue.IsNull() && !prototypeValue.IsObject() {
panic(call.runtime.panicTypeError("Object.Create is nil"))
}
obj := call.runtime.newObject()
obj.prototype = prototypeValue.object()
propertiesValue := call.Argument(1)
if propertiesValue.IsDefined() {
properties := call.runtime.toObject(propertiesValue)
properties.enumerate(false, func(name string) bool {
descriptor := toPropertyDescriptor(call.runtime, properties.get(name))
obj.defineOwnProperty(name, descriptor, true)
return true
})
}
return objectValue(obj)
}
func builtinObjectIsExtensible(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
return boolValue(obj.extensible)
}
panic(call.runtime.panicTypeError("Object.IsExtensible is nil"))
}
func builtinObjectPreventExtensions(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
obj.extensible = false
return val
}
panic(call.runtime.panicTypeError("Object.PreventExtensions is nil"))
}
func builtinObjectAssign(call FunctionCall) Value {
if len(call.ArgumentList) < 1 {
panic(call.runtime.panicTypeError("Object.assign TypeError: Cannot convert undefined or null to object"))
}
target := call.ArgumentList[0]
targetObj := target.object()
if !target.IsObject() && target.IsNull() && target.IsUndefined() {
panic(call.runtime.panicTypeError("Object.assign TypeError: Cannot convert undefined or null to object"))
}
for _, source := range call.ArgumentList[1:] {
if source.IsString() {
sourceStr := source.string()
for i, r := range sourceStr {
targetObj.put(strconv.Itoa(i), stringValue(string(r)), true)
}
} else if source.IsObject() {
sourceObj := source.object()
sourceObj.enumerate(false, func(name string) bool {
descriptor := sourceObj.getOwnProperty(name)
ok := targetObj.defineOwnProperty(name, *descriptor, true)
if !ok {
panic(call.runtime.panicTypeError("Object.assign TypeError: Cannot convert undefined or null to object"))
}
return true
})
}
}
return objectValue(targetObj)
}
func builtinObjectIsSealed(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
if obj.extensible {
return boolValue(false)
}
result := true
obj.enumerate(true, func(name string) bool {
prop := obj.getProperty(name)
if prop.configurable() {
result = false
}
return true
})
return boolValue(result)
}
panic(call.runtime.panicTypeError("Object.IsSealed is nil"))
}
func builtinObjectSeal(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
obj.enumerate(true, func(name string) bool {
if prop := obj.getOwnProperty(name); nil != prop && prop.configurable() {
prop.configureOff()
obj.defineOwnProperty(name, *prop, true)
}
return true
})
obj.extensible = false
return val
}
panic(call.runtime.panicTypeError("Object.Seal is nil"))
}
func builtinObjectIsFrozen(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
if obj.extensible {
return boolValue(false)
}
result := true
obj.enumerate(true, func(name string) bool {
prop := obj.getProperty(name)
if prop.configurable() || prop.writable() {
result = false
}
return true
})
return boolValue(result)
}
panic(call.runtime.panicTypeError("Object.IsFrozen is nil"))
}
func builtinObjectFreeze(call FunctionCall) Value {
val := call.Argument(0)
if obj := val.object(); obj != nil {
obj.enumerate(true, func(name string) bool {
if prop, update := obj.getOwnProperty(name), false; nil != prop {
if prop.isDataDescriptor() && prop.writable() {
prop.writeOff()
update = true
}
if prop.configurable() {
prop.configureOff()
update = true
}
if update {
obj.defineOwnProperty(name, *prop, true)
}
}
return true
})
obj.extensible = false
return val
}
panic(call.runtime.panicTypeError("Object.Freeze is nil"))
}
func builtinObjectKeys(call FunctionCall) Value {
if obj, keys := call.Argument(0).object(), []Value(nil); nil != obj {
obj.enumerate(false, func(name string) bool {
keys = append(keys, stringValue(name))
return true
})
return objectValue(call.runtime.newArrayOf(keys))
}
panic(call.runtime.panicTypeError("Object.Keys is nil"))
}
func builtinObjectValues(call FunctionCall) Value {
if obj, values := call.Argument(0).object(), []Value(nil); nil != obj {
obj.enumerate(false, func(name string) bool {
values = append(values, obj.get(name))
return true
})
return objectValue(call.runtime.newArrayOf(values))
}
panic(call.runtime.panicTypeError("Object.Values is nil"))
}
func builtinObjectGetOwnPropertyNames(call FunctionCall) Value {
if obj, propertyNames := call.Argument(0).object(), []Value(nil); nil != obj {
obj.enumerate(true, func(name string) bool {
if obj.hasOwnProperty(name) {
propertyNames = append(propertyNames, stringValue(name))
}
return true
})
return objectValue(call.runtime.newArrayOf(propertyNames))
}
// Default to empty array for non object types.
return objectValue(call.runtime.newArray(0))
}