#include "kjs_window.h"
#include "kjs_events.h"
#include "kjs_events.lut.h"
#include "kjs_views.h"
#include "kjs_proxy.h"
#include "xml/dom_nodeimpl.h"
#include "xml/dom_docimpl.h"
#include "xml/dom2_eventsimpl.h"
#include "rendering/render_object.h"
#include <kdebug.h>
using namespace KJS;
JSEventListener::JSEventListener(Object _listener, const Object &_win, bool _html)
{
listener = _listener;
html = _html;
win = _win;
static_cast<Window*>(win.imp())->jsEventListeners.append(this);
}
JSEventListener::~JSEventListener()
{
static_cast<Window*>(win.imp())->jsEventListeners.removeRef(this);
}
void JSEventListener::handleEvent(DOM::Event &evt, bool isWindowEvent)
{
#ifdef KJS_DEBUGGER
if (KJSDebugWin::instance() && KJSDebugWin::instance()->inSession())
return;
#endif
KHTMLPart *part = static_cast<Window*>(win.imp())->part();
KJSProxy *proxy = 0L;
if (part)
proxy = KJSProxy::proxy( part );
if (proxy && listener.implementsCall()) {
ref();
KJS::ScriptInterpreter *interpreter = static_cast<KJS::ScriptInterpreter *>(proxy->interpreter());
ExecState *exec = interpreter->globalExec();
List args;
args.append(getDOMEvent(exec,evt));
ScopeChain oldScope = listener.scope();
Object thisObj;
if (isWindowEvent) {
thisObj = win;
} else {
thisObj = Object::dynamicCast(getDOMNode(exec,evt.currentTarget()));
if ( !thisObj.isNull() ) {
ScopeChain scope = oldScope;
static_cast<DOMNode*>(thisObj.imp())->pushEventHandlerScope(exec, scope);
listener.setScope( scope );
}
}
Window *window = static_cast<Window*>(win.imp());
window->setCurrentEvent( &evt );
interpreter->setCurrentEvent( &evt );
Value retval = listener.call(exec, thisObj, args);
listener.setScope( oldScope );
window->setCurrentEvent( 0 );
interpreter->setCurrentEvent( 0 );
#if APPLE_CHANGES
if ( exec->hadException() ) {
if (Interpreter::shouldPrintExceptions()) {
char *message = exec->exception().toObject(exec).get(exec, messagePropertyName).toString(exec).ascii();
printf("(event handler):%s\n", message);
}
exec->clearException();
}
#else
if ( exec->hadException() )
exec->clearException();
#endif
else if (html)
{
QVariant ret = ValueToVariant(exec, retval);
if (ret.type() == QVariant::Bool && ret.toBool() == false)
evt.preventDefault();
}
DOM::DocumentImpl::updateDocumentsRendering();
deref();
}
}
DOM::DOMString JSEventListener::eventListenerType()
{
if (html)
return "_khtml_HTMLEventListener";
else
return "_khtml_JSEventListener";
}
Value KJS::getNodeEventListener(DOM::Node n, int eventId)
{
DOM::EventListener *listener = n.handle()->getHTMLEventListener(eventId);
if (listener)
return static_cast<JSEventListener*>(listener)->listenerObj();
else
return Null();
}
const ClassInfo EventConstructor::info = { "EventConstructor", 0, &EventConstructorTable, 0 };
Value EventConstructor::tryGet(ExecState *exec, const Identifier &p) const
{
return DOMObjectLookupGetValue<EventConstructor, DOMObject>(exec,p,&EventConstructorTable,this);
}
Value EventConstructor::getValueProperty(ExecState *, int token) const
{
return Number(token);
}
Value KJS::getEventConstructor(ExecState *exec)
{
return cacheGlobalObject<EventConstructor>(exec, "[[event.constructor]]");
}
const ClassInfo DOMEvent::info = { "Event", 0, &DOMEventTable, 0 };
DEFINE_PROTOTYPE("DOMEvent", DOMEventProto)
IMPLEMENT_PROTOFUNC(DOMEventProtoFunc)
IMPLEMENT_PROTOTYPE(DOMEventProto, DOMEventProtoFunc)
DOMEvent::DOMEvent(ExecState *exec, DOM::Event e)
: DOMObject(DOMEventProto::self(exec)), event(e) { }
DOMEvent::~DOMEvent()
{
ScriptInterpreter::forgetDOMObject(event.handle());
}
Value DOMEvent::tryGet(ExecState *exec, const Identifier &p) const
{
#ifdef KJS_VERBOSE
kdDebug() << "KJS::DOMEvent::tryGet " << p.qstring() << endl;
#endif
return DOMObjectLookupGetValue<DOMEvent,DOMObject>(exec, p, &DOMEventTable, this );
}
Value DOMEvent::getValueProperty(ExecState *exec, int token) const
{
switch (token) {
case Type:
return String(event.type());
case Target:
case SrcElement:
return getDOMNode(exec,event.target());
case CurrentTarget:
return getDOMNode(exec,event.currentTarget());
case EventPhase:
return Number((unsigned int)event.eventPhase());
case Bubbles:
case CancelBubble: return Boolean(event.bubbles());
case Cancelable:
return Boolean(event.cancelable());
case TimeStamp:
return Number((long unsigned int)event.timeStamp()); default:
kdWarning() << "Unhandled token in DOMEvent::getValueProperty : " << token << endl;
return Value();
}
}
void DOMEvent::tryPut(ExecState *exec, const Identifier &propertyName,
const Value& value, int attr)
{
DOMObjectLookupPut<DOMEvent, DOMObject>(exec, propertyName, value, attr,
&DOMEventTable, this);
}
void DOMEvent::putValue(ExecState *exec, int token, const Value& value, int)
{
switch (token) {
case ReturnValue:
if (value.toBoolean(exec))
event.preventDefault();
break;
case CancelBubble:
if (value.toBoolean(exec))
event.stopPropagation();
break;
default:
break;
}
}
Value DOMEventProtoFunc::tryCall(ExecState *exec, Object & thisObj, const List &args)
{
if (!thisObj.inherits(&KJS::DOMEvent::info)) {
Object err = Error::create(exec,TypeError);
exec->setException(err);
return err;
}
DOM::Event event = static_cast<DOMEvent *>( thisObj.imp() )->toEvent();
switch (id) {
case DOMEvent::StopPropagation:
event.stopPropagation();
case DOMEvent::PreventDefault:
event.preventDefault();
return Undefined();
case DOMEvent::InitEvent:
event.initEvent(args[0].toString(exec).string(),args[1].toBoolean(exec),args[2].toBoolean(exec));
return Undefined();
};
return Undefined();
}
Value KJS::getDOMEvent(ExecState *exec, DOM::Event e)
{
DOMObject *ret;
if (e.isNull())
return Null();
ScriptInterpreter* interp = static_cast<ScriptInterpreter *>(exec->interpreter());
if ((ret = interp->getDOMObject(e.handle())))
return Value(ret);
DOM::DOMString module = e.eventModuleName();
if (module == "UIEvents")
ret = new DOMUIEvent(exec, static_cast<DOM::UIEvent>(e));
else if (module == "MouseEvents")
ret = new DOMMouseEvent(exec, static_cast<DOM::MouseEvent>(e));
else if (module == "MutationEvents")
ret = new DOMMutationEvent(exec, static_cast<DOM::MutationEvent>(e));
else
ret = new DOMEvent(exec, e);
interp->putDOMObject(e.handle(),ret);
return Value(ret);
}
DOM::Event KJS::toEvent(const Value& val)
{
Object obj = Object::dynamicCast(val);
if (obj.isNull() || !obj.inherits(&DOMEvent::info))
return DOM::Event();
const DOMEvent *dobj = static_cast<const DOMEvent*>(obj.imp());
return dobj->toEvent();
}
const ClassInfo EventExceptionConstructor::info = { "EventExceptionConstructor", 0, &EventExceptionConstructorTable, 0 };
Value EventExceptionConstructor::tryGet(ExecState *exec, const Identifier &p) const
{
return DOMObjectLookupGetValue<EventExceptionConstructor, DOMObject>(exec,p,&EventExceptionConstructorTable,this);
}
Value EventExceptionConstructor::getValueProperty(ExecState *, int token) const
{
return Number(token);
}
Value KJS::getEventExceptionConstructor(ExecState *exec)
{
return cacheGlobalObject<EventExceptionConstructor>(exec, "[[eventException.constructor]]");
}
const ClassInfo DOMUIEvent::info = { "UIEvent", &DOMEvent::info, &DOMUIEventTable, 0 };
DEFINE_PROTOTYPE("DOMUIEvent",DOMUIEventProto)
IMPLEMENT_PROTOFUNC(DOMUIEventProtoFunc)
IMPLEMENT_PROTOTYPE_WITH_PARENT(DOMUIEventProto,DOMUIEventProtoFunc,DOMEventProto)
DOMUIEvent::~DOMUIEvent()
{
}
Value DOMUIEvent::tryGet(ExecState *exec, const Identifier &p) const
{
return DOMObjectLookupGetValue<DOMUIEvent,DOMEvent>(exec,p,&DOMUIEventTable,this);
}
Value DOMUIEvent::getValueProperty(ExecState *exec, int token) const
{
switch (token) {
case View:
return getDOMAbstractView(exec,static_cast<DOM::UIEvent>(event).view());
case Detail:
return Number(static_cast<DOM::UIEvent>(event).detail());
case KeyCode:
return Number(static_cast<DOM::UIEvent>(event).keyCode());
case LayerX:
return Number(static_cast<DOM::UIEvent>(event).layerX());
case LayerY:
return Number(static_cast<DOM::UIEvent>(event).layerY());
case PageX:
return Number(static_cast<DOM::UIEvent>(event).pageX());
case PageY:
return Number(static_cast<DOM::UIEvent>(event).pageY());
case Which:
return Number(static_cast<DOM::UIEvent>(event).which());
default:
kdWarning() << "Unhandled token in DOMUIEvent::getValueProperty : " << token << endl;
return Undefined();
}
}
Value DOMUIEventProtoFunc::tryCall(ExecState *exec, Object &thisObj, const List &args)
{
if (!thisObj.inherits(&KJS::DOMUIEvent::info)) {
Object err = Error::create(exec,TypeError);
exec->setException(err);
return err;
}
DOM::UIEvent uiEvent = static_cast<DOMUIEvent *>(thisObj.imp())->toUIEvent();
switch (id) {
case DOMUIEvent::InitUIEvent: {
DOM::AbstractView v = toAbstractView(args[3]);
static_cast<DOM::UIEvent>(uiEvent).initUIEvent(args[0].toString(exec).string(),
args[1].toBoolean(exec),
args[2].toBoolean(exec),
v,
args[4].toInteger(exec));
}
return Undefined();
}
return Undefined();
}
const ClassInfo DOMMouseEvent::info = { "MouseEvent", &DOMUIEvent::info, &DOMMouseEventTable, 0 };
DEFINE_PROTOTYPE("DOMMouseEvent",DOMMouseEventProto)
IMPLEMENT_PROTOFUNC(DOMMouseEventProtoFunc)
IMPLEMENT_PROTOTYPE_WITH_PARENT(DOMMouseEventProto,DOMMouseEventProtoFunc,DOMUIEventProto)
DOMMouseEvent::~DOMMouseEvent()
{
}
Value DOMMouseEvent::tryGet(ExecState *exec, const Identifier &p) const
{
#ifdef KJS_VERBOSE
kdDebug(6070) << "DOMMouseEvent::tryGet " << p.qstring() << endl;
#endif
return DOMObjectLookupGetValue<DOMMouseEvent,DOMUIEvent>(exec,p,&DOMMouseEventTable,this);
}
Value DOMMouseEvent::getValueProperty(ExecState *exec, int token) const
{
switch (token) {
case ScreenX:
return Number(static_cast<DOM::MouseEvent>(event).screenX());
case ScreenY:
return Number(static_cast<DOM::MouseEvent>(event).screenY());
case ClientX:
case X:
return Number(static_cast<DOM::MouseEvent>(event).clientX());
case ClientY:
case Y:
return Number(static_cast<DOM::MouseEvent>(event).clientY());
case OffsetX:
case OffsetY: {
DOM::Node node = event.target();
node.handle()->getDocument()->updateRendering();
khtml::RenderObject *rend = node.handle() ? node.handle()->renderer() : 0L;
int x = static_cast<DOM::MouseEvent>(event).clientX();
int y = static_cast<DOM::MouseEvent>(event).clientY();
if ( rend ) {
int xPos, yPos;
if ( rend->absolutePosition( xPos, yPos ) ) {
kdDebug() << "DOMMouseEvent::getValueProperty rend=" << rend << " xPos=" << xPos << " yPos=" << yPos << endl;
x -= xPos;
y -= yPos;
}
}
return Number( token == OffsetX ? x : y );
}
case CtrlKey:
return Boolean(static_cast<DOM::MouseEvent>(event).ctrlKey());
case ShiftKey:
return Boolean(static_cast<DOM::MouseEvent>(event).shiftKey());
case AltKey:
return Boolean(static_cast<DOM::MouseEvent>(event).altKey());
case MetaKey:
return Boolean(static_cast<DOM::MouseEvent>(event).metaKey());
case Button:
{
int domButton = static_cast<DOM::MouseEvent>(event).button();
int button = domButton==0 ? 1 : domButton==1 ? 4 : domButton==2 ? 2 : 0;
return Number( (unsigned int)button );
}
case ToElement:
if (event.handle()->id() == DOM::EventImpl::MOUSEOUT_EVENT)
return getDOMNode(exec,static_cast<DOM::MouseEvent>(event).relatedTarget());
return getDOMNode(exec,static_cast<DOM::MouseEvent>(event).target());
case FromElement:
if (event.handle()->id() == DOM::EventImpl::MOUSEOUT_EVENT)
return getDOMNode(exec,static_cast<DOM::MouseEvent>(event).target());
case RelatedTarget:
return getDOMNode(exec,static_cast<DOM::MouseEvent>(event).relatedTarget());
default:
kdWarning() << "Unhandled token in DOMMouseEvent::getValueProperty : " << token << endl;
return Value();
}
}
Value DOMMouseEventProtoFunc::tryCall(ExecState *exec, Object &thisObj, const List &args)
{
if (!thisObj.inherits(&KJS::DOMMouseEvent::info)) {
Object err = Error::create(exec,TypeError);
exec->setException(err);
return err;
}
DOM::MouseEvent mouseEvent = static_cast<DOMMouseEvent *>(thisObj.imp())->toMouseEvent();
switch (id) {
case DOMMouseEvent::InitMouseEvent:
mouseEvent.initMouseEvent(args[0].toString(exec).string(), args[1].toBoolean(exec), args[2].toBoolean(exec), toAbstractView(args[3]), args[4].toInteger(exec), args[5].toInteger(exec), args[6].toInteger(exec), args[7].toInteger(exec), args[8].toInteger(exec), args[9].toBoolean(exec), args[10].toBoolean(exec), args[11].toBoolean(exec), args[12].toBoolean(exec), args[13].toInteger(exec), toNode(args[14])); return Undefined();
}
return Undefined();
}
const ClassInfo MutationEventConstructor::info = { "MutationEventConstructor", 0, &MutationEventConstructorTable, 0 };
Value MutationEventConstructor::tryGet(ExecState *exec, const Identifier &p) const
{
return DOMObjectLookupGetValue<MutationEventConstructor,DOMObject>(exec,p,&MutationEventConstructorTable,this);
}
Value MutationEventConstructor::getValueProperty(ExecState *, int token) const
{
return Number(token);
}
Value KJS::getMutationEventConstructor(ExecState *exec)
{
return cacheGlobalObject<MutationEventConstructor>(exec, "[[mutationEvent.constructor]]");
}
const ClassInfo DOMMutationEvent::info = { "MutationEvent", &DOMEvent::info, &DOMMutationEventTable, 0 };
DEFINE_PROTOTYPE("DOMMutationEvent",DOMMutationEventProto)
IMPLEMENT_PROTOFUNC(DOMMutationEventProtoFunc)
IMPLEMENT_PROTOTYPE_WITH_PARENT(DOMMutationEventProto,DOMMutationEventProtoFunc,DOMEventProto)
DOMMutationEvent::~DOMMutationEvent()
{
}
Value DOMMutationEvent::tryGet(ExecState *exec, const Identifier &p) const
{
return DOMObjectLookupGetValue<DOMMutationEvent,DOMEvent>(exec,p,&DOMMutationEventTable,this);
}
Value DOMMutationEvent::getValueProperty(ExecState *exec, int token) const
{
switch (token) {
case RelatedNode:
return getDOMNode(exec,static_cast<DOM::MutationEvent>(event).relatedNode());
case PrevValue:
return String(static_cast<DOM::MutationEvent>(event).prevValue());
case NewValue:
return String(static_cast<DOM::MutationEvent>(event).newValue());
case AttrName:
return String(static_cast<DOM::MutationEvent>(event).attrName());
case AttrChange:
return Number((unsigned int)static_cast<DOM::MutationEvent>(event).attrChange());
default:
kdWarning() << "Unhandled token in DOMMutationEvent::getValueProperty : " << token << endl;
return Value();
}
}
Value DOMMutationEventProtoFunc::tryCall(ExecState *exec, Object &thisObj, const List &args)
{
if (!thisObj.inherits(&KJS::DOMMutationEvent::info)) {
Object err = Error::create(exec,TypeError);
exec->setException(err);
return err;
}
DOM::MutationEvent mutationEvent = static_cast<DOMMutationEvent *>(thisObj.imp())->toMutationEvent();
switch (id) {
case DOMMutationEvent::InitMutationEvent:
mutationEvent.initMutationEvent(args[0].toString(exec).string(), args[1].toBoolean(exec), args[2].toBoolean(exec), toNode(args[3]), args[4].toString(exec).string(), args[5].toString(exec).string(), args[6].toString(exec).string(), args[7].toInteger(exec)); return Undefined();
}
return Undefined();
}