// 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/fxfa/cxfa_fffield.h" #include #include #include "constants/ascii.h" #include "third_party/base/check.h" #include "xfa/fgas/graphics/cfgas_gecolor.h" #include "xfa/fgas/graphics/cfgas_gegraphics.h" #include "xfa/fgas/graphics/cfgas_gepath.h" #include "xfa/fwl/cfwl_edit.h" #include "xfa/fwl/cfwl_eventmouse.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_picturebox.h" #include "xfa/fwl/cfwl_widgetmgr.h" #include "xfa/fwl/fwl_widgetdef.h" #include "xfa/fxfa/cxfa_ffapp.h" #include "xfa/fxfa/cxfa_ffdoc.h" #include "xfa/fxfa/cxfa_ffdocview.h" #include "xfa/fxfa/cxfa_ffpageview.h" #include "xfa/fxfa/cxfa_ffwidget.h" #include "xfa/fxfa/cxfa_fwltheme.h" #include "xfa/fxfa/cxfa_textlayout.h" #include "xfa/fxfa/parser/cxfa_border.h" #include "xfa/fxfa/parser/cxfa_calculate.h" #include "xfa/fxfa/parser/cxfa_caption.h" #include "xfa/fxfa/parser/cxfa_margin.h" #include "xfa/fxfa/parser/cxfa_node.h" #include "xfa/fxfa/parser/cxfa_script.h" namespace { constexpr float kMinUIHeight = 4.32f; constexpr float kDefaultUIHeight = 2.0f; } // namespace CXFA_FFField::CXFA_FFField(CXFA_Node* pNode) : CXFA_FFWidget(pNode) {} CXFA_FFField::~CXFA_FFField() = default; CXFA_FFDropDown* CXFA_FFField::AsDropDown() { return nullptr; } CXFA_FFField* CXFA_FFField::AsField() { return this; } void CXFA_FFField::Trace(cppgc::Visitor* visitor) const { CXFA_FFWidget::Trace(visitor); visitor->Trace(m_pNormalWidget); } CFX_RectF CXFA_FFField::GetBBox(FocusOption focus) { if (focus == kDoNotDrawFocus) return CXFA_FFWidget::GetBBox(kDoNotDrawFocus); switch (m_pNode->GetFFWidgetType()) { case XFA_FFWidgetType::kButton: case XFA_FFWidgetType::kCheckButton: case XFA_FFWidgetType::kImageEdit: case XFA_FFWidgetType::kSignature: case XFA_FFWidgetType::kChoiceList: return GetRotateMatrix().TransformRect(m_UIRect); default: return CFX_RectF(); } } void CXFA_FFField::RenderWidget(CFGAS_GEGraphics* pGS, const CFX_Matrix& matrix, HighlightOption highlight) { if (!HasVisibleStatus()) return; CFX_Matrix mtRotate = GetRotateMatrix(); mtRotate.Concat(matrix); CXFA_FFWidget::RenderWidget(pGS, mtRotate, highlight); DrawBorder(pGS, m_pNode->GetUIBorder(), m_UIRect, mtRotate); RenderCaption(pGS, mtRotate); DrawHighlight(pGS, mtRotate, highlight, kSquareShape); CFX_RectF rtWidget = GetNormalWidget()->GetWidgetRect(); CFX_Matrix mt(1, 0, 0, 1, rtWidget.left, rtWidget.top); mt.Concat(mtRotate); GetApp()->GetFWLWidgetMgr()->OnDrawWidget(GetNormalWidget(), pGS, mt); } void CXFA_FFField::DrawHighlight(CFGAS_GEGraphics* pGS, const CFX_Matrix& pMatrix, HighlightOption highlight, ShapeOption shape) { if (highlight == kNoHighlight) return; if (m_UIRect.IsEmpty() || !GetDoc()->GetXFADoc()->IsInteractive() || !m_pNode->IsOpenAccess()) { return; } pGS->SetFillColor(CFGAS_GEColor(GetDoc()->GetHighlightColor())); CFGAS_GEPath path; if (shape == kRoundShape) path.AddEllipse(m_UIRect); else path.AddRectangle(m_UIRect.left, m_UIRect.top, m_UIRect.width, m_UIRect.height); pGS->FillPath(path, CFX_FillRenderOptions::FillType::kWinding, pMatrix); } CFWL_Widget* CXFA_FFField::GetNormalWidget() { return m_pNormalWidget; } const CFWL_Widget* CXFA_FFField::GetNormalWidget() const { return m_pNormalWidget; } void CXFA_FFField::SetNormalWidget(CFWL_Widget* widget) { m_pNormalWidget = widget; } bool CXFA_FFField::IsLoaded() { return GetNormalWidget() && CXFA_FFWidget::IsLoaded(); } bool CXFA_FFField::LoadWidget() { m_pNode->LoadCaption(GetDoc()); PerformLayout(); return true; } void CXFA_FFField::SetEditScrollOffset() { XFA_FFWidgetType eType = m_pNode->GetFFWidgetType(); if (eType != XFA_FFWidgetType::kTextEdit && eType != XFA_FFWidgetType::kNumericEdit && eType != XFA_FFWidgetType::kPasswordEdit) { return; } float fScrollOffset = 0; CXFA_ContentLayoutItem* pItem = GetLayoutItem()->GetPrev(); CXFA_FFField* pPrev = pItem ? ToField(pItem->GetFFWidget()) : nullptr; if (pPrev) fScrollOffset = -(m_pNode->GetUIMargin().top); while (pPrev) { fScrollOffset += pPrev->m_UIRect.height; pItem = pPrev->GetLayoutItem()->GetPrev(); pPrev = pItem ? ToField(pItem->GetFFWidget()) : nullptr; } static_cast(GetNormalWidget())->SetScrollOffset(fScrollOffset); } bool CXFA_FFField::PerformLayout() { CXFA_FFWidget::PerformLayout(); CapPlacement(); LayoutCaption(); SetFWLRect(); SetEditScrollOffset(); if (GetNormalWidget()) GetNormalWidget()->Update(); return true; } void CXFA_FFField::CapPlacement() { CFX_RectF rtWidget = GetRectWithoutRotate(); CXFA_Margin* margin = m_pNode->GetMarginIfExists(); if (margin) { CXFA_ContentLayoutItem* pItem = GetLayoutItem(); float fLeftInset = margin->GetLeftInset(); float fRightInset = margin->GetRightInset(); float fTopInset = margin->GetTopInset(); float fBottomInset = margin->GetBottomInset(); if (!pItem->GetPrev() && !pItem->GetNext()) { rtWidget.Deflate(fLeftInset, fTopInset, fRightInset, fBottomInset); } else { if (!pItem->GetPrev()) rtWidget.Deflate(fLeftInset, fTopInset, fRightInset, 0); else if (!pItem->GetNext()) rtWidget.Deflate(fLeftInset, 0, fRightInset, fBottomInset); else rtWidget.Deflate(fLeftInset, 0, fRightInset, 0); } } XFA_AttributeValue iCapPlacement = XFA_AttributeValue::Unknown; float fCapReserve = 0; CXFA_Caption* caption = m_pNode->GetCaptionIfExists(); if (caption && !caption->IsHidden()) { iCapPlacement = caption->GetPlacementType(); if ((iCapPlacement == XFA_AttributeValue::Top && GetLayoutItem()->GetPrev()) || (iCapPlacement == XFA_AttributeValue::Bottom && GetLayoutItem()->GetNext())) { m_CaptionRect = CFX_RectF(); } else { fCapReserve = caption->GetReserve(); if (iCapPlacement == XFA_AttributeValue::Top || iCapPlacement == XFA_AttributeValue::Bottom) { fCapReserve = std::min(fCapReserve, rtWidget.height); } else { fCapReserve = std::min(fCapReserve, rtWidget.width); } CXFA_ContentLayoutItem* pItem = GetLayoutItem(); if (!pItem->GetPrev() && !pItem->GetNext()) { m_CaptionRect = rtWidget; } else { pItem = pItem->GetFirst(); m_CaptionRect = pItem->GetAbsoluteRect(); pItem = pItem->GetNext(); while (pItem) { m_CaptionRect.height += pItem->GetAbsoluteRect().Height(); pItem = pItem->GetNext(); } XFA_RectWithoutMargin(&m_CaptionRect, margin); } CXFA_TextLayout* pCapTextLayout = m_pNode->GetCaptionTextLayout(); if (fCapReserve <= 0 && pCapTextLayout) { CFX_SizeF minSize; CFX_SizeF maxSize; CFX_SizeF size = pCapTextLayout->CalcSize(minSize, maxSize); if (iCapPlacement == XFA_AttributeValue::Top || iCapPlacement == XFA_AttributeValue::Bottom) { fCapReserve = size.height; } else { fCapReserve = size.width; } } } } m_UIRect = rtWidget; CXFA_Margin* capMargin = caption ? caption->GetMarginIfExists() : nullptr; switch (iCapPlacement) { case XFA_AttributeValue::Left: { m_CaptionRect.width = fCapReserve; CapLeftRightPlacement(capMargin, rtWidget, iCapPlacement); m_UIRect.width -= fCapReserve; m_UIRect.left += fCapReserve; break; } case XFA_AttributeValue::Top: { m_CaptionRect.height = fCapReserve; CapTopBottomPlacement(capMargin, rtWidget, iCapPlacement); m_UIRect.top += fCapReserve; m_UIRect.height -= fCapReserve; break; } case XFA_AttributeValue::Right: { m_CaptionRect.left = m_CaptionRect.right() - fCapReserve; m_CaptionRect.width = fCapReserve; CapLeftRightPlacement(capMargin, rtWidget, iCapPlacement); m_UIRect.width -= fCapReserve; break; } case XFA_AttributeValue::Bottom: { m_CaptionRect.top = m_CaptionRect.bottom() - fCapReserve; m_CaptionRect.height = fCapReserve; CapTopBottomPlacement(capMargin, rtWidget, iCapPlacement); m_UIRect.height -= fCapReserve; break; } case XFA_AttributeValue::Inline: break; default: break; } CXFA_Border* borderUI = m_pNode->GetUIBorder(); if (borderUI) { CXFA_Margin* borderMargin = borderUI->GetMarginIfExists(); XFA_RectWithoutMargin(&m_UIRect, borderMargin); } m_UIRect.Normalize(); } void CXFA_FFField::CapTopBottomPlacement(const CXFA_Margin* margin, const CFX_RectF& rtWidget, XFA_AttributeValue iCapPlacement) { CFX_RectF rtUIMargin = m_pNode->GetUIMargin(); m_CaptionRect.left += rtUIMargin.left; if (margin) { XFA_RectWithoutMargin(&m_CaptionRect, margin); if (m_CaptionRect.height < 0) m_CaptionRect.top += m_CaptionRect.height; } float fWidth = rtUIMargin.left + rtUIMargin.width; float fHeight = m_CaptionRect.height + rtUIMargin.top + rtUIMargin.height; if (fWidth > rtWidget.width) m_UIRect.width += fWidth - rtWidget.width; if (fHeight == kDefaultUIHeight && m_UIRect.height < kMinUIHeight) { m_UIRect.height = kMinUIHeight; m_CaptionRect.top += rtUIMargin.top + rtUIMargin.height; } else if (fHeight > rtWidget.height) { m_UIRect.height += fHeight - rtWidget.height; if (iCapPlacement == XFA_AttributeValue::Bottom) m_CaptionRect.top += fHeight - rtWidget.height; } } void CXFA_FFField::CapLeftRightPlacement(const CXFA_Margin* margin, const CFX_RectF& rtWidget, XFA_AttributeValue iCapPlacement) { CFX_RectF rtUIMargin = m_pNode->GetUIMargin(); m_CaptionRect.top += rtUIMargin.top; m_CaptionRect.height -= rtUIMargin.top; if (margin) { XFA_RectWithoutMargin(&m_CaptionRect, margin); if (m_CaptionRect.height < 0) m_CaptionRect.top += m_CaptionRect.height; } float fWidth = m_CaptionRect.width + rtUIMargin.left + rtUIMargin.width; float fHeight = rtUIMargin.top + rtUIMargin.height; if (fWidth > rtWidget.width) { m_UIRect.width += fWidth - rtWidget.width; if (iCapPlacement == XFA_AttributeValue::Right) m_CaptionRect.left += fWidth - rtWidget.width; } if (fHeight == kDefaultUIHeight && m_UIRect.height < kMinUIHeight) { m_UIRect.height = kMinUIHeight; m_CaptionRect.top += rtUIMargin.top + rtUIMargin.height; } else if (fHeight > rtWidget.height) { m_UIRect.height += fHeight - rtWidget.height; } } void CXFA_FFField::UpdateFWL() { if (GetNormalWidget()) GetNormalWidget()->Update(); } uint32_t CXFA_FFField::UpdateUIProperty() { CXFA_Node* pUiNode = m_pNode->GetUIChildNode(); if (pUiNode && pUiNode->GetElementType() == XFA_Element::DefaultUi) return FWL_STYLEEXT_EDT_ReadOnly; return 0; } void CXFA_FFField::SetFWLRect() { if (!GetNormalWidget()) return; CFX_RectF rtUi = m_UIRect; rtUi.width = std::max(rtUi.width, 1.0f); if (!GetDoc()->GetXFADoc()->IsInteractive()) { float fFontSize = m_pNode->GetFontSize(); rtUi.height = std::max(rtUi.height, fFontSize); } GetNormalWidget()->SetWidgetRect(rtUi); } bool CXFA_FFField::OnMouseEnter() { if (!GetNormalWidget()) return false; CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kEnter, Mask(), CFX_PointF()); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnMouseExit() { if (!GetNormalWidget()) return false; CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kLeave, Mask(), CFX_PointF()); SendMessageToFWLWidget(&msg); return true; } CFX_PointF CXFA_FFField::FWLToClient(const CFX_PointF& point) { return GetNormalWidget() ? point - GetNormalWidget()->GetWidgetRect().TopLeft() : point; } bool CXFA_FFField::AcceptsFocusOnButtonDown( Mask dwFlags, const CFX_PointF& point, CFWL_MessageMouse::MouseCommand command) { if (!GetNormalWidget()) return false; if (!m_pNode->IsOpenAccess() || !GetDoc()->GetXFADoc()->IsInteractive()) return false; if (!PtInActiveRect(point)) return false; return true; } bool CXFA_FFField::OnLButtonDown(Mask dwFlags, const CFX_PointF& point) { SetButtonDown(true); CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kLeftButtonDown, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnLButtonUp(Mask dwFlags, const CFX_PointF& point) { if (!GetNormalWidget()) return false; if (!IsButtonDown()) return false; SetButtonDown(false); CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kLeftButtonUp, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnLButtonDblClk(Mask dwFlags, const CFX_PointF& point) { if (!GetNormalWidget()) return false; CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kLeftButtonDblClk, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnMouseMove(Mask dwFlags, const CFX_PointF& point) { if (!GetNormalWidget()) return false; CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kMove, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnMouseWheel(Mask dwFlags, const CFX_PointF& point, const CFX_Vector& delta) { if (!GetNormalWidget()) return false; CFWL_MessageMouseWheel msg(GetNormalWidget(), FWLToClient(point), delta); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnRButtonDown(Mask dwFlags, const CFX_PointF& point) { SetButtonDown(true); CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kRightButtonDown, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnRButtonUp(Mask dwFlags, const CFX_PointF& point) { if (!GetNormalWidget()) return false; if (!IsButtonDown()) return false; SetButtonDown(false); CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kRightButtonUp, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnRButtonDblClk(Mask dwFlags, const CFX_PointF& point) { if (!GetNormalWidget()) return false; CFWL_MessageMouse msg(GetNormalWidget(), CFWL_MessageMouse::MouseCommand::kRightButtonDblClk, dwFlags, FWLToClient(point)); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnSetFocus(CXFA_FFWidget* pOldWidget) { if (!CXFA_FFWidget::OnSetFocus(pOldWidget)) return false; if (!GetNormalWidget()) return false; CFWL_MessageSetFocus msg(GetNormalWidget()); SendMessageToFWLWidget(&msg); GetLayoutItem()->SetStatusBits(XFA_WidgetStatus::kFocused); InvalidateRect(); return true; } bool CXFA_FFField::OnKillFocus(CXFA_FFWidget* pNewWidget) { if (GetNormalWidget()) { CFWL_MessageKillFocus msg(GetNormalWidget()); SendMessageToFWLWidget(&msg); GetLayoutItem()->ClearStatusBits(XFA_WidgetStatus::kFocused); InvalidateRect(); } return pNewWidget && CXFA_FFWidget::OnKillFocus(pNewWidget); } bool CXFA_FFField::OnKeyDown(XFA_FWL_VKEYCODE dwKeyCode, Mask dwFlags) { if (!GetNormalWidget() || !GetDoc()->GetXFADoc()->IsInteractive()) return false; CFWL_MessageKey msg(GetNormalWidget(), CFWL_MessageKey::KeyCommand::kKeyDown, dwFlags, dwKeyCode); SendMessageToFWLWidget(&msg); return true; } bool CXFA_FFField::OnChar(uint32_t dwChar, Mask dwFlags) { if (!GetDoc()->GetXFADoc()->IsInteractive()) return false; if (dwChar == pdfium::ascii::kTab) return true; if (!GetNormalWidget()) return false; if (!m_pNode->IsOpenAccess()) return false; CFWL_MessageKey msg(GetNormalWidget(), CFWL_MessageKey::KeyCommand::kChar, dwFlags, dwChar); SendMessageToFWLWidget(&msg); return true; } FWL_WidgetHit CXFA_FFField::HitTest(const CFX_PointF& point) { auto* pNorm = GetNormalWidget(); if (pNorm && pNorm->HitTest(FWLToClient(point)) != FWL_WidgetHit::Unknown) return FWL_WidgetHit::Client; if (!GetRectWithoutRotate().Contains(point)) return FWL_WidgetHit::Unknown; if (m_CaptionRect.Contains(point)) return FWL_WidgetHit::Titlebar; return FWL_WidgetHit::Border; } bool CXFA_FFField::PtInActiveRect(const CFX_PointF& point) { return GetNormalWidget() && GetNormalWidget()->GetWidgetRect().Contains(point); } void CXFA_FFField::LayoutCaption() { CXFA_TextLayout* pCapTextLayout = m_pNode->GetCaptionTextLayout(); if (!pCapTextLayout) return; float fHeight = pCapTextLayout->Layout(m_CaptionRect.Size()); m_CaptionRect.height = std::max(m_CaptionRect.height, fHeight); } void CXFA_FFField::RenderCaption(CFGAS_GEGraphics* pGS, const CFX_Matrix& pMatrix) { CXFA_TextLayout* pCapTextLayout = m_pNode->GetCaptionTextLayout(); if (!pCapTextLayout) return; CXFA_Caption* caption = m_pNode->GetCaptionIfExists(); if (!caption || !caption->IsVisible()) return; if (!pCapTextLayout->IsLoaded()) pCapTextLayout->Layout(m_CaptionRect.Size()); CFX_RectF rtClip = m_CaptionRect; rtClip.Intersect(GetRectWithoutRotate()); CFX_RenderDevice* pRenderDevice = pGS->GetRenderDevice(); CFX_Matrix mt(1, 0, 0, 1, m_CaptionRect.left, m_CaptionRect.top); rtClip = pMatrix.TransformRect(rtClip); mt.Concat(pMatrix); pCapTextLayout->DrawString(pRenderDevice, mt, rtClip, 0); } bool CXFA_FFField::ProcessCommittedData() { if (!m_pNode->IsOpenAccess()) return false; if (!IsDataChanged()) return false; m_pDocView->SetChangeMark(); m_pDocView->AddValidateNode(m_pNode.Get()); if (CalculateOverride() != 1) return false; return CommitData(); } int32_t CXFA_FFField::CalculateOverride() { CXFA_Node* exclNode = m_pNode->GetExclGroupIfExists(); if (!exclNode || !exclNode->IsWidgetReady()) return CalculateNode(m_pNode.Get()); if (CalculateNode(exclNode) == 0) return 0; CXFA_Node* pNode = exclNode->GetExclGroupFirstMember(); if (!pNode) return 1; while (pNode) { if (!pNode->IsWidgetReady()) return 1; if (CalculateNode(pNode) == 0) return 0; pNode = pNode->GetExclGroupNextMember(pNode); } return 1; } int32_t CXFA_FFField::CalculateNode(CXFA_Node* pNode) { CXFA_Calculate* calc = pNode->GetCalculateIfExists(); if (!calc) return 1; XFA_VERSION version = GetDoc()->GetXFADoc()->GetCurVersionMode(); switch (calc->GetOverride()) { case XFA_AttributeValue::Error: { if (version <= XFA_VERSION_204) return 1; CXFA_FFApp::CallbackIface* pAppProvider = GetAppProvider(); if (pAppProvider) { pAppProvider->MsgBox( WideString::FromASCII("You are not allowed to modify this field."), WideString::FromASCII("Calculate Override"), static_cast(AlertIcon::kWarning), static_cast(AlertButton::kOK)); } return 0; } case XFA_AttributeValue::Warning: { if (version <= XFA_VERSION_204) { CXFA_Script* script = calc->GetScriptIfExists(); if (!script || script->GetExpression().IsEmpty()) return 1; } if (pNode->IsUserInteractive()) return 1; CXFA_FFApp::CallbackIface* pAppProvider = GetAppProvider(); if (!pAppProvider) return 0; WideString wsMessage = calc->GetMessageText(); if (!wsMessage.IsEmpty()) wsMessage += L"\r\n"; wsMessage += WideString::FromASCII("Are you sure you want to modify this field?"); if (pAppProvider->MsgBox(wsMessage, WideString::FromASCII("Calculate Override"), static_cast(AlertIcon::kWarning), static_cast(AlertButton::kYesNo)) == static_cast(AlertReturn::kYes)) { pNode->SetFlag(XFA_NodeFlag::kUserInteractive); return 1; } return 0; } case XFA_AttributeValue::Ignore: return 0; case XFA_AttributeValue::Disabled: pNode->SetFlag(XFA_NodeFlag::kUserInteractive); return 1; default: return 1; } } bool CXFA_FFField::CommitData() { return false; } bool CXFA_FFField::IsDataChanged() { return false; } void CXFA_FFField::SendMessageToFWLWidget(CFWL_Message* pMessage) { DCHECK(pMessage); GetApp()->GetFWLWidgetMgr()->OnProcessMessageToForm(pMessage); } void CXFA_FFField::OnProcessMessage(CFWL_Message* pMessage) {} void CXFA_FFField::OnProcessEvent(CFWL_Event* pEvent) { switch (pEvent->GetType()) { case CFWL_Event::Type::Mouse: { CFWL_EventMouse* event = static_cast(pEvent); CFWL_MessageMouse::MouseCommand cmd = event->GetCommand(); if (cmd == CFWL_MessageMouse::MouseCommand::kEnter) { CXFA_EventParam eParam; eParam.m_eType = XFA_EVENT_MouseEnter; m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::MouseEnter, &eParam); } else if (cmd == CFWL_MessageMouse::MouseCommand::kLeave) { CXFA_EventParam eParam; eParam.m_eType = XFA_EVENT_MouseExit; m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::MouseExit, &eParam); } else if (cmd == CFWL_MessageMouse::MouseCommand::kLeftButtonDown) { CXFA_EventParam eParam; eParam.m_eType = XFA_EVENT_MouseDown; m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::MouseDown, &eParam); } else if (cmd == CFWL_MessageMouse::MouseCommand::kLeftButtonUp) { CXFA_EventParam eParam; eParam.m_eType = XFA_EVENT_MouseUp; m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::MouseUp, &eParam); } break; } case CFWL_Event::Type::Click: { CXFA_EventParam eParam; eParam.m_eType = XFA_EVENT_Click; m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::Click, &eParam); break; } default: break; } } void CXFA_FFField::OnDrawWidget(CFGAS_GEGraphics* pGraphics, const CFX_Matrix& matrix) {}