Ecosyste.ms: Awesome
An open API service indexing awesome lists of open source software.
https://github.com/tadija/AERecord
Super awesome Swift minion for Core Data (iOS, macOS, tvOS)
https://github.com/tadija/AERecord
Last synced: 3 months ago
JSON representation
Super awesome Swift minion for Core Data (iOS, macOS, tvOS)
- Host: GitHub
- URL: https://github.com/tadija/AERecord
- Owner: tadija
- License: mit
- Created: 2014-11-07T01:08:26.000Z (almost 10 years ago)
- Default Branch: master
- Last Pushed: 2018-09-18T20:50:43.000Z (about 6 years ago)
- Last Synced: 2024-08-07T17:54:03.356Z (3 months ago)
- Language: Swift
- Homepage:
- Size: 347 KB
- Stars: 303
- Watchers: 23
- Forks: 55
- Open Issues: 5
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- License: LICENSE
Awesome Lists containing this project
- awesome-ios - AERecord - Super awesome Core Data wrapper in Swift. (Core Data / Linter)
- awesome-ios-star - AERecord - Super awesome Core Data wrapper in Swift. (Core Data / Linter)
- awesome-swift-cn - AECoreDataUI - Core Data driven UI made in Swift. (Libs / UI)
- awesome-swift - AECoreDataUI - Core Data driven UI. (Libs / UI)
- awesome-swift - AECoreDataUI - Core Data driven UI. (Libs / UI)
- fucking-awesome-swift - AECoreDataUI - Core Data driven UI. (Libs / UI)
- fucking-awesome-ios - AERecord - Super awesome Core Data wrapper in Swift. (Core Data / Linter)
- fucking-awesome-ios - AERecord - Super awesome Core Data wrapper in Swift. (Core Data / Linter)
- awesome-swift - AECoreDataUI - Super awesome Swift minion for Core Data (iOS, macOS, tvOS) ` 📝 3 years ago ` (UI [🔝](#readme))
README
[![Swift 4.2](https://img.shields.io/badge/Swift-4.2-orange.svg?style=flat)](https://swift.org)
[![Platforms iOS | watchOS | tvOS | macOS](https://img.shields.io/badge/Platforms-iOS%20%7C%20watchOS%20%7C%20tvOS%20%7C%20macOS-lightgray.svg?style=flat)](http://www.apple.com)
[![CocoaPods](https://img.shields.io/cocoapods/v/AERecord.svg?style=flat)](https://cocoapods.org/pods/AERecord)
[![Carthage](https://img.shields.io/badge/Carthage-compatible-brightgreen.svg?style=flat)](https://github.com/Carthage/Carthage)
[![Swift Package Manager](https://img.shields.io/badge/SPM-compatible-brightgreen.svg)](https://github.com/apple/swift-package-manager)
[![License MIT](https://img.shields.io/badge/License-MIT-lightgrey.svg?style=flat)](https://github.com/tadija/AERecord/blob/master/LICENSE)> ⚠️ Since this repository is going to be archived soon, I suggest migrating to [NSPersistentContainer](https://developer.apple.com/documentation/coredata/nspersistentcontainer) instead (available since iOS 10).
> For other convenience helpers, beside managing stack, I'm currently just using [this](https://gist.github.com/tadija/6003830264d67a87193ff0c3d20373e7).# AERecord
**Super awesome Swift minion for Core Data (iOS, macOS, tvOS)**> I made this for personal use, but feel free to use it or contribute.
> For more examples check out [Sources](Sources) and [Tests](Tests).## Index
- [Intro](#intro)
- [Features](#features)
- [Usage](#usage)
- [Create Core Data stack](#create-core-data-stack)
- [Context operations](#context-operations)
- [Easy Queries](#easy-queries)
- [General](#general)
- [Create](#create)
- [Find first](#find-first)
- [Find all](#find-all)
- [Delete](#delete)
- [Count](#count)
- [Distinct](#distinct)
- [Auto increment](#auto-increment)
- [Turn managed object into fault](#turn-managed-object-into-fault)
- [Batch update](#batch-update)
- [Installation](#installation)
- [License](#license)## Intro
[AECoreDataUI](https://github.com/tadija/AECoreDataUI) was previously part of **AERecord**, so you may want to check that also.
Why do we need yet another one Core Data wrapper? You tell me!
Inspired by many different (spoiler alert) **magical** solutions, I wanted something which combines complexity and functionality just about right.
All that boilerplate code for setting up of Core Data stack, passing the right `NSManagedObjectContext` all accross the project and different threads, not to mention that boring `NSFetchRequest` boilerplates for any kind of creating or querying the data - should be more simple with this.## Features
- Create default or custom Core Data stack **(or more stacks)** easily accessible from everywhere
- Have **[main and background contexts](http://floriankugler.com/2013/04/29/concurrent-core-data-stack-performance-shootout/)**, always **in sync**, but don't worry about it
- [CRUD](https://en.wikipedia.org/wiki/Create,_read,_update_and_delete) data in many ways with **generic one liners**
- **iCloud** support
- Covered with **unit tests**
- Covered with inline docs## Usage
You may see [this demo project](https://github.com/tadija/AECoreDataDemo) for example.
### Create Core Data stack
Almost everything in `AERecord` is made with 'optional' parameters (which have default values if you don't specify anything).
So you can load (create if doesn't already exist) CoreData stack like this:```swift
do {
try AERecord.loadCoreDataStack()
} catch {
print(error)
}
```or like this:
```swift
let myModel: NSManagedObjectModel = AERecord.modelFromBundle(for: MyClass.self)
let myStoreType = NSInMemoryStoreType
let myConfiguration = ...
let myStoreURL = AERecord.storeURL(for: "MyName")
let myOptions = [NSMigratePersistentStoresAutomaticallyOption : true]
do {
try AERecord.loadCoreDataStack(managedObjectModel: myModel, storeType: myStoreType, configuration: myConfiguration, storeURL: myStoreURL, options: myOptions)
} catch {
print(error)
}
```or any combination of these.
If for any reason you want to completely remove your stack and start over (separate demo data stack for example) you can do it as simple as this:
```swift
do {
try AERecord.destroyCoreDataStack() // destroy default stack
} catch {
print(error)
}do {
let demoStoreURL = AERecord.storeURL(for: "Demo")
try AERecord.destroyCoreDataStack(storeURL: demoStoreURL) // destroy custom stack
} catch {
print(error)
}
```Similarly you can delete all data from all entities (without messing with the stack) like this:
```swift
AERecord.truncateAllData()
```### Context operations
Context for current thread (`Context.default`) is used if you don't specify any (all examples below are using `Context.default`).```swift
// get context
AERecord.Context.main // get NSManagedObjectContext for main thread
AERecord.Context.background // get NSManagedObjectContext for background thread
AERecord.Context.default // get NSManagedObjectContext for current thread// execute NSFetchRequest
let request = ...
let managedObjects = AERecord.execute(fetchRequest: request) // returns array of objects// save context
AERecord.save() // save default context
AERecord.saveAndWait() // save default context and wait for save to finish// turn managed objects into faults (you don't need this often, but sometimes you do)
let objectIDs = ...
AERecord.refreshObjects(with: [objectIDs], mergeChanges: true) // turn objects for given IDs into faults
AERecord.refreshRegisteredObjects(mergeChanges: true) // turn all registered objects into faults
```### Easy Queries
Easy querying helpers are created as `NSManagedObject` extension.
All queries are called on generic `NSManagedObject`, and `Context.default` is used if you don't specify any (all examples below are using `Context.default`). All finders have optional parameter for `NSSortDescriptor` which is not used in these examples.
For even more examples check out unit tests.#### General
If you need custom `NSFetchRequest`, you can use `createPredicate(with:)` and `createFetchRequest(predicate:sortdDescriptors:)`, tweak it as you wish and execute with `AERecord`.```swift
// create request for any entity type
let attributes = ...
let predicate = NSManagedObject.createPredicate(with: attributes)
let sortDescriptors = ...
let request = NSManagedObject.createFetchRequest(predicate: predicate, sortDescriptors: sortDescriptors)// set some custom request properties
request.someProperty = someValue// execute request and get array of entity objects
let managedObjects = AERecord.execute(fetchRequest: request)
```Of course, all of the often needed requests for creating, finding, counting or deleting entities are already there, so just keep reading.
#### Create
```swift
NSManagedObject.create() // create new objectlet attributes = ...
NSManagedObject.create(with: attributes) // create new object and sets it's attributesNSManagedObject.firstOrCreate(with: "city", value: "Belgrade") // get existing object (or create new if it doesn't already exist) with given attribute
let attributes = ...
NSManagedObject.firstOrCreate(with: attributes) // get existing object (or create new if it doesn't already exist) with given attributes
```#### Find first
```swift
NSManagedObject.first() // get first objectlet predicate = ...
NSManagedObject.first(with: predicate) // get first object with predicateNSManagedObject.first(with: "bike", value: "KTM") // get first object with given attribute name and value
let attributes = ...
NSManagedObject.first(with: attributes) // get first object with given attributesNSManagedObject.first(orderedBy: "speed", ascending: false) // get first object ordered by given attribute name
```#### Find all
```swift
NSManagedObject.all() // get all objectslet predicate = ...
NSManagedObject.all(with: predicate) // get all objects with predicateNSManagedObject.all(with: "year", value: 1984) // get all objects with given attribute name and value
let attributes = ...
NSManagedObject.all(with: attributes) // get all objects with given attributes
```#### Delete
```swift
let managedObject = ...
managedObject.delete() // delete object (call on instance)NSManagedObject.deleteAll() // delete all objects
NSManagedObject.deleteAll(with: "fat", value: true) // delete all objects with given attribute name and value
let attributes = ...
NSManagedObject.deleteAll(with: attributes) // delete all objects with given attributeslet predicate = ...
NSManagedObject.deleteAll(with: predicate) // delete all objects with given predicate
```#### Count
```swift
NSManagedObject.count() // count all objectslet predicate = ...
NSManagedObject.count(with: predicate) // count all objects with predicateNSManagedObject.count(with: "selected", value: true) // count all objects with given attribute name and value
let attributes = ...
NSManagedObject.count(with: attributes) // count all objects with given attributes
```#### Distinct
```swift
do {
try NSManagedObject.distinctValues(for: "city") // get array of all distinct values for given attribute name
} catch {
print(error)
}do {
let attributes = ["country", "city"]
try NSManagedObject.distinctRecords(for: attributes) // get dictionary with name and values of all distinct records for multiple given attributes
} catch {
print(error)
}
```#### Auto Increment
If you need to have auto incremented attribute, just create one with Int type and get next ID like this:```swift
NSManagedObject.autoIncrementedInteger(for: "myCustomAutoID") // returns next ID for given attribute of Integer type
```#### Turn managed object into fault
`NSFetchedResultsController` is designed to watch only one entity at a time, but when there is a bit more complex UI (ex. showing data from related entities too),
you sometimes have to manually refresh this related data, which can be done by turning 'watched' entity object into fault.
This is shortcut for doing just that (`mergeChanges` parameter defaults to `true`). You can read more about turning objects into faults in Core Data documentation.```swift
let managedObject = ...
managedObject.refresh() // turns instance of managed object into fault
```#### Batch update
Batch updating is the 'new' feature from iOS 8. It's doing stuff directly in persistent store, so be carefull with this and read the docs first. Btw, `NSPredicate` is also optional parameter here.```swift
NSManagedObject.batchUpdate(properties: ["timeStamp" : NSDate()]) // returns NSBatchUpdateResult?NSManagedObject.objectsCountForBatchUpdate(properties: ["timeStamp" : NSDate()]) // returns count of updated objects
NSManagedObject.batchUpdateAndRefreshObjects(properties: ["timeStamp" : NSDate()]) // turns updated objects into faults after updating them in persistent store
```## Installation
- [Swift Package Manager](https://swift.org/package-manager/):
```
.Package(url: "https://github.com/tadija/AERecord.git", majorVersion: 4)
```
- [Carthage](https://github.com/Carthage/Carthage):```ogdl
github "tadija/AERecord"
```- [CocoaPods](http://cocoapods.org/):
```ruby
pod 'AERecord'
```## License
AERecord is released under the MIT license. See [LICENSE](LICENSE) for details.