RLMAccessor.hpp 6.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160
  1. ////////////////////////////////////////////////////////////////////////////
  2. //
  3. // Copyright 2017 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 "RLMAccessor.h"
  19. #import "RLMClassInfo.hpp"
  20. #import "RLMDecimal128_Private.hpp"
  21. #import "RLMObjectId_Private.hpp"
  22. #import "RLMUUID_Private.hpp"
  23. #import "RLMUtil.hpp"
  24. #import <realm/object-store/object_accessor.hpp>
  25. @class RLMRealm;
  26. class RLMClassInfo;
  27. class RLMObservationTracker;
  28. typedef NS_ENUM(NSUInteger, RLMUpdatePolicy);
  29. // realm::util::Optional<id> doesn't work because Objective-C types can't
  30. // be members of unions with ARC, so this covers the subset of Optional that we
  31. // actually need.
  32. struct RLMOptionalId {
  33. id value;
  34. RLMOptionalId(id value) : value(value) { }
  35. explicit operator bool() const noexcept { return value; }
  36. id operator*() const noexcept { return value; }
  37. };
  38. // The subset of RLMAccessorContext which does not require any member variables.
  39. // Use this if you require to box/unbox types and you do not have access to the
  40. // parent object or realm.
  41. struct RLMStatelessAccessorContext {
  42. static id box(bool v) { return @(v); }
  43. static id box(double v) { return @(v); }
  44. static id box(float v) { return @(v); }
  45. static id box(long long v) { return @(v); }
  46. static id box(realm::StringData v) { return RLMStringDataToNSString(v) ?: NSNull.null; }
  47. static id box(realm::BinaryData v) { return RLMBinaryDataToNSData(v) ?: NSNull.null; }
  48. static id box(realm::Timestamp v) { return RLMTimestampToNSDate(v) ?: NSNull.null; }
  49. static id box(realm::Decimal128 v) { return v.is_null() ? NSNull.null : [[RLMDecimal128 alloc] initWithDecimal128:v]; }
  50. static id box(realm::ObjectId v) { return [[RLMObjectId alloc] initWithValue:v]; }
  51. static id box(realm::UUID v) { return [[NSUUID alloc] initWithRealmUUID:v]; }
  52. static id box(realm::util::Optional<bool> v) { return v ? @(*v) : NSNull.null; }
  53. static id box(realm::util::Optional<double> v) { return v ? @(*v) : NSNull.null; }
  54. static id box(realm::util::Optional<float> v) { return v ? @(*v) : NSNull.null; }
  55. static id box(realm::util::Optional<int64_t> v) { return v ? @(*v) : NSNull.null; }
  56. static id box(realm::util::Optional<realm::ObjectId> v) { return v ? box(*v) : NSNull.null; }
  57. static id box(realm::util::Optional<realm::UUID> v) { return v ? box(*v) : NSNull.null; }
  58. template<typename T>
  59. static T unbox(id v);
  60. template<typename Func>
  61. static void enumerate_collection(__unsafe_unretained const id v, Func&& func) {
  62. id enumerable = RLMAsFastEnumeration(v) ?: v;
  63. for (id value in enumerable) {
  64. func(value);
  65. }
  66. }
  67. template<typename Func>
  68. static void enumerate_dictionary(__unsafe_unretained const id v, Func&& func) {
  69. id enumerable = RLMAsFastEnumeration(v) ?: v;
  70. for (id key in enumerable) {
  71. func(unbox<realm::StringData>(key), v[key]);
  72. }
  73. }
  74. static bool is_null(id v) noexcept { return v == NSNull.null; }
  75. static id null_value() noexcept { return NSNull.null; }
  76. static id no_value() noexcept { return nil; }
  77. static bool allow_missing(id v) noexcept { return [v isKindOfClass:[NSArray class]]; }
  78. static bool is_same_list(realm::List const& list, id v) noexcept;
  79. static bool is_same_dictionary(realm::object_store::Dictionary const&, id) noexcept;
  80. static bool is_same_set(realm::object_store::Set const&, id) noexcept;
  81. static std::string print(id obj) { return [obj description].UTF8String; }
  82. };
  83. class RLMAccessorContext : public RLMStatelessAccessorContext {
  84. public:
  85. ~RLMAccessorContext();
  86. // Accessor context interface
  87. RLMAccessorContext(RLMAccessorContext& parent, realm::Obj const& parent_obj, realm::Property const& property);
  88. using RLMStatelessAccessorContext::box;
  89. id box(realm::List&&);
  90. id box(realm::Results&&);
  91. id box(realm::Object&&);
  92. id box(realm::Obj&&);
  93. id box(realm::object_store::Dictionary&&);
  94. id box(realm::object_store::Set&&);
  95. id box(realm::Mixed);
  96. void will_change(realm::Obj const&, realm::Property const&);
  97. void will_change(realm::Object& obj, realm::Property const& prop) { will_change(obj.obj(), prop); }
  98. void did_change();
  99. RLMOptionalId value_for_property(id dict, realm::Property const&, size_t prop_index);
  100. RLMOptionalId default_value_for_property(realm::ObjectSchema const&,
  101. realm::Property const& prop);
  102. template<typename T>
  103. T unbox(__unsafe_unretained id const v, realm::CreatePolicy = realm::CreatePolicy::Skip, realm::ObjKey = {}) {
  104. return RLMStatelessAccessorContext::unbox<T>(v);
  105. }
  106. template<>
  107. realm::Obj unbox(id v, realm::CreatePolicy, realm::ObjKey);
  108. template<>
  109. realm::Mixed unbox(id v, realm::CreatePolicy, realm::ObjKey);
  110. realm::Obj create_embedded_object();
  111. // Internal API
  112. RLMAccessorContext(RLMObjectBase *parentObject, const realm::Property *property = nullptr);
  113. RLMAccessorContext(RLMObjectBase *parentObject, realm::ColKey);
  114. RLMAccessorContext(RLMClassInfo& info);
  115. // The property currently being accessed; needed for KVO things for boxing
  116. // List and Results
  117. RLMProperty *currentProperty;
  118. std::pair<realm::Obj, bool>
  119. createObject(id value, realm::CreatePolicy policy, bool forceCreate=false, realm::ObjKey existingKey={});
  120. private:
  121. __unsafe_unretained RLMRealm *const _realm;
  122. RLMClassInfo& _info;
  123. realm::Obj _parentObject;
  124. RLMClassInfo* _parentObjectInfo = nullptr;
  125. realm::ColKey _colKey;
  126. // Cached default values dictionary to avoid having to call the class method
  127. // for every property
  128. NSDictionary *_defaultValues;
  129. std::unique_ptr<RLMObservationTracker> _observationHelper;
  130. id defaultValue(NSString *key);
  131. id propertyValue(id obj, size_t propIndex, __unsafe_unretained RLMProperty *const prop);
  132. };