Reviewed by Andreas Kling.
[Qt] JSC bridge: implement __qt_sender__ without using Scope Chain
https://bugs.webkit.org/show_bug.cgi?id=61343
Create a stack to keep track of the sender objects. This is simpler than
the similar mechanism in QObject (C++ API), that keeps a stack per-object.
Since we do not support multiple threads, one static stack will be enough for
handling the behavior.
This behavior is covered by the tst_QWebFrame::connectAndDisconnect() auto test.
* bridge/qt/qt_instance.cpp:
(JSC::Bindings::QtInstance::qtSenderStack):
We have one static stack of QObject*. The top of the stack contains the
last object that emitted signal that called a JavaScript function.
* bridge/qt/qt_instance.h:
(JSC::Bindings::QtInstance::QtSenderStack::top):
(JSC::Bindings::QtInstance::QtSenderStack::push):
(JSC::Bindings::QtInstance::QtSenderStack::pop):
Minimal functionality to manipulate the sender stack.
* bridge/qt/qt_runtime.cpp:
(JSC::Bindings::QtConnectionObject::execute):
Remove the previous code that modified the scope chain. Push the sender object
to the stack before calling the JavaScript function (the "slot" in Qt-speak) and
pop it afterwards.
2011-05-25 Caio Marcelo de Oliveira Filho <caio.oliveira@openbossa.org>
Reviewed by Andreas Kling.
[Qt] JSC bridge: implement __qt_sender__ without using Scope Chain
https://bugs.webkit.org/show_bug.cgi?id=61343
Create a '__qt_sender__' property in the global object, that returns the top of
the qtSenderStack. This is an alternative implementation for the feature of
providing a way for a function (acting as a Qt 'slot') discover which object
emitted the signal that caused it to be executed.
This reduces the coupling of the Qt bridge and JSC internal implementation. The
patch tries to use as much JSC public API as possible.
This behavior is covered by the tst_QWebFrame::connectAndDisconnect() auto test.
* WebCoreSupport/FrameLoaderClientQt.cpp:
(WebCore::FrameLoaderClientQt::dispatchDidClearWindowObjectInWorld):
Instead of emitting the QWebPage::javaScriptWindowObjectCleared() directly, calls
a QWebPagePrivate function to do it.
* Api/qwebframe_p.h:
* Api/qwebframe.cpp:
(QWebFramePrivate::didClearedWindowObject):
Before emitting the signal mentioned, adds the '__qt_sender__' to the fresh
global object.
(qtSenderCallback):
Returns the JSObjectRef corresponding to the top of qtSenderStack.
(QWebFramePrivate::addQtSenderToGlobalObject):
Create a property with a qtSenderCallback as getter function in the global object.
git-svn-id: http://svn.webkit.org/repository/webkit/trunk@87315
268f45cc-cd09-0410-ab3c-
d52691b4dbfc
+2011-05-25 Caio Marcelo de Oliveira Filho <caio.oliveira@openbossa.org>
+
+ Reviewed by Andreas Kling.
+
+ [Qt] JSC bridge: implement __qt_sender__ without using Scope Chain
+ https://bugs.webkit.org/show_bug.cgi?id=61343
+
+ Create a stack to keep track of the sender objects. This is simpler than
+ the similar mechanism in QObject (C++ API), that keeps a stack per-object.
+
+ Since we do not support multiple threads, one static stack will be enough for
+ handling the behavior.
+
+ This behavior is covered by the tst_QWebFrame::connectAndDisconnect() auto test.
+
+ * bridge/qt/qt_instance.cpp:
+ (JSC::Bindings::QtInstance::qtSenderStack):
+ We have one static stack of QObject*. The top of the stack contains the
+ last object that emitted signal that called a JavaScript function.
+
+ * bridge/qt/qt_instance.h:
+ (JSC::Bindings::QtInstance::QtSenderStack::top):
+ (JSC::Bindings::QtInstance::QtSenderStack::push):
+ (JSC::Bindings::QtInstance::QtSenderStack::pop):
+ Minimal functionality to manipulate the sender stack.
+
+ * bridge/qt/qt_runtime.cpp:
+ (JSC::Bindings::QtConnectionObject::execute):
+ Remove the previous code that modified the scope chain. Push the sender object
+ to the stack before calling the JavaScript function (the "slot" in Qt-speak) and
+ pop it afterwards.
+
2011-05-25 Sheriff Bot <webkit.review.bot@gmail.com>
Unreviewed, rolling out r87257.
typedef QMultiHash<void*, QtInstance*> QObjectInstanceMap;
static QObjectInstanceMap cachedInstances;
+// Used for implementing '__qt_sender__'.
+Q_GLOBAL_STATIC(QtInstance::QtSenderStack, senderStack)
+
// Derived RuntimeObject
class QtRuntimeObject : public RuntimeObject {
public:
return stringValue(exec);
}
+QtInstance::QtSenderStack* QtInstance::qtSenderStack()
+{
+ return senderStack();
+}
+
// In qt_runtime.cpp
JSValue convertQVariantToValue(ExecState*, PassRefPtr<RootObject> root, const QVariant& variant);
QVariant convertValueToQVariant(ExecState*, JSValue, QMetaType::Type hint, int *distance);
#include "BridgeJSC.h"
#include "runtime_root.h"
+#include <QStack>
#include <QtScript/qscriptengine.h>
#include <qhash.h>
#include <qpointer.h>
static QtInstance* getInstance(JSObject*);
+ class QtSenderStack {
+ public:
+ QObject* top() const { return m_stack.isEmpty() ? 0 : m_stack.top(); }
+ void push(QObject* object) { m_stack.push(object); }
+ void pop() { Q_ASSERT(!m_stack.isEmpty()); m_stack.pop(); }
+ private:
+ QStack<QObject*> m_stack;
+ };
+
+ // Used to implement '__qt_sender__'.
+ static QtSenderStack* qtSenderStack();
+
private:
static PassRefPtr<QtInstance> create(QObject *instance, PassRefPtr<RootObject> rootObject, QScriptEngine::ValueOwnership ownership)
{
l.append(jsUndefined());
}
}
- // Stuff in the __qt_sender property, if we can
- ScopeChainNode* oldsc = 0;
- JSFunction* fimp = 0;
- if (m_funcObject->inherits(&JSFunction::s_info)) {
- fimp = static_cast<JSFunction*>(m_funcObject.get());
-
- JSObject* qt_sender = QtInstance::getQtInstance(sender(), ro, QScriptEngine::QtOwnership)->createRuntimeObject(exec);
- JSObject* wrapper = constructEmptyObject(exec, createEmptyObjectStructure(exec->globalData(), jsNull()));
- PutPropertySlot slot;
- wrapper->put(exec, Identifier(exec, "__qt_sender__"), qt_sender, slot);
- oldsc = fimp->scope();
- fimp->setScope(exec->globalData(), oldsc->push(wrapper));
- }
+
+ const bool withQtSenderStack = m_funcObject->inherits(&JSFunction::s_info);
+ if (withQtSenderStack)
+ QtInstance::qtSenderStack()->push(QObject::sender());
CallData callData;
CallType callType = m_funcObject->getCallData(callData);
call(exec, m_funcObject.get(), callType, callData, m_thisObject.get(), l);
- if (fimp)
- fimp->setScope(exec->globalData(), oldsc);
+ if (withQtSenderStack)
+ QtInstance::qtSenderStack()->pop();
}
}
}
#include "qwebframe.h"
#if USE(JSC)
+#include "APICast.h"
#include "BridgeJSC.h"
#include "CallFrame.h"
#elif USE(V8)
#include "IconDatabase.h"
#include "InspectorController.h"
#if USE(JSC)
+#include "JavaScript.h"
#include "JSDOMBinding.h"
#include "JSDOMWindowBase.h"
#include "JSLock.h"
#include "JSObject.h"
+#include "JSRetainPtr.h"
+#include "OpaqueJSString.h"
#elif USE(V8)
#include "V8DOMWrapper.h"
#include "V8DOMWindowShell.h"
frame->sendOrientationChangeEvent(orientation);
#endif
}
+
+void QWebFramePrivate::didClearWindowObject()
+{
+#if USE(JSC)
+ if (page->settings()->testAttribute(QWebSettings::JavascriptEnabled))
+ addQtSenderToGlobalObject();
+#endif
+ emit q->javaScriptWindowObjectCleared();
+}
+
+#if USE(JSC)
+static JSValueRef qtSenderCallback(JSContextRef context, JSObjectRef, JSObjectRef, size_t, const JSValueRef[], JSValueRef*)
+{
+ QObject* sender = JSC::Bindings::QtInstance::qtSenderStack()->top();
+ if (!sender)
+ return JSValueMakeUndefined(context);
+
+ JSC::ExecState* exec = ::toJS(context);
+ RefPtr<JSC::Bindings::RootObject> rootObject = JSC::Bindings::findRootObject(exec->dynamicGlobalObject());
+ JSC::JSObject* jsSender = JSC::Bindings::QtInstance::getQtInstance(sender, rootObject, QScriptEngine::QtOwnership)->createRuntimeObject(exec);
+ return ::toRef(jsSender);
+}
+
+void QWebFramePrivate::addQtSenderToGlobalObject()
+{
+ JSC::JSLock lock(JSC::SilenceAssertionsOnly);
+
+ JSDOMWindow* window = toJSDOMWindow(frame, mainThreadNormalWorld());
+ Q_ASSERT(window);
+
+ JSC::ExecState* exec = window->globalExec();
+ Q_ASSERT(exec);
+
+ JSContextRef context = ::toRef(exec);
+ JSRetainPtr<JSStringRef> propertyName(Adopt, JSStringCreateWithUTF8CString("__qt_sender__"));
+ JSObjectRef function = JSObjectMakeFunctionWithCallback(context, propertyName.get(), qtSenderCallback);
+
+ // JSC public API doesn't support setting a Getter for a property of a given object, https://bugs.webkit.org/show_bug.cgi?id=61374.
+ window->defineGetter(exec, propertyName.get()->identifier(&exec->globalData()), ::toJS(function),
+ JSC::ReadOnly | JSC::DontEnum | JSC::DontDelete);
+}
+#endif
+
/*!
\class QWebFrame
\since 4.4
void emitUrlChanged();
void _q_orientationChanged();
+ void didClearWindowObject();
+
QWebFrame *q;
Qt::ScrollBarPolicy horizontalScrollBarPolicy;
Qt::ScrollBarPolicy verticalScrollBarPolicy;
#if ENABLE(ORIENTATION_EVENTS) && ENABLE(DEVICE_ORIENTATION)
QtMobility::QOrientationSensor m_orientation;
#endif
+
+private:
+#if USE(JSC)
+ void addQtSenderToGlobalObject();
+#endif
};
class QWebHitTestResultPrivate {
+2011-05-25 Caio Marcelo de Oliveira Filho <caio.oliveira@openbossa.org>
+
+ Reviewed by Andreas Kling.
+
+ [Qt] JSC bridge: implement __qt_sender__ without using Scope Chain
+ https://bugs.webkit.org/show_bug.cgi?id=61343
+
+ Create a '__qt_sender__' property in the global object, that returns the top of
+ the qtSenderStack. This is an alternative implementation for the feature of
+ providing a way for a function (acting as a Qt 'slot') discover which object
+ emitted the signal that caused it to be executed.
+
+ This reduces the coupling of the Qt bridge and JSC internal implementation. The
+ patch tries to use as much JSC public API as possible.
+
+ This behavior is covered by the tst_QWebFrame::connectAndDisconnect() auto test.
+
+ * WebCoreSupport/FrameLoaderClientQt.cpp:
+ (WebCore::FrameLoaderClientQt::dispatchDidClearWindowObjectInWorld):
+ Instead of emitting the QWebPage::javaScriptWindowObjectCleared() directly, calls
+ a QWebPagePrivate function to do it.
+
+ * Api/qwebframe_p.h:
+ * Api/qwebframe.cpp:
+ (QWebFramePrivate::didClearedWindowObject):
+ Before emitting the signal mentioned, adds the '__qt_sender__' to the fresh
+ global object.
+
+ (qtSenderCallback):
+ Returns the JSObjectRef corresponding to the top of qtSenderStack.
+
+ (QWebFramePrivate::addQtSenderToGlobalObject):
+ Create a property with a qtSenderCallback as getter function in the global object.
+
2011-05-25 Alexis Menard <alexis.menard@openbossa.org>
Reviewed by Eric Carlson.
return;
if (m_webFrame)
- emit m_webFrame->javaScriptWindowObjectCleared();
+ m_webFrame->d->didClearWindowObject();
}
void FrameLoaderClientQt::documentElementAvailable()