| // Copyright 2005 Google Inc. |
| // All Rights Reserved. |
| |
| #include "FunctionObject.h" |
| #include <kjs/array_object.h> |
| |
| using namespace KJS; |
| |
| const ClassInfo FunctionObject::info = {"Function", 0, 0, 0}; |
| |
| class CallFunction : public FunctionObject { |
| public: |
| CallFunction(): FunctionObject("call") { |
| } |
| |
| virtual JSValue *callAsFunction(ExecState *exec, JSObject *thisObj, const List &args) { |
| // Copied from FunctionProtoFunc::callAsFunction() |
| JSValue *thisArg = args[0]; |
| JSObject *func = thisObj; |
| |
| if (!func->implementsCall()) |
| return throwError(exec, TypeError); |
| |
| JSObject *callThis; |
| if (thisArg->isUndefinedOrNull()) |
| callThis = exec->dynamicInterpreter()->globalObject(); |
| else |
| callThis = thisArg->toObject(exec); |
| |
| return func->call(exec, callThis, args.copyTail()); |
| } |
| }; |
| |
| class ApplyFunction : public FunctionObject { |
| public: |
| ApplyFunction(): FunctionObject("apply") { |
| } |
| |
| virtual JSValue *callAsFunction(ExecState *exec, JSObject *thisObj, const List &args) { |
| // Copied from FunctionProtoFunc::callAsFunction() |
| JSObject *func = thisObj; |
| if (!func->implementsCall()) |
| return throwError(exec, TypeError); |
| |
| JSValue *thisArg = args[0]; |
| JSObject *applyThis; |
| if (thisArg->isUndefinedOrNull()) |
| applyThis = exec->dynamicInterpreter()->globalObject(); |
| else |
| applyThis = thisArg->toObject(exec); |
| |
| JSValue *argArray = args[1]; |
| List applyArgs; |
| if (!argArray->isUndefinedOrNull()) { |
| if (!argArray->isObject(&ArrayInstance::info)) |
| return throwError(exec, TypeError); |
| |
| JSObject *argArrayObj = static_cast<JSObject *>(argArray); |
| unsigned int length = argArrayObj->get(exec, lengthPropertyName)->toUInt32(exec); |
| for (unsigned int i = 0; i < length; ++i) { |
| applyArgs.append(argArrayObj->get(exec,i)); |
| } |
| } |
| return func->call(exec, applyThis, applyArgs); |
| } |
| }; |
| |
| |
| static UString makeFunctionString(const UString& name) { |
| return "\nfunction " + name + "() {\n [native code]\n}\n"; |
| } |
| |
| JSValue *FunctionObject::getter(ExecState* exec, JSObject* obj, const Identifier& propertyName, const PropertySlot& slot) { |
| if (propertyName.ustring() == "toString") { |
| return new ToStringFunction(); |
| } else if (propertyName.ustring() == "call") { |
| return new CallFunction(); |
| } else if (propertyName.ustring() == "apply") { |
| return new ApplyFunction(); |
| } |
| return jsUndefined(); |
| } |
| |
| FunctionObject::FunctionObject(const UString& name): name(name) { |
| } |
| |
| bool FunctionObject::getOwnPropertySlot(ExecState *exec, const Identifier& propertyName, PropertySlot& slot) { |
| if (propertyName.ustring() == "toString") { |
| slot.setCustom(this, getter); |
| return true; |
| } |
| if (propertyName.ustring() == "call") { |
| slot.setCustom(this, getter); |
| return true; |
| } |
| if (propertyName.ustring() == "apply") { |
| slot.setCustom(this, getter); |
| return true; |
| } |
| return false; |
| } |
| |
| bool FunctionObject::canPut(ExecState *exec, const Identifier &propertyName) const { |
| return false; |
| } |
| |
| void FunctionObject::put(ExecState *exec, const Identifier &propertyName, JSValue *value, int attr) { |
| } |
| |
| bool FunctionObject::deleteProperty(ExecState *exec, const Identifier &propertyName) { |
| return false; |
| } |
| |
| JSValue *FunctionObject::defaultValue(ExecState *exec, JSType hint) const { |
| return jsString(makeFunctionString(name)); |
| } |
| |
| bool FunctionObject::implementsCall() const { |
| return true; |
| } |
| |
| // ToStringFunction |
| |
| ToStringFunction::ToStringFunction(): FunctionObject("toString") { |
| } |
| |
| JSValue *ToStringFunction::callAsFunction(ExecState *exec, JSObject *thisObj, const List &args) { |
| if (!thisObj) { |
| return throwError(exec, TypeError); |
| } |
| return jsString(thisObj->toString(exec)); |
| } |