1
0
mirror of https://github.com/robertkrimen/otto synced 2025-10-19 19:55:30 +08:00
otto/object_class.go

377 lines
8.0 KiB
Go

package otto
type _objectClass struct {
getOwnProperty func(*_object, string) *_property
getProperty func(*_object, string) *_property
get func(*_object, string) Value
canPut func(*_object, string) bool
put func(*_object, string, Value, bool)
hasProperty func(*_object, string) bool
hasOwnProperty func(*_object, string) bool
defineOwnProperty func(*_object, string, _property, bool) bool
delete func(*_object, string, bool) bool
enumerate func(*_object, bool, func(string) bool)
clone func(*_object, *_object, *_clone) *_object
}
func objectEnumerate(self *_object, all bool, each func(string) bool) {
for _, name := range self.propertyOrder {
if all || self.property[name].enumerable() {
if !each(name) {
return
}
}
}
}
var (
_classObject,
_classArray,
_classString,
_classArguments,
_classGoStruct,
_classGoMap,
_classGoArray,
_classGoSlice,
_ *_objectClass
)
func init() {
_classObject = &_objectClass{
objectGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
objectDefineOwnProperty,
objectDelete,
objectEnumerate,
objectClone,
}
_classArray = &_objectClass{
objectGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
arrayDefineOwnProperty,
objectDelete,
objectEnumerate,
objectClone,
}
_classString = &_objectClass{
stringGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
objectDefineOwnProperty,
objectDelete,
stringEnumerate,
objectClone,
//stringClone,
}
_classArguments = &_objectClass{
argumentsGetOwnProperty,
objectGetProperty,
argumentsGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
argumentsDefineOwnProperty,
argumentsDelete,
argumentsEnumerate,
//argumentsClone
objectClone,
}
_classGoStruct = &_objectClass{
goStructGetOwnProperty,
objectGetProperty,
objectGet,
goStructCanPut,
goStructPut,
objectHasProperty,
objectHasOwnProperty,
objectDefineOwnProperty,
objectDelete,
goStructEnumerate,
objectClone,
}
_classGoMap = &_objectClass{
goMapGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
goMapDefineOwnProperty,
goMapDelete,
goMapEnumerate,
objectClone,
}
_classGoArray = &_objectClass{
goArrayGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
goArrayDefineOwnProperty,
goArrayDelete,
goArrayEnumerate,
objectClone,
}
_classGoSlice = &_objectClass{
goSliceGetOwnProperty,
objectGetProperty,
objectGet,
objectCanPut,
objectPut,
objectHasProperty,
objectHasOwnProperty,
goSliceDefineOwnProperty,
goSliceDelete,
goSliceEnumerate,
objectClone,
}
}
// Allons-y
// 8.12.1
func objectGetOwnProperty(self *_object, name string) *_property {
// Return a _copy_ of the property
property, exists := self._read(name)
if !exists {
return nil
}
return &property
}
// 8.12.2
func objectGetProperty(self *_object, name string) *_property {
property := self.getOwnProperty(name)
if property != nil {
return property
}
if self.prototype != nil {
return self.prototype.getProperty(name)
}
return nil
}
// 8.12.3
func objectGet(self *_object, name string) Value {
property := self.getProperty(name)
if property != nil {
return property.value.(Value)
}
return UndefinedValue()
}
// 8.12.4
func objectCanPut(self *_object, name string) bool {
property := self.getOwnProperty(name)
if property != nil {
switch value := property.value.(type) {
case Value:
return property.writable()
case _propertyGetSet:
return value[1] != nil
default:
panic(newTypeError())
}
}
if self.prototype == nil {
return self.extensible
}
property = self.prototype.getOwnProperty(name)
if property == nil {
return self.extensible
}
switch value := property.value.(type) {
case Value:
if !self.extensible {
return false
}
return property.writable()
case _propertyGetSet:
return value[1] != nil
default:
panic(newTypeError())
}
return false
}
// 8.12.5
func objectPut(self *_object, name string, value Value, throw bool) {
if !self.canPut(name) {
typeErrorResult(throw)
return
}
// TODO This is not exacting to the specification
// TODO Shortcut?
property := self.getOwnProperty(name)
if property == nil {
self.defineProperty(name, value, 0111, throw)
} else {
property.value = value
self.defineOwnProperty(name, *property, throw)
}
}
// 8.12.6
func objectHasProperty(self *_object, name string) bool {
return self.getProperty(name) != nil
}
func objectHasOwnProperty(self *_object, name string) bool {
return self.getOwnProperty(name) != nil
}
// 8.12.9
func objectDefineOwnProperty(self *_object, name string, descriptor _property, throw bool) bool {
property, exists := self._read(name)
{
if !exists {
if !self.extensible {
goto Reject
}
self._write(name, descriptor.value, descriptor.mode)
return true
}
if descriptor.isEmpty() {
return true
}
// TODO Per 8.12.9.6 - We should shortcut here (returning true) if
// the current and new (define) properties are the same
configurable := property.configurable()
if !configurable {
if descriptor.configurable() {
goto Reject
}
// Test that, if enumerable is set on the property descriptor, then it should
// be the same as the existing property
if descriptor.enumerateSet() && descriptor.enumerable() != property.enumerable() {
goto Reject
}
}
value, isDataDescriptor := property.value.(Value)
getSet, _ := property.value.(_propertyGetSet)
if descriptor.isGenericDescriptor() {
// GenericDescriptor
} else if isDataDescriptor != descriptor.isDataDescriptor() {
// FIXME This branch is non-functioning
panic("isDataDescriptor != descriptor.isDataDescriptor")
var interface_ interface{}
if isDataDescriptor {
property.writeOff()
property.value = interface_
} else {
property.writeOn()
property.value = interface_
}
} else if isDataDescriptor && descriptor.isDataDescriptor() {
if !configurable {
if !property.writable() && descriptor.writable() {
goto Reject
}
if !property.writable() {
if !sameValue(value, descriptor.value.(Value)) {
goto Reject
}
}
}
} else {
if !configurable {
defineGetSet, _ := descriptor.value.(_propertyGetSet)
if getSet[0] != defineGetSet[0] || getSet[1] != defineGetSet[1] {
goto Reject
}
}
}
mode1 := descriptor.mode
if mode1&0222 != 0 {
// TODO Factor this out into somewhere testable
// (Maybe put into switch ...)
mode0 := property.mode
if mode1&0200 != 0 {
mode1 |= (mode0 & 0100)
}
if mode1&020 != 0 {
mode1 |= (mode0 & 010)
}
if mode1&02 != 0 {
mode1 |= (mode0 & 01)
}
mode1 &= 0111
}
self._write(name, descriptor.value, mode1)
return true
}
Reject:
if throw {
panic(newTypeError())
}
return false
}
func objectDelete(self *_object, name string, throw bool) bool {
property_ := self.getOwnProperty(name)
if property_ == nil {
return true
}
if property_.configurable() {
self._delete(name)
return true
}
return typeErrorResult(throw)
}
func objectClone(self0 *_object, self1 *_object, clone *_clone) *_object {
*self1 = *self0
self1.runtime = clone.runtime
if self1.prototype != nil {
self1.prototype = clone.object(self0.prototype)
}
self1.property = make(map[string]_property, len(self0.property))
self1.propertyOrder = make([]string, len(self0.propertyOrder))
copy(self1.propertyOrder, self0.propertyOrder)
for index, property := range self0.property {
self1.property[index] = clone.property(property)
}
switch value := self0.value.(type) {
case _functionObject:
self1.value = value.clone(clone)
case _argumentsObject:
self1.value = value.clone(clone)
}
return self1
}