gecko-dev/content/events/src/nsDOMEvent.cpp

1600 lines
44 KiB
C++
Raw Normal View History

/* -*- Mode: C++; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- */
/* ***** BEGIN LICENSE BLOCK *****
* Version: MPL 1.1/GPL 2.0/LGPL 2.1
*
* The contents of this file are subject to the Mozilla Public License Version
* 1.1 (the "License"); you may not use this file except in compliance with
* the License. You may obtain a copy of the License at
* http://www.mozilla.org/MPL/
*
* Software distributed under the License is distributed on an "AS IS" basis,
* WITHOUT WARRANTY OF ANY KIND, either express or implied. See the License
* for the specific language governing rights and limitations under the
* License.
*
* The Original Code is mozilla.org code.
*
* The Initial Developer of the Original Code is
* Netscape Communications Corporation.
* Portions created by the Initial Developer are Copyright (C) 1998
* the Initial Developer. All Rights Reserved.
*
* Contributor(s):
* Steve Clark (buster@netscape.com)
*
* Alternatively, the contents of this file may be used under the terms of
* either of the GNU General Public License Version 2 or later (the "GPL"),
* or the GNU Lesser General Public License Version 2.1 or later (the "LGPL"),
* in which case the provisions of the GPL or the LGPL are applicable instead
* of those above. If you wish to allow use of your version of this file only
* under the terms of either the GPL or the LGPL, and not to allow others to
* use your version of this file under the terms of the MPL, indicate your
* decision by deleting the provisions above and replace them with the notice
* and other provisions required by the GPL or the LGPL. If you do not delete
* the provisions above, a recipient may use your version of this file under
* the terms of any one of the MPL, the GPL or the LGPL.
*
* ***** END LICENSE BLOCK ***** */
#include "nsCOMPtr.h"
#include "nsDOMEvent.h"
#include "nsIDOMNode.h"
#include "nsIEventStateManager.h"
#include "nsIFrame.h"
#include "nsIContent.h"
#include "nsIRenderingContext.h"
#include "nsIWidget.h"
#include "nsIPresShell.h"
#include "nsPrivateTextRange.h"
#include "nsIDocument.h"
#include "nsIViewManager.h"
#include "nsIPrivateCompositionEvent.h"
#include "nsIScrollableView.h"
#include "nsIDOMEventTarget.h"
#include "nsIInterfaceRequestor.h"
#include "nsIInterfaceRequestorUtils.h"
#include "nsIDOMWindowInternal.h"
#include "nsIDOMAbstractView.h"
#include "prmem.h"
#include "nsLayoutAtoms.h"
2000-11-27 07:55:20 +00:00
#include "nsMutationEvent.h"
#include "nsContentUtils.h"
#include "nsIDOMKeyEvent.h"
#include "nsIDOMMutationEvent.h"
#include "nsIURI.h"
static const char* const sEventNames[] = {
"mousedown", "mouseup", "click", "dblclick", "mouseover",
"mouseout", "mousemove", "contextmenu", "keydown", "keyup", "keypress",
"focus", "blur", "load", "beforeunload", "unload", "abort", "error",
"submit", "reset", "change", "select", "input", "paint" ,"text",
"popupshowing", "popupshown", "popuphiding", "popuphidden", "close", "command", "broadcast", "commandupdate",
"dragenter", "dragover", "dragexit", "dragdrop", "draggesture", "resize",
2000-11-27 07:55:20 +00:00
"scroll","overflow", "underflow", "overflowchanged",
"DOMSubtreeModified", "DOMNodeInserted", "DOMNodeRemoved",
"DOMNodeRemovedFromDocument", "DOMNodeInsertedIntoDocument",
"DOMAttrModified", "DOMCharacterDataModified",
"popupBlocked", "DOMActivate", "DOMFocusIn", "DOMFocusOut"
1999-09-02 03:51:03 +00:00
};
/** event pool used as a simple recycler for objects of this class */
static PRUint8 gEventPool[ sizeof(nsDOMEvent) ];
/* declare static class data */
PRBool nsDOMEvent::gEventPoolInUse=PR_FALSE;
#ifdef NS_DEBUG // metrics for measuring event pool use
static PRInt32 numEvents=0;
static PRInt32 numNewEvents=0;
static PRInt32 numDelEvents=0;
static PRInt32 numAllocFromPool=0;
//#define NOISY_EVENT_LEAKS // define NOISY_EVENT_LEAKS to get metrics printed to stdout for all nsDOMEvent allocations
#endif
// allocate the memory for the object from the recycler, if possible
// otherwise, just grab it from the heap.
void*
nsDOMEvent::operator new(size_t aSize) CPP_THROW_NEW
{
#ifdef NS_DEBUG
numEvents++;
#endif
void *result = nsnull;
if (!gEventPoolInUse && aSize <= sizeof(gEventPool)) {
#ifdef NS_DEBUG
numAllocFromPool++;
#endif
result = &gEventPool;
gEventPoolInUse = PR_TRUE;
}
else {
#ifdef NS_DEBUG
numNewEvents++;
#endif
result = ::operator new(aSize);
}
if (result) {
memset(result, 0, aSize);
}
return result;
}
// Overridden to prevent the global delete from being called on objects from
// the recycler. Otherwise, just pass through to the global delete operator.
void
nsDOMEvent::operator delete(void* aPtr)
{
if (aPtr==&gEventPool) {
gEventPoolInUse = PR_FALSE;
}
else {
#ifdef NS_DEBUG
numDelEvents++;
#endif
::operator delete(aPtr);
}
#if defined(NS_DEBUG) && defined(NOISY_EVENT_LEAKS)
printf("total events =%d, from pool = %d, concurrent live events = %d\n",
numEvents, numAllocFromPool, numNewEvents-numDelEvents);
#endif
}
nsDOMEvent::nsDOMEvent(nsPresContext* aPresContext, nsEvent* aEvent,
const nsAString& aEventType)
{
mPresContext = aPresContext;
mEventIsTrusted = PR_FALSE;
if (aEvent) {
mEvent = aEvent;
mEventIsTrusted = PR_TRUE;
}
else {
mEventIsInternal = PR_TRUE;
AllocateEvent(aEventType);
}
// Get the explicit original target (if it's anonymous make it null)
{
mExplicitOriginalTarget = GetTargetFromFrame();
mTmpRealOriginalTarget = mExplicitOriginalTarget;
nsCOMPtr<nsIContent> content = do_QueryInterface(mExplicitOriginalTarget);
if (content) {
if (content->IsNativeAnonymous()) {
mExplicitOriginalTarget = nsnull;
}
if (content->GetBindingParent()) {
mExplicitOriginalTarget = nsnull;
}
}
}
mText = nsnull;
mButton = -1;
if (aEvent) {
mScreenPoint.x = aEvent->refPoint.x;
mScreenPoint.y = aEvent->refPoint.y;
mClientPoint.x = aEvent->point.x;
mClientPoint.y = aEvent->point.y;
} else
mScreenPoint.x = mScreenPoint.y = mClientPoint.x = mClientPoint.y = 0;
if (aEvent && aEvent->eventStructType == NS_TEXT_EVENT) {
//
// extract the IME composition string
//
nsTextEvent *te = (nsTextEvent*)aEvent;
mText = new nsString(te->theText);
//
// build the range list -- ranges need to be DOM-ified since the
// IME transaction will hold a ref, the widget representation
// isn't persistent
//
nsIPrivateTextRange** tempTextRangeList =
new nsIPrivateTextRange*[te->rangeCount];
if (tempTextRangeList) {
PRUint16 i;
for(i = 0; i < te->rangeCount; i++) {
nsPrivateTextRange* tempPrivateTextRange = new
nsPrivateTextRange(te->rangeArray[i].mStartOffset,
te->rangeArray[i].mEndOffset,
te->rangeArray[i].mRangeType);
if (tempPrivateTextRange) {
NS_ADDREF(tempPrivateTextRange);
tempTextRangeList[i] = (nsIPrivateTextRange*)tempPrivateTextRange;
}
}
}
// We need to create mTextRange even rangeCount is 0.
// If rangeCount is 0, mac carbon will return 0 for new and
// tempTextRangeList will be null. but we should still create
// mTextRange, otherwise, we will crash it later when some code
// call GetInputRange and AddRef to the result
mTextRange = new nsPrivateTextRangeList(te->rangeCount ,tempTextRangeList);
}
}
nsDOMEvent::~nsDOMEvent()
{
NS_ASSERT_OWNINGTHREAD(nsDOMEvent);
if (mEventIsInternal) {
if (mEvent->userType) {
delete mEvent->userType;
}
if (mEvent->eventStructType == NS_POPUPBLOCKED_EVENT) {
nsPopupBlockedEvent* event = NS_STATIC_CAST(nsPopupBlockedEvent*, mEvent);
NS_IF_RELEASE(event->mRequestingWindowURI);
NS_IF_RELEASE(event->mPopupWindowURI);
}
delete mEvent;
}
delete mText;
}
NS_IMPL_ADDREF(nsDOMEvent)
NS_IMPL_RELEASE(nsDOMEvent)
NS_INTERFACE_MAP_BEGIN(nsDOMEvent)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsIDOMEvent, nsIDOMMouseEvent)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsIDOMUIEvent, nsIDOMMouseEvent)
NS_INTERFACE_MAP_ENTRY(nsIDOMMouseEvent)
NS_INTERFACE_MAP_ENTRY(nsIDOMKeyEvent)
NS_INTERFACE_MAP_ENTRY(nsIDOMNSUIEvent)
NS_INTERFACE_MAP_ENTRY(nsIDOMNSEvent)
NS_INTERFACE_MAP_ENTRY(nsIDOMPopupBlockedEvent)
NS_INTERFACE_MAP_ENTRY(nsIPrivateDOMEvent)
NS_INTERFACE_MAP_ENTRY(nsIPrivateTextEvent)
NS_INTERFACE_MAP_ENTRY(nsIPrivateCompositionEvent)
NS_INTERFACE_MAP_ENTRY_AMBIGUOUS(nsISupports, nsIDOMMouseEvent)
NS_INTERFACE_MAP_ENTRY_CONTENT_CLASSINFO(Event)
NS_INTERFACE_MAP_END
// nsIDOMEventInterface
NS_METHOD nsDOMEvent::GetType(nsAString& aType)
{
const char* name = GetEventName(mEvent->message);
if (name) {
CopyASCIItoUTF16(name, aType);
return NS_OK;
}
else {
if (mEvent->message == NS_USER_DEFINED_EVENT && mEvent->userType) {
aType.Assign(NS_STATIC_CAST(nsStringKey*, mEvent->userType)->GetString());
return NS_OK;
}
}
return NS_ERROR_FAILURE;
}
NS_METHOD nsDOMEvent::GetTarget(nsIDOMEventTarget** aTarget)
{
if (nsnull != mTarget) {
*aTarget = mTarget;
NS_ADDREF(*aTarget);
return NS_OK;
}
*aTarget = nsnull;
nsCOMPtr<nsIContent> targetContent;
if (mPresContext) {
mPresContext->EventStateManager()->
GetEventTargetContent(mEvent, getter_AddRefs(targetContent));
}
if (targetContent) {
mTarget = do_QueryInterface(targetContent);
if (mTarget) {
*aTarget = mTarget;
NS_ADDREF(*aTarget);
}
}
else {
//Always want a target. Use document if nothing else.
nsIPresShell *presShell;
if (mPresContext && (presShell = mPresContext->GetPresShell())) {
nsIDocument *doc = presShell->GetDocument();
if (doc) {
mTarget = do_QueryInterface(doc);
if (mTarget) {
*aTarget = mTarget;
NS_ADDREF(*aTarget);
}
}
}
}
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetCurrentTarget(nsIDOMEventTarget** aCurrentTarget)
{
*aCurrentTarget = mCurrentTarget;
NS_IF_ADDREF(*aCurrentTarget);
return NS_OK;
}
//
// Get the actual event target node (may have been retargeted for mouse events)
//
already_AddRefed<nsIDOMEventTarget>
nsDOMEvent::GetTargetFromFrame()
{
if (!mPresContext) { return nsnull; }
// Get the target frame (have to get the ESM first)
nsIFrame* targetFrame = nsnull;
mPresContext->EventStateManager()->GetEventTarget(&targetFrame);
if (!targetFrame) { return nsnull; }
// get the real content
nsCOMPtr<nsIContent> realEventContent;
targetFrame->GetContentForEvent(mPresContext, mEvent, getter_AddRefs(realEventContent));
if (!realEventContent) { return nsnull; }
// Finally, we have the real content. QI it and return.
nsIDOMEventTarget* target = nsnull;
CallQueryInterface(realEventContent, &target);
return target;
}
NS_IMETHODIMP
nsDOMEvent::GetExplicitOriginalTarget(nsIDOMEventTarget** aRealEventTarget)
{
if (mExplicitOriginalTarget) {
*aRealEventTarget = mExplicitOriginalTarget;
NS_ADDREF(*aRealEventTarget);
return NS_OK;
}
return GetTarget(aRealEventTarget);
}
NS_IMETHODIMP
nsDOMEvent::GetTmpRealOriginalTarget(nsIDOMEventTarget** aRealEventTarget)
{
if (mTmpRealOriginalTarget) {
*aRealEventTarget = mTmpRealOriginalTarget;
NS_ADDREF(*aRealEventTarget);
return NS_OK;
}
return GetOriginalTarget(aRealEventTarget);
}
NS_IMETHODIMP
nsDOMEvent::GetOriginalTarget(nsIDOMEventTarget** aOriginalTarget)
{
2000-09-07 09:20:56 +00:00
if (!mOriginalTarget)
return GetTarget(aOriginalTarget);
*aOriginalTarget = mOriginalTarget;
NS_IF_ADDREF(*aOriginalTarget);
return NS_OK;
}
2000-09-07 09:20:56 +00:00
NS_IMETHODIMP
nsDOMEvent::HasOriginalTarget(PRBool* aResult)
{
*aResult = (mOriginalTarget != nsnull);
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::IsTrustedEvent(PRBool* aResult)
{
*aResult = mEventIsTrusted;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::SetTrusted(PRBool aTrusted)
{
mEventIsTrusted = aTrusted;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetEventPhase(PRUint16* aEventPhase)
{
if (mEvent->flags & NS_EVENT_FLAG_CAPTURE) {
if (mEvent->flags & NS_EVENT_FLAG_BUBBLE) {
*aEventPhase = nsIDOMMouseEvent::AT_TARGET;
}
else {
*aEventPhase = nsIDOMMouseEvent::CAPTURING_PHASE;
}
}
else if (mEvent->flags & NS_EVENT_FLAG_BUBBLE) {
*aEventPhase = nsIDOMMouseEvent::BUBBLING_PHASE;
}
else {
*aEventPhase = 0;
}
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetBubbles(PRBool* aBubbles)
{
*aBubbles = !(mEvent->flags & NS_EVENT_FLAG_CANT_BUBBLE);
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetCancelable(PRBool* aCancelable)
{
*aCancelable = !(mEvent->flags & NS_EVENT_FLAG_CANT_CANCEL);
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetTimeStamp(PRUint64* aTimeStamp)
{
LL_UI2L(*aTimeStamp, mEvent->time);
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::StopPropagation()
{
mEvent->flags |= NS_EVENT_FLAG_STOP_DISPATCH;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::PreventBubble()
{
if (mEvent->flags & NS_EVENT_FLAG_BUBBLE || mEvent->flags & NS_EVENT_FLAG_INIT) {
mEvent->flags |= NS_EVENT_FLAG_STOP_DISPATCH;
}
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::PreventCapture()
{
if (mEvent->flags & NS_EVENT_FLAG_CAPTURE) {
mEvent->flags |= NS_EVENT_FLAG_STOP_DISPATCH;
}
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::PreventDefault()
{
if (!(mEvent->flags & NS_EVENT_FLAG_CANT_CANCEL)) {
mEvent->flags |= NS_EVENT_FLAG_NO_DEFAULT;
}
return NS_OK;
}
2000-06-20 00:45:47 +00:00
NS_IMETHODIMP
nsDOMEvent::GetView(nsIDOMAbstractView** aView)
{
NS_ENSURE_ARG_POINTER(aView);
*aView = nsnull;
nsresult rv = NS_OK;
if (mPresContext) {
nsCOMPtr<nsISupports> container = mPresContext->GetContainer();
NS_ENSURE_TRUE(container, NS_OK);
nsCOMPtr<nsIDOMWindowInternal> window = do_GetInterface(container);
NS_ENSURE_TRUE(window, NS_OK);
CallQueryInterface(window, aView);
}
return rv;
}
NS_IMETHODIMP
nsDOMEvent::GetDetail(PRInt32* aDetail)
{
if (!mEvent) {
*aDetail = 0;
return NS_OK;
}
switch(mEvent->eventStructType)
{
case NS_SCROLLPORT_EVENT:
{
nsScrollPortEvent* scrollEvent = (nsScrollPortEvent*)mEvent;
*aDetail = (PRInt32)scrollEvent->orient;
return NS_OK;
}
case NS_MOUSE_EVENT:
{
switch (mEvent->message) {
case NS_MOUSE_LEFT_BUTTON_UP:
case NS_MOUSE_LEFT_BUTTON_DOWN:
case NS_MOUSE_LEFT_CLICK:
case NS_MOUSE_LEFT_DOUBLECLICK:
case NS_MOUSE_MIDDLE_BUTTON_UP:
case NS_MOUSE_MIDDLE_BUTTON_DOWN:
case NS_MOUSE_MIDDLE_CLICK:
case NS_MOUSE_MIDDLE_DOUBLECLICK:
case NS_MOUSE_RIGHT_BUTTON_UP:
case NS_MOUSE_RIGHT_BUTTON_DOWN:
case NS_MOUSE_RIGHT_CLICK:
case NS_MOUSE_RIGHT_DOUBLECLICK:
case NS_USER_DEFINED_EVENT:
*aDetail = ((nsMouseEvent*)mEvent)->clickCount;
break;
default:
break;
}
2000-06-23 22:59:18 +00:00
return NS_OK;
}
case NS_MOUSE_SCROLL_EVENT:
*aDetail = ((nsMouseScrollEvent*)mEvent)->delta;
break;
case NS_DOMUI_EVENT:
*aDetail = ((nsDOMUIEvent*)mEvent)->detail;
break;
default:
*aDetail = 0;
return NS_OK;
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetText(nsString& aText)
{
if (mEvent->message == NS_TEXT_TEXT) {
aText = *mText;
return NS_OK;
}
aText.Truncate();
return NS_ERROR_FAILURE;
}
NS_METHOD nsDOMEvent::GetInputRange(nsIPrivateTextRangeList** aInputRange)
{
NS_ENSURE_ARG_POINTER(aInputRange);
if (mEvent->message == NS_TEXT_TEXT) {
*aInputRange = mTextRange;
return NS_OK;
}
*aInputRange = 0;
return NS_ERROR_FAILURE;
}
NS_METHOD nsDOMEvent::GetEventReply(nsTextEventReply** aReply)
{
NS_ENSURE_ARG_POINTER(aReply);
if (mEvent->message==NS_TEXT_TEXT) {
*aReply = &(((nsTextEvent*)mEvent)->theReply);
return NS_OK;
}
*aReply = 0;
return NS_ERROR_FAILURE;
}
NS_METHOD nsDOMEvent::GetCompositionReply(nsTextEventReply** aReply)
{
NS_ENSURE_ARG_POINTER(aReply);
if((mEvent->message==NS_COMPOSITION_START) ||
(mEvent->message==NS_COMPOSITION_QUERY)) {
*aReply = &(((nsCompositionEvent*)mEvent)->theReply);
return NS_OK;
}
*aReply = 0;
return NS_ERROR_FAILURE;
}
NS_METHOD
nsDOMEvent::GetReconversionReply(nsReconversionEventReply** aReply)
{
NS_ENSURE_ARG_POINTER(aReply);
*aReply = &(((nsReconversionEvent*)mEvent)->theReply);
return NS_OK;
}
nsPoint nsDOMEvent::GetScreenPoint() {
if (!mEvent ||
(mEvent->eventStructType != NS_MOUSE_EVENT &&
mEvent->eventStructType != NS_POPUP_EVENT &&
!NS_IS_DRAG_EVENT(mEvent))) {
return nsPoint(0, 0);
}
if (!((nsGUIEvent*)mEvent)->widget ) {
return mScreenPoint;
}
nsRect bounds(mEvent->refPoint, nsSize(1, 1));
nsRect offset;
((nsGUIEvent*)mEvent)->widget->WidgetToScreen ( bounds, offset );
return offset.TopLeft();
}
NS_METHOD nsDOMEvent::GetScreenX(PRInt32* aScreenX)
{
NS_ENSURE_ARG_POINTER(aScreenX);
*aScreenX = GetScreenPoint().x;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetScreenY(PRInt32* aScreenY)
{
NS_ENSURE_ARG_POINTER(aScreenY);
*aScreenY = GetScreenPoint().y;
return NS_OK;
}
nsPoint nsDOMEvent::GetClientPoint() {
if (!mEvent ||
(mEvent->eventStructType != NS_MOUSE_EVENT &&
mEvent->eventStructType != NS_POPUP_EVENT &&
!NS_IS_DRAG_EVENT(mEvent)) ||
!mPresContext) {
return nsPoint(0, 0);
}
if (!((nsGUIEvent*)mEvent)->widget ) {
return mClientPoint;
}
//My god, man, there *must* be a better way to do this.
nsCOMPtr<nsIWidget> docWidget;
nsIPresShell *presShell = mPresContext->GetPresShell();
if (presShell) {
nsIViewManager* vm = presShell->GetViewManager();
if (vm) {
vm->GetWidget(getter_AddRefs(docWidget));
}
}
nsPoint pt = mEvent->refPoint;
nsCOMPtr<nsIWidget> eventWidget = ((nsGUIEvent*)mEvent)->widget;
while (eventWidget && docWidget != eventWidget) {
2001-10-07 06:32:36 +00:00
nsWindowType windowType;
eventWidget->GetWindowType(windowType);
2001-10-07 06:32:36 +00:00
if (windowType == eWindowType_popup)
break;
nsRect bounds;
eventWidget->GetBounds(bounds);
pt += bounds.TopLeft();
eventWidget = dont_AddRef(eventWidget->GetParent());
}
if (eventWidget != docWidget) {
// docWidget wasn't on the chain from the event widget to the root
// of the widget tree (or the nearest popup). OK, so now pt is
// relative to eventWidget; to get it relative to docWidget, we
// need to subtract docWidget's offset from eventWidget.
while (docWidget && docWidget != eventWidget) {
nsWindowType windowType;
docWidget->GetWindowType(windowType);
if (windowType == eWindowType_popup) {
// oh dear. the doc and the event were in different popups?
// That shouldn't happen.
NS_NOTREACHED("doc widget and event widget are in different popups. That's dumb.");
break;
}
nsRect bounds;
docWidget->GetBounds(bounds);
pt -= bounds.TopLeft();
docWidget = docWidget->GetParent();
}
}
return pt;
}
NS_METHOD nsDOMEvent::GetClientX(PRInt32* aClientX)
{
NS_ENSURE_ARG_POINTER(aClientX);
*aClientX = GetClientPoint().x;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetClientY(PRInt32* aClientY)
{
NS_ENSURE_ARG_POINTER(aClientY);
*aClientY = GetClientPoint().y;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetAltKey(PRBool* aIsDown)
{
NS_ENSURE_ARG_POINTER(aIsDown);
*aIsDown = ((nsInputEvent*)mEvent)->isAlt;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetCtrlKey(PRBool* aIsDown)
{
NS_ENSURE_ARG_POINTER(aIsDown);
*aIsDown = ((nsInputEvent*)mEvent)->isControl;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetShiftKey(PRBool* aIsDown)
{
NS_ENSURE_ARG_POINTER(aIsDown);
*aIsDown = ((nsInputEvent*)mEvent)->isShift;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetMetaKey(PRBool* aIsDown)
{
NS_ENSURE_ARG_POINTER(aIsDown);
*aIsDown = ((nsInputEvent*)mEvent)->isMeta;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetCharCode(PRUint32* aCharCode)
{
NS_ENSURE_ARG_POINTER(aCharCode);
if (!mEvent || mEvent->eventStructType != NS_KEY_EVENT) {
*aCharCode = 0;
return NS_OK;
}
switch (mEvent->message) {
case NS_KEY_UP:
case NS_KEY_DOWN:
2000-02-15 05:06:45 +00:00
#if defined(NS_DEBUG) && defined(DEBUG_brade)
printf("GetCharCode used for wrong key event; should use onkeypress.\n");
#endif
*aCharCode = 0;
break;
case NS_KEY_PRESS:
*aCharCode = ((nsKeyEvent*)mEvent)->charCode;
break;
default:
break;
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetKeyCode(PRUint32* aKeyCode)
{
NS_ENSURE_ARG_POINTER(aKeyCode);
if (!mEvent || mEvent->eventStructType != NS_KEY_EVENT) {
*aKeyCode = 0;
return NS_OK;
}
switch (mEvent->message) {
case NS_KEY_UP:
case NS_KEY_PRESS:
case NS_KEY_DOWN:
*aKeyCode = ((nsKeyEvent*)mEvent)->keyCode;
break;
default:
break;
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetButton(PRUint16* aButton)
{
NS_ENSURE_ARG_POINTER(aButton);
if (!mEvent || mEvent->eventStructType != NS_MOUSE_EVENT) {
NS_WARNING("Tried to get mouse button for null or non-mouse event!");
*aButton = (PRUint16)-1;
return NS_OK;
}
// If button has been set then use that instead.
if (mButton > 0) {
*aButton = (PRUint16)mButton;
}
else {
switch (mEvent->message) {
case NS_MOUSE_LEFT_BUTTON_UP:
case NS_MOUSE_LEFT_BUTTON_DOWN:
case NS_MOUSE_LEFT_CLICK:
case NS_MOUSE_LEFT_DOUBLECLICK:
*aButton = 0;
break;
case NS_MOUSE_MIDDLE_BUTTON_UP:
case NS_MOUSE_MIDDLE_BUTTON_DOWN:
case NS_MOUSE_MIDDLE_CLICK:
case NS_MOUSE_MIDDLE_DOUBLECLICK:
*aButton = 1;
break;
case NS_MOUSE_RIGHT_BUTTON_UP:
case NS_MOUSE_RIGHT_BUTTON_DOWN:
case NS_MOUSE_RIGHT_CLICK:
case NS_MOUSE_RIGHT_DOUBLECLICK:
*aButton = 2;
break;
default:
break;
}
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetRelatedTarget(nsIDOMEventTarget** aRelatedTarget)
{
NS_ENSURE_ARG_POINTER(aRelatedTarget);
*aRelatedTarget = nsnull;
if (!mPresContext) {
return NS_OK;
}
nsCOMPtr<nsIContent> relatedContent;
mPresContext->EventStateManager()->
GetEventRelatedContent(getter_AddRefs(relatedContent));
if (!relatedContent) {
return NS_OK;
}
return CallQueryInterface(relatedContent, aRelatedTarget);
}
// nsINSEventInterface
NS_METHOD nsDOMEvent::GetLayerX(PRInt32* aLayerX)
{
NS_ENSURE_ARG_POINTER(aLayerX);
if (!mEvent || (mEvent->eventStructType != NS_MOUSE_EVENT) ||
!mPresContext) {
*aLayerX = 0;
return NS_OK;
}
float t2p;
t2p = mPresContext->TwipsToPixels();
*aLayerX = NSTwipsToIntPixels(mEvent->point.x, t2p);
return NS_OK;
}
NS_METHOD nsDOMEvent::GetLayerY(PRInt32* aLayerY)
{
NS_ENSURE_ARG_POINTER(aLayerY);
if (!mEvent || (mEvent->eventStructType != NS_MOUSE_EVENT) ||
!mPresContext) {
*aLayerY = 0;
return NS_OK;
}
float t2p;
t2p = mPresContext->TwipsToPixels();
*aLayerY = NSTwipsToIntPixels(mEvent->point.y, t2p);
return NS_OK;
}
nsresult nsDOMEvent::GetScrollInfo(nsIScrollableView** aScrollableView,
float* aP2T, float* aT2P)
{
NS_ENSURE_ARG_POINTER(aScrollableView);
NS_ENSURE_ARG_POINTER(aP2T);
NS_ENSURE_ARG_POINTER(aT2P);
if (!mPresContext) {
*aScrollableView = nsnull;
return NS_ERROR_FAILURE;
}
*aP2T = mPresContext->PixelsToTwips();
*aT2P = mPresContext->TwipsToPixels();
nsIPresShell *presShell = mPresContext->GetPresShell();
if (presShell) {
nsIViewManager* vm = presShell->GetViewManager();
if(vm) {
return vm->GetRootScrollableView(aScrollableView);
}
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetPageX(PRInt32* aPageX)
{
NS_ENSURE_ARG_POINTER(aPageX);
nsresult ret = NS_OK;
PRInt32 scrollX = 0;
nsIScrollableView* view = nsnull;
float p2t, t2p;
GetScrollInfo(&view, &p2t, &t2p);
if(view) {
nscoord xPos, yPos;
ret = view->GetScrollPosition(xPos, yPos);
scrollX = NSTwipsToIntPixels(xPos, t2p);
}
if (NS_SUCCEEDED(ret)) {
ret = GetClientX(aPageX);
}
if (NS_SUCCEEDED(ret)) {
*aPageX += scrollX;
}
return ret;
}
NS_METHOD nsDOMEvent::GetPageY(PRInt32* aPageY)
{
NS_ENSURE_ARG_POINTER(aPageY);
nsresult ret = NS_OK;
PRInt32 scrollY = 0;
nsIScrollableView* view = nsnull;
float p2t, t2p;
GetScrollInfo(&view, &p2t, &t2p);
if(view) {
nscoord xPos, yPos;
ret = view->GetScrollPosition(xPos, yPos);
scrollY = NSTwipsToIntPixels(yPos, t2p);
}
if (NS_SUCCEEDED(ret)) {
ret = GetClientY(aPageY);
}
if (NS_SUCCEEDED(ret)) {
*aPageY += scrollY;
}
return ret;
}
NS_METHOD nsDOMEvent::GetWhich(PRUint32* aWhich)
{
NS_ENSURE_ARG_POINTER(aWhich);
switch (mEvent->eventStructType) {
case NS_KEY_EVENT:
switch (mEvent->message) {
case NS_KEY_UP:
case NS_KEY_DOWN:
return GetKeyCode(aWhich);
case NS_KEY_PRESS:
//Special case for 4xp bug 62878. Try to make value of which
//more closely mirror the values that 4.x gave for RETURN and BACKSPACE
{
PRUint32 keyCode = ((nsKeyEvent*)mEvent)->keyCode;
if (keyCode == NS_VK_RETURN || keyCode == NS_VK_BACK) {
*aWhich = keyCode;
return NS_OK;
}
return GetCharCode(aWhich);
}
default:
break;
}
case NS_MOUSE_EVENT:
{
PRUint16 button;
(void) GetButton(&button);
*aWhich = button + 1;
break;
}
default:
*aWhich = 0;
break;
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetRangeParent(nsIDOMNode** aRangeParent)
{
NS_ENSURE_ARG_POINTER(aRangeParent);
nsIFrame* targetFrame = nsnull;
if (mPresContext) {
mPresContext->EventStateManager()->GetEventTarget(&targetFrame);
}
*aRangeParent = nsnull;
if (targetFrame) {
nsCOMPtr<nsIContent> parent;
PRInt32 offset, endOffset;
PRBool beginOfContent;
if (NS_SUCCEEDED(targetFrame->GetContentAndOffsetsFromPoint(mPresContext,
mEvent->point,
getter_AddRefs(parent),
offset,
endOffset,
beginOfContent))) {
if (parent) {
return CallQueryInterface(parent, aRangeParent);
}
}
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetRangeOffset(PRInt32* aRangeOffset)
{
NS_ENSURE_ARG_POINTER(aRangeOffset);
nsIFrame* targetFrame = nsnull;
if (mPresContext) {
mPresContext->EventStateManager()->GetEventTarget(&targetFrame);
}
if (targetFrame) {
nsIContent* parent = nsnull;
1999-06-09 07:51:52 +00:00
PRInt32 endOffset;
PRBool beginOfContent;
if (NS_SUCCEEDED(targetFrame->GetContentAndOffsetsFromPoint(mPresContext,
mEvent->point,
&parent,
*aRangeOffset,
endOffset,
beginOfContent))) {
NS_IF_RELEASE(parent);
return NS_OK;
}
}
*aRangeOffset = 0;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetCancelBubble(PRBool* aCancelBubble)
{
NS_ENSURE_ARG_POINTER(aCancelBubble);
if (mEvent->flags & NS_EVENT_FLAG_BUBBLE || mEvent->flags & NS_EVENT_FLAG_INIT) {
*aCancelBubble = (mEvent->flags &= NS_EVENT_FLAG_STOP_DISPATCH) ? PR_TRUE : PR_FALSE;
}
else {
*aCancelBubble = PR_FALSE;
}
return NS_OK;
}
NS_METHOD nsDOMEvent::SetCancelBubble(PRBool aCancelBubble)
{
if (mEvent->flags & NS_EVENT_FLAG_BUBBLE || mEvent->flags & NS_EVENT_FLAG_INIT) {
if (aCancelBubble) {
mEvent->flags |= NS_EVENT_FLAG_STOP_DISPATCH;
}
else {
mEvent->flags &= ~NS_EVENT_FLAG_STOP_DISPATCH;
}
}
return NS_OK;
}
NS_METHOD nsDOMEvent::GetIsChar(PRBool* aIsChar)
{
NS_ENSURE_ARG_POINTER(aIsChar);
if (!mEvent) {
*aIsChar = PR_FALSE;
return NS_OK;
}
if (mEvent->eventStructType == NS_KEY_EVENT) {
*aIsChar = ((nsKeyEvent*)mEvent)->isChar;
return NS_OK;
}
if (mEvent->eventStructType == NS_TEXT_EVENT) {
*aIsChar = ((nsTextEvent*)mEvent)->isChar;
return NS_OK;
}
*aIsChar = PR_FALSE;
return NS_OK;
}
NS_METHOD nsDOMEvent::GetPreventDefault(PRBool* aReturn)
{
NS_ENSURE_ARG_POINTER(aReturn);
if (!mEvent) {
*aReturn = PR_FALSE;
}
else {
*aReturn = (mEvent->flags & NS_EVENT_FLAG_NO_DEFAULT) ? PR_TRUE : PR_FALSE;
}
return NS_OK;
}
nsresult
nsDOMEvent::SetEventType(const nsAString& aEventTypeArg)
{
nsCOMPtr<nsIAtom> atom= do_GetAtom(NS_LITERAL_STRING("on") + aEventTypeArg);
mEvent->message = NS_USER_DEFINED_EVENT;
if (mEvent->eventStructType == NS_MOUSE_EVENT) {
if (atom == nsLayoutAtoms::onmousedown)
mEvent->message = NS_MOUSE_LEFT_BUTTON_DOWN;
else if (atom == nsLayoutAtoms::onmouseup)
mEvent->message = NS_MOUSE_LEFT_BUTTON_UP;
else if (atom == nsLayoutAtoms::onclick)
mEvent->message = NS_MOUSE_LEFT_CLICK;
else if (atom == nsLayoutAtoms::ondblclick)
mEvent->message = NS_MOUSE_LEFT_DOUBLECLICK;
else if (atom == nsLayoutAtoms::onmouseover)
mEvent->message = NS_MOUSE_ENTER_SYNTH;
else if (atom == nsLayoutAtoms::onmouseout)
mEvent->message = NS_MOUSE_EXIT_SYNTH;
else if (atom == nsLayoutAtoms::onmousemove)
mEvent->message = NS_MOUSE_MOVE;
else if (atom == nsLayoutAtoms::oncontextmenu)
mEvent->message = NS_CONTEXTMENU;
} else if (mEvent->eventStructType == NS_KEY_EVENT) {
if (atom == nsLayoutAtoms::onkeydown)
mEvent->message = NS_KEY_DOWN;
else if (atom == nsLayoutAtoms::onkeyup)
mEvent->message = NS_KEY_UP;
else if (atom == nsLayoutAtoms::onkeypress)
mEvent->message = NS_KEY_PRESS;
} else if (mEvent->eventStructType == NS_EVENT) {
if (atom == nsLayoutAtoms::onfocus)
mEvent->message = NS_FOCUS_CONTENT;
else if (atom == nsLayoutAtoms::onblur)
mEvent->message = NS_BLUR_CONTENT;
else if (atom == nsLayoutAtoms::onsubmit)
mEvent->message = NS_FORM_SUBMIT;
else if (atom == nsLayoutAtoms::onreset)
mEvent->message = NS_FORM_RESET;
else if (atom == nsLayoutAtoms::onchange)
mEvent->message = NS_FORM_CHANGE;
else if (atom == nsLayoutAtoms::onselect)
mEvent->message = NS_FORM_SELECTED;
else if (atom == nsLayoutAtoms::onload)
mEvent->message = NS_PAGE_LOAD;
else if (atom == nsLayoutAtoms::onunload)
mEvent->message = NS_PAGE_UNLOAD;
else if (atom == nsLayoutAtoms::onabort)
mEvent->message = NS_IMAGE_ABORT;
else if (atom == nsLayoutAtoms::onerror)
mEvent->message = NS_IMAGE_ERROR;
} else if (mEvent->eventStructType == NS_MUTATION_EVENT) {
if (atom == nsLayoutAtoms::onDOMAttrModified)
mEvent->message = NS_MUTATION_ATTRMODIFIED;
else if (atom == nsLayoutAtoms::onDOMCharacterDataModified)
mEvent->message = NS_MUTATION_CHARACTERDATAMODIFIED;
else if (atom == nsLayoutAtoms::onDOMNodeInserted)
mEvent->message = NS_MUTATION_NODEINSERTED;
else if (atom == nsLayoutAtoms::onDOMNodeRemoved)
mEvent->message = NS_MUTATION_NODEREMOVED;
else if (atom == nsLayoutAtoms::onDOMNodeInsertedIntoDocument)
mEvent->message = NS_MUTATION_NODEINSERTEDINTODOCUMENT;
else if (atom == nsLayoutAtoms::onDOMNodeRemovedFromDocument)
mEvent->message = NS_MUTATION_NODEREMOVEDFROMDOCUMENT;
else if (atom == nsLayoutAtoms::onDOMSubtreeModified)
mEvent->message = NS_MUTATION_SUBTREEMODIFIED;
} else if (mEvent->eventStructType == NS_DOMUI_EVENT) {
if (atom == nsLayoutAtoms::onDOMActivate)
mEvent->message = NS_DOMUI_ACTIVATE;
else if (atom == nsLayoutAtoms::onDOMFocusIn)
mEvent->message = NS_DOMUI_FOCUSIN;
else if (atom == nsLayoutAtoms::onDOMFocusOut)
mEvent->message = NS_DOMUI_FOCUSOUT;
}
if (mEvent->message == NS_USER_DEFINED_EVENT)
mEvent->userType = new nsStringKey(aEventTypeArg);
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::InitEvent(const nsAString& aEventTypeArg, PRBool aCanBubbleArg, PRBool aCancelableArg)
{
NS_ENSURE_SUCCESS(SetEventType(aEventTypeArg), NS_ERROR_FAILURE);
mEvent->flags |= aCanBubbleArg ? NS_EVENT_FLAG_NONE : NS_EVENT_FLAG_CANT_BUBBLE;
mEvent->flags |= aCancelableArg ? NS_EVENT_FLAG_NONE : NS_EVENT_FLAG_CANT_CANCEL;
2000-06-20 00:45:47 +00:00
mEvent->internalAppFlags |= NS_APP_EVENT_FLAG_NONE;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::InitUIEvent(const nsAString& aTypeArg, PRBool aCanBubbleArg, PRBool aCancelableArg,
nsIDOMAbstractView* aViewArg, PRInt32 aDetailArg)
{
nsresult rv = nsDOMEvent::InitEvent(aTypeArg, aCanBubbleArg, aCancelableArg);
NS_ENSURE_SUCCESS(rv, NS_ERROR_FAILURE);
if (mEvent->eventStructType == NS_DOMUI_EVENT) {
nsDOMUIEvent *event = NS_STATIC_CAST(nsDOMUIEvent*, mEvent);
event->detail = aDetailArg;
return NS_OK;
}
NS_ERROR("event type mismatch");
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP
nsDOMEvent::InitMouseEvent(const nsAString & aTypeArg, PRBool aCanBubbleArg, PRBool aCancelableArg,
nsIDOMAbstractView *aViewArg, PRInt32 aDetailArg, PRInt32 aScreenXArg,
PRInt32 aScreenYArg, PRInt32 aClientXArg, PRInt32 aClientYArg,
PRBool aCtrlKeyArg, PRBool aAltKeyArg, PRBool aShiftKeyArg,
PRBool aMetaKeyArg, PRUint16 aButtonArg, nsIDOMEventTarget *aRelatedTargetArg)
{
nsresult rv = nsDOMEvent::InitEvent(aTypeArg, aCanBubbleArg, aCancelableArg);
NS_ENSURE_SUCCESS(rv, NS_ERROR_FAILURE);
NS_ASSERTION(mEvent->eventStructType == NS_MOUSE_EVENT || mEvent->eventStructType == NS_MOUSE_SCROLL_EVENT, "event type mismatch");
if (mEvent->eventStructType == NS_MOUSE_EVENT || mEvent->eventStructType == NS_MOUSE_SCROLL_EVENT) {
nsInputEvent* inputEvent = NS_STATIC_CAST(nsInputEvent*, mEvent);
inputEvent->isControl = aCtrlKeyArg;
inputEvent->isAlt = aAltKeyArg;
inputEvent->isShift = aShiftKeyArg;
inputEvent->isMeta = aMetaKeyArg;
inputEvent->point.x = aClientXArg;
inputEvent->point.y = aClientYArg;
inputEvent->refPoint.x = aScreenXArg;
inputEvent->refPoint.y = aScreenYArg;
mScreenPoint.x = aScreenXArg;
mScreenPoint.y = aScreenYArg;
mClientPoint.x = aClientXArg;
mClientPoint.y = aClientYArg;
mButton = aButtonArg;
if (mEvent->eventStructType == NS_MOUSE_SCROLL_EVENT) {
nsMouseScrollEvent* scrollEvent = NS_STATIC_CAST(nsMouseScrollEvent*, mEvent);
scrollEvent->delta = aDetailArg;
} else {
nsMouseEvent* mouseEvent = NS_STATIC_CAST(nsMouseEvent*, mEvent);
mouseEvent->clickCount = aDetailArg;
}
return NS_OK;
}
//include a way to set view once we have more than one
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP
nsDOMEvent::InitKeyEvent(const nsAString& aTypeArg, PRBool aCanBubbleArg, PRBool aCancelableArg,
nsIDOMAbstractView* aViewArg, PRBool aCtrlKeyArg, PRBool aAltKeyArg,
PRBool aShiftKeyArg, PRBool aMetaKeyArg,
PRUint32 aKeyCodeArg, PRUint32 aCharCodeArg)
{
nsresult rv = nsDOMEvent::InitEvent(aTypeArg, aCanBubbleArg, aCancelableArg);
NS_ENSURE_SUCCESS(rv, NS_ERROR_FAILURE);
NS_ASSERTION(mEvent->eventStructType == NS_KEY_EVENT, "event type mismatch");
if (mEvent->eventStructType == NS_KEY_EVENT) {
nsKeyEvent* keyEvent = NS_STATIC_CAST(nsKeyEvent*, mEvent);
keyEvent->isControl = aCtrlKeyArg;
keyEvent->isAlt = aAltKeyArg;
keyEvent->isShift = aShiftKeyArg;
keyEvent->isMeta = aMetaKeyArg;
keyEvent->keyCode = aKeyCodeArg;
keyEvent->charCode = aCharCodeArg;
return NS_OK;
}
//include a way to set view once we have more than one
return NS_ERROR_FAILURE;
}
NS_IMETHODIMP nsDOMEvent::InitPopupBlockedEvent(const nsAString & aTypeArg,
PRBool aCanBubbleArg, PRBool aCancelableArg,
nsIURI *aRequestingWindowURI,
nsIURI *aPopupWindowURI,
const nsAString & aPopupWindowFeatures)
{
nsresult rv = nsDOMEvent::InitEvent(aTypeArg, aCanBubbleArg, aCancelableArg);
NS_ENSURE_SUCCESS(rv, NS_ERROR_FAILURE);
NS_ASSERTION(mEvent->eventStructType == NS_POPUPBLOCKED_EVENT, "event type mismatch");
if (mEvent->eventStructType == NS_POPUPBLOCKED_EVENT) {
nsPopupBlockedEvent* event = NS_STATIC_CAST(nsPopupBlockedEvent*, mEvent);
event->mRequestingWindowURI = aRequestingWindowURI;
event->mPopupWindowURI = aPopupWindowURI;
NS_IF_ADDREF(event->mRequestingWindowURI);
NS_IF_ADDREF(event->mPopupWindowURI);
event->mPopupWindowFeatures = aPopupWindowFeatures;
return NS_OK;
}
return NS_ERROR_FAILURE;
}
/* readonly attribute nsIURI requestingWindowURI; */
NS_IMETHODIMP nsDOMEvent::GetRequestingWindowURI(nsIURI **aRequestingWindowURI)
{
NS_ENSURE_ARG_POINTER(aRequestingWindowURI);
if (mEvent->eventStructType == NS_POPUPBLOCKED_EVENT) {
nsPopupBlockedEvent* event = NS_STATIC_CAST(nsPopupBlockedEvent*, mEvent);
*aRequestingWindowURI = event->mRequestingWindowURI;
NS_IF_ADDREF(*aRequestingWindowURI);
return NS_OK;
}
*aRequestingWindowURI = 0;
return NS_OK; // Don't throw an exception
}
/* readonly attribute nsIURI popupWindowURI; */
NS_IMETHODIMP nsDOMEvent::GetPopupWindowURI(nsIURI **aPopupWindowURI)
{
NS_ENSURE_ARG_POINTER(aPopupWindowURI);
if (mEvent->eventStructType == NS_POPUPBLOCKED_EVENT) {
nsPopupBlockedEvent* event = NS_STATIC_CAST(nsPopupBlockedEvent*, mEvent);
*aPopupWindowURI = event->mPopupWindowURI;
NS_IF_ADDREF(*aPopupWindowURI);
return NS_OK;
}
*aPopupWindowURI = 0;
return NS_OK; // Don't throw an exception
}
/* readonly attribute DOMString popupFeatures; */
NS_IMETHODIMP nsDOMEvent::GetPopupWindowFeatures(nsAString &aPopupWindowFeatures)
{
if (mEvent->eventStructType == NS_POPUPBLOCKED_EVENT) {
nsPopupBlockedEvent* event = NS_STATIC_CAST(nsPopupBlockedEvent*, mEvent);
aPopupWindowFeatures = event->mPopupWindowFeatures;
return NS_OK;
}
aPopupWindowFeatures.Truncate();
return NS_OK; // Don't throw an exception
}
NS_METHOD nsDOMEvent::DuplicatePrivateData()
{
//XXX Write me!
//XXX And when you do, make sure to copy over the event target here, too!
return NS_OK;
}
NS_METHOD nsDOMEvent::SetTarget(nsIDOMEventTarget* aTarget)
{
mTarget = aTarget;
return NS_OK;
}
NS_METHOD nsDOMEvent::SetCurrentTarget(nsIDOMEventTarget* aCurrentTarget)
{
mCurrentTarget = aCurrentTarget;
return NS_OK;
}
NS_METHOD nsDOMEvent::SetOriginalTarget(nsIDOMEventTarget* aOriginalTarget)
{
mOriginalTarget = aOriginalTarget;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::IsDispatchStopped(PRBool* aIsDispatchStopped)
{
if (mEvent->flags & NS_EVENT_FLAG_STOP_DISPATCH) {
*aIsDispatchStopped = PR_TRUE;
} else {
*aIsDispatchStopped = PR_FALSE;
}
return NS_OK;
}
2000-06-20 00:45:47 +00:00
NS_IMETHODIMP
nsDOMEvent::IsHandled(PRBool* aIsHandled)
{
if (mEvent->internalAppFlags | NS_APP_EVENT_FLAG_HANDLED) {
*aIsHandled = PR_TRUE;
} else {
*aIsHandled = PR_FALSE;
}
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::SetHandled(PRBool aHandled)
{
if(aHandled)
mEvent->internalAppFlags |= NS_APP_EVENT_FLAG_HANDLED;
else
mEvent->internalAppFlags &= ~NS_APP_EVENT_FLAG_HANDLED;
return NS_OK;
}
NS_IMETHODIMP
nsDOMEvent::GetInternalNSEvent(nsEvent** aNSEvent)
{
NS_ENSURE_ARG_POINTER(aNSEvent);
*aNSEvent = mEvent;
return NS_OK;
}
const char* nsDOMEvent::GetEventName(PRUint32 aEventType)
{
switch(aEventType) {
case NS_MOUSE_LEFT_BUTTON_DOWN:
case NS_MOUSE_MIDDLE_BUTTON_DOWN:
case NS_MOUSE_RIGHT_BUTTON_DOWN:
return sEventNames[eDOMEvents_mousedown];
case NS_MOUSE_LEFT_BUTTON_UP:
case NS_MOUSE_MIDDLE_BUTTON_UP:
case NS_MOUSE_RIGHT_BUTTON_UP:
return sEventNames[eDOMEvents_mouseup];
case NS_MOUSE_LEFT_CLICK:
case NS_MOUSE_MIDDLE_CLICK:
case NS_MOUSE_RIGHT_CLICK:
return sEventNames[eDOMEvents_click];
case NS_MOUSE_LEFT_DOUBLECLICK:
case NS_MOUSE_MIDDLE_DOUBLECLICK:
case NS_MOUSE_RIGHT_DOUBLECLICK:
return sEventNames[eDOMEvents_dblclick];
case NS_MOUSE_ENTER_SYNTH:
return sEventNames[eDOMEvents_mouseover];
case NS_MOUSE_EXIT_SYNTH:
return sEventNames[eDOMEvents_mouseout];
case NS_MOUSE_MOVE:
return sEventNames[eDOMEvents_mousemove];
case NS_KEY_UP:
return sEventNames[eDOMEvents_keyup];
case NS_KEY_DOWN:
return sEventNames[eDOMEvents_keydown];
case NS_KEY_PRESS:
return sEventNames[eDOMEvents_keypress];
case NS_FOCUS_CONTENT:
return sEventNames[eDOMEvents_focus];
case NS_BLUR_CONTENT:
return sEventNames[eDOMEvents_blur];
case NS_XUL_CLOSE:
return sEventNames[eDOMEvents_close];
case NS_PAGE_LOAD:
case NS_IMAGE_LOAD:
case NS_SCRIPT_LOAD:
return sEventNames[eDOMEvents_load];
case NS_BEFORE_PAGE_UNLOAD:
return sEventNames[eDOMEvents_beforeunload];
case NS_PAGE_UNLOAD:
return sEventNames[eDOMEvents_unload];
case NS_IMAGE_ABORT:
return sEventNames[eDOMEvents_abort];
case NS_IMAGE_ERROR:
case NS_SCRIPT_ERROR:
return sEventNames[eDOMEvents_error];
case NS_FORM_SUBMIT:
return sEventNames[eDOMEvents_submit];
case NS_FORM_RESET:
return sEventNames[eDOMEvents_reset];
1998-11-02 23:05:46 +00:00
case NS_FORM_CHANGE:
return sEventNames[eDOMEvents_change];
2004-06-21 23:35:56 +00:00
case NS_FORM_SELECTED:
return sEventNames[eDOMEvents_select];
case NS_FORM_INPUT:
return sEventNames[eDOMEvents_input];
1998-10-06 21:01:59 +00:00
case NS_PAINT:
return sEventNames[eDOMEvents_paint];
case NS_RESIZE_EVENT:
return sEventNames[eDOMEvents_resize];
case NS_SCROLL_EVENT:
return sEventNames[eDOMEvents_scroll];
case NS_TEXT_TEXT:
return sEventNames[eDOMEvents_text];
2001-08-06 21:49:35 +00:00
case NS_XUL_POPUP_SHOWING:
return sEventNames[eDOMEvents_popupShowing];
2001-08-06 21:49:35 +00:00
case NS_XUL_POPUP_SHOWN:
return sEventNames[eDOMEvents_popupShown];
2001-08-06 21:49:35 +00:00
case NS_XUL_POPUP_HIDING:
return sEventNames[eDOMEvents_popupHiding];
2001-08-06 21:49:35 +00:00
case NS_XUL_POPUP_HIDDEN:
return sEventNames[eDOMEvents_popupHidden];
2001-08-06 21:49:35 +00:00
case NS_XUL_COMMAND:
return sEventNames[eDOMEvents_command];
case NS_XUL_BROADCAST:
return sEventNames[eDOMEvents_broadcast];
case NS_XUL_COMMAND_UPDATE:
return sEventNames[eDOMEvents_commandupdate];
1999-09-02 03:51:03 +00:00
case NS_DRAGDROP_ENTER:
return sEventNames[eDOMEvents_dragenter];
case NS_DRAGDROP_OVER_SYNTH:
return sEventNames[eDOMEvents_dragover];
case NS_DRAGDROP_EXIT_SYNTH:
return sEventNames[eDOMEvents_dragexit];
1999-09-02 03:51:03 +00:00
case NS_DRAGDROP_DROP:
return sEventNames[eDOMEvents_dragdrop];
1999-09-02 03:51:03 +00:00
case NS_DRAGDROP_GESTURE:
return sEventNames[eDOMEvents_draggesture];
case NS_SCROLLPORT_OVERFLOW:
return sEventNames[eDOMEvents_overflow];
case NS_SCROLLPORT_UNDERFLOW:
return sEventNames[eDOMEvents_underflow];
case NS_SCROLLPORT_OVERFLOWCHANGED:
return sEventNames[eDOMEvents_overflowchanged];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_SUBTREEMODIFIED:
return sEventNames[eDOMEvents_subtreemodified];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_NODEINSERTED:
return sEventNames[eDOMEvents_nodeinserted];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_NODEREMOVED:
return sEventNames[eDOMEvents_noderemoved];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_NODEREMOVEDFROMDOCUMENT:
return sEventNames[eDOMEvents_noderemovedfromdocument];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_NODEINSERTEDINTODOCUMENT:
return sEventNames[eDOMEvents_nodeinsertedintodocument];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_ATTRMODIFIED:
return sEventNames[eDOMEvents_attrmodified];
2000-11-27 07:55:20 +00:00
case NS_MUTATION_CHARACTERDATAMODIFIED:
return sEventNames[eDOMEvents_characterdatamodified];
2001-03-30 02:30:17 +00:00
case NS_CONTEXTMENU:
case NS_CONTEXTMENU_KEY:
return sEventNames[eDOMEvents_contextmenu];
case NS_DOMUI_ACTIVATE:
return sEventNames[eDOMEvents_DOMActivate];
case NS_DOMUI_FOCUSIN:
return sEventNames[eDOMEvents_DOMFocusIn];
case NS_DOMUI_FOCUSOUT:
return sEventNames[eDOMEvents_DOMFocusOut];
default:
break;
}
return nsnull;
}
nsresult
NS_NewDOMUIEvent(nsIDOMEvent** aInstancePtrResult,
nsPresContext* aPresContext, const nsAString& aEventType,
nsEvent *aEvent)
{
nsDOMEvent* it = new nsDOMEvent(aPresContext, aEvent, aEventType);
if (nsnull == it) {
return NS_ERROR_OUT_OF_MEMORY;
}
return CallQueryInterface(it, aInstancePtrResult);
}
void
nsDOMEvent::AllocateEvent(const nsAString& aEventType)
{
//Allocate internal event
nsAutoString eventType(aEventType);
if (eventType.LowerCaseEqualsLiteral("mouseevents")) {
mEvent = new nsMouseEvent();
}
else if (eventType.LowerCaseEqualsLiteral("mousescrollevents")) {
mEvent = new nsMouseScrollEvent();
}
else if (eventType.LowerCaseEqualsLiteral("keyevents")) {
mEvent = new nsKeyEvent();
}
else if (eventType.LowerCaseEqualsLiteral("mutationevents")) {
mEvent = new nsMutationEvent();
}
else if (eventType.LowerCaseEqualsLiteral("popupevents")) {
mEvent = new nsGUIEvent();
}
else if (eventType.LowerCaseEqualsLiteral("popupblockedevents")) {
mEvent = new nsPopupBlockedEvent();
}
else if (eventType.LowerCaseEqualsLiteral("uievents")) {
mEvent = new nsDOMUIEvent();
}
else {
mEvent = new nsEvent();
}
mEvent->time = PR_Now();
}