forked from shripalsoni04/nativescript-webview-interface
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathindex-common.js
186 lines (163 loc) · 6.23 KB
/
index-common.js
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
/**
* Parses json string to object if valid.
*/
function parseJSON(data) {
var oData;
try {
oData = JSON.parse(data);
} catch (e) {
return false;
}
return oData;
}
/**
* WebViewInterface Class containing common functionalities for Android and iOS
*/
function WebViewInterface(webView) {
/**
* WebView to setup interface for
*/
this.webView = webView;
/**
* Mapping of webView event/command and its native handler
*/
this.eventListenerMap = {};
/**
* Mapping of js call request id and its success handler.
* Based on this mapping, the registered success handler will be called
* on successful response from the js call
*/
this.jsCallReqIdSuccessCallbackMap = {};
/**
* Mapping of js call request id and its error handler.
* Based on this mapping, the error handler will be called
* on error from the js call
*/
this.jsCallReqIdErrorCallbackMap = {};
/**
* Web-view instance unique id to handle scenarios of multiple webview on single page.
*/
this.id = ++WebViewInterface.cntWebViewId;
/**
* Maintaining mapping of webview instance and its id, to handle scenarios of multiple webview on single page.
*/
WebViewInterface.webViewInterfaceIdMap[this.id] = this;
}
/**
* Prepares call to a function in webView, which handles native event/command calls
*/
WebViewInterface.prototype._prepareEmitEventJSCall = function (eventName, data) {
data = typeof data === 'object' ? JSON.stringify(data) : '"' + data + '"';
return 'window.nsWebViewInterface._onNativeEvent("' + eventName + '",' + data + ');'
};
/**
* Prepares call to a function in webView, which calls the specified function in the webView
*/
WebViewInterface.prototype._prepareJSFunctionCall = function (functionName, arrArgs, successHandler, errorHandler) {
arrArgs = arrArgs || [];
// converts non array argument to array
if (typeof arrArgs !== 'object' || arrArgs.length === void (0)) {
arrArgs = [arrArgs];
}
var strArgs = JSON.stringify(arrArgs);
// creating id with combination of web-view id and req id
var reqId = '"'+this.id+'#'+ (++WebViewInterface.cntJSCallReqId)+'"';
this.jsCallReqIdSuccessCallbackMap[reqId] = successHandler;
this.jsCallReqIdErrorCallbackMap[reqId] = errorHandler;
return 'window.nsWebViewInterface._callJSFunction(' + reqId + ',"' + functionName + '",' + strArgs + ');'
}
/**
* Handles response/event/command from webView.
*/
WebViewInterface.prototype._onWebViewEvent = function (eventName, data) {
var oData = parseJSON(data) || data;
// in case of JS call result, eventName will be _jsCallResponse
if (eventName === '_jsCallResponse') {
var reqId = '"'+oData.reqId+'"';
var callback;
if(oData.isError){
callback = this.jsCallReqIdErrorCallbackMap[reqId];
}else{
callback = this.jsCallReqIdSuccessCallbackMap[reqId];
}
if (callback) {
callback(oData.response);
}
} else {
var lstCallbacks = this.eventListenerMap[eventName] || [];
for (var i = 0; i < lstCallbacks.length; i++) {
var retnVal = lstCallbacks[i](oData);
if (retnVal === false) {
break;
}
}
}
};
/**
* Registers handler for event/command emitted from webview
* @param {string} eventName - Any event name except reserved '_jsCallResponse'
* @param {function} callback - Callback function to be executed on event/command receive.
*/
WebViewInterface.prototype.on = function (eventName, callback) {
if(eventName === '_jsCallResponse'){
throw new Error('_jsCallResponse eventName is reserved for internal use. You cannot attach listeners to it.');
}
(this.eventListenerMap[eventName] || (this.eventListenerMap[eventName] = [])).push(callback);
};
/**
* Deregisters handler for event/command emitted from webview
* @param {string} eventName - Any event name except reserved '_jsCallResponse'
* @param {function} callback - Callback function to be executed on event/command receive.
**/
WebViewInterface.prototype.off = function (eventName, callback) {
if(eventName === '_jsCallResponse'){
throw new Error('_jsCallResponse eventName is reserved for internal use. You cannot deattach listeners to it.');
}
if (!this.eventListenerMap[eventName] || this.eventListenerMap[eventName].length === 0) {
return;
}
if (callback) {
this.eventListenerMap[eventName] = this.eventListenerMap[eventName].filter(function(oldCallback) {
return oldCallback !== callback;
});
} else {
delete this.eventListenerMap[eventName];
}
};
/**
* Emits event/command with payload to webView.
* @param {string} eventName - Any event name
* @param {any} data - Payload to send wiht event/command
*/
WebViewInterface.prototype.emit = function (eventName, data) {
var strJSFunction = this._prepareEmitEventJSCall(eventName, data);
this._executeJS(strJSFunction);
}
/**
* Calls function in webView
* @param {string} functionName - Function should be in global scope in webView
* @param {any[]} args - Arguments of the function
* @param {function} callback - Function to call on result from webView
*/
WebViewInterface.prototype.callJSFunction = function (functionName, args, successHandler, errorHandler) {
var strJSFunction = this._prepareJSFunctionCall(functionName, args, successHandler, errorHandler);
this._executeJS(strJSFunction);
};
/**
* Clears mappings of callbacks and webview.
* This needs to be called in navigatedFrom event handler in page where webviewInterface plugin is used.
*/
WebViewInterface.prototype.destroy = function(){
this.eventListenerMap = null;
this.jsCallReqIdSuccessCallbackMap = null;
this.jsCallReqIdErrorCallbackMap = null;
delete WebViewInterface.webViewInterfaceIdMap[this.id];
};
/**
* Counter to create unique requestId for each JS call to webView.
*/
WebViewInterface.cntJSCallReqId = 0;
WebViewInterface.cntWebViewId = 0;
WebViewInterface.webViewInterfaceIdMap = {};
exports.WebViewInterface = WebViewInterface;
exports.parseJSON = parseJSON;