RLMSet.h 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2020 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. @class RLMObject, RLMResults<RLMObjectType>;
  21. /**
  22. A collection datatype used for storing distinct objects.
  23. - Note:
  24. `RLMSet` supports storing primitive and `RLMObject` types. `RLMSet` does not support storing
  25. Embedded Realm Objects.
  26. */
  27. @interface RLMSet<RLMObjectType> : NSObject<RLMCollection>
  28. #pragma mark - Properties
  29. /**
  30. The number of objects in the set.
  31. */
  32. @property (nonatomic, readonly, assign) NSUInteger count;
  33. /**
  34. The type of the objects in the set.
  35. */
  36. @property (nonatomic, readonly, assign) RLMPropertyType type;
  37. /**
  38. Indicates whether the objects in the collection can be `nil`.
  39. */
  40. @property (nonatomic, readonly, getter = isOptional) BOOL optional;
  41. /**
  42. The objects in the RLMSet as an NSArray value.
  43. */
  44. @property (nonatomic, readonly) NSArray<RLMObjectType> *allObjects;
  45. /**
  46. The class name of the objects contained in the set.
  47. Will be `nil` if `type` is not RLMPropertyTypeObject.
  48. */
  49. @property (nonatomic, readonly, copy, nullable) NSString *objectClassName;
  50. /**
  51. The Realm which manages the set. Returns `nil` for unmanaged set.
  52. */
  53. @property (nonatomic, readonly, nullable) RLMRealm *realm;
  54. /**
  55. Indicates if the set can no longer be accessed.
  56. */
  57. @property (nonatomic, readonly, getter = isInvalidated) BOOL invalidated;
  58. /**
  59. Indicates if the set is frozen.
  60. Frozen sets are immutable and can be accessed from any thread. Frozen sets
  61. are created by calling `-freeze` on a managed live set. Unmanaged sets are
  62. never frozen.
  63. */
  64. @property (nonatomic, readonly, getter = isFrozen) BOOL frozen;
  65. #pragma mark - Adding, Removing, and Replacing Objects in a Set
  66. /**
  67. Adds an object to the set if it is not already present.
  68. @warning This method may only be called during a write transaction.
  69. @param object An object of the type contained in the set.
  70. */
  71. - (void)addObject:(RLMObjectType)object;
  72. /**
  73. Adds an array of distinct objects to the set.
  74. @warning This method may only be called during a write transaction.
  75. @param objects An enumerable object such as `NSArray`, `NSSet` or `RLMResults` which contains objects of the
  76. same class as the set.
  77. */
  78. - (void)addObjects:(id<NSFastEnumeration>)objects;
  79. /**
  80. Removes a given object from the set.
  81. @warning This method may only be called during a write transaction.
  82. @param object The object in the set that you want to remove.
  83. */
  84. - (void)removeObject:(RLMObjectType)object;
  85. /**
  86. Removes all objects from the set.
  87. @warning This method may only be called during a write transaction.
  88. */
  89. - (void)removeAllObjects;
  90. /**
  91. Empties the receiving set, then adds each object contained in another given set.
  92. @warning This method may only be called during a write transaction.
  93. @param set The RLMSet whose members replace the receiving set's content.
  94. */
  95. - (void)setSet:(RLMSet<RLMObjectType> *)set;
  96. /**
  97. Removes from the receiving set each object that isn’t a member of another given set.
  98. @warning This method may only be called during a write transaction.
  99. @param set The RLMSet with which to perform the intersection.
  100. */
  101. - (void)intersectSet:(RLMSet<RLMObjectType> *)set;
  102. /**
  103. Removes each object in another given set from the receiving set, if present.
  104. @warning This method may only be called during a write transaction.
  105. @param set The set of objects to remove from the receiving set.
  106. */
  107. - (void)minusSet:(RLMSet<RLMObjectType> *)set;
  108. /**
  109. Adds each object in another given set to the receiving set, if not present.
  110. @warning This method may only be called during a write transaction.
  111. @param set The set of objects to add to the receiving set.
  112. */
  113. - (void)unionSet:(RLMSet<RLMObjectType> *)set;
  114. #pragma mark - Querying a Set
  115. /// :nodoc:
  116. - (RLMResults<RLMObjectType> *)objectsWhere:(NSString *)predicateFormat, ...;
  117. /// :nodoc:
  118. - (RLMResults<RLMObjectType> *)objectsWhere:(NSString *)predicateFormat args:(va_list)args;
  119. /// :nodoc:
  120. - (RLMResults<RLMObjectType> *)objectsWithPredicate:(NSPredicate *)predicate;
  121. /// :nodoc:
  122. - (RLMResults<RLMObjectType> *)sortedResultsUsingKeyPath:(NSString *)keyPath ascending:(BOOL)ascending;
  123. /// :nodoc:
  124. - (RLMResults<RLMObjectType> *)sortedResultsUsingDescriptors:(NSArray<RLMSortDescriptor *> *)properties;
  125. /// :nodoc:
  126. - (RLMResults<RLMObjectType> *)distinctResultsUsingKeyPaths:(NSArray<NSString *> *)keyPaths;
  127. /**
  128. Returns a Boolean value that indicates whether at least one object in the receiving set is also present in another given set.
  129. @param set The RLMSet to compare the receiving set to.
  130. @return YES if at least one object in the receiving set is also present in otherSet, otherwise NO.
  131. */
  132. - (BOOL)intersectsSet:(RLMSet<RLMObjectType> *)set;
  133. /**
  134. Returns a Boolean value that indicates whether every object in the receiving set is also present in another given set.
  135. @param set The RLMSet to compare the receiving set to.
  136. @return YES if every object in the receiving set is also present in otherSet, otherwise NO.
  137. */
  138. - (BOOL)isSubsetOfSet:(RLMSet<RLMObjectType> *)set;
  139. /**
  140. Returns a Boolean value that indicates whether a given object is present in the set.
  141. @param anObject An object to look for in the set.
  142. @return YES if anObject is present in the set, otherwise NO.
  143. */
  144. - (BOOL)containsObject:(RLMObjectType)anObject;
  145. /**
  146. Compares the receiving set to another set.
  147. @param otherSet The set with which to compare the receiving set.
  148. @return YES if the contents of otherSet are equal to the contents of the receiving set, otherwise NO.
  149. */
  150. - (BOOL)isEqualToSet:(RLMSet<RLMObjectType> *)otherSet;
  151. #pragma mark - Sectioning a Set
  152. /**
  153. Sorts and sections this collection from a given property key path, returning the result
  154. as an instance of `RLMSectionedResults`.
  155. @param keyPath The property key path to sort on.
  156. @param ascending The direction to sort in.
  157. @param keyBlock A callback which is invoked on each element in the Results collection.
  158. This callback is to return the section key for the element in the collection.
  159. @return An instance of RLMSectionedResults.
  160. */
  161. - (RLMSectionedResults *)sectionedResultsSortedUsingKeyPath:(NSString *)keyPath
  162. ascending:(BOOL)ascending
  163. keyBlock:(RLMSectionedResultsKeyBlock)keyBlock;
  164. /**
  165. Sorts and sections this collection from a given array of sort descriptors, returning the result
  166. as an instance of `RLMSectionedResults`.
  167. @param sortDescriptors An array of `RLMSortDescriptor`s to sort by.
  168. @param keyBlock A callback which is invoked on each element in the Results collection.
  169. This callback is to return the section key for the element in the collection.
  170. @note The primary sort descriptor must be responsible for determining the section key.
  171. @return An instance of RLMSectionedResults.
  172. */
  173. - (RLMSectionedResults *)sectionedResultsUsingSortDescriptors:(NSArray<RLMSortDescriptor *> *)sortDescriptors
  174. keyBlock:(RLMSectionedResultsKeyBlock)keyBlock;
  175. #pragma mark - Notifications
  176. /**
  177. Registers a block to be called each time the set changes.
  178. The block will be asynchronously called with the initial set, and then
  179. called again after each write transaction which changes any of the objects in
  180. the set, which objects are in the results, or the order of the objects in the
  181. set.
  182. The `changes` parameter will be `nil` the first time the block is called.
  183. For each call after that, it will contain information about
  184. which rows in the set were added, removed or modified. If a write transaction
  185. did not modify any objects in the set, the block is not called at all.
  186. See the `RLMCollectionChange` documentation for information on how the changes
  187. are reported and an example of updating a `UITableView`.
  188. The error parameter is present only for backwards compatibility and will always
  189. be `nil`.
  190. Notifications are delivered via the standard run loop, and so can't be
  191. delivered while the run loop is blocked by other activity. When
  192. notifications can't be delivered instantly, multiple notifications may be
  193. coalesced into a single notification. This can include the notification
  194. with the initial results. For example, the following code performs a write
  195. transaction immediately after adding the notification block, so there is no
  196. opportunity for the initial notification to be delivered first. As a
  197. result, the initial notification will reflect the state of the Realm after
  198. the write transaction.
  199. Person *person = [[Person allObjectsInRealm:realm] firstObject];
  200. NSLog(@"person.dogs.count: %zu", person.dogs.count); // => 0
  201. self.token = [person.dogs addNotificationBlock(RLMSet<Dog *> *dogs,
  202. RLMCollectionChange *changes,
  203. NSError *error) {
  204. // Only fired once for the example
  205. NSLog(@"dogs.count: %zu", dogs.count) // => 1
  206. }];
  207. [realm transactionWithBlock:^{
  208. Dog *dog = [[Dog alloc] init];
  209. dog.name = @"Rex";
  210. [person.dogs addObject:dog];
  211. }];
  212. // end of run loop execution context
  213. You must retain the returned token for as long as you want updates to continue
  214. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  215. @warning This method cannot be called during a write transaction, or when the
  216. containing Realm is read-only.
  217. @warning This method may only be called on a non-frozen managed set.
  218. @param block The block to be called each time the set changes.
  219. @return A token which must be held for as long as you want updates to be delivered.
  220. */
  221. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  222. RLMCollectionChange *_Nullable changes,
  223. NSError *_Nullable error))block
  224. __attribute__((warn_unused_result));
  225. /**
  226. Registers a block to be called each time the set changes.
  227. The block will be asynchronously called with the initial set, and then
  228. called again after each write transaction which changes any of the objects in
  229. the set, which objects are in the results, or the order of the objects in the
  230. set.
  231. The `changes` parameter will be `nil` the first time the block is called.
  232. For each call after that, it will contain information about
  233. which rows in the set were added, removed or modified. If a write transaction
  234. did not modify any objects in the set, the block is not called at all.
  235. See the `RLMCollectionChange` documentation for information on how the changes
  236. are reported and an example of updating a `UITableView`.
  237. The error parameter is present only for backwards compatibility and will always
  238. be `nil`.
  239. Notifications are delivered on the given queue. If the queue is blocked and
  240. notifications can't be delivered instantly, multiple notifications may be
  241. coalesced into a single notification.
  242. You must retain the returned token for as long as you want updates to continue
  243. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  244. @warning This method cannot be called when the containing Realm is read-only or frozen.
  245. @warning The queue must be a serial queue.
  246. @param block The block to be called whenever a change occurs.
  247. @param queue The serial queue to deliver notifications to.
  248. @return A token which must be held for as long as you want updates to be delivered.
  249. */
  250. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  251. RLMCollectionChange *_Nullable changes,
  252. NSError *_Nullable error))block
  253. queue:(nullable dispatch_queue_t)queue
  254. __attribute__((warn_unused_result));
  255. /**
  256. Registers a block to be called each time the set changes.
  257. The block will be asynchronously called with the initial set, and then
  258. called again after each write transaction which changes any of the objects in
  259. the set, which objects are in the results, or the order of the objects in the
  260. set.
  261. The `changes` parameter will be `nil` the first time the block is called.
  262. For each call after that, it will contain information about
  263. which rows in the set were added, removed or modified. If a write transaction
  264. did not modify any objects in the set, the block is not called at all.
  265. See the `RLMCollectionChange` documentation for information on how the changes
  266. are reported and an example of updating a `UITableView`.
  267. The error parameter is present only for backwards compatibility and will always
  268. be `nil`.
  269. Notifications are delivered on the given queue. If the queue is blocked and
  270. notifications can't be delivered instantly, multiple notifications may be
  271. coalesced into a single notification.
  272. You must retain the returned token for as long as you want updates to continue
  273. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  274. @warning This method cannot be called when the containing Realm is read-only or frozen.
  275. @warning The queue must be a serial queue.
  276. @param block The block to be called whenever a change occurs.
  277. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  278. key paths are given, notifications are delivered for every property key path.
  279. @param queue The serial queue to deliver notifications to.
  280. @return A token which must be held for as long as you want updates to be delivered.
  281. */
  282. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  283. RLMCollectionChange *_Nullable changes,
  284. NSError *_Nullable error))block
  285. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  286. queue:(nullable dispatch_queue_t)queue
  287. __attribute__((warn_unused_result));
  288. /**
  289. Registers a block to be called each time the set changes.
  290. The block will be asynchronously called with the initial set, and then
  291. called again after each write transaction which changes any of the objects in
  292. the set, which objects are in the results, or the order of the objects in the
  293. set.
  294. The `changes` parameter will be `nil` the first time the block is called.
  295. For each call after that, it will contain information about
  296. which rows in the set were added, removed or modified. If a write transaction
  297. did not modify any objects in the set, the block is not called at all.
  298. See the `RLMCollectionChange` documentation for information on how the changes
  299. are reported and an example of updating a `UITableView`.
  300. The error parameter is present only for backwards compatibility and will always
  301. be `nil`.
  302. Notifications are delivered via the standard run loop, and so can't be
  303. delivered while the run loop is blocked by other activity. When
  304. notifications can't be delivered instantly, multiple notifications may be
  305. coalesced into a single notification. This can include the notification
  306. with the initial results. For example, the following code performs a write
  307. transaction immediately after adding the notification block, so there is no
  308. opportunity for the initial notification to be delivered first. As a
  309. result, the initial notification will reflect the state of the Realm after
  310. the write transaction.
  311. You must retain the returned token for as long as you want updates to continue
  312. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  313. @warning This method cannot be called when the containing Realm is read-only or frozen.
  314. @warning The queue must be a serial queue.
  315. @param block The block to be called whenever a change occurs.
  316. @param keyPaths The block will be called for changes occurring on these keypaths. If no
  317. key paths are given, notifications are delivered for every property key path.
  318. @return A token which must be held for as long as you want updates to be delivered.
  319. */
  320. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  321. RLMCollectionChange *_Nullable changes,
  322. NSError *_Nullable error))block
  323. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  324. __attribute__((warn_unused_result));
  325. #pragma mark - Aggregating Property Values
  326. /**
  327. Returns the minimum (lowest) value of the given property among all the objects in the set.
  328. NSNumber *min = [object.setProperty minOfProperty:@"age"];
  329. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet`, and `NSData` properties.
  330. @param property The property whose minimum value is desired. Only properties of
  331. types `int`, `float`, `double`, and `NSDate` are supported.
  332. @return The minimum value of the property, or `nil` if the set is empty.
  333. */
  334. - (nullable id)minOfProperty:(NSString *)property;
  335. /**
  336. Returns the maximum (highest) value of the given property among all the objects in the set.
  337. NSNumber *max = [object.setProperty maxOfProperty:@"age"];
  338. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet`, and `NSData` properties.
  339. @param property The property whose maximum value is desired. Only properties of
  340. types `int`, `float`, `double`, and `NSDate` are supported.
  341. @return The maximum value of the property, or `nil` if the set is empty.
  342. */
  343. - (nullable id)maxOfProperty:(NSString *)property;
  344. /**
  345. Returns the sum of distinct values of a given property over all the objects in the set.
  346. NSNumber *sum = [object.setProperty sumOfProperty:@"age"];
  347. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet and `NSData` properties.
  348. @param property The property whose values should be summed. Only properties of
  349. types `int`, `float`, and `double` are supported.
  350. @return The sum of the given property.
  351. */
  352. - (NSNumber *)sumOfProperty:(NSString *)property;
  353. /**
  354. Returns the average value of a given property over the objects in the set.
  355. NSNumber *average = [object.setProperty averageOfProperty:@"age"];
  356. @warning You cannot use this method on `RLMObject`, `RLMSet`, `RLMArray`, and `NSData` properties.
  357. @param property The property whose average value should be calculated. Only
  358. properties of types `int`, `float`, and `double` are supported.
  359. @return The average value of the given property, or `nil` if the set is empty.
  360. */
  361. - (nullable NSNumber *)averageOfProperty:(NSString *)property;
  362. #pragma mark - Freeze
  363. /**
  364. Returns a frozen (immutable) snapshot of this set.
  365. The frozen copy is an immutable set which contains the same data as this
  366. et currently contains, but will not update when writes are made to the
  367. containing Realm. Unlike live sets, frozen sets can be accessed from any
  368. thread.
  369. @warning This method cannot be called during a write transaction, or when the
  370. containing Realm is read-only.
  371. @warning This method may only be called on a managed set.
  372. @warning Holding onto a frozen set for an extended period while performing
  373. write transaction on the Realm may result in the Realm file growing
  374. to large sizes. See `RLMRealmConfiguration.maximumNumberOfActiveVersions`
  375. for more information.
  376. */
  377. - (instancetype)freeze;
  378. /**
  379. Returns a live version of this frozen collection.
  380. This method resolves a reference to a live copy of the same frozen collection.
  381. If called on a live collection, will return itself.
  382. */
  383. - (instancetype)thaw;
  384. #pragma mark - Unavailable Methods
  385. /**
  386. `-[RLMSet init]` is not available because `RLMSet`s cannot be created directly.
  387. ``RLMSet` properties on `RLMObject`s are lazily created when accessed.
  388. */
  389. - (instancetype)init __attribute__((unavailable("RLMSets cannot be created directly")));
  390. /**
  391. `+[RLMSet new]` is not available because `RLMSet`s cannot be created directly.
  392. `RLMSet` properties on `RLMObject`s are lazily created when accessed.
  393. */
  394. + (instancetype)new __attribute__((unavailable("RLMSet cannot be created directly")));
  395. @end
  396. /// :nodoc:
  397. @interface RLMSet (Swift)
  398. // for use only in Swift class definitions
  399. - (instancetype)initWithObjectClassName:(NSString *)objectClassName;
  400. @end
  401. RLM_HEADER_AUDIT_END(nullability, sendability)