Home | History | Annotate | Download | only in dom
      1 /*
      2  * Copyright (C) 2001 Peter Kelly (pmk (at) post.com)
      3  * Copyright (C) 2001 Tobias Anton (anton (at) stud.fbi.fh-darmstadt.de)
      4  * Copyright (C) 2006 Samuel Weinig (sam.weinig (at) gmail.com)
      5  * Copyright (C) 2003, 2005, 2006, 2008 Apple Inc. All rights reserved.
      6  *
      7  * This library is free software; you can redistribute it and/or
      8  * modify it under the terms of the GNU Library General Public
      9  * License as published by the Free Software Foundation; either
     10  * version 2 of the License, or (at your option) any later version.
     11  *
     12  * This library is distributed in the hope that it will be useful,
     13  * but WITHOUT ANY WARRANTY; without even the implied warranty of
     14  * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
     15  * Library General Public License for more details.
     16  *
     17  * You should have received a copy of the GNU Library General Public License
     18  * along with this library; see the file COPYING.LIB.  If not, write to
     19  * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor,
     20  * Boston, MA 02110-1301, USA.
     21  */
     22 
     23 #include "config.h"
     24 #include "MouseEvent.h"
     25 
     26 #include "EventDispatcher.h"
     27 #include "EventNames.h"
     28 #include "Frame.h"
     29 #include "FrameView.h"
     30 #include "PlatformMouseEvent.h"
     31 
     32 namespace WebCore {
     33 
     34 PassRefPtr<MouseEvent> MouseEvent::create(const AtomicString& eventType, PassRefPtr<AbstractView> view, const PlatformMouseEvent& event, int detail, PassRefPtr<Node> relatedTarget)
     35 {
     36     ASSERT(event.eventType() == MouseEventMoved || event.button() != NoButton);
     37 
     38     bool isCancelable = eventType != eventNames().mousemoveEvent;
     39 
     40     return MouseEvent::create(eventType, true, isCancelable, view,
     41         detail, event.globalX(), event.globalY(), event.x(), event.y(),
     42         event.ctrlKey(), event.altKey(), event.shiftKey(), event.metaKey(), event.button(),
     43         relatedTarget, 0, false);
     44 }
     45 
     46 MouseEvent::MouseEvent()
     47     : m_button(0)
     48     , m_buttonDown(false)
     49 {
     50 }
     51 
     52 MouseEvent::MouseEvent(const AtomicString& eventType, bool canBubble, bool cancelable, PassRefPtr<AbstractView> view,
     53                        int detail, int screenX, int screenY, int pageX, int pageY,
     54                        bool ctrlKey, bool altKey, bool shiftKey, bool metaKey,
     55                        unsigned short button, PassRefPtr<EventTarget> relatedTarget,
     56                        PassRefPtr<Clipboard> clipboard, bool isSimulated)
     57     : MouseRelatedEvent(eventType, canBubble, cancelable, view, detail, screenX, screenY,
     58                         pageX, pageY, ctrlKey, altKey, shiftKey, metaKey, isSimulated)
     59     , m_button(button == (unsigned short)-1 ? 0 : button)
     60     , m_buttonDown(button != (unsigned short)-1)
     61     , m_relatedTarget(relatedTarget)
     62     , m_clipboard(clipboard)
     63 {
     64 }
     65 
     66 MouseEvent::~MouseEvent()
     67 {
     68 }
     69 
     70 void MouseEvent::initMouseEvent(const AtomicString& type, bool canBubble, bool cancelable, PassRefPtr<AbstractView> view,
     71                                 int detail, int screenX, int screenY, int clientX, int clientY,
     72                                 bool ctrlKey, bool altKey, bool shiftKey, bool metaKey,
     73                                 unsigned short button, PassRefPtr<EventTarget> relatedTarget)
     74 {
     75     if (dispatched())
     76         return;
     77 
     78     initUIEvent(type, canBubble, cancelable, view, detail);
     79 
     80     m_screenX = screenX;
     81     m_screenY = screenY;
     82     m_ctrlKey = ctrlKey;
     83     m_altKey = altKey;
     84     m_shiftKey = shiftKey;
     85     m_metaKey = metaKey;
     86     m_button = button == (unsigned short)-1 ? 0 : button;
     87     m_buttonDown = button != (unsigned short)-1;
     88     m_relatedTarget = relatedTarget;
     89 
     90     initCoordinates(clientX, clientY);
     91 
     92     // FIXME: m_isSimulated is not set to false here.
     93     // FIXME: m_clipboard is not set to 0 here.
     94 }
     95 
     96 bool MouseEvent::isMouseEvent() const
     97 {
     98     return true;
     99 }
    100 
    101 bool MouseEvent::isDragEvent() const
    102 {
    103     const AtomicString& t = type();
    104     return t == eventNames().dragenterEvent || t == eventNames().dragoverEvent || t == eventNames().dragleaveEvent || t == eventNames().dropEvent
    105                || t == eventNames().dragstartEvent|| t == eventNames().dragEvent || t == eventNames().dragendEvent;
    106 }
    107 
    108 int MouseEvent::which() const
    109 {
    110     // For the DOM, the return values for left, middle and right mouse buttons are 0, 1, 2, respectively.
    111     // For the Netscape "which" property, the return values for left, middle and right mouse buttons are 1, 2, 3, respectively.
    112     // So we must add 1.
    113     return m_button + 1;
    114 }
    115 
    116 Node* MouseEvent::toElement() const
    117 {
    118     // MSIE extension - "the object toward which the user is moving the mouse pointer"
    119     if (type() == eventNames().mouseoutEvent)
    120         return relatedTarget() ? relatedTarget()->toNode() : 0;
    121 
    122     return target() ? target()->toNode() : 0;
    123 }
    124 
    125 Node* MouseEvent::fromElement() const
    126 {
    127     // MSIE extension - "object from which activation or the mouse pointer is exiting during the event" (huh?)
    128     if (type() != eventNames().mouseoutEvent)
    129         return relatedTarget() ? relatedTarget()->toNode() : 0;
    130 
    131     return target() ? target()->toNode() : 0;
    132 }
    133 
    134 PassRefPtr<SimulatedMouseEvent> SimulatedMouseEvent::create(const AtomicString& eventType, PassRefPtr<AbstractView> view, PassRefPtr<Event> underlyingEvent)
    135 {
    136     return adoptRef(new SimulatedMouseEvent(eventType, view, underlyingEvent));
    137 }
    138 
    139 SimulatedMouseEvent::~SimulatedMouseEvent()
    140 {
    141 }
    142 
    143 SimulatedMouseEvent::SimulatedMouseEvent(const AtomicString& eventType, PassRefPtr<AbstractView> view, PassRefPtr<Event> underlyingEvent)
    144     : MouseEvent(eventType, true, true, view, 0, 0, 0, 0, 0, false, false, false, false, 0, 0, 0, true)
    145 {
    146     if (UIEventWithKeyState* keyStateEvent = findEventWithKeyState(underlyingEvent.get())) {
    147         m_ctrlKey = keyStateEvent->ctrlKey();
    148         m_altKey = keyStateEvent->altKey();
    149         m_shiftKey = keyStateEvent->shiftKey();
    150         m_metaKey = keyStateEvent->metaKey();
    151     }
    152     setUnderlyingEvent(underlyingEvent);
    153 }
    154 
    155 MouseEventDispatchMediator::MouseEventDispatchMediator(PassRefPtr<MouseEvent> mouseEvent)
    156     : EventDispatchMediator(mouseEvent)
    157 {
    158 }
    159 
    160 MouseEvent* MouseEventDispatchMediator::event() const
    161 {
    162     return static_cast<MouseEvent*>(EventDispatchMediator::event());
    163 }
    164 
    165 bool MouseEventDispatchMediator::dispatchEvent(EventDispatcher* dispatcher) const
    166 {
    167     if (dispatcher->node()->disabled()) // Don't even send DOM events for disabled controls..
    168         return true;
    169 
    170     if (event()->type().isEmpty())
    171         return false; // Shouldn't happen.
    172 
    173     RefPtr<EventTarget> relatedTarget = dispatcher->adjustRelatedTarget(event(), event()->relatedTarget());
    174     event()->setRelatedTarget(relatedTarget);
    175 
    176     dispatcher->dispatchEvent(event());
    177     bool swallowEvent = event()->defaultHandled() || event()->defaultPrevented();
    178 
    179     // Special case: If it's a double click event, we also send the dblclick event. This is not part
    180     // of the DOM specs, but is used for compatibility with the ondblclick="" attribute. This is treated
    181     // as a separate event in other DOM-compliant browsers like Firefox, and so we do the same.
    182     if (event()->type() == eventNames().clickEvent && event()->detail() == 2) {
    183         RefPtr<MouseEvent> doubleClickEvent = MouseEvent::create();
    184         doubleClickEvent->initMouseEvent(eventNames().dblclickEvent, event()->bubbles(), event()->cancelable(), event()->view(),
    185                 event()->detail(), event()->screenX(), event()->screenY(), event()->clientX(), event()->clientY(),
    186                 event()->ctrlKey(), event()->altKey(), event()->shiftKey(), event()->metaKey(),
    187                 event()->button(), relatedTarget);
    188         if (event()->defaultHandled())
    189             doubleClickEvent->setDefaultHandled();
    190         dispatcher->dispatchEvent(doubleClickEvent);
    191         if (doubleClickEvent->defaultHandled() || doubleClickEvent->defaultPrevented())
    192             swallowEvent = true;
    193     }
    194 
    195     return swallowEvent;
    196 }
    197 
    198 } // namespace WebCore
    199