// 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 #ifndef FPDFSDK_PWL_CPWL_EDIT_IMPL_H_ #define FPDFSDK_PWL_CPWL_EDIT_IMPL_H_ #include <deque> #include <memory> #include <vector> #include "core/fpdfdoc/cpdf_variabletext.h" #include "core/fpdfdoc/cpvt_wordrange.h" #include "core/fxcrt/unowned_ptr.h" #include "core/fxge/fx_dib.h" #define FX_EDIT_ISLATINWORD(u) \ (u == 0x2D || (u <= 0x005A && u >= 0x0041) || \ (u <= 0x007A && u >= 0x0061) || (u <= 0x02AF && u >= 0x00C0)) class CFFL_FormFiller; class CPWL_EditImpl; class CPWL_EditImpl_Iterator; class CPWL_EditImpl_Provider; class CFX_RenderDevice; class CFX_SystemHandler; class CPWL_Edit; class CPWL_EditCtrl; class IFX_Edit_UndoItem; struct CPWL_EditImpl_LineRect { CPWL_EditImpl_LineRect(const CPVT_WordRange& wrLine, const CFX_FloatRect& rcLine) : m_wrLine(wrLine), m_rcLine(rcLine) {} CPVT_WordRange m_wrLine; CFX_FloatRect m_rcLine; }; class CPWL_EditImpl_Refresh { public: CPWL_EditImpl_Refresh(); ~CPWL_EditImpl_Refresh(); void BeginRefresh(); void Push(const CPVT_WordRange& linerange, const CFX_FloatRect& rect); void NoAnalyse(); std::vector<CFX_FloatRect>* GetRefreshRects(); void EndRefresh(); private: void Add(const CFX_FloatRect& new_rect); std::vector<CPWL_EditImpl_LineRect> m_NewLineRects; std::vector<CPWL_EditImpl_LineRect> m_OldLineRects; std::vector<CFX_FloatRect> m_RefreshRects; }; class CPWL_EditImpl_Select { public: CPWL_EditImpl_Select(); explicit CPWL_EditImpl_Select(const CPVT_WordRange& range); void Reset(); void Set(const CPVT_WordPlace& begin, const CPVT_WordPlace& end); void SetEndPos(const CPVT_WordPlace& end); CPVT_WordRange ConvertToWordRange() const; bool IsEmpty() const; CPVT_WordPlace BeginPos; CPVT_WordPlace EndPos; }; class CPWL_EditImpl_Undo { public: CPWL_EditImpl_Undo(); ~CPWL_EditImpl_Undo(); void AddItem(std::unique_ptr<IFX_Edit_UndoItem> pItem); void Undo(); void Redo(); bool CanUndo() const; bool CanRedo() const; private: void RemoveHeads(); void RemoveTails(); std::deque<std::unique_ptr<IFX_Edit_UndoItem>> m_UndoItemStack; size_t m_nCurUndoPos; bool m_bWorking; }; class IFX_Edit_UndoItem { public: virtual ~IFX_Edit_UndoItem() {} virtual void Undo() = 0; virtual void Redo() = 0; }; class CFXEU_InsertWord : public IFX_Edit_UndoItem { public: CFXEU_InsertWord(CPWL_EditImpl* pEdit, const CPVT_WordPlace& wpOldPlace, const CPVT_WordPlace& wpNewPlace, uint16_t word, int32_t charset); ~CFXEU_InsertWord() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordPlace m_wpOld; CPVT_WordPlace m_wpNew; uint16_t m_Word; int32_t m_nCharset; }; class CFXEU_InsertReturn : public IFX_Edit_UndoItem { public: CFXEU_InsertReturn(CPWL_EditImpl* pEdit, const CPVT_WordPlace& wpOldPlace, const CPVT_WordPlace& wpNewPlace); ~CFXEU_InsertReturn() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordPlace m_wpOld; CPVT_WordPlace m_wpNew; }; class CFXEU_Backspace : public IFX_Edit_UndoItem { public: CFXEU_Backspace(CPWL_EditImpl* pEdit, const CPVT_WordPlace& wpOldPlace, const CPVT_WordPlace& wpNewPlace, uint16_t word, int32_t charset); ~CFXEU_Backspace() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordPlace m_wpOld; CPVT_WordPlace m_wpNew; uint16_t m_Word; int32_t m_nCharset; }; class CFXEU_Delete : public IFX_Edit_UndoItem { public: CFXEU_Delete(CPWL_EditImpl* pEdit, const CPVT_WordPlace& wpOldPlace, const CPVT_WordPlace& wpNewPlace, uint16_t word, int32_t charset, bool bSecEnd); ~CFXEU_Delete() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordPlace m_wpOld; CPVT_WordPlace m_wpNew; uint16_t m_Word; int32_t m_nCharset; bool m_bSecEnd; }; class CFXEU_Clear : public IFX_Edit_UndoItem { public: CFXEU_Clear(CPWL_EditImpl* pEdit, const CPVT_WordRange& wrSel, const WideString& swText); ~CFXEU_Clear() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordRange m_wrSel; WideString m_swText; }; class CFXEU_InsertText : public IFX_Edit_UndoItem { public: CFXEU_InsertText(CPWL_EditImpl* pEdit, const CPVT_WordPlace& wpOldPlace, const CPVT_WordPlace& wpNewPlace, const WideString& swText, int32_t charset); ~CFXEU_InsertText() override; // IFX_Edit_UndoItem: void Redo() override; void Undo() override; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPVT_WordPlace m_wpOld; CPVT_WordPlace m_wpNew; WideString m_swText; int32_t m_nCharset; }; class CPWL_EditImpl { public: static void DrawEdit(CFX_RenderDevice* pDevice, const CFX_Matrix& mtUser2Device, CPWL_EditImpl* pEdit, FX_COLORREF crTextFill, const CFX_FloatRect& rcClip, const CFX_PointF& ptOffset, const CPVT_WordRange* pRange, CFX_SystemHandler* pSystemHandler, CFFL_FormFiller* pFFLData); CPWL_EditImpl(); ~CPWL_EditImpl(); void SetFontMap(IPVT_FontMap* pFontMap); void SetNotify(CPWL_EditCtrl* pNotify); void SetOperationNotify(CPWL_Edit* pOperationNotify); // Returns an iterator for the contents. Should not be released. CPWL_EditImpl_Iterator* GetIterator(); IPVT_FontMap* GetFontMap(); void Initialize(); // Set the bounding box of the text area. void SetPlateRect(const CFX_FloatRect& rect); void SetScrollPos(const CFX_PointF& point); // Set the horizontal text alignment. (nFormat [0:left, 1:middle, 2:right]) void SetAlignmentH(int32_t nFormat, bool bPaint); // Set the vertical text alignment. (nFormat [0:left, 1:middle, 2:right]) void SetAlignmentV(int32_t nFormat, bool bPaint); // Set the substitution character for hidden text. void SetPasswordChar(uint16_t wSubWord, bool bPaint); // Set the maximum number of words in the text. void SetLimitChar(int32_t nLimitChar); void SetCharArray(int32_t nCharArray); void SetCharSpace(float fCharSpace); void SetMultiLine(bool bMultiLine, bool bPaint); void SetAutoReturn(bool bAuto, bool bPaint); void SetAutoFontSize(bool bAuto, bool bPaint); void SetAutoScroll(bool bAuto, bool bPaint); void SetFontSize(float fFontSize); void SetTextOverflow(bool bAllowed, bool bPaint); void OnMouseDown(const CFX_PointF& point, bool bShift, bool bCtrl); void OnMouseMove(const CFX_PointF& point, bool bShift, bool bCtrl); void OnVK_UP(bool bShift, bool bCtrl); void OnVK_DOWN(bool bShift, bool bCtrl); void OnVK_LEFT(bool bShift, bool bCtrl); void OnVK_RIGHT(bool bShift, bool bCtrl); void OnVK_HOME(bool bShift, bool bCtrl); void OnVK_END(bool bShift, bool bCtrl); void SetText(const WideString& sText); bool InsertWord(uint16_t word, int32_t charset); bool InsertReturn(); bool Backspace(); bool Delete(); bool ClearSelection(); bool InsertText(const WideString& sText, int32_t charset); bool Redo(); bool Undo(); CPVT_WordPlace WordIndexToWordPlace(int32_t index) const; CPVT_WordPlace SearchWordPlace(const CFX_PointF& point) const; int32_t GetCaret() const; CPVT_WordPlace GetCaretWordPlace() const; WideString GetSelectedText() const; WideString GetText() const; float GetFontSize() const; uint16_t GetPasswordChar() const; CFX_PointF GetScrollPos() const; int32_t GetCharArray() const; CFX_FloatRect GetContentRect() const; WideString GetRangeText(const CPVT_WordRange& range) const; int32_t GetHorzScale() const; float GetCharSpace() const; void SetSelection(int32_t nStartChar, int32_t nEndChar); void GetSelection(int32_t& nStartChar, int32_t& nEndChar) const; void SelectAll(); void SelectNone(); bool IsSelected() const; void Paint(); void EnableRefresh(bool bRefresh); void RefreshWordRange(const CPVT_WordRange& wr); CPVT_WordRange GetWholeWordRange() const; CPVT_WordRange GetSelectWordRange() const; void EnableUndo(bool bUndo); bool IsTextFull() const; bool IsTextOverflow() const; bool CanUndo() const; bool CanRedo() const; CPVT_WordRange GetVisibleWordRange() const; bool Empty(); CPVT_WordPlace DoInsertText(const CPVT_WordPlace& place, const WideString& sText, int32_t charset); int32_t GetCharSetFromUnicode(uint16_t word, int32_t nOldCharset); int32_t GetTotalLines() const; ByteString GetPDFWordString(int32_t nFontIndex, uint16_t Word, uint16_t SubWord); void SetSelection(const CPVT_WordPlace& begin, const CPVT_WordPlace& end); bool Delete(bool bAddUndo, bool bPaint); bool Clear(bool bAddUndo, bool bPaint); bool InsertText(const WideString& sText, int32_t charset, bool bAddUndo, bool bPaint); bool InsertWord(uint16_t word, int32_t charset, bool bAddUndo, bool bPaint); bool InsertReturn(bool bAddUndo, bool bPaint); bool Backspace(bool bAddUndo, bool bPaint); void SetCaret(const CPVT_WordPlace& place); CFX_PointF VTToEdit(const CFX_PointF& point) const; private: void RearrangeAll(); void RearrangePart(const CPVT_WordRange& range); void ScrollToCaret(); void SetScrollInfo(); void SetScrollPosX(float fx); void SetScrollPosY(float fy); void SetScrollLimit(); void SetContentChanged(); void PaintInsertText(const CPVT_WordPlace& wpOld, const CPVT_WordPlace& wpNew); CFX_PointF EditToVT(const CFX_PointF& point) const; CFX_FloatRect VTToEdit(const CFX_FloatRect& rect) const; void Refresh(); void RefreshPushLineRects(const CPVT_WordRange& wr); void SetCaretInfo(); void SetCaretOrigin(); void AddEditUndoItem(std::unique_ptr<IFX_Edit_UndoItem> pEditUndoItem); std::unique_ptr<CPDF_VariableText> m_pVT; UnownedPtr<CPWL_EditCtrl> m_pNotify; UnownedPtr<CPWL_Edit> m_pOperationNotify; std::unique_ptr<CPWL_EditImpl_Provider> m_pVTProvider; CPVT_WordPlace m_wpCaret; CPVT_WordPlace m_wpOldCaret; CPWL_EditImpl_Select m_SelState; CFX_PointF m_ptScrollPos; CFX_PointF m_ptRefreshScrollPos; bool m_bEnableScroll; std::unique_ptr<CPWL_EditImpl_Iterator> m_pIterator; CPWL_EditImpl_Refresh m_Refresh; CFX_PointF m_ptCaret; CPWL_EditImpl_Undo m_Undo; int32_t m_nAlignment; bool m_bNotifyFlag; bool m_bEnableOverflow; bool m_bEnableRefresh; CFX_FloatRect m_rcOldContent; bool m_bEnableUndo; }; class CPWL_EditImpl_Iterator { public: CPWL_EditImpl_Iterator(CPWL_EditImpl* pEdit, CPDF_VariableText::Iterator* pVTIterator); ~CPWL_EditImpl_Iterator(); bool NextWord(); bool PrevWord(); bool GetWord(CPVT_Word& word) const; bool GetLine(CPVT_Line& line) const; void SetAt(int32_t nWordIndex); void SetAt(const CPVT_WordPlace& place); const CPVT_WordPlace& GetAt() const; private: UnownedPtr<CPWL_EditImpl> m_pEdit; CPDF_VariableText::Iterator* m_pVTIterator; }; class CPWL_EditImpl_Provider : public CPDF_VariableText::Provider { public: explicit CPWL_EditImpl_Provider(IPVT_FontMap* pFontMap); ~CPWL_EditImpl_Provider() override; IPVT_FontMap* GetFontMap() const; // CPDF_VariableText::Provider: int32_t GetCharWidth(int32_t nFontIndex, uint16_t word) override; int32_t GetTypeAscent(int32_t nFontIndex) override; int32_t GetTypeDescent(int32_t nFontIndex) override; int32_t GetWordFontIndex(uint16_t word, int32_t charset, int32_t nFontIndex) override; int32_t GetDefaultFontIndex() override; bool IsLatinWord(uint16_t word) override; private: IPVT_FontMap* m_pFontMap; }; #endif // FPDFSDK_PWL_CPWL_EDIT_IMPL_H_