// 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 "fpdfsdk/pwl/cpwl_list_impl.h" #include #include #include "core/fpdfdoc/cpvt_word.h" #include "core/fxcrt/fx_extension.h" #include "fpdfsdk/pwl/cpwl_edit_impl.h" #include "fpdfsdk/pwl/cpwl_list_box.h" #include "third_party/base/ptr_util.h" #include "third_party/base/stl_util.h" CPWL_ListCtrl::Item::Item() : m_pEdit(pdfium::MakeUnique()) { m_pEdit->SetAlignmentV(1, true); m_pEdit->Initialize(); } CPWL_ListCtrl::Item::~Item() = default; void CPWL_ListCtrl::Item::SetFontMap(IPVT_FontMap* pFontMap) { m_pEdit->SetFontMap(pFontMap); } void CPWL_ListCtrl::Item::SetText(const WideString& text) { m_pEdit->SetText(text); } void CPWL_ListCtrl::Item::SetFontSize(float fFontSize) { m_pEdit->SetFontSize(fFontSize); } float CPWL_ListCtrl::Item::GetItemHeight() const { return m_pEdit->GetContentRect().Height(); } uint16_t CPWL_ListCtrl::Item::GetFirstChar() const { CPVT_Word word; CPWL_EditImpl_Iterator* pIterator = m_pEdit->GetIterator(); pIterator->SetAt(1); pIterator->GetWord(word); return word.Word; } WideString CPWL_ListCtrl::Item::GetText() const { return m_pEdit->GetText(); } CPLST_Select::CPLST_Select() {} CPLST_Select::~CPLST_Select() {} void CPLST_Select::Add(int32_t nItemIndex) { m_Items[nItemIndex] = SELECTING; } void CPLST_Select::Add(int32_t nBeginIndex, int32_t nEndIndex) { if (nBeginIndex > nEndIndex) std::swap(nBeginIndex, nEndIndex); for (int32_t i = nBeginIndex; i <= nEndIndex; ++i) Add(i); } void CPLST_Select::Sub(int32_t nItemIndex) { auto it = m_Items.find(nItemIndex); if (it != m_Items.end()) it->second = DESELECTING; } void CPLST_Select::Sub(int32_t nBeginIndex, int32_t nEndIndex) { if (nBeginIndex > nEndIndex) std::swap(nBeginIndex, nEndIndex); for (int32_t i = nBeginIndex; i <= nEndIndex; ++i) Sub(i); } void CPLST_Select::DeselectAll() { for (auto& item : m_Items) item.second = DESELECTING; } void CPLST_Select::Done() { auto it = m_Items.begin(); while (it != m_Items.end()) { if (it->second == DESELECTING) it = m_Items.erase(it); else (it++)->second = NORMAL; } } CPWL_ListCtrl::CPWL_ListCtrl() : m_pNotify(nullptr), m_bNotifyFlag(false), m_nSelItem(-1), m_nFootIndex(-1), m_bCtrlSel(false), m_nCaretIndex(-1), m_fFontSize(0.0f), m_pFontMap(nullptr), m_bMultiple(false) {} CPWL_ListCtrl::~CPWL_ListCtrl() { Clear(); } CFX_PointF CPWL_ListCtrl::InToOut(const CFX_PointF& point) const { CFX_FloatRect rcPlate = m_rcPlate; return CFX_PointF(point.x - (m_ptScrollPos.x - rcPlate.left), point.y - (m_ptScrollPos.y - rcPlate.top)); } CFX_PointF CPWL_ListCtrl::OutToIn(const CFX_PointF& point) const { CFX_FloatRect rcPlate = m_rcPlate; return CFX_PointF(point.x + (m_ptScrollPos.x - rcPlate.left), point.y + (m_ptScrollPos.y - rcPlate.top)); } CFX_FloatRect CPWL_ListCtrl::InToOut(const CFX_FloatRect& rect) const { CFX_PointF ptLeftBottom = InToOut(CFX_PointF(rect.left, rect.bottom)); CFX_PointF ptRightTop = InToOut(CFX_PointF(rect.right, rect.top)); return CFX_FloatRect(ptLeftBottom.x, ptLeftBottom.y, ptRightTop.x, ptRightTop.y); } CFX_FloatRect CPWL_ListCtrl::OutToIn(const CFX_FloatRect& rect) const { CFX_PointF ptLeftBottom = OutToIn(CFX_PointF(rect.left, rect.bottom)); CFX_PointF ptRightTop = OutToIn(CFX_PointF(rect.right, rect.top)); return CFX_FloatRect(ptLeftBottom.x, ptLeftBottom.y, ptRightTop.x, ptRightTop.y); } CFX_PointF CPWL_ListCtrl::InnerToOuter(const CFX_PointF& point) const { return CFX_PointF(point.x + GetBTPoint().x, GetBTPoint().y - point.y); } CFX_PointF CPWL_ListCtrl::OuterToInner(const CFX_PointF& point) const { return CFX_PointF(point.x - GetBTPoint().x, GetBTPoint().y - point.y); } CFX_FloatRect CPWL_ListCtrl::InnerToOuter(const CFX_FloatRect& rect) const { CFX_PointF ptLeftTop = InnerToOuter(CFX_PointF(rect.left, rect.top)); CFX_PointF ptRightBottom = InnerToOuter(CFX_PointF(rect.right, rect.bottom)); return CFX_FloatRect(ptLeftTop.x, ptRightBottom.y, ptRightBottom.x, ptLeftTop.y); } CFX_FloatRect CPWL_ListCtrl::OuterToInner(const CFX_FloatRect& rect) const { CFX_PointF ptLeftTop = OuterToInner(CFX_PointF(rect.left, rect.top)); CFX_PointF ptRightBottom = OuterToInner(CFX_PointF(rect.right, rect.bottom)); return CFX_FloatRect(ptLeftTop.x, ptRightBottom.y, ptRightBottom.x, ptLeftTop.y); } void CPWL_ListCtrl::OnMouseDown(const CFX_PointF& point, bool bShift, bool bCtrl) { int32_t nHitIndex = GetItemIndex(point); if (IsMultipleSel()) { if (bCtrl) { if (IsItemSelected(nHitIndex)) { m_aSelItems.Sub(nHitIndex); SelectItems(); m_bCtrlSel = false; } else { m_aSelItems.Add(nHitIndex); SelectItems(); m_bCtrlSel = true; } m_nFootIndex = nHitIndex; } else if (bShift) { m_aSelItems.DeselectAll(); m_aSelItems.Add(m_nFootIndex, nHitIndex); SelectItems(); } else { m_aSelItems.DeselectAll(); m_aSelItems.Add(nHitIndex); SelectItems(); m_nFootIndex = nHitIndex; } SetCaret(nHitIndex); } else { SetSingleSelect(nHitIndex); } if (!IsItemVisible(nHitIndex)) ScrollToListItem(nHitIndex); } void CPWL_ListCtrl::OnMouseMove(const CFX_PointF& point, bool bShift, bool bCtrl) { int32_t nHitIndex = GetItemIndex(point); if (IsMultipleSel()) { if (bCtrl) { if (m_bCtrlSel) m_aSelItems.Add(m_nFootIndex, nHitIndex); else m_aSelItems.Sub(m_nFootIndex, nHitIndex); SelectItems(); } else { m_aSelItems.DeselectAll(); m_aSelItems.Add(m_nFootIndex, nHitIndex); SelectItems(); } SetCaret(nHitIndex); } else { SetSingleSelect(nHitIndex); } if (!IsItemVisible(nHitIndex)) ScrollToListItem(nHitIndex); } void CPWL_ListCtrl::OnVK(int32_t nItemIndex, bool bShift, bool bCtrl) { if (IsMultipleSel()) { if (nItemIndex >= 0 && nItemIndex < GetCount()) { if (bCtrl) { } else if (bShift) { m_aSelItems.DeselectAll(); m_aSelItems.Add(m_nFootIndex, nItemIndex); SelectItems(); } else { m_aSelItems.DeselectAll(); m_aSelItems.Add(nItemIndex); SelectItems(); m_nFootIndex = nItemIndex; } SetCaret(nItemIndex); } } else { SetSingleSelect(nItemIndex); } if (!IsItemVisible(nItemIndex)) ScrollToListItem(nItemIndex); } void CPWL_ListCtrl::OnVK_UP(bool bShift, bool bCtrl) { OnVK(IsMultipleSel() ? GetCaret() - 1 : GetSelect() - 1, bShift, bCtrl); } void CPWL_ListCtrl::OnVK_DOWN(bool bShift, bool bCtrl) { OnVK(IsMultipleSel() ? GetCaret() + 1 : GetSelect() + 1, bShift, bCtrl); } void CPWL_ListCtrl::OnVK_LEFT(bool bShift, bool bCtrl) { OnVK(0, bShift, bCtrl); } void CPWL_ListCtrl::OnVK_RIGHT(bool bShift, bool bCtrl) { OnVK(GetCount() - 1, bShift, bCtrl); } void CPWL_ListCtrl::OnVK_HOME(bool bShift, bool bCtrl) { OnVK(0, bShift, bCtrl); } void CPWL_ListCtrl::OnVK_END(bool bShift, bool bCtrl) { OnVK(GetCount() - 1, bShift, bCtrl); } bool CPWL_ListCtrl::OnChar(uint16_t nChar, bool bShift, bool bCtrl) { int32_t nIndex = GetLastSelected(); int32_t nFindIndex = FindNext(nIndex, nChar); if (nFindIndex != nIndex) { OnVK(nFindIndex, bShift, bCtrl); return true; } return false; } void CPWL_ListCtrl::SetPlateRect(const CFX_FloatRect& rect) { m_rcPlate = rect; m_ptScrollPos.x = rect.left; SetScrollPos(CFX_PointF(rect.left, rect.top)); ReArrange(0); InvalidateItem(-1); } CFX_FloatRect CPWL_ListCtrl::GetItemRect(int32_t nIndex) const { return InToOut(GetItemRectInternal(nIndex)); } CFX_FloatRect CPWL_ListCtrl::GetItemRectInternal(int32_t nIndex) const { if (!IsValid(nIndex)) return CFX_FloatRect(); CFX_FloatRect rcItem = m_ListItems[nIndex]->GetRect(); rcItem.left = 0.0f; rcItem.right = m_rcPlate.Width(); return InnerToOuter(rcItem); } void CPWL_ListCtrl::AddString(const WideString& str) { AddItem(str); ReArrange(GetCount() - 1); } void CPWL_ListCtrl::SetMultipleSelect(int32_t nItemIndex, bool bSelected) { if (!IsValid(nItemIndex)) return; if (bSelected != IsItemSelected(nItemIndex)) { if (bSelected) { SetItemSelect(nItemIndex, true); InvalidateItem(nItemIndex); } else { SetItemSelect(nItemIndex, false); InvalidateItem(nItemIndex); } } } void CPWL_ListCtrl::SetSingleSelect(int32_t nItemIndex) { if (!IsValid(nItemIndex)) return; if (m_nSelItem != nItemIndex) { if (m_nSelItem >= 0) { SetItemSelect(m_nSelItem, false); InvalidateItem(m_nSelItem); } SetItemSelect(nItemIndex, true); InvalidateItem(nItemIndex); m_nSelItem = nItemIndex; } } void CPWL_ListCtrl::SetCaret(int32_t nItemIndex) { if (!IsValid(nItemIndex)) return; if (IsMultipleSel()) { int32_t nOldIndex = m_nCaretIndex; if (nOldIndex != nItemIndex) { m_nCaretIndex = nItemIndex; InvalidateItem(nOldIndex); InvalidateItem(nItemIndex); } } } void CPWL_ListCtrl::InvalidateItem(int32_t nItemIndex) { if (m_pNotify) { if (nItemIndex == -1) { if (!m_bNotifyFlag) { m_bNotifyFlag = true; CFX_FloatRect rcRefresh = m_rcPlate; m_pNotify->IOnInvalidateRect(&rcRefresh); m_bNotifyFlag = false; } } else { if (!m_bNotifyFlag) { m_bNotifyFlag = true; CFX_FloatRect rcRefresh = GetItemRect(nItemIndex); rcRefresh.left -= 1.0f; rcRefresh.right += 1.0f; rcRefresh.bottom -= 1.0f; rcRefresh.top += 1.0f; m_pNotify->IOnInvalidateRect(&rcRefresh); m_bNotifyFlag = false; } } } } void CPWL_ListCtrl::SelectItems() { for (const auto& item : m_aSelItems) { if (item.second != CPLST_Select::NORMAL) SetMultipleSelect(item.first, item.second == CPLST_Select::SELECTING); } m_aSelItems.Done(); } void CPWL_ListCtrl::Select(int32_t nItemIndex) { if (!IsValid(nItemIndex)) return; if (IsMultipleSel()) { m_aSelItems.Add(nItemIndex); SelectItems(); } else { SetSingleSelect(nItemIndex); } } void CPWL_ListCtrl::Deselect(int32_t nItemIndex) { if (!IsItemSelected(nItemIndex)) return; SetMultipleSelect(nItemIndex, false); if (!IsMultipleSel()) m_nSelItem = -1; } bool CPWL_ListCtrl::IsItemVisible(int32_t nItemIndex) const { CFX_FloatRect rcPlate = m_rcPlate; CFX_FloatRect rcItem = GetItemRect(nItemIndex); return rcItem.bottom >= rcPlate.bottom && rcItem.top <= rcPlate.top; } void CPWL_ListCtrl::ScrollToListItem(int32_t nItemIndex) { if (!IsValid(nItemIndex)) return; CFX_FloatRect rcPlate = m_rcPlate; CFX_FloatRect rcItem = GetItemRectInternal(nItemIndex); CFX_FloatRect rcItemCtrl = GetItemRect(nItemIndex); if (IsFloatSmaller(rcItemCtrl.bottom, rcPlate.bottom)) { if (IsFloatSmaller(rcItemCtrl.top, rcPlate.top)) { SetScrollPosY(rcItem.bottom + rcPlate.Height()); } } else if (IsFloatBigger(rcItemCtrl.top, rcPlate.top)) { if (IsFloatBigger(rcItemCtrl.bottom, rcPlate.bottom)) { SetScrollPosY(rcItem.top); } } } void CPWL_ListCtrl::SetScrollInfo() { if (m_pNotify) { CFX_FloatRect rcPlate = m_rcPlate; CFX_FloatRect rcContent = GetContentRectInternal(); if (!m_bNotifyFlag) { m_bNotifyFlag = true; m_pNotify->IOnSetScrollInfoY(rcPlate.bottom, rcPlate.top, rcContent.bottom, rcContent.top, GetFirstHeight(), rcPlate.Height()); m_bNotifyFlag = false; } } } void CPWL_ListCtrl::SetScrollPos(const CFX_PointF& point) { SetScrollPosY(point.y); } void CPWL_ListCtrl::SetScrollPosY(float fy) { if (!IsFloatEqual(m_ptScrollPos.y, fy)) { CFX_FloatRect rcPlate = m_rcPlate; CFX_FloatRect rcContent = GetContentRectInternal(); if (rcPlate.Height() > rcContent.Height()) { fy = rcPlate.top; } else { if (IsFloatSmaller(fy - rcPlate.Height(), rcContent.bottom)) { fy = rcContent.bottom + rcPlate.Height(); } else if (IsFloatBigger(fy, rcContent.top)) { fy = rcContent.top; } } m_ptScrollPos.y = fy; InvalidateItem(-1); if (m_pNotify) { if (!m_bNotifyFlag) { m_bNotifyFlag = true; m_pNotify->IOnSetScrollPosY(fy); m_bNotifyFlag = false; } } } } CFX_FloatRect CPWL_ListCtrl::GetContentRectInternal() const { return InnerToOuter(m_rcContent); } CFX_FloatRect CPWL_ListCtrl::GetContentRect() const { return InToOut(GetContentRectInternal()); } void CPWL_ListCtrl::ReArrange(int32_t nItemIndex) { float fPosY = 0.0f; if (IsValid(nItemIndex - 1)) fPosY = m_ListItems[nItemIndex - 1]->GetRect().bottom; for (const auto& pListItem : m_ListItems) { float fListItemHeight = pListItem->GetItemHeight(); pListItem->SetRect( CFX_FloatRect(0.0f, fPosY + fListItemHeight, 0.0f, fPosY)); fPosY += fListItemHeight; } SetContentRect(CFX_FloatRect(0.0f, fPosY, 0.0f, 0.0f)); SetScrollInfo(); } void CPWL_ListCtrl::SetTopItem(int32_t nIndex) { if (IsValid(nIndex)) { CFX_FloatRect rcItem = GetItemRectInternal(nIndex); SetScrollPosY(rcItem.top); } } int32_t CPWL_ListCtrl::GetTopItem() const { int32_t nItemIndex = GetItemIndex(GetBTPoint()); if (!IsItemVisible(nItemIndex) && IsItemVisible(nItemIndex + 1)) nItemIndex += 1; return nItemIndex; } void CPWL_ListCtrl::Clear() { m_ListItems.clear(); InvalidateItem(-1); } void CPWL_ListCtrl::Cancel() { m_aSelItems.DeselectAll(); } int32_t CPWL_ListCtrl::GetItemIndex(const CFX_PointF& point) const { CFX_PointF pt = OuterToInner(OutToIn(point)); bool bFirst = true; bool bLast = true; for (const auto& pListItem : m_ListItems) { CFX_FloatRect rcListItem = pListItem->GetRect(); if (IsFloatBigger(pt.y, rcListItem.top)) bFirst = false; if (IsFloatSmaller(pt.y, rcListItem.bottom)) bLast = false; if (pt.y >= rcListItem.top && pt.y < rcListItem.bottom) return &pListItem - &m_ListItems.front(); } if (bFirst) return 0; if (bLast) return GetCount() - 1; return -1; } WideString CPWL_ListCtrl::GetText() const { if (IsMultipleSel()) return GetItemText(m_nCaretIndex); return GetItemText(m_nSelItem); } void CPWL_ListCtrl::AddItem(const WideString& str) { auto pListItem = pdfium::MakeUnique(); pListItem->SetFontMap(m_pFontMap.Get()); pListItem->SetFontSize(m_fFontSize); pListItem->SetText(str); m_ListItems.push_back(std::move(pListItem)); } CPWL_EditImpl* CPWL_ListCtrl::GetItemEdit(int32_t nIndex) const { if (!IsValid(nIndex)) return nullptr; return m_ListItems[nIndex]->GetEdit(); } int32_t CPWL_ListCtrl::GetCount() const { return pdfium::CollectionSize(m_ListItems); } float CPWL_ListCtrl::GetFirstHeight() const { if (m_ListItems.empty()) return 1.0f; return m_ListItems.front()->GetItemHeight(); } int32_t CPWL_ListCtrl::GetFirstSelected() const { int32_t i = 0; for (const auto& pListItem : m_ListItems) { if (pListItem->IsSelected()) return i; ++i; } return -1; } int32_t CPWL_ListCtrl::GetLastSelected() const { for (auto iter = m_ListItems.rbegin(); iter != m_ListItems.rend(); ++iter) { if ((*iter)->IsSelected()) return &*iter - &m_ListItems.front(); } return -1; } int32_t CPWL_ListCtrl::FindNext(int32_t nIndex, wchar_t nChar) const { int32_t nCircleIndex = nIndex; int32_t sz = GetCount(); for (int32_t i = 0; i < sz; i++) { nCircleIndex++; if (nCircleIndex >= sz) nCircleIndex = 0; if (Item* pListItem = m_ListItems[nCircleIndex].get()) { if (FXSYS_towupper(pListItem->GetFirstChar()) == FXSYS_towupper(nChar)) return nCircleIndex; } } return nCircleIndex; } bool CPWL_ListCtrl::IsItemSelected(int32_t nIndex) const { return IsValid(nIndex) && m_ListItems[nIndex]->IsSelected(); } void CPWL_ListCtrl::SetItemSelect(int32_t nIndex, bool bSelected) { if (IsValid(nIndex)) m_ListItems[nIndex]->SetSelect(bSelected); } bool CPWL_ListCtrl::IsValid(int32_t nItemIndex) const { return pdfium::IndexInBounds(m_ListItems, nItemIndex); } WideString CPWL_ListCtrl::GetItemText(int32_t nIndex) const { if (IsValid(nIndex)) return m_ListItems[nIndex]->GetText(); return WideString(); }