table_view.hpp 29 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756
  1. /*************************************************************************
  2. *
  3. * Copyright 2016 Realm Inc.
  4. *
  5. * Licensed under the Apache License, Version 2.0 (the "License");
  6. * you may not use this file except in compliance with the License.
  7. * You may obtain a copy of the License at
  8. *
  9. * http://www.apache.org/licenses/LICENSE-2.0
  10. *
  11. * Unless required by applicable law or agreed to in writing, software
  12. * distributed under the License is distributed on an "AS IS" BASIS,
  13. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  14. * See the License for the specific language governing permissions and
  15. * limitations under the License.
  16. *
  17. **************************************************************************/
  18. #ifndef REALM_TABLE_VIEW_HPP
  19. #define REALM_TABLE_VIEW_HPP
  20. #include <realm/sort_descriptor.hpp>
  21. #include <realm/table.hpp>
  22. #include <realm/util/features.h>
  23. #include <realm/obj_list.hpp>
  24. #include <realm/list.hpp>
  25. #include <realm/set.hpp>
  26. namespace realm {
  27. // Views, tables and synchronization between them:
  28. //
  29. // Views are built through queries against either tables or another view.
  30. // Views may be restricted to only hold entries provided by another view.
  31. // this other view is called the "restricting view".
  32. // Views may be sorted in ascending or descending order of values in one ore more columns.
  33. //
  34. // Views remember the query from which it was originally built.
  35. // Views remember the table from which it was originally built.
  36. // Views remember a restricting view if one was used when it was originally built.
  37. // Views remember the sorting criteria (columns and direction)
  38. //
  39. // A view may be operated in one of two distinct modes: *reflective* and *imperative*.
  40. // Sometimes the term "reactive" is used instead of "reflective" with the same meaning.
  41. //
  42. // Reflective views:
  43. // - A reflective view *always* *reflect* the result of running the query.
  44. // If the underlying tables or tableviews change, the reflective view changes as well.
  45. // A reflective view may need to rerun the query it was generated from, a potentially
  46. // costly operation which happens on demand.
  47. // - It does not matter whether changes are explicitly done within the transaction, or
  48. // occur implicitly as part of advance_read() or promote_to_write().
  49. //
  50. // Imperative views:
  51. // - An imperative view only *initially* holds the result of the query. An imperative
  52. // view *never* reruns the query. To force the view to match it's query (by rerunning it),
  53. // the view must be operated in reflective mode.
  54. // An imperative view can be modified explicitly. References can be added, removed or
  55. // changed.
  56. //
  57. // - In imperative mode, the references in the view tracks movement of the referenced data:
  58. // If you delete an entry which is referenced from a view, said reference is detached,
  59. // not removed.
  60. // - It does not matter whether the delete is done in-line (as part of the current transaction),
  61. // or if it is done implicitly as part of advance_read() or promote_to_write().
  62. //
  63. // The choice between reflective and imperative views might eventually be represented by a
  64. // switch on the tableview, but isn't yet. For now, clients (bindings) must call sync_if_needed()
  65. // to get reflective behavior.
  66. //
  67. // Use cases:
  68. //
  69. // 1. Presenting data
  70. // The first use case (and primary motivator behind the reflective view) is to just track
  71. // and present the state of the database. In this case, the view is operated in reflective
  72. // mode, it is not modified within the transaction, and it is not used to modify data in
  73. // other parts of the database.
  74. //
  75. // 2. Background execution
  76. // This is the second use case. The implicit rerun of the query in our first use case
  77. // may be too costly to be acceptable on the main thread. Instead you want to run the query
  78. // on a worker thread, but display it on the main thread. To achieve this, you need two
  79. // Transactions locked on to the same version of the database. If you have that, you can
  80. // import_copy_of() a view from one transaction to the other. See also db.hpp for more
  81. // information. Technically, you can also import_copy_of into a transaction locked to a
  82. // different version. The imported view will automatically match the importing version.
  83. //
  84. // 3. Iterating a view and changing data
  85. // The third use case (and a motivator behind the imperative view) is when you want
  86. // to make changes to the database in accordance with a query result. Imagine you want to
  87. // find all employees with a salary below a limit and raise their salaries to the limit (pseudocode):
  88. //
  89. // promote_to_write();
  90. // view = table.where().less_than(salary_column,limit).find_all();
  91. // for (size_t i = 0; i < view.size(); ++i) {
  92. // view[i].set(salary_column, limit);
  93. // // add this to get reflective mode: view.sync_if_needed();
  94. // }
  95. // commit_and_continue_as_read();
  96. //
  97. // This is idiomatic imperative code and it works if the view is operated in imperative mode.
  98. //
  99. // If the view is operated in reflective mode, the behaviour surprises most people: When the
  100. // first salary is changed, the entry no longer fullfills the query, so it is dropped from the
  101. // view implicitly. view[0] is removed, view[1] moves to view[0] and so forth. But the next
  102. // loop iteration has i=1 and refers to view[1], thus skipping view[0]. The end result is that
  103. // every other employee get a raise, while the others don't.
  104. //
  105. // 4. Iterating intermixed with implicit updates
  106. // This leads us to use case 4, which is similar to use case 3, but uses promote_to_write()
  107. // intermixed with iterating a view. This is actually quite important to some, who do not want
  108. // to end up with a large write transaction.
  109. //
  110. // view = table.where().less_than(salary_column,limit).find_all();
  111. // for (size_t i = 0; i < view.size(); ++i) {
  112. // promote_to_write();
  113. // view[i].set(salary_column, limit);
  114. // commit_and_continue_as_write();
  115. // }
  116. //
  117. // Anything can happen at the call to promote_to_write(). The key question then becomes: how
  118. // do we support a safe way of realising the original goal (raising salaries) ?
  119. //
  120. // using the imperative operating mode:
  121. //
  122. // view = table.where().less_than(salary_column,limit).find_all();
  123. // for (size_t i = 0; i < view.size(); ++i) {
  124. // promote_to_write();
  125. // // add r.sync_if_needed(); to get reflective mode
  126. // if (r.is_obj_valid(i)) {
  127. // auto r = view[i];
  128. // view[i].set(salary_column, limit);
  129. // }
  130. // commit_and_continue_as_write();
  131. // }
  132. //
  133. // This is safe, and we just aim for providing low level safety: is_obj_valid() can tell
  134. // if the reference is valid, and the references in the view continue to point to the
  135. // same object at all times, also following implicit updates. The rest is up to the
  136. // application logic.
  137. //
  138. // It is important to see, that there is no guarantee that all relevant employees get
  139. // their raise in cases whith concurrent updates. At every call to promote_to_write() new
  140. // employees may be added to the underlying table, but as the view is in imperative mode,
  141. // these new employees are not added to the view. Also at promote_to_write() an existing
  142. // employee could recieve a (different, larger) raise which would then be overwritten and lost.
  143. // However, these are problems that you should expect, since the activity is spread over multiple
  144. // transactions.
  145. /// A ConstTableView gives read access to the parent table, but no
  146. /// write access. The view itself, though, can be changed, for
  147. /// example, it can be sorted.
  148. ///
  149. /// Note that methods are declared 'const' if, and only if they leave
  150. /// the view unmodified, and this is irrespective of whether they
  151. /// modify the parent table.
  152. ///
  153. /// A ConstTableView has both copy and move semantics. See TableView
  154. /// for more on this.
  155. class ConstTableView : public ObjList {
  156. public:
  157. /// Construct null view (no memory allocated).
  158. ConstTableView()
  159. {
  160. }
  161. /// Construct empty view, ready for addition of row indices.
  162. ConstTableView(ConstTableRef parent);
  163. ConstTableView(ConstTableRef parent, Query& query, size_t start, size_t end, size_t limit);
  164. ConstTableView(ConstTableRef parent, ColKey column, const Obj& obj);
  165. ConstTableView(ConstTableRef parent, LnkLstPtr link_list);
  166. ConstTableView(ConstTableRef parent, LnkSetPtr link_list);
  167. /// Copy constructor.
  168. ConstTableView(const ConstTableView&);
  169. /// Move constructor.
  170. ConstTableView(ConstTableView&&) noexcept;
  171. ConstTableView& operator=(const ConstTableView&);
  172. ConstTableView& operator=(ConstTableView&&) noexcept;
  173. ConstTableView(ConstTableView& source, Transaction* tr, PayloadPolicy mode);
  174. ~ConstTableView()
  175. {
  176. }
  177. TableRef get_target_table() const override
  178. {
  179. return m_table.cast_away_const();
  180. }
  181. size_t size() const override
  182. {
  183. return m_key_values.size();
  184. }
  185. bool is_empty() const noexcept
  186. {
  187. return m_key_values.size() == 0;
  188. }
  189. // Tells if the table that this TableView points at still exists or has been deleted.
  190. bool is_attached() const noexcept
  191. {
  192. return bool(m_table);
  193. }
  194. ObjKey get_key(size_t ndx) const override
  195. {
  196. return m_key_values.get(ndx);
  197. }
  198. bool is_obj_valid(size_t ndx) const noexcept override
  199. {
  200. return m_table->is_valid(get_key(ndx));
  201. }
  202. Obj get_object(size_t ndx) const override
  203. {
  204. return m_table.cast_away_const()->get_object(get_key(ndx));
  205. }
  206. // Get the query used to create this TableView
  207. // The query will have a null source table if this tv was not created from
  208. // a query
  209. const Query& get_query() const noexcept
  210. {
  211. return m_query;
  212. }
  213. // Change the TableView to be backed by another query
  214. // only works if the TableView is already backed by a query, and both
  215. // queries points to the same Table
  216. void update_query(const Query& q);
  217. std::unique_ptr<ConstTableView> clone() const
  218. {
  219. return std::unique_ptr<ConstTableView>(new ConstTableView(*this));
  220. }
  221. LinkCollectionPtr clone_obj_list() const final
  222. {
  223. return std::unique_ptr<ConstTableView>(new ConstTableView(*this));
  224. }
  225. // import_copy_of() machinery entry points based on dynamic type. These methods:
  226. // a) forward their calls to the static type entry points.
  227. // b) new/delete patch data structures.
  228. std::unique_ptr<ConstTableView> clone_for_handover(Transaction* tr, PayloadPolicy mode)
  229. {
  230. std::unique_ptr<ConstTableView> retval(new ConstTableView(*this, tr, mode));
  231. return retval;
  232. }
  233. template <Action action, typename T, typename R>
  234. R aggregate(ColKey column_key, size_t* result_count = nullptr, ObjKey* return_key = nullptr) const;
  235. template <typename T>
  236. size_t aggregate_count(ColKey column_key, T count_target) const;
  237. int64_t sum_int(ColKey column_key) const;
  238. int64_t maximum_int(ColKey column_key, ObjKey* return_key = nullptr) const;
  239. int64_t minimum_int(ColKey column_key, ObjKey* return_key = nullptr) const;
  240. double average_int(ColKey column_key, size_t* value_count = nullptr) const;
  241. size_t count_int(ColKey column_key, int64_t target) const;
  242. double sum_float(ColKey column_key) const;
  243. float maximum_float(ColKey column_key, ObjKey* return_key = nullptr) const;
  244. float minimum_float(ColKey column_key, ObjKey* return_key = nullptr) const;
  245. double average_float(ColKey column_key, size_t* value_count = nullptr) const;
  246. size_t count_float(ColKey column_key, float target) const;
  247. double sum_double(ColKey column_key) const;
  248. double maximum_double(ColKey column_key, ObjKey* return_key = nullptr) const;
  249. double minimum_double(ColKey column_key, ObjKey* return_key = nullptr) const;
  250. double average_double(ColKey column_key, size_t* value_count = nullptr) const;
  251. size_t count_double(ColKey column_key, double target) const;
  252. Timestamp minimum_timestamp(ColKey column_key, ObjKey* return_key = nullptr) const;
  253. Timestamp maximum_timestamp(ColKey column_key, ObjKey* return_key = nullptr) const;
  254. size_t count_timestamp(ColKey column_key, Timestamp target) const;
  255. Decimal128 sum_decimal(ColKey column_key) const;
  256. Decimal128 maximum_decimal(ColKey column_key, ObjKey* return_key = nullptr) const;
  257. Decimal128 minimum_decimal(ColKey column_key, ObjKey* return_key = nullptr) const;
  258. Decimal128 average_decimal(ColKey column_key, size_t* value_count = nullptr) const;
  259. size_t count_decimal(ColKey column_key, Decimal128 target) const;
  260. Decimal128 sum_mixed(ColKey column_key) const;
  261. Mixed maximum_mixed(ColKey column_key, ObjKey* return_key = nullptr) const;
  262. Mixed minimum_mixed(ColKey column_key, ObjKey* return_key = nullptr) const;
  263. Decimal128 average_mixed(ColKey column_key, size_t* value_count = nullptr) const;
  264. size_t count_mixed(ColKey column_key, Mixed target) const;
  265. /// Search this view for the specified key. If found, the index of that row
  266. /// within this view is returned, otherwise `realm::not_found` is returned.
  267. size_t find_by_source_ndx(ObjKey key) const noexcept
  268. {
  269. return m_key_values.find_first(key);
  270. }
  271. // Conversion
  272. void to_json(std::ostream&, size_t link_depth = 0) const;
  273. // Determine if the view is 'in sync' with the underlying table
  274. // as well as other views used to generate the view. Note that updates
  275. // through views maintains synchronization between view and table.
  276. // It doesnt by itself maintain other views as well. So if a view
  277. // is generated from another view (not a table), updates may cause
  278. // that view to be outdated, AND as the generated view depends upon
  279. // it, it too will become outdated.
  280. bool is_in_sync() const override;
  281. // A TableView is frozen if it is a) obtained from a query against a frozen table
  282. // and b) is synchronized (is_in_sync())
  283. bool is_frozen()
  284. {
  285. return m_table->is_frozen() && is_in_sync();
  286. }
  287. // Tells if this TableView depends on a LinkList or row that has been deleted.
  288. bool depends_on_deleted_object() const;
  289. // Synchronize a view to match a table or tableview from which it
  290. // has been derived. Synchronization is achieved by rerunning the
  291. // query used to generate the view. If derived from another view, that
  292. // view will be synchronized as well.
  293. //
  294. // "live" or "reactive" views are implemented by calling sync_if_needed()
  295. // before any of the other access-methods whenever the view may have become
  296. // outdated.
  297. void sync_if_needed() const override;
  298. // Return the version of the source it was created from.
  299. TableVersions get_dependency_versions() const
  300. {
  301. TableVersions ret;
  302. get_dependencies(ret);
  303. return ret;
  304. }
  305. // Sort m_key_values according to one column
  306. void sort(ColKey column, bool ascending = true);
  307. // Sort m_key_values according to multiple columns
  308. void sort(SortDescriptor order);
  309. // Get the number of total results which have been filtered out because a number of "LIMIT" operations have
  310. // been applied. This number only applies to the last sync.
  311. size_t get_num_results_excluded_by_limit() const noexcept
  312. {
  313. return m_limit_count;
  314. }
  315. // Remove rows that are duplicated with respect to the column set passed as argument.
  316. // distinct() will preserve the original order of the row pointers, also if the order is a result of sort()
  317. // If two rows are indentical (for the given set of distinct-columns), then the last row is removed.
  318. // You can call sync_if_needed() to update the distinct view, just like you can for a sorted view.
  319. // Each time you call distinct() it will compound on the previous calls
  320. void distinct(ColKey column);
  321. void distinct(DistinctDescriptor columns);
  322. void limit(LimitDescriptor limit);
  323. // Replace the order of sort and distinct operations, bypassing manually
  324. // calling sort and distinct. This is a convenience method for bindings.
  325. void apply_descriptor_ordering(const DescriptorOrdering& new_ordering);
  326. // Gets a readable and parsable string which completely describes the sort and
  327. // distinct operations applied to this view.
  328. std::string get_descriptor_ordering_description() const;
  329. // Returns whether the rows are guaranteed to be in table order.
  330. // This is true only of unsorted TableViews created from either:
  331. // - Table::find_all()
  332. // - Query::find_all() when the query is not restricted to a view.
  333. bool is_in_table_order() const;
  334. bool is_backlink_view() const
  335. {
  336. return m_source_column_key != ColKey();
  337. }
  338. protected:
  339. // This TableView can be "born" from 4 different sources:
  340. // - LinkView
  341. // - Query::find_all()
  342. // - Table::get_distinct_view()
  343. // - Table::get_backlink_view()
  344. void get_dependencies(TableVersions&) const override;
  345. void do_sync();
  346. void do_sort(const DescriptorOrdering&);
  347. mutable ConstTableRef m_table;
  348. // The source column index that this view contain backlinks for.
  349. ColKey m_source_column_key;
  350. // The target object that rows in this view link to.
  351. ObjKey m_linked_obj_key;
  352. ConstTableRef m_linked_table;
  353. // If this TableView was created from a LnkLst, then this reference points to it. Otherwise it's 0
  354. mutable LnkLstPtr m_linklist_source;
  355. // If this TableView was created from a LnkSet, then this reference points to it. Otherwise it's 0
  356. mutable LnkSetPtr m_linkset_source;
  357. // Stores the ordering criteria of applied sort and distinct operations.
  358. DescriptorOrdering m_descriptor_ordering;
  359. size_t m_limit_count = 0;
  360. // A valid query holds a reference to its table which must match our m_table.
  361. // hence we can use a query with a null table reference to indicate that the view
  362. // was NOT generated by a query, but follows a table directly.
  363. Query m_query;
  364. // parameters for findall, needed to rerun the query
  365. size_t m_start = 0;
  366. size_t m_end = size_t(-1);
  367. size_t m_limit = size_t(-1);
  368. // FIXME: This class should eventually be replaced by std::vector<ObjKey>
  369. // It implements a vector of ObjKey, where the elements are held in the
  370. // heap (default allocator is the only option)
  371. class KeyValues : public KeyColumn {
  372. public:
  373. KeyValues()
  374. : KeyColumn(Allocator::get_default())
  375. {
  376. }
  377. KeyValues(const KeyValues&) = delete;
  378. ~KeyValues()
  379. {
  380. destroy();
  381. }
  382. void move_from(KeyValues&);
  383. void copy_from(const KeyValues&);
  384. };
  385. mutable TableVersions m_last_seen_versions;
  386. KeyValues m_key_values;
  387. private:
  388. ObjKey find_first_integer(ColKey column_key, int64_t value) const;
  389. template <class oper>
  390. Timestamp minmax_timestamp(ColKey column_key, ObjKey* return_key) const;
  391. util::RaceDetector m_race_detector;
  392. friend class Table;
  393. friend class Obj;
  394. friend class Query;
  395. friend class DB;
  396. friend class ObjList;
  397. };
  398. enum class RemoveMode { ordered, unordered };
  399. /// A TableView gives read and write access to the parent table.
  400. ///
  401. /// A 'const TableView' cannot be changed (e.g. sorted), nor can the
  402. /// parent table be modified through it.
  403. ///
  404. /// A TableView is both copyable and movable.
  405. class TableView : public ConstTableView {
  406. public:
  407. using ConstTableView::ConstTableView;
  408. TableView() = default;
  409. TableRef get_parent() noexcept
  410. {
  411. return m_table.cast_away_const();
  412. }
  413. // Rows
  414. Obj get(size_t row_ndx);
  415. Obj front();
  416. Obj back();
  417. Obj operator[](size_t row_ndx);
  418. /// \defgroup table_view_removes
  419. //@{
  420. /// \brief Remove the specified row (or rows) from the underlying table.
  421. ///
  422. /// remove() removes the specified row from the underlying table,
  423. /// remove_last() removes the last row in the table view from the underlying
  424. /// table, and clear removes all the rows in the table view from the
  425. /// underlying table.
  426. ///
  427. /// When rows are removed from the underlying table, they will by necessity
  428. /// also be removed from the table view. The order of the remaining rows in
  429. /// the the table view will be maintained.
  430. ///
  431. /// \param row_ndx The index within this table view of the row to be removed.
  432. void remove(size_t row_ndx);
  433. void remove_last();
  434. void clear();
  435. //@}
  436. std::unique_ptr<TableView> clone() const
  437. {
  438. return std::unique_ptr<TableView>(new TableView(*this));
  439. }
  440. std::unique_ptr<TableView> clone_for_handover(Transaction* tr, PayloadPolicy policy)
  441. {
  442. std::unique_ptr<TableView> retval(new TableView(*this, tr, policy));
  443. return retval;
  444. }
  445. private:
  446. TableView(TableRef parent);
  447. TableView(TableRef parent, Query& query, size_t start, size_t end, size_t limit);
  448. TableView(TableRef parent, LnkLstPtr);
  449. friend class ConstTableView;
  450. friend class Table;
  451. friend class Query;
  452. friend class LnkLst;
  453. };
  454. // ================================================================================================
  455. // ConstTableView Implementation:
  456. inline ConstTableView::ConstTableView(ConstTableRef parent)
  457. : m_table(parent) // Throws
  458. {
  459. m_key_values.create();
  460. if (m_table) {
  461. m_last_seen_versions.emplace_back(m_table->get_key(), m_table->get_content_version());
  462. }
  463. }
  464. inline ConstTableView::ConstTableView(ConstTableRef parent, Query& query, size_t start, size_t end, size_t lim)
  465. : m_table(parent)
  466. , m_query(query)
  467. , m_start(start)
  468. , m_end(end)
  469. , m_limit(lim)
  470. {
  471. m_key_values.create();
  472. }
  473. inline ConstTableView::ConstTableView(ConstTableRef src_table, ColKey src_column_key, const Obj& obj)
  474. : m_table(src_table) // Throws
  475. , m_source_column_key(src_column_key)
  476. , m_linked_obj_key(obj.get_key())
  477. , m_linked_table(obj.get_table())
  478. {
  479. m_key_values.create();
  480. if (m_table) {
  481. m_last_seen_versions.emplace_back(m_table->get_key(), m_table->get_content_version());
  482. m_last_seen_versions.emplace_back(obj.get_table()->get_key(), obj.get_table()->get_content_version());
  483. }
  484. }
  485. inline ConstTableView::ConstTableView(ConstTableRef parent, LnkLstPtr link_list)
  486. : m_table(parent) // Throws
  487. , m_linklist_source(std::move(link_list))
  488. {
  489. REALM_ASSERT(m_linklist_source);
  490. m_key_values.create();
  491. if (m_table) {
  492. m_last_seen_versions.emplace_back(m_table->get_key(), m_table->get_content_version());
  493. }
  494. }
  495. inline ConstTableView::ConstTableView(ConstTableRef parent, LnkSetPtr link_set)
  496. : m_table(parent) // Throws
  497. , m_linkset_source(std::move(link_set))
  498. {
  499. REALM_ASSERT(m_linkset_source);
  500. m_key_values.create();
  501. if (m_table) {
  502. m_last_seen_versions.emplace_back(m_table->get_key(), m_table->get_content_version());
  503. }
  504. }
  505. inline ConstTableView::ConstTableView(const ConstTableView& tv)
  506. : m_table(tv.m_table)
  507. , m_source_column_key(tv.m_source_column_key)
  508. , m_linked_obj_key(tv.m_linked_obj_key)
  509. , m_linked_table(tv.m_linked_table)
  510. , m_linklist_source(tv.m_linklist_source ? tv.m_linklist_source->clone_linklist() : LnkLstPtr{})
  511. , m_linkset_source(tv.m_linkset_source ? tv.m_linkset_source->clone_linkset() : LnkSetPtr{})
  512. , m_descriptor_ordering(tv.m_descriptor_ordering)
  513. , m_query(tv.m_query)
  514. , m_start(tv.m_start)
  515. , m_end(tv.m_end)
  516. , m_limit(tv.m_limit)
  517. , m_last_seen_versions(tv.m_last_seen_versions)
  518. {
  519. m_key_values.copy_from(tv.m_key_values);
  520. m_limit_count = tv.m_limit_count;
  521. }
  522. inline ConstTableView::ConstTableView(ConstTableView&& tv) noexcept
  523. : m_table(tv.m_table)
  524. , m_source_column_key(tv.m_source_column_key)
  525. , m_linked_obj_key(tv.m_linked_obj_key)
  526. , m_linked_table(tv.m_linked_table)
  527. , m_linklist_source(std::move(tv.m_linklist_source))
  528. , m_linkset_source(std::move(tv.m_linkset_source))
  529. , m_descriptor_ordering(std::move(tv.m_descriptor_ordering))
  530. , m_query(std::move(tv.m_query))
  531. , m_start(tv.m_start)
  532. , m_end(tv.m_end)
  533. , m_limit(tv.m_limit)
  534. // if we are created from a table view which is outdated, take care to use the outdated
  535. // version number so that we can later trigger a sync if needed.
  536. , m_last_seen_versions(std::move(tv.m_last_seen_versions))
  537. {
  538. m_key_values.move_from(tv.m_key_values);
  539. m_limit_count = tv.m_limit_count;
  540. }
  541. inline ConstTableView& ConstTableView::operator=(ConstTableView&& tv) noexcept
  542. {
  543. m_table = std::move(tv.m_table);
  544. m_key_values.move_from(tv.m_key_values);
  545. m_query = std::move(tv.m_query);
  546. m_last_seen_versions = tv.m_last_seen_versions;
  547. m_start = tv.m_start;
  548. m_end = tv.m_end;
  549. m_limit = tv.m_limit;
  550. m_limit_count = tv.m_limit_count;
  551. m_source_column_key = tv.m_source_column_key;
  552. m_linked_obj_key = tv.m_linked_obj_key;
  553. m_linked_table = tv.m_linked_table;
  554. m_linklist_source = std::move(tv.m_linklist_source);
  555. m_linkset_source = std::move(tv.m_linkset_source);
  556. m_descriptor_ordering = std::move(tv.m_descriptor_ordering);
  557. return *this;
  558. }
  559. inline ConstTableView& ConstTableView::operator=(const ConstTableView& tv)
  560. {
  561. if (this == &tv)
  562. return *this;
  563. m_key_values.copy_from(tv.m_key_values);
  564. m_query = tv.m_query;
  565. m_last_seen_versions = tv.m_last_seen_versions;
  566. m_start = tv.m_start;
  567. m_end = tv.m_end;
  568. m_limit = tv.m_limit;
  569. m_limit_count = tv.m_limit_count;
  570. m_source_column_key = tv.m_source_column_key;
  571. m_linked_obj_key = tv.m_linked_obj_key;
  572. m_linked_table = tv.m_linked_table;
  573. m_linklist_source = tv.m_linklist_source ? tv.m_linklist_source->clone_linklist() : LnkLstPtr{};
  574. m_linkset_source = tv.m_linkset_source ? tv.m_linkset_source->clone_linkset() : LnkSetPtr{};
  575. m_descriptor_ordering = tv.m_descriptor_ordering;
  576. return *this;
  577. }
  578. #define REALM_ASSERT_COLUMN(column_key) \
  579. m_table.check(); \
  580. REALM_ASSERT(m_table->colkey2ndx(column_key))
  581. #define REALM_ASSERT_ROW(row_ndx) \
  582. m_table.check(); \
  583. REALM_ASSERT(row_ndx < m_key_values.size())
  584. #define REALM_ASSERT_COLUMN_AND_TYPE(column_key, column_type) \
  585. REALM_ASSERT_COLUMN(column_key); \
  586. REALM_DIAG_PUSH(); \
  587. REALM_DIAG_IGNORE_TAUTOLOGICAL_COMPARE(); \
  588. REALM_ASSERT(m_table->get_column_type(column_key) == column_type); \
  589. REALM_DIAG_POP()
  590. #define REALM_ASSERT_INDEX(column_key, row_ndx) \
  591. REALM_ASSERT_COLUMN(column_key); \
  592. REALM_ASSERT(row_ndx < m_key_values.size())
  593. #define REALM_ASSERT_INDEX_AND_TYPE(column_key, row_ndx, column_type) \
  594. REALM_ASSERT_COLUMN_AND_TYPE(column_key, column_type); \
  595. REALM_ASSERT(row_ndx < m_key_values.size())
  596. #define REALM_ASSERT_INDEX_AND_TYPE_TABLE_OR_MIXED(column_key, row_ndx) \
  597. REALM_ASSERT_COLUMN(column_key); \
  598. REALM_DIAG_PUSH(); \
  599. REALM_DIAG_IGNORE_TAUTOLOGICAL_COMPARE(); \
  600. REALM_ASSERT(m_table->get_column_type(column_key) == type_Table || \
  601. (m_table->get_column_type(column_key) == type_Mixed)); \
  602. REALM_DIAG_POP(); \
  603. REALM_ASSERT(row_ndx < m_key_values.size())
  604. //-------------------------- TableView, ConstTableView implementation:
  605. template <class T>
  606. ConstTableView ObjList::find_all(ColKey column_key, T value) const
  607. {
  608. ConstTableView tv(get_target_table());
  609. auto& keys = tv.m_key_values;
  610. for_each([column_key, value, &keys](const Obj& o) {
  611. if (o.get<T>(column_key) == value) {
  612. keys.add(o.get_key());
  613. }
  614. return false;
  615. });
  616. return tv;
  617. }
  618. inline void TableView::remove_last()
  619. {
  620. if (!is_empty())
  621. remove(size() - 1);
  622. }
  623. inline TableView::TableView(TableRef parent)
  624. : ConstTableView(parent)
  625. {
  626. }
  627. inline TableView::TableView(TableRef parent, Query& query, size_t start, size_t end, size_t lim)
  628. : ConstTableView(parent, query, start, end, lim)
  629. {
  630. }
  631. inline TableView::TableView(TableRef parent, LnkLstPtr link_list)
  632. : ConstTableView(parent, std::move(link_list))
  633. {
  634. }
  635. // Rows
  636. inline Obj TableView::get(size_t row_ndx)
  637. {
  638. REALM_ASSERT_ROW(row_ndx);
  639. ObjKey key(m_key_values.get(row_ndx));
  640. REALM_ASSERT(key != realm::null_key);
  641. return get_parent()->get_object(key);
  642. }
  643. inline Obj TableView::front()
  644. {
  645. return get(0);
  646. }
  647. inline Obj TableView::back()
  648. {
  649. size_t last_row_ndx = size() - 1;
  650. return get(last_row_ndx);
  651. }
  652. inline Obj TableView::operator[](size_t row_ndx)
  653. {
  654. return get(row_ndx);
  655. }
  656. } // namespace realm
  657. #endif // REALM_TABLE_VIEW_HPP