blob: e87ef42f5a9d3c6932bf1e706dba669edcf767d3 [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/fxfa/cxfa_ffnotify.h"
#include <utility>
#include "core/fxcrt/check.h"
#include "xfa/fxfa/cxfa_ffapp.h"
#include "xfa/fxfa/cxfa_ffarc.h"
#include "xfa/fxfa/cxfa_ffbarcode.h"
#include "xfa/fxfa/cxfa_ffcheckbutton.h"
#include "xfa/fxfa/cxfa_ffcombobox.h"
#include "xfa/fxfa/cxfa_ffdatetimeedit.h"
#include "xfa/fxfa/cxfa_ffdoc.h"
#include "xfa/fxfa/cxfa_ffdocview.h"
#include "xfa/fxfa/cxfa_ffexclgroup.h"
#include "xfa/fxfa/cxfa_fffield.h"
#include "xfa/fxfa/cxfa_ffimage.h"
#include "xfa/fxfa/cxfa_ffimageedit.h"
#include "xfa/fxfa/cxfa_ffline.h"
#include "xfa/fxfa/cxfa_fflistbox.h"
#include "xfa/fxfa/cxfa_ffnumericedit.h"
#include "xfa/fxfa/cxfa_ffpageview.h"
#include "xfa/fxfa/cxfa_ffpasswordedit.h"
#include "xfa/fxfa/cxfa_ffpushbutton.h"
#include "xfa/fxfa/cxfa_ffrectangle.h"
#include "xfa/fxfa/cxfa_ffsignature.h"
#include "xfa/fxfa/cxfa_fftext.h"
#include "xfa/fxfa/cxfa_ffwidget.h"
#include "xfa/fxfa/cxfa_ffwidgethandler.h"
#include "xfa/fxfa/cxfa_fwladapterwidgetmgr.h"
#include "xfa/fxfa/cxfa_textlayout.h"
#include "xfa/fxfa/cxfa_textprovider.h"
#include "xfa/fxfa/layout/cxfa_layoutprocessor.h"
#include "xfa/fxfa/parser/cxfa_barcode.h"
#include "xfa/fxfa/parser/cxfa_binditems.h"
#include "xfa/fxfa/parser/cxfa_button.h"
#include "xfa/fxfa/parser/cxfa_checkbutton.h"
#include "xfa/fxfa/parser/cxfa_node.h"
#include "xfa/fxfa/parser/cxfa_passwordedit.h"
CXFA_FFNotify::CXFA_FFNotify(CXFA_FFDoc* pDoc) : m_pDoc(pDoc) {}
CXFA_FFNotify::~CXFA_FFNotify() = default;
void CXFA_FFNotify::Trace(cppgc::Visitor* visitor) const {
visitor->Trace(m_pDoc);
}
void CXFA_FFNotify::OnPageViewEvent(CXFA_ViewLayoutItem* pSender,
CXFA_FFDoc::PageViewEvent eEvent) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView(pSender->GetLayout());
if (pDocView)
pDocView->OnPageViewEvent(pSender, eEvent);
}
void CXFA_FFNotify::OnWidgetListItemAdded(CXFA_Node* pSender,
const WideString& wsLabel,
int32_t iIndex) {
if (pSender->GetFFWidgetType() != XFA_FFWidgetType::kChoiceList)
return;
CXFA_FFWidget* pWidget = m_pDoc->GetDocView()->GetWidgetForNode(pSender);
for (; pWidget; pWidget = pWidget->GetNextFFWidget()) {
if (pWidget->IsLoaded())
ToDropDown(ToField(pWidget))->InsertItem(wsLabel, iIndex);
}
}
void CXFA_FFNotify::OnWidgetListItemRemoved(CXFA_Node* pSender,
int32_t iIndex) {
if (pSender->GetFFWidgetType() != XFA_FFWidgetType::kChoiceList)
return;
CXFA_FFWidget* pWidget = m_pDoc->GetDocView()->GetWidgetForNode(pSender);
for (; pWidget; pWidget = pWidget->GetNextFFWidget()) {
if (pWidget->IsLoaded())
ToDropDown(ToField(pWidget))->DeleteItem(iIndex);
}
}
CXFA_FFPageView* CXFA_FFNotify::OnCreateViewLayoutItem(CXFA_Node* pNode) {
if (pNode->GetElementType() != XFA_Element::PageArea)
return nullptr;
auto* pLayout = CXFA_LayoutProcessor::FromDocument(m_pDoc->GetXFADoc());
return cppgc::MakeGarbageCollected<CXFA_FFPageView>(
m_pDoc->GetHeap()->GetAllocationHandle(), m_pDoc->GetDocView(pLayout),
pNode);
}
CXFA_FFWidget* CXFA_FFNotify::OnCreateContentLayoutItem(CXFA_Node* pNode) {
DCHECK(pNode->GetElementType() != XFA_Element::ContentArea);
DCHECK(pNode->GetElementType() != XFA_Element::PageArea);
// We only need to create the widget for certain types of objects.
if (!pNode->HasCreatedUIWidget())
return nullptr;
CXFA_FFWidget* pWidget = nullptr;
switch (pNode->GetFFWidgetType()) {
case XFA_FFWidgetType::kBarcode: {
auto* child = CXFA_Barcode::FromNode(pNode->GetUIChildNode());
if (!child)
return nullptr;
pWidget = cppgc::MakeGarbageCollected<CXFA_FFBarcode>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode, child);
break;
}
case XFA_FFWidgetType::kButton: {
auto* child = CXFA_Button::FromNode(pNode->GetUIChildNode());
if (!child)
return nullptr;
pWidget = cppgc::MakeGarbageCollected<CXFA_FFPushButton>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode, child);
break;
}
case XFA_FFWidgetType::kCheckButton: {
auto* child = CXFA_CheckButton::FromNode(pNode->GetUIChildNode());
if (!child)
return nullptr;
pWidget = cppgc::MakeGarbageCollected<CXFA_FFCheckButton>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode, child);
break;
}
case XFA_FFWidgetType::kChoiceList: {
if (pNode->IsListBox()) {
pWidget = cppgc::MakeGarbageCollected<CXFA_FFListBox>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
} else {
pWidget = cppgc::MakeGarbageCollected<CXFA_FFComboBox>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
}
break;
}
case XFA_FFWidgetType::kDateTimeEdit:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFDateTimeEdit>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kImageEdit:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFImageEdit>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kNumericEdit:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFNumericEdit>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kPasswordEdit: {
auto* child = CXFA_PasswordEdit::FromNode(pNode->GetUIChildNode());
if (!child)
return nullptr;
pWidget = cppgc::MakeGarbageCollected<CXFA_FFPasswordEdit>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode, child);
break;
}
case XFA_FFWidgetType::kSignature:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFSignature>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kTextEdit:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFTextEdit>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kArc:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFArc>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kLine:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFLine>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kRectangle:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFRectangle>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kText:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFText>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kImage:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFImage>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kSubform:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFWidget>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kExclGroup:
pWidget = cppgc::MakeGarbageCollected<CXFA_FFExclGroup>(
m_pDoc->GetHeap()->GetAllocationHandle(), pNode);
break;
case XFA_FFWidgetType::kNone:
return nullptr;
}
auto* pLayout = CXFA_LayoutProcessor::FromDocument(m_pDoc->GetXFADoc());
pWidget->SetDocView(m_pDoc->GetDocView(pLayout));
return pWidget;
}
void CXFA_FFNotify::StartFieldDrawLayout(CXFA_Node* pItem,
float* pCalcWidth,
float* pCalcHeight) {
pItem->StartWidgetLayout(m_pDoc.Get(), pCalcWidth, pCalcHeight);
}
bool CXFA_FFNotify::RunScript(CXFA_Script* script, CXFA_Node* item) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return false;
CXFA_EventParam EventParam(XFA_EVENT_Unknown);
EventParam.m_bTargeted = false;
CXFA_Node::BoolScriptResult result =
item->ExecuteBoolScript(pDocView, script, &EventParam);
return result.xfa_event_result == XFA_EventError::kSuccess &&
result.script_result;
}
XFA_EventError CXFA_FFNotify::ExecEventByDeepFirst(CXFA_Node* pFormNode,
XFA_EVENTTYPE eEventType,
bool bIsFormReady,
bool bRecursive) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return XFA_EventError::kNotExist;
return pDocView->ExecEventActivityByDeepFirst(pFormNode, eEventType,
bIsFormReady, bRecursive);
}
void CXFA_FFNotify::AddCalcValidate(CXFA_Node* pNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
pDocView->AddCalculateNode(pNode);
pDocView->AddValidateNode(pNode);
}
CXFA_FFApp::CallbackIface* CXFA_FFNotify::GetAppProvider() {
return m_pDoc->GetApp()->GetAppProvider();
}
void CXFA_FFNotify::HandleWidgetEvent(CXFA_Node* pNode,
CXFA_EventParam* pParam) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
CXFA_FFWidgetHandler* pHandler = pDocView->GetWidgetHandler();
if (!pHandler)
return;
pHandler->ProcessEvent(pNode, pParam);
}
void CXFA_FFNotify::OpenDropDownList(CXFA_Node* pNode) {
auto* pDocLayout = CXFA_LayoutProcessor::FromDocument(m_pDoc->GetXFADoc());
CXFA_LayoutItem* pLayoutItem = pDocLayout->GetLayoutItem(pNode);
if (!pLayoutItem)
return;
CXFA_FFWidget* hWidget = CXFA_FFWidget::FromLayoutItem(pLayoutItem);
if (!hWidget)
return;
CXFA_FFDoc* hDoc = GetFFDoc();
hDoc->SetFocusWidget(hWidget);
if (hWidget->GetNode()->GetFFWidgetType() != XFA_FFWidgetType::kChoiceList)
return;
if (!hWidget->IsLoaded())
return;
CXFA_FFDropDown* pDropDown = ToDropDown(ToField(hWidget));
CXFA_FFComboBox* pComboBox = ToComboBox(pDropDown);
if (!pComboBox)
return;
CXFA_FFDocView::UpdateScope scope(m_pDoc->GetDocView());
pComboBox->OpenDropDownList();
}
void CXFA_FFNotify::ResetData(CXFA_Node* pNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
pDocView->ResetNode(pNode);
}
CXFA_FFDocView::LayoutStatus CXFA_FFNotify::GetLayoutStatus() {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
return pDocView ? pDocView->GetLayoutStatus()
: CXFA_FFDocView::LayoutStatus::kNone;
}
void CXFA_FFNotify::RunNodeInitialize(CXFA_Node* pNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
pDocView->AddNewFormNode(pNode);
}
void CXFA_FFNotify::RunSubformIndexChange(CXFA_Subform* pSubformNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
pDocView->AddIndexChangedSubform(pSubformNode);
}
CXFA_Node* CXFA_FFNotify::GetFocusWidgetNode() {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
return pDocView ? pDocView->GetFocusNode() : nullptr;
}
void CXFA_FFNotify::SetFocusWidgetNode(CXFA_Node* pNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
pDocView->SetFocusNode(pNode);
}
void CXFA_FFNotify::OnNodeReady(CXFA_Node* pNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
if (pNode->HasCreatedUIWidget()) {
pNode->SetWidgetReady();
return;
}
switch (pNode->GetElementType()) {
case XFA_Element::BindItems:
pDocView->AddBindItem(static_cast<CXFA_BindItems*>(pNode));
break;
case XFA_Element::Validate:
pNode->SetFlag(XFA_NodeFlag::kNeedsInitApp);
break;
default:
break;
}
}
void CXFA_FFNotify::OnValueChanging(CXFA_Node* pSender, XFA_Attribute eAttr) {
if (eAttr != XFA_Attribute::Presence)
return;
if (pSender->GetPacketType() == XFA_PacketType::Datasets)
return;
if (!pSender->IsFormContainer())
return;
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
if (pDocView->GetLayoutStatus() != CXFA_FFDocView::LayoutStatus::kEnd)
return;
CXFA_FFWidget* pWidget = m_pDoc->GetDocView()->GetWidgetForNode(pSender);
for (; pWidget; pWidget = pWidget->GetNextFFWidget()) {
if (pWidget->IsLoaded())
pWidget->InvalidateRect();
}
}
void CXFA_FFNotify::OnValueChanged(CXFA_Node* pSender,
XFA_Attribute eAttr,
CXFA_Node* pParentNode,
CXFA_Node* pWidgetNode) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
if (pSender->GetPacketType() != XFA_PacketType::Form) {
if (eAttr == XFA_Attribute::Value)
pDocView->AddCalculateNodeNotify(pSender);
return;
}
XFA_Element eType = pParentNode->GetElementType();
bool bIsContainerNode = pParentNode->IsContainerNode();
bool bUpdateProperty = false;
pDocView->SetChangeMark();
switch (eType) {
case XFA_Element::Caption: {
CXFA_TextLayout* pCapOut = pWidgetNode->GetCaptionTextLayout();
if (!pCapOut)
return;
pCapOut->Unload();
break;
}
case XFA_Element::Ui:
case XFA_Element::Para:
bUpdateProperty = true;
break;
default:
break;
}
if (bIsContainerNode && eAttr == XFA_Attribute::Access)
bUpdateProperty = true;
if (eAttr == XFA_Attribute::Value) {
pDocView->AddCalculateNodeNotify(pSender);
if (eType == XFA_Element::Value || bIsContainerNode) {
if (bIsContainerNode) {
m_pDoc->GetDocView()->UpdateUIDisplay(pWidgetNode, nullptr);
pDocView->AddCalculateNode(pWidgetNode);
pDocView->AddValidateNode(pWidgetNode);
} else if (pWidgetNode->GetParent()->GetElementType() ==
XFA_Element::ExclGroup) {
m_pDoc->GetDocView()->UpdateUIDisplay(pWidgetNode, nullptr);
}
return;
}
}
CXFA_FFWidget* pWidget = m_pDoc->GetDocView()->GetWidgetForNode(pWidgetNode);
for (; pWidget; pWidget = pWidget->GetNextFFWidget()) {
if (!pWidget->IsLoaded())
continue;
if (bUpdateProperty)
pWidget->UpdateWidgetProperty();
pWidget->PerformLayout();
pWidget->InvalidateRect();
}
}
void CXFA_FFNotify::OnContainerChanged() {
m_pDoc->GetXFADoc()->GetLayoutProcessor()->SetHasChangedContainer();
}
void CXFA_FFNotify::OnChildAdded(CXFA_Node* pSender) {
if (!pSender->IsFormContainer())
return;
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
const bool bLayoutReady =
!pDocView->InLayoutStatus() &&
pDocView->GetLayoutStatus() == CXFA_FFDocView::LayoutStatus::kEnd;
if (bLayoutReady)
m_pDoc->SetChangeMark();
}
void CXFA_FFNotify::OnChildRemoved() {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView();
if (!pDocView)
return;
const bool bLayoutReady =
!pDocView->InLayoutStatus() &&
pDocView->GetLayoutStatus() == CXFA_FFDocView::LayoutStatus::kEnd;
if (bLayoutReady)
m_pDoc->SetChangeMark();
}
void CXFA_FFNotify::OnLayoutItemAdded(CXFA_LayoutProcessor* pLayout,
CXFA_LayoutItem* pSender,
int32_t iPageIdx,
Mask<XFA_WidgetStatus> dwStatus) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView(pLayout);
if (!pDocView)
return;
CXFA_FFWidget* pWidget = CXFA_FFWidget::FromLayoutItem(pSender);
if (!pWidget)
return;
CXFA_FFPageView* pNewPageView = pDocView->GetPageView(iPageIdx);
constexpr Mask<XFA_WidgetStatus> kRemove{XFA_WidgetStatus::kVisible,
XFA_WidgetStatus::kViewable,
XFA_WidgetStatus::kPrintable};
pWidget->ModifyStatus(dwStatus, kRemove);
CXFA_FFPageView* pPrePageView = pWidget->GetPageView();
if (pPrePageView != pNewPageView ||
dwStatus.TestAll(
{XFA_WidgetStatus::kVisible, XFA_WidgetStatus::kViewable})) {
pWidget->SetPageView(pNewPageView);
m_pDoc->WidgetPostAdd(pWidget);
}
if (pDocView->GetLayoutStatus() != CXFA_FFDocView::LayoutStatus::kEnd ||
!(dwStatus & XFA_WidgetStatus::kVisible)) {
return;
}
if (pWidget->IsLoaded()) {
if (pWidget->GetWidgetRect() != pWidget->RecacheWidgetRect())
pWidget->PerformLayout();
} else {
pWidget->LoadWidget();
}
pWidget->InvalidateRect();
}
void CXFA_FFNotify::OnLayoutItemRemoving(CXFA_LayoutProcessor* pLayout,
CXFA_LayoutItem* pSender) {
CXFA_FFDocView* pDocView = m_pDoc->GetDocView(pLayout);
if (!pDocView)
return;
CXFA_FFWidget* pWidget = CXFA_FFWidget::FromLayoutItem(pSender);
if (!pWidget)
return;
pDocView->DeleteLayoutItem(pWidget);
m_pDoc->WidgetPreRemove(pWidget);
pWidget->InvalidateRect();
}