|
|
|
/*
|
|
|
|
* This file is part of the KDE libraries
|
|
|
|
* Copyright (C) 1999-2000 Harri Porten (porten@kde.org)
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library; if not, write to the Free Software
|
|
|
|
* Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
|
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "value.h"
|
|
|
|
#include "object.h"
|
|
|
|
#include "types.h"
|
|
|
|
#include "interpreter.h"
|
|
|
|
#include "operations.h"
|
|
|
|
#include "object_object.h"
|
|
|
|
#include "function_object.h"
|
|
|
|
#include "lookup.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <assert.h>
|
|
|
|
|
|
|
|
using namespace KJS;
|
|
|
|
|
|
|
|
// ------------------------------ ObjectPrototypeImp --------------------------------
|
|
|
|
|
|
|
|
ObjectPrototypeImp::ObjectPrototypeImp(ExecState *exec,
|
|
|
|
FunctionPrototypeImp *funcProto)
|
|
|
|
: ObjectImp() // [[Prototype]] is Null()
|
|
|
|
{
|
|
|
|
Value protect(this);
|
|
|
|
putDirect(toStringPropertyName, new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::ToString,
|
|
|
|
0, toStringPropertyName), DontEnum);
|
|
|
|
putDirect(toLocaleStringPropertyName, new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::ToLocaleString,
|
|
|
|
0, toLocaleStringPropertyName), DontEnum);
|
|
|
|
putDirect(valueOfPropertyName, new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::ValueOf,
|
|
|
|
0, valueOfPropertyName), DontEnum);
|
|
|
|
putDirect("hasOwnProperty", new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::HasOwnProperty,
|
|
|
|
1,"hasOwnProperty"),DontEnum);
|
|
|
|
putDirect("isPrototypeOf", new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::IsPrototypeOf,
|
|
|
|
1,"isPrototypeOf"),DontEnum);
|
|
|
|
putDirect("propertyIsEnumerable", new ObjectProtoFuncImp(exec,funcProto,ObjectProtoFuncImp::PropertyIsEnumerable,
|
|
|
|
1,"propertyIsEnumerable"),DontEnum);
|
|
|
|
|
|
|
|
#ifndef KJS_PURE_ECMA // standard compliance location is the Global object
|
|
|
|
// see http://www.devguru.com/Technologies/ecmascript/quickref/object.html
|
|
|
|
put(exec, "eval",
|
|
|
|
Object(new GlobalFuncImp(exec, funcProto,GlobalFuncImp::Eval, 1, "eval")),
|
|
|
|
DontEnum);
|
|
|
|
#endif
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------ ObjectProtoFuncImp --------------------------------
|
|
|
|
|
|
|
|
ObjectProtoFuncImp::ObjectProtoFuncImp(ExecState * /*exec*/,
|
|
|
|
FunctionPrototypeImp *funcProto,
|
|
|
|
int i, int len, const Identifier &_ident)
|
|
|
|
: InternalFunctionImp(funcProto), id(i)
|
|
|
|
{
|
|
|
|
Value protect(this);
|
|
|
|
putDirect(lengthPropertyName, len, DontDelete|ReadOnly|DontEnum);
|
|
|
|
ident = _ident;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ObjectProtoFuncImp::implementsCall() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ECMA 15.2.4.2 + 15.2.4.3
|
|
|
|
|
|
|
|
Value ObjectProtoFuncImp::call(ExecState *exec, Object &thisObj, const List &args)
|
|
|
|
{
|
|
|
|
switch (id) {
|
|
|
|
case ToString:
|
|
|
|
// fall through
|
|
|
|
case ToLocaleString:
|
|
|
|
return String("[object "+thisObj.className()+"]");
|
|
|
|
case ValueOf:
|
|
|
|
return thisObj;
|
|
|
|
case HasOwnProperty: {
|
|
|
|
// Same as hasProperty() but without checking the prototype
|
|
|
|
Identifier propertyName(args[0].toString(exec));
|
|
|
|
Value tempProto(thisObj.imp()->prototype());
|
|
|
|
thisObj.imp()->setPrototype(Value());
|
|
|
|
bool exists = thisObj.hasProperty(exec,propertyName);
|
|
|
|
thisObj.imp()->setPrototype(tempProto);
|
|
|
|
return Value(exists ? BooleanImp::staticTrue : BooleanImp::staticFalse);
|
|
|
|
}
|
|
|
|
case IsPrototypeOf: {
|
|
|
|
Value v = args[0];
|
|
|
|
for (; v.isValid() && v.isA(ObjectType); v = Object::dynamicCast(v).prototype()) {
|
|
|
|
if (v.imp() == thisObj.imp())
|
|
|
|
return Value(BooleanImp::staticTrue);
|
|
|
|
}
|
|
|
|
return Value(BooleanImp::staticFalse);
|
|
|
|
}
|
|
|
|
case PropertyIsEnumerable: {
|
|
|
|
Identifier propertyName(args[0].toString(exec));
|
|
|
|
ObjectImp *obj = static_cast<ObjectImp*>(thisObj.imp());
|
|
|
|
|
|
|
|
int attributes;
|
|
|
|
ValueImp *v = obj->_prop.get(propertyName,attributes);
|
|
|
|
if (v)
|
|
|
|
return Value((attributes & DontEnum) ?
|
|
|
|
BooleanImp::staticFalse : BooleanImp::staticTrue);
|
|
|
|
|
|
|
|
if (propertyName == specialPrototypePropertyName)
|
|
|
|
return Value(BooleanImp::staticFalse);
|
|
|
|
|
|
|
|
const HashEntry *entry = obj->findPropertyHashEntry(propertyName);
|
|
|
|
return Value((entry && !(entry->attr & DontEnum)) ?
|
|
|
|
BooleanImp::staticTrue : BooleanImp::staticFalse);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return Undefined();
|
|
|
|
}
|
|
|
|
|
|
|
|
// ------------------------------ ObjectObjectImp --------------------------------
|
|
|
|
|
|
|
|
ObjectObjectImp::ObjectObjectImp(ExecState * /*exec*/,
|
|
|
|
ObjectPrototypeImp *objProto,
|
|
|
|
FunctionPrototypeImp *funcProto)
|
|
|
|
: InternalFunctionImp(funcProto)
|
|
|
|
{
|
|
|
|
Value protect(this);
|
|
|
|
// ECMA 15.2.3.1
|
|
|
|
putDirect(prototypePropertyName, objProto, DontEnum|DontDelete|ReadOnly);
|
|
|
|
|
|
|
|
// no. of arguments for constructor
|
|
|
|
putDirect(lengthPropertyName, NumberImp::one(), ReadOnly|DontDelete|DontEnum);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool ObjectObjectImp::implementsConstruct() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ECMA 15.2.2
|
|
|
|
Object ObjectObjectImp::construct(ExecState *exec, const List &args)
|
|
|
|
{
|
|
|
|
// if no arguments have been passed ...
|
|
|
|
if (args.isEmpty()) {
|
|
|
|
Object proto = exec->lexicalInterpreter()->builtinObjectPrototype();
|
|
|
|
Object result(new ObjectImp(proto));
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
Value arg = *(args.begin());
|
|
|
|
Object obj = Object::dynamicCast(arg);
|
|
|
|
if (obj.isValid())
|
|
|
|
return obj;
|
|
|
|
|
|
|
|
switch (arg.type()) {
|
|
|
|
case StringType:
|
|
|
|
case BooleanType:
|
|
|
|
case NumberType:
|
|
|
|
return arg.toObject(exec);
|
|
|
|
default:
|
|
|
|
assert(!"unhandled switch case in ObjectConstructor");
|
|
|
|
case NullType:
|
|
|
|
case UndefinedType:
|
|
|
|
Object proto = exec->lexicalInterpreter()->builtinObjectPrototype();
|
|
|
|
return Object(new ObjectImp(proto));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
bool ObjectObjectImp::implementsCall() const
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
Value ObjectObjectImp::call(ExecState *exec, Object &/*thisObj*/, const List &args)
|
|
|
|
{
|
|
|
|
Value result;
|
|
|
|
|
|
|
|
List argList;
|
|
|
|
// Construct a new Object
|
|
|
|
if (args.isEmpty()) {
|
|
|
|
result = construct(exec,argList);
|
|
|
|
} else {
|
|
|
|
Value arg = args[0];
|
|
|
|
if (arg.type() == NullType || arg.type() == UndefinedType) {
|
|
|
|
argList.append(arg);
|
|
|
|
result = construct(exec,argList);
|
|
|
|
} else
|
|
|
|
result = arg.toObject(exec);
|
|
|
|
}
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|