| Index: xfa/fwl/core/ifwl_combolist.cpp
|
| diff --git a/xfa/fwl/core/ifwl_combolist.cpp b/xfa/fwl/core/ifwl_combolist.cpp
|
| new file mode 100644
|
| index 0000000000000000000000000000000000000000..9090622b10fa696b528d0ac0574d9bd34396a587
|
| --- /dev/null
|
| +++ b/xfa/fwl/core/ifwl_combolist.cpp
|
| @@ -0,0 +1,298 @@
|
| +// Copyright 2016 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/fwl/core/ifwl_combolist.h"
|
| +
|
| +#include "xfa/fwl/core/ifwl_combobox.h"
|
| +#include "xfa/fwl/core/ifwl_comboedit.h"
|
| +
|
| +// static
|
| +IFWL_ComboList* IFWL_ComboList::Create(
|
| + const CFWL_WidgetImpProperties& properties,
|
| + IFWL_Widget* pOuter) {
|
| + return new IFWL_ComboList(properties, pOuter);
|
| +}
|
| +
|
| +IFWL_ComboList::IFWL_ComboList(const CFWL_WidgetImpProperties& properties,
|
| + IFWL_Widget* pOuter)
|
| + : IFWL_ListBox(properties, pOuter), m_bNotifyOwner(TRUE) {
|
| + ASSERT(pOuter);
|
| +}
|
| +
|
| +FWL_Error IFWL_ComboList::Initialize() {
|
| + if (IFWL_ListBox::Initialize() != FWL_Error::Succeeded)
|
| + return FWL_Error::Indefinite;
|
| + delete m_pDelegate;
|
| + m_pDelegate = new CFWL_ComboListImpDelegate(this);
|
| + return FWL_Error::Succeeded;
|
| +}
|
| +
|
| +FWL_Error IFWL_ComboList::Finalize() {
|
| + delete m_pDelegate;
|
| + m_pDelegate = nullptr;
|
| + return IFWL_ListBox::Finalize();
|
| +}
|
| +
|
| +int32_t IFWL_ComboList::MatchItem(const CFX_WideString& wsMatch) {
|
| + if (wsMatch.IsEmpty()) {
|
| + return -1;
|
| + }
|
| + if (!m_pProperties->m_pDataProvider)
|
| + return -1;
|
| + IFWL_ListBoxDP* pData =
|
| + static_cast<IFWL_ListBoxDP*>(m_pProperties->m_pDataProvider);
|
| + int32_t iCount = pData->CountItems(this);
|
| + for (int32_t i = 0; i < iCount; i++) {
|
| + IFWL_ListItem* hItem = pData->GetItem(this, i);
|
| + CFX_WideString wsText;
|
| + pData->GetItemText(this, hItem, wsText);
|
| + FX_STRSIZE pos = wsText.Find(wsMatch.c_str());
|
| + if (!pos) {
|
| + return i;
|
| + }
|
| + }
|
| + return -1;
|
| +}
|
| +
|
| +void IFWL_ComboList::ChangeSelected(int32_t iSel) {
|
| + if (!m_pProperties->m_pDataProvider)
|
| + return;
|
| + IFWL_ListBoxDP* pData =
|
| + static_cast<IFWL_ListBoxDP*>(m_pProperties->m_pDataProvider);
|
| + IFWL_ListItem* hItem = pData->GetItem(this, iSel);
|
| + CFX_RectF rtInvalidate;
|
| + rtInvalidate.Reset();
|
| + IFWL_ListItem* hOld = GetSelItem(0);
|
| + int32_t iOld = pData->GetItemIndex(this, hOld);
|
| + if (iOld == iSel) {
|
| + return;
|
| + } else if (iOld > -1) {
|
| + GetItemRect(iOld, rtInvalidate);
|
| + SetSelItem(hOld, FALSE);
|
| + }
|
| + if (hItem) {
|
| + CFX_RectF rect;
|
| + GetItemRect(iSel, rect);
|
| + rtInvalidate.Union(rect);
|
| + IFWL_ListItem* hSel = pData->GetItem(this, iSel);
|
| + SetSelItem(hSel, TRUE);
|
| + }
|
| + if (!rtInvalidate.IsEmpty()) {
|
| + Repaint(&rtInvalidate);
|
| + }
|
| +}
|
| +
|
| +int32_t IFWL_ComboList::CountItems() {
|
| + IFWL_ListBoxDP* pData =
|
| + static_cast<IFWL_ListBoxDP*>(m_pProperties->m_pDataProvider);
|
| + return pData ? pData->CountItems(this) : 0;
|
| +}
|
| +
|
| +void IFWL_ComboList::GetItemRect(int32_t nIndex, CFX_RectF& rtItem) {
|
| + IFWL_ListBoxDP* pData =
|
| + static_cast<IFWL_ListBoxDP*>(m_pProperties->m_pDataProvider);
|
| + IFWL_ListItem* hItem = pData->GetItem(this, nIndex);
|
| + pData->GetItemRect(this, hItem, rtItem);
|
| +}
|
| +
|
| +void IFWL_ComboList::ClientToOuter(FX_FLOAT& fx, FX_FLOAT& fy) {
|
| + fx += m_pProperties->m_rtWidget.left, fy += m_pProperties->m_rtWidget.top;
|
| + IFWL_Widget* pOwner = GetOwner();
|
| + if (!pOwner)
|
| + return;
|
| + pOwner->TransformTo(m_pOuter, fx, fy);
|
| +}
|
| +
|
| +void IFWL_ComboList::SetFocus(FX_BOOL bSet) {
|
| + IFWL_Widget::SetFocus(bSet);
|
| +}
|
| +
|
| +CFWL_ComboListImpDelegate::CFWL_ComboListImpDelegate(IFWL_ComboList* pOwner)
|
| + : CFWL_ListBoxImpDelegate(pOwner), m_pOwner(pOwner) {}
|
| +
|
| +void CFWL_ComboListImpDelegate::OnProcessMessage(CFWL_Message* pMessage) {
|
| + if (!pMessage)
|
| + return;
|
| +
|
| + CFWL_MessageType dwHashCode = pMessage->GetClassID();
|
| + FX_BOOL backDefault = TRUE;
|
| + if (dwHashCode == CFWL_MessageType::SetFocus ||
|
| + dwHashCode == CFWL_MessageType::KillFocus) {
|
| + OnDropListFocusChanged(pMessage, dwHashCode == CFWL_MessageType::SetFocus);
|
| + } else if (dwHashCode == CFWL_MessageType::Mouse) {
|
| + CFWL_MsgMouse* pMsg = static_cast<CFWL_MsgMouse*>(pMessage);
|
| + if (m_pOwner->IsShowScrollBar(TRUE) && m_pOwner->m_pVertScrollBar) {
|
| + CFX_RectF rect;
|
| + m_pOwner->m_pVertScrollBar->GetWidgetRect(rect);
|
| + if (rect.Contains(pMsg->m_fx, pMsg->m_fy)) {
|
| + pMsg->m_fx -= rect.left;
|
| + pMsg->m_fy -= rect.top;
|
| + IFWL_WidgetDelegate* pDelegate =
|
| + m_pOwner->m_pVertScrollBar->SetDelegate(nullptr);
|
| + pDelegate->OnProcessMessage(pMsg);
|
| + return;
|
| + }
|
| + }
|
| + switch (pMsg->m_dwCmd) {
|
| + case FWL_MouseCommand::Move: {
|
| + backDefault = FALSE;
|
| + OnDropListMouseMove(pMsg);
|
| + break;
|
| + }
|
| + case FWL_MouseCommand::LeftButtonDown: {
|
| + backDefault = FALSE;
|
| + OnDropListLButtonDown(pMsg);
|
| + break;
|
| + }
|
| + case FWL_MouseCommand::LeftButtonUp: {
|
| + backDefault = FALSE;
|
| + OnDropListLButtonUp(pMsg);
|
| + break;
|
| + }
|
| + default:
|
| + break;
|
| + }
|
| + } else if (dwHashCode == CFWL_MessageType::Key) {
|
| + backDefault = !OnDropListKey(static_cast<CFWL_MsgKey*>(pMessage));
|
| + }
|
| + if (backDefault)
|
| + CFWL_ListBoxImpDelegate::OnProcessMessage(pMessage);
|
| +}
|
| +
|
| +void CFWL_ComboListImpDelegate::OnDropListFocusChanged(CFWL_Message* pMsg,
|
| + FX_BOOL bSet) {
|
| + if (!bSet) {
|
| + CFWL_MsgKillFocus* pKill = static_cast<CFWL_MsgKillFocus*>(pMsg);
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + if (pKill->m_pSetFocus == m_pOwner->m_pOuter ||
|
| + pKill->m_pSetFocus == pOuter->m_pEdit.get()) {
|
| + pOuter->ShowDropList(FALSE);
|
| + }
|
| + }
|
| +}
|
| +
|
| +int32_t CFWL_ComboListImpDelegate::OnDropListMouseMove(CFWL_MsgMouse* pMsg) {
|
| + if (m_pOwner->m_rtClient.Contains(pMsg->m_fx, pMsg->m_fy)) {
|
| + if (m_pOwner->m_bNotifyOwner) {
|
| + m_pOwner->m_bNotifyOwner = FALSE;
|
| + }
|
| + if (m_pOwner->IsShowScrollBar(TRUE) && m_pOwner->m_pVertScrollBar) {
|
| + CFX_RectF rect;
|
| + m_pOwner->m_pVertScrollBar->GetWidgetRect(rect);
|
| + if (rect.Contains(pMsg->m_fx, pMsg->m_fy)) {
|
| + return 1;
|
| + }
|
| + }
|
| + IFWL_ListItem* hItem = m_pOwner->GetItemAtPoint(pMsg->m_fx, pMsg->m_fy);
|
| + if (hItem) {
|
| + if (!m_pOwner->m_pProperties->m_pDataProvider)
|
| + return 0;
|
| + IFWL_ListBoxDP* pData = static_cast<IFWL_ListBoxDP*>(
|
| + m_pOwner->m_pProperties->m_pDataProvider);
|
| + int32_t iSel = pData->GetItemIndex(m_pOwner, hItem);
|
| + CFWL_EvtCmbHoverChanged event;
|
| + event.m_pSrcTarget = m_pOwner->m_pOuter;
|
| + event.m_iCurHover = iSel;
|
| + m_pOwner->DispatchEvent(&event);
|
| + m_pOwner->ChangeSelected(iSel);
|
| + }
|
| + } else if (m_pOwner->m_bNotifyOwner) {
|
| + m_pOwner->ClientToOuter(pMsg->m_fx, pMsg->m_fy);
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + pOuter->m_pDelegate->OnProcessMessage(pMsg);
|
| + }
|
| + return 1;
|
| +}
|
| +
|
| +int32_t CFWL_ComboListImpDelegate::OnDropListLButtonDown(CFWL_MsgMouse* pMsg) {
|
| + if (m_pOwner->m_rtClient.Contains(pMsg->m_fx, pMsg->m_fy)) {
|
| + return 0;
|
| + }
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + pOuter->ShowDropList(FALSE);
|
| + return 1;
|
| +}
|
| +
|
| +int32_t CFWL_ComboListImpDelegate::OnDropListLButtonUp(CFWL_MsgMouse* pMsg) {
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + if (m_pOwner->m_bNotifyOwner) {
|
| + m_pOwner->ClientToOuter(pMsg->m_fx, pMsg->m_fy);
|
| + pOuter->m_pDelegate->OnProcessMessage(pMsg);
|
| + } else {
|
| + if (m_pOwner->IsShowScrollBar(TRUE) && m_pOwner->m_pVertScrollBar) {
|
| + CFX_RectF rect;
|
| + m_pOwner->m_pVertScrollBar->GetWidgetRect(rect);
|
| + if (rect.Contains(pMsg->m_fx, pMsg->m_fy)) {
|
| + return 1;
|
| + }
|
| + }
|
| + pOuter->ShowDropList(FALSE);
|
| + IFWL_ListItem* hItem = m_pOwner->GetItemAtPoint(pMsg->m_fx, pMsg->m_fy);
|
| + if (hItem) {
|
| + pOuter->ProcessSelChanged(TRUE);
|
| + }
|
| + }
|
| + return 1;
|
| +}
|
| +
|
| +int32_t CFWL_ComboListImpDelegate::OnDropListKey(CFWL_MsgKey* pKey) {
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + FX_BOOL bPropagate = FALSE;
|
| + if (pKey->m_dwCmd == FWL_KeyCommand::KeyDown) {
|
| + uint32_t dwKeyCode = pKey->m_dwKeyCode;
|
| + switch (dwKeyCode) {
|
| + case FWL_VKEY_Return:
|
| + case FWL_VKEY_Escape: {
|
| + pOuter->ShowDropList(FALSE);
|
| + return 1;
|
| + }
|
| + case FWL_VKEY_Up:
|
| + case FWL_VKEY_Down: {
|
| + OnDropListKeyDown(pKey);
|
| + pOuter->SetDelegate(nullptr);
|
| + pOuter->ProcessSelChanged(FALSE);
|
| + return 1;
|
| + }
|
| + default: { bPropagate = TRUE; }
|
| + }
|
| + } else if (pKey->m_dwCmd == FWL_KeyCommand::Char) {
|
| + bPropagate = TRUE;
|
| + }
|
| + if (bPropagate) {
|
| + pKey->m_pDstTarget = m_pOwner->m_pOuter;
|
| + pOuter->m_pDelegate->OnProcessMessage(pKey);
|
| + return 1;
|
| + }
|
| + return 0;
|
| +}
|
| +
|
| +void CFWL_ComboListImpDelegate::OnDropListKeyDown(CFWL_MsgKey* pKey) {
|
| + uint32_t dwKeyCode = pKey->m_dwKeyCode;
|
| + switch (dwKeyCode) {
|
| + case FWL_VKEY_Up:
|
| + case FWL_VKEY_Down:
|
| + case FWL_VKEY_Home:
|
| + case FWL_VKEY_End: {
|
| + IFWL_ComboBox* pOuter = static_cast<IFWL_ComboBox*>(m_pOwner->m_pOuter);
|
| + IFWL_ListBoxDP* pData = static_cast<IFWL_ListBoxDP*>(
|
| + m_pOwner->m_pProperties->m_pDataProvider);
|
| + IFWL_ListItem* hItem = pData->GetItem(m_pOwner, pOuter->m_iCurSel);
|
| + hItem = m_pOwner->GetItem(hItem, dwKeyCode);
|
| + if (!hItem) {
|
| + break;
|
| + }
|
| + m_pOwner->SetSelection(hItem, hItem, TRUE);
|
| + m_pOwner->ScrollToVisible(hItem);
|
| + CFX_RectF rtInvalidate;
|
| + rtInvalidate.Set(0, 0, m_pOwner->m_pProperties->m_rtWidget.width,
|
| + m_pOwner->m_pProperties->m_rtWidget.height);
|
| + m_pOwner->Repaint(&rtInvalidate);
|
| + break;
|
| + }
|
| + default:
|
| + break;
|
| + }
|
| +}
|
|
|