ItemListModel.h 4.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  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. {
  17. requires !IsVoid<ColumnNameListType>;
  18. data.at(0).at(0);
  19. data.at(0).size();
  20. };
  21. // Substitute 'void' for a dummy u8.
  22. using ColumnNamesT = Conditional<IsVoid<ColumnNameListType>, u8, ColumnNameListType>;
  23. static NonnullRefPtr<ItemListModel> create(Container const& data, ColumnNamesT const& column_names, Optional<size_t> const& row_count = {}) 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 = {}) requires(!IsTwoDimensional)
  28. {
  29. return adopt_ref(*new ItemListModel<T, Container>(data, row_count));
  30. }
  31. virtual ~ItemListModel() override = default;
  32. virtual int row_count(ModelIndex const& index) const override
  33. {
  34. if (!index.is_valid())
  35. return m_provided_row_count.has_value() ? *m_provided_row_count : m_data.size();
  36. return 0;
  37. }
  38. virtual int column_count(ModelIndex const& index) const override
  39. {
  40. // if it's 2D (e.g. Vector<Vector<T>>)
  41. if constexpr (IsTwoDimensional) {
  42. if (index.is_valid())
  43. return m_data.at(index.row()).size();
  44. if (m_data.size())
  45. return m_data.at(0).size();
  46. return 0;
  47. }
  48. // Otherwise, let's just assume it's 1D.
  49. return 1;
  50. }
  51. virtual String column_name(int index) const override
  52. {
  53. if constexpr (IsTwoDimensional)
  54. return m_column_names[index];
  55. return "Data";
  56. }
  57. virtual Variant data(ModelIndex const& index, ModelRole role) const override
  58. {
  59. if (role == ModelRole::TextAlignment)
  60. return Gfx::TextAlignment::CenterLeft;
  61. if (role == ModelRole::Display) {
  62. if constexpr (IsTwoDimensional)
  63. return m_data.at(index.row()).at(index.column());
  64. else
  65. return m_data.at(index.row());
  66. }
  67. return {};
  68. }
  69. virtual TriState data_matches(GUI::ModelIndex const& index, GUI::Variant const& term) const override
  70. {
  71. if (index.data().as_string().contains(term.as_string(), CaseSensitivity::CaseInsensitive))
  72. return TriState::True;
  73. return TriState::False;
  74. }
  75. virtual bool is_searchable() const override { return true; }
  76. virtual Vector<GUI::ModelIndex> matches(StringView searching, unsigned flags, GUI::ModelIndex const&) override
  77. {
  78. Vector<GUI::ModelIndex> found_indices;
  79. if constexpr (IsTwoDimensional) {
  80. for (auto it = m_data.begin(); it != m_data.end(); ++it) {
  81. for (auto it2d = (*it).begin(); it2d != (*it).end(); ++it2d) {
  82. GUI::ModelIndex index = this->index(it.index(), it2d.index());
  83. if (!string_matches(data(index, ModelRole::Display).to_string(), searching, flags))
  84. continue;
  85. found_indices.append(index);
  86. if (flags & FirstMatchOnly)
  87. return found_indices;
  88. }
  89. }
  90. } else {
  91. for (auto it = m_data.begin(); it != m_data.end(); ++it) {
  92. GUI::ModelIndex index = this->index(it.index());
  93. if (!string_matches(data(index, ModelRole::Display).to_string(), searching, flags))
  94. continue;
  95. found_indices.append(index);
  96. if (flags & FirstMatchOnly)
  97. return found_indices;
  98. }
  99. }
  100. return found_indices;
  101. }
  102. protected:
  103. explicit ItemListModel(Container const& data, Optional<size_t> row_count = {}) requires(!IsTwoDimensional)
  104. : m_data(data)
  105. , m_provided_row_count(move(row_count))
  106. {
  107. }
  108. explicit ItemListModel(Container const& data, ColumnNamesT const& column_names, Optional<size_t> row_count = {}) requires(IsTwoDimensional)
  109. : m_data(data)
  110. , m_column_names(column_names)
  111. , m_provided_row_count(move(row_count))
  112. {
  113. }
  114. Container const& m_data;
  115. ColumnNamesT m_column_names;
  116. Optional<size_t> m_provided_row_count;
  117. };
  118. }