@operato/scene-restful 0.1.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +8 -0
- package/LICENSE +21 -0
- package/README.md +13 -0
- package/boards/boards.json +12 -0
- package/boards/groups.json +7 -0
- package/boards/models/SAMPLE.json +70 -0
- package/demo/index.html +116 -0
- package/demo/things-scene-restful.html +5 -0
- package/demo/things-theme.html +178 -0
- package/dist/index.d.ts +1 -0
- package/dist/index.js +5 -0
- package/dist/index.js.map +1 -0
- package/dist/jsonp.d.ts +19 -0
- package/dist/jsonp.js +73 -0
- package/dist/jsonp.js.map +1 -0
- package/dist/restful.d.ts +68 -0
- package/dist/restful.js +223 -0
- package/dist/restful.js.map +1 -0
- package/helps/scene/component/restful.ko.md +12 -0
- package/helps/scene/component/restful.md +11 -0
- package/helps/scene/component/restful.zh.md +13 -0
- package/icons/restful.png +0 -0
- package/package.json +62 -0
- package/src/index.ts +5 -0
- package/src/jsonp.ts +85 -0
- package/src/restful.ts +265 -0
- package/test/basic-test.html +67 -0
- package/test/index.html +24 -0
- package/test/unit/test-restful.js +33 -0
- package/test/unit/util.js +22 -0
- package/things-scene.config.js +24 -0
- package/translations/en.json +6 -0
- package/translations/ko.json +6 -0
- package/translations/zh.json +6 -0
- package/tsconfig.json +22 -0
- package/tsconfig.tsbuildinfo +1 -0
package/dist/restful.js
ADDED
|
@@ -0,0 +1,223 @@
|
|
|
1
|
+
import { Component, DataSource, RectPath, Shape, warn } from '@hatiolab/things-scene';
|
|
2
|
+
import jsonp from './jsonp';
|
|
3
|
+
const NATURE = {
|
|
4
|
+
mutable: false,
|
|
5
|
+
resizable: true,
|
|
6
|
+
rotatable: true,
|
|
7
|
+
properties: [
|
|
8
|
+
{
|
|
9
|
+
type: 'string',
|
|
10
|
+
label: 'url',
|
|
11
|
+
name: 'url'
|
|
12
|
+
},
|
|
13
|
+
{
|
|
14
|
+
type: 'number',
|
|
15
|
+
label: 'period',
|
|
16
|
+
name: 'period',
|
|
17
|
+
placeholder: 'SECONDS'
|
|
18
|
+
},
|
|
19
|
+
{
|
|
20
|
+
type: 'select',
|
|
21
|
+
label: 'data-format',
|
|
22
|
+
name: 'dataFormat',
|
|
23
|
+
property: {
|
|
24
|
+
options: [
|
|
25
|
+
{
|
|
26
|
+
display: 'Plain Text',
|
|
27
|
+
value: 'text'
|
|
28
|
+
},
|
|
29
|
+
{
|
|
30
|
+
display: 'JSON',
|
|
31
|
+
value: 'json'
|
|
32
|
+
},
|
|
33
|
+
{
|
|
34
|
+
display: 'JSONP',
|
|
35
|
+
value: 'jsonp'
|
|
36
|
+
}
|
|
37
|
+
]
|
|
38
|
+
}
|
|
39
|
+
},
|
|
40
|
+
{
|
|
41
|
+
type: 'checkbox',
|
|
42
|
+
label: 'with-credentials',
|
|
43
|
+
name: 'withCredentials'
|
|
44
|
+
}
|
|
45
|
+
],
|
|
46
|
+
'value-property': 'url',
|
|
47
|
+
help: 'scene/component/restful'
|
|
48
|
+
};
|
|
49
|
+
const REST_IMAGE = '';
|
|
50
|
+
const WARN_NO_URL = 'Valid URL property required';
|
|
51
|
+
export default class Restful extends DataSource(RectPath(Shape)) {
|
|
52
|
+
constructor() {
|
|
53
|
+
super(...arguments);
|
|
54
|
+
this._isStarted = false;
|
|
55
|
+
}
|
|
56
|
+
static get image() {
|
|
57
|
+
if (!Restful._image) {
|
|
58
|
+
Restful._image = new Image();
|
|
59
|
+
Restful._image.src = REST_IMAGE;
|
|
60
|
+
}
|
|
61
|
+
return Restful._image;
|
|
62
|
+
}
|
|
63
|
+
get url() {
|
|
64
|
+
return this.getState('url');
|
|
65
|
+
}
|
|
66
|
+
set url(url) {
|
|
67
|
+
this.setState('url', url);
|
|
68
|
+
this._initRestful();
|
|
69
|
+
}
|
|
70
|
+
get period() {
|
|
71
|
+
return this.state.period * 1000;
|
|
72
|
+
}
|
|
73
|
+
set period(period) {
|
|
74
|
+
this.setState('period', period);
|
|
75
|
+
this._initRestful();
|
|
76
|
+
}
|
|
77
|
+
get withCredentials() {
|
|
78
|
+
return !!this.getState('withCredentials');
|
|
79
|
+
}
|
|
80
|
+
set withCredentials(withCredentials) {
|
|
81
|
+
this.setState('withCredentials', withCredentials);
|
|
82
|
+
this._initRestful();
|
|
83
|
+
}
|
|
84
|
+
get repeatTimer() {
|
|
85
|
+
return this._repeatTimer;
|
|
86
|
+
}
|
|
87
|
+
set repeatTimer(repeatTimer) {
|
|
88
|
+
this._stopRepeater();
|
|
89
|
+
this._repeatTimer = repeatTimer;
|
|
90
|
+
}
|
|
91
|
+
get httpRequest() {
|
|
92
|
+
return this._httpRequest;
|
|
93
|
+
}
|
|
94
|
+
set httpRequest(httpRequest) {
|
|
95
|
+
this._httpRequest = httpRequest;
|
|
96
|
+
}
|
|
97
|
+
ready() {
|
|
98
|
+
this._initRestful();
|
|
99
|
+
}
|
|
100
|
+
_initRestful() {
|
|
101
|
+
if (!this.app.isViewMode)
|
|
102
|
+
return;
|
|
103
|
+
if (!this.url) {
|
|
104
|
+
warn(WARN_NO_URL);
|
|
105
|
+
return;
|
|
106
|
+
}
|
|
107
|
+
this._stopRepeater();
|
|
108
|
+
this._startRepeater();
|
|
109
|
+
}
|
|
110
|
+
dispose() {
|
|
111
|
+
super.dispose();
|
|
112
|
+
this._stopRepeater();
|
|
113
|
+
}
|
|
114
|
+
_startRepeater() {
|
|
115
|
+
this._isStarted = true;
|
|
116
|
+
var self = this;
|
|
117
|
+
// requestAnimationFrame 이 호출되지 않을 때는 ajax 호출도 하지 않도록 함.
|
|
118
|
+
function _() {
|
|
119
|
+
if (!self._isStarted) {
|
|
120
|
+
return;
|
|
121
|
+
}
|
|
122
|
+
self.callAjax();
|
|
123
|
+
if (!self.period) {
|
|
124
|
+
self._stopRepeater();
|
|
125
|
+
return;
|
|
126
|
+
}
|
|
127
|
+
self._repeatTimer = setTimeout(() => {
|
|
128
|
+
requestAnimationFrame(_);
|
|
129
|
+
}, self.period);
|
|
130
|
+
}
|
|
131
|
+
requestAnimationFrame(_);
|
|
132
|
+
}
|
|
133
|
+
_stopRepeater() {
|
|
134
|
+
this._abortRequest();
|
|
135
|
+
if (this.repeatTimer)
|
|
136
|
+
clearTimeout(this._repeatTimer);
|
|
137
|
+
this._isStarted = false;
|
|
138
|
+
}
|
|
139
|
+
_makeRequest(url) {
|
|
140
|
+
if (window.XMLHttpRequest) {
|
|
141
|
+
// Mozilla, Safari, ...
|
|
142
|
+
this.httpRequest = new XMLHttpRequest();
|
|
143
|
+
// @ts-ignore
|
|
144
|
+
}
|
|
145
|
+
else if (window.ActiveXObject) {
|
|
146
|
+
// IE
|
|
147
|
+
try {
|
|
148
|
+
// @ts-ignore
|
|
149
|
+
this.httpRequest = new ActiveXObject('Msxml2.XMLHTTP');
|
|
150
|
+
}
|
|
151
|
+
catch (e) {
|
|
152
|
+
try {
|
|
153
|
+
// @ts-ignore
|
|
154
|
+
this.httpRequest = new ActiveXObject('Microsoft.XMLHTTP');
|
|
155
|
+
}
|
|
156
|
+
catch (e) { }
|
|
157
|
+
}
|
|
158
|
+
}
|
|
159
|
+
if (!this.httpRequest) {
|
|
160
|
+
warn('Giving up :( Cannot create an XMLHTTP instance');
|
|
161
|
+
return false;
|
|
162
|
+
}
|
|
163
|
+
this.httpRequest.withCredentials = this.withCredentials;
|
|
164
|
+
this.httpRequest.open('GET', url);
|
|
165
|
+
this.httpRequest.onreadystatechange = this.onDataReceived.bind(this);
|
|
166
|
+
return true;
|
|
167
|
+
}
|
|
168
|
+
_makeRequestJsonp(url) {
|
|
169
|
+
jsonp(url, {}, (self, data) => {
|
|
170
|
+
if (!data)
|
|
171
|
+
return;
|
|
172
|
+
this.data = data;
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
_abortRequest() {
|
|
176
|
+
if (this.httpRequest)
|
|
177
|
+
this.httpRequest.abort();
|
|
178
|
+
}
|
|
179
|
+
onDataReceived() {
|
|
180
|
+
var { dataFormat = 'text' } = this.state;
|
|
181
|
+
if (this.httpRequest.readyState === 4) {
|
|
182
|
+
if (this.httpRequest.status === 200) {
|
|
183
|
+
var data = this.httpRequest.responseText;
|
|
184
|
+
if (!data)
|
|
185
|
+
return;
|
|
186
|
+
this.data = this._convertDataFormat(data, dataFormat);
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
}
|
|
190
|
+
callAjax() {
|
|
191
|
+
var { dataFormat = 'text' } = this.state;
|
|
192
|
+
if (dataFormat == 'jsonp') {
|
|
193
|
+
// @ts-ignore TODO add substitute property to things-scene.d.ts
|
|
194
|
+
this._makeRequestJsonp(this.substitute(this.url, this));
|
|
195
|
+
}
|
|
196
|
+
else {
|
|
197
|
+
// @ts-ignore TODO add substitute property to things-scene.d.ts
|
|
198
|
+
if (!this._makeRequest(this.substitute(this.url, this)))
|
|
199
|
+
return;
|
|
200
|
+
this.httpRequest.send();
|
|
201
|
+
}
|
|
202
|
+
}
|
|
203
|
+
_draw(context) {
|
|
204
|
+
var { left, top, width, height } = this.bounds;
|
|
205
|
+
context.beginPath();
|
|
206
|
+
this.drawImage(context, Restful.image, left, top, width, height);
|
|
207
|
+
}
|
|
208
|
+
ondblclick(e) {
|
|
209
|
+
if (!this.url) {
|
|
210
|
+
warn(WARN_NO_URL);
|
|
211
|
+
return;
|
|
212
|
+
}
|
|
213
|
+
this.callAjax();
|
|
214
|
+
}
|
|
215
|
+
get controls() {
|
|
216
|
+
return [];
|
|
217
|
+
}
|
|
218
|
+
get nature() {
|
|
219
|
+
return NATURE;
|
|
220
|
+
}
|
|
221
|
+
}
|
|
222
|
+
Component.register('restful', Restful);
|
|
223
|
+
//# sourceMappingURL=restful.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"restful.js","sourceRoot":"","sources":["../src/restful.ts"],"names":[],"mappings":"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","sourcesContent":["import { Component, DataSource, RectPath, Shape, warn } from '@hatiolab/things-scene'\n\nimport jsonp from './jsonp'\n\nconst NATURE = {\n mutable: false,\n resizable: true,\n rotatable: true,\n properties: [\n {\n type: 'string',\n label: 'url',\n name: 'url'\n },\n {\n type: 'number',\n label: 'period',\n name: 'period',\n placeholder: 'SECONDS'\n },\n {\n type: 'select',\n label: 'data-format',\n name: 'dataFormat',\n property: {\n options: [\n {\n display: 'Plain Text',\n value: 'text'\n },\n {\n display: 'JSON',\n value: 'json'\n },\n {\n display: 'JSONP',\n value: 'jsonp'\n }\n ]\n }\n },\n {\n type: 'checkbox',\n label: 'with-credentials',\n name: 'withCredentials'\n }\n ],\n 'value-property': 'url',\n help: 'scene/component/restful'\n}\n\nconst REST_IMAGE =\n ''\n\nconst WARN_NO_URL = 'Valid URL property required'\n\nexport default class Restful extends DataSource(RectPath(Shape)) {\n static _image: HTMLImageElement\n\n static get image() {\n if (!Restful._image) {\n Restful._image = new Image()\n Restful._image.src = REST_IMAGE\n }\n\n return Restful._image\n }\n\n get url() {\n return this.getState('url')\n }\n\n set url(url) {\n this.setState('url', url)\n this._initRestful()\n }\n\n get period() {\n return this.state.period * 1000\n }\n\n set period(period) {\n this.setState('period', period)\n this._initRestful()\n }\n\n get withCredentials() {\n return !!this.getState('withCredentials')\n }\n\n set withCredentials(withCredentials) {\n this.setState('withCredentials', withCredentials)\n this._initRestful()\n }\n\n _repeatTimer!: NodeJS.Timeout\n _httpRequest?: any\n _isStarted = false\n _convertDataFormat?: any\n\n get repeatTimer() {\n return this._repeatTimer\n }\n\n set repeatTimer(repeatTimer) {\n this._stopRepeater()\n this._repeatTimer = repeatTimer\n }\n\n get httpRequest() {\n return this._httpRequest\n }\n\n set httpRequest(httpRequest) {\n this._httpRequest = httpRequest\n }\n\n ready() {\n this._initRestful()\n }\n\n _initRestful() {\n if (!this.app.isViewMode) return\n\n if (!this.url) {\n warn(WARN_NO_URL)\n return\n }\n\n this._stopRepeater()\n this._startRepeater()\n }\n\n dispose() {\n super.dispose()\n this._stopRepeater()\n }\n\n _startRepeater() {\n this._isStarted = true\n\n var self = this\n\n // requestAnimationFrame 이 호출되지 않을 때는 ajax 호출도 하지 않도록 함.\n function _() {\n if (!self._isStarted) {\n return\n }\n self.callAjax()\n\n if (!self.period) {\n self._stopRepeater()\n return\n }\n\n self._repeatTimer = setTimeout(() => {\n requestAnimationFrame(_)\n }, self.period)\n }\n\n requestAnimationFrame(_)\n }\n\n _stopRepeater() {\n this._abortRequest()\n\n if (this.repeatTimer) clearTimeout(this._repeatTimer)\n this._isStarted = false\n }\n\n _makeRequest(url: string) {\n if (window.XMLHttpRequest) {\n // Mozilla, Safari, ...\n this.httpRequest = new XMLHttpRequest()\n // @ts-ignore\n } else if (window.ActiveXObject) {\n // IE\n try {\n // @ts-ignore\n this.httpRequest = new ActiveXObject('Msxml2.XMLHTTP')\n } catch (e) {\n try {\n // @ts-ignore\n this.httpRequest = new ActiveXObject('Microsoft.XMLHTTP')\n } catch (e) {}\n }\n }\n\n if (!this.httpRequest) {\n warn('Giving up :( Cannot create an XMLHTTP instance')\n return false\n }\n this.httpRequest.withCredentials = this.withCredentials\n this.httpRequest.open('GET', url)\n this.httpRequest.onreadystatechange = this.onDataReceived.bind(this)\n\n return true\n }\n\n _makeRequestJsonp(url: string) {\n jsonp(url, {}, (self, data) => {\n if (!data) return\n\n this.data = data\n })\n }\n\n _abortRequest() {\n if (this.httpRequest) this.httpRequest.abort()\n }\n\n onDataReceived() {\n var { dataFormat = 'text' } = this.state\n\n if (this.httpRequest.readyState === 4) {\n if (this.httpRequest.status === 200) {\n var data = this.httpRequest.responseText\n\n if (!data) return\n\n this.data = this._convertDataFormat(data, dataFormat)\n }\n }\n }\n\n callAjax() {\n var { dataFormat = 'text' } = this.state\n\n if (dataFormat == 'jsonp') {\n // @ts-ignore TODO add substitute property to things-scene.d.ts\n this._makeRequestJsonp(this.substitute(this.url, this))\n } else {\n // @ts-ignore TODO add substitute property to things-scene.d.ts\n if (!this._makeRequest(this.substitute(this.url, this))) return\n\n this.httpRequest.send()\n }\n }\n\n _draw(context: CanvasRenderingContext2D) {\n var { left, top, width, height } = this.bounds\n\n context.beginPath()\n this.drawImage(context, Restful.image, left, top, width, height)\n }\n\n ondblclick(e: Event) {\n if (!this.url) {\n warn(WARN_NO_URL)\n return\n }\n\n this.callAjax()\n }\n\n get controls() {\n return []\n }\n\n get nature() {\n return NATURE\n }\n}\n\nComponent.register('restful', Restful)\n"]}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
# restful
|
|
2
|
+
URL를 제공 받고 URL로만 데이터를 조회할 경우 사용하는 컴포넌트.
|
|
3
|
+
|
|
4
|
+
## properties
|
|
5
|
+
|
|
6
|
+
- url : 데이터 조회 URL
|
|
7
|
+
- period : 데이터 조회 주기
|
|
8
|
+
- data format : Plain Text, JSON, JSONP등 데이터 포멧
|
|
9
|
+
- Plain Text : 서버에서 데이터를 Plain Text로 반환할 경우 사용
|
|
10
|
+
- JSON : 서버에서 데이터를 JSON으로 제공할 경우
|
|
11
|
+
- JSONP : 데이터를 function형태로 전달 받을 경우 해당 function을 받아서 처리해 주는 기능
|
|
12
|
+
- with credentials : Cross site request 설정
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
# restful
|
|
2
|
+
A component used when a URL is provided and data is retrieved only with a URL.
|
|
3
|
+
|
|
4
|
+
## properties
|
|
5
|
+
- url : Data inquiry URL
|
|
6
|
+
- period : Data inquiry cycle
|
|
7
|
+
- data format : Plain Text, JSON, JSONP Data format
|
|
8
|
+
- Plain Text : Used when the server returns data in Plain Text.
|
|
9
|
+
- JSON : When the server provides data as JSON
|
|
10
|
+
- JSONP : When data is received in the form of a function, the function is received and processed.
|
|
11
|
+
- with credentials : Cross site request
|
|
Binary file
|
package/package.json
ADDED
|
@@ -0,0 +1,62 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@operato/scene-restful",
|
|
3
|
+
"description": "Restful Client Component for Things Scene",
|
|
4
|
+
"license": "MIT",
|
|
5
|
+
"author": "heartyoh",
|
|
6
|
+
"version": "0.1.2",
|
|
7
|
+
"main": "dist/index.js",
|
|
8
|
+
"module": "dist/index.js",
|
|
9
|
+
"things-scene": true,
|
|
10
|
+
"publishConfig": {
|
|
11
|
+
"access": "public",
|
|
12
|
+
"@oprato:registry": "https://registry.npmjs.org"
|
|
13
|
+
},
|
|
14
|
+
"repository": {
|
|
15
|
+
"type": "git",
|
|
16
|
+
"url": "git+https://github.com/things-scene/operato-scene.git",
|
|
17
|
+
"directory": "packages/restful"
|
|
18
|
+
},
|
|
19
|
+
"scripts": {
|
|
20
|
+
"serve": "tsc && things-factory-dev",
|
|
21
|
+
"start": "tsc && concurrently -k -r \"tsc --watch --preserveWatchOutput\" \"wds\"",
|
|
22
|
+
"build": "tsc",
|
|
23
|
+
"prepublish": "tsc",
|
|
24
|
+
"lint": "eslint --ext .ts,.html . --ignore-path .gitignore && prettier \"**/*.ts\" --check --ignore-path .gitignore",
|
|
25
|
+
"format": "eslint --ext .ts,.html . --fix --ignore-path .gitignore && prettier \"**/*.ts\" --write --ignore-path .gitignore",
|
|
26
|
+
"migration": "things-factory-migration"
|
|
27
|
+
},
|
|
28
|
+
"dependencies": {
|
|
29
|
+
"@hatiolab/things-scene": "^2.7.31"
|
|
30
|
+
},
|
|
31
|
+
"devDependencies": {
|
|
32
|
+
"@hatiolab/prettier-config": "^1.0.0",
|
|
33
|
+
"@operato/board": "^0.3.7",
|
|
34
|
+
"@things-factory/builder": "^4.0.27",
|
|
35
|
+
"@things-factory/operato-board": "^4.0.27",
|
|
36
|
+
"@types/chance": "^1.1.3",
|
|
37
|
+
"@typescript-eslint/eslint-plugin": "^4.33.0",
|
|
38
|
+
"@typescript-eslint/parser": "^4.33.0",
|
|
39
|
+
"@web/dev-server": "^0.1.28",
|
|
40
|
+
"concurrently": "^5.3.0",
|
|
41
|
+
"eslint": "^7.32.0",
|
|
42
|
+
"eslint-config-prettier": "^8.3.0",
|
|
43
|
+
"husky": "^4.3.8",
|
|
44
|
+
"lint-staged": "^10.5.4",
|
|
45
|
+
"prettier": "^2.4.1",
|
|
46
|
+
"tslib": "^2.3.1",
|
|
47
|
+
"typescript": "^4.5.2"
|
|
48
|
+
},
|
|
49
|
+
"prettier": "@hatiolab/prettier-config",
|
|
50
|
+
"husky": {
|
|
51
|
+
"hooks": {
|
|
52
|
+
"pre-commit": "lint-staged"
|
|
53
|
+
}
|
|
54
|
+
},
|
|
55
|
+
"lint-staged": {
|
|
56
|
+
"*.ts": [
|
|
57
|
+
"eslint --fix",
|
|
58
|
+
"prettier --write"
|
|
59
|
+
]
|
|
60
|
+
},
|
|
61
|
+
"gitHead": "a08c67f644a77255e0ccd00a9ebcdb81f89b8adc"
|
|
62
|
+
}
|
package/src/index.ts
ADDED
package/src/jsonp.ts
ADDED
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
import { debug } from '@hatiolab/things-scene'
|
|
2
|
+
|
|
3
|
+
var count = 0
|
|
4
|
+
|
|
5
|
+
function noop() {}
|
|
6
|
+
|
|
7
|
+
/**
|
|
8
|
+
* JSONP handler
|
|
9
|
+
*
|
|
10
|
+
* Options:
|
|
11
|
+
* - param {String} qs parameter (`callback`)
|
|
12
|
+
* - prefix {String} qs parameter (`__jp`)
|
|
13
|
+
* - name {String} qs parameter (`prefix` + incr)
|
|
14
|
+
* - timeout {Number} how long after a timeout error is emitted (`60000`)
|
|
15
|
+
*
|
|
16
|
+
* @param {String} url
|
|
17
|
+
* @param {Object|Function} optional options / callback
|
|
18
|
+
* @param {Function} optional callback
|
|
19
|
+
*/
|
|
20
|
+
|
|
21
|
+
export default function jsonp(
|
|
22
|
+
url: string,
|
|
23
|
+
opts: { prefix?: string; name?: string; param?: string; timeout?: number },
|
|
24
|
+
fn: (self: any, data?: any) => void
|
|
25
|
+
) {
|
|
26
|
+
if ('function' == typeof opts) {
|
|
27
|
+
fn = opts
|
|
28
|
+
opts = {}
|
|
29
|
+
}
|
|
30
|
+
if (!opts) opts = {}
|
|
31
|
+
|
|
32
|
+
var prefix = opts.prefix || '__jp'
|
|
33
|
+
|
|
34
|
+
// use the callback name that was passed if one was provided.
|
|
35
|
+
// otherwise generate a unique name by incrementing our counter.
|
|
36
|
+
var id = opts.name || prefix + count++
|
|
37
|
+
|
|
38
|
+
var param = opts.param || 'callback'
|
|
39
|
+
var timeout = null != opts.timeout ? opts.timeout : 60000
|
|
40
|
+
var enc = encodeURIComponent
|
|
41
|
+
var target = document.getElementsByTagName('script')[0] || document.head
|
|
42
|
+
var script: any
|
|
43
|
+
var timer: any
|
|
44
|
+
|
|
45
|
+
if (timeout) {
|
|
46
|
+
timer = setTimeout(function () {
|
|
47
|
+
cleanup()
|
|
48
|
+
if (fn) fn(new Error('Timeout'))
|
|
49
|
+
}, timeout)
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
function cleanup() {
|
|
53
|
+
if (script.parentNode) script.parentNode.removeChild(script)
|
|
54
|
+
//@ts-ignore
|
|
55
|
+
window[id] = noop
|
|
56
|
+
if (timer) clearTimeout(timer)
|
|
57
|
+
}
|
|
58
|
+
|
|
59
|
+
function cancel() {
|
|
60
|
+
// @ts-ignore
|
|
61
|
+
if (window[id]) {
|
|
62
|
+
cleanup()
|
|
63
|
+
}
|
|
64
|
+
}
|
|
65
|
+
|
|
66
|
+
//@ts-ignore
|
|
67
|
+
window[id] = function (data) {
|
|
68
|
+
debug('jsonp got', data)
|
|
69
|
+
cleanup()
|
|
70
|
+
if (fn) fn(null, data)
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// add qs component
|
|
74
|
+
url += (~url.indexOf('?') ? '&' : '?') + param + '=' + enc(id)
|
|
75
|
+
url = url.replace('?&', '?')
|
|
76
|
+
|
|
77
|
+
debug('jsonp req "%s"', url)
|
|
78
|
+
|
|
79
|
+
// create script
|
|
80
|
+
script = document.createElement('script')
|
|
81
|
+
script.src = url
|
|
82
|
+
target.parentNode?.insertBefore(script, target)
|
|
83
|
+
|
|
84
|
+
return cancel
|
|
85
|
+
}
|