// 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_edit_ctrl.h" #include #include "core/fpdfdoc/cpvt_word.h" #include "core/fxge/fx_font.h" #include "fpdfsdk/pwl/cpwl_caret.h" #include "fpdfsdk/pwl/cpwl_edit_impl.h" #include "fpdfsdk/pwl/cpwl_scroll_bar.h" #include "fpdfsdk/pwl/cpwl_wnd.h" #include "public/fpdf_fwlevent.h" #include "third_party/base/ptr_util.h" CPWL_EditCtrl::CPWL_EditCtrl( const CreateParams& cp, std::unique_ptr pAttachedData) : CPWL_Wnd(cp, std::move(pAttachedData)), m_pEdit(pdfium::MakeUnique()) { GetCreationParams()->eCursorType = FXCT_VBEAM; } CPWL_EditCtrl::~CPWL_EditCtrl() = default; void CPWL_EditCtrl::OnCreated() { SetFontSize(GetCreationParams()->fFontSize); m_pEdit->SetFontMap(GetFontMap()); m_pEdit->SetNotify(this); m_pEdit->Initialize(); } bool CPWL_EditCtrl::IsWndHorV() const { CFX_Matrix mt = GetWindowMatrix(); return mt.Transform(CFX_PointF(1, 1)).y == mt.Transform(CFX_PointF(0, 1)).y; } void CPWL_EditCtrl::SetCursor() { if (IsValid()) GetSystemHandler()->SetCursor(IsWndHorV() ? FXCT_VBEAM : FXCT_HBEAM); } WideString CPWL_EditCtrl::GetSelectedText() { return m_pEdit->GetSelectedText(); } void CPWL_EditCtrl::ReplaceSelection(const WideString& text) { m_pEdit->ReplaceSelection(text); } bool CPWL_EditCtrl::RePosChildWnd() { m_pEdit->SetPlateRect(GetClientRect()); return true; } void CPWL_EditCtrl::SetScrollInfo(const PWL_SCROLL_INFO& info) { if (CPWL_Wnd* pChild = GetVScrollBar()) pChild->SetScrollInfo(info); } void CPWL_EditCtrl::SetScrollPosition(float pos) { if (CPWL_Wnd* pChild = GetVScrollBar()) pChild->SetScrollPosition(pos); } void CPWL_EditCtrl::ScrollWindowVertically(float pos) { m_pEdit->SetScrollPos(CFX_PointF(m_pEdit->GetScrollPos().x, pos)); } void CPWL_EditCtrl::CreateChildWnd(const CreateParams& cp) { if (!IsReadOnly()) CreateEditCaret(cp); } void CPWL_EditCtrl::CreateEditCaret(const CreateParams& cp) { if (m_pEditCaret) return; CreateParams ecp = cp; ecp.dwFlags = PWS_CHILD | PWS_NOREFRESHCLIP; ecp.dwBorderWidth = 0; ecp.nBorderStyle = BorderStyle::SOLID; ecp.rcRectWnd = CFX_FloatRect(); auto pCaret = pdfium::MakeUnique(ecp, CloneAttachedData()); m_pEditCaret = pCaret.get(); m_pEditCaret->SetInvalidRect(GetClientRect()); AddChild(std::move(pCaret)); m_pEditCaret->Realize(); } void CPWL_EditCtrl::SetFontSize(float fFontSize) { m_pEdit->SetFontSize(fFontSize); } float CPWL_EditCtrl::GetFontSize() const { return m_pEdit->GetFontSize(); } bool CPWL_EditCtrl::OnKeyDown(uint16_t nChar, uint32_t nFlag) { if (m_bMouseDown) return true; bool bRet = CPWL_Wnd::OnKeyDown(nChar, nFlag); // FILTER switch (nChar) { default: return false; case FWL_VKEY_Delete: case FWL_VKEY_Up: case FWL_VKEY_Down: case FWL_VKEY_Left: case FWL_VKEY_Right: case FWL_VKEY_Home: case FWL_VKEY_End: case FWL_VKEY_Insert: case 'C': case 'V': case 'X': case 'A': case 'Z': case 'c': case 'v': case 'x': case 'a': case 'z': break; } if (nChar == FWL_VKEY_Delete && m_pEdit->IsSelected()) nChar = FWL_VKEY_Unknown; switch (nChar) { case FWL_VKEY_Delete: Delete(); return true; case FWL_VKEY_Insert: if (IsSHIFTpressed(nFlag)) PasteText(); return true; case FWL_VKEY_Up: m_pEdit->OnVK_UP(IsSHIFTpressed(nFlag), false); return true; case FWL_VKEY_Down: m_pEdit->OnVK_DOWN(IsSHIFTpressed(nFlag), false); return true; case FWL_VKEY_Left: m_pEdit->OnVK_LEFT(IsSHIFTpressed(nFlag), false); return true; case FWL_VKEY_Right: m_pEdit->OnVK_RIGHT(IsSHIFTpressed(nFlag), false); return true; case FWL_VKEY_Home: m_pEdit->OnVK_HOME(IsSHIFTpressed(nFlag), IsCTRLpressed(nFlag)); return true; case FWL_VKEY_End: m_pEdit->OnVK_END(IsSHIFTpressed(nFlag), IsCTRLpressed(nFlag)); return true; case FWL_VKEY_Unknown: if (!IsSHIFTpressed(nFlag)) ClearSelection(); else CutText(); return true; default: break; } return bRet; } bool CPWL_EditCtrl::OnChar(uint16_t nChar, uint32_t nFlag) { if (m_bMouseDown) return true; CPWL_Wnd::OnChar(nChar, nFlag); // FILTER switch (nChar) { case 0x0A: case 0x1B: return false; default: break; } bool bCtrl = IsCTRLpressed(nFlag); bool bAlt = IsALTpressed(nFlag); bool bShift = IsSHIFTpressed(nFlag); uint16_t word = nChar; if (bCtrl && !bAlt) { switch (nChar) { case 'C' - 'A' + 1: CopyText(); return true; case 'V' - 'A' + 1: PasteText(); return true; case 'X' - 'A' + 1: CutText(); return true; case 'A' - 'A' + 1: SelectAll(); return true; case 'Z' - 'A' + 1: if (bShift) Redo(); else Undo(); return true; default: if (nChar < 32) return false; } } if (IsReadOnly()) return true; if (m_pEdit->IsSelected() && word == FWL_VKEY_Back) word = FWL_VKEY_Unknown; ClearSelection(); switch (word) { case FWL_VKEY_Back: Backspace(); break; case FWL_VKEY_Return: InsertReturn(); break; case FWL_VKEY_Unknown: break; default: InsertWord(word, GetCharSet()); break; } return true; } bool CPWL_EditCtrl::OnLButtonDown(const CFX_PointF& point, uint32_t nFlag) { CPWL_Wnd::OnLButtonDown(point, nFlag); if (ClientHitTest(point)) { if (m_bMouseDown && !InvalidateRect(nullptr)) return true; m_bMouseDown = true; SetCapture(); m_pEdit->OnMouseDown(point, IsSHIFTpressed(nFlag), IsCTRLpressed(nFlag)); } return true; } bool CPWL_EditCtrl::OnLButtonUp(const CFX_PointF& point, uint32_t nFlag) { CPWL_Wnd::OnLButtonUp(point, nFlag); if (m_bMouseDown) { // can receive keybord message if (ClientHitTest(point) && !IsFocused()) SetFocus(); ReleaseCapture(); m_bMouseDown = false; } return true; } bool CPWL_EditCtrl::OnMouseMove(const CFX_PointF& point, uint32_t nFlag) { CPWL_Wnd::OnMouseMove(point, nFlag); if (m_bMouseDown) m_pEdit->OnMouseMove(point, false, false); return true; } void CPWL_EditCtrl::SetEditCaret(bool bVisible) { CFX_PointF ptHead; CFX_PointF ptFoot; if (bVisible) GetCaretInfo(&ptHead, &ptFoot); SetCaret(bVisible, ptHead, ptFoot); // Note, |this| may no longer be viable at this point. If more work needs to // be done, check the return value of SetCaret(). } void CPWL_EditCtrl::GetCaretInfo(CFX_PointF* ptHead, CFX_PointF* ptFoot) const { CPWL_EditImpl_Iterator* pIterator = m_pEdit->GetIterator(); pIterator->SetAt(m_pEdit->GetCaret()); CPVT_Word word; CPVT_Line line; if (pIterator->GetWord(word)) { ptHead->x = word.ptWord.x + word.fWidth; ptHead->y = word.ptWord.y + word.fAscent; ptFoot->x = word.ptWord.x + word.fWidth; ptFoot->y = word.ptWord.y + word.fDescent; } else if (pIterator->GetLine(line)) { ptHead->x = line.ptLine.x; ptHead->y = line.ptLine.y + line.fLineAscent; ptFoot->x = line.ptLine.x; ptFoot->y = line.ptLine.y + line.fLineDescent; } } bool CPWL_EditCtrl::SetCaret(bool bVisible, const CFX_PointF& ptHead, const CFX_PointF& ptFoot) { if (!m_pEditCaret) return true; if (!IsFocused() || m_pEdit->IsSelected()) bVisible = false; ObservedPtr thisObserved(this); m_pEditCaret->SetCaret(bVisible, ptHead, ptFoot); if (!thisObserved) return false; return true; } WideString CPWL_EditCtrl::GetText() { return m_pEdit->GetText(); } void CPWL_EditCtrl::SetSelection(int32_t nStartChar, int32_t nEndChar) { m_pEdit->SetSelection(nStartChar, nEndChar); } void CPWL_EditCtrl::GetSelection(int32_t& nStartChar, int32_t& nEndChar) const { m_pEdit->GetSelection(nStartChar, nEndChar); } void CPWL_EditCtrl::ClearSelection() { if (!IsReadOnly()) m_pEdit->ClearSelection(); } void CPWL_EditCtrl::SelectAll() { m_pEdit->SelectAll(); } void CPWL_EditCtrl::SetScrollPos(const CFX_PointF& point) { m_pEdit->SetScrollPos(point); } CFX_PointF CPWL_EditCtrl::GetScrollPos() const { return m_pEdit->GetScrollPos(); } void CPWL_EditCtrl::CopyText() {} void CPWL_EditCtrl::PasteText() {} void CPWL_EditCtrl::CutText() {} void CPWL_EditCtrl::InsertWord(uint16_t word, int32_t nCharset) { if (!IsReadOnly()) m_pEdit->InsertWord(word, nCharset); } void CPWL_EditCtrl::InsertReturn() { if (!IsReadOnly()) m_pEdit->InsertReturn(); } void CPWL_EditCtrl::Delete() { if (!IsReadOnly()) m_pEdit->Delete(); } void CPWL_EditCtrl::Backspace() { if (!IsReadOnly()) m_pEdit->Backspace(); } bool CPWL_EditCtrl::CanUndo() { return !IsReadOnly() && m_pEdit->CanUndo(); } bool CPWL_EditCtrl::CanRedo() { return !IsReadOnly() && m_pEdit->CanRedo(); } bool CPWL_EditCtrl::Undo() { return CanUndo() && m_pEdit->Undo(); } bool CPWL_EditCtrl::Redo() { return CanRedo() && m_pEdit->Redo(); } int32_t CPWL_EditCtrl::GetCharSet() const { return m_nCharSet < 0 ? FX_CHARSET_Default : m_nCharSet; } void CPWL_EditCtrl::SetReadyToInput() { if (m_bMouseDown) { ReleaseCapture(); m_bMouseDown = false; } }