#pragma once #include #include #include #include #include #include #include #include #include #include class GTextEditor; class GTextDocument; class GTextDocumentLine; struct GTextDocumentSpan { GTextRange range; Color color; Optional background_color; bool is_skippable { false }; const Font* font { nullptr }; void* data { nullptr }; }; class GTextDocumentUndoCommand : public GCommand { public: GTextDocumentUndoCommand(GTextDocument&); virtual ~GTextDocumentUndoCommand(); protected: GTextDocument& m_document; }; class InsertCharacterCommand : public GTextDocumentUndoCommand { public: InsertCharacterCommand(GTextDocument&, char, GTextPosition); virtual void undo() override; virtual void redo() override; private: char m_character; GTextPosition m_text_position; }; class RemoveCharacterCommand : public GTextDocumentUndoCommand { public: RemoveCharacterCommand(GTextDocument&, char, GTextPosition); virtual void undo() override; virtual void redo() override; private: char m_character; GTextPosition m_text_position; }; class RemoveLineCommand : public GTextDocumentUndoCommand { public: RemoveLineCommand(GTextDocument&, String, GTextPosition, bool has_merged_content); virtual void undo() override; virtual void redo() override; private: String m_line_content; GTextPosition m_text_position; bool m_has_merged_content; }; class CreateLineCommand : public GTextDocumentUndoCommand { public: CreateLineCommand(GTextDocument&, Vector line_content, GTextPosition); virtual void undo() override; virtual void redo() override; private: Vector m_line_content; GTextPosition m_text_position; }; class GTextDocument : public RefCounted { public: enum class SearchShouldWrap { No = 0, Yes }; class Client { public: virtual ~Client(); virtual void document_did_append_line() = 0; virtual void document_did_insert_line(int) = 0; virtual void document_did_remove_line(int) = 0; virtual void document_did_remove_all_lines() = 0; virtual void document_did_change() = 0; virtual void document_did_set_text() = 0; virtual void document_did_set_cursor(const GTextPosition&) = 0; }; static NonnullRefPtr create(Client* client = nullptr) { return adopt(*new GTextDocument(client)); } int line_count() const { return m_lines.size(); } const GTextDocumentLine& line(int line_index) const { return m_lines[line_index]; } GTextDocumentLine& line(int line_index) { return m_lines[line_index]; } void set_spans(const Vector& spans) { m_spans = spans; } void set_text(const StringView&); const NonnullOwnPtrVector& lines() const { return m_lines; } NonnullOwnPtrVector& lines() { return m_lines; } bool has_spans() const { return !m_spans.is_empty(); } const Vector& spans() const { return m_spans; } void set_span_at_index(int index, GTextDocumentSpan span) { m_spans[index] = move(span); } void append_line(NonnullOwnPtr); void remove_line(int line_index); void remove_all_lines(); void insert_line(int line_index, NonnullOwnPtr); void register_client(Client&); void unregister_client(Client&); void update_views(Badge); String text_in_range(const GTextRange&) const; Vector find_all(const StringView& needle) const; GTextRange find_next(const StringView&, const GTextPosition& start = {}, SearchShouldWrap = SearchShouldWrap::Yes) const; GTextRange find_previous(const StringView&, const GTextPosition& start = {}, SearchShouldWrap = SearchShouldWrap::Yes) const; GTextPosition next_position_after(const GTextPosition&, SearchShouldWrap = SearchShouldWrap::Yes) const; GTextPosition previous_position_before(const GTextPosition&, SearchShouldWrap = SearchShouldWrap::Yes) const; char character_at(const GTextPosition&) const; Optional first_non_skippable_span_before(const GTextPosition&) const; Optional first_non_skippable_span_after(const GTextPosition&) const; void add_to_undo_stack(NonnullOwnPtr); bool can_undo() const { return m_undo_stack.can_undo(); } bool can_redo() const { return m_undo_stack.can_redo(); } void undo(); void redo(); void notify_did_change(); void set_all_cursors(const GTextPosition&); private: explicit GTextDocument(Client* client); void update_undo_timer(); NonnullOwnPtrVector m_lines; Vector m_spans; HashTable m_clients; bool m_client_notifications_enabled { true }; GUndoStack m_undo_stack; RefPtr m_undo_timer; }; class GTextDocumentLine { friend class GTextEditor; friend class GTextDocument; public: explicit GTextDocumentLine(GTextDocument&); explicit GTextDocumentLine(GTextDocument&, const StringView&); StringView view() const { return { characters(), length() }; } const char* characters() const { return m_text.data(); } int length() const { return m_text.size() - 1; } void set_text(GTextDocument&, const StringView&); void append(GTextDocument&, char); void prepend(GTextDocument&, char); void insert(GTextDocument&, int index, char); void remove(GTextDocument&, int index); void append(GTextDocument&, const char*, int); void truncate(GTextDocument&, int length); void clear(GTextDocument&); int first_non_whitespace_column() const; private: // NOTE: This vector is null terminated. Vector m_text; };