// 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 #include #include #include "core/fxcrt/stl_util.h" #include "third_party/base/cxx17_backports.h" #include "third_party/base/numerics/safe_conversions.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 { 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(m_pHorzScrollBar); visitor->Trace(m_pVertScrollBar); } FWL_Type CFWL_ListBox::GetClassID() const { return FWL_Type::ListBox; } void CFWL_ListBox::Update() { if (IsLocked()) return; switch (m_Properties.m_dwStyleExts & FWL_STYLEEXT_LTB_AlignMask) { case FWL_STYLEEXT_LTB_LeftAlign: m_iTTOAligns = FDE_TextAlignment::kCenterLeft; break; case FWL_STYLEEXT_LTB_RightAlign: m_iTTOAligns = FDE_TextAlignment::kCenterRight; break; case FWL_STYLEEXT_LTB_CenterAlign: default: m_iTTOAligns = FDE_TextAlignment::kCenter; break; } m_TTOStyles.single_line_ = true; m_fScorllBarWidth = GetScrollWidth(); CalcSize(); } FWL_WidgetHit CFWL_ListBox::HitTest(const CFX_PointF& point) { if (IsShowHorzScrollBar()) { CFX_RectF rect = m_pHorzScrollBar->GetWidgetRect(); if (rect.Contains(point)) return FWL_WidgetHit::HScrollBar; } if (IsShowVertScrollBar()) { CFX_RectF rect = m_pVertScrollBar->GetWidgetRect(); if (rect.Contains(point)) return FWL_WidgetHit::VScrollBar; } if (m_ClientRect.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(m_ContentRect); if (IsShowHorzScrollBar()) rtClip.height -= m_fScorllBarWidth; if (IsShowVertScrollBar()) rtClip.width -= m_fScorllBarWidth; pGraphics->SetClipRect(matrix.TransformRect(rtClip)); if ((m_Properties.m_dwStyles & 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 m_Properties.m_dwStyleExts & 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 - m_ContentRect.TopLeft(); float fPosX = 0.0f; if (m_pHorzScrollBar) fPosX = m_pHorzScrollBar->GetPos(); float fPosY = 0.0; if (m_pVertScrollBar) fPosY = m_pVertScrollBar->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 (!m_pVertScrollBar) return false; CFX_RectF rtItem = pItem ? pItem->GetRect() : CFX_RectF(); bool bScroll = false; float fPosY = m_pVertScrollBar->GetPos(); rtItem.Offset(0, -fPosY + m_ContentRect.top); if (rtItem.top < m_ContentRect.top) { fPosY += rtItem.top - m_ContentRect.top; bScroll = true; } else if (rtItem.bottom() > m_ContentRect.bottom()) { fPosY += rtItem.bottom() - m_ContentRect.bottom(); bScroll = true; } if (!bScroll) return false; m_pVertScrollBar->SetPos(fPosY); m_pVertScrollBar->SetTrackPos(fPosY); RepaintRect(m_ClientRect); 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.m_matrix = mtMatrix; param.m_PartRect = m_ClientRect; if (IsShowHorzScrollBar() && IsShowVertScrollBar()) param.m_pRtData = &m_StaticRect; if (!IsEnabled()) param.m_dwStates = CFWL_PartState::kDisabled; GetThemeProvider()->DrawBackground(param); } void CFWL_ListBox::DrawItems(CFGAS_GEGraphics* pGraphics, const CFX_Matrix& mtMatrix) { float fPosX = 0.0f; if (m_pHorzScrollBar) fPosX = m_pHorzScrollBar->GetPos(); float fPosY = 0.0f; if (m_pVertScrollBar) fPosY = m_pVertScrollBar->GetPos(); CFX_RectF rtView(m_ContentRect); if (m_pHorzScrollBar) rtView.height -= m_fScorllBarWidth; if (m_pVertScrollBar) rtView.width -= m_fScorllBarWidth; 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(m_ContentRect.left - fPosX, m_ContentRect.top - fPosY); if (rtItem.bottom() < m_ContentRect.top) continue; if (rtItem.top >= m_ContentRect.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 dwPartStates = CFWL_PartState::kNormal; if (m_Properties.m_dwStates & FWL_STATE_WGT_Disabled) dwPartStates = CFWL_PartState::kDisabled; else if (pItem && pItem->IsSelected()) dwPartStates = CFWL_PartState::kSelected; if ((m_Properties.m_dwStates & 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.m_dwStates = dwPartStates; bg_param.m_matrix = mtMatrix; bg_param.m_PartRect = rtItem; bg_param.m_bMaximize = true; bg_param.m_pRtData = &rtFocus; if (m_pVertScrollBar && !m_pHorzScrollBar && (dwPartStates & CFWL_PartState::kFocused)) { bg_param.m_PartRect.left += 1; bg_param.m_PartRect.width -= (m_fScorllBarWidth + 1); rtFocus.Deflate(0.5, 0.5, 1 + m_fScorllBarWidth, 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.m_dwStates = dwPartStates; textParam.m_matrix = mtMatrix; textParam.m_PartRect = rtText; textParam.m_wsText = std::move(wsText); textParam.m_dwTTOStyles = m_TTOStyles; textParam.m_iTTOAlign = m_iTTOAligns; textParam.m_bMaximize = true; pTheme->DrawText(textParam); } CFX_SizeF CFWL_ListBox::CalcSize() { m_ClientRect = GetClientRect(); m_ContentRect = m_ClientRect; CFX_RectF rtUIMargin; if (!GetOuter()) { CFWL_ThemePart part(CFWL_ThemePart::Part::kNone, this); CFX_RectF pUIMargin = GetThemeProvider()->GetUIMargin(part); m_ContentRect.Deflate(pUIMargin.left, pUIMargin.top, pUIMargin.width, pUIMargin.height); } float fWidth = GetMaxTextWidth(); fWidth += 2 * kItemTextMargin; float fActualWidth = m_ClientRect.width - rtUIMargin.left - rtUIMargin.width; fWidth = std::max(fWidth, fActualWidth); m_fItemHeight = 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, m_fItemHeight); } float iHeight = m_ClientRect.height; bool bShowVertScr = false; bool bShowHorzScr = false; if (!bShowVertScr && (m_Properties.m_dwStyles & FWL_STYLE_WGT_VScroll)) bShowVertScr = (fs.height > iHeight); float fMax = 0.0f; if (bShowVertScr) { if (!m_pVertScrollBar) InitVerticalScrollBar(); CFX_RectF rtScrollBar(m_ClientRect.right() - m_fScorllBarWidth, m_ClientRect.top, m_fScorllBarWidth, m_ClientRect.height - 1); if (bShowHorzScr) rtScrollBar.height -= m_fScorllBarWidth; m_pVertScrollBar->SetWidgetRect(rtScrollBar); fMax = std::max(fs.height - m_ContentRect.height, m_fItemHeight); m_pVertScrollBar->SetRange(0.0f, fMax); m_pVertScrollBar->SetPageSize(rtScrollBar.height * 9 / 10); m_pVertScrollBar->SetStepSize(m_fItemHeight); float fPos = pdfium::clamp(m_pVertScrollBar->GetPos(), 0.0f, fMax); m_pVertScrollBar->SetPos(fPos); m_pVertScrollBar->SetTrackPos(fPos); if ((m_Properties.m_dwStyleExts & FWL_STYLEEXT_LTB_ShowScrollBarFocus) == 0 || (m_Properties.m_dwStates & FWL_STATE_WGT_Focused)) { m_pVertScrollBar->RemoveStates(FWL_STATE_WGT_Invisible); } m_pVertScrollBar->Update(); } else if (m_pVertScrollBar) { m_pVertScrollBar->SetPos(0); m_pVertScrollBar->SetTrackPos(0); m_pVertScrollBar->SetStates(FWL_STATE_WGT_Invisible); } if (bShowHorzScr) { if (!m_pHorzScrollBar) InitHorizontalScrollBar(); CFX_RectF rtScrollBar(m_ClientRect.left, m_ClientRect.bottom() - m_fScorllBarWidth, m_ClientRect.width, m_fScorllBarWidth); if (bShowVertScr) rtScrollBar.width -= m_fScorllBarWidth; m_pHorzScrollBar->SetWidgetRect(rtScrollBar); fMax = fs.width - rtScrollBar.width; m_pHorzScrollBar->SetRange(0.0f, fMax); m_pHorzScrollBar->SetPageSize(fWidth * 9 / 10); m_pHorzScrollBar->SetStepSize(fWidth / 10); float fPos = pdfium::clamp(m_pHorzScrollBar->GetPos(), 0.0f, fMax); m_pHorzScrollBar->SetPos(fPos); m_pHorzScrollBar->SetTrackPos(fPos); if ((m_Properties.m_dwStyleExts & FWL_STYLEEXT_LTB_ShowScrollBarFocus) == 0 || (m_Properties.m_dwStates & FWL_STATE_WGT_Focused)) { m_pHorzScrollBar->RemoveStates(FWL_STATE_WGT_Invisible); } m_pHorzScrollBar->Update(); } else if (m_pHorzScrollBar) { m_pHorzScrollBar->SetPos(0); m_pHorzScrollBar->SetTrackPos(0); m_pHorzScrollBar->SetStates(FWL_STATE_WGT_Invisible); } if (bShowVertScr && bShowHorzScr) { m_StaticRect = CFX_RectF(m_ClientRect.right() - m_fScorllBarWidth, m_ClientRect.bottom() - m_fScorllBarWidth, m_fScorllBarWidth, m_fScorllBarWidth); } 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 (m_pVertScrollBar) return; m_pVertScrollBar = cppgc::MakeGarbageCollected( GetFWLApp()->GetHeap()->GetAllocationHandle(), GetFWLApp(), Properties{0, FWL_STYLEEXT_SCB_Vert, FWL_STATE_WGT_Invisible}, this); } void CFWL_ListBox::InitHorizontalScrollBar() { if (m_pHorzScrollBar) return; m_pHorzScrollBar = cppgc::MakeGarbageCollected( GetFWLApp()->GetHeap()->GetAllocationHandle(), GetFWLApp(), Properties{0, FWL_STYLEEXT_SCB_Horz, FWL_STATE_WGT_Invisible}, this); } bool CFWL_ListBox::IsShowVertScrollBar() const { return m_pVertScrollBar && m_pVertScrollBar->IsVisible() && ScrollBarPropertiesPresent(); } bool CFWL_ListBox::IsShowHorzScrollBar() const { return m_pHorzScrollBar && m_pHorzScrollBar->IsVisible() && ScrollBarPropertiesPresent(); } bool CFWL_ListBox::ScrollBarPropertiesPresent() const { return !(m_Properties.m_dwStyleExts & FWL_STYLEEXT_LTB_ShowScrollBarFocus) || (m_Properties.m_dwStates & 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(pMessage); switch (pMsg->m_dwCmd) { 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(pMessage)); break; case CFWL_Message::Type::kKey: { CFWL_MessageKey* pMsg = static_cast(pMessage); if (pMsg->m_dwCmd == CFWL_MessageKey::KeyCommand::kKeyDown) OnKeyDown(pMsg); break; } default: 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 == m_pVertScrollBar && m_pVertScrollBar) || (pSrcTarget == m_pHorzScrollBar && m_pHorzScrollBar)) { CFWL_EventScroll* pScrollEvent = static_cast(pEvent); OnScroll(static_cast(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 (m_pVertScrollBar) m_pVertScrollBar->RemoveStates(FWL_STATE_WGT_Invisible); if (m_pHorzScrollBar) m_pHorzScrollBar->RemoveStates(FWL_STATE_WGT_Invisible); } m_Properties.m_dwStates |= FWL_STATE_WGT_Focused; RepaintRect(m_ClientRect); } void CFWL_ListBox::OnFocusLost() { if (GetStyleExts() & FWL_STYLEEXT_LTB_ShowScrollBarFocus) { if (m_pVertScrollBar) m_pVertScrollBar->SetStates(FWL_STATE_WGT_Invisible); if (m_pHorzScrollBar) m_pHorzScrollBar->SetStates(FWL_STATE_WGT_Invisible); } m_Properties.m_dwStates &= ~FWL_STATE_WGT_Focused; RepaintRect(m_ClientRect); } void CFWL_ListBox::OnLButtonDown(CFWL_MessageMouse* pMsg) { m_bLButtonDown = true; Item* pItem = GetItemAtPoint(pMsg->m_pos); if (!pItem) return; if (IsMultiSelection()) { if (pMsg->m_dwFlags & XFA_FWL_KeyFlag::kCtrl) { pItem->SetSelected(!pItem->IsSelected()); m_hAnchor = pItem; } else if (pMsg->m_dwFlags & XFA_FWL_KeyFlag::kShift) { if (m_hAnchor) SetSelection(m_hAnchor, pItem, true); else pItem->SetSelected(true); } else { SetSelection(pItem, pItem, true); m_hAnchor = pItem; } } else { SetSelection(pItem, pItem, true); } SetFocusItem(pItem); ScrollToVisible(pItem); SetGrab(true); RepaintRect(m_ClientRect); } void CFWL_ListBox::OnLButtonUp(CFWL_MessageMouse* pMsg) { if (!m_bLButtonDown) return; m_bLButtonDown = false; SetGrab(false); } void CFWL_ListBox::OnMouseWheel(CFWL_MessageMouseWheel* pMsg) { if (IsShowVertScrollBar()) m_pVertScrollBar->GetDelegate()->OnProcessMessage(pMsg); } void CFWL_ListBox::OnKeyDown(CFWL_MessageKey* pMsg) { auto dwKeyCode = static_cast(pMsg->m_dwKeyCodeOrChar); 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->m_dwFlags & XFA_FWL_KeyFlag::kShift); bool bCtrl = !!(pMsg->m_dwFlags & 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 (m_hAnchor) SetSelection(m_hAnchor, pItem, true); else pItem->SetSelected(true); } else { SetSelection(pItem, pItem, true); m_hAnchor = pItem; } } else { SetSelection(pItem, pItem, true); } SetFocusItem(pItem); ScrollToVisible(pItem); RepaintRect(CFX_RectF(0, 0, m_WidgetRect.width, m_WidgetRect.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(m_ClientRect); } return true; } int32_t CFWL_ListBox::CountItems(const CFWL_Widget* pWidget) const { return fxcrt::CollectionSize(m_ItemArray); } CFWL_ListBox::Item* CFWL_ListBox::GetItem(const CFWL_Widget* pWidget, int32_t nIndex) const { if (nIndex < 0 || nIndex >= CountItems(pWidget)) return nullptr; return m_ItemArray[nIndex].get(); } int32_t CFWL_ListBox::GetItemIndex(CFWL_Widget* pWidget, Item* pItem) { auto it = std::find_if(m_ItemArray.begin(), m_ItemArray.end(), [pItem](const std::unique_ptr& candidate) { return candidate.get() == pItem; }); return it != m_ItemArray.end() ? pdfium::base::checked_cast(it - m_ItemArray.begin()) : -1; } CFWL_ListBox::Item* CFWL_ListBox::AddString(const WideString& wsAdd) { m_ItemArray.push_back(std::make_unique(wsAdd)); return m_ItemArray.back().get(); } void CFWL_ListBox::RemoveAt(int32_t iIndex) { if (iIndex < 0 || static_cast(iIndex) >= m_ItemArray.size()) return; m_ItemArray.erase(m_ItemArray.begin() + iIndex); } void CFWL_ListBox::DeleteString(Item* pItem) { int32_t nIndex = GetItemIndex(this, pItem); if (nIndex < 0 || static_cast(nIndex) >= m_ItemArray.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); } m_ItemArray.erase(m_ItemArray.begin() + nIndex); } void CFWL_ListBox::DeleteAll() { m_ItemArray.clear(); } CFWL_ListBox::Item::Item(const WideString& text) : m_wsText(text) {} CFWL_ListBox::Item::~Item() = default;