GBoxLayout.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132
  1. #include <LibGUI/GBoxLayout.h>
  2. #include <LibGUI/GWidget.h>
  3. #include <stdio.h>
  4. //#define GBOXLAYOUT_DEBUG
  5. GBoxLayout::GBoxLayout(Orientation orientation)
  6. : m_orientation(orientation)
  7. {
  8. }
  9. GBoxLayout::~GBoxLayout()
  10. {
  11. }
  12. void GBoxLayout::run(GWidget& widget)
  13. {
  14. bool should_log = false;
  15. #ifdef GBOXLAYOUT_DEBUG
  16. should_log = true;
  17. #endif
  18. if (should_log)
  19. dbgprintf("GBoxLayout: running layout on %s{%p}, entry count: %d\n", widget.class_name(), &widget, m_entries.size());
  20. if (m_entries.is_empty())
  21. return;
  22. Size available_size = widget.size();
  23. int number_of_entries_with_fixed_size = 0;
  24. int number_of_visible_entries = 0;
  25. if (should_log)
  26. dbgprintf("GBoxLayout: Starting with available size: %s\n", available_size.to_string().characters());
  27. for (auto& entry : m_entries) {
  28. if (entry.type == Entry::Type::Spacer) {
  29. ++number_of_visible_entries;
  30. }
  31. if (!entry.widget)
  32. continue;
  33. if (!entry.widget->is_visible())
  34. continue;
  35. ++number_of_visible_entries;
  36. if (entry.widget && entry.widget->size_policy(orientation()) == SizePolicy::Fixed) {
  37. if (should_log) {
  38. dbgprintf("GBoxLayout: Subtracting for fixed %s{%p}, size: %s\n", entry.widget->class_name(), entry.widget.ptr(), entry.widget->preferred_size().to_string().characters());
  39. dbgprintf("GBoxLayout: Available size before: %s\n", available_size.to_string().characters());
  40. }
  41. available_size -= entry.widget->preferred_size();
  42. if (should_log)
  43. dbgprintf("GBoxLayout: Available size after: %s\n", available_size.to_string().characters());
  44. ++number_of_entries_with_fixed_size;
  45. }
  46. available_size -= { spacing(), spacing() };
  47. }
  48. available_size += { spacing(), spacing() };
  49. available_size -= { margins().left() + margins().right(), margins().top() + margins().bottom() };
  50. if (should_log)
  51. dbgprintf("GBoxLayout: Number of visible: %d/%d\n", number_of_visible_entries, m_entries.size());
  52. int number_of_entries_with_automatic_size = number_of_visible_entries - number_of_entries_with_fixed_size;
  53. if (should_log)
  54. dbgprintf("GBoxLayout: available_size=%s, fixed=%d, fill=%d\n", available_size.to_string().characters(), number_of_entries_with_fixed_size, number_of_entries_with_automatic_size);
  55. Size automatic_size;
  56. if (number_of_entries_with_automatic_size) {
  57. if (m_orientation == Orientation::Horizontal) {
  58. automatic_size.set_width(available_size.width() / number_of_entries_with_automatic_size);
  59. automatic_size.set_height(widget.height());
  60. } else {
  61. automatic_size.set_width(widget.width());
  62. automatic_size.set_height(available_size.height() / number_of_entries_with_automatic_size);
  63. }
  64. }
  65. if (should_log)
  66. dbgprintf("GBoxLayout: automatic_size=%s\n", automatic_size.to_string().characters());
  67. int current_x = margins().left();
  68. int current_y = margins().top();
  69. for (auto& entry : m_entries) {
  70. if (entry.type == Entry::Type::Spacer) {
  71. current_x += automatic_size.width();
  72. current_y += automatic_size.height();
  73. }
  74. if (!entry.widget)
  75. continue;
  76. if (!entry.widget->is_visible())
  77. continue;
  78. Rect rect(current_x, current_y, 0, 0);
  79. if (entry.layout) {
  80. // FIXME: Implement recursive layout.
  81. ASSERT_NOT_REACHED();
  82. }
  83. ASSERT(entry.widget);
  84. rect.set_size(automatic_size);
  85. if (entry.widget->size_policy(Orientation::Vertical) == SizePolicy::Fixed)
  86. rect.set_height(entry.widget->preferred_size().height());
  87. if (entry.widget->size_policy(Orientation::Horizontal) == SizePolicy::Fixed)
  88. rect.set_width(entry.widget->preferred_size().width());
  89. if (orientation() == Orientation::Horizontal) {
  90. if (entry.widget->size_policy(Orientation::Vertical) == SizePolicy::Fill)
  91. rect.set_height(widget.height() - margins().top() - margins().bottom());
  92. rect.center_vertically_within(widget.rect());
  93. } else {
  94. if (entry.widget->size_policy(Orientation::Horizontal) == SizePolicy::Fill)
  95. rect.set_width(widget.width() - margins().left() - margins().right());
  96. rect.center_horizontally_within(widget.rect());
  97. }
  98. if (should_log)
  99. dbgprintf("GBoxLayout: apply, %s{%p} <- %s\n", entry.widget->class_name(), entry.widget.ptr(), rect.to_string().characters());
  100. entry.widget->set_relative_rect(rect);
  101. if (orientation() == Orientation::Horizontal)
  102. current_x += rect.width() + spacing();
  103. else
  104. current_y += rect.height() + spacing();
  105. }
  106. }