blob: db71ab9ce5a07ab3e99cec94bf9c13c6f740761f [file] [log] [blame]
// 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 <memory>
#include <utility>
#include <vector>
#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;
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() {
ASSERT(!IsLoaded());
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
auto pNew = std::make_unique<CFWL_ComboBox>(GetFWLApp());
CFWL_ComboBox* pComboBox = pNew.get();
SetNormalWidget(std::move(pNew));
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_Raw));
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()->ModifyStylesEx(dwExtendedStyle, 0xFFFFFFFF);
if (!m_pNode->IsHorizontalScrollPolicyOff())
dwEditStyles |= FWL_STYLEEXT_EDT_AutoHScroll;
pComboBox->EditModifyStylesEx(dwEditStyles, 0xFFFFFFFF);
}
bool CXFA_FFComboBox::OnRButtonUp(uint32_t dwFlags, const CFX_PointF& point) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
if (!CXFA_FFField::OnRButtonUp(dwFlags, point))
return false;
GetDoc()->GetDocEnvironment()->PopupMenu(this, point);
return true;
}
bool CXFA_FFComboBox::OnKillFocus(CXFA_FFWidget* pNewWidget) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
ObservedPtr<CXFA_FFWidget> pNewWatched(pNewWidget);
if (!ProcessCommittedData())
UpdateFWLData();
return pNewWatched && CXFA_FFField::OnKillFocus(pNewWatched.Get());
}
void CXFA_FFComboBox::OpenDropDownList() {
ToComboBox(GetNormalWidget())->OpenDropDownList(true);
}
bool CXFA_FFComboBox::CommitData() {
return m_pNode->SetValue(XFA_VALUEPICTURE_Raw, m_wsNewValue);
}
bool CXFA_FFComboBox::IsDataChanged() {
WideString wsText = GetCurrentText();
if (m_pNode->GetValue(XFA_VALUEPICTURE_Raw) == 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;
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
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_Raw));
}
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() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
return m_pNode->IsChoiceListAllowTextEntry() &&
ToComboBox(GetNormalWidget())->EditUndo();
}
bool CXFA_FFComboBox::Redo() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
return m_pNode->IsChoiceListAllowTextEntry() &&
ToComboBox(GetNormalWidget())->EditRedo();
}
Optional<WideString> CXFA_FFComboBox::Copy() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
return ToComboBox(GetNormalWidget())->EditCopy();
}
Optional<WideString> CXFA_FFComboBox::Cut() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
if (!m_pNode->IsChoiceListAllowTextEntry())
return {};
return ToComboBox(GetNormalWidget())->EditCut();
}
bool CXFA_FFComboBox::Paste(const WideString& wsPaste) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
return m_pNode->IsChoiceListAllowTextEntry() &&
ToComboBox(GetNormalWidget())->EditPaste(wsPaste);
}
void CXFA_FFComboBox::SelectAll() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
ToComboBox(GetNormalWidget())->EditSelectAll();
}
void CXFA_FFComboBox::Delete() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
ToComboBox(GetNormalWidget())->EditDelete();
}
void CXFA_FFComboBox::DeSelect() {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
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) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
CXFA_EventParam eParam;
eParam.m_wsPrevText = m_pNode->GetValue(XFA_VALUEPICTURE_Raw);
eParam.m_wsChange = wsChanged;
FWLEventSelChange(&eParam);
}
void CXFA_FFComboBox::OnSelectChanged(CFWL_Widget* pWidget, bool bLButtonUp) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
CXFA_EventParam eParam;
eParam.m_wsPrevText = m_pNode->GetValue(XFA_VALUEPICTURE_Raw);
FWLEventSelChange(&eParam);
if (m_pNode->IsChoiceListCommitOnSelect() && bLButtonUp)
m_pDocView->SetFocusNode(nullptr);
}
void CXFA_FFComboBox::OnPreOpen(CFWL_Widget* pWidget) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
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) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
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) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
m_pOldDelegate->OnProcessMessage(pMessage);
}
void CXFA_FFComboBox::OnProcessEvent(CFWL_Event* pEvent) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retain_layout(m_pLayoutItem.Get());
CXFA_FFField::OnProcessEvent(pEvent);
switch (pEvent->GetType()) {
case CFWL_Event::Type::SelectChanged: {
auto* postEvent = static_cast<CFWL_EventSelectChanged*>(pEvent);
OnSelectChanged(GetNormalWidget(), postEvent->bLButtonUp);
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(CXFA_Graphics* pGraphics,
const CFX_Matrix& matrix) {
// Prevents destruction of the CXFA_ContentLayoutItem that owns |this|.
RetainPtr<CXFA_ContentLayoutItem> retainer(m_pLayoutItem.Get());
m_pOldDelegate->OnDrawWidget(pGraphics, matrix);
}