-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathpandora_media_session.user.js
340 lines (307 loc) · 21.3 KB
/
pandora_media_session.user.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
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
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
// ==UserScript==
// @name Pandora Media Session Support
// @namespace https://github.com/snaphat/pandora_media_session
// @version 0.5.0
// @description Shows media session information from Pandora Radio.
// @author Aaron Landwehr
// @icon https://raw.githubusercontent.com/snaphat/pandora_media_session_packager/main/assets/pandora_64x64.png
// @match *://*.pandora.com/*
// @grant none
// ==/UserScript==
// Note: document-idle breaks this script for firefox.
"use strict";
/**
* Determines if any audio elements on the page are currently playing.
*
* This function checks all audio elements in the document and returns a boolean indicating whether at least one audio
* element is not paused (i.e., is playing).
*
* @returns {boolean} True if any audio element is playing, false otherwise.
*/
function isPlayingPageAudio() {
return Array.from(document.querySelectorAll('audio')).some(audio => !audio.paused);
}
/**
* Checks if any audio is currently playing on the page.
*
* This function searches for an element with the class name "PlayButton" and checks its 'data-qa' attribute. The
* presence of a "PlayButton" with a 'data-qa' attribute value of "pause_button" indicates that audio is currently
* playing, as the button in this state is used to pause the audio.
*
* @returns {boolean} True if an element indicating playing audio is found, false otherwise.
*/
function isPlayingSomeAudio() {
// Fetch all elements with the class "PlayButton"
var e = document.getElementsByClassName("PlayButton");
// Check if the first element exists and its 'data-qa' attribute is "pause_button"
return e && e[0] && e[0].getAttribute('data-qa') == "pause_button";
}
/**
* Attaches a set of event listeners to a specified audio element to monitor various state changes.
*
* This function adds listeners for a wide range of audio-related events to the provided audio element. When any of
* these events occur, a callback function is executed, updating the state of the stubAudio and pausing it in order to
* synchronize the state of the stubAudio with the state of the page's primary audio element. If the stub audio was the
* last played, the media session metadata is reset to ensure the media session in chromium-based browsers updates
* correctly.
*
* This is a workaround to address a specific issue with chromium-based browsers. In these browsers, the media session
* tends to stop displaying if multiple audio elements play simultaneously within a single tab. To prevent this, the
* function ensures that whenever a page audio event occurs, the stub audio is immediately paused and a variable named
* `lastPageAudioEventTime` is set with the current time. Users of this function should always ensure that at least 1
* second has passed since the last page audio event occurred before calling stubAudio.play(). Additionally, if the
* stubAudio was the last audio played, the media session's metadata is reset to null to prompt a media session reset.
*
* @param {HTMLAudioElement} audio - The audio element to which the event listeners will be attached.
* @param {HTMLAudioElement} stubAudio - The stub audio element whose state will be updated in response to audio events.
*/
function addStateChangeEventListenerstoPageAudio(audio, stubAudio) {
// Callback to execute when any audio event occurs
var stateChangeCallback = () => {
// Record the time of the last page audio event
if (stubAudio.lastPlayed) { // if we are going to pause stub and it was the played last.
navigator.mediaSession.metadata = null; // Reset metadata to reset media session in chromium-based browsers.
}
stubAudio.lastPageAudioEventTime = Date.now();
stubAudio.lastPlayed = false;
// Pause the stub audio to avoid conflicts or overlapping with the main audio
stubAudio.pause();
};
// List of audio events to monitor
const audioEvents = [
"audioprocess", "abort", "canplay", "canplaythrough", "complete", "durationchange", "emptied", "ended", "error",
"loadeddata", "loadedmetadata", "loadstart", "pause", "play", "playing", "ratechange", "progress", "ratechange",
"resize", "seeked", "seeking", "stalled", "suspend", "timeupdate", "volumechange", "waiting"
];
// Attach the state change callback to each event type on the audio element
audioEvents.forEach(event => { audio.addEventListener(event, stateChangeCallback); });
}
/**
* Retrieves the source URL of the first image within the first element of a given class.
*
* This function is designed to navigate through the DOM starting from the first element of the specified class. It then
* attempts to access the first child's first child assuming it's an image element and returns its source URL. If any of
* these elements are not found or the structure is different, it returns an empty string.
*
* @param {string} cls - The class name of the element from which the image source is to be extracted.
* @returns {string} The source URL of the image if found, otherwise an empty string.
*/
function getArt(cls) {
let e = document.getElementsByClassName(cls);
return (e && e[0] && e[0].firstChild && e[0].firstChild.firstChild) ? e[0].firstChild.firstChild.src : "";
}
/**
* Retrieves the text content from the first element of a given class.
*
* This function selects the first element with the specified class name and returns its text content. If the element is
* not found, it returns an empty string. This is useful for extracting text from specific elements identified by their
* class name.
*
* @param {string} cls - The class name of the element whose text content is to be retrieved.
* @returns {string} The text content of the element if found, otherwise an empty string.
*/
function getText(cls) {
let e = document.getElementsByClassName(cls);
return e && e[0] ? e[0].textContent : "";
}
/**
* Updates the media session metadata based on the currently playing music.
*
* This function extracts music information such as the title, artist, album, and artwork from the DOM elements
* identified by specific class names. It then updates the media session's metadata with this information. If higher
* quality artwork is available, it replaces the existing artwork URL with the higher resolution version.
*/
const updateMetadata = (function () {
let isEvenCall = true;
return function () {
// Retrieve music information from the DOM
let title = getText('Tuner__Audio__TrackDetail__title');
let artist = getText('Tuner__Audio__TrackDetail__artist');
let album = getText('nowPlayingTopInfo__current__albumName');
let art = getArt('Tuner__Audio__TrackDetail__img');
// Attempt to get higher quality artwork if available
if (art) art = art.replace("90W", "500W").replace("90H", "500H");
let artwork = [{ src: art, sizes: '500x500', type: 'image/jpeg' }];
// Append a space to the title, artist, and album every other call, and duplicate the artwork entry in the array.
// These actions are individual workarounds to ensure that each piece of media metadata (title, artist, album,
// and artwork) updates correctly. This approach addresses issues in chromium-based browsers, where the displayed
// metadata does not always update, especially when the browser is not in focus. By making slight alterations
// to the metadata fields, the browser is more likely to recognize these changes and update the media session
// display accordingly.
if (!isEvenCall) {
title += " ";
artist += " ";
album += " ";
artwork[1] = artwork[0]; // Duplicate the artwork entry to reinforce the metadata update
}
// Update the media session metadata
navigator.mediaSession.metadata = new MediaMetadata({
title: title,
artist: artist,
album: album,
artwork: artwork
});
// Toggle isEvenCall for the next invocation
isEvenCall = !isEvenCall;
};
})();
/**
* Updates the playback state of the stub audio based on the current audio playback state on the page.
*
* This function first updates the media session playback state to "playing" or "paused" based on whether any audio is
* currently playing on the page, as determined by the 'isPlayingSomeAudio' function.
*
* It then checks whether at least 1 second has elapsed since the last event on any page audio element. If less than 1
* second has passed, it exits without making further changes to the stub audio. This delay is a workaround for
* chromium-based browsers, where the media session may stop displaying if multiple audio elements play simultaneously
* within a single tab.
*
* If the required time has elapsed, the stub audio is either played or paused based on the current audio state of the
* page, and whether any other audio is playing. Additionally, if the stub audio is about to play and it was not the
* last played audio, the media session's metadata is reset to prompt an update in the media session display.
*
* @param {HTMLAudioElement} stubAudio - The stub audio element whose playback state will be modified.
*/
function updateStubAudioState(stubAudio) {
// Update the media session playback state based on the current audio playback state on the page
let isPlaying = isPlayingSomeAudio();
if (isPlaying) {
navigator.mediaSession.playbackState = "playing";
} else {
navigator.mediaSession.playbackState = "paused";
}
// Check if at least 1 second has passed since the last page audio event
if ((Date.now() - stubAudio.lastPageAudioEventTime) < 1000) return;
// Play or pause the stub audio based on the audio playback state and the last played status
if (isPlaying && !isPlayingPageAudio()) {
if (!stubAudio.lastPlayed) { // if we are going to play stub but it wasn't played last.
navigator.mediaSession.metadata = null; // Reset metadata to reset media session in chromium-based browsers.
}
stubAudio.play(); // Play if no other audio is playing
stubAudio.lastPlayed = true; // set as last played (1 second interval passed).
} else {
stubAudio.pause();
}
}
/**
* Simulates a click event on the first element of a given class.
*
* This function creates a new 'click' MouseEvent and dispatches it on the first element found with the specified class
* name. This can be used to programmatically trigger click events on elements that match the given class. The event
* bubbles and is cancelable, mimicking the behavior of a standard user-initiated click event.
*
* @param {string} cls - The class name of the element on which to simulate the click event.
*/
function simulateClick(cls) {
// Create a new click event
let clickEvent = new MouseEvent('click', { view: null, bubbles: true, cancelable: true });
// Get the first element with the specified class
let e = document.getElementsByClassName(cls)[0];
// Dispatch the event on the element if it exists
if (e) e.dispatchEvent(clickEvent);
}
/**
* Sets up media session action handlers to control audio playback.
*
* This function configures the action handlers for the browser's media session, enabling control over audio playback
* through standard media controls such as play, pause, previous track, and next track. The play and pause actions are
* linked to simulated click events on specific DOM elements representing these controls, allowing integration with
* custom audio controls on the web page.
*
* To address a race condition in Firefox where rapid, programmatic click events can cause playback issues, a delay
* check is included. This check ensures that at least 200 milliseconds have passed since the last media session event
* before simulating another click. While this delay improves reliability in Firefox, it also means that the On-Screen
* Display (OSD) updates are not immediate and are dependent on the timing of these simulated clicks.
*/
function setupMediaSessionEventHandlers(stubAudio) {
// Set action handler for 'play' action
navigator.mediaSession.setActionHandler('play', () => {
// Check if at least 200ms has passed since the last media session click
if ((Date.now() - stubAudio.lastMediaSessionEventTime) < 200) return;
stubAudio.lastMediaSessionEventTime = Date.now();
simulateClick("PlayButton"); // Simulates a click on the Play button
});
// Set action handler for 'pause' action
navigator.mediaSession.setActionHandler('pause', () => {
// Check if at least 200ms has passed since the last media session click
if ((Date.now() - stubAudio.lastMediaSessionEventTime) < 200) return;
stubAudio.lastMediaSessionEventTime = Date.now();
simulateClick("PlayButton"); // Simulates a click on the Pause button
});
// Set action handler for 'previoustrack' (previous track or replay)
navigator.mediaSession.setActionHandler('previoustrack', () => {
simulateClick("ReplayButton"); // Simulates a click on the Replay button (Station)
simulateClick("Tuner__Control__SkipBack__Button"); // Simulates click on the Skip Back button (Playlist)
});
// Set action handler for 'nexttrack' (next track)
navigator.mediaSession.setActionHandler('nexttrack', () => {
simulateClick("Tuner__Control__Skip__Button"); // Simulates a click on the Skip button (Station)
simulateClick("Tuner__Control__SkipForward__Button"); // Simulates click on the Skip Forward button (Playlist)
});
}
/**
* Initializes the functionality for enhancing media session support.
* 1. Creates and adds a stub audio element to the DOM. This element is used to maintain a consistent media session
* and to enable On-Screen Display (OSD) features in browsers like Firefox.
* 2. Sets up a MutationObserver to monitor DOM changes for new audio elements and the removal of 'PlayButton' elements.
* It modifies the play functionality of added audio elements and adjusts the playback state of 'stubAudio when
* 'PlayButton' elements are removed.
* 3. Attaches custom play listeners to all existing and dynamically added audio elements, managing playback via the
* last audio element that was played, tracked by the 'lastPlayingAudio' variable.
* 4. Periodically updates media metadata based on the state of the real audio elements.
* 5. Configures media session event handlers for actions like play, pause, next track, and previous track, which also
* interact with the last played audio element to synchronize the media session's state.
*/
function initialize() {
/**
* Add a 'stub' audio element to the DOM. This stub audio is a small, silent audio file used to trick browsers into
* thinking there is always an audio element playing. This is necessary for enabling On-Screen Display (OSD)
* features, as some browsers require an actual audio file to be playing. The audio file used is very short and
* silent to be unobtrusive.
*/
let stubAudio = document.createElement('audio');
stubAudio.loop = true;
stubAudio.volume = 0.1;
stubAudio.src = "data:audio/ogg;base64,T2dnUwACAAAAAAAAAABsbAAAAAAAALBXT0MBHgF2b3JiaXMAAAAAARErAAAAAAAAIE4AAAAAAACZAU9nZ1MAAAAAAAAAAAAAbGwAAAEAAAC8MMvOCzv///////////+1A3ZvcmJpcysAAABYaXBoLk9yZyBsaWJWb3JiaXMgSSAyMDEyMDIwMyAoT21uaXByZXNlbnQpAAAAAAEFdm9yYmlzEkJDVgEAAAEADFIUISUZU0pjCJVSUikFHWNQW0cdY9Q5RiFkEFOISRmle08qlVhKyBFSWClFHVNMU0mVUpYpRR1jFFNIIVPWMWWhcxRLhkkJJWxNrnQWS+iZY5YxRh1jzlpKnWPWMUUdY1JSSaFzGDpmJWQUOkbF6GJ8MDqVokIovsfeUukthYpbir3XGlPrLYQYS2nBCGFz7bXV3EpqxRhjjDHGxeJTKILQkFUAAAEAAEAEAUJDVgEACgAAwlAMRVGA0JBVAEAGAIAAFEVxFMdxHEeSJMsCQkNWAQBAAAACAAAojuEokiNJkmRZlmVZlqZ5lqi5qi/7ri7rru3qug6EhqwEAMgAABiGIYfeScyQU5BJJilVzDkIofUOOeUUZNJSxphijFHOkFMMMQUxhtAphRDUTjmlDCIIQ0idZM4gSz3o4GLnOBAasiIAiAIAAIxBjCHGkHMMSgYhco5JyCBEzjkpnZRMSiittJZJCS2V1iLnnJROSialtBZSy6SU1kIrBQAABDgAAARYCIWGrAgAogAAEIOQUkgpxJRiTjGHlFKOKceQUsw5xZhyjDHoIFTMMcgchEgpxRhzTjnmIGQMKuYchAwyAQAAAQ4AAAEWQqEhKwKAOAEAgyRpmqVpomhpmih6pqiqoiiqquV5pumZpqp6oqmqpqq6rqmqrmx5nml6pqiqnimqqqmqrmuqquuKqmrLpqvatumqtuzKsm67sqzbnqrKtqm6sm6qrm27smzrrizbuuR5quqZput6pum6quvasuq6su2ZpuuKqivbpuvKsuvKtq3Ksq5rpum6oqvarqm6su3Krm27sqz7puvqturKuq7Ksu7btq77sq0Lu+i6tq7Krq6rsqzrsi3rtmzbQsnzVNUzTdf1TNN1Vde1bdV1bVszTdc1XVeWRdV1ZdWVdV11ZVv3TNN1TVeVZdNVZVmVZd12ZVeXRde1bVWWfV11ZV+Xbd33ZVnXfdN1dVuVZdtXZVn3ZV33hVm3fd1TVVs3XVfXTdfVfVvXfWG2bd8XXVfXVdnWhVWWdd/WfWWYdZ0wuq6uq7bs66os676u68Yw67owrLpt/K6tC8Or68ax676u3L6Patu+8Oq2Mby6bhy7sBu/7fvGsamqbZuuq+umK+u6bOu+b+u6cYyuq+uqLPu66sq+b+u68Ou+Lwyj6+q6Ksu6sNqyr8u6Lgy7rhvDatvC7tq6cMyyLgy37yvHrwtD1baF4dV1o6vbxm8Lw9I3dr4AAIABBwCAABPKQKEhKwKAOAEABiEIFWMQKsYghBBSCiGkVDEGIWMOSsYclBBKSSGU0irGIGSOScgckxBKaKmU0EoopaVQSkuhlNZSai2m1FoMobQUSmmtlNJaaim21FJsFWMQMuekZI5JKKW0VkppKXNMSsagpA5CKqWk0kpJrWXOScmgo9I5SKmk0lJJqbVQSmuhlNZKSrGl0kptrcUaSmktpNJaSam11FJtrbVaI8YgZIxByZyTUkpJqZTSWuaclA46KpmDkkopqZWSUqyYk9JBKCWDjEpJpbWSSiuhlNZKSrGFUlprrdWYUks1lJJaSanFUEprrbUaUys1hVBSC6W0FkpprbVWa2ottlBCa6GkFksqMbUWY22txRhKaa2kElspqcUWW42ttVhTSzWWkmJsrdXYSi051lprSi3W0lKMrbWYW0y5xVhrDSW0FkpprZTSWkqtxdZaraGU1koqsZWSWmyt1dhajDWU0mIpKbWQSmyttVhbbDWmlmJssdVYUosxxlhzS7XVlFqLrbVYSys1xhhrbjXlUgAAwIADAECACWWg0JCVAEAUAABgDGOMQWgUcsw5KY1SzjknJXMOQggpZc5BCCGlzjkIpbTUOQehlJRCKSmlFFsoJaXWWiwAAKDAAQAgwAZNicUBCg1ZCQBEAQAgxijFGITGIKUYg9AYoxRjECqlGHMOQqUUY85ByBhzzkEpGWPOQSclhBBCKaWEEEIopZQCAAAKHAAAAmzQlFgcoNCQFQFAFAAAYAxiDDGGIHRSOikRhExKJ6WREloLKWWWSoolxsxaia3E2EgJrYXWMmslxtJiRq3EWGIqAADswAEA7MBCKDRkJQCQBwBAGKMUY845ZxBizDkIITQIMeYchBAqxpxzDkIIFWPOOQchhM455yCEEELnnHMQQgihgxBCCKWU0kEIIYRSSukghBBCKaV0EEIIoZRSCgAAKnAAAAiwUWRzgpGgQkNWAgB5AACAMUo5JyWlRinGIKQUW6MUYxBSaq1iDEJKrcVYMQYhpdZi7CCk1FqMtXYQUmotxlpDSq3FWGvOIaXWYqw119RajLXm3HtqLcZac865AADcBQcAsAMbRTYnGAkqNGQlAJAHAEAgpBRjjDmHlGKMMeecQ0oxxphzzinGGHPOOecUY4w555xzjDHnnHPOOcaYc84555xzzjnnoIOQOeecc9BB6JxzzjkIIXTOOecchBAKAAAqcAAACLBRZHOCkaBCQ1YCAOEAAIAxlFJKKaWUUkqoo5RSSimllFICIaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKaWUUkoppZRSSimllFJKKZVSSimllFJKKaWUUkoppQAg3woHAP8HG2dYSTorHA0uNGQlABAOAAAYwxiEjDknJaWGMQildE5KSSU1jEEopXMSUkopg9BaaqWk0lJKGYSUYgshlZRaCqW0VmspqbWUUigpxRpLSqml1jLnJKSSWkuttpg5B6Wk1lpqrcUQQkqxtdZSa7F1UlJJrbXWWm0tpJRaay3G1mJsJaWWWmupxdZaTKm1FltLLcbWYkutxdhiizHGGgsA4G5wAIBIsHGGlaSzwtHgQkNWAgAhAQAEMko555yDEEIIIVKKMeeggxBCCCFESjHmnIMQQgghhIwx5yCEEEIIoZSQMeYchBBCCCGEUjrnIIRQSgmllFJK5xyEEEIIpZRSSgkhhBBCKKWUUkopIYQQSimllFJKKSWEEEIopZRSSimlhBBCKKWUUkoppZQQQiillFJKKaWUEkIIoZRSSimllFJCCKWUUkoppZRSSighhFJKKaWUUkoJJZRSSimllFJKKSGUUkoppZRSSimlAACAAwcAgAAj6CSjyiJsNOHCAxAAAAACAAJMAIEBgoJRCAKEEQgAAAAAAAgA+AAASAqAiIho5gwOEBIUFhgaHB4gIiQAAAAAAAAAAAAAAAAET2dnUwAEM4EAAAAAAABsbAAAAgAAAP+KmWqDLgEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQEBAQGGDY8GQaXSYC4suK7ruq7ruq7ruq7ruq7ruq7ruq7ruq77+biui33f7+7u7goAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAAA==";
stubAudio.lastPageAudioEventTime = Date.now(); // keep track of last time page audio played.
stubAudio.lastMediaSessionEventTime = Date.now(); // keep track of last media session interaction.
stubAudio.lastPlayed = false; // set played last to false.
// Set up a MutationObserver to monitor DOM changes for audio playback.
let observer = new MutationObserver(mutations => {
mutations.forEach(mutation => {
// Process new audio elements added to the DOM
mutation.addedNodes.forEach(node => {
if (node.nodeName === 'AUDIO') {
// Attach a custom play event listener to all new audio elements
addStateChangeEventListenerstoPageAudio(node, stubAudio);
}
});
});
});
// Start observing the document body for DOM changes
observer.observe(document.body, { attributes: true, childList: true, subtree: true });
// Attach a custom play event listener to all existing audio elements
document.querySelectorAll('audio').forEach(node => {
addStateChangeEventListenerstoPageAudio(node, stubAudio);
});
// Periodically updates media metadata at 1s interval (avoids flickering).
setInterval(updateMetadata, 1000);
// Periodically updates stub audio state at 100ms intervals (to capture play/pause changes).
setInterval(() => updateStubAudioState(stubAudio), 100);
// Sets up media session event handlers
setupMediaSessionEventHandlers(stubAudio);
}
// This self-invoking function ensures that the initialization process starts at the right time in the document's
// loading phase.
(function () {
/**
* Checks if the document body is already available. If it is, it means the DOM is sufficiently loaded to run the
* initialize function. If the document body isn't available yet, it adds an event listener for the
* 'DOMContentLoaded' event. This event fires when the initial HTML document has been completely loaded and parsed,
* without waiting for stylesheets, images, and subframes to finish loading.
*
* The initialize function is then executed either immediately (if the document body is available) or after the
* 'DOMContentLoaded' event fires, ensuring that the initialization logic runs at the appropriate time.
*/
document.body ? initialize() : document.addEventListener('DOMContentLoaded', initialize);
})();