Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/artemstepanenko/Dekoter
NSCoding's counterpart for Swift structs.
https://github.com/artemstepanenko/Dekoter
decoding swift swift-extensions swift-library swift-structs userdefaults
Last synced: about 1 month ago
JSON representation
NSCoding's counterpart for Swift structs.
- Host: GitHub
- URL: https://github.com/artemstepanenko/Dekoter
- Owner: artemstepanenko
- License: mit
- Created: 2016-12-27T19:27:51.000Z (about 8 years ago)
- Default Branch: master
- Last Pushed: 2019-09-27T15:49:28.000Z (over 5 years ago)
- Last Synced: 2024-07-28T17:21:34.034Z (6 months ago)
- Topics: decoding, swift, swift-extensions, swift-library, swift-structs, userdefaults
- Language: Swift
- Homepage:
- Size: 366 KB
- Stars: 25
- Watchers: 4
- Forks: 1
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
- awesome-ios - Dekoter - `NSCoding`'s counterpart for Swift structs. (Data Structures / Algorithms / Getting Started)
- awesome-ios-star - Dekoter - `NSCoding`'s counterpart for Swift structs. (Data Structures / Algorithms / Getting Started)
- fucking-awesome-ios - Dekoter - `NSCoding`'s counterpart for Swift structs. (Data Structures / Algorithms / Getting Started)
- fucking-awesome-ios - Dekoter - `NSCoding`'s counterpart for Swift structs. (Data Structures / Algorithms / Getting Started)
README
# Dekoter
[![Build Status](https://travis-ci.org/artemstepanenko/Dekoter.svg?branch=master)](https://travis-ci.org/artemstepanenko/Dekoter)
[![Version](https://img.shields.io/cocoapods/v/Dekoter.svg?style=flat)](http://cocoapods.org/pods/Dekoter)
[![License](https://img.shields.io/cocoapods/l/Dekoter.svg?style=flat)](http://cocoapods.org/pods/Dekoter)
[![Platform](https://img.shields.io/cocoapods/p/Dekoter.svg?style=flat)](http://cocoapods.org/pods/Dekoter)- [Why You Might Be Interested](#why-you-might-be-interested)
- [How Much Familiar It Feels](#how-much-familiar-it-feels)
- [One More Example](#one-more-example)
- [What We've Learned from It](#what-weve-learned-from-it)
- [Features](#features)
- [Save an Object to UserDefaults](#save-an-object-to-userdefaults)
- [Archive and Unarchive an Object](#archive-and-unarchive-an-object)
- [JSON](#json)
- [Micromission](#micromission)
- [Why Dekoter](#why-dekoter)
- [Installation](#installation)
- [CocoaPods](#cocoapods)
- [Collaboration](#collaboration)
- [License](#license)## Why You Might Be Interested
Fills a gap left by the missing `NSCoding`'s support for Swift structs.
If you've ever implemented `NSCoding`, `Koting` will be familiar to you as well.## How Much Familiar It Feels
A quick reminder how to implement `NSCoding`:
```swift
class Cat: NSObject, NSCoding {let name: String
init(name: String) {
self.name = name
}// MARK: - NSCoding
private struct Key {
static let name = "name"
}func encode(with aCoder: NSCoder) {
aCoder.encode(name, forKey: Key.name)
}required convenience init?(coder aDecoder: NSCoder) {
guard let name = aDecoder.decodeObject(forKey: Key.name) as? String else {
return nil
}
self.init(name: name)
}
}
```Let's compare it to `Koting`:
```swift
struct Cat: Koting {let name: String
// MARK: - Koting
private struct Key {
static let name = "name"
}init?(koter: Koter) {
guard let name: String = koter.dekotObject(forKey: Key.name) else {
return nil
}
self.init(name: name)
}func enkot(with koter: Koter) {
koter.enkotObject(name, forKey: Key.name)
}
}
```Thus, not much different besides naming.
To summarize:
- Add the `Koting` protocol to the class declaration.
- Implement `init?(koter:)` and func `enkot(with:)`.
- Done!Once it's done, the compiler is happy, and you can convert objects to `Data` and back.
```swift
let puss = Cat(name: "Puss")
let data = NSKeyedArchiver.de_archivedData(withRootObject: puss)
guard let againPuss: Cat = NSKeyedUnarchiver.de_unarchiveObject(with: data) else { return }
```## One More Example
This one is going to depict most of the Dekoter's features.
```swift
struct Cat {enum Sex: Int {
case male
case female
}let name: String
let surname: String?
let sex: Sex
let nationality: String
let birthPlace: Place?// MARK: - Koting
private struct Key {
static let name = "name"
static let surname = "surname"
static let sex = "sex"
static let nationality = "nationality"
static let birthPlace = "birthPlace"
}init?(koter: Koter) {
guard let name: String = koter.dekotObject(forKey: Key.name),
let nationality: String = koter.dekotObject(forKey: Key.nationality),
let sexValue: Int = koter.dekotObject(forKey: Key.sex),
let sex = Sex(rawValue: sexValue) else {return nil
}
let surname: String? = koter.dekotObject(forKey: Key.surname)
let birthPlace: Place? = koter.dekotObject(forKey: Key.birthPlace)
self.init(name: name, surname: surname, sex: sex, nationality: nationality, birthPlace: birthPlace)
}func enkot(with koter: Koter) {
koter.enkotObject(name, forKey: Key.name)
koter.enkotObject(surname, forKey: Key.surname)
koter.enkotObject(sex.rawValue, forKey: Key.sex)
koter.enkotObject(nationality, forKey: Key.nationality)
koter.enkotObject(birthPlace, forKey: Key.birthPlace)
}
}
```### What We've Learned from It
- *It's okay to have optional properties.*
As you can see, there're two optional properties. To encode them you don't do anything special, `enkotObject(_, forKey:)` takes optional as the first argument. For decoding you use `dekotObject(forKey:)` which also returns optional and it's up to you how whether you unwrap it or not.
- *`Koter` supports the same parameter types as `NSCoding` and additionally types which implement `Koting`.*
In the example above `Cat` has an optional `birthPlace` property of a type `Place`.
- *There's only one method for encoding and one – for decoding.*
Regardless the type, you use the same methods: `enkotObject(_, forKey:)` for encoding and `dekotObject(forKey:)` for decoding. These methods are generic, they derive a type based on the expected return value, that's why you should always explicitly specify it.
## Features
### Save an Object to UserDefaults
There are two methods implemented in a `UserDefaults` extension: `de_set(_, forKey:)` and `de_object(forKey:)`
```swift
let murzik = Cat(name: "Murzik", surname: nil, sex: .male, nationality: "GER", birthPlace: nil)
userDefaults.de_set(murzik, forKey: "cat")
let againMurzik: Cat? = userDefaults.de_object(forKey: "cat")
```and
```swift
let sonya = Cat(name: "Sonya", surname: "Kryvonis", sex: .female, nationality: "UA", birthPlace: Place(country: "Ukraine", city: "Lviv"))
let puff: Cat = Cat(name: "Puff", surname: nil, sex: .female, nationality: "US", birthPlace: nil)
let cats = [ sonya, puff ]
userDefaults.de_set(cats, forKey: Key.cat)
guard let againCats: [Cat] = userDefaults.de_object(forKey: Key.cat) else { return }
```### Archive and Unarchive an Object
The library contains two extensions for `NSKeyedArchiver` and `NSKeyedUnarchiver` with methods for objects which implement the `Koting` protocol.
```swift
let emma = Cat(name: "Emma", surname: "Lambert", sex: .female, nationality: "FR", birthPlace: Place(country: "France", city: "Marseille"))
let data = NSKeyedArchiver.de_archivedData(withRootObject: emma)
guard let againEmma: Cat = NSKeyedUnarchiver.de_unarchiveObject(with: data) else { return }
```and
```swift
let sonya = Cat(name: "Sonya", surname: "Kryvonis", sex: .female, nationality: "UA", birthPlace: Place(country: "Ukraine", city: "Lviv"))
let puff: Cat = Cat(name: "Puff", surname: nil, sex: .female, nationality: "US", birthPlace: nil)
let cats = [ sonya, puff ]
let data = NSKeyedArchiver.de_archivedData(withRootObject: cats)
guard let againCats: [Cat] = NSKeyedUnarchiver.de_unarchiveObject(with: data) else { return }
```### JSON
A `JSONSerialization` extension makes deserialization from JSON very easy.
```swift
let oneCat: Cat? = JSONSerialization.de_jsonObject(with: oneCatData)
let cats: [Cat]? = JSONSerialization.de_jsonObject(with: catsData)
```For structs which make use only of this feature there's no need to implement the `Koting` protocol (contains 2 methods), instead implement a `Dekoting` protocol (only 1 method).
## Micromission
The library is small but proud of its mission, though the latter is also not that big. It's willing to serve developers as good as `NSCoding` does. Developers shouldn't feel lost and disappointed without a convenient tool to convert their Swift structs to `Data` and back.
## Why Dekoter
You might have noticed a few cats here and there. There's a reason.
"Kot" in some slavic languages means "cat", and it sounds similar to "code"."enkot" -> "encode"
"dekot" -> "decode"
"koter" -> "coder"
"koting" -> "coding"
"dekoter" -> "decoder"
## Installation
### CocoaPods
Add `pod 'Dekoter'` similar to the following to your Podfile:
```ruby
target 'MyApp' do
pod 'Dekoter'
end
```Then run a `pod install` inside your terminal, or from CocoaPods.app.
## Collaboration
Dear friends, your help is more than welcome!
There're multiple ways to support the project.- [Create an issue.](https://github.com/artemstepanenko/Dekoter/issues/new)
if you find a problem, or you know how to improve, or you have a question.
- [Create a pull request.](https://github.com/artemstepanenko/Dekoter/compare)
if you develop something important (previously filed as an issue).
- [Send me an email.](mailto:[email protected])
if you want to share your either positive or negative experience using the library and have a hard time expressing it in a form of issue. Or, maybe, you don't want to make it publicly available.
I'm always happy to read an email from you.
## License
It's available under the MIT license. See the [LICENSE](LICENSE) file for more info.
![](https://github.com/artemstepanenko/Dekoter/blob/master/cat.png)