aboutsummaryrefslogtreecommitdiff
path: root/src/ui/controls/text_common.cpp
blob: d6bcdd03c3569c88c24439e5a7f1d64cd717ac8b (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
#include "cru/ui/controls/text_common.hpp"

#include "cru/common/logger.hpp"
#include "cru/platform/graph/font.hpp"
#include "cru/platform/graph/painter.hpp"
#include "cru/platform/native/ui_application.hpp"
#include "cru/ui/control.hpp"
#include "cru/ui/render/canvas_render_object.hpp"
#include "cru/ui/render/text_render_object.hpp"

#include <cassert>

namespace cru::ui::controls {
using platform::graph::ITextLayout;

constexpr float caret_width = 2;
constexpr long long caret_blink_duration = 500;

TextControlService::TextControlService(Control* control,
                                       ITextControl* text_control)
    : control_(control), text_control_(text_control) {
  event_revoker_guards_.push_back(
      EventRevokerGuard{control->MouseDownEvent()->Direct()->AddHandler(
          [this](event::MouseButtonEventArgs& args) {
            if (this->select_down_button_.has_value()) {
              return;
            } else {
              if (!this->control_->CaptureMouse()) return;
              const auto text_render_object =
                  this->text_control_->GetTextRenderObject();
              this->select_down_button_ = args.GetButton();
              const auto result = text_render_object->TextHitTest(
                  text_render_object->FromRootToContent(args.GetPoint()));
              const auto position = result.position + (result.trailing ? 1 : 0);
              this->select_start_position_ = position;
              log::Debug("Begin to select text. Position: {}", position);
            }
          })});

  event_revoker_guards_.push_back(
      EventRevokerGuard{control->MouseMoveEvent()->Direct()->AddHandler(
          [this](event::MouseEventArgs& args) {
            if (this->select_down_button_.has_value()) {
              const auto text_render_object =
                  this->text_control_->GetTextRenderObject();
              const auto result = text_render_object->TextHitTest(
                  text_render_object->FromRootToContent(args.GetPoint()));
              const auto position = result.position + (result.trailing ? 1 : 0);
              this->caret_position_ = position;
              log::Debug("Text select range: {} : {}.", position,
                         this->select_start_position_);
              this->text_control_->GetTextRenderObject()->SetSelectionRange(
                  TextRange::FromTwoSides(
                      static_cast<unsigned>(position),
                      static_cast<unsigned>(this->select_start_position_)));
              this->text_control_->GetTextRenderObject()->InvalidatePaint();
              this->text_control_->GetCaretRenderObject()->InvalidatePaint();
            }
          })});

  event_revoker_guards_.push_back(
      EventRevokerGuard{control->MouseUpEvent()->Direct()->AddHandler(
          [this](event::MouseButtonEventArgs& args) {
            if (this->select_down_button_.has_value() &&
                this->select_down_button_.value() == args.GetButton()) {
              this->control_->ReleaseMouse();
              this->select_down_button_ = std::nullopt;
              log::Debug("End selecting text.");
            }
          })});
}

TextControlService::~TextControlService() { TearDownCaretTimer(); }

void TextControlService::SetCaretVisible(bool visible) {
  if (visible) {
    SetupCaretTimer();
  } else {
    TearDownCaretTimer();
  }
}

void TextControlService::DrawCaret(platform::graph::IPainter* painter) {
  if (caret_show_) {
    const auto text_render_object = text_control_->GetTextRenderObject();
    const auto point = text_render_object->TextSingleRect(
        caret_position_, false);  // Maybe cache the result???
    painter->FillRectangle(
        Rect{point,
             Size{caret_width, text_render_object->GetFont()->GetFontSize()}},
        text_control_->GetCaretBrush());
  }
}

void TextControlService::SetupCaretTimer() {
  if (caret_visible_) return;

  caret_visible_ = true;
  caret_timer_tag_ =
      platform::native::IUiApplication::GetInstance()->SetInterval(
          std::chrono::milliseconds(caret_blink_duration), [this] {
            this->caret_show_ = !this->caret_show_;
            this->text_control_->GetCaretRenderObject()->InvalidatePaint();
          });
}

void TextControlService::TearDownCaretTimer() {
  if (!caret_visible_) return;
  caret_visible_ = false;
  platform::native::IUiApplication::GetInstance()->CancelTimer(
      caret_timer_tag_);
}
}  // namespace cru::ui::controls