blob: 13f0275d8343a42f60149696032e2d04a1c1b3af [file] [log] [blame] [edit]
// Copyright 2014 The PDFium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
// Original code copyright 2014 Foxit Software Inc. http://www.foxitsoftware.com
#include "xfa/fwl/cfwl_notedriver.h"
#include <algorithm>
#include <utility>
#include "build/build_config.h"
#include "core/fxcrt/fx_extension.h"
#include "fxjs/gc/container_trace.h"
#include "xfa/fwl/cfwl_app.h"
#include "xfa/fwl/cfwl_event.h"
#include "xfa/fwl/cfwl_messagekey.h"
#include "xfa/fwl/cfwl_messagekillfocus.h"
#include "xfa/fwl/cfwl_messagemouse.h"
#include "xfa/fwl/cfwl_messagemousewheel.h"
#include "xfa/fwl/cfwl_messagesetfocus.h"
#include "xfa/fwl/cfwl_widgetmgr.h"
#include "xfa/fwl/fwl_widgetdef.h"
namespace pdfium {
namespace {
uint64_t g_next_listener_key = 1;
} // namespace
CFWL_NoteDriver::CFWL_NoteDriver(CFWL_App* pApp) : app_(pApp) {}
CFWL_NoteDriver::~CFWL_NoteDriver() = default;
void CFWL_NoteDriver::Trace(cppgc::Visitor* visitor) const {
visitor->Trace(app_);
visitor->Trace(hover_);
visitor->Trace(focus_);
visitor->Trace(grab_);
ContainerTrace(visitor, event_targets_);
}
void CFWL_NoteDriver::SendEvent(CFWL_Event* pNote) {
for (const auto& pair : event_targets_) {
if (pair.second->IsValid()) {
pair.second->ProcessEvent(pNote);
}
}
}
void CFWL_NoteDriver::RegisterEventTarget(CFWL_Widget* pListener,
CFWL_Widget* pEventSource) {
uint64_t key = pListener->GetEventKey();
if (key == 0) {
key = g_next_listener_key++;
pListener->SetEventKey(key);
}
if (!event_targets_[key]) {
event_targets_[key] = cppgc::MakeGarbageCollected<Target>(
app_->GetHeap()->GetAllocationHandle(), pListener);
}
event_targets_[key]->SetEventSource(pEventSource);
}
void CFWL_NoteDriver::UnregisterEventTarget(CFWL_Widget* pListener) {
uint64_t key = pListener->GetEventKey();
if (key == 0) {
return;
}
auto it = event_targets_.find(key);
if (it != event_targets_.end()) {
it->second->Invalidate();
}
}
void CFWL_NoteDriver::NotifyTargetHide(CFWL_Widget* pNoteTarget) {
if (focus_ == pNoteTarget) {
focus_ = nullptr;
}
if (hover_ == pNoteTarget) {
hover_ = nullptr;
}
if (grab_ == pNoteTarget) {
grab_ = nullptr;
}
}
void CFWL_NoteDriver::NotifyTargetDestroy(CFWL_Widget* pNoteTarget) {
if (focus_ == pNoteTarget) {
focus_ = nullptr;
}
if (hover_ == pNoteTarget) {
hover_ = nullptr;
}
if (grab_ == pNoteTarget) {
grab_ = nullptr;
}
UnregisterEventTarget(pNoteTarget);
}
void CFWL_NoteDriver::ProcessMessage(CFWL_Message* pMessage) {
CFWL_Widget* pMessageForm = pMessage->GetDstTarget();
if (!pMessageForm) {
return;
}
if (!DispatchMessage(pMessage, pMessageForm)) {
return;
}
if (pMessage->GetType() == CFWL_Message::Type::kMouse) {
MouseSecondary(pMessage);
}
}
bool CFWL_NoteDriver::DispatchMessage(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
switch (pMessage->GetType()) {
case CFWL_Message::Type::kSetFocus: {
if (!DoSetFocus(pMessage, pMessageForm)) {
return false;
}
break;
}
case CFWL_Message::Type::kKillFocus: {
if (!DoKillFocus(pMessage, pMessageForm)) {
return false;
}
break;
}
case CFWL_Message::Type::kKey: {
if (!DoKey(pMessage, pMessageForm)) {
return false;
}
break;
}
case CFWL_Message::Type::kMouse: {
if (!DoMouse(pMessage, pMessageForm)) {
return false;
}
break;
}
case CFWL_Message::Type::kMouseWheel: {
if (!DoWheel(pMessage, pMessageForm)) {
return false;
}
break;
}
}
IFWL_WidgetDelegate* pDelegate = pMessage->GetDstTarget()->GetDelegate();
if (pDelegate) {
pDelegate->OnProcessMessage(pMessage);
}
return true;
}
bool CFWL_NoteDriver::DoSetFocus(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
focus_ = pMessage->GetDstTarget();
return true;
}
bool CFWL_NoteDriver::DoKillFocus(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
if (focus_ == pMessage->GetDstTarget()) {
focus_ = nullptr;
}
return true;
}
bool CFWL_NoteDriver::DoKey(CFWL_Message* pMessage, CFWL_Widget* pMessageForm) {
CFWL_MessageKey* pMsg = static_cast<CFWL_MessageKey*>(pMessage);
#if !BUILDFLAG(IS_APPLE)
if (pMsg->cmd_ == CFWL_MessageKey::KeyCommand::kKeyDown &&
pMsg->key_code_or_char_ == XFA_FWL_VKEY_Tab) {
return true;
}
#endif
if (focus_) {
pMsg->SetDstTarget(focus_.Get());
return true;
}
if (pMsg->cmd_ == CFWL_MessageKey::KeyCommand::kKeyDown &&
pMsg->key_code_or_char_ == XFA_FWL_VKEY_Return) {
CFWL_WidgetMgr* pWidgetMgr = pMessageForm->GetFWLApp()->GetWidgetMgr();
CFWL_Widget* pDefButton = pWidgetMgr->GetDefaultButton(pMessageForm);
if (pDefButton) {
pMsg->SetDstTarget(pDefButton);
return true;
}
}
return false;
}
bool CFWL_NoteDriver::DoMouse(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
CFWL_MessageMouse* pMsg = static_cast<CFWL_MessageMouse*>(pMessage);
if (pMsg->cmd_ == CFWL_MessageMouse::MouseCommand::kLeave ||
pMsg->cmd_ == CFWL_MessageMouse::MouseCommand::kHover ||
pMsg->cmd_ == CFWL_MessageMouse::MouseCommand::kEnter) {
return !!pMsg->GetDstTarget();
}
if (pMsg->GetDstTarget() != pMessageForm) {
pMsg->pos_ = pMsg->GetDstTarget()->TransformTo(pMessageForm, pMsg->pos_);
}
if (!DoMouseEx(pMsg, pMessageForm)) {
pMsg->SetDstTarget(pMessageForm);
}
return true;
}
bool CFWL_NoteDriver::DoWheel(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
CFWL_WidgetMgr* pWidgetMgr = pMessageForm->GetFWLApp()->GetWidgetMgr();
CFWL_MessageMouseWheel* pMsg = static_cast<CFWL_MessageMouseWheel*>(pMessage);
CFWL_Widget* pDst = pWidgetMgr->GetWidgetAtPoint(pMessageForm, pMsg->pos());
if (!pDst) {
return false;
}
pMsg->set_pos(pMessageForm->TransformTo(pDst, pMsg->pos()));
pMsg->SetDstTarget(pDst);
return true;
}
bool CFWL_NoteDriver::DoMouseEx(CFWL_Message* pMessage,
CFWL_Widget* pMessageForm) {
CFWL_WidgetMgr* pWidgetMgr = pMessageForm->GetFWLApp()->GetWidgetMgr();
CFWL_Widget* pTarget = nullptr;
if (grab_) {
pTarget = grab_.Get();
}
CFWL_MessageMouse* pMsg = static_cast<CFWL_MessageMouse*>(pMessage);
if (!pTarget) {
pTarget = pWidgetMgr->GetWidgetAtPoint(pMessageForm, pMsg->pos_);
}
if (!pTarget) {
return false;
}
if (pTarget && pMessageForm != pTarget) {
pMsg->pos_ = pMessageForm->TransformTo(pTarget, pMsg->pos_);
}
pMsg->SetDstTarget(pTarget);
return true;
}
void CFWL_NoteDriver::MouseSecondary(CFWL_Message* pMessage) {
CFWL_Widget* pTarget = pMessage->GetDstTarget();
if (pTarget == hover_) {
return;
}
CFWL_MessageMouse* pMsg = static_cast<CFWL_MessageMouse*>(pMessage);
if (hover_) {
CFWL_MessageMouse msLeave(hover_.Get(),
CFWL_MessageMouse::MouseCommand::kLeave,
Mask<XFA_FWL_KeyFlag>(),
pTarget->TransformTo(hover_.Get(), pMsg->pos_));
DispatchMessage(&msLeave, nullptr);
}
if (pTarget->GetClassID() == FWL_Type::Form) {
hover_ = nullptr;
return;
}
hover_ = pTarget;
CFWL_MessageMouse msHover(pTarget, CFWL_MessageMouse::MouseCommand::kHover,
Mask<XFA_FWL_KeyFlag>(), pMsg->pos_);
DispatchMessage(&msHover, nullptr);
}
CFWL_NoteDriver::Target::Target(CFWL_Widget* pListener)
: listener_(pListener) {}
CFWL_NoteDriver::Target::~Target() = default;
void CFWL_NoteDriver::Target::Trace(cppgc::Visitor* visitor) const {
visitor->Trace(listener_);
for (auto& widget : widgets_) {
visitor->Trace(widget);
}
}
void CFWL_NoteDriver::Target::SetEventSource(CFWL_Widget* pSource) {
if (pSource) {
widgets_.insert(pSource);
}
}
bool CFWL_NoteDriver::Target::ProcessEvent(CFWL_Event* pEvent) {
IFWL_WidgetDelegate* pDelegate = listener_->GetDelegate();
if (!pDelegate) {
return false;
}
if (!widgets_.empty() && widgets_.count(pEvent->GetSrcTarget()) == 0) {
return false;
}
pDelegate->OnProcessEvent(pEvent);
return true;
}
} // namespace pdfium