Statusbar.cpp 5.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176
  1. /*
  2. * Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <LibGUI/BoxLayout.h>
  8. #include <LibGUI/Painter.h>
  9. #include <LibGUI/ResizeCorner.h>
  10. #include <LibGUI/Statusbar.h>
  11. #include <LibGUI/Window.h>
  12. #include <LibGfx/Palette.h>
  13. #include <LibGfx/StylePainter.h>
  14. REGISTER_WIDGET(GUI, Statusbar)
  15. namespace GUI {
  16. Statusbar::Statusbar(int count)
  17. {
  18. set_fixed_height(18);
  19. set_layout<HorizontalBoxLayout>(0, 2);
  20. m_corner = add<ResizeCorner>();
  21. set_segment_count(count);
  22. REGISTER_STRING_PROPERTY("text", text, set_text);
  23. REGISTER_INT_PROPERTY("segment_count", segment_count, set_segment_count);
  24. }
  25. NonnullRefPtr<Statusbar::Segment> Statusbar::create_segment()
  26. {
  27. auto widget = Segment::construct();
  28. insert_child_before(*widget, *m_corner);
  29. return widget;
  30. }
  31. void Statusbar::child_event(Core::ChildEvent& event)
  32. {
  33. auto& event_to_forward = event;
  34. // To ensure that the ResizeCorner is always the last widget, and thus stays in the corner,
  35. // we replace ChildAdded events that do not request specific placement with events that request placement before the corner
  36. if (event.type() == Event::ChildAdded && is<Widget>(*event.child()) && !event.insertion_before_child()) {
  37. Core::ChildEvent new_event(Event::ChildAdded, *event.child(), m_corner.ptr());
  38. event_to_forward = new_event;
  39. }
  40. return Widget::child_event(event_to_forward);
  41. }
  42. void Statusbar::set_segment_count(size_t count)
  43. {
  44. if (count <= 1)
  45. count = 1;
  46. for (auto i = m_segments.size(); i < count; i++) {
  47. auto segment = create_segment();
  48. m_segments.append(move(segment));
  49. }
  50. }
  51. void Statusbar::update_segment(size_t index)
  52. {
  53. auto& segment = m_segments.at(index);
  54. if (segment.mode() == Segment::Mode::Auto) {
  55. if (segment.restored_text().is_empty())
  56. segment.set_visible(false);
  57. else {
  58. constexpr auto horizontal_padding { 10 };
  59. auto width = font().width(segment.restored_text()) + horizontal_padding;
  60. segment.set_restored_width(width);
  61. segment.set_fixed_width(width);
  62. }
  63. } else if (segment.mode() == Segment::Mode::Fixed) {
  64. if (segment.max_width().is_int()) {
  65. segment.set_restored_width(segment.max_width().as_int());
  66. segment.set_fixed_width(segment.max_width());
  67. }
  68. }
  69. if (segment.override_text().is_null()) {
  70. for (size_t i = 1; i < m_segments.size(); i++) {
  71. if (!text(i).is_empty())
  72. m_segments[i].set_visible(true);
  73. }
  74. segment.set_text(String::from_utf8(segment.restored_text()).release_value_but_fixme_should_propagate_errors());
  75. segment.set_frame_shape(Gfx::FrameShape::Panel);
  76. if (segment.mode() != Segment::Mode::Proportional)
  77. segment.set_fixed_width(segment.restored_width());
  78. } else {
  79. for (size_t i = 1; i < m_segments.size(); i++) {
  80. if (!m_segments[i].is_clickable())
  81. m_segments[i].set_visible(false);
  82. }
  83. segment.set_text(String::from_utf8(segment.override_text()).release_value_but_fixme_should_propagate_errors());
  84. segment.set_frame_shape(Gfx::FrameShape::NoFrame);
  85. if (segment.mode() != Segment::Mode::Proportional)
  86. segment.set_fixed_width(SpecialDimension::Grow);
  87. }
  88. }
  89. DeprecatedString Statusbar::text(size_t index) const
  90. {
  91. return m_segments.at(index).text().to_deprecated_string();
  92. }
  93. void Statusbar::set_text(DeprecatedString text)
  94. {
  95. set_text(0, move(text));
  96. }
  97. void Statusbar::set_text(size_t index, DeprecatedString text)
  98. {
  99. m_segments.at(index).m_restored_text = move(text);
  100. update_segment(index);
  101. }
  102. void Statusbar::set_override_text(DeprecatedString override_text)
  103. {
  104. m_segments.at(0).m_override_text = move(override_text);
  105. update_segment(0);
  106. }
  107. void Statusbar::paint_event(PaintEvent& event)
  108. {
  109. Painter painter(*this);
  110. painter.add_clip_rect(event.rect());
  111. painter.fill_rect(rect(), palette().button());
  112. }
  113. void Statusbar::resize_event(ResizeEvent& event)
  114. {
  115. if (auto* window = this->window()) {
  116. m_corner->set_visible(window->is_resizable() && !window->is_maximized());
  117. }
  118. Widget::resize_event(event);
  119. }
  120. Statusbar::Segment::Segment()
  121. {
  122. set_fixed_height(18);
  123. set_focus_policy(GUI::FocusPolicy::NoFocus);
  124. set_button_style(Gfx::ButtonStyle::Tray);
  125. set_text_alignment(Gfx::TextAlignment::CenterLeft);
  126. }
  127. void Statusbar::Segment::paint_event(PaintEvent& event)
  128. {
  129. Painter painter(*this);
  130. painter.add_clip_rect(event.rect());
  131. Gfx::StylePainter::current().paint_frame(painter, rect(), palette(), m_shape, Gfx::FrameShadow::Sunken, m_thickness, spans_entire_window_horizontally());
  132. if (is_clickable())
  133. Button::paint_event(event);
  134. else if (!text().is_empty())
  135. painter.draw_text(rect().shrunken(font().max_glyph_width(), 0), text(), text_alignment(), palette().color(foreground_role()), Gfx::TextElision::Right, Gfx::TextWrapping::DontWrap);
  136. }
  137. void Statusbar::Segment::mousedown_event(MouseEvent& event)
  138. {
  139. if (!is_clickable())
  140. return;
  141. Button::mousedown_event(event);
  142. }
  143. void Statusbar::Segment::mouseup_event(MouseEvent& event)
  144. {
  145. if (!is_clickable())
  146. return;
  147. Button::mouseup_event(event);
  148. }
  149. }