blob: 85a6f402bbf0b2877a47cc92ea218250f8a14dcc [file] [log] [blame]
// Copyright 2013 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "ui/events/event_source.h"
#include <algorithm>
#include "base/stl_util.h"
#include "ui/events/event_rewriter.h"
#include "ui/events/event_sink.h"
namespace ui {
namespace {
bool IsLocatedEventWithDifferentLocations(const Event& event) {
if (!event.IsLocatedEvent())
return false;
const LocatedEvent* located_event = event.AsLocatedEvent();
return located_event->target() &&
located_event->location_f() != located_event->root_location_f();
}
} // namespace
EventSource::EventSource() {}
EventSource::~EventSource() {}
void EventSource::AddEventRewriter(EventRewriter* rewriter) {
DCHECK(rewriter);
DCHECK(!base::ContainsValue(rewriter_list_, rewriter));
rewriter_list_.push_back(rewriter);
}
void EventSource::RemoveEventRewriter(EventRewriter* rewriter) {
auto find = std::find(rewriter_list_.begin(), rewriter_list_.end(), rewriter);
if (find != rewriter_list_.end())
rewriter_list_.erase(find);
}
EventDispatchDetails EventSource::SendEventToSink(Event* event) {
return SendEventToSinkFromRewriter(event, nullptr);
}
EventDispatchDetails EventSource::DeliverEventToSink(Event* event) {
EventSink* sink = GetEventSink();
CHECK(sink);
return sink->OnEventFromSource(event);
}
EventDispatchDetails EventSource::SendEventToSinkFromRewriter(
Event* event,
const EventRewriter* rewriter) {
std::unique_ptr<ui::Event> event_for_rewriting_ptr;
Event* event_for_rewriting = event;
if (!rewriter_list_.empty() && IsLocatedEventWithDifferentLocations(*event)) {
// EventRewriters don't expect an event with differing location and
// root-location (because they don't honor the target). Provide such an
// event for rewriters.
event_for_rewriting_ptr = ui::Event::Clone(*event);
event_for_rewriting_ptr->AsLocatedEvent()->set_location_f(
event_for_rewriting_ptr->AsLocatedEvent()->root_location_f());
event_for_rewriting = event_for_rewriting_ptr.get();
}
std::unique_ptr<Event> rewritten_event;
EventRewriteStatus status = EVENT_REWRITE_CONTINUE;
EventRewriterList::const_iterator it = rewriter_list_.begin(),
end = rewriter_list_.end();
// If a rewriter reposted |event|, only send it to subsequent rewriters.
bool send_to_rewriter = rewriter == nullptr;
for (; it != end; ++it) {
if (!send_to_rewriter) {
send_to_rewriter |= (*it == rewriter);
continue;
}
status = (*it)->RewriteEvent(*event_for_rewriting, &rewritten_event);
if (status == EVENT_REWRITE_DISCARD) {
CHECK(!rewritten_event);
EventDispatchDetails details;
details.event_discarded = true;
return details;
}
if (status == EVENT_REWRITE_CONTINUE) {
CHECK(!rewritten_event);
continue;
}
break;
}
CHECK((it == end && !rewritten_event) || rewritten_event);
EventDispatchDetails details =
DeliverEventToSink(rewritten_event ? rewritten_event.get() : event);
if (details.dispatcher_destroyed)
return details;
while (status == EVENT_REWRITE_DISPATCH_ANOTHER) {
std::unique_ptr<Event> new_event;
status = (*it)->NextDispatchEvent(*rewritten_event, &new_event);
if (status == EVENT_REWRITE_DISCARD)
return EventDispatchDetails();
CHECK_NE(EVENT_REWRITE_CONTINUE, status);
CHECK(new_event);
details = DeliverEventToSink(new_event.get());
if (details.dispatcher_destroyed)
return details;
rewritten_event = std::move(new_event);
}
return EventDispatchDetails();
}
} // namespace ui