RLMSectionedResults.h 46 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2022 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. #import <Realm/RLMCollection.h>
  19. RLM_HEADER_AUDIT_BEGIN(nullability, sendability)
  20. @protocol RLMValue;
  21. @class RLMResults<RLMObjectType>;
  22. /**
  23. A `RLMSectionedResultsChange` object encapsulates information about changes to sectioned
  24. results that are reported by Realm notifications.
  25. `RLMSectionedResultsChange` is passed to the notification blocks registered with
  26. `-addNotificationBlock` on `RLMSectionedResults`, and reports what sections and rows in the
  27. collection changed since the last time the notification block was called.
  28. A complete example of updating a `UITableView` named `tv`:
  29. [tv beginUpdates];
  30. [tv deleteRowsAtIndexPaths:changes.deletions withRowAnimation:UITableViewRowAnimationAutomatic];
  31. [tv insertRowsAtIndexPaths:changes.insertions withRowAnimation:UITableViewRowAnimationAutomatic];
  32. [tv reloadRowsAtIndexPaths:changes.modifications withRowAnimation:UITableViewRowAnimationAutomatic];
  33. [tv insertSections:changes.sectionsToInsert withRowAnimation:UITableViewRowAnimationAutomatic];
  34. [tv deleteSections:changes.sectionsToRemove withRowAnimation:UITableViewRowAnimationAutomatic];
  35. [tv endUpdates];
  36. All of the arrays in an `RLMSectionedResultsChange` are always sorted in ascending order.
  37. */
  38. @interface RLMSectionedResultsChange : NSObject
  39. /// The index paths of objects in the previous version of the collection which have
  40. /// been removed from this one.
  41. @property (nonatomic, readonly) NSArray<NSIndexPath *> *deletions;
  42. /// The index paths in the new version of the collection which were newly inserted.
  43. @property (nonatomic, readonly) NSArray<NSIndexPath *> *insertions;
  44. /// The index paths in the old version of the collection which were modified.
  45. @property (nonatomic, readonly) NSArray<NSIndexPath *> *modifications;
  46. /// The indices of the sections to be inserted.
  47. @property (nonatomic, readonly) NSIndexSet *sectionsToInsert;
  48. /// The indices of the sections to be removed.
  49. @property (nonatomic, readonly) NSIndexSet *sectionsToRemove;
  50. /// Returns the index paths of the deletion indices in the given section.
  51. - (NSArray<NSIndexPath *> *)deletionsInSection:(NSUInteger)section;
  52. /// Returns the index paths of the insertion indices in the given section.
  53. - (NSArray<NSIndexPath *> *)insertionsInSection:(NSUInteger)section;
  54. /// Returns the index paths of the modification indices in the given section.
  55. - (NSArray<NSIndexPath *> *)modificationsInSection:(NSUInteger)section;
  56. @end
  57. /// The `RLMSectionedResult` protocol defines properties and methods common to both `RLMSectionedResults and RLMSection`
  58. @protocol RLMSectionedResult <NSFastEnumeration, RLMThreadConfined>
  59. #pragma mark - Object Access
  60. /// The count of objects in the collection.
  61. @property (nonatomic, readonly) NSUInteger count;
  62. /// Returns the object for a given index in the collection.
  63. - (id)objectAtIndexedSubscript:(NSUInteger)index;
  64. /// Returns the object for a given index in the collection.
  65. - (id)objectAtIndex:(NSUInteger)index;
  66. #pragma mark - Freeze
  67. /**
  68. Returns a frozen (immutable) snapshot of this collection.
  69. The frozen copy is an immutable collection which contains the same data as this
  70. collection currently contains, but will not update when writes are made to the
  71. containing Realm. Unlike live arrays, frozen collections can be accessed from any
  72. thread.
  73. @warning This method cannot be called during a write transaction, or when the
  74. containing Realm is read-only.
  75. @warning Holding onto a frozen collection for an extended period while performing
  76. write transaction on the Realm may result in the Realm file growing
  77. to large sizes. See `RLMRealmConfiguration.maximumNumberOfActiveVersions`
  78. for more information.
  79. */
  80. - (instancetype)freeze;
  81. /**
  82. Returns a live version of this frozen collection.
  83. This method resolves a reference to a live copy of the same frozen collection.
  84. If called on a live collection, will return itself.
  85. */
  86. - (instancetype)thaw;
  87. /**
  88. Indicates if the underlying collection is frozen.
  89. Frozen collections are immutable and can be accessed from any thread.
  90. */
  91. @property (nonatomic, readonly, getter = isFrozen) BOOL frozen;
  92. #pragma mark - Sectioned Results Notifications
  93. /**
  94. Registers a block to be called each time the collection changes.
  95. The block will be asynchronously called with the initial sectioned results collection,
  96. and then called again after each write transaction which changes either any
  97. of the objects in the results, or which objects are in the results.
  98. The `change` parameter will be `nil` the first time the block is called.
  99. For each call after that, it will contain information about
  100. which rows in the section were added, removed or modified. If a
  101. write transaction did not modify any objects in the section,
  102. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  103. information on how the changes are reported and an example of updating a
  104. `UITableView`.
  105. At the time when the block is called, the `RLMSection` / `RLMSectionedResults` object will be fully
  106. evaluated and up-to-date.
  107. Notifications are delivered via the standard run loop, and so can't be
  108. delivered while the run loop is blocked by other activity. When
  109. notifications can't be delivered instantly, multiple notifications may be
  110. coalesced into a single notification. This can include the notification
  111. with the initial results. For example, the following code performs a write
  112. transaction immediately after adding the notification block, so there is no
  113. opportunity for the initial notification to be delivered first. As a
  114. result, the initial notification will reflect the state of the Realm after
  115. the write transaction.
  116. RLMResults<Dog *> *results = [Dog allObjects];
  117. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  118. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  119. // Only fired once for the example
  120. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  121. }];
  122. [realm transactionWithBlock:^{
  123. Dog *dog = [[Dog alloc] init];
  124. dog.name = @"Rex";
  125. dog.age = 5;
  126. [realm addObject:dog];
  127. }];
  128. // end of run loop execution context
  129. You must retain the returned token for as long as you want updates to continue
  130. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  131. @warning This method cannot be called during a write transaction, or when the
  132. containing Realm is read-only.
  133. @warning The queue must be a serial queue.
  134. @param block The block to be called whenever a change occurs.
  135. @return A token which must be held for as long as you want updates to be delivered.
  136. */
  137. - (RLMNotificationToken *)addNotificationBlock:(void (^)(id<RLMSectionedResult>, RLMSectionedResultsChange *))block __attribute__((warn_unused_result));
  138. /**
  139. Registers a block to be called each time the collection changes.
  140. The block will be asynchronously called with the initial sectioned results collection,
  141. and then called again after each write transaction which changes either any
  142. of the objects in the results, or which objects are in the results.
  143. The `change` parameter will be `nil` the first time the block is called.
  144. For each call after that, it will contain information about
  145. which rows in the section were added, removed or modified. If a
  146. write transaction did not modify any objects in the section,
  147. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  148. information on how the changes are reported and an example of updating a
  149. `UITableView`.
  150. At the time when the block is called, the `RLMSection` / `RLMSectionedResults` object will be fully
  151. evaluated and up-to-date.
  152. Notifications are delivered via the standard run loop, and so can't be
  153. delivered while the run loop is blocked by other activity. When
  154. notifications can't be delivered instantly, multiple notifications may be
  155. coalesced into a single notification. This can include the notification
  156. with the initial results. For example, the following code performs a write
  157. transaction immediately after adding the notification block, so there is no
  158. opportunity for the initial notification to be delivered first. As a
  159. result, the initial notification will reflect the state of the Realm after
  160. the write transaction.
  161. RLMResults<Dog *> *results = [Dog allObjects];
  162. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  163. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  164. // Only fired once for the example
  165. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  166. }];
  167. [realm transactionWithBlock:^{
  168. Dog *dog = [[Dog alloc] init];
  169. dog.name = @"Rex";
  170. dog.age = 5;
  171. [realm addObject:dog];
  172. }];
  173. // end of run loop execution context
  174. You must retain the returned token for as long as you want updates to continue
  175. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  176. @warning This method cannot be called during a write transaction, or when the
  177. containing Realm is read-only.
  178. @warning The queue must be a serial queue.
  179. @param block The block to be called whenever a change occurs.
  180. @param queue The serial queue to deliver notifications to.
  181. @return A token which must be held for as long as you want updates to be delivered.
  182. */
  183. - (RLMNotificationToken *)addNotificationBlock:(void (^)(id<RLMSectionedResult>, RLMSectionedResultsChange *))block
  184. queue:(dispatch_queue_t)queue __attribute__((warn_unused_result));
  185. /**
  186. Registers a block to be called each time the collection changes.
  187. The block will be asynchronously called with the initial sectioned results collection,
  188. and then called again after each write transaction which changes either any
  189. of the objects in the results, or which objects are in the results.
  190. The `change` parameter will be `nil` the first time the block is called.
  191. For each call after that, it will contain information about
  192. which rows in the section were added, removed or modified. If a
  193. write transaction did not modify any objects in the section,
  194. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  195. information on how the changes are reported and an example of updating a
  196. `UITableView`.
  197. At the time when the block is called, the `RLMSection` / `RLMSectionedResults` object will be fully
  198. evaluated and up-to-date.
  199. Notifications are delivered via the standard run loop, and so can't be
  200. delivered while the run loop is blocked by other activity. When
  201. notifications can't be delivered instantly, multiple notifications may be
  202. coalesced into a single notification. This can include the notification
  203. with the initial results. For example, the following code performs a write
  204. transaction immediately after adding the notification block, so there is no
  205. opportunity for the initial notification to be delivered first. As a
  206. result, the initial notification will reflect the state of the Realm after
  207. the write transaction.
  208. RLMResults<Dog *> *results = [Dog allObjects];
  209. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  210. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  211. // Only fired once for the example
  212. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  213. }];
  214. [realm transactionWithBlock:^{
  215. Dog *dog = [[Dog alloc] init];
  216. dog.name = @"Rex";
  217. dog.age = 5;
  218. [realm addObject:dog];
  219. }];
  220. // end of run loop execution context
  221. You must retain the returned token for as long as you want updates to continue
  222. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  223. @warning This method cannot be called during a write transaction, or when the
  224. containing Realm is read-only.
  225. @warning The queue must be a serial queue.
  226. @param block The block to be called whenever a change occurs.
  227. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  228. key paths are given, notifications are delivered for every property key path.
  229. @return A token which must be held for as long as you want updates to be delivered.
  230. */
  231. - (RLMNotificationToken *)addNotificationBlock:(void (^)(id<RLMSectionedResult>, RLMSectionedResultsChange *))block
  232. keyPaths:(NSArray<NSString *> *)keyPaths __attribute__((warn_unused_result));
  233. /**
  234. Registers a block to be called each time the collection changes.
  235. The block will be asynchronously called with the initial sectioned results collection,
  236. and then called again after each write transaction which changes either any
  237. of the objects in the results, or which objects are in the results.
  238. The `change` parameter will be `nil` the first time the block is called.
  239. For each call after that, it will contain information about
  240. which rows in the section were added, removed or modified. If a
  241. write transaction did not modify any objects in the section,
  242. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  243. information on how the changes are reported and an example of updating a
  244. `UITableView`.
  245. At the time when the block is called, the `RLMSection` / `RLMSectionedResults` object will be fully
  246. evaluated and up-to-date.
  247. Notifications are delivered via the standard run loop, and so can't be
  248. delivered while the run loop is blocked by other activity. When
  249. notifications can't be delivered instantly, multiple notifications may be
  250. coalesced into a single notification. This can include the notification
  251. with the initial results. For example, the following code performs a write
  252. transaction immediately after adding the notification block, so there is no
  253. opportunity for the initial notification to be delivered first. As a
  254. result, the initial notification will reflect the state of the Realm after
  255. the write transaction.
  256. RLMResults<Dog *> *results = [Dog allObjects];
  257. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  258. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  259. // Only fired once for the example
  260. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  261. }];
  262. [realm transactionWithBlock:^{
  263. Dog *dog = [[Dog alloc] init];
  264. dog.name = @"Rex";
  265. dog.age = 5;
  266. [realm addObject:dog];
  267. }];
  268. // end of run loop execution context
  269. You must retain the returned token for as long as you want updates to continue
  270. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  271. @warning This method cannot be called during a write transaction, or when the
  272. containing Realm is read-only.
  273. @warning The queue must be a serial queue.
  274. @note When filtering with key paths a notification will be fired in the following scenarios:
  275. - An object in the collection has been modified at the filtered properties.
  276. - An object has been modified on the section key path property, and the result of that modification has changed it's position in the section, or the object may need to move to another section.
  277. - An object of the same observed type has been inserted or deleted from the Realm.
  278. @param block The block to be called whenever a change occurs.
  279. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  280. key paths are given, notifications are delivered for every property key path.
  281. @param queue The serial queue to deliver notifications to.
  282. @return A token which must be held for as long as you want updates to be delivered.
  283. */
  284. - (RLMNotificationToken *)addNotificationBlock:(void (^)(id<RLMSectionedResult>, RLMSectionedResultsChange *))block
  285. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  286. queue:(nullable dispatch_queue_t)queue __attribute__((warn_unused_result));
  287. @end
  288. /// An RLMSection contains the objects which belong to a specified section key.
  289. @interface RLMSection<RLMKeyType: id<RLMValue>, RLMObjectType> : NSObject<RLMSectionedResult>
  290. /// The value that represents the key in this section.
  291. @property (nonatomic, readonly) RLMKeyType key;
  292. /// The count of objects in the section.
  293. @property (nonatomic, readonly) NSUInteger count;
  294. /// Returns the object for a given index in the section.
  295. - (RLMObjectType)objectAtIndexedSubscript:(NSUInteger)index;
  296. /// Returns the object for a given index in the section.
  297. - (RLMObjectType)objectAtIndex:(NSUInteger)index;
  298. #pragma mark - Freeze
  299. /**
  300. Returns a frozen (immutable) snapshot of this section.
  301. The frozen copy is an immutable section which contains the same data as this
  302. section currently contains, but will not update when writes are made to the
  303. containing Realm. Unlike live arrays, frozen collections can be accessed from any
  304. thread.
  305. @warning This method cannot be called during a write transaction, or when the
  306. containing Realm is read-only.
  307. @warning Holding onto a frozen section for an extended period while performing
  308. write transaction on the Realm may result in the Realm file growing
  309. to large sizes. See `RLMRealmConfiguration.maximumNumberOfActiveVersions`
  310. for more information.
  311. */
  312. - (instancetype)freeze;
  313. /**
  314. Returns a live version of this frozen section.
  315. This method resolves a reference to a live copy of the same frozen section.
  316. If called on a live section, will return itself.
  317. */
  318. - (instancetype)thaw;
  319. /**
  320. Indicates if the underlying section is frozen.
  321. Frozen sections are immutable and can be accessed from any thread.
  322. */
  323. @property (nonatomic, readonly, getter = isFrozen) BOOL frozen;
  324. #pragma mark - Section Notifications
  325. /**
  326. Registers a block to be called each time the section changes.
  327. The block will be asynchronously called with the initial section,
  328. and then called again after each write transaction which changes either any
  329. of the objects in the results, or which objects are in the results.
  330. The `change` parameter will be `nil` the first time the block is called.
  331. For each call after that, it will contain information about
  332. which rows in the section were added, removed or modified. If a
  333. write transaction did not modify any objects in the section,
  334. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  335. information on how the changes are reported and an example of updating a
  336. `UITableView`.
  337. At the time when the block is called, the `RLMSection` object will be fully
  338. evaluated and up-to-date.
  339. Notifications are delivered via the standard run loop, and so can't be
  340. delivered while the run loop is blocked by other activity. When
  341. notifications can't be delivered instantly, multiple notifications may be
  342. coalesced into a single notification. This can include the notification
  343. with the initial results. For example, the following code performs a write
  344. transaction immediately after adding the notification block, so there is no
  345. opportunity for the initial notification to be delivered first. As a
  346. result, the initial notification will reflect the state of the Realm after
  347. the write transaction.
  348. RLMResults<Dog *> *results = [Dog allObjects];
  349. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  350. RLMSection<Dog *> *section = sectionedResults[0] // section with dogs aged '5' already exists.
  351. self.token = [section addNotificationBlock:^(RLMSection *section, RLMSectionedResultsChange *changes) {
  352. // Only fired once for the example
  353. NSLog(@"section.count: %zu", section.count); // => 2
  354. }];
  355. [realm transactionWithBlock:^{
  356. Dog *dog = [[Dog alloc] init];
  357. dog.name = @"Rex";
  358. dog.age = 5;
  359. [realm addObject:dog];
  360. }];
  361. // end of run loop execution context
  362. You must retain the returned token for as long as you want updates to continue
  363. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  364. @warning This method cannot be called during a write transaction, or when the
  365. containing Realm is read-only.
  366. @warning The queue must be a serial queue.
  367. @param block The block to be called whenever a change occurs.
  368. @return A token which must be held for as long as you want updates to be delivered.
  369. */
  370. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSection<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block __attribute__((warn_unused_result));
  371. /**
  372. Registers a block to be called each time the section changes.
  373. The block will be asynchronously called with the initial section,
  374. and then called again after each write transaction which changes either any
  375. of the objects in the results, or which objects are in the results.
  376. The `change` parameter will be `nil` the first time the block is called.
  377. For each call after that, it will contain information about
  378. which rows in the section were added, removed or modified. If a
  379. write transaction did not modify any objects in the section,
  380. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  381. information on how the changes are reported and an example of updating a
  382. `UITableView`.
  383. At the time when the block is called, the `RLMSection` object will be fully
  384. evaluated and up-to-date.
  385. Notifications are delivered via the standard run loop, and so can't be
  386. delivered while the run loop is blocked by other activity. When
  387. notifications can't be delivered instantly, multiple notifications may be
  388. coalesced into a single notification. This can include the notification
  389. with the initial results. For example, the following code performs a write
  390. transaction immediately after adding the notification block, so there is no
  391. opportunity for the initial notification to be delivered first. As a
  392. result, the initial notification will reflect the state of the Realm after
  393. the write transaction.
  394. RLMResults<Dog *> *results = [Dog allObjects];
  395. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  396. RLMSection<Dog *> *section = sectionedResults[0] // section with dogs aged '5' already exists.
  397. self.token = [section addNotificationBlock:^(RLMSection *section, RLMSectionedResultsChange *changes) {
  398. // Only fired once for the example
  399. NSLog(@"section.count: %zu", section.count); // => 2
  400. }];
  401. [realm transactionWithBlock:^{
  402. Dog *dog = [[Dog alloc] init];
  403. dog.name = @"Rex";
  404. dog.age = 5;
  405. [realm addObject:dog];
  406. }];
  407. // end of run loop execution context
  408. You must retain the returned token for as long as you want updates to continue
  409. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  410. @warning This method cannot be called during a write transaction, or when the
  411. containing Realm is read-only.
  412. @warning The queue must be a serial queue.
  413. @param block The block to be called whenever a change occurs.
  414. @param queue The serial queue to deliver notifications to.
  415. @return A token which must be held for as long as you want updates to be delivered.
  416. */
  417. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSection<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  418. queue:(dispatch_queue_t)queue __attribute__((warn_unused_result));
  419. /**
  420. Registers a block to be called each time the section changes.
  421. The block will be asynchronously called with the initial section,
  422. and then called again after each write transaction which changes either any
  423. of the objects in the results, or which objects are in the results.
  424. The `change` parameter will be `nil` the first time the block is called.
  425. For each call after that, it will contain information about
  426. which rows in the section were added, removed or modified. If a
  427. write transaction did not modify any objects in the section,
  428. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  429. information on how the changes are reported and an example of updating a
  430. `UITableView`.
  431. At the time when the block is called, the `RLMSection` object will be fully
  432. evaluated and up-to-date.
  433. Notifications are delivered via the standard run loop, and so can't be
  434. delivered while the run loop is blocked by other activity. When
  435. notifications can't be delivered instantly, multiple notifications may be
  436. coalesced into a single notification. This can include the notification
  437. with the initial results. For example, the following code performs a write
  438. transaction immediately after adding the notification block, so there is no
  439. opportunity for the initial notification to be delivered first. As a
  440. result, the initial notification will reflect the state of the Realm after
  441. the write transaction.
  442. RLMResults<Dog *> *results = [Dog allObjects];
  443. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  444. RLMSection<Dog *> *section = sectionedResults[0] // section with dogs aged '5' already exists.
  445. self.token = [section addNotificationBlock:^(RLMSection *section, RLMSectionedResultsChange *changes) {
  446. // Only fired once for the example
  447. NSLog(@"section.count: %zu", section.count); // => 2
  448. }];
  449. [realm transactionWithBlock:^{
  450. Dog *dog = [[Dog alloc] init];
  451. dog.name = @"Rex";
  452. dog.age = 5;
  453. [realm addObject:dog];
  454. }];
  455. // end of run loop execution context
  456. You must retain the returned token for as long as you want updates to continue
  457. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  458. @warning This method cannot be called during a write transaction, or when the
  459. containing Realm is read-only.
  460. @warning The queue must be a serial queue.
  461. @note When filtering with key paths a notification will be fired in the following scenarios:
  462. - An object in the collection has been modified at the filtered properties.
  463. - An object has been modified on the section key path property, and the result of that modification has changed it's position in the section, or the object may need to move to another section.
  464. - An object of the same observed type has been inserted or deleted from the Realm.
  465. @param block The block to be called whenever a change occurs.
  466. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  467. key paths are given, notifications are delivered for every property key path.
  468. @return A token which must be held for as long as you want updates to be delivered.
  469. */
  470. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSection<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  471. keyPaths:(NSArray<NSString *> *)keyPaths __attribute__((warn_unused_result));
  472. /**
  473. Registers a block to be called each time the section changes.
  474. The block will be asynchronously called with the initial section,
  475. and then called again after each write transaction which changes either any
  476. of the objects in the results, or which objects are in the results.
  477. The `change` parameter will be `nil` the first time the block is called.
  478. For each call after that, it will contain information about
  479. which rows in the section were added, removed or modified. If a
  480. write transaction did not modify any objects in the section,
  481. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  482. information on how the changes are reported and an example of updating a
  483. `UITableView`.
  484. At the time when the block is called, the `RLMSection` object will be fully
  485. evaluated and up-to-date.
  486. Notifications are delivered via the standard run loop, and so can't be
  487. delivered while the run loop is blocked by other activity. When
  488. notifications can't be delivered instantly, multiple notifications may be
  489. coalesced into a single notification. This can include the notification
  490. with the initial results. For example, the following code performs a write
  491. transaction immediately after adding the notification block, so there is no
  492. opportunity for the initial notification to be delivered first. As a
  493. result, the initial notification will reflect the state of the Realm after
  494. the write transaction.
  495. RLMResults<Dog *> *results = [Dog allObjects];
  496. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  497. RLMSection<Dog *> *section = sectionedResults[0] // section with dogs aged '5' already exists.
  498. self.token = [section addNotificationBlock:^(RLMSection *section, RLMSectionedResultsChange *changes) {
  499. // Only fired once for the example
  500. NSLog(@"section.count: %zu", section.count); // => 2
  501. }];
  502. [realm transactionWithBlock:^{
  503. Dog *dog = [[Dog alloc] init];
  504. dog.name = @"Rex";
  505. dog.age = 5;
  506. [realm addObject:dog];
  507. }];
  508. // end of run loop execution context
  509. You must retain the returned token for as long as you want updates to continue
  510. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  511. @warning This method cannot be called during a write transaction, or when the
  512. containing Realm is read-only.
  513. @warning The queue must be a serial queue.
  514. @note When filtering with key paths a notification will be fired in the following scenarios:
  515. - An object in the collection has been modified at the filtered properties.
  516. - An object has been modified on the section key path property, and the result of that modification has changed it's position in the section, or the object may need to move to another section.
  517. - An object of the same observed type has been inserted or deleted from the Realm.
  518. @param block The block to be called whenever a change occurs.
  519. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  520. key paths are given, notifications are delivered for every property key path.
  521. @param queue The serial queue to deliver notifications to.
  522. @return A token which must be held for as long as you want updates to be delivered.
  523. */
  524. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSection<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  525. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  526. queue:(nullable dispatch_queue_t)queue __attribute__((warn_unused_result));
  527. @end
  528. /// A lazily evaluated collection that holds elements in sections determined by a section key.
  529. @interface RLMSectionedResults<RLMKeyType: id<RLMValue>, RLMObjectType: id<RLMValue>> : NSObject<RLMSectionedResult>
  530. /// An array of all keys in the sectioned results collection.
  531. @property (nonatomic) NSArray<RLMKeyType> *allKeys;
  532. /// The total amount of sections in this collection.
  533. @property (nonatomic, readonly, assign) NSUInteger count;
  534. /// Returns the section at a given index.
  535. - (RLMSection<RLMKeyType, RLMObjectType> *)objectAtIndexedSubscript:(NSUInteger)index;
  536. /// Returns the section at a given index.
  537. - (RLMSection<RLMKeyType, RLMObjectType> *)objectAtIndex:(NSUInteger)index;
  538. #pragma mark - Freeze
  539. /**
  540. Returns a frozen (immutable) snapshot of this sectioned results collection.
  541. The frozen copy is an immutable sectioned results collection which contains the same data as this
  542. sectioned results collection currently contains, but will not update when writes are made to the
  543. containing Realm. Unlike live sectioned results collections, frozen sectioned results collection
  544. can be accessed from any thread.
  545. @warning This method cannot be called during a write transaction, or when the
  546. containing Realm is read-only.
  547. @warning Holding onto a frozen sectioned results collection for an extended period while performing
  548. write transaction on the Realm may result in the Realm file growing
  549. to large sizes. See `RLMRealmConfiguration.maximumNumberOfActiveVersions`
  550. for more information.
  551. */
  552. - (instancetype)freeze;
  553. /**
  554. Returns a live version of this frozen sectioned results collection.
  555. This method resolves a reference to a live copy of the same frozen sectioned results collection.
  556. If called on a live section, will return itself.
  557. */
  558. - (instancetype)thaw;
  559. /**
  560. Indicates if the underlying sectioned results collection is frozen.
  561. Frozen sectioned results collections are immutable and can be accessed from any thread.
  562. */
  563. @property (nonatomic, readonly, getter = isFrozen) BOOL frozen;
  564. #pragma mark - Sectioned Results Notifications
  565. /**
  566. Registers a block to be called each time the sectioned results collection changes.
  567. The block will be asynchronously called with the initial sectioned results collection,
  568. and then called again after each write transaction which changes either any
  569. of the objects in the results, or which objects are in the results.
  570. The `change` parameter will be `nil` the first time the block is called.
  571. For each call after that, it will contain information about
  572. which rows in the section were added, removed or modified. If a
  573. write transaction did not modify any objects in the section,
  574. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  575. information on how the changes are reported and an example of updating a
  576. `UITableView`.
  577. At the time when the block is called, the `RLMSectionedResults` object will be fully
  578. evaluated and up-to-date.
  579. Notifications are delivered via the standard run loop, and so can't be
  580. delivered while the run loop is blocked by other activity. When
  581. notifications can't be delivered instantly, multiple notifications may be
  582. coalesced into a single notification. This can include the notification
  583. with the initial results. For example, the following code performs a write
  584. transaction immediately after adding the notification block, so there is no
  585. opportunity for the initial notification to be delivered first. As a
  586. result, the initial notification will reflect the state of the Realm after
  587. the write transaction.
  588. RLMResults<Dog *> *results = [Dog allObjects];
  589. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  590. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  591. // Only fired once for the example
  592. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  593. }];
  594. [realm transactionWithBlock:^{
  595. Dog *dog = [[Dog alloc] init];
  596. dog.name = @"Rex";
  597. dog.age = 5;
  598. [realm addObject:dog];
  599. }];
  600. // end of run loop execution context
  601. You must retain the returned token for as long as you want updates to continue
  602. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  603. @warning This method cannot be called during a write transaction, or when the
  604. containing Realm is read-only.
  605. @warning The queue must be a serial queue.
  606. @param block The block to be called whenever a change occurs.
  607. @return A token which must be held for as long as you want updates to be delivered.
  608. */
  609. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSectionedResults<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block __attribute__((warn_unused_result));
  610. /**
  611. Registers a block to be called each time the sectioned results collection changes.
  612. The block will be asynchronously called with the initial sectioned results collection,
  613. and then called again after each write transaction which changes either any
  614. of the objects in the results, or which objects are in the results.
  615. The `change` parameter will be `nil` the first time the block is called.
  616. For each call after that, it will contain information about
  617. which rows in the section were added, removed or modified. If a
  618. write transaction did not modify any objects in the section,
  619. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  620. information on how the changes are reported and an example of updating a
  621. `UITableView`.
  622. At the time when the block is called, the `RLMSectionedResults` object will be fully
  623. evaluated and up-to-date.
  624. Notifications are delivered via the standard run loop, and so can't be
  625. delivered while the run loop is blocked by other activity. When
  626. notifications can't be delivered instantly, multiple notifications may be
  627. coalesced into a single notification. This can include the notification
  628. with the initial results. For example, the following code performs a write
  629. transaction immediately after adding the notification block, so there is no
  630. opportunity for the initial notification to be delivered first. As a
  631. result, the initial notification will reflect the state of the Realm after
  632. the write transaction.
  633. RLMResults<Dog *> *results = [Dog allObjects];
  634. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  635. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  636. // Only fired once for the example
  637. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  638. }];
  639. [realm transactionWithBlock:^{
  640. Dog *dog = [[Dog alloc] init];
  641. dog.name = @"Rex";
  642. dog.age = 5;
  643. [realm addObject:dog];
  644. }];
  645. // end of run loop execution context
  646. You must retain the returned token for as long as you want updates to continue
  647. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  648. @warning This method cannot be called during a write transaction, or when the
  649. containing Realm is read-only.
  650. @warning The queue must be a serial queue.
  651. @param block The block to be called whenever a change occurs.
  652. @param queue The serial queue to deliver notifications to.
  653. @return A token which must be held for as long as you want updates to be delivered.
  654. */
  655. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSectionedResults<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  656. queue:(dispatch_queue_t)queue __attribute__((warn_unused_result));
  657. /**
  658. Registers a block to be called each time the sectioned results collection changes.
  659. The block will be asynchronously called with the initial sectioned results collection,
  660. and then called again after each write transaction which changes either any
  661. of the objects in the results, or which objects are in the results.
  662. The `change` parameter will be `nil` the first time the block is called.
  663. For each call after that, it will contain information about
  664. which rows in the section were added, removed or modified. If a
  665. write transaction did not modify any objects in the section,
  666. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  667. information on how the changes are reported and an example of updating a
  668. `UITableView`.
  669. At the time when the block is called, the `RLMSectionedResults` object will be fully
  670. evaluated and up-to-date.
  671. Notifications are delivered via the standard run loop, and so can't be
  672. delivered while the run loop is blocked by other activity. When
  673. notifications can't be delivered instantly, multiple notifications may be
  674. coalesced into a single notification. This can include the notification
  675. with the initial results. For example, the following code performs a write
  676. transaction immediately after adding the notification block, so there is no
  677. opportunity for the initial notification to be delivered first. As a
  678. result, the initial notification will reflect the state of the Realm after
  679. the write transaction.
  680. RLMResults<Dog *> *results = [Dog allObjects];
  681. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  682. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  683. // Only fired once for the example
  684. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  685. }];
  686. [realm transactionWithBlock:^{
  687. Dog *dog = [[Dog alloc] init];
  688. dog.name = @"Rex";
  689. dog.age = 5;
  690. [realm addObject:dog];
  691. }];
  692. // end of run loop execution context
  693. You must retain the returned token for as long as you want updates to continue
  694. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  695. @warning This method cannot be called during a write transaction, or when the
  696. containing Realm is read-only.
  697. @warning The queue must be a serial queue.
  698. @note When filtering with key paths a notification will be fired in the following scenarios:
  699. - An object in the collection has been modified at the filtered properties.
  700. - An object has been modified on the section key path property, and the result of that modification has changed it's position in the section, or the object may need to move to another section.
  701. - An object of the same observed type has been inserted or deleted from the Realm.
  702. @param block The block to be called whenever a change occurs.
  703. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  704. key paths are given, notifications are delivered for every property key path.
  705. @return A token which must be held for as long as you want updates to be delivered.
  706. */
  707. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSectionedResults<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  708. keyPaths:(NSArray<NSString *> *)keyPaths __attribute__((warn_unused_result));
  709. /**
  710. Registers a block to be called each time the sectioned results collection changes.
  711. The block will be asynchronously called with the initial sectioned results collection,
  712. and then called again after each write transaction which changes either any
  713. of the objects in the results, or which objects are in the results.
  714. The `change` parameter will be `nil` the first time the block is called.
  715. For each call after that, it will contain information about
  716. which rows in the section were added, removed or modified. If a
  717. write transaction did not modify any objects in the section,
  718. the block is not called at all. See the `RLMSectionedResultsChange` documentation for
  719. information on how the changes are reported and an example of updating a
  720. `UITableView`.
  721. At the time when the block is called, the `RLMSectionedResults` object will be fully
  722. evaluated and up-to-date.
  723. Notifications are delivered via the standard run loop, and so can't be
  724. delivered while the run loop is blocked by other activity. When
  725. notifications can't be delivered instantly, multiple notifications may be
  726. coalesced into a single notification. This can include the notification
  727. with the initial results. For example, the following code performs a write
  728. transaction immediately after adding the notification block, so there is no
  729. opportunity for the initial notification to be delivered first. As a
  730. result, the initial notification will reflect the state of the Realm after
  731. the write transaction.
  732. RLMResults<Dog *> *results = [Dog allObjects];
  733. RLMSectionedResults<Dog *> *sectionedResults = [results sectionedResultsUsingKeyPath:@"age" ascending:YES];
  734. self.token = [sectionedResults addNotificationBlock:^(RLMSectionedResults *sectionedResults, RLMSectionedResultsChange *changes) {
  735. // Only fired once for the example
  736. NSLog(@"sectionedResults.count: %zu", sectionedResults.count); // => 1
  737. }];
  738. [realm transactionWithBlock:^{
  739. Dog *dog = [[Dog alloc] init];
  740. dog.name = @"Rex";
  741. dog.age = 5;
  742. [realm addObject:dog];
  743. }];
  744. // end of run loop execution context
  745. You must retain the returned token for as long as you want updates to continue
  746. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  747. @warning This method cannot be called during a write transaction, or when the
  748. containing Realm is read-only.
  749. @warning The queue must be a serial queue.
  750. @note When filtering with key paths a notification will be fired in the following scenarios:
  751. - An object in the collection has been modified at the filtered properties.
  752. - An object has been modified on the section key path property, and the result of that modification has changed it's position in the section, or the object may need to move to another section.
  753. - An object of the same observed type has been inserted or deleted from the Realm.
  754. @param block The block to be called whenever a change occurs.
  755. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  756. key paths are given, notifications are delivered for every property key path.
  757. @param queue The serial queue to deliver notifications to.
  758. @return A token which must be held for as long as you want updates to be delivered.
  759. */
  760. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSectionedResults<RLMKeyType, RLMObjectType> *, RLMSectionedResultsChange *))block
  761. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  762. queue:(nullable dispatch_queue_t)queue __attribute__((warn_unused_result));
  763. @end
  764. RLM_HEADER_AUDIT_END(nullability, sendability)