/* * Copyright (C) 2011 Apple Inc. All rights reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in the * documentation and/or other materials provided with the distribution. * * THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS'' * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, * THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR * PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS * BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #ifndef JSDictionary_h #define JSDictionary_h #include "MessagePort.h" #include #include #include #include #include #include namespace Deprecated { class ScriptValue; } namespace WebCore { class ArrayValue; class CSSFontFaceRule; class Dictionary; class DOMError; class DOMWindow; class EventTarget; class Gamepad; class FetchHeaders; class MediaKeyError; class MediaStream; class MediaStreamTrack; class RTCRtpReceiver; class Node; class SerializedScriptValue; class Storage; class TrackBase; class VoidCallback; class JSDictionary { public: JSDictionary(JSC::ExecState* exec, JSC::JSObject* initializerObject) : m_exec(exec) { if (exec && initializerObject) m_initializerObject = JSC::Strong(exec->vm(), initializerObject); } // Returns false if any exceptions were thrown, regardless of whether the property was found. template bool tryGetProperty(const char* propertyName, Result&) const; template bool tryGetProperty(const char* propertyName, T* context, void (*setter)(T* context, const Result&)) const; // Returns true if the property was found in the dictionary, and the value could be converted to the desired type. template bool get(const char* propertyName, Result&) const; bool getWithUndefinedOrNullCheck(const char* propertyName, String& value) const; JSC::ExecState* execState() const { return m_exec; } JSC::JSObject* initializerObject() const { return m_initializerObject.get(); } bool isValid() const { return m_exec && m_initializerObject; } private: template struct IdentitySetter { static void identitySetter(Result* context, const Result& result) { *context = result; } }; enum GetPropertyResult { ExceptionThrown, NoPropertyFound, PropertyFound }; template GetPropertyResult tryGetPropertyAndResult(const char* propertyName, T* context, void (*setter)(T* context, const Result&)) const; GetPropertyResult tryGetProperty(const char* propertyName, JSC::JSValue&) const; static void convertValue(JSC::ExecState*, JSC::JSValue, bool& result); static void convertValue(JSC::ExecState*, JSC::JSValue, int& result); static void convertValue(JSC::ExecState*, JSC::JSValue, unsigned& result); static void convertValue(JSC::ExecState*, JSC::JSValue, unsigned short& result); static void convertValue(JSC::ExecState*, JSC::JSValue, unsigned long& result); static void convertValue(JSC::ExecState*, JSC::JSValue, unsigned long long& result); static void convertValue(JSC::ExecState*, JSC::JSValue, double& result); static void convertValue(JSC::ExecState*, JSC::JSValue, Dictionary& result); static void convertValue(JSC::ExecState*, JSC::JSValue, String& result); static void convertValue(JSC::ExecState*, JSC::JSValue, Deprecated::ScriptValue& result); static void convertValue(JSC::ExecState*, JSC::JSValue, Vector& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, MessagePortArray& result); #if ENABLE(VIDEO_TRACK) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #endif static void convertValue(JSC::ExecState*, JSC::JSValue, HashSet& result); static void convertValue(JSC::ExecState*, JSC::JSValue, ArrayValue& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #if ENABLE(ENCRYPTED_MEDIA) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #endif #if ENABLE(FETCH_API) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #endif #if ENABLE(MEDIA_STREAM) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #endif #if ENABLE(FONT_LOAD_EVENTS) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr& result); #endif #if ENABLE(GAMEPAD) static void convertValue(JSC::ExecState*, JSC::JSValue, RefPtr&); #endif static void convertValue(JSC::ExecState*, JSC::JSValue, JSC::JSFunction*&); JSC::ExecState* m_exec; JSC::Strong m_initializerObject; }; template bool JSDictionary::tryGetProperty(const char* propertyName, T* context, void (*setter)(T* context, const Result&)) const { return tryGetPropertyAndResult(propertyName, context, setter) != ExceptionThrown; } template bool JSDictionary::tryGetProperty(const char* propertyName, Result& finalResult) const { return tryGetPropertyAndResult(propertyName, &finalResult, IdentitySetter::identitySetter) != ExceptionThrown; } template bool JSDictionary::get(const char* propertyName, Result& finalResult) const { return tryGetPropertyAndResult(propertyName, &finalResult, IdentitySetter::identitySetter) == PropertyFound; } template <> inline bool JSDictionary::get(const char* propertyName, JSC::JSValue& finalResult) const { return tryGetProperty(propertyName, finalResult) == PropertyFound; } template JSDictionary::GetPropertyResult JSDictionary::tryGetPropertyAndResult(const char* propertyName, T* context, void (*setter)(T* context, const Result&)) const { JSC::JSValue value; GetPropertyResult getPropertyResult = tryGetProperty(propertyName, value); switch (getPropertyResult) { case ExceptionThrown: return getPropertyResult; case PropertyFound: { Result result; convertValue(m_exec, value, result); if (m_exec->hadException()) return ExceptionThrown; setter(context, result); break; } case NoPropertyFound: break; } return getPropertyResult; } } // namespace WebCore #endif // JSDictionary_h