| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <LibGUI/GWidget.h>
 | 
					
						
							|  |  |  | #include <AK/Function.h>
 | 
					
						
							|  |  |  | #include <AK/HashMap.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class GScrollBar; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class GTextPosition { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     GTextPosition() { } | 
					
						
							|  |  |  |     GTextPosition(int line, int column) | 
					
						
							|  |  |  |         : m_line(line) | 
					
						
							|  |  |  |         , m_column(column) | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool is_valid() const { return m_line >= 0 && m_column >= 0; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int line() const { return m_line; } | 
					
						
							|  |  |  |     int column() const { return m_column; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     void set_line(int line) { m_line = line; } | 
					
						
							|  |  |  |     void set_column(int column) { m_column = column; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 15:03:38 +01:00
										 |  |  |     bool operator==(const GTextPosition& other) const { return m_line == other.m_line && m_column == other.m_column; } | 
					
						
							| 
									
										
										
										
											2019-03-08 00:49:45 +01:00
										 |  |  |     bool operator<(const GTextPosition& other) const { return m_line < other.m_line || (m_line == other.m_line && m_column < other.m_column); } | 
					
						
							| 
									
										
										
										
											2019-03-07 15:03:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     int m_line { -1 }; | 
					
						
							|  |  |  |     int m_column { -1 }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class GTextEditor : public GWidget { | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     explicit GTextEditor(GWidget* parent); | 
					
						
							|  |  |  |     virtual ~GTextEditor() override; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 00:46:29 +01:00
										 |  |  |     Function<void(GTextEditor&)> on_cursor_change; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 00:46:29 +01:00
										 |  |  |     void set_text(const String&); | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     int content_width() const; | 
					
						
							| 
									
										
										
										
											2019-03-07 13:54:02 +01:00
										 |  |  |     int content_height() const; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     Rect visible_content_rect() const; | 
					
						
							| 
									
										
										
										
											2019-03-07 13:13:25 +01:00
										 |  |  |     void scroll_cursor_into_view(); | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     int line_count() const { return m_lines.size(); } | 
					
						
							| 
									
										
										
										
											2019-03-07 01:05:35 +01:00
										 |  |  |     int line_spacing() const { return m_line_spacing; } | 
					
						
							|  |  |  |     int line_height() const { return font().glyph_height() + m_line_spacing; } | 
					
						
							| 
									
										
										
										
											2019-03-07 13:54:02 +01:00
										 |  |  |     int padding() const { return 3; } | 
					
						
							| 
									
										
										
										
											2019-03-07 00:46:29 +01:00
										 |  |  |     GTextPosition cursor() const { return m_cursor; } | 
					
						
							| 
									
										
										
										
											2019-03-08 00:49:45 +01:00
										 |  |  |     GTextPosition selection_start() const { return m_selection_start; } | 
					
						
							| 
									
										
										
										
											2019-03-07 14:02:10 +01:00
										 |  |  |     int glyph_width() const { return font().glyph_width('x'); } | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 17:06:11 +01:00
										 |  |  |     bool write_to_file(const String& path); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | private: | 
					
						
							|  |  |  |     virtual void paint_event(GPaintEvent&) override; | 
					
						
							|  |  |  |     virtual void resize_event(GResizeEvent&) override; | 
					
						
							|  |  |  |     virtual void mousedown_event(GMouseEvent&) override; | 
					
						
							|  |  |  |     virtual void keydown_event(GKeyEvent&) override; | 
					
						
							| 
									
										
										
										
											2019-03-07 01:05:35 +01:00
										 |  |  |     virtual void focusin_event(GEvent&) override; | 
					
						
							|  |  |  |     virtual void focusout_event(GEvent&) override; | 
					
						
							|  |  |  |     virtual void timer_event(GTimerEvent&) override; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     virtual bool accepts_focus() const override { return true; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     class Line { | 
					
						
							| 
									
										
										
										
											2019-03-07 16:04:21 +01:00
										 |  |  |         friend class GTextEditor; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     public: | 
					
						
							| 
									
										
										
										
											2019-03-07 14:35:32 +01:00
										 |  |  |         Line(); | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 14:35:32 +01:00
										 |  |  |         const char* characters() const { return m_text.data(); } | 
					
						
							|  |  |  |         int length() const { return m_text.size() - 1; } | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |         int width(const Font&) const; | 
					
						
							|  |  |  |         void set_text(const String&); | 
					
						
							| 
									
										
										
										
											2019-03-07 16:04:21 +01:00
										 |  |  |         void append(char); | 
					
						
							|  |  |  |         void prepend(char); | 
					
						
							|  |  |  |         void insert(int index, char); | 
					
						
							|  |  |  |         void remove(int index); | 
					
						
							| 
									
										
										
										
											2019-03-07 16:15:25 +01:00
										 |  |  |         void append(const char*, int); | 
					
						
							| 
									
										
										
										
											2019-03-07 16:49:04 +01:00
										 |  |  |         void truncate(int length); | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     private: | 
					
						
							| 
									
										
										
										
											2019-03-07 14:35:32 +01:00
										 |  |  |         // NOTE: This vector is null terminated.
 | 
					
						
							|  |  |  |         Vector<char> m_text; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     }; | 
					
						
							| 
									
										
										
										
											2019-03-07 13:21:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     void update_scrollbar_ranges(); | 
					
						
							|  |  |  |     Rect line_content_rect(int item_index) const; | 
					
						
							|  |  |  |     Rect line_widget_rect(int line_index) const; | 
					
						
							|  |  |  |     Rect cursor_content_rect() const; | 
					
						
							|  |  |  |     void update_cursor(); | 
					
						
							|  |  |  |     void set_cursor(int line, int column); | 
					
						
							| 
									
										
										
										
											2019-03-07 15:03:38 +01:00
										 |  |  |     void set_cursor(const GTextPosition&); | 
					
						
							| 
									
										
										
										
											2019-03-07 15:52:11 +01:00
										 |  |  |     Line& current_line() { return *m_lines[m_cursor.line()]; } | 
					
						
							|  |  |  |     const Line& current_line() const { return *m_lines[m_cursor.line()]; } | 
					
						
							| 
									
										
										
										
											2019-03-07 15:03:38 +01:00
										 |  |  |     GTextPosition text_position_at(const Point&) const; | 
					
						
							|  |  |  |     void insert_at_cursor(char); | 
					
						
							| 
									
										
										
										
											2019-03-07 20:05:05 +01:00
										 |  |  |     int ruler_width() const; | 
					
						
							|  |  |  |     Rect ruler_content_rect(int line) const; | 
					
						
							| 
									
										
										
										
											2019-03-08 00:49:45 +01:00
										 |  |  |     void toggle_selection_if_needed_for_event(const GKeyEvent&); | 
					
						
							| 
									
										
										
										
											2019-03-07 13:21:51 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     GScrollBar* m_vertical_scrollbar { nullptr }; | 
					
						
							|  |  |  |     GScrollBar* m_horizontal_scrollbar { nullptr }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 15:52:11 +01:00
										 |  |  |     Vector<OwnPtr<Line>> m_lines; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  |     GTextPosition m_cursor; | 
					
						
							| 
									
										
										
										
											2019-03-07 01:05:35 +01:00
										 |  |  |     bool m_cursor_state { true }; | 
					
						
							|  |  |  |     int m_line_spacing { 2 }; | 
					
						
							| 
									
										
										
										
											2019-03-08 00:49:45 +01:00
										 |  |  |     GTextPosition m_selection_start; | 
					
						
							| 
									
										
										
										
											2019-03-07 00:31:06 +01:00
										 |  |  | }; |