/* * \brief Text dialog * \author Norman Feske * \date 2020-01-14 */ /* * Copyright (C) 2020 Genode Labs GmbH * * This file is part of the Genode OS framework, which is distributed * under the terms of the GNU Affero General Public License version 3. */ #ifndef _DIALOG_H_ #define _DIALOG_H_ /* Genode includes */ #include #include #include #include #include #include #include /* local includes */ #include #include #include namespace Text_area { struct Dialog; } struct Text_area::Dialog : private Dynamic_rom_session::Xml_producer { public: Dynamic_rom_session rom_session; struct Trigger_copy : Interface, Noncopyable { virtual void trigger_copy() = 0; }; struct Trigger_paste : Interface, Noncopyable { virtual void trigger_paste() = 0; }; struct Trigger_save : Interface, Noncopyable { virtual void trigger_save() = 0; }; private: Allocator &_alloc; Trigger_copy &_trigger_copy; Trigger_paste &_trigger_paste; Trigger_save &_trigger_save; struct Character : Codepoint { Character(Codepoint &codepoint) : Codepoint(codepoint) { } void print(Output &out) const { if (value == '"') Genode::print(out, """); else if (value == 9) Genode::print(out, " "); else Codepoint::print(out); } }; using Line = Dynamic_array; using Text = Dynamic_array; Text _text { _alloc }; struct Position { Line::Index x; Text::Index y; Position(Line::Index x, Text::Index y) : x(x), y(y) { } Position(Position const &other) : x(other.x), y(other.y) { } bool operator != (Position const &other) const { return (x.value != other.x.value) || (y.value != other.y.value); } }; Position _cursor { { 0 }, { 0 } }; Position _scroll { { 0 }, { 0 } }; Constructible _hovered_position { }; unsigned _max_lines = ~0U; bool _editable = false; unsigned _modification_count = 0; struct Selection { Constructible start; Constructible end; void clear() { start.destruct(); end .destruct(); } bool defined() const { return start.constructed() && end.constructed() && (*start != *end); } template void for_each_selected_line(FN const &) const; template void with_selection_at_line(Text::Index y, Line const &, FN const &) const; /* generate dialog model */ void gen_selected_line(Xml_generator &, Text::Index, Line const &) const; }; bool _drag = false; bool _shift = false; bool _control = false; bool _text_hovered = false; Selection _selection { }; void produce_xml(Xml_generator &xml) override; static bool _printable(Codepoint code) { if (!code.valid()) return false; if (code.value == '\t') return true; return (code.value >= 0x20 && code.value < 0xf000); } bool _cursor_at_last_line() const { return (_cursor.y.value + 1 >= _text.upper_bound().value); } bool _cursor_at_end_of_line() const { bool result = false; _text.apply(_cursor.y, [&] (Line &line) { result = (_cursor.x.value >= line.upper_bound().value); }); return result; } void _tie_cursor_to_end_of_line() { _text.apply(_cursor.y, [&] (Line &line) { if (_cursor.x.value > line.upper_bound().value) _cursor.x = line.upper_bound(); }); } bool _end_of_text() const { return _cursor_at_last_line() && _cursor_at_end_of_line(); } void _clamp_scroll_position_to_upper_bound() { if (_max_lines != ~0U) if (_scroll.y.value + _max_lines > _text.upper_bound().value) _scroll.y.value = max(_text.upper_bound().value, _max_lines) - _max_lines; } void _move_characters(Line &, Line &); void _delete_selection(); void _insert_printable(Codepoint); void _handle_printable(Codepoint); void _handle_backspace(); void _handle_delete(); void _handle_newline(); void _handle_left(); void _handle_right(); void _handle_up(); void _handle_down(); void _handle_pageup(); void _handle_pagedown(); void _handle_home(); void _handle_end(); public: Dialog(Entrypoint &, Ram_allocator &, Region_map &, Allocator &, Trigger_copy &, Trigger_paste &, Trigger_save &); void editable(bool editable) { _editable = editable; } unsigned modification_count() const { return _modification_count; } void max_lines(unsigned max_lines) { _max_lines = max_lines; } void handle_input_event(Input::Event const &); void handle_hover(Xml_node const &hover); void clear(); void append_newline() { _text.append(_alloc); } void append_character(Codepoint c) { if (_printable(c)) { Text::Index const y { _text.upper_bound().value - 1 }; _text.apply(y, [&] (Line &line) { line.append(c); }); } } /* insert character and advance cursor */ void insert_at_cursor_position(Codepoint); void gen_clipboard_content(Xml_generator &xml) const; template void for_each_character(FN const &fn) const { _text.for_each([&] (Text::Index at, Line const &line) { line.for_each([&] (Line::Index, Character c) { fn(c); }); if (at.value + 1 < _text.upper_bound().value) fn(Codepoint{'\n'}); }); } }; #endif /* _DIALOG_H_ */