Aliases.swift 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2014 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 Foundation
  19. import Realm
  20. // These types don't change when wrapping in Swift
  21. // so we just typealias them to remove the 'RLM' prefix
  22. // MARK: Aliases
  23. /**
  24. `PropertyType` is an enum describing all property types supported in Realm models.
  25. For more information, see [Realm Models](https://realm.io/docs/swift/latest/#models).
  26. ### Primitive types
  27. * `Int`
  28. * `Bool`
  29. * `Float`
  30. * `Double`
  31. ### Object types
  32. * `String`
  33. * `Data`
  34. * `Date`
  35. * `Decimal128`
  36. * `ObjectId`
  37. ### Relationships: Array (in Swift, `List`) and `Object` types
  38. * `Object`
  39. * `Array`
  40. */
  41. public typealias PropertyType = RLMPropertyType
  42. /**
  43. An opaque token which is returned from methods which subscribe to changes to a Realm.
  44. - see: `Realm.observe(_:)`
  45. */
  46. public typealias NotificationToken = RLMNotificationToken
  47. /// :nodoc:
  48. public typealias ObjectBase = RLMObjectBase
  49. extension ObjectBase {
  50. /**
  51. Registers a block to be called each time the object changes.
  52. The block will be asynchronously called after each write transaction which
  53. deletes the object or modifies any of the managed properties of the object,
  54. including self-assignments that set a property to its existing value.
  55. For write transactions performed on different threads or in different
  56. processes, the block will be called when the managing Realm is
  57. (auto)refreshed to a version including the changes, while for local write
  58. transactions it will be called at some point in the future after the write
  59. transaction is committed.
  60. If no key paths are given, the block will be executed on any insertion,
  61. modification, or deletion for all object properties and the properties of
  62. any nested, linked objects. If a key path or key paths are provided,
  63. then the block will be called for changes which occur only on the
  64. provided key paths. For example, if:
  65. ```swift
  66. class Dog: Object {
  67. @Persisted var name: String
  68. @Persisted var adopted: Bool
  69. @Persisted var siblings: List<Dog>
  70. }
  71. // ... where `dog` is a managed Dog object.
  72. dog.observe(keyPaths: ["adopted"], { changes in
  73. // ...
  74. })
  75. ```
  76. - The above notification block fires for changes to the
  77. `adopted` property, but not for any changes made to `name`.
  78. - If the observed key path were `["siblings"]`, then any insertion,
  79. deletion, or modification to the `siblings` list will trigger the block. A change to
  80. `someSibling.name` would not trigger the block (where `someSibling`
  81. is an element contained in `siblings`)
  82. - If the observed key path were `["siblings.name"]`, then any insertion or
  83. deletion to the `siblings` list would trigger the block. For objects
  84. contained in the `siblings` list, only modifications to their `name` property
  85. will trigger the block.
  86. - note: Multiple notification tokens on the same object which filter for
  87. separate key paths *do not* filter exclusively. If one key path
  88. change is satisfied for one notification token, then all notification
  89. token blocks for that object will execute.
  90. If no queue is given, notifications are delivered via the standard run
  91. loop, and so can't be delivered while the run loop is blocked by other
  92. activity. If a queue is given, notifications are delivered to that queue
  93. instead. When notifications can't be delivered instantly, multiple
  94. notifications may be coalesced into a single notification.
  95. Unlike with `List` and `Results`, there is no "initial" callback made after
  96. you add a new notification block.
  97. Only objects which are managed by a Realm can be observed in this way. You
  98. must retain the returned token for as long as you want updates to be sent
  99. to the block. To stop receiving updates, call `invalidate()` on the token.
  100. It is safe to capture a strong reference to the observed object within the
  101. callback block. There is no retain cycle due to that the callback is
  102. retained by the returned token and not by the object itself.
  103. - warning: This method cannot be called during a write transaction, or when
  104. the containing Realm is read-only.
  105. - parameter keyPaths: Only properties contained in the key paths array will trigger
  106. the block when they are modified. If `nil`, notifications
  107. will be delivered for any property change on the object.
  108. String key paths which do not correspond to a valid a property
  109. will throw an exception.
  110. See description above for more detail on linked properties.
  111. - parameter queue: The serial dispatch queue to receive notification on. If
  112. `nil`, notifications are delivered to the current thread.
  113. - parameter block: The block to call with information about changes to the object.
  114. - returns: A token which must be held for as long as you want updates to be delivered.
  115. */
  116. // swiftlint:disable:next identifier_name
  117. internal func _observe<T: ObjectBase>(keyPaths: [String]? = nil,
  118. on queue: DispatchQueue? = nil,
  119. _ block: @escaping (ObjectChange<T>) -> Void) -> NotificationToken {
  120. return RLMObjectBaseAddNotificationBlock(self, keyPaths, queue) { object, names, oldValues, newValues, error in
  121. if let error = error {
  122. block(.error(error as NSError))
  123. return
  124. }
  125. guard let names = names, let newValues = newValues else {
  126. block(.deleted)
  127. return
  128. }
  129. block(.change(object as! T, (0..<newValues.count).map { i in
  130. PropertyChange(name: names[i], oldValue: oldValues?[i], newValue: newValues[i])
  131. }))
  132. }
  133. }
  134. }