ScrollableContainerWidget.cpp 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138
  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. auto new_size = Widget::content_size();
  34. if (m_widget->layout()) {
  35. auto preferred_size = m_widget->layout()->preferred_size();
  36. if (m_widget->is_shrink_to_fit()) {
  37. if (preferred_size.width() != -1)
  38. new_size.set_width(preferred_size.width());
  39. if (preferred_size.height() != -1)
  40. new_size.set_height(preferred_size.height());
  41. } else {
  42. new_size = Gfx::Size {
  43. max(new_size.width(), preferred_size.width()),
  44. max(new_size.height(), preferred_size.height())
  45. };
  46. }
  47. }
  48. auto min_size = m_widget->min_size();
  49. new_size = Gfx::Size {
  50. max(new_size.width(), min_size.width()),
  51. max(new_size.height(), min_size.height()),
  52. };
  53. m_widget->resize(new_size);
  54. set_content_size(new_size);
  55. }
  56. void ScrollableContainerWidget::resize_event(GUI::ResizeEvent& event)
  57. {
  58. AbstractScrollableWidget::resize_event(event);
  59. update_widget_position();
  60. update_widget_size();
  61. }
  62. void ScrollableContainerWidget::set_widget(GUI::Widget* widget)
  63. {
  64. if (m_widget == widget)
  65. return;
  66. if (m_widget)
  67. remove_child(*m_widget);
  68. m_widget = widget;
  69. if (m_widget) {
  70. add_child(*m_widget);
  71. m_widget->move_to_back();
  72. }
  73. update_widget_size();
  74. update_widget_position();
  75. }
  76. bool ScrollableContainerWidget::load_from_gml_ast(NonnullRefPtr<GUI::GML::Node> ast, RefPtr<Core::Object> (*unregistered_child_handler)(String const&))
  77. {
  78. if (is<GUI::GML::GMLFile>(ast.ptr()))
  79. return load_from_gml_ast(static_ptr_cast<GUI::GML::GMLFile>(ast)->main_class(), unregistered_child_handler);
  80. VERIFY(is<GUI::GML::Object>(ast.ptr()));
  81. auto object = static_ptr_cast<GUI::GML::Object>(ast);
  82. object->for_each_property([&](auto key, auto value) {
  83. set_property(key, value);
  84. });
  85. auto content_widget_value = object->get_property("content_widget"sv);
  86. if (!content_widget_value.is_null() && !is<GUI::GML::Object>(content_widget_value.ptr())) {
  87. dbgln("content widget is not an object");
  88. return false;
  89. }
  90. auto has_children = false;
  91. object->for_each_child_object([&](auto) { has_children = true; });
  92. if (has_children) {
  93. dbgln("children specified for ScrollableContainerWidget, but only 1 widget as content_widget is supported");
  94. return false;
  95. }
  96. if (!content_widget_value.is_null() && is<GUI::GML::Object>(content_widget_value.ptr())) {
  97. auto content_widget = static_ptr_cast<GUI::GML::Object>(content_widget_value);
  98. auto class_name = content_widget->name();
  99. RefPtr<Core::Object> child;
  100. if (auto* registration = Core::ObjectClassRegistration::find(class_name)) {
  101. child = registration->construct();
  102. } else {
  103. child = unregistered_child_handler(class_name);
  104. }
  105. if (!child)
  106. return false;
  107. auto widget_ptr = verify_cast<GUI::Widget>(child.ptr());
  108. set_widget(widget_ptr);
  109. static_ptr_cast<Widget>(child)->load_from_gml_ast(content_widget.release_nonnull(), unregistered_child_handler);
  110. return true;
  111. }
  112. return true;
  113. }
  114. }