#Palau: NSUserDefaults with Wings!
Features | Included Types | Installation | Validators and Defaults | Custom Types | DidSet Callback |
-------- Easily store your Custom Types in NSUserDefaults
- Most Standard Types Out-of-the-box
- Per-property based Chainable Rules
- Supports NSCoding and RawRepresentable
- 300% Type Safe :P
- 100% Unit Test Coverage
- Swift 3 features coming!
- Bool
- Int
- UInt
- Float
- Double
- String
- Array
- Dictionary
- NSNumber
- NSString
- NSArray
- NSDictionary
- NSDate
- NSData
- UIColor
- Swift 2.2
- iOS 8.0+ / tvOS 9.0+ / watchOS 2.0+
- Xcode 7.3+
To integrate Palau into your project using Carthage, add to your Cartfile:
github "symentis/Palau" ~> 1.0
Run carthage update to build the framework and drag the built Palau.framework into your Xcode project.
See more instructions on the Carthage page.
To integrate Palau into your project using CocoaPods, add to your Podfile:
use_frameworks!
pod 'Palau', '~> 1.0'Run pod install to install the framework into your Xcode workspace.
Import Palau
import PalauOnce you import the framework you can setup PalauDefaults like:
/// Your defaults are defined as extension of PalauDefaults
///
/// - The generic type of your PalauDefaultsEntry must conform 
///   to the protocol PalauDefaultable.
/// - We provide support for the most common types. 
/// - `value` is a helper function defined in PalauDefaults
/// - The String "backingName" is the key used in NSUserDefaults
/// - The empty `set` is used to please the compiler
extension PalauDefaults {
  /// a NSUserDefaults Entry of Type String with the key "backingName"
  public static var name: PalauDefaultsEntry<String> {
    get { return value("backingName") }
    set { }
  }
}Every value of a PalauDefaultsEntry will always be optional.
If you want to set a value you call:
PalauDefaults.name.value = "I am a great String value!"Getting your value back is as easy as:
/// name is an Optional<String>
let name = PalauDefaults.name.valueYou can delete a property by setting it to nil:
PalauDefaults.name.value = nilOr
/// skip custom rules and delete
PalauDefaults.name.clear()If you want to provide a default, when there is no value set, you can write a custom rule. This allows fine granular control on your values.
We include two rule types by default: whenNil and ensure
import Palau
extension PalauDefaults {
  public static var color: PalauDefaultsEntry<UIColor> {
    /// whenNil provides a value that will be returned
    /// when the related NSUserDefaults value is nil 
    /// (e.g. the 1st time, or after clear)
    get { return value("color").whenNil(use: UIColor.redColor())  }
    set { }
  }
}
/// is UIColor.redColor() 
let color: UIColor? = PalauDefaults.color.valueYou can also build up arbitrary rules for your value like:
/// Custom Validator Closure
let lessThan10: Int? -> Bool = {
  return $0.map { $0 < 10 } ?? false
}
/// the PalauDefaultsEntry with the key "intValueMin10" has 2 rules
/// - 1. when the value is nil - we will get or set 10
/// - 2. when the value is less than 10 (see lessThan10 closure) - we will also get or set 10
/// - Add as many chainable rules as you like
public static var intValueMin10: PalauDefaultsEntry<Int> {
  get { return value("intValue")
    .whenNil(use: 10)
    .ensure(when: lessThan10, use: 10) }
  set { }
}
/// try setting the property to 8
PalauDefaults.intValueMin10.value = 8
/// property ensured to be >= 10
assert(PalauDefaults.intValueMin10.value == 10)
/// try setting the property to 11
PalauDefaults.intValueMin10.value = 11
/// property changed to 11
assert(PalauDefaults.intValueMin10.value == 11)In Swift 2.2 Classes and Protocols can be used to constrain the ValueType.
For example this is how Palau adds support for RawRepresentable via an Extension:
/// Extension for RawRepresentable types aka enums
extension PalauDefaultable where ValueType: RawRepresentable {
  public static func get(key: String, from defaults: NSUD) -> ValueType? {
    guard let val = defaults.objectForKey(key) as? ValueType.RawValue else { return nil }
    return ValueType(rawValue: val)
  }
  public static func set(value: ValueType?, forKey key: String, in defaults: NSUD) -> Void {
    guard let value = value?.rawValue as? AnyObject else { return defaults.removeObjectForKey(key) }
    defaults.setObject(value, forKey: key)
  }
}Generally for Types which conform to NSCoding you can usually just provide an
extension like so:
/// Make UIColor PalauDefaultable
extension UIColor: PalauDefaultable {
  public typealias ValueType = UIColor
}For custom types you can provide an extension on your type for PalauDefaultable,
to implement a get and a get function.
// example Struct called Structy for demonstrating we can save a Struct with Palau
public struct Structy {
  let tuple: (String, String)
}
// our Structy PalauDefaultable extension allowing the mapping between PalauDefaults and the Type
// here we just map the two values to two keys named "1" and "2"
extension Structy: PalauDefaultable {
  public static func get(key: String, from defaults: NSUD) -> Structy? {
    guard let d = defaults.objectForKey(key) as? [String: AnyObject] ,
      let t1 = d["1"] as? String,
      let t2 = d["2"] as? String else { return nil }
    return Structy(tuple: (t1, t2))
  }
  public static func set(value: Structy?, forKey key: String, in defaults: NSUD) -> Void {
    guard let value = value else { return defaults.setObject(nil, forKey: key) }
    defaults.setObject(["1": value.tuple.0, "2": value.tuple.1], forKey: key)
  }
}
// now create a property on PalauDefaults
extension PalauDefaults {
  public static var structWithTuple: PalauDefaultsEntry<Structy> {
    get { return value("structy") }
    set { }
  }
}You can easily register a didSet callback, which gets fired when the value has changed.
extension PalauDefaults {
  public static var strings: PalauDefaultsEntry<[String]> {
    get { return value("strings").didSet({ print("changed to:", $0, "from:", $1) }) }
    set { }
  }
}We are waiting for more Swift 3 generics features like extensions on Generic types.... yay!
Then we get even more type saftey on arrays and dictionaries.
Plus we might be able to make generic types conform to PalauDefaultable.
Palau is named after the Palau swiftlet, a species of swift, endemic to the island of Palau.
Btw - if you really don`t like the name, you can use a typealias
typealias Defaults = PalauDefaults
typealias Defaultable = PalauDefaultable
/// for Swift 3 even:
/// typealias DefaultsEntry<T> = PalauDefaultsEntry<T>Palau is owned and maintained by Symentis GmbH.
Developed by: Elmar Kretzer & Madhava Jay
Follow for more Swift Goodness:
##Logo
Awesome Logo by: 4th motion
Palau is released under the Apache 2.0 license. See LICENSE for details.
