DefaultsBridges.swift 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305
  1. //
  2. // SwiftyUserDefaults
  3. //
  4. // Copyright (c) 2015-present Radosław Pietruszewski, Łukasz Mróz
  5. //
  6. // Permission is hereby granted, free of charge, to any person obtaining a copy
  7. // of this software and associated documentation files (the "Software"), to deal
  8. // in the Software without restriction, including without limitation the rights
  9. // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
  10. // copies of the Software, and to permit persons to whom the Software is
  11. // furnished to do so, subject to the following conditions:
  12. //
  13. // The above copyright notice and this permission notice shall be included in all
  14. // copies or substantial portions of the Software.
  15. //
  16. // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
  17. // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
  18. // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
  19. // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
  20. // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
  21. // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
  22. // SOFTWARE.
  23. //
  24. import Foundation
  25. /// Class important for saving and getting values from UserDefaults. Be careful when you
  26. /// subclass your own!
  27. open class DefaultsBridge<T> {
  28. public init() {}
  29. /// This method provides a way of saving your data in UserDefaults. Usually needed
  30. /// when you want to create your custom Bridge, so you'll have to override it.
  31. open func save(key: String, value: T?, userDefaults: UserDefaults) {
  32. fatalError("This Bridge wasn't subclassed! Please do so before using it in your type.")
  33. }
  34. /// This method provides a way of saving your data in UserDefaults. Usually needed
  35. /// when you want to create your custom Bridge, so you'll have to override it.
  36. open func get(key: String, userDefaults: UserDefaults) -> T? {
  37. fatalError("This Bridge wasn't subclassed! Please do so before using it in your type.")
  38. }
  39. /// Override this function if your data is represented differently in UserDefaults
  40. /// and you map it in save/get methods.
  41. ///
  42. /// For instance, if you store it as Data in UserDefaults, but your type is not Data in your
  43. /// defaults key, then you need to `return true` here and provide `deserialize(_:)` method as well.
  44. ///
  45. /// Similar if you store your array of type as e.g. `[String]` but the type you use is actually `[SomeClassThatHasOnlyOneStringProperty]`.
  46. ///
  47. /// See `DefaultsRawRepresentableBridge` or `DefaultsCodableBridge` for examples.
  48. open func isSerialized() -> Bool {
  49. return false
  50. }
  51. /// Override this function if you've returned `true` in `isSerialized()` method.
  52. ///
  53. /// See `isSerialized()` method description for more details.
  54. open func deserialize(_ object: Any) -> T? {
  55. fatalError("You set `isSerialized` to true, now you have to implement `deserialize` method.")
  56. }
  57. }
  58. open class DefaultsObjectBridge<T>: DefaultsBridge<T> {
  59. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  60. userDefaults.set(value, forKey: key)
  61. }
  62. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  63. return userDefaults.object(forKey: key) as? T
  64. }
  65. }
  66. open class DefaultsArrayBridge<T: Collection>: DefaultsBridge<T> {
  67. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  68. userDefaults.set(value, forKey: key)
  69. }
  70. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  71. return userDefaults.array(forKey: key) as? T
  72. }
  73. }
  74. open class DefaultsStringBridge: DefaultsBridge<String> {
  75. open override func save(key: String, value: String?, userDefaults: UserDefaults) {
  76. userDefaults.set(value, forKey: key)
  77. }
  78. open override func get(key: String, userDefaults: UserDefaults) -> String? {
  79. return userDefaults.string(forKey: key)
  80. }
  81. }
  82. open class DefaultsIntBridge: DefaultsBridge<Int> {
  83. open override func save(key: String, value: Int?, userDefaults: UserDefaults) {
  84. userDefaults.set(value, forKey: key)
  85. }
  86. open override func get(key: String, userDefaults: UserDefaults) -> Int? {
  87. if let int = userDefaults.number(forKey: key)?.intValue {
  88. return int
  89. }
  90. // Fallback for launch arguments
  91. if let string = userDefaults.object(forKey: key) as? String,
  92. let int = Int(string) {
  93. return int
  94. }
  95. return nil
  96. }
  97. }
  98. open class DefaultsDoubleBridge: DefaultsBridge<Double> {
  99. open override func save(key: String, value: Double?, userDefaults: UserDefaults) {
  100. userDefaults.set(value, forKey: key)
  101. }
  102. open override func get(key: String, userDefaults: UserDefaults) -> Double? {
  103. if let double = userDefaults.number(forKey: key)?.doubleValue {
  104. return double
  105. }
  106. // Fallback for launch arguments
  107. if let string = userDefaults.object(forKey: key) as? String,
  108. let double = Double(string) {
  109. return double
  110. }
  111. return nil
  112. }
  113. }
  114. open class DefaultsBoolBridge: DefaultsBridge<Bool> {
  115. open override func save(key: String, value: Bool?, userDefaults: UserDefaults) {
  116. userDefaults.set(value, forKey: key)
  117. }
  118. open override func get(key: String, userDefaults: UserDefaults) -> Bool? {
  119. // @warning we use number(forKey:) instead of bool(forKey:), because
  120. // bool(forKey:) will always return value, even if it's not set
  121. //
  122. // Now, let's see if there is value in defaults that converts to Bool first:
  123. if let bool = userDefaults.number(forKey: key)?.boolValue {
  124. return bool
  125. }
  126. // If not, fallback for values saved in a plist (e.g. for testing)
  127. // For instance, few of the string("YES", "true", "NO", "false") convert to Bool from a property list
  128. return (userDefaults.object(forKey: key) as? String)?.bool
  129. }
  130. }
  131. open class DefaultsDataBridge: DefaultsBridge<Data> {
  132. open override func save(key: String, value: Data?, userDefaults: UserDefaults) {
  133. userDefaults.set(value, forKey: key)
  134. }
  135. open override func get(key: String, userDefaults: UserDefaults) -> Data? {
  136. return userDefaults.data(forKey: key)
  137. }
  138. }
  139. open class DefaultsUrlBridge: DefaultsBridge<URL> {
  140. open override func save(key: String, value: URL?, userDefaults: UserDefaults) {
  141. userDefaults.set(value, forKey: key)
  142. }
  143. open override func get(key: String, userDefaults: UserDefaults) -> URL? {
  144. return userDefaults.url(forKey: key)
  145. }
  146. open override func isSerialized() -> Bool {
  147. return true
  148. }
  149. open override func deserialize(_ object: Any) -> URL? {
  150. if let object = object as? URL {
  151. return object
  152. }
  153. if let object = object as? Data {
  154. return NSKeyedUnarchiver.unarchiveObject(with: object) as? URL
  155. }
  156. if let object = object as? NSString {
  157. let path = object.expandingTildeInPath
  158. return URL(fileURLWithPath: path)
  159. }
  160. return nil
  161. }
  162. }
  163. open class DefaultsCodableBridge<T: Codable>: DefaultsBridge<T> {
  164. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  165. guard let value = value else {
  166. userDefaults.removeObject(forKey: key)
  167. return
  168. }
  169. userDefaults.set(encodable: value, forKey: key)
  170. }
  171. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  172. guard let data = userDefaults.data(forKey: key) else {
  173. return nil
  174. }
  175. return deserialize(data)
  176. }
  177. open override func isSerialized() -> Bool {
  178. return true
  179. }
  180. open override func deserialize(_ object: Any) -> T? {
  181. guard let data = object as? Data else { return nil }
  182. return try? JSONDecoder().decode(T.self, from: data)
  183. }
  184. }
  185. open class DefaultsKeyedArchiverBridge<T>: DefaultsBridge<T> {
  186. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  187. guard let data = userDefaults.data(forKey: key) else {
  188. return nil
  189. }
  190. return deserialize(data)
  191. }
  192. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  193. guard let value = value else {
  194. userDefaults.removeObject(forKey: key)
  195. return
  196. }
  197. // Needed because Quick/Nimble have min target 10.10...
  198. if #available(OSX 10.11, *) {
  199. userDefaults.set(NSKeyedArchiver.archivedData(withRootObject: value), forKey: key)
  200. } else {
  201. fatalError("Shouldn't really happen. We do not support macOS 10.10, if it happened to you please report your use-case on GitHub issues.")
  202. }
  203. }
  204. open override func isSerialized() -> Bool {
  205. return true
  206. }
  207. open override func deserialize(_ object: Any) -> T? {
  208. guard let data = object as? Data else { return nil }
  209. return NSKeyedUnarchiver.unarchiveObject(with: data) as? T
  210. }
  211. }
  212. open class DefaultsRawRepresentableBridge<T: RawRepresentable>: DefaultsBridge<T> {
  213. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  214. guard let object = userDefaults.object(forKey: key) else { return nil }
  215. return deserialize(object)
  216. }
  217. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  218. userDefaults.set(value?.rawValue, forKey: key)
  219. }
  220. open override func isSerialized() -> Bool {
  221. return true
  222. }
  223. open override func deserialize(_ object: Any) -> T? {
  224. guard let rawValue = object as? T.RawValue else { return nil }
  225. return T(rawValue: rawValue)
  226. }
  227. }
  228. open class DefaultsRawRepresentableArrayBridge<T: Collection>: DefaultsBridge<T> where T.Element: RawRepresentable {
  229. open override func get(key: String, userDefaults: UserDefaults) -> T? {
  230. guard let object = userDefaults.array(forKey: key) else { return nil }
  231. return deserialize(object)
  232. }
  233. open override func save(key: String, value: T?, userDefaults: UserDefaults) {
  234. let raw = value?.map { $0.rawValue }
  235. userDefaults.set(raw, forKey: key)
  236. }
  237. open override func isSerialized() -> Bool {
  238. return true
  239. }
  240. open override func deserialize(_ object: Any) -> T? {
  241. guard let rawValue = object as? [T.Element.RawValue] else { return nil }
  242. return rawValue.compactMap { T.Element(rawValue: $0) } as? T
  243. }
  244. }