123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594 |
- ////////////////////////////////////////////////////////////////////////////
- //
- // Copyright 2021 Realm Inc.
- //
- // Licensed under the Apache License, Version 2.0 (the "License");
- // you may not use this file except in compliance with the License.
- // You may obtain a copy of the License at
- //
- // http://www.apache.org/licenses/LICENSE-2.0
- //
- // Unless required by applicable law or agreed to in writing, software
- // distributed under the License is distributed on an "AS IS" BASIS,
- // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- // See the License for the specific language governing permissions and
- // limitations under the License.
- //
- ////////////////////////////////////////////////////////////////////////////
- import Foundation
- import Realm
- /**
- A homogenous key-value collection of `Object`s which can be retrieved, filtered, sorted, and operated upon.
- */
- public protocol RealmKeyedCollection: Sequence, ThreadConfined, CustomStringConvertible {
- /// The type of key associated with this collection
- associatedtype Key: _MapKey
- /// The type of value associated with this collection.
- associatedtype Value: RealmCollectionValue
- // MARK: Properties
- /// The Realm which manages the map, or `nil` if the map is unmanaged.
- var realm: Realm? { get }
- /// Indicates if the map can no longer be accessed.
- var isInvalidated: Bool { get }
- /// Returns the number of key-value pairs in this map.
- var count: Int { get }
- /// A human-readable description of the objects contained in the Map.
- var description: String { get }
- // MARK: Mutation
- /**
- Updates the value stored in the dictionary for the given key, or adds a new key-value pair if the key does not exist.
- - Note:If the value being added to the dictionary is an unmanaged object and the dictionary is managed
- then that unmanaged object will be added to the Realm.
- - warning: This method may only be called during a write transaction.
- - parameter value: a value's key path predicate.
- - parameter forKey: The direction to sort in.
- */
- func updateValue(_ value: Value, forKey key: Key)
- /**
- Removes the given key and its associated object, only if the key exists in the dictionary. If the key does not
- exist, the dictionary will not be modified.
- - warning: This method may only be called during a write transaction.
- */
- func removeObject(for key: Key)
- /**
- Removes all objects from the dictionary. The objects are not removed from the Realm that manages them.
- - warning: This method may only be called during a write transaction.
- */
- func removeAll()
- /**
- Returns the value for a given key, or sets a value for a key should the subscript be used for an assign.
- - Note:If the value being added to the dictionary is an unmanaged object and the dictionary is managed
- then that unmanaged object will be added to the Realm.
- - Note:If the value being assigned for a key is `nil` then that key will be removed from the dictionary.
- - warning: This method may only be called during a write transaction.
- - parameter key: The key.
- */
- subscript(key: Key) -> Value? { get set }
- // MARK: KVC
- /**
- Returns a type of `Value` for a specified key if it exists in the map.
- Note that when using key-value coding, the key must be a string.
- - parameter key: The key to the property whose values are desired.
- */
- func value(forKey key: String) -> AnyObject?
- /**
- Returns a type of `Value` for a specified key if it exists in the map.
- - parameter keyPath: The key to the property whose values are desired.
- */
- func value(forKeyPath keyPath: String) -> AnyObject?
- /**
- Adds a given key-value pair to the dictionary or updates a given key should it already exist.
- - warning: This method can only be called during a write transaction.
- - parameter value: The object value.
- - parameter key: The name of the property whose value should be set on each object.
- */
- func setValue(_ value: Any?, forKey key: String)
- // MARK: Filtering
- /**
- Returns a `Results` containing all matching values in the dictionary with the given predicate.
- - Note: This will return the values in the dictionary, and not the key-value pairs.
- - parameter predicate: The predicate with which to filter the values.
- */
- func filter(_ predicate: NSPredicate) -> Results<Value>
- /**
- Returns a Boolean value indicating whether the Map contains the key-value pair
- satisfies the given predicate
- - parameter where: a closure that test if any key-pair of the given map represents the match.
- */
- func contains(where predicate: @escaping (_ key: Key, _ value: Value) -> Bool) -> Bool
- // MARK: Sorting
- /**
- Returns a `Results` containing the objects in the dictionary, but sorted.
- Objects are sorted based on their values. For example, to sort a dictionary of `Date`s from
- neweset to oldest based, you might call `dates.sorted(ascending: true)`.
- - parameter ascending: The direction to sort in.
- */
- func sorted(ascending: Bool) -> Results<Value>
- /**
- Returns a `Results` containing the objects in the dictionary, but sorted.
- Objects are sorted based on the values of the given key path. For example, to sort a dictionary of `Student`s from
- youngest to oldest based on their `age` property, you might call
- `students.sorted(byKeyPath: "age", ascending: true)`.
- - warning: Dictionaries may only be sorted by properties of boolean, `Date`, `NSDate`, single and double-precision
- floating point, integer, and string types.
- - parameter keyPath: The key path to sort by.
- - parameter ascending: The direction to sort in.
- */
- func sorted(byKeyPath keyPath: String, ascending: Bool) -> Results<Value>
- /**
- Returns a `Results` containing the objects in the dictionary, but sorted.
- - warning: Dictionaries may only be sorted by properties of boolean, `Date`, `NSDate`, single and double-precision
- floating point, integer, and string types.
- - see: `sorted(byKeyPath:ascending:)`
- */
- func sorted<S: Sequence>(by sortDescriptors: S) -> Results<Value>
- where S.Iterator.Element == SortDescriptor
- /// Returns all of the keys in this dictionary.
- var keys: [Key] { get }
- /// Returns all of the values in the dictionary.
- var values: [Value] { get }
- // MARK: Aggregate Operations
- /**
- Returns the minimum (lowest) value of the given property among all the objects in the collection, or `nil` if the
- dictionary is empty.
- - warning: Only a property whose type conforms to the `MinMaxType` protocol can be specified.
- - parameter property: The name of a property whose minimum value is desired.
- */
- func min<T: MinMaxType>(ofProperty property: String) -> T?
- /**
- Returns the maximum (highest) value of the given property among all the objects in the dictionary, or `nil` if the
- dictionary is empty.
- - warning: Only a property whose type conforms to the `MinMaxType` protocol can be specified.
- - parameter property: The name of a property whose minimum value is desired.
- */
- func max<T: MinMaxType>(ofProperty property: String) -> T?
- /**
- Returns the sum of the given property for objects in the dictionary, or `nil` if the dictionary is empty.
- - warning: Only names of properties of a type conforming to the `AddableType` protocol can be used.
- - parameter property: The name of a property conforming to `AddableType` to calculate sum on.
- */
- func sum<T: AddableType>(ofProperty property: String) -> T
- /**
- Returns the average value of a given property over all the objects in the dictionary, or `nil` if
- the dictionary is empty.
- - warning: Only a property whose type conforms to the `AddableType` protocol can be specified.
- - parameter property: The name of a property whose values should be summed.
- */
- func average<T: AddableType>(ofProperty property: String) -> T?
- // MARK: Notifications
- /**
- Registers a block to be called each time the dictionary changes.
- The block will be asynchronously called with the initial dictionary, and then called again after each write
- transaction which changes either any of the keys or values in the dictionary.
- The `change` parameter that is passed to the block reports, in the form of keys within the dictionary, which of
- the key-value pairs were added, removed, or modified during each write transaction.
- At the time when the block is called, the dictionary will be fully evaluated and up-to-date, and as long as you do
- not perform a write transaction on the same thread or explicitly call `realm.refresh()`, accessing it will never
- perform blocking work.
- If no queue is given, notifications are delivered via the standard run loop, and so can't be delivered while the
- run loop is blocked by other activity. If a queue is given, notifications are delivered to that queue instead. When
- notifications can't be delivered instantly, multiple notifications may be coalesced into a single notification.
- This can include the notification with the initial collection.
- For example, the following code performs a write transaction immediately after adding the notification block, so
- there is no opportunity for the initial notification to be delivered first. As a result, the initial notification
- will reflect the state of the Realm after the write transaction.
- ```swift
- let myStringMap = myObject.stringMap
- print("myStringMap.count: \(myStringMap?.count)") // => 0
- let token = myStringMap.observe { changes in
- switch changes {
- case .initial(let myStringMap):
- // Will print "myStringMap.count: 1"
- print("myStringMap.count: \(myStringMap.count)")
- print("Dog Name: \(myStringMap["nameOfDog"])") // => "Rex"
- break
- case .update:
- // Will not be hit in this example
- break
- case .error:
- break
- }
- }
- try! realm.write {
- myStringMap["nameOfDog"] = "Rex"
- }
- // end of run loop execution context
- ```
- You must retain the returned token for as long as you want updates to be sent to the block. To stop receiving
- updates, call `invalidate()` on the token.
- - warning: This method cannot be called during a write transaction, or when the containing Realm is read-only.
- - parameter queue: The serial dispatch queue to receive notification on. If
- `nil`, notifications are delivered to the current thread.
- - parameter block: The block to be called whenever a change occurs.
- - returns: A token which must be held for as long as you want updates to be delivered.
- */
- func observe(on queue: DispatchQueue?,
- _ block: @escaping (RealmMapChange<Self>) -> Void)
- -> NotificationToken
- /**
- Registers a block to be called each time the dictionary changes.
- The block will be asynchronously called with the initial dictionary, and then called again after each write
- transaction which changes either any of the keys or values in the dictionary.
- The `change` parameter that is passed to the block reports, in the form of keys within the dictionary, which of
- the key-value pairs were added, removed, or modified during each write transaction.
- At the time when the block is called, the dictionary will be fully evaluated and up-to-date, and as long as you do
- not perform a write transaction on the same thread or explicitly call `realm.refresh()`, accessing it will never
- perform blocking work.
- If no queue is given, notifications are delivered via the standard run loop, and so can't be delivered while the
- run loop is blocked by other activity. If a queue is given, notifications are delivered to that queue instead. When
- notifications can't be delivered instantly, multiple notifications may be coalesced into a single notification.
- This can include the notification with the initial collection.
- For example, the following code performs a write transaction immediately after adding the notification block, so
- there is no opportunity for the initial notification to be delivered first. As a result, the initial notification
- will reflect the state of the Realm after the write transaction.
- ```swift
- let myStringMap = myObject.stringMap
- print("myStringMap.count: \(myStringMap?.count)") // => 0
- let token = myStringMap.observe { changes in
- switch changes {
- case .initial(let myStringMap):
- // Will print "myStringMap.count: 1"
- print("myStringMap.count: \(myStringMap.count)")
- print("Dog Name: \(myStringMap["nameOfDog"])") // => "Rex"
- break
- case .update:
- // Will not be hit in this example
- break
- case .error:
- break
- }
- }
- try! realm.write {
- myStringMap["nameOfDog"] = "Rex"
- }
- // end of run loop execution context
- ```
- If no key paths are given, the block will be executed on any insertion,
- modification, or deletion for all object properties and the properties of
- any nested, linked objects. If a key path or key paths are provided,
- then the block will be called for changes which occur only on the
- provided key paths. For example, if:
- ```swift
- class Dog: Object {
- @Persisted var name: String
- @Persisted var age: Int
- @Persisted var toys: List<Toy>
- }
- // ...
- let dogs = myObject.mapOfDogs
- let token = dogs.observe(keyPaths: ["name"]) { changes in
- switch changes {
- case .initial(let dogs):
- // ...
- case .update:
- // This case is hit:
- // - after the token is intialized
- // - when the name property of an object in the
- // collection is modified
- // - when an element is inserted or removed
- // from the collection.
- // This block is not triggered:
- // - when a value other than name is modified on
- // one of the elements.
- case .error:
- // ...
- }
- }
- // end of run loop execution context
- ```
- - If the observed key path were `["toys.brand"]`, then any insertion or
- deletion to the `toys` list on any of the collection's elements would trigger the block.
- Changes to the `brand` value on any `Toy` that is linked to a `Dog` in this
- collection will trigger the block. Changes to a value other than `brand` on any `Toy` that
- is linked to a `Dog` in this collection would not trigger the block.
- Any insertion or removal to the `Dog` type collection being observed
- would also trigger a notification.
- - If the above example observed the `["toys"]` key path, then any insertion,
- deletion, or modification to the `toys` list for any element in the collection
- would trigger the block.
- Changes to any value on any `Toy` that is linked to a `Dog` in this collection
- would *not* trigger the block.
- Any insertion or removal to the `Dog` type collection being observed
- would still trigger a notification.
- - note: Multiple notification tokens on the same object which filter for
- separate key paths *do not* filter exclusively. If one key path
- change is satisfied for one notification token, then all notification
- token blocks for that object will execute.
- You must retain the returned token for as long as you want updates to be sent to the block. To stop receiving
- updates, call `invalidate()` on the token.
- - warning: This method cannot be called during a write transaction, or when the containing Realm is read-only.
- - parameter keyPaths: Only properties contained in the key paths array will trigger
- the block when they are modified. If `nil`, notifications
- will be delivered for any property change on the object.
- String key paths which do not correspond to a valid a property
- will throw an exception.
- See description above for more detail on linked properties.
- - note: The keyPaths parameter refers to object properties of the collection type and
- *does not* refer to particular key/value pairs within the collection.
- - parameter queue: The serial dispatch queue to receive notification on. If
- `nil`, notifications are delivered to the current thread.
- - parameter block: The block to be called whenever a change occurs.
- - returns: A token which must be held for as long as you want updates to be delivered.
- */
- func observe(keyPaths: [String]?,
- on queue: DispatchQueue?,
- _ block: @escaping (RealmMapChange<Self>) -> Void) -> NotificationToken
- // MARK: Frozen Objects
- /// Returns if this collection is frozen
- var isFrozen: Bool { get }
- /**
- Returns a frozen (immutable) snapshot of this collection.
- The frozen copy is an immutable collection which contains the same data as this collection
- currently contains, but will not update when writes are made to the containing Realm. Unlike
- live collections, frozen collections can be accessed from any thread.
- - warning: This method cannot be called during a write transaction, or when the containing
- Realm is read-only.
- - warning: Holding onto a frozen collection for an extended period while performing write
- transaction on the Realm may result in the Realm file growing to large sizes. See
- `Realm.Configuration.maximumNumberOfActiveVersions` for more information.
- */
- func freeze() -> Self
- /**
- Returns a live (mutable) version of this frozen collection.
- This method resolves a reference to a live copy of the same frozen collection.
- If called on a live collection, will return itself.
- */
- func thaw() -> Self?
- }
- /**
- Protocol for RealmKeyedCollections where the Value is of an Object type that
- enables aggregatable operations.
- */
- public extension RealmKeyedCollection where Value: OptionalProtocol, Value.Wrapped: ObjectBase, Value.Wrapped: RealmCollectionValue {
- /**
- Returns the minimum (lowest) value of the given property among all the objects in the collection, or `nil` if the
- collection is empty.
- - warning: Only a property whose type conforms to the `MinMaxType` protocol can be specified.
- - parameter keyPath: The keyPath of a property whose minimum value is desired.
- */
- func min<T: MinMaxType>(of keyPath: KeyPath<Value.Wrapped, T>) -> T? {
- min(ofProperty: _name(for: keyPath))
- }
- /**
- Returns the maximum (highest) value of the given property among all the objects in the collection, or `nil` if the
- collection is empty.
- - warning: Only a property whose type conforms to the `MinMaxType` protocol can be specified.
- - parameter keyPath: The keyPath of a property whose minimum value is desired.
- */
- func max<T: MinMaxType>(of keyPath: KeyPath<Value.Wrapped, T>) -> T? {
- max(ofProperty: _name(for: keyPath))
- }
- /**
- Returns the sum of the given property for objects in the collection, or `nil` if the collection is empty.
- - warning: Only names of properties of a type conforming to the `AddableType` protocol can be used.
- - parameter keyPath: The keyPath of a property conforming to `AddableType` to calculate sum on.
- */
- func sum<T: AddableType>(of keyPath: KeyPath<Value.Wrapped, T>) -> T {
- sum(ofProperty: _name(for: keyPath))
- }
- /**
- Returns the average value of a given property over all the objects in the collection, or `nil` if
- the collection is empty.
- - warning: Only a property whose type conforms to the `AddableType` protocol can be specified.
- - parameter keyPath: The keyPath of a property whose values should be summed.
- */
- func average<T: AddableType>(of keyPath: KeyPath<Value.Wrapped, T>) -> T? {
- average(ofProperty: _name(for: keyPath))
- }
- }
- // MARK: Sortable
- /**
- Protocol for RealmKeyedCollections where the Value is of an Object type that
- enables sortable operations.
- */
- public extension RealmKeyedCollection where Value: OptionalProtocol, Value.Wrapped: ObjectBase, Value.Wrapped: RealmCollectionValue {
- /**
- Returns a `Results` containing the objects in the collection, but sorted.
- Objects are sorted based on the values of the given key path. For example, to sort a collection of `Student`s from
- youngest to oldest based on their `age` property, you might call
- `students.sorted(byKeyPath: "age", ascending: true)`.
- - warning: Collections may only be sorted by properties of boolean, `Date`, `NSDate`, single and double-precision
- floating point, integer, and string types.
- - parameter keyPath: The key path to sort by.
- - parameter ascending: The direction to sort in.
- */
- func sorted<T: Comparable>(by keyPath: KeyPath<Value.Wrapped, T>, ascending: Bool) -> Results<Value> {
- sorted(byKeyPath: _name(for: keyPath), ascending: ascending)
- }
- }
- public extension RealmKeyedCollection where Value: MinMaxType {
- /**
- Returns the minimum (lowest) value of the collection, or `nil` if the collection is empty.
- */
- func min() -> Value? {
- return min(ofProperty: "self")
- }
- /**
- Returns the maximum (highest) value of the collection, or `nil` if the collection is empty.
- */
- func max() -> Value? {
- return max(ofProperty: "self")
- }
- }
- public extension RealmKeyedCollection where Value: OptionalProtocol, Value.Wrapped: MinMaxType {
- /**
- Returns the minimum (lowest) value of the collection, or `nil` if the collection is empty.
- */
- func min() -> Value.Wrapped? {
- return min(ofProperty: "self")
- }
- /**
- Returns the maximum (highest) value of the collection, or `nil` if the collection is empty.
- */
- func max() -> Value.Wrapped? {
- return max(ofProperty: "self")
- }
- }
- public extension RealmKeyedCollection where Value: AddableType {
- /**
- Returns the sum of the values in the collection, or `nil` if the collection is empty.
- */
- func sum() -> Value {
- return sum(ofProperty: "self")
- }
- /**
- Returns the average of all of the values in the collection.
- */
- func average<T: AddableType>() -> T? {
- return average(ofProperty: "self")
- }
- }
- public extension RealmKeyedCollection where Value: OptionalProtocol, Value.Wrapped: AddableType {
- /**
- Returns the sum of the values in the collection, or `nil` if the collection is empty.
- */
- func sum() -> Value.Wrapped {
- return sum(ofProperty: "self")
- }
- /**
- Returns the average of all of the values in the collection.
- */
- func average<T: AddableType>() -> T? {
- return average(ofProperty: "self")
- }
- }
- public extension RealmKeyedCollection where Value: Comparable {
- /**
- Returns a `Results` containing the objects in the collection, but sorted.
- Objects are sorted based on their values. For example, to sort a collection of `Date`s from
- neweset to oldest based, you might call `dates.sorted(ascending: true)`.
- - parameter ascending: The direction to sort in.
- */
- func sorted(ascending: Bool = true) -> Results<Value> {
- return sorted(byKeyPath: "self", ascending: ascending)
- }
- }
- public extension RealmKeyedCollection where Value: OptionalProtocol, Value.Wrapped: Comparable {
- /**
- Returns a `Results` containing the objects in the collection, but sorted.
- Objects are sorted based on their values. For example, to sort a collection of `Date`s from
- neweset to oldest based, you might call `dates.sorted(ascending: true)`.
- - parameter ascending: The direction to sort in.
- */
- func sorted(ascending: Bool = true) -> Results<Value> {
- return sorted(byKeyPath: "self", ascending: ascending)
- }
- }
|