| // Copyright 2017 The PDFium Authors |
| // 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 XFA_FDE_CFDE_TEXTEDITENGINE_H_ |
| #define XFA_FDE_CFDE_TEXTEDITENGINE_H_ |
| |
| #include <limits> |
| #include <memory> |
| #include <utility> |
| #include <vector> |
| |
| #include "core/fxcrt/retain_ptr.h" |
| #include "core/fxcrt/unowned_ptr.h" |
| #include "core/fxcrt/widestring.h" |
| #include "core/fxge/dib/fx_dib.h" |
| #include "xfa/fgas/layout/cfgas_txtbreak.h" |
| |
| class CFGAS_GEFont; |
| class TextCharPos; |
| |
| struct FDE_TEXTEDITPIECE { |
| FDE_TEXTEDITPIECE(); |
| FDE_TEXTEDITPIECE(const FDE_TEXTEDITPIECE& that); |
| ~FDE_TEXTEDITPIECE(); |
| |
| CFX_RectF rtPiece; |
| int32_t nStart = 0; |
| int32_t nCount = 0; |
| int32_t nBidiLevel = 0; |
| uint32_t dwCharStyles = 0; |
| }; |
| |
| inline FDE_TEXTEDITPIECE::FDE_TEXTEDITPIECE() = default; |
| inline FDE_TEXTEDITPIECE::FDE_TEXTEDITPIECE(const FDE_TEXTEDITPIECE& that) = |
| default; |
| inline FDE_TEXTEDITPIECE::~FDE_TEXTEDITPIECE() = default; |
| |
| class CFDE_TextEditEngine final : public CFGAS_TxtBreak::Engine { |
| public: |
| class Iterator { |
| public: |
| explicit Iterator(const CFDE_TextEditEngine* engine); |
| ~Iterator(); |
| |
| void Next(bool bPrev); |
| wchar_t GetChar() const; |
| void SetAt(size_t nIndex); |
| size_t FindNextBreakPos(bool bPrev); |
| bool IsEOF(bool bPrev) const; |
| |
| private: |
| UnownedPtr<const CFDE_TextEditEngine> const engine_; |
| int32_t current_position_ = -1; |
| }; |
| |
| class Operation { |
| public: |
| virtual ~Operation() = default; |
| virtual void Redo() const = 0; |
| virtual void Undo() const = 0; |
| }; |
| |
| struct TextChange { |
| WideString text; |
| WideString previous_text; |
| size_t selection_start; |
| size_t selection_end; |
| bool cancelled; |
| }; |
| |
| class Delegate { |
| public: |
| virtual ~Delegate() = default; |
| virtual void NotifyTextFull() = 0; |
| virtual void OnCaretChanged() = 0; |
| virtual void OnTextWillChange(TextChange* change) = 0; |
| virtual void OnTextChanged() = 0; |
| virtual void OnSelChanged() = 0; |
| virtual bool OnValidate(const WideString& wsText) = 0; |
| virtual void SetScrollOffset(float fScrollOffset) = 0; |
| }; |
| |
| enum class RecordOperation { kInsertRecord, kSkipRecord, kSkipNotify }; |
| |
| CFDE_TextEditEngine(); |
| ~CFDE_TextEditEngine() override; |
| |
| // CFGAS_TxtBreak::Engine: |
| wchar_t GetChar(size_t idx) const override; |
| int32_t GetWidthOfChar(size_t idx) override; |
| |
| void SetDelegate(Delegate* delegate) { delegate_ = delegate; } |
| void Clear(); |
| |
| void Insert(size_t idx, |
| const WideString& text, |
| RecordOperation add_operation = RecordOperation::kInsertRecord); |
| WideString Delete( |
| size_t start_idx, |
| size_t length, |
| RecordOperation add_operation = RecordOperation::kInsertRecord); |
| WideString GetText() const; |
| size_t GetLength() const; |
| |
| // Non-const so we can force a layout. |
| CFX_RectF GetContentsBoundingBox(); |
| void SetAvailableWidth(size_t width); |
| |
| void SetFont(RetainPtr<CFGAS_GEFont> font); |
| RetainPtr<CFGAS_GEFont> GetFont() const; |
| void SetFontSize(float size); |
| float GetFontSize() const { return font_size_; } |
| void SetFontColor(FX_ARGB color) { font_color_ = color; } |
| FX_ARGB GetFontColor() const { return font_color_; } |
| |
| void SetAlignment(uint32_t alignment); |
| float GetLineSpace() const { return line_spacing_; } |
| void SetLineSpace(float space) { line_spacing_ = space; } |
| void SetAliasChar(wchar_t alias) { password_alias_ = alias; } |
| void SetHasCharacterLimit(bool limit); |
| void SetCharacterLimit(size_t limit); |
| void SetCombText(bool enable); |
| void SetTabWidth(float width); |
| void SetVisibleLineCount(size_t lines); |
| |
| void EnableValidation(bool val) { validation_enabled_ = val; } |
| void EnablePasswordMode(bool val) { password_mode_ = val; } |
| void EnableMultiLine(bool val); |
| void EnableLineWrap(bool val); |
| void LimitHorizontalScroll(bool val); |
| void LimitVerticalScroll(bool val); |
| |
| bool CanUndo() const; |
| bool CanRedo() const; |
| bool Redo(); |
| bool Undo(); |
| void ClearOperationRecords(); |
| |
| size_t GetIndexLeft(size_t pos) const; |
| size_t GetIndexRight(size_t pos) const; |
| size_t GetIndexUp(size_t pos) const; |
| size_t GetIndexDown(size_t pos) const; |
| size_t GetIndexAtStartOfLine(size_t pos) const; |
| size_t GetIndexAtEndOfLine(size_t pos) const; |
| |
| void SelectAll(); |
| void SetSelection(size_t start_idx, size_t count); |
| void ClearSelection(); |
| bool HasSelection() const { return has_selection_; } |
| // Returns <start_idx, count> of the selection. |
| std::pair<size_t, size_t> GetSelection() const { |
| return {selection_.start_idx, selection_.count}; |
| } |
| WideString GetSelectedText() const; |
| WideString DeleteSelectedText( |
| RecordOperation add_operation = RecordOperation::kInsertRecord); |
| void ReplaceSelectedText(const WideString& str); |
| |
| void Layout(); |
| |
| // Non-const so we can force a Layout() if needed. |
| size_t GetIndexForPoint(const CFX_PointF& point); |
| // <start_idx, count> |
| std::pair<size_t, size_t> BoundsForWordAt(size_t idx) const; |
| |
| // Note that if CanGenerateCharacterInfo() returns false, then |
| // GetCharacterInfo() cannot be called. |
| bool CanGenerateCharacterInfo() const { return text_length_ > 0 && font_; } |
| |
| // Returns <bidi level, character rect> |
| std::pair<int32_t, CFX_RectF> GetCharacterInfo(int32_t start_idx); |
| std::vector<CFX_RectF> GetCharacterRectsInRange(int32_t start_idx, |
| int32_t count); |
| |
| const std::vector<FDE_TEXTEDITPIECE>& GetTextPieces() { |
| // Force a layout if needed. |
| Layout(); |
| return text_piece_info_; |
| } |
| |
| std::vector<TextCharPos> GetDisplayPos(const FDE_TEXTEDITPIECE& info); |
| |
| void SetMaxEditOperationsForTesting(size_t max); |
| |
| private: |
| struct Selection { |
| size_t start_idx; |
| size_t count; |
| }; |
| |
| static constexpr size_t kGapSize = 128; |
| static constexpr size_t kMaxEditOperations = 128; |
| static constexpr size_t kPageWidthMax = 0xffff; |
| |
| void SetCombTextWidth(); |
| void AdjustGap(size_t idx, size_t length); |
| void RebuildPieces(); |
| size_t CountCharsExceedingSize(const WideString& str, size_t num_to_check); |
| void AddOperationRecord(std::unique_ptr<Operation> op); |
| |
| bool IsAlignedRight() const { |
| return !!(character_alignment_ & CFX_TxtLineAlignment_Right); |
| } |
| |
| bool IsAlignedCenter() const { |
| return !!(character_alignment_ & CFX_TxtLineAlignment_Center); |
| } |
| std::vector<CFX_RectF> GetCharRects(const FDE_TEXTEDITPIECE& piece); |
| |
| CFX_RectF contents_bounding_box_; |
| UnownedPtr<Delegate> delegate_; |
| std::vector<FDE_TEXTEDITPIECE> text_piece_info_; |
| std::vector<int32_t> char_widths_; // May be negative for combining chars. |
| CFGAS_TxtBreak text_break_; |
| RetainPtr<CFGAS_GEFont> font_; |
| FX_ARGB font_color_ = 0xff000000; |
| float font_size_ = 10.0f; |
| float line_spacing_ = 10.0f; |
| std::vector<WideString::CharType> content_; |
| size_t text_length_ = 0; |
| |
| // See e.g. https://en.wikipedia.org/wiki/Gap_buffer |
| size_t gap_position_ = 0; |
| size_t gap_size_ = kGapSize; |
| |
| size_t available_width_ = kPageWidthMax; |
| size_t character_limit_ = std::numeric_limits<size_t>::max(); |
| size_t visible_line_count_ = 1; |
| // Ring buffer of edit operations |
| std::vector<std::unique_ptr<Operation>> operation_buffer_; |
| // Next edit operation to undo. |
| size_t next_operation_index_to_undo_ = kMaxEditOperations - 1; |
| // Next index to insert an edit operation into. |
| size_t next_operation_index_to_insert_ = 0; |
| size_t max_edit_operations_ = kMaxEditOperations; |
| uint32_t character_alignment_ = CFX_TxtLineAlignment_Left; |
| bool has_character_limit_ = false; |
| bool is_comb_text_ = false; |
| bool is_dirty_ = false; |
| bool validation_enabled_ = false; |
| bool is_multiline_ = false; |
| bool is_linewrap_enabled_ = false; |
| bool limit_horizontal_area_ = false; |
| bool limit_vertical_area_ = false; |
| bool password_mode_ = false; |
| wchar_t password_alias_ = L'*'; |
| bool has_selection_ = false; |
| Selection selection_{0, 0}; |
| }; |
| |
| #endif // XFA_FDE_CFDE_TEXTEDITENGINE_H_ |