ItemListModel.h 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. /*
  2. * Copyright (c) 2019-2020, Jesse Buhgaiar <jooster669@gmail.com>
  3. * Copyright (c) 2022, the SerenityOS developers.
  4. *
  5. * SPDX-License-Identifier: BSD-2-Clause
  6. */
  7. #pragma once
  8. #include <AK/NonnullRefPtr.h>
  9. #include <AK/Vector.h>
  10. #include <LibGUI/Model.h>
  11. namespace GUI {
  12. template<typename T, typename Container = Vector<T>, typename ColumnNameListType = void>
  13. class ItemListModel : public Model {
  14. public:
  15. static constexpr auto IsTwoDimensional = requires(Container data) {
  16. requires !IsVoid<ColumnNameListType>;
  17. data.at(0).at(0);
  18. data.at(0).size();
  19. };
  20. // Substitute 'void' for a dummy u8.
  21. using ColumnNamesT = Conditional<IsVoid<ColumnNameListType>, u8, ColumnNameListType>;
  22. static NonnullRefPtr<ItemListModel> create(Container const& data, ColumnNamesT const& column_names, Optional<size_t> const& row_count = {})
  23. requires(IsTwoDimensional)
  24. {
  25. return adopt_ref(*new ItemListModel<T, Container, ColumnNameListType>(data, column_names, row_count));
  26. }
  27. static NonnullRefPtr<ItemListModel> create(Container const& data, Optional<size_t> const& row_count = {})
  28. requires(!IsTwoDimensional)
  29. {
  30. return adopt_ref(*new ItemListModel<T, Container>(data, row_count));
  31. }
  32. virtual ~ItemListModel() override = default;
  33. virtual int row_count(ModelIndex const& index) const override
  34. {
  35. if (!index.is_valid())
  36. return m_provided_row_count.has_value() ? *m_provided_row_count : m_data.size();
  37. return 0;
  38. }
  39. virtual int column_count(ModelIndex const& index) const override
  40. {
  41. // if it's 2D (e.g. Vector<Vector<T>>)
  42. if constexpr (IsTwoDimensional) {
  43. if (index.is_valid())
  44. return m_data.at(index.row()).size();
  45. if (m_data.size())
  46. return m_data.at(0).size();
  47. return 0;
  48. }
  49. // Otherwise, let's just assume it's 1D.
  50. return 1;
  51. }
  52. virtual DeprecatedString column_name(int index) const override
  53. {
  54. if constexpr (IsTwoDimensional)
  55. return m_column_names[index];
  56. return "Data";
  57. }
  58. virtual Variant data(ModelIndex const& index, ModelRole role) const override
  59. {
  60. if (role == ModelRole::TextAlignment)
  61. return Gfx::TextAlignment::CenterLeft;
  62. if (role == ModelRole::Display) {
  63. if constexpr (IsTwoDimensional)
  64. return m_data.at(index.row()).at(index.column());
  65. else
  66. return m_data.at(index.row());
  67. }
  68. return {};
  69. }
  70. virtual TriState data_matches(GUI::ModelIndex const& index, GUI::Variant const& term) const override
  71. {
  72. if (index.data().as_string().contains(term.as_string(), CaseSensitivity::CaseInsensitive))
  73. return TriState::True;
  74. return TriState::False;
  75. }
  76. virtual bool is_searchable() const override { return true; }
  77. virtual Vector<GUI::ModelIndex> matches(StringView searching, unsigned flags, GUI::ModelIndex const&) override
  78. {
  79. Vector<GUI::ModelIndex> found_indices;
  80. if constexpr (IsTwoDimensional) {
  81. for (auto it = m_data.begin(); it != m_data.end(); ++it) {
  82. for (auto it2d = (*it).begin(); it2d != (*it).end(); ++it2d) {
  83. GUI::ModelIndex index = this->index(it.index(), it2d.index());
  84. if (!string_matches(data(index, ModelRole::Display).to_deprecated_string(), searching, flags))
  85. continue;
  86. found_indices.append(index);
  87. if (flags & FirstMatchOnly)
  88. return found_indices;
  89. }
  90. }
  91. } else {
  92. for (auto it = m_data.begin(); it != m_data.end(); ++it) {
  93. GUI::ModelIndex index = this->index(it.index());
  94. if (!string_matches(data(index, ModelRole::Display).to_deprecated_string(), searching, flags))
  95. continue;
  96. found_indices.append(index);
  97. if (flags & FirstMatchOnly)
  98. return found_indices;
  99. }
  100. }
  101. return found_indices;
  102. }
  103. protected:
  104. explicit ItemListModel(Container const& data, Optional<size_t> row_count = {})
  105. requires(!IsTwoDimensional)
  106. : m_data(data)
  107. , m_provided_row_count(move(row_count))
  108. {
  109. }
  110. explicit ItemListModel(Container const& data, ColumnNamesT const& column_names, Optional<size_t> row_count = {})
  111. requires(IsTwoDimensional)
  112. : m_data(data)
  113. , m_column_names(column_names)
  114. , m_provided_row_count(move(row_count))
  115. {
  116. }
  117. Container const& m_data;
  118. ColumnNamesT m_column_names;
  119. Optional<size_t> m_provided_row_count;
  120. };
  121. }