-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathSwiftAppcuesFlutterPlugin.swift
217 lines (186 loc) · 8.04 KB
/
SwiftAppcuesFlutterPlugin.swift
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
import Flutter
import UIKit
import AppcuesKit
public class SwiftAppcuesFlutterPlugin: NSObject, FlutterPlugin {
private var implementation: Appcues?
private var analyticsChannel: FlutterEventChannel?
private var eventSink: FlutterEventSink?
public static func register(with registrar: FlutterPluginRegistrar) {
let methodChannel = FlutterMethodChannel(name: "appcues_flutter", binaryMessenger: registrar.messenger())
let instance = SwiftAppcuesFlutterPlugin()
instance.analyticsChannel = FlutterEventChannel(name: "appcues_analytics", binaryMessenger: registrar.messenger())
registrar.addMethodCallDelegate(instance, channel: methodChannel)
if #available(iOS 13.0, *) {
Appcues.elementTargeting = FlutterElementTargeting()
}
let factory = AppcuesFrameViewFactory(plugin: instance, messenger: registrar.messenger())
registrar.register(factory, withId: "AppcuesFrameView")
}
public func handle(_ call: FlutterMethodCall, result: @escaping FlutterResult) {
// init is a special case, which creates the Appcues instance, must be done first
if call.method == "initialize" {
if let accountID = call["accountId"], let applicationID = call["applicationId"] {
let config = Appcues.Config(accountID: accountID, applicationID: applicationID)
var enableUniversalLinks = false
if let arguments = call.arguments as? [String: Any] {
if let options = arguments["options"] as? [String: Any?] {
if let logging = options["logging"] as? Bool {
config.logging(logging)
}
if let apiHost = options["apiHost"] as? String, let url = URL(string: apiHost) {
config.apiHost(url)
}
if let sessionTimeout = options["sessionTimeout"] as? UInt {
config.sessionTimeout(sessionTimeout)
}
if let activityStorageMaxSize = options["activityStorageMaxSize"] as? UInt {
config.activityStorageMaxSize(activityStorageMaxSize)
}
if let activityStorageMaxAge = options["activityStorageMaxAge"] as? UInt {
config.activityStorageMaxAge(activityStorageMaxAge)
}
enableUniversalLinks = options["enableUniversalLinks"] as? Bool ?? false
}
if let additionalAutoProperties = arguments["additionalAutoProperties"] as? [String: Any?] {
config.additionalAutoProperties(additionalAutoProperties.compactMapValues { $0 })
}
}
config.enableUniversalLinks(enableUniversalLinks)
implementation = Appcues(config: config)
analyticsChannel?.setStreamHandler(self)
result(nil)
} else {
result(missingArgs(names: "accountId, applicationId"))
}
return
}
guard let implementation = implementation else {
result(FlutterError(code: "notInitialized",
message: "the initialize function must be called before any other Appcues SDK calls",
details: nil))
return
}
switch call.method {
case "identify":
if let userId = call["userId"] {
implementation.identify(userID: userId, properties: call.properties)
result(nil)
} else {
result(missingArgs(names: "userId"))
}
case "group":
implementation.group(groupID: call["groupId"], properties: call.properties)
result(nil)
case "track":
if let name = call["name"] {
implementation.track(name: name, properties: call.properties)
result(nil)
} else {
result(missingArgs(names: "name"))
}
case "screen":
if let title = call["title"] {
implementation.screen(title: title, properties: call.properties)
result(nil)
} else {
result(missingArgs(names: "title"))
}
case "anonymous":
implementation.anonymous()
result(nil)
case "reset":
implementation.reset()
result(nil)
case "version":
result(implementation.version())
case "debug":
implementation.debug()
result(nil)
case "show":
if let experienceID = call["experienceId"] {
implementation.show(experienceID: experienceID) { success, error in
if success {
result(nil)
} else {
result(FlutterError(
code: "show-experience-failure",
message: "unable to show experience \(experienceID)",
details: error?.localizedDescription))
}
}
} else {
result(missingArgs(names: "experienceId"))
}
case "didHandleURL":
if let urlString = call["url"], let url = URL(string: urlString) {
result(implementation.didHandleURL(url))
} else {
result(missingArgs(names: "url"))
}
case "setTargetElements":
guard #available(iOS 13.0, *),
let flutterElementTargeting = Appcues.elementTargeting as? FlutterElementTargeting,
let viewElements = call.parameters?["viewElements"] as? Array<Dictionary<String, Any>> else {
return
}
flutterElementTargeting.setTargetElements(viewElements: viewElements)
default:
result(FlutterMethodNotImplemented)
}
}
internal func register(frameID: String, for view: AppcuesFrameView, on parentViewController: UIViewController) {
implementation?.register(frameID: frameID, for: view, on: parentViewController)
}
private func missingArgs(names: String) -> FlutterError {
return FlutterError(code: "bad-args", message: "missing one or more required args", details: names)
}
}
extension SwiftAppcuesFlutterPlugin: AppcuesAnalyticsDelegate {
public func didTrack(analytic: AppcuesKit.AppcuesAnalytic, value: String?, properties: [String : Any]?, isInternal: Bool) {
let analyticName: String
switch analytic {
case .event:
analyticName = "EVENT"
case .screen:
analyticName = "SCREEN"
case .identify:
analyticName = "IDENTIFY"
case .group:
analyticName = "GROUP"
}
eventSink?([
"analytic": analyticName,
"value": value ?? "",
"properties": properties ?? [:],
"isInternal": isInternal
])
}
}
extension SwiftAppcuesFlutterPlugin: FlutterStreamHandler {
public func onListen(withArguments arguments: Any?, eventSink events: @escaping FlutterEventSink) -> FlutterError? {
eventSink = events
implementation?.analyticsDelegate = self
return nil
}
public func onCancel(withArguments arguments: Any?) -> FlutterError? {
implementation?.analyticsDelegate = nil
return nil
}
}
private extension FlutterMethodCall {
var parameters: Dictionary<String, Any>? {
arguments as? Dictionary<String, Any>
}
var properties: Dictionary<String, Any>? {
parameters?["properties"] as? Dictionary<String, Any>
}
func getParam(_ name: String) -> String? {
guard let parameters = parameters else { return nil }
return parameters[name] as? String
}
subscript(index: String) -> String? {
get {
getParam(index)
}
}
}