blob: 6f00ed87a3e6a9c920567ca8b28b288cf465333a [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_listbox.h"
#include <algorithm>
#include <memory>
#include <utility>
#include "core/fxcrt/numerics/safe_conversions.h"
#include "core/fxcrt/stl_util.h"
#include "v8/include/cppgc/visitor.h"
#include "xfa/fde/cfde_textout.h"
#include "xfa/fgas/graphics/cfgas_gegraphics.h"
#include "xfa/fwl/cfwl_app.h"
#include "xfa/fwl/cfwl_messagekey.h"
#include "xfa/fwl/cfwl_messagemouse.h"
#include "xfa/fwl/cfwl_messagemousewheel.h"
#include "xfa/fwl/cfwl_themebackground.h"
#include "xfa/fwl/cfwl_themepart.h"
#include "xfa/fwl/cfwl_themetext.h"
#include "xfa/fwl/fwl_widgetdef.h"
#include "xfa/fwl/ifwl_themeprovider.h"
namespace pdfium {
namespace {
const int kItemTextMargin = 2;
} // namespace
CFWL_ListBox::CFWL_ListBox(CFWL_App* app,
const Properties& properties,
CFWL_Widget* pOuter)
: CFWL_Widget(app, properties, pOuter) {}
CFWL_ListBox::~CFWL_ListBox() = default;
void CFWL_ListBox::Trace(cppgc::Visitor* visitor) const {
CFWL_Widget::Trace(visitor);
visitor->Trace(horz_scroll_bar_);
visitor->Trace(vert_scroll_bar_);
}
FWL_Type CFWL_ListBox::GetClassID() const {
return FWL_Type::ListBox;
}
void CFWL_ListBox::Update() {
if (IsLocked())
return;
switch (properties_.style_exts_ & FWL_STYLEEXT_LTB_AlignMask) {
case FWL_STYLEEXT_LTB_LeftAlign:
ttoaligns_ = FDE_TextAlignment::kCenterLeft;
break;
case FWL_STYLEEXT_LTB_RightAlign:
ttoaligns_ = FDE_TextAlignment::kCenterRight;
break;
case FWL_STYLEEXT_LTB_CenterAlign:
default:
ttoaligns_ = FDE_TextAlignment::kCenter;
break;
}
tto_styles_.single_line_ = true;
scorll_bar_width_ = GetScrollWidth();
CalcSize();
}
FWL_WidgetHit CFWL_ListBox::HitTest(const CFX_PointF& point) {
if (IsShowHorzScrollBar()) {
CFX_RectF rect = horz_scroll_bar_->GetWidgetRect();
if (rect.Contains(point))
return FWL_WidgetHit::HScrollBar;
}
if (IsShowVertScrollBar()) {
CFX_RectF rect = vert_scroll_bar_->GetWidgetRect();
if (rect.Contains(point))
return FWL_WidgetHit::VScrollBar;
}
if (client_rect_.Contains(point)) {
return FWL_WidgetHit::Client;
}
return FWL_WidgetHit::Unknown;
}
void CFWL_ListBox::DrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
if (!pGraphics)
return;
CFGAS_GEGraphics::StateRestorer restorer(pGraphics);
if (HasBorder())
DrawBorder(pGraphics, CFWL_ThemePart::Part::kBorder, matrix);
CFX_RectF rtClip(content_rect_);
if (IsShowHorzScrollBar())
rtClip.height -= scorll_bar_width_;
if (IsShowVertScrollBar())
rtClip.width -= scorll_bar_width_;
pGraphics->SetClipRect(matrix.TransformRect(rtClip));
if ((properties_.styles_ & FWL_STYLE_WGT_NoBackground) == 0) {
DrawBkground(pGraphics, matrix);
}
DrawItems(pGraphics, matrix);
}
int32_t CFWL_ListBox::CountSelItems() {
int32_t iRet = 0;
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (pItem && pItem->IsSelected())
iRet++;
}
return iRet;
}
CFWL_ListBox::Item* CFWL_ListBox::GetSelItem(int32_t nIndexSel) {
int32_t idx = GetSelIndex(nIndexSel);
if (idx < 0)
return nullptr;
return GetItem(this, idx);
}
int32_t CFWL_ListBox::GetSelIndex(int32_t nIndex) {
int32_t index = 0;
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (!pItem)
return -1;
if (pItem->IsSelected()) {
if (index == nIndex)
return i;
index++;
}
}
return -1;
}
void CFWL_ListBox::SetSelItem(Item* pItem, bool bSelect) {
if (!pItem) {
if (bSelect) {
SelectAll();
} else {
ClearSelection();
SetFocusItem(nullptr);
}
return;
}
if (IsMultiSelection())
pItem->SetSelected(bSelect);
else
SetSelection(pItem, pItem, bSelect);
}
CFWL_ListBox::Item* CFWL_ListBox::GetListItem(Item* pItem,
XFA_FWL_VKEYCODE dwKeyCode) {
Item* hRet = nullptr;
switch (dwKeyCode) {
case XFA_FWL_VKEY_Up:
case XFA_FWL_VKEY_Down:
case XFA_FWL_VKEY_Home:
case XFA_FWL_VKEY_End: {
const bool bUp = dwKeyCode == XFA_FWL_VKEY_Up;
const bool bDown = dwKeyCode == XFA_FWL_VKEY_Down;
const bool bHome = dwKeyCode == XFA_FWL_VKEY_Home;
int32_t iDstItem = -1;
if (bUp || bDown) {
int32_t index = GetItemIndex(this, pItem);
iDstItem = dwKeyCode == XFA_FWL_VKEY_Up ? index - 1 : index + 1;
} else if (bHome) {
iDstItem = 0;
} else {
int32_t iCount = CountItems(this);
iDstItem = iCount - 1;
}
hRet = GetItem(this, iDstItem);
break;
}
default:
break;
}
return hRet;
}
void CFWL_ListBox::SetSelection(Item* hStart, Item* hEnd, bool bSelected) {
int32_t iStart = GetItemIndex(this, hStart);
int32_t iEnd = GetItemIndex(this, hEnd);
if (iStart > iEnd)
std::swap(iStart, iEnd);
if (bSelected) {
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (pItem)
pItem->SetSelected(false);
}
}
while (iStart <= iEnd) {
Item* pItem = GetItem(this, iStart);
if (pItem)
pItem->SetSelected(bSelected);
++iStart;
}
}
bool CFWL_ListBox::IsMultiSelection() const {
return properties_.style_exts_ & FWL_STYLEEXT_LTB_MultiSelection;
}
void CFWL_ListBox::ClearSelection() {
bool bMulti = IsMultiSelection();
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (!pItem)
continue;
if (!pItem->IsSelected())
continue;
pItem->SetSelected(false);
if (!bMulti)
return;
}
}
void CFWL_ListBox::SelectAll() {
if (!IsMultiSelection())
return;
int32_t iCount = CountItems(this);
if (iCount <= 0)
return;
Item* pItemStart = GetItem(this, 0);
Item* pItemEnd = GetItem(this, iCount - 1);
SetSelection(pItemStart, pItemEnd, false);
}
CFWL_ListBox::Item* CFWL_ListBox::GetFocusedItem() {
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (!pItem)
break;
if (pItem->IsFocused())
return pItem;
}
return nullptr;
}
void CFWL_ListBox::SetFocusItem(Item* pItem) {
Item* hFocus = GetFocusedItem();
if (pItem == hFocus)
return;
if (hFocus)
hFocus->SetFocused(false);
if (pItem)
pItem->SetFocused(true);
}
CFWL_ListBox::Item* CFWL_ListBox::GetItemAtPoint(const CFX_PointF& point) {
CFX_PointF pos = point - content_rect_.TopLeft();
float fPosX = 0.0f;
if (horz_scroll_bar_) {
fPosX = horz_scroll_bar_->GetPos();
}
float fPosY = 0.0;
if (vert_scroll_bar_) {
fPosY = vert_scroll_bar_->GetPos();
}
int32_t nCount = CountItems(this);
for (int32_t i = 0; i < nCount; i++) {
Item* pItem = GetItem(this, i);
if (!pItem)
continue;
CFX_RectF rtItem = pItem->GetRect();
rtItem.Offset(-fPosX, -fPosY);
if (rtItem.Contains(pos))
return pItem;
}
return nullptr;
}
bool CFWL_ListBox::ScrollToVisible(Item* pItem) {
if (!vert_scroll_bar_) {
return false;
}
CFX_RectF rtItem = pItem ? pItem->GetRect() : CFX_RectF();
bool bScroll = false;
float fPosY = vert_scroll_bar_->GetPos();
rtItem.Offset(0, -fPosY + content_rect_.top);
if (rtItem.top < content_rect_.top) {
fPosY += rtItem.top - content_rect_.top;
bScroll = true;
} else if (rtItem.bottom() > content_rect_.bottom()) {
fPosY += rtItem.bottom() - content_rect_.bottom();
bScroll = true;
}
if (!bScroll)
return false;
vert_scroll_bar_->SetPos(fPosY);
vert_scroll_bar_->SetTrackPos(fPosY);
RepaintRect(client_rect_);
return true;
}
void CFWL_ListBox::DrawBkground(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& mtMatrix) {
if (!pGraphics)
return;
CFWL_ThemeBackground param(CFWL_ThemePart::Part::kBackground, this,
pGraphics);
param.matrix_ = mtMatrix;
param.part_rect_ = client_rect_;
if (IsShowHorzScrollBar() && IsShowVertScrollBar())
param.data_rect_ = &static_rect_;
if (!IsEnabled())
param.states_ = CFWL_PartState::kDisabled;
GetThemeProvider()->DrawBackground(param);
}
void CFWL_ListBox::DrawItems(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& mtMatrix) {
float fPosX = 0.0f;
if (horz_scroll_bar_) {
fPosX = horz_scroll_bar_->GetPos();
}
float fPosY = 0.0f;
if (vert_scroll_bar_) {
fPosY = vert_scroll_bar_->GetPos();
}
CFX_RectF rtView(content_rect_);
if (horz_scroll_bar_) {
rtView.height -= scorll_bar_width_;
}
if (vert_scroll_bar_) {
rtView.width -= scorll_bar_width_;
}
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
CFWL_ListBox::Item* pItem = GetItem(this, i);
if (!pItem)
continue;
CFX_RectF rtItem = pItem->GetRect();
rtItem.Offset(content_rect_.left - fPosX, content_rect_.top - fPosY);
if (rtItem.bottom() < content_rect_.top) {
continue;
}
if (rtItem.top >= content_rect_.bottom()) {
break;
}
DrawItem(pGraphics, pItem, i, rtItem, mtMatrix);
}
}
void CFWL_ListBox::DrawItem(CFGAS_GEGraphics* pGraphics,
Item* pItem,
int32_t Index,
const CFX_RectF& rtItem,
const CFX_Matrix& mtMatrix) {
Mask<CFWL_PartState> dwPartStates = CFWL_PartState::kNormal;
if (properties_.states_ & FWL_STATE_WGT_Disabled) {
dwPartStates = CFWL_PartState::kDisabled;
} else if (pItem && pItem->IsSelected()) {
dwPartStates = CFWL_PartState::kSelected;
}
if ((properties_.states_ & FWL_STATE_WGT_Focused) && pItem &&
pItem->IsFocused()) {
dwPartStates |= CFWL_PartState::kFocused;
}
CFX_RectF rtFocus(rtItem); // Must outlive |bg_param|.
CFWL_ThemeBackground bg_param(CFWL_ThemePart::Part::kListItem, this,
pGraphics);
bg_param.states_ = dwPartStates;
bg_param.matrix_ = mtMatrix;
bg_param.part_rect_ = rtItem;
bg_param.maximize_ = true;
bg_param.data_rect_ = &rtFocus;
if (vert_scroll_bar_ && !horz_scroll_bar_ &&
(dwPartStates & CFWL_PartState::kFocused)) {
bg_param.part_rect_.left += 1;
bg_param.part_rect_.width -= (scorll_bar_width_ + 1);
rtFocus.Deflate(0.5, 0.5, 1 + scorll_bar_width_, 1);
}
IFWL_ThemeProvider* pTheme = GetThemeProvider();
pTheme->DrawBackground(bg_param);
if (!pItem)
return;
WideString wsText = pItem->GetText();
if (wsText.GetLength() <= 0)
return;
CFX_RectF rtText(rtItem);
rtText.Deflate(kItemTextMargin, kItemTextMargin);
CFWL_ThemeText textParam(CFWL_ThemePart::Part::kListItem, this, pGraphics);
textParam.states_ = dwPartStates;
textParam.matrix_ = mtMatrix;
textParam.part_rect_ = rtText;
textParam.text_ = std::move(wsText);
textParam.tto_styles_ = tto_styles_;
textParam.tto_align_ = ttoaligns_;
textParam.maximize_ = true;
pTheme->DrawText(textParam);
}
CFX_SizeF CFWL_ListBox::CalcSize() {
client_rect_ = GetClientRect();
content_rect_ = client_rect_;
CFX_RectF rtUIMargin;
if (!GetOuter()) {
CFWL_ThemePart part(CFWL_ThemePart::Part::kNone, this);
CFX_RectF pUIMargin = GetThemeProvider()->GetUIMargin(part);
content_rect_.Deflate(pUIMargin.left, pUIMargin.top, pUIMargin.width,
pUIMargin.height);
}
float fWidth = GetMaxTextWidth();
fWidth += 2 * kItemTextMargin;
float fActualWidth = client_rect_.width - rtUIMargin.left - rtUIMargin.width;
fWidth = std::max(fWidth, fActualWidth);
item_height_ = CalcItemHeight();
int32_t iCount = CountItems(this);
CFX_SizeF fs;
for (int32_t i = 0; i < iCount; i++) {
Item* htem = GetItem(this, i);
UpdateItemSize(htem, fs, fWidth, item_height_);
}
float iHeight = client_rect_.height;
bool bShowVertScr = false;
bool bShowHorzScr = false;
if (!bShowVertScr && (properties_.styles_ & FWL_STYLE_WGT_VScroll)) {
bShowVertScr = (fs.height > iHeight);
}
float fMax = 0.0f;
if (bShowVertScr) {
if (!vert_scroll_bar_) {
InitVerticalScrollBar();
}
CFX_RectF rtScrollBar(client_rect_.right() - scorll_bar_width_,
client_rect_.top, scorll_bar_width_,
client_rect_.height - 1);
if (bShowHorzScr)
rtScrollBar.height -= scorll_bar_width_;
vert_scroll_bar_->SetWidgetRect(rtScrollBar);
fMax = std::max(fs.height - content_rect_.height, item_height_);
vert_scroll_bar_->SetRange(0.0f, fMax);
vert_scroll_bar_->SetPageSize(rtScrollBar.height * 9 / 10);
vert_scroll_bar_->SetStepSize(item_height_);
float fPos = std::clamp(vert_scroll_bar_->GetPos(), 0.0f, fMax);
vert_scroll_bar_->SetPos(fPos);
vert_scroll_bar_->SetTrackPos(fPos);
if ((properties_.style_exts_ & FWL_STYLEEXT_LTB_ShowScrollBarFocus) == 0 ||
(properties_.states_ & FWL_STATE_WGT_Focused)) {
vert_scroll_bar_->RemoveStates(FWL_STATE_WGT_Invisible);
}
vert_scroll_bar_->Update();
} else if (vert_scroll_bar_) {
vert_scroll_bar_->SetPos(0);
vert_scroll_bar_->SetTrackPos(0);
vert_scroll_bar_->SetStates(FWL_STATE_WGT_Invisible);
}
if (bShowHorzScr) {
if (!horz_scroll_bar_) {
InitHorizontalScrollBar();
}
CFX_RectF rtScrollBar(client_rect_.left,
client_rect_.bottom() - scorll_bar_width_,
client_rect_.width, scorll_bar_width_);
if (bShowVertScr)
rtScrollBar.width -= scorll_bar_width_;
horz_scroll_bar_->SetWidgetRect(rtScrollBar);
fMax = fs.width - rtScrollBar.width;
horz_scroll_bar_->SetRange(0.0f, fMax);
horz_scroll_bar_->SetPageSize(fWidth * 9 / 10);
horz_scroll_bar_->SetStepSize(fWidth / 10);
float fPos = std::clamp(horz_scroll_bar_->GetPos(), 0.0f, fMax);
horz_scroll_bar_->SetPos(fPos);
horz_scroll_bar_->SetTrackPos(fPos);
if ((properties_.style_exts_ & FWL_STYLEEXT_LTB_ShowScrollBarFocus) == 0 ||
(properties_.states_ & FWL_STATE_WGT_Focused)) {
horz_scroll_bar_->RemoveStates(FWL_STATE_WGT_Invisible);
}
horz_scroll_bar_->Update();
} else if (horz_scroll_bar_) {
horz_scroll_bar_->SetPos(0);
horz_scroll_bar_->SetTrackPos(0);
horz_scroll_bar_->SetStates(FWL_STATE_WGT_Invisible);
}
if (bShowVertScr && bShowHorzScr) {
static_rect_ = CFX_RectF(client_rect_.right() - scorll_bar_width_,
client_rect_.bottom() - scorll_bar_width_,
scorll_bar_width_, scorll_bar_width_);
}
return fs;
}
void CFWL_ListBox::UpdateItemSize(Item* pItem,
CFX_SizeF& size,
float fWidth,
float fItemHeight) const {
if (pItem) {
CFX_RectF rtItem(0, size.height, fWidth, fItemHeight);
pItem->SetRect(rtItem);
}
size.width = fWidth;
size.height += fItemHeight;
}
float CFWL_ListBox::GetMaxTextWidth() {
float fRet = 0.0f;
int32_t iCount = CountItems(this);
for (int32_t i = 0; i < iCount; i++) {
Item* pItem = GetItem(this, i);
if (!pItem)
continue;
CFX_SizeF sz = CalcTextSize(pItem->GetText(), false);
fRet = std::max(fRet, sz.width);
}
return fRet;
}
float CFWL_ListBox::GetScrollWidth() {
return GetThemeProvider()->GetScrollBarWidth();
}
float CFWL_ListBox::CalcItemHeight() {
CFWL_ThemePart part(CFWL_ThemePart::Part::kNone, this);
return GetThemeProvider()->GetFontSize(part) + 2 * kItemTextMargin;
}
void CFWL_ListBox::InitVerticalScrollBar() {
if (vert_scroll_bar_) {
return;
}
vert_scroll_bar_ = cppgc::MakeGarbageCollected<CFWL_ScrollBar>(
GetFWLApp()->GetHeap()->GetAllocationHandle(), GetFWLApp(),
Properties{0, FWL_STYLEEXT_SCB_Vert, FWL_STATE_WGT_Invisible}, this);
}
void CFWL_ListBox::InitHorizontalScrollBar() {
if (horz_scroll_bar_) {
return;
}
horz_scroll_bar_ = cppgc::MakeGarbageCollected<CFWL_ScrollBar>(
GetFWLApp()->GetHeap()->GetAllocationHandle(), GetFWLApp(),
Properties{0, FWL_STYLEEXT_SCB_Horz, FWL_STATE_WGT_Invisible}, this);
}
bool CFWL_ListBox::IsShowVertScrollBar() const {
return vert_scroll_bar_ && vert_scroll_bar_->IsVisible() &&
ScrollBarPropertiesPresent();
}
bool CFWL_ListBox::IsShowHorzScrollBar() const {
return horz_scroll_bar_ && horz_scroll_bar_->IsVisible() &&
ScrollBarPropertiesPresent();
}
bool CFWL_ListBox::ScrollBarPropertiesPresent() const {
return !(properties_.style_exts_ & FWL_STYLEEXT_LTB_ShowScrollBarFocus) ||
(properties_.states_ & FWL_STATE_WGT_Focused);
}
void CFWL_ListBox::OnProcessMessage(CFWL_Message* pMessage) {
if (!IsEnabled())
return;
switch (pMessage->GetType()) {
case CFWL_Message::Type::kSetFocus:
OnFocusGained();
break;
case CFWL_Message::Type::kKillFocus:
OnFocusLost();
break;
case CFWL_Message::Type::kMouse: {
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::kMouseWheel:
OnMouseWheel(static_cast<CFWL_MessageMouseWheel*>(pMessage));
break;
case CFWL_Message::Type::kKey: {
CFWL_MessageKey* pMsg = static_cast<CFWL_MessageKey*>(pMessage);
if (pMsg->cmd_ == CFWL_MessageKey::KeyCommand::kKeyDown) {
OnKeyDown(pMsg);
}
break;
}
}
// Dst target could be |this|, continue only if not destroyed by above.
if (pMessage->GetDstTarget())
CFWL_Widget::OnProcessMessage(pMessage);
}
void CFWL_ListBox::OnProcessEvent(CFWL_Event* pEvent) {
if (!pEvent)
return;
if (pEvent->GetType() != CFWL_Event::Type::Scroll)
return;
CFWL_Widget* pSrcTarget = pEvent->GetSrcTarget();
if ((pSrcTarget == vert_scroll_bar_ && vert_scroll_bar_) ||
(pSrcTarget == horz_scroll_bar_ && horz_scroll_bar_)) {
CFWL_EventScroll* pScrollEvent = static_cast<CFWL_EventScroll*>(pEvent);
OnScroll(static_cast<CFWL_ScrollBar*>(pSrcTarget),
pScrollEvent->GetScrollCode(), pScrollEvent->GetPos());
}
}
void CFWL_ListBox::OnDrawWidget(CFGAS_GEGraphics* pGraphics,
const CFX_Matrix& matrix) {
DrawWidget(pGraphics, matrix);
}
void CFWL_ListBox::OnFocusGained() {
if (GetStyleExts() & FWL_STYLEEXT_LTB_ShowScrollBarFocus) {
if (vert_scroll_bar_) {
vert_scroll_bar_->RemoveStates(FWL_STATE_WGT_Invisible);
}
if (horz_scroll_bar_) {
horz_scroll_bar_->RemoveStates(FWL_STATE_WGT_Invisible);
}
}
properties_.states_ |= FWL_STATE_WGT_Focused;
RepaintRect(client_rect_);
}
void CFWL_ListBox::OnFocusLost() {
if (GetStyleExts() & FWL_STYLEEXT_LTB_ShowScrollBarFocus) {
if (vert_scroll_bar_) {
vert_scroll_bar_->SetStates(FWL_STATE_WGT_Invisible);
}
if (horz_scroll_bar_) {
horz_scroll_bar_->SetStates(FWL_STATE_WGT_Invisible);
}
}
properties_.states_ &= ~FWL_STATE_WGT_Focused;
RepaintRect(client_rect_);
}
void CFWL_ListBox::OnLButtonDown(CFWL_MessageMouse* pMsg) {
lbutton_down_ = true;
Item* pItem = GetItemAtPoint(pMsg->pos_);
if (!pItem)
return;
if (IsMultiSelection()) {
if (pMsg->flags_ & XFA_FWL_KeyFlag::kCtrl) {
pItem->SetSelected(!pItem->IsSelected());
h_anchor_ = pItem;
} else if (pMsg->flags_ & XFA_FWL_KeyFlag::kShift) {
if (h_anchor_) {
SetSelection(h_anchor_, pItem, true);
} else {
pItem->SetSelected(true);
}
} else {
SetSelection(pItem, pItem, true);
h_anchor_ = pItem;
}
} else {
SetSelection(pItem, pItem, true);
}
SetFocusItem(pItem);
ScrollToVisible(pItem);
SetGrab(true);
RepaintRect(client_rect_);
}
void CFWL_ListBox::OnLButtonUp(CFWL_MessageMouse* pMsg) {
if (!lbutton_down_) {
return;
}
lbutton_down_ = false;
SetGrab(false);
}
void CFWL_ListBox::OnMouseWheel(CFWL_MessageMouseWheel* pMsg) {
if (IsShowVertScrollBar())
vert_scroll_bar_->GetDelegate()->OnProcessMessage(pMsg);
}
void CFWL_ListBox::OnKeyDown(CFWL_MessageKey* pMsg) {
auto dwKeyCode = static_cast<XFA_FWL_VKEYCODE>(pMsg->key_code_or_char_);
switch (dwKeyCode) {
case XFA_FWL_VKEY_Tab:
case XFA_FWL_VKEY_Up:
case XFA_FWL_VKEY_Down:
case XFA_FWL_VKEY_Home:
case XFA_FWL_VKEY_End: {
Item* pItem = GetListItem(GetFocusedItem(), dwKeyCode);
bool bShift = !!(pMsg->flags_ & XFA_FWL_KeyFlag::kShift);
bool bCtrl = !!(pMsg->flags_ & XFA_FWL_KeyFlag::kCtrl);
OnVK(pItem, bShift, bCtrl);
break;
}
default:
break;
}
}
void CFWL_ListBox::OnVK(Item* pItem, bool bShift, bool bCtrl) {
if (!pItem)
return;
if (IsMultiSelection()) {
if (bCtrl) {
// Do nothing.
} else if (bShift) {
if (h_anchor_) {
SetSelection(h_anchor_, pItem, true);
} else {
pItem->SetSelected(true);
}
} else {
SetSelection(pItem, pItem, true);
h_anchor_ = pItem;
}
} else {
SetSelection(pItem, pItem, true);
}
SetFocusItem(pItem);
ScrollToVisible(pItem);
RepaintRect(CFX_RectF(0, 0, widget_rect_.width, widget_rect_.height));
}
bool CFWL_ListBox::OnScroll(CFWL_ScrollBar* pScrollBar,
CFWL_EventScroll::Code dwCode,
float fPos) {
float fMin;
float fMax;
pScrollBar->GetRange(&fMin, &fMax);
float iCurPos = pScrollBar->GetPos();
float fStep = pScrollBar->GetStepSize();
switch (dwCode) {
case CFWL_EventScroll::Code::Min: {
fPos = fMin;
break;
}
case CFWL_EventScroll::Code::Max: {
fPos = fMax;
break;
}
case CFWL_EventScroll::Code::StepBackward: {
fPos -= fStep;
if (fPos < fMin + fStep / 2)
fPos = fMin;
break;
}
case CFWL_EventScroll::Code::StepForward: {
fPos += fStep;
if (fPos > fMax - fStep / 2)
fPos = fMax;
break;
}
case CFWL_EventScroll::Code::PageBackward: {
fPos -= pScrollBar->GetPageSize();
if (fPos < fMin)
fPos = fMin;
break;
}
case CFWL_EventScroll::Code::PageForward: {
fPos += pScrollBar->GetPageSize();
if (fPos > fMax)
fPos = fMax;
break;
}
case CFWL_EventScroll::Code::Pos:
case CFWL_EventScroll::Code::TrackPos:
case CFWL_EventScroll::Code::None:
break;
case CFWL_EventScroll::Code::EndScroll:
return false;
}
if (iCurPos != fPos) {
pScrollBar->SetPos(fPos);
pScrollBar->SetTrackPos(fPos);
RepaintRect(client_rect_);
}
return true;
}
int32_t CFWL_ListBox::CountItems(const CFWL_Widget* pWidget) const {
return fxcrt::CollectionSize<int32_t>(item_array_);
}
CFWL_ListBox::Item* CFWL_ListBox::GetItem(const CFWL_Widget* pWidget,
int32_t nIndex) const {
if (nIndex < 0 || nIndex >= CountItems(pWidget))
return nullptr;
return item_array_[nIndex].get();
}
int32_t CFWL_ListBox::GetItemIndex(CFWL_Widget* pWidget, Item* pItem) {
auto it = std::find_if(item_array_.begin(), item_array_.end(),
[pItem](const std::unique_ptr<Item>& candidate) {
return candidate.get() == pItem;
});
return it != item_array_.end()
? checked_cast<int32_t>(it - item_array_.begin())
: -1;
}
CFWL_ListBox::Item* CFWL_ListBox::AddString(const WideString& wsAdd) {
item_array_.push_back(std::make_unique<Item>(wsAdd));
return item_array_.back().get();
}
void CFWL_ListBox::RemoveAt(int32_t iIndex) {
if (iIndex < 0 || static_cast<size_t>(iIndex) >= item_array_.size()) {
return;
}
item_array_.erase(item_array_.begin() + iIndex);
}
void CFWL_ListBox::DeleteString(Item* pItem) {
int32_t nIndex = GetItemIndex(this, pItem);
if (nIndex < 0 || static_cast<size_t>(nIndex) >= item_array_.size()) {
return;
}
int32_t iSel = nIndex + 1;
if (iSel >= CountItems(this))
iSel = nIndex - 1;
if (iSel >= 0) {
Item* item = GetItem(this, iSel);
if (item)
item->SetSelected(true);
}
item_array_.erase(item_array_.begin() + nIndex);
}
void CFWL_ListBox::DeleteAll() {
item_array_.clear();
}
CFWL_ListBox::Item::Item(const WideString& text) : text_(text) {}
CFWL_ListBox::Item::~Item() = default;
} // namespace pdfium