1 | (function () {
|
---|
2 |
|
---|
3 | if (typeof Prism === 'undefined' || typeof document === 'undefined') {
|
---|
4 | return;
|
---|
5 | }
|
---|
6 |
|
---|
7 | /**
|
---|
8 | * @callback Adapter
|
---|
9 | * @param {any} response
|
---|
10 | * @param {HTMLPreElement} [pre]
|
---|
11 | * @returns {string | null}
|
---|
12 | */
|
---|
13 |
|
---|
14 | /**
|
---|
15 | * The list of adapter which will be used if `data-adapter` is not specified.
|
---|
16 | *
|
---|
17 | * @type {Array<{adapter: Adapter, name: string}>}
|
---|
18 | */
|
---|
19 | var adapters = [];
|
---|
20 |
|
---|
21 | /**
|
---|
22 | * Adds a new function to the list of adapters.
|
---|
23 | *
|
---|
24 | * If the given adapter is already registered or not a function or there is an adapter with the given name already,
|
---|
25 | * nothing will happen.
|
---|
26 | *
|
---|
27 | * @param {Adapter} adapter The adapter to be registered.
|
---|
28 | * @param {string} [name] The name of the adapter. Defaults to the function name of `adapter`.
|
---|
29 | */
|
---|
30 | function registerAdapter(adapter, name) {
|
---|
31 | name = name || adapter.name;
|
---|
32 | if (typeof adapter === 'function' && !getAdapter(adapter) && !getAdapter(name)) {
|
---|
33 | adapters.push({ adapter: adapter, name: name });
|
---|
34 | }
|
---|
35 | }
|
---|
36 | /**
|
---|
37 | * Returns the given adapter itself, if registered, or a registered adapter with the given name.
|
---|
38 | *
|
---|
39 | * If no fitting adapter is registered, `null` will be returned.
|
---|
40 | *
|
---|
41 | * @param {string|Function} adapter The adapter itself or the name of an adapter.
|
---|
42 | * @returns {Adapter} A registered adapter or `null`.
|
---|
43 | */
|
---|
44 | function getAdapter(adapter) {
|
---|
45 | if (typeof adapter === 'function') {
|
---|
46 | for (var i = 0, item; (item = adapters[i++]);) {
|
---|
47 | if (item.adapter.valueOf() === adapter.valueOf()) {
|
---|
48 | return item.adapter;
|
---|
49 | }
|
---|
50 | }
|
---|
51 | } else if (typeof adapter === 'string') {
|
---|
52 | // eslint-disable-next-line no-redeclare
|
---|
53 | for (var i = 0, item; (item = adapters[i++]);) {
|
---|
54 | if (item.name === adapter) {
|
---|
55 | return item.adapter;
|
---|
56 | }
|
---|
57 | }
|
---|
58 | }
|
---|
59 | return null;
|
---|
60 | }
|
---|
61 | /**
|
---|
62 | * Remove the given adapter or the first registered adapter with the given name from the list of
|
---|
63 | * registered adapters.
|
---|
64 | *
|
---|
65 | * @param {string|Function} adapter The adapter itself or the name of an adapter.
|
---|
66 | */
|
---|
67 | function removeAdapter(adapter) {
|
---|
68 | if (typeof adapter === 'string') {
|
---|
69 | adapter = getAdapter(adapter);
|
---|
70 | }
|
---|
71 | if (typeof adapter === 'function') {
|
---|
72 | var index = adapters.findIndex(function (item) {
|
---|
73 | return item.adapter === adapter;
|
---|
74 | });
|
---|
75 | if (index >= 0) {
|
---|
76 | adapters.splice(index, 1);
|
---|
77 | }
|
---|
78 | }
|
---|
79 | }
|
---|
80 |
|
---|
81 | registerAdapter(function github(rsp) {
|
---|
82 | if (rsp && rsp.meta && rsp.data) {
|
---|
83 | if (rsp.meta.status && rsp.meta.status >= 400) {
|
---|
84 | return 'Error: ' + (rsp.data.message || rsp.meta.status);
|
---|
85 | } else if (typeof (rsp.data.content) === 'string') {
|
---|
86 | return typeof (atob) === 'function'
|
---|
87 | ? atob(rsp.data.content.replace(/\s/g, ''))
|
---|
88 | : 'Your browser cannot decode base64';
|
---|
89 | }
|
---|
90 | }
|
---|
91 | return null;
|
---|
92 | }, 'github');
|
---|
93 | registerAdapter(function gist(rsp, el) {
|
---|
94 | if (rsp && rsp.meta && rsp.data && rsp.data.files) {
|
---|
95 | if (rsp.meta.status && rsp.meta.status >= 400) {
|
---|
96 | return 'Error: ' + (rsp.data.message || rsp.meta.status);
|
---|
97 | }
|
---|
98 |
|
---|
99 | var files = rsp.data.files;
|
---|
100 | var filename = el.getAttribute('data-filename');
|
---|
101 | if (filename == null) {
|
---|
102 | // Maybe in the future we can somehow render all files
|
---|
103 | // But the standard <script> include for gists does that nicely already,
|
---|
104 | // so that might be getting beyond the scope of this plugin
|
---|
105 | for (var key in files) {
|
---|
106 | if (files.hasOwnProperty(key)) {
|
---|
107 | filename = key;
|
---|
108 | break;
|
---|
109 | }
|
---|
110 | }
|
---|
111 | }
|
---|
112 |
|
---|
113 | if (files[filename] !== undefined) {
|
---|
114 | return files[filename].content;
|
---|
115 | }
|
---|
116 | return 'Error: unknown or missing gist file ' + filename;
|
---|
117 | }
|
---|
118 | return null;
|
---|
119 | }, 'gist');
|
---|
120 | registerAdapter(function bitbucket(rsp) {
|
---|
121 | if (rsp && rsp.node && typeof (rsp.data) === 'string') {
|
---|
122 | return rsp.data;
|
---|
123 | }
|
---|
124 | return null;
|
---|
125 | }, 'bitbucket');
|
---|
126 |
|
---|
127 |
|
---|
128 | var jsonpCallbackCounter = 0;
|
---|
129 | /**
|
---|
130 | * Makes a JSONP request.
|
---|
131 | *
|
---|
132 | * @param {string} src The URL of the resource to request.
|
---|
133 | * @param {string | undefined | null} callbackParameter The name of the callback parameter. If falsy, `"callback"`
|
---|
134 | * will be used.
|
---|
135 | * @param {(data: unknown) => void} onSuccess
|
---|
136 | * @param {(reason: "timeout" | "network") => void} onError
|
---|
137 | * @returns {void}
|
---|
138 | */
|
---|
139 | function jsonp(src, callbackParameter, onSuccess, onError) {
|
---|
140 | var callbackName = 'prismjsonp' + jsonpCallbackCounter++;
|
---|
141 |
|
---|
142 | var uri = document.createElement('a');
|
---|
143 | uri.href = src;
|
---|
144 | uri.href += (uri.search ? '&' : '?') + (callbackParameter || 'callback') + '=' + callbackName;
|
---|
145 |
|
---|
146 | var script = document.createElement('script');
|
---|
147 | script.src = uri.href;
|
---|
148 | script.onerror = function () {
|
---|
149 | cleanup();
|
---|
150 | onError('network');
|
---|
151 | };
|
---|
152 |
|
---|
153 | var timeoutId = setTimeout(function () {
|
---|
154 | cleanup();
|
---|
155 | onError('timeout');
|
---|
156 | }, Prism.plugins.jsonphighlight.timeout);
|
---|
157 |
|
---|
158 | function cleanup() {
|
---|
159 | clearTimeout(timeoutId);
|
---|
160 | document.head.removeChild(script);
|
---|
161 | delete window[callbackName];
|
---|
162 | }
|
---|
163 |
|
---|
164 | // the JSONP callback function
|
---|
165 | window[callbackName] = function (response) {
|
---|
166 | cleanup();
|
---|
167 | onSuccess(response);
|
---|
168 | };
|
---|
169 |
|
---|
170 | document.head.appendChild(script);
|
---|
171 | }
|
---|
172 |
|
---|
173 | var LOADING_MESSAGE = 'Loading…';
|
---|
174 | var MISSING_ADAPTER_MESSAGE = function (name) {
|
---|
175 | return '✖ Error: JSONP adapter function "' + name + '" doesn\'t exist';
|
---|
176 | };
|
---|
177 | var TIMEOUT_MESSAGE = function (url) {
|
---|
178 | return '✖ Error: Timeout loading ' + url;
|
---|
179 | };
|
---|
180 | var UNKNOWN_FAILURE_MESSAGE = '✖ Error: Cannot parse response (perhaps you need an adapter function?)';
|
---|
181 |
|
---|
182 | var STATUS_ATTR = 'data-jsonp-status';
|
---|
183 | var STATUS_LOADING = 'loading';
|
---|
184 | var STATUS_LOADED = 'loaded';
|
---|
185 | var STATUS_FAILED = 'failed';
|
---|
186 |
|
---|
187 | var SELECTOR = 'pre[data-jsonp]:not([' + STATUS_ATTR + '="' + STATUS_LOADED + '"])'
|
---|
188 | + ':not([' + STATUS_ATTR + '="' + STATUS_LOADING + '"])';
|
---|
189 |
|
---|
190 |
|
---|
191 | Prism.hooks.add('before-highlightall', function (env) {
|
---|
192 | env.selector += ', ' + SELECTOR;
|
---|
193 | });
|
---|
194 |
|
---|
195 | Prism.hooks.add('before-sanity-check', function (env) {
|
---|
196 | var pre = /** @type {HTMLPreElement} */ (env.element);
|
---|
197 | if (pre.matches(SELECTOR)) {
|
---|
198 | env.code = ''; // fast-path the whole thing and go to complete
|
---|
199 |
|
---|
200 | // mark as loading
|
---|
201 | pre.setAttribute(STATUS_ATTR, STATUS_LOADING);
|
---|
202 |
|
---|
203 | // add code element with loading message
|
---|
204 | var code = pre.appendChild(document.createElement('CODE'));
|
---|
205 | code.textContent = LOADING_MESSAGE;
|
---|
206 |
|
---|
207 | // set language
|
---|
208 | var language = env.language;
|
---|
209 | code.className = 'language-' + language;
|
---|
210 |
|
---|
211 | // preload the language
|
---|
212 | var autoloader = Prism.plugins.autoloader;
|
---|
213 | if (autoloader) {
|
---|
214 | autoloader.loadLanguages(language);
|
---|
215 | }
|
---|
216 |
|
---|
217 | var adapterName = pre.getAttribute('data-adapter');
|
---|
218 | var adapter = null;
|
---|
219 | if (adapterName) {
|
---|
220 | if (typeof window[adapterName] === 'function') {
|
---|
221 | adapter = window[adapterName];
|
---|
222 | } else {
|
---|
223 | // mark as failed
|
---|
224 | pre.setAttribute(STATUS_ATTR, STATUS_FAILED);
|
---|
225 |
|
---|
226 | code.textContent = MISSING_ADAPTER_MESSAGE(adapterName);
|
---|
227 | return;
|
---|
228 | }
|
---|
229 | }
|
---|
230 |
|
---|
231 | var src = pre.getAttribute('data-jsonp');
|
---|
232 |
|
---|
233 | jsonp(
|
---|
234 | src,
|
---|
235 | pre.getAttribute('data-callback'),
|
---|
236 | function (response) {
|
---|
237 | // interpret the received data using the adapter(s)
|
---|
238 | var data = null;
|
---|
239 | if (adapter) {
|
---|
240 | data = adapter(response, pre);
|
---|
241 | } else {
|
---|
242 | for (var i = 0, l = adapters.length; i < l; i++) {
|
---|
243 | data = adapters[i].adapter(response, pre);
|
---|
244 | if (data !== null) {
|
---|
245 | break;
|
---|
246 | }
|
---|
247 | }
|
---|
248 | }
|
---|
249 |
|
---|
250 | if (data === null) {
|
---|
251 | // mark as failed
|
---|
252 | pre.setAttribute(STATUS_ATTR, STATUS_FAILED);
|
---|
253 |
|
---|
254 | code.textContent = UNKNOWN_FAILURE_MESSAGE;
|
---|
255 | } else {
|
---|
256 | // mark as loaded
|
---|
257 | pre.setAttribute(STATUS_ATTR, STATUS_LOADED);
|
---|
258 |
|
---|
259 | code.textContent = data;
|
---|
260 | Prism.highlightElement(code);
|
---|
261 | }
|
---|
262 | },
|
---|
263 | function () {
|
---|
264 | // mark as failed
|
---|
265 | pre.setAttribute(STATUS_ATTR, STATUS_FAILED);
|
---|
266 |
|
---|
267 | code.textContent = TIMEOUT_MESSAGE(src);
|
---|
268 | }
|
---|
269 | );
|
---|
270 | }
|
---|
271 | });
|
---|
272 |
|
---|
273 |
|
---|
274 | Prism.plugins.jsonphighlight = {
|
---|
275 | /**
|
---|
276 | * The timeout after which an error message will be displayed.
|
---|
277 | *
|
---|
278 | * __Note:__ If the request succeeds after the timeout, it will still be processed and will override any
|
---|
279 | * displayed error messages.
|
---|
280 | */
|
---|
281 | timeout: 5000,
|
---|
282 | registerAdapter: registerAdapter,
|
---|
283 | removeAdapter: removeAdapter,
|
---|
284 |
|
---|
285 | /**
|
---|
286 | * Highlights all `pre` elements under the given container with a `data-jsonp` attribute by requesting the
|
---|
287 | * specified JSON and using the specified adapter or a registered adapter to extract the code to highlight
|
---|
288 | * from the response. The highlighted code will be inserted into the `pre` element.
|
---|
289 | *
|
---|
290 | * Note: Elements which are already loaded or currently loading will not be touched by this method.
|
---|
291 | *
|
---|
292 | * @param {Element | Document} [container=document]
|
---|
293 | */
|
---|
294 | highlight: function (container) {
|
---|
295 | var elements = (container || document).querySelectorAll(SELECTOR);
|
---|
296 |
|
---|
297 | for (var i = 0, element; (element = elements[i++]);) {
|
---|
298 | Prism.highlightElement(element);
|
---|
299 | }
|
---|
300 | }
|
---|
301 | };
|
---|
302 |
|
---|
303 | }());
|
---|