Slide 1

Slide 1 text

Moya + RxSwi,Ͱ࣮ݱ͢Δʂ Reac%veͳAPIϦΫΤετ Զίϯ Vol.1 Day. 1 @satoshin21 @satoshin21 1

Slide 2

Slide 2 text

INTRODUCE • @satoshin21 • eureka, Inc. • Pairs JP iOS Applica

Slide 3

Slide 3 text

@satoshin21 3

Slide 4

Slide 4 text

What's Moya? @satoshin21 4

Slide 5

Slide 5 text

Whats' Moya? • Network abstrac.on layer • RxSwi6, Reac.veSwi6ͱͷ࿈ܞ͕σϑΥϧτͰՄೳ • encapsulates Alamofire • ϓϥάΠϯΛ࡞੒Մೳ • ίϛϡχςΟ͕׆ൃ ( ༷ʑͳExtension ) @satoshin21 5

Slide 6

Slide 6 text

How to implement Moya.TargetType public enum GitHub { case zen case userProfile(String) case userRepositories(String) } @satoshin21 6

Slide 7

Slide 7 text

How to implement Moya.TargetType extension GitHub: TargetType { public var baseURL: URL { return URL(string: "https://api.github.com")! } public var path: String { switch self { case .userProfile(let name): return "/users/\(name.urlEscaped)" ... } } public var method: Moya.Method = .get public var parameters: [String: Any]? { ... } } @satoshin21 7

Slide 8

Slide 8 text

How to request with RxSwi0 & Moya let githubProvider = MoyaProvider() githubProvider.request(.userProfile("satoshin21")).subscribe { event in switch event { case let .next(response): let jsonString = try? response.mapString() message = jsonString ?? message case let .error(error): print(error) default: break } } @satoshin21 8

Slide 9

Slide 9 text

@satoshin21 9

Slide 10

Slide 10 text

That's it? @satoshin21 10

Slide 11

Slide 11 text

Sampleͷ՝୊ • enumʹTargetTypeΛ࣮૷ • ΤϯυϙΠϯτ͕૿͑Δͱ஍ࠈ • Observable • Response ParseΛຖ౓ߦ͏ • APIKitͷΑ͏ʹType SafeͰ͸ͳ͍ @satoshin21 11

Slide 12

Slide 12 text

ཧ૝ܗ DoSomething() .asDriver(onErrorJustReturn: nil) .drive(imageView.rx.image) .addDisposableTo(disposeBag) @satoshin21 12

Slide 13

Slide 13 text

Moya Customiza-on for Pairs(jp) • enum to struct • Type safeͳObservable • Type Erasure @satoshin21 13

Slide 14

Slide 14 text

@satoshin21 14

Slide 15

Slide 15 text

enum to struct enum API { } extension API { enum Community {} } // Community API extension API.Community { /// ίϛϡχςΟ৘ใऔಘ struct GetSingle: PairsTargetType { ... } /// Search struct Search: PairsTargetType { ... } } @satoshin21 15

Slide 16

Slide 16 text

enum to struct let communitySearch = API.Community.Search() @satoshin21 16

Slide 17

Slide 17 text

PairsTargetType protocol PairsTargetType: TargetType, ObservableType { /// DecodedResponseType associatedtype D /// ObservableType.EʹόΠϯυͤ͞ΔͨΊ associatedtype E /// Ϩεϙϯε͔ΒೝΊ͍ͯΔܗࣜ΁σίʔυ͢ΔʢJSONͳͲʣ func decodeResponse(_ response: Moya.Response) throws -> D /// Ϩεϙϯε৘ใ͔Β׬ྃ·ͰͷObservableΛ࡞੒͢Δ func didDecodeData(_ responseData: D) -> Observable } @satoshin21 17

Slide 18

Slide 18 text

implements PairsTargetType struct GetCampaign: PairsTargetType { // Moya.Response -> JSON typealias D = JSON // JSON -> [Community] typealias E = [Community] func didDecodeData(_ responseData: JSON) -> Observable<[Community]> { // JSON͔Β[Community]΁ͷϚοϐϯά΍DBͷอଘͳͲ let communities = ... return communities } } @satoshin21 18

Slide 19

Slide 19 text

PairsTargetTypeͷObservableԽ extension PairsTargetType { func subscribe(_ observer: O) -> Disposable where O.E == E { // APIϦΫΤετ -> decodeResponse(response)ͰMoya.Response͔Βσίʔυ let requestData = Observable.create { (observer) in let cancellable = API.provider.request(.init(target: self), completion: { (result) in switch result { case .success(let response): let responseData = try self.decodeResponse(response) observer.onNext(responseData) observer.onCompleted() } ) return Disposables.create(with: cancellable.cancel) } // σίʔυ͞ΕͨΦϒδΣΫτ(ओʹJSON)͔ΒϚοϐϯά͞ΕͨΦϒδΣΫτ΁ return requestData .flatMap(self.didDecodeData) .subscribe(observer) } @satoshin21 19

Slide 20

Slide 20 text

RxMoyaProvider extension API { fileprivate static let provider = API.Provider() // PairsTargetTypeΛ࣮૷ͨ͠ϦΫΤετͷॲཧΛߦ͏ fileprivate class Provider: RxMoyaProvider { init() { } } } @satoshin21 20

Slide 21

Slide 21 text

Binding API.Image.Get() .asDriver(onErrorJustReturn: nil) .drive(imageView.rx.image) .addDisposableTo(disposeBag) @satoshin21 21

Slide 22

Slide 22 text

How we use Moya in Pairs // ίϛϡχςΟͱΧςΰϦʔΛऔಘ Observable .zip( API.Community.GetCommunities(), API.Community.GetCategories(), resultSelector: { $0 }) .subscribe( onNext: { (communities: [Community], categories: [Category]) in // do something! }).addDisposableTo(disposeBag) @satoshin21 22

Slide 23

Slide 23 text

Summary • Moyaࣗମ΋ڧྗ͕ͩɺΧελϚΠζͰΑΓReac)veʹ @satoshin21 23

Slide 24

Slide 24 text

@satoshin21 24