Ecosyste.ms: Awesome

An open API service indexing awesome lists of open source software.

Awesome Lists | Featured Topics | Projects

https://github.com/k-lpmg/RealmWrapper

Safe and easy wrappers for RealmSwift
https://github.com/k-lpmg/RealmWrapper

realm-mobile-database realmcocoa realmswift swift wrapper-library

Last synced: about 2 months ago
JSON representation

Safe and easy wrappers for RealmSwift

Awesome Lists containing this project

README

        

# RealmWrapper

[![Cocoapods](https://img.shields.io/cocoapods/v/RealmWrapper.svg?style=flat)](https://cocoapods.org/pods/RealmWrapper)
[![Carthage compatible](https://img.shields.io/badge/Carthage-compatible-4BC51D.svg?style=flat)](https://github.com/Carthage/Carthage)
![Swift](https://img.shields.io/badge/Swift-5.0-orange.svg)
[![GitHub license](https://img.shields.io/badge/license-MIT-lightgrey.svg?style=flat)](https://raw.githubusercontent.com/k-lpmg/RealmWrapper/master/LICENSE)
[![Build Status](https://travis-ci.org/k-lpmg/RealmWrapper.svg?branch=master)](https://travis-ci.org/k-lpmg/RealmWrapper)

RealmWrapper is wrapper library for [RealmSwift](https://github.com/realm/realm-cocoa/tree/master/RealmSwift) in [realm-cocoa](https://github.com/realm/realm-cocoa)

If you use [RealmWrapper](https://github.com/k-lpmg/RealmWrapper), you can easily use UI update through Notification and Transaction processing.
Also, you do not have to worry about the retain cycle when using self in the Notification block.

- [RealmWrapper](#realmwrapper)
- [At a Glance](#at-a-glance)
- [Threading](#threading)
- [Getting Started](#getting-started)
- [Usage](#usage)
- [RealmManageable Property](#realmmanageable-property)
- [Example](#example)
- [Installation](#installation)
- [CocoaPods (iOS 9+)](#cocoapods-ios-9)
- [Carthage (iOS 9+)](#carthage-ios-9)
- [Swift Package Manager (Swift 5.2+, iOS 11+)](#swift-package-manager-swift-52-ios-11)
- [LICENSE](#license)

## At a Glance

If you use [RealmSwift](https://github.com/realm/realm-cocoa/tree/master/RealmSwift), you have to be careful about try statement and thread processing every transaction.
However, In [RealmManageable](https://github.com/k-lpmg/RealmWrapper/blob/master/Sources/RealmManageable.swift) which manages one realm file, transaction processing is performed using Realm-only DispatchQueue.
[RealmProxiable](https://github.com/k-lpmg/RealmWrapper/blob/master/Sources/RealmProxiable.swift), which owns [RealmManageable](https://github.com/k-lpmg/RealmWrapper/blob/master/Sources/RealmManageable.swift), can easily add, modify or delete models without having to worry about try statement and thread processing.

> User Model
```swift
@objcMembers
class User: Object {
dynamic var id: String?
dynamic var name: String?

override static func primaryKey() -> String? {
return "id"
}
}

var user = User()
user.id = UUID().uuidString
user.name = "Kevin"
```

> Using RealmSwift
```swift
let realm = try! Realm(configuration: Realm.Configuration(fileURL: URL(fileURLWithPath: RLMRealmPathForFile("user.realm")), schemaVersion: 1, objectTypes: [User.self]))
try! realm.write {
realm.add(user)
}
```

> Using RealmWrapper
```swift
UserRealmManager().transaction(writeHandler: { (realm) in
realm.add(user)
})
```

```swift
UserRealmProxy().append(user)
```

## Threading

- By default, you can use the transaction function to process a Realm Transaction in MainThread
```swift
UserRealmManager().transaction(writeHandler: { (realm) in
realm.add(user)
})
```

- It can be implemented by a background thread using DispatchQueue and isSync parameters.
```swift
UserRealmManager().transaction(isSync: false, writeHandler: { (realm) in
realm.add(user)
})
```

```swift
UserRealmManager().transaction(writeQueue: DispatchQueue(label: "background"), isSync: false, writeHandler: { (realm) in
realm.add(user)
})
```

- You can add completion closure.
```swift
UserRealmManager().transaction(isSync: false, writeHandler: { (realm) in
realm.add(user)
}) { (realm, error) in
self.tableView.reloadData()
}
```

## Getting Started

1. Create a `RealmManager` that manages one realm file.

```swift
final class UserRealmManager: RealmManageable {

var isUseInMemory: Bool {
return false
}
var schemaVersion: UInt64 {
return 1
}
var fileName: String {
return "user"
}
var objectTypes: [Object.Type]? {
return [User.self]
}

}
```

2. Create a `RealmProxy` that is responsible for the CRUD function to be accessed by the Controller.

```swift
struct UserRealmProxy: RealmProxiable {

var users: RealmQuery {
return query(sortProperty: "date", ordering: .ascending)
}

func append(_ user: User) {
rm.transaction(writeHandler: { (realm) in
realm.add(user, update: .all)
})
}

func delete(_ user: User) {
rm.transaction(writeHandler: { (realm) in
realm.delete(user)
})
}

func updateName(id: String, name: String, age: Int) {
guard let user = userFromId(id) else {return}

rm.transaction(writeHandler: { (realm) in
user.name = name
user.age = age
realm.add(user, update: .all)
})
}

func userFromId(_ id: String) -> User? {
return query(filter: "id == '\(id)'").results.first
}

}

var user = User()
user.id = UUID().uuidString
user.name = "Kevin"

UserRealmProxy().append(user)
UserRealmProxy().delete(user)
UserRealmProxy().updateName(id: user.id, name: "Kris")
```

3. If you want to register the notification of the status of the Realm object in the controller, you can register the notification in [RealmQuery](https://github.com/k-lpmg/RealmWrapper/blob/master/Sources/RealmWrapper/RealmQuery.swift).

```swift
let users: RealmQuery = UserRealmProxy().users

users.setSection(0)
.addInsertNotificationBlock(self) { (self, insertions) in
self.tableView.reloadData()
}
.addModificateNotificationBlock(self) { (self, modifications) in
self.tableView.reloadRows(at: modifications, with: .fade)
}
.addDeleteNotificationBlock(self) { (self, deletions) in
self.tableView.deleteRows(at: deletions, with: .fade)
}
.registerNotification()
```

Also, since [RealmQuery](https://github.com/k-lpmg/RealmWrapper/blob/master/Sources/RealmWrapper/RealmQuery.swift) is doing weak handling to prevent the retain cycle, you can use closures without worrying about the retain cycle if you pass only self.

```swift

public func addDeleteNotificationBlock(_ object: Object, block: @escaping (Object, [IndexPath]) -> Void) -> Self {
deleteNotificationBlock = { [weak object] (deletions) in
guard let weakObject = object else {return}

block(weakObject, deletions)
}

return self
}
```

## Usage

#### RealmManageable Property

| Property | Type | Default | Description |
| -------- | ---- | ------- | ---------- |
| `isUseInMemory` | `Bool` | `required` |`Use InMemory Realm` |
| `fileName` | `Bool` | `required` |`Realm File Name` |
| `appGroupIdentifier` | `String?` | `nil` |`Value for Realm file directory for App Extension`|

## Example

1. run carthage update
```console
$ carthage update --platform iOS
```

2. open RealmWrapper.xcodeproj
```console
$ open RealmWrapper.xcodeproj
```

3. run RealmWrapperExample

## Installation

#### CocoaPods (iOS 9+)

```ruby
platform :ios, '9.0'
use_frameworks!

target '' do
pod 'RealmWrapper'
end
```

#### Carthage (iOS 9+)

```ruby
github "k-lpmg/RealmWrapper"
```

#### Swift Package Manager (Swift 5.2+, iOS 11+)

```swift
.package(url: "https://github.com/k-lpmg/RealmWrapper.git", .upToNextMajor(from: "1.4.4")),
```

## LICENSE

These works are available under the MIT license. See the [LICENSE][license] file
for more info.

[license]: LICENSE