RLMSet.h 20 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506
  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. NS_ASSUME_NONNULL_BEGIN
  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 - Notifications
  152. /**
  153. Registers a block to be called each time the set changes.
  154. The block will be asynchronously called with the initial set, and then
  155. called again after each write transaction which changes any of the objects in
  156. the set, which objects are in the results, or the order of the objects in the
  157. set.
  158. The `changes` parameter will be `nil` the first time the block is called.
  159. For each call after that, it will contain information about
  160. which rows in the set were added, removed or modified. If a write transaction
  161. did not modify any objects in the set, the block is not called at all.
  162. See the `RLMCollectionChange` documentation for information on how the changes
  163. are reported and an example of updating a `UITableView`.
  164. If an error occurs the block will be called with `nil` for the results
  165. parameter and a non-`nil` error. Currently the only errors that can occur are
  166. when opening the Realm on the background worker thread.
  167. Notifications are delivered via the standard run loop, and so can't be
  168. delivered while the run loop is blocked by other activity. When
  169. notifications can't be delivered instantly, multiple notifications may be
  170. coalesced into a single notification. This can include the notification
  171. with the initial results. For example, the following code performs a write
  172. transaction immediately after adding the notification block, so there is no
  173. opportunity for the initial notification to be delivered first. As a
  174. result, the initial notification will reflect the state of the Realm after
  175. the write transaction.
  176. Person *person = [[Person allObjectsInRealm:realm] firstObject];
  177. NSLog(@"person.dogs.count: %zu", person.dogs.count); // => 0
  178. self.token = [person.dogs addNotificationBlock(RLMSet<Dog *> *dogs,
  179. RLMCollectionChange *changes,
  180. NSError *error) {
  181. // Only fired once for the example
  182. NSLog(@"dogs.count: %zu", dogs.count) // => 1
  183. }];
  184. [realm transactionWithBlock:^{
  185. Dog *dog = [[Dog alloc] init];
  186. dog.name = @"Rex";
  187. [person.dogs addObject:dog];
  188. }];
  189. // end of run loop execution context
  190. You must retain the returned token for as long as you want updates to continue
  191. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  192. @warning This method cannot be called during a write transaction, or when the
  193. containing Realm is read-only.
  194. @warning This method may only be called on a non-frozen managed set.
  195. @param block The block to be called each time the set changes.
  196. @return A token which must be held for as long as you want updates to be delivered.
  197. */
  198. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  199. RLMCollectionChange *_Nullable changes,
  200. NSError *_Nullable error))block
  201. __attribute__((warn_unused_result));
  202. /**
  203. Registers a block to be called each time the set changes.
  204. The block will be asynchronously called with the initial set, and then
  205. called again after each write transaction which changes any of the objects in
  206. the set, which objects are in the results, or the order of the objects in the
  207. set.
  208. The `changes` parameter will be `nil` the first time the block is called.
  209. For each call after that, it will contain information about
  210. which rows in the set were added, removed or modified. If a write transaction
  211. did not modify any objects in the set, the block is not called at all.
  212. See the `RLMCollectionChange` documentation for information on how the changes
  213. are reported and an example of updating a `UITableView`.
  214. If an error occurs the block will be called with `nil` for the results
  215. parameter and a non-`nil` error. Currently the only errors that can occur are
  216. when opening the Realm on the background worker thread.
  217. Notifications are delivered on the given queue. If the queue is blocked and
  218. notifications can't be delivered instantly, multiple notifications may be
  219. coalesced into a single notification.
  220. You must retain the returned token for as long as you want updates to continue
  221. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  222. @warning This method cannot be called when the containing Realm is read-only or frozen.
  223. @warning The queue must be a serial queue.
  224. @param block The block to be called whenever a change occurs.
  225. @param queue The serial queue to deliver notifications to.
  226. @return A token which must be held for as long as you want updates to be delivered.
  227. */
  228. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  229. RLMCollectionChange *_Nullable changes,
  230. NSError *_Nullable error))block
  231. queue:(nullable dispatch_queue_t)queue
  232. __attribute__((warn_unused_result));
  233. /**
  234. Registers a block to be called each time the set changes.
  235. The block will be asynchronously called with the initial set, and then
  236. called again after each write transaction which changes any of the objects in
  237. the set, which objects are in the results, or the order of the objects in the
  238. set.
  239. The `changes` parameter will be `nil` the first time the block is called.
  240. For each call after that, it will contain information about
  241. which rows in the set were added, removed or modified. If a write transaction
  242. did not modify any objects in the set, the block is not called at all.
  243. See the `RLMCollectionChange` documentation for information on how the changes
  244. are reported and an example of updating a `UITableView`.
  245. If an error occurs the block will be called with `nil` for the results
  246. parameter and a non-`nil` error. Currently the only errors that can occur are
  247. when opening the Realm on the background worker thread.
  248. Notifications are delivered on the given queue. If the queue is blocked and
  249. notifications can't be delivered instantly, multiple notifications may be
  250. coalesced into a single notification.
  251. You must retain the returned token for as long as you want updates to continue
  252. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  253. @warning This method cannot be called when the containing Realm is read-only or frozen.
  254. @warning The queue must be a serial queue.
  255. @param block The block to be called whenever a change occurs.
  256. @param keyPaths The block will be called for changes occuring on these keypaths. If no
  257. key paths are given, notifications are delivered for every property key path.
  258. @param queue The serial queue to deliver notifications to.
  259. @return A token which must be held for as long as you want updates to be delivered.
  260. */
  261. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  262. RLMCollectionChange *_Nullable changes,
  263. NSError *_Nullable error))block
  264. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  265. queue:(nullable dispatch_queue_t)queue
  266. __attribute__((warn_unused_result));
  267. /**
  268. Registers a block to be called each time the set changes.
  269. The block will be asynchronously called with the initial set, and then
  270. called again after each write transaction which changes any of the objects in
  271. the set, which objects are in the results, or the order of the objects in the
  272. set.
  273. The `changes` parameter will be `nil` the first time the block is called.
  274. For each call after that, it will contain information about
  275. which rows in the set were added, removed or modified. If a write transaction
  276. did not modify any objects in the set, the block is not called at all.
  277. See the `RLMCollectionChange` documentation for information on how the changes
  278. are reported and an example of updating a `UITableView`.
  279. If an error occurs the block will be called with `nil` for the results
  280. parameter and a non-`nil` error. Currently the only errors that can occur are
  281. when opening the Realm on the background worker thread.
  282. Notifications are delivered via the standard run loop, and so can't be
  283. delivered while the run loop is blocked by other activity. When
  284. notifications can't be delivered instantly, multiple notifications may be
  285. coalesced into a single notification. This can include the notification
  286. with the initial results. For example, the following code performs a write
  287. transaction immediately after adding the notification block, so there is no
  288. opportunity for the initial notification to be delivered first. As a
  289. result, the initial notification will reflect the state of the Realm after
  290. the write transaction.
  291. You must retain the returned token for as long as you want updates to continue
  292. to be sent to the block. To stop receiving updates, call `-invalidate` on the token.
  293. @warning This method cannot be called when the containing Realm is read-only or frozen.
  294. @warning The queue must be a serial queue.
  295. @param block The block to be called whenever a change occurs.
  296. @param keyPaths The block will be called for changes occuring on these keypaths. If no
  297. key paths are given, notifications are delivered for every property key path.
  298. @return A token which must be held for as long as you want updates to be delivered.
  299. */
  300. - (RLMNotificationToken *)addNotificationBlock:(void (^)(RLMSet<RLMObjectType> *_Nullable set,
  301. RLMCollectionChange *_Nullable changes,
  302. NSError *_Nullable error))block
  303. keyPaths:(nullable NSArray<NSString *> *)keyPaths
  304. __attribute__((warn_unused_result));
  305. #pragma mark - Aggregating Property Values
  306. /**
  307. Returns the minimum (lowest) value of the given property among all the objects in the set.
  308. NSNumber *min = [object.setProperty minOfProperty:@"age"];
  309. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet`, and `NSData` properties.
  310. @param property The property whose minimum value is desired. Only properties of
  311. types `int`, `float`, `double`, and `NSDate` are supported.
  312. @return The minimum value of the property, or `nil` if the set is empty.
  313. */
  314. - (nullable id)minOfProperty:(NSString *)property;
  315. /**
  316. Returns the maximum (highest) value of the given property among all the objects in the set.
  317. NSNumber *max = [object.setProperty maxOfProperty:@"age"];
  318. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet`, and `NSData` properties.
  319. @param property The property whose maximum value is desired. Only properties of
  320. types `int`, `float`, `double`, and `NSDate` are supported.
  321. @return The maximum value of the property, or `nil` if the set is empty.
  322. */
  323. - (nullable id)maxOfProperty:(NSString *)property;
  324. /**
  325. Returns the sum of distinct values of a given property over all the objects in the set.
  326. NSNumber *sum = [object.setProperty sumOfProperty:@"age"];
  327. @warning You cannot use this method on `RLMObject`, `RLMArray`, `RLMSet and `NSData` properties.
  328. @param property The property whose values should be summed. Only properties of
  329. types `int`, `float`, and `double` are supported.
  330. @return The sum of the given property.
  331. */
  332. - (NSNumber *)sumOfProperty:(NSString *)property;
  333. /**
  334. Returns the average value of a given property over the objects in the set.
  335. NSNumber *average = [object.setProperty averageOfProperty:@"age"];
  336. @warning You cannot use this method on `RLMObject`, `RLMSet`, `RLMArray`, and `NSData` properties.
  337. @param property The property whose average value should be calculated. Only
  338. properties of types `int`, `float`, and `double` are supported.
  339. @return The average value of the given property, or `nil` if the set is empty.
  340. */
  341. - (nullable NSNumber *)averageOfProperty:(NSString *)property;
  342. #pragma mark - Freeze
  343. /**
  344. Returns a frozen (immutable) snapshot of this set.
  345. The frozen copy is an immutable set which contains the same data as this
  346. et currently contains, but will not update when writes are made to the
  347. containing Realm. Unlike live sets, frozen sets can be accessed from any
  348. thread.
  349. @warning This method cannot be called during a write transaction, or when the
  350. containing Realm is read-only.
  351. @warning This method may only be called on a managed set.
  352. @warning Holding onto a frozen set for an extended period while performing
  353. write transaction on the Realm may result in the Realm file growing
  354. to large sizes. See `RLMRealmConfiguration.maximumNumberOfActiveVersions`
  355. for more information.
  356. */
  357. - (instancetype)freeze;
  358. /**
  359. Returns a live version of this frozen collection.
  360. This method resolves a reference to a live copy of the same frozen collection.
  361. If called on a live collection, will return itself.
  362. */
  363. - (instancetype)thaw;
  364. #pragma mark - Unavailable Methods
  365. /**
  366. `-[RLMSet init]` is not available because `RLMSet`s cannot be created directly.
  367. ``RLMSet` properties on `RLMObject`s are lazily created when accessed.
  368. */
  369. - (instancetype)init __attribute__((unavailable("RLMSets cannot be created directly")));
  370. /**
  371. `+[RLMSet new]` is not available because `RLMSet`s cannot be created directly.
  372. `RLMSet` properties on `RLMObject`s are lazily created when accessed.
  373. */
  374. + (instancetype)new __attribute__((unavailable("RLMSet cannot be created directly")));
  375. @end
  376. /// :nodoc:
  377. @interface RLMSet (Swift)
  378. // for use only in Swift class definitions
  379. - (instancetype)initWithObjectClassName:(NSString *)objectClassName;
  380. @end
  381. NS_ASSUME_NONNULL_END