arabica/include/DOM/Events/EventTargetImpl.hpp
Stefan Radomski 2e486cfd96 More work on DOM Level2 events
- dispatchEvent follows standard now
- actually dsipatch events when modifying the dom
2013-01-01 23:35:42 +01:00

173 lines
6.4 KiB
C++

#ifndef JEZUK_DOM_SimpleDOM_EVENTTARGET_IMPL_H
#define JEZUK_DOM_SimpleDOM_EVENTTARGET_IMPL_H
////////////////////////////
// C++ EventTarget implementation
////////////////////////////
#include <DOM/Node.hpp>
#include <DOM/Events/EventTarget.hpp>
#include <DOM/DOMException.hpp>
#include <XML/XMLCharacterClasses.hpp>
#include <deque>
#include <algorithm>
#include <map>
#include <list>
//#include <iostream>
namespace Arabica
{
namespace DOM
{
namespace Events
{
////////////////////////////////////////////////////////////////////
template<class stringT, class string_adaptorT> class DocumentImpl;
template<class stringT, class string_adaptorT>
class EventTargetImpl : virtual public DOM::Events::EventTarget_impl<stringT, string_adaptorT>
{
public:
typedef DocumentImpl<stringT, string_adaptorT> DocumentImplT;
typedef DOM::Node<stringT, string_adaptorT> DOMNodeT;
typedef DOM::Node_impl<stringT, string_adaptorT> DOMNode_implT;
typedef DOM::Text_impl<stringT, string_adaptorT> DOMText_implT;
typedef DOM::Document_impl<stringT, string_adaptorT> DOMDocument_implT;
typedef DOM::NamedNodeMap_impl<stringT, string_adaptorT> DOMNamedNodeMap_implT;
typedef DOM::NodeList_impl<stringT, string_adaptorT> DOMNodeList_implT;
typedef DOM::Events::Event<stringT, string_adaptorT> EventT;
typedef DOM::Events::Event_impl<stringT, string_adaptorT> Event_implT;
typedef DOM::Events::EventImpl<stringT, string_adaptorT> EventImplT;
typedef DOM::Events::EventTarget<stringT, string_adaptorT> EventTargetT;
typedef DOM::Events::EventTargetImpl<stringT, string_adaptorT> EventTargetImplT;
typedef DOM::Events::EventTarget_impl<stringT, string_adaptorT> EventTarget_implT;
typedef DOM::Events::EventListener<stringT, string_adaptorT> EventListenerT;
EventTargetImpl()
{
//std::cout << std::endl << "born " << this << std::endl;
} // EventTargetImpl
virtual ~EventTargetImpl()
{
//std::cout << std::endl << "die " << this << std::endl;
}
///////////////////////////////////////////////////////
// EventTarget methods
virtual void addEventListener(const stringT type,
EventListenerT& listener,
bool useCapture)
{
typedef typename std::multimap<stringT, EventListenerT*>::iterator iterator;
if (useCapture) {
std::pair<iterator, iterator> range = eventCapturers_.equal_range(type);
for (iterator it = range.first; it != range.second; ++it) {
if (it->second == &listener)
return;
}
eventCapturers_.insert(std::make_pair(type, &listener));
} else {
std::pair<iterator, iterator> range = eventListeners_.equal_range(type);
for (iterator it = range.first; it != range.second; ++it) {
if (it->second == &listener)
return;
}
eventListeners_.insert(std::make_pair(type, &listener));
}
}
virtual void removeEventListener(const stringT type,
EventListenerT& listener,
bool useCapture)
{
typedef typename std::multimap<stringT, EventListenerT*>::iterator iterator;
if (useCapture) {
std::pair<iterator, iterator> range = eventCapturers_.equal_range(type);
for (iterator it = range.first; it != range.second; ++it) {
if (it->second == &listener)
eventCapturers_.erase(it);
return;
}
} else {
std::pair<iterator, iterator> range = eventListeners_.equal_range(type);
for (iterator it = range.first; it != range.second; ++it) {
if (it->second == &listener)
eventListeners_.erase(it);
return;
}
}
}
virtual bool dispatchEvent(EventT& event)
{
typedef typename std::multimap<stringT, EventListenerT*>::iterator eventListenerIterator;
EventImplT* eventImpl = dynamic_cast<EventImplT*>(*event);
eventImpl->target_ = EventTargetT(this);
std::list<EventTargetT> eventAncestorChain;
DOMNode_implT* curr = dynamic_cast<DOMNode_implT*>(this);
while((curr = curr->getParentNode())) {
eventAncestorChain.push_front(EventTargetT(curr));
}
// capturing
eventImpl->phase_ = EventT::CAPTURING_PHASE;
typename std::list<EventTargetT>::iterator it = eventAncestorChain.begin();
while(it != eventAncestorChain.end()) {
EventTargetImplT* nodeImpl = dynamic_cast<EventTargetImplT*>(it->Impl());
if (nodeImpl != NULL) {
std::pair<eventListenerIterator, eventListenerIterator> range = nodeImpl->eventCapturers_.equal_range(event.getType());
for (eventListenerIterator it = range.first; it != range.second; ++it) {
it->second->handleEvent(event);
}
// stopPropagation was called on the event
if (eventImpl->stopped_)
return false;
}
++it;
}
// at target
eventImpl->phase_ = EventT::AT_TARGET;
std::pair<eventListenerIterator, eventListenerIterator> range = eventListeners_.equal_range(event.getType());
for (eventListenerIterator rit = range.first; rit != range.second; ++rit) {
rit->second->handleEvent(event);
}
// bubbling
if (eventImpl->getBubbles()) {
eventImpl->phase_ = EventT::BUBBLING_PHASE;
typename std::list<EventTargetT>::reverse_iterator rit = eventAncestorChain.rbegin();
while(rit != eventAncestorChain.rend()) {
EventTargetImplT* nodeImpl = dynamic_cast<EventTargetImplT*>(rit->Impl());
if (nodeImpl != NULL) {
std::pair<eventListenerIterator, eventListenerIterator> range = nodeImpl->eventListeners_.equal_range(event.getType());
for (eventListenerIterator rit = range.first; rit != range.second; ++rit) {
rit->second->handleEvent(event);
}
// stopPropagation was called on the event
if (eventImpl->stopped_)
return false;
}
++rit;
}
}
return true;
}
protected:
std::multimap<stringT, EventListenerT*> eventListeners_;
std::multimap<stringT, EventListenerT*> eventCapturers_;
}; // class EventTargetImpl
} // namespace DOM
} // namespace Events
} // namespace Arabica
#endif // JEZUK_DOM_SimpleDOM_EVENTTARGET_H
// end of file