| // 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_wnd.h" |
| |
| #include <map> |
| #include <sstream> |
| #include <vector> |
| |
| #include "core/fxge/cfx_renderdevice.h" |
| #include "fpdfsdk/pwl/cpwl_scroll_bar.h" |
| #include "third_party/base/ptr_util.h" |
| #include "third_party/base/stl_util.h" |
| |
| namespace { |
| |
| constexpr float kDefaultFontSize = 9.0f; |
| constexpr int kInvalidationInflate = 2; |
| |
| } // namespace |
| |
| PWL_CREATEPARAM::PWL_CREATEPARAM() |
| : rcRectWnd(0, 0, 0, 0), |
| pSystemHandler(nullptr), |
| pFontMap(nullptr), |
| pProvider(nullptr), |
| pFocusHandler(nullptr), |
| dwFlags(0), |
| sBackgroundColor(), |
| pAttachedWidget(nullptr), |
| nBorderStyle(BorderStyle::SOLID), |
| dwBorderWidth(1), |
| sBorderColor(), |
| sTextColor(), |
| nTransparency(255), |
| fFontSize(kDefaultFontSize), |
| sDash(3, 0, 0), |
| pAttachedData(nullptr), |
| pParentWnd(nullptr), |
| pMsgControl(nullptr), |
| eCursorType(FXCT_ARROW) {} |
| |
| PWL_CREATEPARAM::PWL_CREATEPARAM(const PWL_CREATEPARAM& other) = default; |
| |
| class CPWL_MsgControl { |
| friend class CPWL_Wnd; |
| |
| public: |
| explicit CPWL_MsgControl(CPWL_Wnd* pWnd) { |
| m_pCreatedWnd = pWnd; |
| Default(); |
| } |
| |
| ~CPWL_MsgControl() { Default(); } |
| |
| void Default() { |
| m_aMousePath.clear(); |
| m_aKeyboardPath.clear(); |
| m_pMainMouseWnd = nullptr; |
| m_pMainKeyboardWnd = nullptr; |
| } |
| |
| bool IsWndCreated(const CPWL_Wnd* pWnd) const { |
| return m_pCreatedWnd == pWnd; |
| } |
| |
| bool IsMainCaptureMouse(const CPWL_Wnd* pWnd) const { |
| return pWnd == m_pMainMouseWnd; |
| } |
| |
| bool IsWndCaptureMouse(const CPWL_Wnd* pWnd) const { |
| return pWnd && pdfium::ContainsValue(m_aMousePath, pWnd); |
| } |
| |
| bool IsMainCaptureKeyboard(const CPWL_Wnd* pWnd) const { |
| return pWnd == m_pMainKeyboardWnd; |
| } |
| |
| bool IsWndCaptureKeyboard(const CPWL_Wnd* pWnd) const { |
| return pWnd && pdfium::ContainsValue(m_aKeyboardPath, pWnd); |
| } |
| |
| void SetFocus(CPWL_Wnd* pWnd) { |
| m_aKeyboardPath.clear(); |
| if (pWnd) { |
| m_pMainKeyboardWnd = pWnd; |
| CPWL_Wnd* pParent = pWnd; |
| while (pParent) { |
| m_aKeyboardPath.push_back(pParent); |
| pParent = pParent->GetParentWindow(); |
| } |
| pWnd->OnSetFocus(); |
| } |
| } |
| |
| void KillFocus() { |
| if (!m_aKeyboardPath.empty()) |
| if (CPWL_Wnd* pWnd = m_aKeyboardPath[0]) |
| pWnd->OnKillFocus(); |
| |
| m_pMainKeyboardWnd = nullptr; |
| m_aKeyboardPath.clear(); |
| } |
| |
| void SetCapture(CPWL_Wnd* pWnd) { |
| m_aMousePath.clear(); |
| if (pWnd) { |
| m_pMainMouseWnd = pWnd; |
| CPWL_Wnd* pParent = pWnd; |
| while (pParent) { |
| m_aMousePath.push_back(pParent); |
| pParent = pParent->GetParentWindow(); |
| } |
| } |
| } |
| |
| void ReleaseCapture() { |
| m_pMainMouseWnd = nullptr; |
| m_aMousePath.clear(); |
| } |
| |
| private: |
| std::vector<CPWL_Wnd*> m_aMousePath; |
| std::vector<CPWL_Wnd*> m_aKeyboardPath; |
| CFX_UnownedPtr<CPWL_Wnd> m_pCreatedWnd; |
| CFX_UnownedPtr<CPWL_Wnd> m_pMainMouseWnd; |
| CFX_UnownedPtr<CPWL_Wnd> m_pMainKeyboardWnd; |
| }; |
| |
| CPWL_Wnd::CPWL_Wnd() |
| : m_rcWindow(), |
| m_rcClip(), |
| m_bCreated(false), |
| m_bVisible(false), |
| m_bNotifying(false), |
| m_bEnabled(true) {} |
| |
| CPWL_Wnd::~CPWL_Wnd() { |
| ASSERT(!m_bCreated); |
| } |
| |
| CFX_ByteString CPWL_Wnd::GetClassName() const { |
| return "CPWL_Wnd"; |
| } |
| |
| void CPWL_Wnd::Create(const PWL_CREATEPARAM& cp) { |
| if (IsValid()) |
| return; |
| |
| m_sPrivateParam = cp; |
| OnCreate(m_sPrivateParam); |
| |
| m_sPrivateParam.rcRectWnd.Normalize(); |
| m_rcWindow = m_sPrivateParam.rcRectWnd; |
| m_rcClip = m_rcWindow; |
| if (!m_rcClip.IsEmpty()) { |
| m_rcClip.Inflate(1.0f, 1.0f); |
| m_rcClip.Normalize(); |
| } |
| CreateMsgControl(); |
| |
| if (m_sPrivateParam.pParentWnd) |
| m_sPrivateParam.pParentWnd->AddChild(this); |
| |
| PWL_CREATEPARAM ccp = m_sPrivateParam; |
| |
| ccp.dwFlags &= 0xFFFF0000L; // remove sub styles |
| CreateScrollBar(ccp); |
| CreateChildWnd(ccp); |
| |
| m_bVisible = HasFlag(PWS_VISIBLE); |
| OnCreated(); |
| |
| RePosChildWnd(); |
| m_bCreated = true; |
| } |
| |
| void CPWL_Wnd::OnCreate(PWL_CREATEPARAM& cp) {} |
| |
| void CPWL_Wnd::OnCreated() {} |
| |
| void CPWL_Wnd::OnDestroy() {} |
| |
| void CPWL_Wnd::InvalidateFocusHandler(IPWL_FocusHandler* handler) { |
| if (m_sPrivateParam.pFocusHandler == handler) |
| m_sPrivateParam.pFocusHandler = nullptr; |
| } |
| |
| void CPWL_Wnd::InvalidateProvider(IPWL_Provider* provider) { |
| if (m_sPrivateParam.pProvider.Get() == provider) |
| m_sPrivateParam.pProvider.Reset(); |
| } |
| |
| void CPWL_Wnd::Destroy() { |
| KillFocus(); |
| OnDestroy(); |
| if (m_bCreated) { |
| m_pVScrollBar = nullptr; |
| for (auto it = m_Children.rbegin(); it != m_Children.rend(); ++it) { |
| if (CPWL_Wnd* pChild = *it) { |
| *it = nullptr; |
| pChild->Destroy(); |
| delete pChild; |
| } |
| } |
| if (m_sPrivateParam.pParentWnd) |
| m_sPrivateParam.pParentWnd->RemoveChild(this); |
| |
| m_bCreated = false; |
| } |
| DestroyMsgControl(); |
| m_sPrivateParam.Reset(); |
| m_Children.clear(); |
| } |
| |
| void CPWL_Wnd::Move(const CFX_FloatRect& rcNew, bool bReset, bool bRefresh) { |
| if (!IsValid()) |
| return; |
| |
| CFX_FloatRect rcOld = GetWindowRect(); |
| m_rcWindow = rcNew; |
| m_rcWindow.Normalize(); |
| |
| if (bReset) { |
| if (rcOld.left != rcNew.left || rcOld.right != rcNew.right || |
| rcOld.top != rcNew.top || rcOld.bottom != rcNew.bottom) { |
| RePosChildWnd(); |
| } |
| } |
| if (bRefresh) |
| InvalidateRectMove(rcOld, rcNew); |
| |
| m_sPrivateParam.rcRectWnd = m_rcWindow; |
| } |
| |
| void CPWL_Wnd::InvalidateRectMove(const CFX_FloatRect& rcOld, |
| const CFX_FloatRect& rcNew) { |
| CFX_FloatRect rcUnion = rcOld; |
| rcUnion.Union(rcNew); |
| |
| InvalidateRect(&rcUnion); |
| } |
| |
| void CPWL_Wnd::DrawAppearance(CFX_RenderDevice* pDevice, |
| CFX_Matrix* pUser2Device) { |
| if (IsValid() && IsVisible()) { |
| DrawThisAppearance(pDevice, pUser2Device); |
| DrawChildAppearance(pDevice, pUser2Device); |
| } |
| } |
| |
| void CPWL_Wnd::DrawThisAppearance(CFX_RenderDevice* pDevice, |
| CFX_Matrix* pUser2Device) { |
| CFX_FloatRect rectWnd = GetWindowRect(); |
| if (rectWnd.IsEmpty()) |
| return; |
| |
| if (HasFlag(PWS_BACKGROUND)) { |
| float width = static_cast<float>(GetBorderWidth() + GetInnerBorderWidth()); |
| pDevice->DrawFillRect(pUser2Device, rectWnd.GetDeflated(width, width), |
| GetBackgroundColor(), GetTransparency()); |
| } |
| |
| if (HasFlag(PWS_BORDER)) { |
| pDevice->DrawBorder(pUser2Device, rectWnd, |
| static_cast<float>(GetBorderWidth()), GetBorderColor(), |
| GetBorderLeftTopColor(GetBorderStyle()), |
| GetBorderRightBottomColor(GetBorderStyle()), |
| GetBorderStyle(), GetTransparency()); |
| } |
| } |
| |
| void CPWL_Wnd::DrawChildAppearance(CFX_RenderDevice* pDevice, |
| CFX_Matrix* pUser2Device) { |
| for (CPWL_Wnd* pChild : m_Children) { |
| if (!pChild) |
| continue; |
| |
| CFX_Matrix mt = pChild->GetChildMatrix(); |
| if (mt.IsIdentity()) { |
| pChild->DrawAppearance(pDevice, pUser2Device); |
| } else { |
| mt.Concat(*pUser2Device); |
| pChild->DrawAppearance(pDevice, &mt); |
| } |
| } |
| } |
| |
| void CPWL_Wnd::InvalidateRect(CFX_FloatRect* pRect) { |
| if (!IsValid()) |
| return; |
| |
| CFX_FloatRect rcRefresh = pRect ? *pRect : GetWindowRect(); |
| |
| if (!HasFlag(PWS_NOREFRESHCLIP)) { |
| CFX_FloatRect rcClip = GetClipRect(); |
| if (!rcClip.IsEmpty()) { |
| rcRefresh.Intersect(rcClip); |
| } |
| } |
| |
| FX_RECT rcWin = PWLtoWnd(rcRefresh); |
| rcWin.left -= kInvalidationInflate; |
| rcWin.top -= kInvalidationInflate; |
| rcWin.right += kInvalidationInflate; |
| rcWin.bottom += kInvalidationInflate; |
| |
| if (CFX_SystemHandler* pSH = GetSystemHandler()) { |
| if (CPDFSDK_Widget* widget = static_cast<CPDFSDK_Widget*>( |
| m_sPrivateParam.pAttachedWidget.Get())) { |
| pSH->InvalidateRect(widget, rcWin); |
| } |
| } |
| } |
| |
| #define PWL_IMPLEMENT_KEY_METHOD(key_method_name) \ |
| bool CPWL_Wnd::key_method_name(uint16_t nChar, uint32_t nFlag) { \ |
| if (!IsValid() || !IsVisible() || !IsEnabled()) \ |
| return false; \ |
| if (!IsWndCaptureKeyboard(this)) \ |
| return false; \ |
| for (auto* pChild : m_Children) { \ |
| if (pChild && IsWndCaptureKeyboard(pChild)) \ |
| return pChild->key_method_name(nChar, nFlag); \ |
| } \ |
| return false; \ |
| } |
| |
| PWL_IMPLEMENT_KEY_METHOD(OnKeyDown) |
| PWL_IMPLEMENT_KEY_METHOD(OnChar) |
| #undef PWL_IMPLEMENT_KEY_METHOD |
| |
| #define PWL_IMPLEMENT_MOUSE_METHOD(mouse_method_name) \ |
| bool CPWL_Wnd::mouse_method_name(const CFX_PointF& point, uint32_t nFlag) { \ |
| if (!IsValid() || !IsVisible() || !IsEnabled()) \ |
| return false; \ |
| if (IsWndCaptureMouse(this)) { \ |
| for (auto* pChild : m_Children) { \ |
| if (pChild && IsWndCaptureMouse(pChild)) { \ |
| return pChild->mouse_method_name(pChild->ParentToChild(point), \ |
| nFlag); \ |
| } \ |
| } \ |
| SetCursor(); \ |
| return false; \ |
| } \ |
| for (auto* pChild : m_Children) { \ |
| if (pChild && pChild->WndHitTest(pChild->ParentToChild(point))) { \ |
| return pChild->mouse_method_name(pChild->ParentToChild(point), nFlag); \ |
| } \ |
| } \ |
| if (WndHitTest(point)) \ |
| SetCursor(); \ |
| return false; \ |
| } |
| |
| PWL_IMPLEMENT_MOUSE_METHOD(OnLButtonDblClk) |
| PWL_IMPLEMENT_MOUSE_METHOD(OnLButtonDown) |
| PWL_IMPLEMENT_MOUSE_METHOD(OnLButtonUp) |
| PWL_IMPLEMENT_MOUSE_METHOD(OnRButtonDown) |
| PWL_IMPLEMENT_MOUSE_METHOD(OnRButtonUp) |
| PWL_IMPLEMENT_MOUSE_METHOD(OnMouseMove) |
| #undef PWL_IMPLEMENT_MOUSE_METHOD |
| |
| CFX_WideString CPWL_Wnd::GetSelectedText() { |
| return CFX_WideString(); |
| } |
| |
| void CPWL_Wnd::ReplaceSelection(const CFX_WideString& text) {} |
| |
| bool CPWL_Wnd::OnMouseWheel(short zDelta, |
| const CFX_PointF& point, |
| uint32_t nFlag) { |
| if (!IsValid() || !IsVisible() || !IsEnabled()) |
| return false; |
| |
| SetCursor(); |
| if (!IsWndCaptureKeyboard(this)) |
| return false; |
| |
| for (auto* pChild : m_Children) { |
| if (pChild && IsWndCaptureKeyboard(pChild)) |
| return pChild->OnMouseWheel(zDelta, pChild->ParentToChild(point), nFlag); |
| } |
| return false; |
| } |
| |
| void CPWL_Wnd::AddChild(CPWL_Wnd* pWnd) { |
| m_Children.push_back(pWnd); |
| } |
| |
| void CPWL_Wnd::RemoveChild(CPWL_Wnd* pWnd) { |
| for (auto it = m_Children.rbegin(); it != m_Children.rend(); ++it) { |
| if (*it && *it == pWnd) { |
| m_Children.erase(std::next(it).base()); |
| break; |
| } |
| } |
| } |
| |
| void CPWL_Wnd::SetScrollInfo(const PWL_SCROLL_INFO& info) {} |
| |
| void CPWL_Wnd::SetScrollPosition(float pos) {} |
| |
| void CPWL_Wnd::ScrollWindowVertically(float pos) {} |
| |
| void CPWL_Wnd::NotifyLButtonDown(CPWL_Wnd* child, const CFX_PointF& pos) {} |
| |
| void CPWL_Wnd::NotifyLButtonUp(CPWL_Wnd* child, const CFX_PointF& pos) {} |
| |
| void CPWL_Wnd::NotifyMouseMove(CPWL_Wnd* child, const CFX_PointF& pos) {} |
| |
| bool CPWL_Wnd::IsValid() const { |
| return m_bCreated; |
| } |
| |
| const PWL_CREATEPARAM& CPWL_Wnd::GetCreationParam() const { |
| return m_sPrivateParam; |
| } |
| |
| CPWL_Wnd* CPWL_Wnd::GetParentWindow() const { |
| return m_sPrivateParam.pParentWnd; |
| } |
| |
| CFX_FloatRect CPWL_Wnd::GetWindowRect() const { |
| return m_rcWindow; |
| } |
| |
| CFX_FloatRect CPWL_Wnd::GetClientRect() const { |
| CFX_FloatRect rcWindow = GetWindowRect(); |
| |
| float width = static_cast<float>(GetBorderWidth() + GetInnerBorderWidth()); |
| CFX_FloatRect rcClient = rcWindow.GetDeflated(width, width); |
| if (CPWL_ScrollBar* pVSB = GetVScrollBar()) |
| rcClient.right -= pVSB->GetScrollBarWidth(); |
| |
| rcClient.Normalize(); |
| return rcWindow.Contains(rcClient) ? rcClient : CFX_FloatRect(); |
| } |
| |
| CFX_PointF CPWL_Wnd::GetCenterPoint() const { |
| CFX_FloatRect rcClient = GetClientRect(); |
| return CFX_PointF((rcClient.left + rcClient.right) * 0.5f, |
| (rcClient.top + rcClient.bottom) * 0.5f); |
| } |
| |
| bool CPWL_Wnd::HasFlag(uint32_t dwFlags) const { |
| return (m_sPrivateParam.dwFlags & dwFlags) != 0; |
| } |
| |
| void CPWL_Wnd::RemoveFlag(uint32_t dwFlags) { |
| m_sPrivateParam.dwFlags &= ~dwFlags; |
| } |
| |
| void CPWL_Wnd::AddFlag(uint32_t dwFlags) { |
| m_sPrivateParam.dwFlags |= dwFlags; |
| } |
| |
| CFX_Color CPWL_Wnd::GetBackgroundColor() const { |
| return m_sPrivateParam.sBackgroundColor; |
| } |
| |
| void CPWL_Wnd::SetBackgroundColor(const CFX_Color& color) { |
| m_sPrivateParam.sBackgroundColor = color; |
| } |
| |
| CFX_Color CPWL_Wnd::GetTextColor() const { |
| return m_sPrivateParam.sTextColor; |
| } |
| |
| BorderStyle CPWL_Wnd::GetBorderStyle() const { |
| return m_sPrivateParam.nBorderStyle; |
| } |
| |
| void CPWL_Wnd::SetBorderStyle(BorderStyle nBorderStyle) { |
| if (HasFlag(PWS_BORDER)) |
| m_sPrivateParam.nBorderStyle = nBorderStyle; |
| } |
| |
| int32_t CPWL_Wnd::GetBorderWidth() const { |
| return HasFlag(PWS_BORDER) ? m_sPrivateParam.dwBorderWidth : 0; |
| } |
| |
| int32_t CPWL_Wnd::GetInnerBorderWidth() const { |
| return 0; |
| } |
| |
| CFX_Color CPWL_Wnd::GetBorderColor() const { |
| return HasFlag(PWS_BORDER) ? m_sPrivateParam.sBorderColor : CFX_Color(); |
| } |
| |
| const CPWL_Dash& CPWL_Wnd::GetBorderDash() const { |
| return m_sPrivateParam.sDash; |
| } |
| |
| void* CPWL_Wnd::GetAttachedData() const { |
| return m_sPrivateParam.pAttachedData; |
| } |
| |
| CPWL_ScrollBar* CPWL_Wnd::GetVScrollBar() const { |
| return HasFlag(PWS_VSCROLL) ? m_pVScrollBar.Get() : nullptr; |
| } |
| |
| void CPWL_Wnd::CreateScrollBar(const PWL_CREATEPARAM& cp) { |
| CreateVScrollBar(cp); |
| } |
| |
| void CPWL_Wnd::CreateVScrollBar(const PWL_CREATEPARAM& cp) { |
| if (m_pVScrollBar || !HasFlag(PWS_VSCROLL)) |
| return; |
| |
| PWL_CREATEPARAM scp = cp; |
| |
| // flags |
| scp.dwFlags = |
| PWS_CHILD | PWS_BACKGROUND | PWS_AUTOTRANSPARENT | PWS_NOREFRESHCLIP; |
| |
| scp.pParentWnd = this; |
| scp.sBackgroundColor = PWL_DEFAULT_WHITECOLOR; |
| scp.eCursorType = FXCT_ARROW; |
| scp.nTransparency = PWL_SCROLLBAR_TRANSPARENCY; |
| |
| m_pVScrollBar = new CPWL_ScrollBar(SBT_VSCROLL); |
| m_pVScrollBar->Create(scp); |
| } |
| |
| void CPWL_Wnd::SetCapture() { |
| if (CPWL_MsgControl* pMsgCtrl = GetMsgControl()) |
| pMsgCtrl->SetCapture(this); |
| } |
| |
| void CPWL_Wnd::ReleaseCapture() { |
| for (auto* pChild : m_Children) { |
| if (pChild) |
| pChild->ReleaseCapture(); |
| } |
| if (CPWL_MsgControl* pMsgCtrl = GetMsgControl()) |
| pMsgCtrl->ReleaseCapture(); |
| } |
| |
| void CPWL_Wnd::SetFocus() { |
| if (CPWL_MsgControl* pMsgCtrl = GetMsgControl()) { |
| if (!pMsgCtrl->IsMainCaptureKeyboard(this)) |
| pMsgCtrl->KillFocus(); |
| pMsgCtrl->SetFocus(this); |
| } |
| } |
| |
| void CPWL_Wnd::KillFocus() { |
| if (CPWL_MsgControl* pMsgCtrl = GetMsgControl()) { |
| if (pMsgCtrl->IsWndCaptureKeyboard(this)) |
| pMsgCtrl->KillFocus(); |
| } |
| } |
| |
| void CPWL_Wnd::OnSetFocus() {} |
| |
| void CPWL_Wnd::OnKillFocus() {} |
| |
| bool CPWL_Wnd::WndHitTest(const CFX_PointF& point) const { |
| return IsValid() && IsVisible() && GetWindowRect().Contains(point); |
| } |
| |
| bool CPWL_Wnd::ClientHitTest(const CFX_PointF& point) const { |
| return IsValid() && IsVisible() && GetClientRect().Contains(point); |
| } |
| |
| const CPWL_Wnd* CPWL_Wnd::GetRootWnd() const { |
| auto* pParent = m_sPrivateParam.pParentWnd; |
| return pParent ? pParent->GetRootWnd() : this; |
| } |
| |
| void CPWL_Wnd::SetVisible(bool bVisible) { |
| if (!IsValid()) |
| return; |
| |
| for (auto* pChild : m_Children) { |
| if (pChild) |
| pChild->SetVisible(bVisible); |
| } |
| if (bVisible != m_bVisible) { |
| m_bVisible = bVisible; |
| RePosChildWnd(); |
| InvalidateRect(); |
| } |
| } |
| |
| void CPWL_Wnd::SetClipRect(const CFX_FloatRect& rect) { |
| m_rcClip = rect; |
| m_rcClip.Normalize(); |
| } |
| |
| const CFX_FloatRect& CPWL_Wnd::GetClipRect() const { |
| return m_rcClip; |
| } |
| |
| bool CPWL_Wnd::IsReadOnly() const { |
| return HasFlag(PWS_READONLY); |
| } |
| |
| void CPWL_Wnd::RePosChildWnd() { |
| CPWL_ScrollBar* pVSB = GetVScrollBar(); |
| if (!pVSB) |
| return; |
| |
| CFX_FloatRect rcContent = GetWindowRect(); |
| if (!rcContent.IsEmpty()) { |
| float width = static_cast<float>(GetBorderWidth() + GetInnerBorderWidth()); |
| rcContent.Deflate(width, width); |
| rcContent.Normalize(); |
| } |
| CFX_FloatRect rcVScroll = |
| CFX_FloatRect(rcContent.right - PWL_SCROLLBAR_WIDTH, rcContent.bottom, |
| rcContent.right - 1.0f, rcContent.top); |
| pVSB->Move(rcVScroll, true, false); |
| } |
| |
| void CPWL_Wnd::CreateChildWnd(const PWL_CREATEPARAM& cp) {} |
| |
| void CPWL_Wnd::SetCursor() { |
| if (IsValid()) { |
| if (CFX_SystemHandler* pSH = GetSystemHandler()) { |
| int32_t nCursorType = GetCreationParam().eCursorType; |
| pSH->SetCursor(nCursorType); |
| } |
| } |
| } |
| |
| void CPWL_Wnd::CreateMsgControl() { |
| if (!m_sPrivateParam.pMsgControl) |
| m_sPrivateParam.pMsgControl = new CPWL_MsgControl(this); |
| } |
| |
| void CPWL_Wnd::DestroyMsgControl() { |
| CPWL_MsgControl* pMsgControl = GetMsgControl(); |
| if (pMsgControl && pMsgControl->IsWndCreated(this)) |
| delete pMsgControl; |
| } |
| |
| CPWL_MsgControl* CPWL_Wnd::GetMsgControl() const { |
| return m_sPrivateParam.pMsgControl; |
| } |
| |
| bool CPWL_Wnd::IsCaptureMouse() const { |
| return IsWndCaptureMouse(this); |
| } |
| |
| bool CPWL_Wnd::IsWndCaptureMouse(const CPWL_Wnd* pWnd) const { |
| CPWL_MsgControl* pCtrl = GetMsgControl(); |
| return pCtrl ? pCtrl->IsWndCaptureMouse(pWnd) : false; |
| } |
| |
| bool CPWL_Wnd::IsWndCaptureKeyboard(const CPWL_Wnd* pWnd) const { |
| CPWL_MsgControl* pCtrl = GetMsgControl(); |
| return pCtrl ? pCtrl->IsWndCaptureKeyboard(pWnd) : false; |
| } |
| |
| bool CPWL_Wnd::IsFocused() const { |
| CPWL_MsgControl* pCtrl = GetMsgControl(); |
| return pCtrl ? pCtrl->IsMainCaptureKeyboard(this) : false; |
| } |
| |
| CFX_FloatRect CPWL_Wnd::GetFocusRect() const { |
| CFX_FloatRect rect = GetWindowRect(); |
| if (!rect.IsEmpty()) { |
| rect.Inflate(1.0f, 1.0f); |
| rect.Normalize(); |
| } |
| return rect; |
| } |
| |
| float CPWL_Wnd::GetFontSize() const { |
| return m_sPrivateParam.fFontSize; |
| } |
| |
| void CPWL_Wnd::SetFontSize(float fFontSize) { |
| m_sPrivateParam.fFontSize = fFontSize; |
| } |
| |
| CFX_SystemHandler* CPWL_Wnd::GetSystemHandler() const { |
| return m_sPrivateParam.pSystemHandler; |
| } |
| |
| IPWL_FocusHandler* CPWL_Wnd::GetFocusHandler() const { |
| return m_sPrivateParam.pFocusHandler; |
| } |
| |
| IPWL_Provider* CPWL_Wnd::GetProvider() const { |
| return m_sPrivateParam.pProvider.Get(); |
| } |
| |
| IPVT_FontMap* CPWL_Wnd::GetFontMap() const { |
| return m_sPrivateParam.pFontMap; |
| } |
| |
| CFX_Color CPWL_Wnd::GetBorderLeftTopColor(BorderStyle nBorderStyle) const { |
| switch (nBorderStyle) { |
| case BorderStyle::BEVELED: |
| return CFX_Color(COLORTYPE_GRAY, 1); |
| case BorderStyle::INSET: |
| return CFX_Color(COLORTYPE_GRAY, 0.5f); |
| default: |
| return CFX_Color(); |
| } |
| } |
| |
| CFX_Color CPWL_Wnd::GetBorderRightBottomColor(BorderStyle nBorderStyle) const { |
| switch (nBorderStyle) { |
| case BorderStyle::BEVELED: |
| return GetBackgroundColor() / 2.0f; |
| case BorderStyle::INSET: |
| return CFX_Color(COLORTYPE_GRAY, 0.75f); |
| default: |
| return CFX_Color(); |
| } |
| } |
| |
| int32_t CPWL_Wnd::GetTransparency() { |
| return m_sPrivateParam.nTransparency; |
| } |
| |
| void CPWL_Wnd::SetTransparency(int32_t nTransparency) { |
| for (auto* pChild : m_Children) { |
| if (pChild) |
| pChild->SetTransparency(nTransparency); |
| } |
| m_sPrivateParam.nTransparency = nTransparency; |
| } |
| |
| CFX_Matrix CPWL_Wnd::GetWindowMatrix() const { |
| CFX_Matrix mt = GetChildToRoot(); |
| if (IPWL_Provider* pProvider = GetProvider()) |
| mt.Concat(pProvider->GetWindowMatrix(GetAttachedData())); |
| return mt; |
| } |
| |
| FX_RECT CPWL_Wnd::PWLtoWnd(const CFX_FloatRect& rect) const { |
| CFX_FloatRect rcTemp = rect; |
| CFX_Matrix mt = GetWindowMatrix(); |
| mt.TransformRect(rcTemp); |
| return FX_RECT((int32_t)(rcTemp.left + 0.5), (int32_t)(rcTemp.bottom + 0.5), |
| (int32_t)(rcTemp.right + 0.5), (int32_t)(rcTemp.top + 0.5)); |
| } |
| |
| CFX_PointF CPWL_Wnd::ParentToChild(const CFX_PointF& point) const { |
| CFX_Matrix mt = GetChildMatrix(); |
| if (mt.IsIdentity()) |
| return point; |
| |
| CFX_Matrix inverse = mt.GetInverse(); |
| if (!inverse.IsIdentity()) |
| mt = inverse; |
| return mt.Transform(point); |
| } |
| |
| CFX_FloatRect CPWL_Wnd::ParentToChild(const CFX_FloatRect& rect) const { |
| CFX_Matrix mt = GetChildMatrix(); |
| if (mt.IsIdentity()) |
| return rect; |
| |
| CFX_Matrix inverse = mt.GetInverse(); |
| if (!inverse.IsIdentity()) |
| mt = inverse; |
| CFX_FloatRect rc = rect; |
| mt.TransformRect(rc); |
| return rc; |
| } |
| |
| CFX_Matrix CPWL_Wnd::GetChildToRoot() const { |
| CFX_Matrix mt; |
| if (HasFlag(PWS_CHILD)) { |
| const CPWL_Wnd* pParent = this; |
| while (pParent) { |
| mt.Concat(pParent->GetChildMatrix()); |
| pParent = pParent->GetParentWindow(); |
| } |
| } |
| return mt; |
| } |
| |
| CFX_Matrix CPWL_Wnd::GetChildMatrix() const { |
| return HasFlag(PWS_CHILD) ? m_sPrivateParam.mtChild : CFX_Matrix(); |
| } |
| |
| void CPWL_Wnd::SetChildMatrix(const CFX_Matrix& mt) { |
| m_sPrivateParam.mtChild = mt; |
| } |
| |
| const CPWL_Wnd* CPWL_Wnd::GetFocused() const { |
| CPWL_MsgControl* pMsgCtrl = GetMsgControl(); |
| return pMsgCtrl ? pMsgCtrl->m_pMainKeyboardWnd.Get() : nullptr; |
| } |
| |
| void CPWL_Wnd::EnableWindow(bool bEnable) { |
| if (m_bEnabled == bEnable) |
| return; |
| |
| for (auto* pChild : m_Children) { |
| if (pChild) |
| pChild->EnableWindow(bEnable); |
| } |
| m_bEnabled = bEnable; |
| } |