ScrollableContainerWidget.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. /*
  2. * Copyright (c) 2021, Andreas Kling <kling@serenityos.org>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #include <AK/RefPtr.h>
  8. #include <LibGUI/Layout.h>
  9. #include <LibGUI/ScrollableContainerWidget.h>
  10. REGISTER_WIDGET(GUI, ScrollableContainerWidget)
  11. namespace GUI {
  12. ScrollableContainerWidget::ScrollableContainerWidget()
  13. {
  14. REGISTER_BOOL_PROPERTY("scrollbars_enabled", is_scrollbars_enabled, set_scrollbars_enabled);
  15. REGISTER_BOOL_PROPERTY("should_hide_unnecessary_scrollbars", should_hide_unnecessary_scrollbars, set_should_hide_unnecessary_scrollbars);
  16. }
  17. void ScrollableContainerWidget::did_scroll()
  18. {
  19. AbstractScrollableWidget::did_scroll();
  20. update_widget_position();
  21. }
  22. void ScrollableContainerWidget::update_widget_position()
  23. {
  24. if (!m_widget)
  25. return;
  26. m_widget->move_to(-horizontal_scrollbar().value() + content_margins().left(), -vertical_scrollbar().value() + content_margins().top());
  27. }
  28. void ScrollableContainerWidget::update_widget_size()
  29. {
  30. if (!m_widget)
  31. return;
  32. m_widget->do_layout();
  33. if (m_widget->is_shrink_to_fit() && m_widget->layout()) {
  34. auto new_size = Widget::content_size();
  35. auto preferred_size = m_widget->layout()->preferred_size();
  36. if (preferred_size.width() != -1)
  37. new_size.set_width(preferred_size.width());
  38. if (preferred_size.height() != -1)
  39. new_size.set_height(preferred_size.height());
  40. m_widget->resize(new_size);
  41. set_content_size(new_size);
  42. } else {
  43. auto inner_size = Widget::content_size();
  44. auto min_size = m_widget->min_size();
  45. auto new_size = Gfx::Size {
  46. max(inner_size.width(), min_size.width()),
  47. max(inner_size.height(), min_size.height())
  48. };
  49. m_widget->resize(new_size);
  50. set_content_size(new_size);
  51. }
  52. }
  53. void ScrollableContainerWidget::resize_event(GUI::ResizeEvent& event)
  54. {
  55. AbstractScrollableWidget::resize_event(event);
  56. update_widget_position();
  57. update_widget_size();
  58. }
  59. void ScrollableContainerWidget::set_widget(GUI::Widget* widget)
  60. {
  61. if (m_widget == widget)
  62. return;
  63. if (m_widget)
  64. remove_child(*m_widget);
  65. m_widget = widget;
  66. if (m_widget) {
  67. add_child(*m_widget);
  68. m_widget->move_to_back();
  69. }
  70. update_widget_size();
  71. update_widget_position();
  72. }
  73. bool ScrollableContainerWidget::load_from_gml_ast(NonnullRefPtr<GUI::GML::Node> ast, RefPtr<Core::Object> (*unregistered_child_handler)(String const&))
  74. {
  75. if (is<GUI::GML::GMLFile>(ast.ptr()))
  76. return load_from_gml_ast(static_ptr_cast<GUI::GML::GMLFile>(ast)->main_class(), unregistered_child_handler);
  77. VERIFY(is<GUI::GML::Object>(ast.ptr()));
  78. auto object = static_ptr_cast<GUI::GML::Object>(ast);
  79. object->for_each_property([&](auto key, auto value) {
  80. set_property(key, value);
  81. });
  82. auto content_widget_value = object->get_property("content_widget"sv);
  83. if (!content_widget_value.is_null() && !is<Object>(content_widget_value.ptr())) {
  84. dbgln("content widget is not an object");
  85. return false;
  86. }
  87. auto has_children = false;
  88. object->for_each_child_object([&](auto) { has_children = true; });
  89. if (has_children) {
  90. dbgln("children specified for ScrollableContainerWidget, but only 1 widget as content_widget is supported");
  91. return false;
  92. }
  93. if (!content_widget_value.is_null() && is<Object>(content_widget_value.ptr())) {
  94. auto content_widget = static_ptr_cast<GUI::GML::Object>(content_widget_value);
  95. auto class_name = content_widget->name();
  96. RefPtr<Core::Object> child;
  97. if (auto* registration = Core::ObjectClassRegistration::find(class_name)) {
  98. child = registration->construct();
  99. } else {
  100. child = unregistered_child_handler(class_name);
  101. }
  102. if (!child)
  103. return false;
  104. auto widget_ptr = verify_cast<GUI::Widget>(child.ptr());
  105. set_widget(widget_ptr);
  106. static_ptr_cast<Widget>(child)->load_from_gml_ast(content_widget.release_nonnull(), unregistered_child_handler);
  107. return true;
  108. }
  109. return true;
  110. }
  111. }