RLMObservation.hpp 7.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2015 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/Foundation.h>
  19. #import <realm/obj.hpp>
  20. #import <realm/object-store/binding_context.hpp>
  21. #import <realm/object-store/impl/deep_change_checker.hpp>
  22. #import <realm/table.hpp>
  23. @class RLMObjectBase, RLMRealm, RLMSchema, RLMProperty, RLMObjectSchema;
  24. class RLMClassInfo;
  25. class RLMSchemaInfo;
  26. namespace realm {
  27. class History;
  28. class SharedGroup;
  29. struct TableKey;
  30. struct ColKey;
  31. }
  32. // RLMObservationInfo stores all of the KVO-related data for RLMObjectBase and
  33. // RLMSet/Array. There is a one-to-one relationship between observed objects and
  34. // RLMObservationInfo instances, so it could be folded into RLMObjectBase, and
  35. // is a separate class mostly to avoid making all accessor objects far larger.
  36. //
  37. // RLMClassInfo stores a vector of pointers to the first observation info
  38. // created for each row. If there are multiple observation infos for a single
  39. // row (such as if there are multiple observed objects backed by a single row,
  40. // or if both an object and an array property of that object are observed),
  41. // they're stored in an intrusive doubly-linked-list in the `next` and `prev`
  42. // members. This is done primarily to make it simpler and faster to loop over
  43. // all of the observed objects for a single row, as that needs to be done for
  44. // every change.
  45. class RLMObservationInfo {
  46. public:
  47. RLMObservationInfo(id object);
  48. RLMObservationInfo(RLMClassInfo &objectSchema, realm::ObjKey row, id object);
  49. ~RLMObservationInfo();
  50. realm::Obj const& getRow() const {
  51. return row;
  52. }
  53. NSString *columnName(realm::ColKey col) const noexcept;
  54. // Send willChange/didChange notifications to all observers for this object/row
  55. // Sends the array versions if indexes is non-nil, normal versions otherwise
  56. void willChange(NSString *key, NSKeyValueChange kind=NSKeyValueChangeSetting, NSIndexSet *indexes=nil) const;
  57. void didChange(NSString *key, NSKeyValueChange kind=NSKeyValueChangeSetting, NSIndexSet *indexes=nil) const;
  58. bool isForRow(realm::ObjKey key) const {
  59. return row.get_key() == key;
  60. }
  61. void recordObserver(realm::Obj& row, RLMClassInfo *objectInfo, RLMObjectSchema *objectSchema, NSString *keyPath);
  62. void removeObserver();
  63. bool hasObservers() const { return observerCount > 0; }
  64. // valueForKey: on observed object and array properties needs to return the
  65. // same object each time for KVO to work at all. Doing this all the time
  66. // requires some odd semantics to avoid reference cycles, so instead we do
  67. // it only to the extent specifically required by KVO. In addition, we
  68. // need to continue to return the same object even if this row is deleted,
  69. // or deleting an object with active observers will explode horribly.
  70. // Once prepareForInvalidation() is called, valueForKey() will always return
  71. // the cached value for object and array properties without checking the
  72. // backing row to verify it's up-to-date.
  73. //
  74. // prepareForInvalidation() must be called on the head of the linked list
  75. // (i.e. on the object pointed to directly by the object schema)
  76. id valueForKey(NSString *key);
  77. void prepareForInvalidation();
  78. private:
  79. // Doubly-linked-list of observed objects for the same row as this
  80. RLMObservationInfo *next = nullptr;
  81. RLMObservationInfo *prev = nullptr;
  82. // Row being observed
  83. realm::Obj row;
  84. RLMClassInfo *objectSchema = nullptr;
  85. // Object doing the observing
  86. __unsafe_unretained id object = nil;
  87. // valueForKey: hack
  88. bool invalidated = false;
  89. size_t observerCount = 0;
  90. NSString *lastKey = nil;
  91. __unsafe_unretained RLMProperty *lastProp = nil;
  92. // objects returned from valueForKey() to keep them alive in case observers
  93. // are added and so that they can still be accessed after row is detached
  94. NSMutableDictionary *cachedObjects;
  95. void setRow(realm::Table const& table, realm::ObjKey newRow);
  96. template<typename F>
  97. void forEach(F&& f) const {
  98. // The user's observation handler may release their last reference to
  99. // the object being observed, which will result in the RLMObservationInfo
  100. // being destroyed. As a result, we need to retain the object which owns
  101. // both `this` and the current info we're looking at.
  102. __attribute__((objc_precise_lifetime)) id self = object, current;
  103. for (auto info = prev; info; info = info->prev) {
  104. current = info->object;
  105. f(info->object);
  106. }
  107. for (auto info = this; info; info = info->next) {
  108. current = info->object;
  109. f(info->object);
  110. }
  111. }
  112. // Default move/copy constructors don't work due to the intrusive linked
  113. // list and we don't need them
  114. RLMObservationInfo(RLMObservationInfo const&) = delete;
  115. RLMObservationInfo(RLMObservationInfo&&) = delete;
  116. RLMObservationInfo& operator=(RLMObservationInfo const&) = delete;
  117. RLMObservationInfo& operator=(RLMObservationInfo&&) = delete;
  118. };
  119. // Get the the observation info chain for the given row
  120. // Will simply return info if it's non-null, and will search ojectSchema's array
  121. // for a matching one otherwise, and return null if there are none
  122. RLMObservationInfo *RLMGetObservationInfo(RLMObservationInfo *info, realm::ObjKey row, RLMClassInfo& objectSchema);
  123. // delete all objects from a single table with change notifications
  124. void RLMClearTable(RLMClassInfo &realm);
  125. class RLMObservationTracker {
  126. public:
  127. RLMObservationTracker(RLMRealm *realm, bool trackDeletions=false);
  128. ~RLMObservationTracker();
  129. void trackDeletions();
  130. void willChange(RLMObservationInfo *info, NSString *key,
  131. NSKeyValueChange kind=NSKeyValueChangeSetting,
  132. NSIndexSet *indexes=nil);
  133. void didChange();
  134. private:
  135. std::vector<std::vector<RLMObservationInfo *> *> _observedTables;
  136. __unsafe_unretained RLMRealm const*_realm;
  137. realm::Group& _group;
  138. RLMObservationInfo *_info = nullptr;
  139. NSString *_key;
  140. NSKeyValueChange _kind = NSKeyValueChangeSetting;
  141. NSIndexSet *_indexes;
  142. struct Change {
  143. RLMObservationInfo *info;
  144. __unsafe_unretained NSString *property;
  145. NSMutableIndexSet *indexes;
  146. };
  147. std::vector<Change> _changes;
  148. std::vector<RLMObservationInfo *> _invalidated;
  149. template<typename CascadeNotification>
  150. void cascadeNotification(CascadeNotification const&);
  151. };
  152. std::vector<realm::BindingContext::ObserverState> RLMGetObservedRows(RLMSchemaInfo const& schema);
  153. void RLMWillChange(std::vector<realm::BindingContext::ObserverState> const& observed, std::vector<void *> const& invalidated);
  154. void RLMDidChange(std::vector<realm::BindingContext::ObserverState> const& observed, std::vector<void *> const& invalidated);
  155. // Used for checking if an `Object` declared with `@StateRealmObject` needs to have
  156. // it's accessors temporarily removed and added back so that the `Object` can be
  157. // managed be the Realm.
  158. [[clang::objc_runtime_visible]]
  159. @interface RLMSwiftUIKVO : NSObject
  160. + (BOOL)removeObserversFromObject:(NSObject *)object;
  161. + (void)addObserversToObject:(NSObject *)object;
  162. @end