blob: 40e800f55fffe3b64f183c24d76a05c6572cf583 [file] [log] [blame]
// 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_combobox.h"
#include "v8/include/cppgc/visitor.h"
#include "xfa/fde/cfde_texteditengine.h"
#include "xfa/fde/cfde_textout.h"
#include "xfa/fwl/cfwl_app.h"
#include "xfa/fwl/cfwl_event.h"
#include "xfa/fwl/cfwl_eventselectchanged.h"
#include "xfa/fwl/cfwl_listbox.h"
#include "xfa/fwl/cfwl_messagekey.h"
#include "xfa/fwl/cfwl_messagekillfocus.h"
#include "xfa/fwl/cfwl_messagemouse.h"
#include "xfa/fwl/cfwl_messagesetfocus.h"
#include "xfa/fwl/cfwl_notedriver.h"
#include "xfa/fwl/cfwl_themebackground.h"
#include "xfa/fwl/cfwl_themepart.h"
#include "xfa/fwl/cfwl_themetext.h"
#include "xfa/fwl/cfwl_widgetmgr.h"
#include "xfa/fwl/fwl_widgetdef.h"
#include "xfa/fwl/ifwl_themeprovider.h"
namespace pdfium {
CFWL_ComboBox::CFWL_ComboBox(CFWL_App* app)
: CFWL_Widget(app, Properties(), nullptr),
edit_(cppgc::MakeGarbageCollected<CFWL_ComboEdit>(
app->GetHeap()->GetAllocationHandle(),
app,
Properties(),
this)),
list_box_(cppgc::MakeGarbageCollected<CFWL_ComboList>(
app->GetHeap()->GetAllocationHandle(),
app,
Properties{FWL_STYLE_WGT_Border | FWL_STYLE_WGT_VScroll, 0,
FWL_STATE_WGT_Invisible},
this)) {}
CFWL_ComboBox::~CFWL_ComboBox() = default;
void CFWL_ComboBox::Trace(cppgc::Visitor* visitor) const {
CFWL_Widget::Trace(visitor);
visitor->Trace(edit_);
visitor->Trace(list_box_);
}
FWL_Type CFWL_ComboBox::GetClassID() const {
return FWL_Type::ComboBox;
}
void CFWL_ComboBox::AddString(const WideString& wsText) {
list_box_->AddString(wsText);
}
void CFWL_ComboBox::RemoveAt(int32_t iIndex) {
list_box_->RemoveAt(iIndex);
}
void CFWL_ComboBox::RemoveAll() {
list_box_->DeleteAll();
}
void CFWL_ComboBox::ModifyStyleExts(uint32_t dwStyleExtsAdded,
uint32_t dwStyleExtsRemoved) {
bool bAddDropDown = !!(dwStyleExtsAdded & FWL_STYLEEXT_CMB_DropDown);
bool bDelDropDown = !!(dwStyleExtsRemoved & FWL_STYLEEXT_CMB_DropDown);
dwStyleExtsRemoved &= ~FWL_STYLEEXT_CMB_DropDown;
properties_.style_exts_ |= FWL_STYLEEXT_CMB_DropDown;
if (bAddDropDown)
edit_->ModifyStyleExts(0, FWL_STYLEEXT_EDT_ReadOnly);
else if (bDelDropDown)
edit_->ModifyStyleExts(FWL_STYLEEXT_EDT_ReadOnly, 0);
CFWL_Widget::ModifyStyleExts(dwStyleExtsAdded, dwStyleExtsRemoved);
}
void CFWL_ComboBox::Update() {
if (IsLocked())
return;
if (edit_) {
ResetEditAlignment();
}
Layout();
}
FWL_WidgetHit CFWL_ComboBox::HitTest(const CFX_PointF& point) {
CFX_RectF rect(0, 0, widget_rect_.width - btn_rect_.width,
widget_rect_.height);
if (rect.Contains(point))
return FWL_WidgetHit::Edit;
if (btn_rect_.Contains(point)) {
return FWL_WidgetHit::Client;
}
if (IsDropListVisible()) {
rect = list_box_->GetWidgetRect();
if (rect.Contains(point))
return FWL_WidgetHit::Client;
}
return FWL_WidgetHit::Unknown;
}
void CFWL_ComboBox::DrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
if (!btn_rect_.IsEmpty(0.1f)) {
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
pGraphics->ConcatMatrix(matrix);
CFWL_ThemeBackground param(CFWL_ThemePart::Part::kDropDownButton, this,
pGraphics);
param.states_ = btn_state_;
param.part_rect_ = btn_rect_;
GetThemeProvider()->DrawBackground(param);
}
if (edit_) {
CFX_RectF rtEdit = edit_->GetWidgetRect();
CFX_Matrix mt(1, 0, 0, 1, rtEdit.left, rtEdit.top);
mt.Concat(matrix);
edit_->DrawWidget(pGraphics, mt);
}
if (list_box_ && IsDropListVisible()) {
CFX_RectF rtList = list_box_->GetWidgetRect();
CFX_Matrix mt(1, 0, 0, 1, rtList.left, rtList.top);
mt.Concat(matrix);
list_box_->DrawWidget(pGraphics, mt);
}
}
WideString CFWL_ComboBox::GetTextByIndex(int32_t iIndex) const {
CFWL_ListBox::Item* pItem = list_box_->GetItem(list_box_, iIndex);
return pItem ? pItem->GetText() : WideString();
}
void CFWL_ComboBox::SetCurSel(int32_t iSel) {
int32_t iCount = list_box_->CountItems(nullptr);
bool bClearSel = iSel < 0 || iSel >= iCount;
if (IsDropDownStyle() && edit_) {
if (bClearSel) {
edit_->SetText(WideString());
} else {
CFWL_ListBox::Item* hItem = list_box_->GetItem(this, iSel);
edit_->SetText(hItem ? hItem->GetText() : WideString());
}
edit_->Update();
}
cur_sel_ = bClearSel ? -1 : iSel;
}
void CFWL_ComboBox::SetStates(uint32_t dwStates) {
if (IsDropDownStyle() && edit_) {
edit_->SetStates(dwStates);
}
if (list_box_) {
list_box_->SetStates(dwStates);
}
CFWL_Widget::SetStates(dwStates);
}
void CFWL_ComboBox::RemoveStates(uint32_t dwStates) {
if (IsDropDownStyle() && edit_) {
edit_->RemoveStates(dwStates);
}
if (list_box_) {
list_box_->RemoveStates(dwStates);
}
CFWL_Widget::RemoveStates(dwStates);
}
void CFWL_ComboBox::SetEditText(const WideString& wsText) {
if (!edit_) {
return;
}
edit_->SetText(wsText);
edit_->Update();
}
WideString CFWL_ComboBox::GetEditText() const {
if (edit_) {
return edit_->GetText();
}
if (!list_box_) {
return WideString();
}
CFWL_ListBox::Item* hItem = list_box_->GetItem(this, cur_sel_);
return hItem ? hItem->GetText() : WideString();
}
CFX_RectF CFWL_ComboBox::GetBBox() const {
CFX_RectF rect = widget_rect_;
if (!list_box_ || !IsDropListVisible()) {
return rect;
}
CFX_RectF rtList = list_box_->GetWidgetRect();
rtList.Offset(rect.left, rect.top);
rect.Union(rtList);
return rect;
}
void CFWL_ComboBox::EditModifyStyleExts(uint32_t dwStyleExtsAdded,
uint32_t dwStyleExtsRemoved) {
if (edit_) {
edit_->ModifyStyleExts(dwStyleExtsAdded, dwStyleExtsRemoved);
}
}
void CFWL_ComboBox::ShowDropDownList() {
if (IsDropListVisible())
return;
CFWL_Event preEvent(CFWL_Event::Type::PreDropDown, this);
DispatchEvent(&preEvent);
if (!preEvent.GetSrcTarget())
return;
CFWL_ComboList* pComboList = list_box_;
int32_t iItems = pComboList->CountItems(nullptr);
if (iItems < 1)
return;
ResetListItemAlignment();
pComboList->ChangeSelected(cur_sel_);
float fItemHeight = pComboList->CalcItemHeight();
float fBorder = GetCXBorderSize();
float fPopupMin = 0.0f;
if (iItems > 3)
fPopupMin = fItemHeight * 3 + fBorder * 2;
float fPopupMax = fItemHeight * iItems + fBorder * 2;
CFX_RectF rtList(client_rect_.left, 0, widget_rect_.width, 0);
GetPopupPos(fPopupMin, fPopupMax, widget_rect_, &rtList);
list_box_->SetWidgetRect(rtList);
list_box_->Update();
list_box_->RemoveStates(FWL_STATE_WGT_Invisible);
CFWL_Event postEvent(CFWL_Event::Type::PostDropDown, this);
DispatchEvent(&postEvent);
RepaintInflatedListBoxRect();
}
void CFWL_ComboBox::HideDropDownList() {
if (!IsDropListVisible())
return;
list_box_->SetStates(FWL_STATE_WGT_Invisible);
RepaintInflatedListBoxRect();
}
void CFWL_ComboBox::RepaintInflatedListBoxRect() {
CFX_RectF rect = list_box_->GetWidgetRect();
rect.Inflate(2, 2);
RepaintRect(rect);
}
void CFWL_ComboBox::MatchEditText() {
WideString wsText = edit_->GetText();
int32_t iMatch = list_box_->MatchItem(wsText.AsStringView());
if (iMatch != cur_sel_) {
list_box_->ChangeSelected(iMatch);
if (iMatch >= 0)
SyncEditText(iMatch);
} else if (iMatch >= 0) {
edit_->SetSelected();
}
cur_sel_ = iMatch;
}
void CFWL_ComboBox::SyncEditText(int32_t iListItem) {
CFWL_ListBox::Item* hItem = list_box_->GetItem(this, iListItem);
edit_->SetText(hItem ? hItem->GetText() : WideString());
edit_->Update();
edit_->SetSelected();
}
void CFWL_ComboBox::Layout() {
client_rect_ = GetClientRect();
content_rect_ = client_rect_;
IFWL_ThemeProvider* theme = GetThemeProvider();
float borderWidth = 1;
float fBtn = theme->GetScrollBarWidth();
if (!(GetStyleExts() & FWL_STYLEEXT_CMB_ReadOnly)) {
btn_rect_ =
CFX_RectF(client_rect_.right() - fBtn, client_rect_.top + borderWidth,
fBtn - borderWidth, client_rect_.height - 2 * borderWidth);
}
CFWL_ThemePart part(CFWL_ThemePart::Part::kNone, this);
CFX_RectF pUIMargin = theme->GetUIMargin(part);
content_rect_.Deflate(pUIMargin.left, pUIMargin.top, pUIMargin.width,
pUIMargin.height);
if (!IsDropDownStyle() || !edit_) {
return;
}
CFX_RectF rtEdit(content_rect_.left, content_rect_.top,
content_rect_.width - fBtn, content_rect_.height);
edit_->SetWidgetRect(rtEdit);
if (cur_sel_ >= 0) {
CFWL_ListBox::Item* hItem = list_box_->GetItem(this, cur_sel_);
ScopedUpdateLock update_lock(edit_);
edit_->SetText(hItem ? hItem->GetText() : WideString());
}
edit_->Update();
}
void CFWL_ComboBox::ResetEditAlignment() {
if (!edit_) {
return;
}
uint32_t dwAdd = 0;
switch (properties_.style_exts_ & FWL_STYLEEXT_CMB_EditHAlignMask) {
case FWL_STYLEEXT_CMB_EditHCenter: {
dwAdd |= FWL_STYLEEXT_EDT_HCenter;
break;
}
default: {
dwAdd |= FWL_STYLEEXT_EDT_HNear;
break;
}
}
switch (properties_.style_exts_ & FWL_STYLEEXT_CMB_EditVAlignMask) {
case FWL_STYLEEXT_CMB_EditVCenter: {
dwAdd |= FWL_STYLEEXT_EDT_VCenter;
break;
}
case FWL_STYLEEXT_CMB_EditVFar: {
dwAdd |= FWL_STYLEEXT_EDT_VFar;
break;
}
default: {
dwAdd |= FWL_STYLEEXT_EDT_VNear;
break;
}
}
if (properties_.style_exts_ & FWL_STYLEEXT_CMB_EditJustified) {
dwAdd |= FWL_STYLEEXT_EDT_Justified;
}
edit_->ModifyStyleExts(dwAdd, FWL_STYLEEXT_EDT_HAlignMask |
FWL_STYLEEXT_EDT_HAlignModeMask |
FWL_STYLEEXT_EDT_VAlignMask);
}
void CFWL_ComboBox::ResetListItemAlignment() {
if (!list_box_) {
return;
}
uint32_t dwAdd = 0;
switch (properties_.style_exts_ & FWL_STYLEEXT_CMB_ListItemAlignMask) {
case FWL_STYLEEXT_CMB_ListItemCenterAlign: {
dwAdd |= FWL_STYLEEXT_LTB_CenterAlign;
break;
}
default: {
dwAdd |= FWL_STYLEEXT_LTB_LeftAlign;
break;
}
}
list_box_->ModifyStyleExts(dwAdd, FWL_STYLEEXT_CMB_ListItemAlignMask);
}
void CFWL_ComboBox::ProcessSelChanged(bool bLButtonUp) {
cur_sel_ = list_box_->GetItemIndex(this, list_box_->GetSelItem(0));
if (!IsDropDownStyle()) {
RepaintRect(client_rect_);
return;
}
CFWL_ListBox::Item* hItem = list_box_->GetItem(this, cur_sel_);
if (!hItem)
return;
if (edit_) {
edit_->SetText(hItem->GetText());
edit_->Update();
edit_->SetSelected();
}
CFWL_EventSelectChanged ev(this, bLButtonUp);
DispatchEvent(&ev);
}
void CFWL_ComboBox::OnProcessMessage(CFWL_Message* pMessage) {
bool backDefault = true;
switch (pMessage->GetType()) {
case CFWL_Message::Type::kSetFocus: {
backDefault = false;
OnFocusGained();
break;
}
case CFWL_Message::Type::kKillFocus: {
backDefault = false;
OnFocusLost();
break;
}
case CFWL_Message::Type::kMouse: {
backDefault = false;
CFWL_MessageMouse* pMsg = static_cast<CFWL_MessageMouse*>(pMessage);
switch (pMsg->cmd_) {
case CFWL_MessageMouse::MouseCommand::kLeftButtonDown:
OnLButtonDown(pMsg);
break;
case CFWL_MessageMouse::MouseCommand::kLeftButtonUp:
OnLButtonUp(pMsg);
break;
default:
break;
}
break;
}
case CFWL_Message::Type::kKey: {
backDefault = false;
CFWL_MessageKey* pKey = static_cast<CFWL_MessageKey*>(pMessage);
if (IsDropListVisible() &&
pKey->cmd_ == CFWL_MessageKey::KeyCommand::kKeyDown) {
bool bListKey = pKey->key_code_or_char_ == XFA_FWL_VKEY_Up ||
pKey->key_code_or_char_ == XFA_FWL_VKEY_Down ||
pKey->key_code_or_char_ == XFA_FWL_VKEY_Return ||
pKey->key_code_or_char_ == XFA_FWL_VKEY_Escape;
if (bListKey) {
list_box_->GetDelegate()->OnProcessMessage(pMessage);
break;
}
}
OnKey(pKey);
break;
}
default:
break;
}
// Dst target could be |this|, continue only if not destroyed by above.
if (backDefault && pMessage->GetDstTarget())
CFWL_Widget::OnProcessMessage(pMessage);
}
void CFWL_ComboBox::OnProcessEvent(CFWL_Event* pEvent) {
CFWL_Event::Type type = pEvent->GetType();
if (type == CFWL_Event::Type::Scroll) {
CFWL_EventScroll* pScrollEvent = static_cast<CFWL_EventScroll*>(pEvent);
CFWL_EventScroll pScrollEv(this, pScrollEvent->GetScrollCode(),
pScrollEvent->GetPos());
DispatchEvent(&pScrollEv);
} else if (type == CFWL_Event::Type::TextWillChange) {
CFWL_Event pTemp(CFWL_Event::Type::EditChanged, this);
DispatchEvent(&pTemp);
}
}
void CFWL_ComboBox::OnDrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
DrawWidget(pGraphics, matrix);
}
void CFWL_ComboBox::OnLButtonUp(CFWL_MessageMouse* pMsg) {
if (btn_rect_.Contains(pMsg->pos_)) {
btn_state_ = CFWL_PartState::kHovered;
} else {
btn_state_ = CFWL_PartState::kNormal;
}
RepaintRect(btn_rect_);
}
void CFWL_ComboBox::OnLButtonDown(CFWL_MessageMouse* pMsg) {
if (IsDropListVisible()) {
if (btn_rect_.Contains(pMsg->pos_)) {
HideDropDownList();
}
return;
}
if (!client_rect_.Contains(pMsg->pos_)) {
return;
}
if (edit_) {
MatchEditText();
}
ShowDropDownList();
}
void CFWL_ComboBox::OnFocusGained() {
properties_.states_ |= FWL_STATE_WGT_Focused;
if ((edit_->GetStates() & FWL_STATE_WGT_Focused) == 0) {
CFWL_MessageSetFocus msg(edit_);
edit_->GetDelegate()->OnProcessMessage(&msg);
}
}
void CFWL_ComboBox::OnFocusLost() {
properties_.states_ &= ~FWL_STATE_WGT_Focused;
HideDropDownList();
CFWL_MessageKillFocus msg(nullptr);
edit_->GetDelegate()->OnProcessMessage(&msg);
}
void CFWL_ComboBox::OnKey(CFWL_MessageKey* pMsg) {
uint32_t dwKeyCode = pMsg->key_code_or_char_;
const bool bUp = dwKeyCode == XFA_FWL_VKEY_Up;
const bool bDown = dwKeyCode == XFA_FWL_VKEY_Down;
if (bUp || bDown) {
CFWL_ComboList* pComboList = list_box_;
int32_t iCount = pComboList->CountItems(nullptr);
if (iCount < 1)
return;
bool bMatchEqual = false;
int32_t iCurSel = cur_sel_;
if (edit_) {
WideString wsText = edit_->GetText();
iCurSel = pComboList->MatchItem(wsText.AsStringView());
if (iCurSel >= 0) {
CFWL_ListBox::Item* item = list_box_->GetSelItem(iCurSel);
bMatchEqual = wsText == (item ? item->GetText() : WideString());
}
}
if (iCurSel < 0) {
iCurSel = 0;
} else if (bMatchEqual) {
if ((bUp && iCurSel == 0) || (bDown && iCurSel == iCount - 1))
return;
if (bUp)
iCurSel--;
else
iCurSel++;
}
cur_sel_ = iCurSel;
SyncEditText(cur_sel_);
return;
}
if (edit_) {
edit_->GetDelegate()->OnProcessMessage(pMsg);
}
}
void CFWL_ComboBox::GetPopupPos(float fMinHeight,
float fMaxHeight,
const CFX_RectF& rtAnchor,
CFX_RectF* pPopupRect) {
GetWidgetMgr()->GetAdapterPopupPos(this, fMinHeight, fMaxHeight, rtAnchor,
pPopupRect);
}
} // namespace pdfium