|  | // Copyright 2014 PDFium Authors. All rights reserved. | 
|  | // 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_ffcombobox.h" | 
|  |  | 
|  | #include <utility> | 
|  | #include <vector> | 
|  |  | 
|  | #include "third_party/base/check.h" | 
|  | #include "v8/include/cppgc/visitor.h" | 
|  | #include "xfa/fwl/cfwl_combobox.h" | 
|  | #include "xfa/fwl/cfwl_eventselectchanged.h" | 
|  | #include "xfa/fwl/cfwl_notedriver.h" | 
|  | #include "xfa/fxfa/cxfa_eventparam.h" | 
|  | #include "xfa/fxfa/cxfa_ffdocview.h" | 
|  | #include "xfa/fxfa/parser/cxfa_para.h" | 
|  |  | 
|  | namespace { | 
|  |  | 
|  | CFWL_ComboBox* ToComboBox(CFWL_Widget* widget) { | 
|  | return static_cast<CFWL_ComboBox*>(widget); | 
|  | } | 
|  |  | 
|  | const CFWL_ComboBox* ToComboBox(const CFWL_Widget* widget) { | 
|  | return static_cast<const CFWL_ComboBox*>(widget); | 
|  | } | 
|  |  | 
|  | }  // namespace | 
|  |  | 
|  | CXFA_FFComboBox::CXFA_FFComboBox(CXFA_Node* pNode) : CXFA_FFDropDown(pNode) {} | 
|  |  | 
|  | CXFA_FFComboBox::~CXFA_FFComboBox() = default; | 
|  |  | 
|  | void CXFA_FFComboBox::Trace(cppgc::Visitor* visitor) const { | 
|  | CXFA_FFDropDown::Trace(visitor); | 
|  | visitor->Trace(m_pOldDelegate); | 
|  | } | 
|  |  | 
|  | CXFA_FFComboBox* CXFA_FFComboBox::AsComboBox() { | 
|  | return this; | 
|  | } | 
|  |  | 
|  | CFX_RectF CXFA_FFComboBox::GetBBox(FocusOption focus) { | 
|  | if (focus == kDrawFocus) | 
|  | return CFX_RectF(); | 
|  | return CXFA_FFWidget::GetBBox(kDoNotDrawFocus); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::PtInActiveRect(const CFX_PointF& point) { | 
|  | auto* pComboBox = ToComboBox(GetNormalWidget()); | 
|  | return pComboBox && pComboBox->GetBBox().Contains(point); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::LoadWidget() { | 
|  | DCHECK(!IsLoaded()); | 
|  |  | 
|  | CFWL_ComboBox* pComboBox = cppgc::MakeGarbageCollected<CFWL_ComboBox>( | 
|  | GetFWLApp()->GetHeap()->GetAllocationHandle(), GetFWLApp()); | 
|  | SetNormalWidget(pComboBox); | 
|  | pComboBox->SetAdapterIface(this); | 
|  |  | 
|  | CFWL_NoteDriver* pNoteDriver = pComboBox->GetFWLApp()->GetNoteDriver(); | 
|  | pNoteDriver->RegisterEventTarget(pComboBox, pComboBox); | 
|  | m_pOldDelegate = pComboBox->GetDelegate(); | 
|  | pComboBox->SetDelegate(this); | 
|  |  | 
|  | { | 
|  | CFWL_Widget::ScopedUpdateLock update_lock(pComboBox); | 
|  | for (const auto& label : m_pNode->GetChoiceListItems(false)) | 
|  | pComboBox->AddString(label); | 
|  |  | 
|  | std::vector<int32_t> iSelArray = m_pNode->GetSelectedItems(); | 
|  | if (iSelArray.empty()) | 
|  | pComboBox->SetEditText(m_pNode->GetValue(XFA_ValuePicture::kRaw)); | 
|  | else | 
|  | pComboBox->SetCurSel(iSelArray.front()); | 
|  |  | 
|  | UpdateWidgetProperty(); | 
|  | } | 
|  |  | 
|  | return CXFA_FFField::LoadWidget(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::UpdateWidgetProperty() { | 
|  | auto* pComboBox = ToComboBox(GetNormalWidget()); | 
|  | if (!pComboBox) | 
|  | return; | 
|  |  | 
|  | uint32_t dwExtendedStyle = 0; | 
|  | uint32_t dwEditStyles = FWL_STYLEEXT_EDT_ReadOnly; | 
|  | dwExtendedStyle |= UpdateUIProperty(); | 
|  | if (m_pNode->IsChoiceListAllowTextEntry()) { | 
|  | dwEditStyles &= ~FWL_STYLEEXT_EDT_ReadOnly; | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_DropDown; | 
|  | } | 
|  | if (!m_pNode->IsOpenAccess() || !GetDoc()->GetXFADoc()->IsInteractive()) { | 
|  | dwEditStyles |= FWL_STYLEEXT_EDT_ReadOnly; | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_ReadOnly; | 
|  | } | 
|  | dwExtendedStyle |= GetAlignment(); | 
|  | GetNormalWidget()->ModifyStyleExts(dwExtendedStyle, 0xFFFFFFFF); | 
|  |  | 
|  | if (!m_pNode->IsHorizontalScrollPolicyOff()) | 
|  | dwEditStyles |= FWL_STYLEEXT_EDT_AutoHScroll; | 
|  |  | 
|  | pComboBox->EditModifyStyleExts(dwEditStyles, 0xFFFFFFFF); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::OnRButtonUp(Mask<XFA_FWL_KeyFlag> dwFlags, | 
|  | const CFX_PointF& point) { | 
|  | if (!CXFA_FFField::OnRButtonUp(dwFlags, point)) | 
|  | return false; | 
|  |  | 
|  | GetDoc()->PopupMenu(this, point); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::OnKillFocus(CXFA_FFWidget* pNewWidget) { | 
|  | if (!ProcessCommittedData()) | 
|  | UpdateFWLData(); | 
|  |  | 
|  | return pNewWidget && CXFA_FFField::OnKillFocus(pNewWidget); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OpenDropDownList() { | 
|  | ToComboBox(GetNormalWidget())->ShowDropDownList(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CommitData() { | 
|  | return m_pNode->SetValue(XFA_ValuePicture::kRaw, m_wsNewValue); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::IsDataChanged() { | 
|  | WideString wsText = GetCurrentText(); | 
|  | if (m_pNode->GetValue(XFA_ValuePicture::kRaw) == wsText) | 
|  | return false; | 
|  |  | 
|  | m_wsNewValue = std::move(wsText); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::FWLEventSelChange(CXFA_EventParam* pParam) { | 
|  | pParam->m_eType = XFA_EVENT_Change; | 
|  | pParam->m_pTarget = m_pNode.Get(); | 
|  | pParam->m_wsPrevText = ToComboBox(GetNormalWidget())->GetEditText(); | 
|  | m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::Change, pParam); | 
|  | } | 
|  |  | 
|  | WideString CXFA_FFComboBox::GetCurrentText() const { | 
|  | auto* pFWLcombobox = ToComboBox(GetNormalWidget()); | 
|  | WideString wsText = pFWLcombobox->GetEditText(); | 
|  | int32_t iCursel = pFWLcombobox->GetCurSel(); | 
|  | if (iCursel >= 0) { | 
|  | WideString wsSel = pFWLcombobox->GetTextByIndex(iCursel); | 
|  | if (wsSel == wsText) | 
|  | wsText = m_pNode->GetChoiceListItem(iCursel, true).value_or(L""); | 
|  | } | 
|  | return wsText; | 
|  | } | 
|  |  | 
|  | uint32_t CXFA_FFComboBox::GetAlignment() { | 
|  | CXFA_Para* para = m_pNode->GetParaIfExists(); | 
|  | if (!para) | 
|  | return 0; | 
|  |  | 
|  | uint32_t dwExtendedStyle = 0; | 
|  | switch (para->GetHorizontalAlign()) { | 
|  | case XFA_AttributeValue::Center: | 
|  | dwExtendedStyle |= | 
|  | FWL_STYLEEXT_CMB_EditHCenter | FWL_STYLEEXT_CMB_ListItemCenterAlign; | 
|  | break; | 
|  | case XFA_AttributeValue::Justify: | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_EditJustified; | 
|  | break; | 
|  | case XFA_AttributeValue::JustifyAll: | 
|  | break; | 
|  | case XFA_AttributeValue::Radix: | 
|  | break; | 
|  | case XFA_AttributeValue::Right: | 
|  | break; | 
|  | default: | 
|  | dwExtendedStyle |= | 
|  | FWL_STYLEEXT_CMB_EditHNear | FWL_STYLEEXT_CMB_ListItemLeftAlign; | 
|  | break; | 
|  | } | 
|  |  | 
|  | switch (para->GetVerticalAlign()) { | 
|  | case XFA_AttributeValue::Middle: | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_EditVCenter; | 
|  | break; | 
|  | case XFA_AttributeValue::Bottom: | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_EditVFar; | 
|  | break; | 
|  | default: | 
|  | dwExtendedStyle |= FWL_STYLEEXT_CMB_EditVNear; | 
|  | break; | 
|  | } | 
|  | return dwExtendedStyle; | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::UpdateFWLData() { | 
|  | auto* pComboBox = ToComboBox(GetNormalWidget()); | 
|  | if (!pComboBox) | 
|  | return false; | 
|  |  | 
|  | std::vector<int32_t> iSelArray = m_pNode->GetSelectedItems(); | 
|  | if (!iSelArray.empty()) { | 
|  | pComboBox->SetCurSel(iSelArray.front()); | 
|  | } else { | 
|  | pComboBox->SetCurSel(-1); | 
|  | pComboBox->SetEditText(m_pNode->GetValue(XFA_ValuePicture::kRaw)); | 
|  | } | 
|  | pComboBox->Update(); | 
|  | return true; | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanUndo() { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditCanUndo(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanRedo() { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditCanRedo(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanCopy() { | 
|  | return ToComboBox(GetNormalWidget())->EditCanCopy(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanCut() { | 
|  | return m_pNode->IsOpenAccess() && m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditCanCut(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanPaste() { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && m_pNode->IsOpenAccess(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::CanSelectAll() { | 
|  | return ToComboBox(GetNormalWidget())->EditCanSelectAll(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::Undo() { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditUndo(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::Redo() { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditRedo(); | 
|  | } | 
|  |  | 
|  | absl::optional<WideString> CXFA_FFComboBox::Copy() { | 
|  | return ToComboBox(GetNormalWidget())->EditCopy(); | 
|  | } | 
|  |  | 
|  | absl::optional<WideString> CXFA_FFComboBox::Cut() { | 
|  | if (!m_pNode->IsChoiceListAllowTextEntry()) | 
|  | return absl::nullopt; | 
|  |  | 
|  | return ToComboBox(GetNormalWidget())->EditCut(); | 
|  | } | 
|  |  | 
|  | bool CXFA_FFComboBox::Paste(const WideString& wsPaste) { | 
|  | return m_pNode->IsChoiceListAllowTextEntry() && | 
|  | ToComboBox(GetNormalWidget())->EditPaste(wsPaste); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::SelectAll() { | 
|  | ToComboBox(GetNormalWidget())->EditSelectAll(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::Delete() { | 
|  | ToComboBox(GetNormalWidget())->EditDelete(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::DeSelect() { | 
|  | ToComboBox(GetNormalWidget())->EditDeSelect(); | 
|  | } | 
|  |  | 
|  | WideString CXFA_FFComboBox::GetText() { | 
|  | return GetCurrentText(); | 
|  | } | 
|  |  | 
|  | FormFieldType CXFA_FFComboBox::GetFormFieldType() { | 
|  | return FormFieldType::kXFA_ComboBox; | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::SetItemState(int32_t nIndex, bool bSelected) { | 
|  | ToComboBox(GetNormalWidget())->SetCurSel(bSelected ? nIndex : -1); | 
|  | GetNormalWidget()->Update(); | 
|  | InvalidateRect(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::InsertItem(const WideString& wsLabel, int32_t nIndex) { | 
|  | ToComboBox(GetNormalWidget())->AddString(wsLabel); | 
|  | GetNormalWidget()->Update(); | 
|  | InvalidateRect(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::DeleteItem(int32_t nIndex) { | 
|  | if (nIndex < 0) | 
|  | ToComboBox(GetNormalWidget())->RemoveAll(); | 
|  | else | 
|  | ToComboBox(GetNormalWidget())->RemoveAt(nIndex); | 
|  |  | 
|  | GetNormalWidget()->Update(); | 
|  | InvalidateRect(); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnTextChanged(CFWL_Widget* pWidget, | 
|  | const WideString& wsChanged) { | 
|  | CXFA_EventParam eParam; | 
|  | eParam.m_wsPrevText = m_pNode->GetValue(XFA_ValuePicture::kRaw); | 
|  | eParam.m_wsChange = wsChanged; | 
|  | FWLEventSelChange(&eParam); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnSelectChanged(CFWL_Widget* pWidget, bool bLButtonUp) { | 
|  | CXFA_EventParam eParam; | 
|  | eParam.m_wsPrevText = m_pNode->GetValue(XFA_ValuePicture::kRaw); | 
|  | FWLEventSelChange(&eParam); | 
|  | if (m_pNode->IsChoiceListCommitOnSelect() && bLButtonUp) | 
|  | m_pDocView->SetFocusNode(nullptr); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnPreOpen(CFWL_Widget* pWidget) { | 
|  | CXFA_EventParam eParam; | 
|  | eParam.m_eType = XFA_EVENT_PreOpen; | 
|  | eParam.m_pTarget = m_pNode.Get(); | 
|  | m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::PreOpen, &eParam); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnPostOpen(CFWL_Widget* pWidget) { | 
|  | CXFA_EventParam eParam; | 
|  | eParam.m_eType = XFA_EVENT_PostOpen; | 
|  | eParam.m_pTarget = m_pNode.Get(); | 
|  | m_pNode->ProcessEvent(GetDocView(), XFA_AttributeValue::PostOpen, &eParam); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnProcessMessage(CFWL_Message* pMessage) { | 
|  | m_pOldDelegate->OnProcessMessage(pMessage); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnProcessEvent(CFWL_Event* pEvent) { | 
|  | CXFA_FFField::OnProcessEvent(pEvent); | 
|  | switch (pEvent->GetType()) { | 
|  | case CFWL_Event::Type::SelectChanged: { | 
|  | auto* postEvent = static_cast<CFWL_EventSelectChanged*>(pEvent); | 
|  | OnSelectChanged(GetNormalWidget(), postEvent->GetLButtonUp()); | 
|  | break; | 
|  | } | 
|  | case CFWL_Event::Type::EditChanged: { | 
|  | WideString wsChanged; | 
|  | OnTextChanged(GetNormalWidget(), wsChanged); | 
|  | break; | 
|  | } | 
|  | case CFWL_Event::Type::PreDropDown: { | 
|  | OnPreOpen(GetNormalWidget()); | 
|  | break; | 
|  | } | 
|  | case CFWL_Event::Type::PostDropDown: { | 
|  | OnPostOpen(GetNormalWidget()); | 
|  | break; | 
|  | } | 
|  | default: | 
|  | break; | 
|  | } | 
|  | m_pOldDelegate->OnProcessEvent(pEvent); | 
|  | } | 
|  |  | 
|  | void CXFA_FFComboBox::OnDrawWidget(CFGAS_GEGraphics* pGraphics, | 
|  | const CFX_Matrix& matrix) { | 
|  | m_pOldDelegate->OnDrawWidget(pGraphics, matrix); | 
|  | } |