diff --git a/.vscode/settings.json b/.vscode/settings.json index 4cb9716..e8b18e1 100644 --- a/.vscode/settings.json +++ b/.vscode/settings.json @@ -5,8 +5,9 @@ "ckeditor", "easings", "linebreak", - "songplayer" + "musicplayer" ], "javascript.preferences.importModuleSpecifierEnding": "js", - "html.format.wrapLineLength": 0 + "html.format.wrapLineLength": 0, + "liveServer.settings.port": 5501 } \ No newline at end of file diff --git a/package-lock.json b/package-lock.json index dc46558..e84f726 100644 --- a/package-lock.json +++ b/package-lock.json @@ -25,6 +25,7 @@ "mocha-junit-reporter": "^2.0.2", "nodemon": "^2.0.15", "style-loader": "^3.3.1", + "sync-directory": "^4.0.12", "webpack": "^5.72.0", "webpack-cli": "^4.9.2", "webpack-merge": "^5.8.0" @@ -2880,6 +2881,12 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=", + "dev": true + }, "node_modules/callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", @@ -3507,6 +3514,12 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/es6-promise": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", + "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", + "dev": true + }, "node_modules/escalade": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", @@ -4049,6 +4062,20 @@ "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", "dev": true }, + "node_modules/fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "dev": true, + "dependencies": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + }, + "engines": { + "node": ">=6 <7 || >=8" + } + }, "node_modules/fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -4497,6 +4524,19 @@ "node": ">= 0.10" } }, + "node_modules/is-absolute": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-absolute/-/is-absolute-1.0.0.tgz", + "integrity": "sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==", + "dev": true, + "dependencies": { + "is-relative": "^1.0.0", + "is-windows": "^1.0.1" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-bigint": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", @@ -4734,6 +4774,18 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/is-relative": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-relative/-/is-relative-1.0.0.tgz", + "integrity": "sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==", + "dev": true, + "dependencies": { + "is-unc-path": "^1.0.0" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-shared-array-buffer": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", @@ -4794,6 +4846,18 @@ "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", "dev": true }, + "node_modules/is-unc-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-unc-path/-/is-unc-path-1.0.0.tgz", + "integrity": "sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==", + "dev": true, + "dependencies": { + "unc-path-regex": "^0.1.2" + }, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-unicode-supported": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", @@ -4818,6 +4882,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/is-yarn-global": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", @@ -6116,6 +6189,23 @@ "node": ">=0.10.0" } }, + "node_modules/readdir-enhanced": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/readdir-enhanced/-/readdir-enhanced-1.5.2.tgz", + "integrity": "sha1-YUYwSGkKxqRVt1ti+nioj43IXlM=", + "dev": true, + "dependencies": { + "call-me-maybe": "^1.0.1", + "es6-promise": "^4.1.0", + "glob-to-regexp": "^0.3.0" + } + }, + "node_modules/readdir-enhanced/node_modules/glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=", + "dev": true + }, "node_modules/readdirp": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", @@ -6597,6 +6687,31 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/sync-directory": { + "version": "4.0.12", + "resolved": "https://registry.npmjs.org/sync-directory/-/sync-directory-4.0.12.tgz", + "integrity": "sha512-ZpBeMXL60flA1jbuhNJoCE6EjzInjTuoh6f7cArruYwQ9V8DEZNzN9FI3Q9CkWz4VzF4IvcKG38xLO1rZaysow==", + "dev": true, + "dependencies": { + "chokidar": "^3.3.1", + "commander": "^6.2.0", + "fs-extra": "^7.0.1", + "is-absolute": "^1.0.0", + "readdir-enhanced": "^1.5.2" + }, + "bin": { + "syncdir": "cmd.js" + } + }, + "node_modules/sync-directory/node_modules/commander": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-6.2.1.tgz", + "integrity": "sha512-U7VdrJFnJgo4xjrHpTzu0yrHPGImdsmD95ZlgYSEajAn2JKzDhDTPG9kBTefmObL2w/ngeZnilk+OV9CG3d7UA==", + "dev": true, + "engines": { + "node": ">= 6" + } + }, "node_modules/tapable": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.1.tgz", @@ -6781,6 +6896,15 @@ "url": "https://github.com/sponsors/ljharb" } }, + "node_modules/unc-path-regex": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/unc-path-regex/-/unc-path-regex-0.1.2.tgz", + "integrity": "sha1-5z3T17DXxe2G+6xrCufYxqadUPo=", + "dev": true, + "engines": { + "node": ">=0.10.0" + } + }, "node_modules/undefsafe": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", @@ -9403,6 +9527,12 @@ "get-intrinsic": "^1.0.2" } }, + "call-me-maybe": { + "version": "1.0.1", + "resolved": "https://registry.npmjs.org/call-me-maybe/-/call-me-maybe-1.0.1.tgz", + "integrity": "sha1-JtII6onje1y95gJQoV8DHBak1ms=", + "dev": true + }, "callsites": { "version": "3.1.0", "resolved": "https://registry.npmjs.org/callsites/-/callsites-3.1.0.tgz", @@ -9870,6 +10000,12 @@ "is-symbol": "^1.0.2" } }, + "es6-promise": { + "version": "4.2.8", + "resolved": "https://registry.npmjs.org/es6-promise/-/es6-promise-4.2.8.tgz", + "integrity": "sha512-HJDGx5daxeIvxdBxvG2cb9g4tEvwIk3i8+nhX0yGrYmZUzbkdg8QbDevheDB8gd0//uPj4c1EQua8Q+MViT0/w==", + "dev": true + }, "escalade": { "version": "3.1.1", "resolved": "https://registry.npmjs.org/escalade/-/escalade-3.1.1.tgz", @@ -10291,6 +10427,17 @@ "integrity": "sha512-WIWGi2L3DyTUvUrwRKgGi9TwxQMUEqPOPQBVi71R96jZXJdFskXEmf54BoZaS1kknGODoIGASGEzBUYdyMCBJg==", "dev": true }, + "fs-extra": { + "version": "7.0.1", + "resolved": "https://registry.npmjs.org/fs-extra/-/fs-extra-7.0.1.tgz", + "integrity": "sha512-YJDaCJZEnBmcbw13fvdAM9AwNOJwOzrE4pqMqBq5nFiEqXUqHwlK4B+3pUw6JNvfSPtX05xFHtYy/1ni01eGCw==", + "dev": true, + "requires": { + "graceful-fs": "^4.1.2", + "jsonfile": "^4.0.0", + "universalify": "^0.1.0" + } + }, "fs.realpath": { "version": "1.0.0", "resolved": "https://registry.npmjs.org/fs.realpath/-/fs.realpath-1.0.0.tgz", @@ -10609,6 +10756,16 @@ "integrity": "sha512-Ju0Bz/cEia55xDwUWEa8+olFpCiQoypjnQySseKtmjNrnps3P+xfpUmGr90T7yjlVJmOtybRvPXhKMbHr+fWnw==", "dev": true }, + "is-absolute": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-absolute/-/is-absolute-1.0.0.tgz", + "integrity": "sha512-dOWoqflvcydARa360Gvv18DZ/gRuHKi2NU/wU5X1ZFzdYfH29nkiNZsF3mp4OJ3H4yo9Mx8A/uAGNzpzPN3yBA==", + "dev": true, + "requires": { + "is-relative": "^1.0.0", + "is-windows": "^1.0.1" + } + }, "is-bigint": { "version": "1.0.4", "resolved": "https://registry.npmjs.org/is-bigint/-/is-bigint-1.0.4.tgz", @@ -10765,6 +10922,15 @@ "has-tostringtag": "^1.0.0" } }, + "is-relative": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-relative/-/is-relative-1.0.0.tgz", + "integrity": "sha512-Kw/ReK0iqwKeu0MITLFuj0jbPAmEiOsIwyIXvvbfa6QfmN9pkD1M+8pdk7Rl/dTKbH34/XBFMbgD4iMJhLQbGA==", + "dev": true, + "requires": { + "is-unc-path": "^1.0.0" + } + }, "is-shared-array-buffer": { "version": "1.0.2", "resolved": "https://registry.npmjs.org/is-shared-array-buffer/-/is-shared-array-buffer-1.0.2.tgz", @@ -10804,6 +10970,15 @@ "integrity": "sha1-5HnICFjfDBsR3dppQPlgEfzaSpo=", "dev": true }, + "is-unc-path": { + "version": "1.0.0", + "resolved": "https://registry.npmjs.org/is-unc-path/-/is-unc-path-1.0.0.tgz", + "integrity": "sha512-mrGpVd0fs7WWLfVsStvgF6iEJnbjDFZh9/emhRDcGWTduTfNHd9CHeUwH3gYIjdbwo4On6hunkztwOaAw0yllQ==", + "dev": true, + "requires": { + "unc-path-regex": "^0.1.2" + } + }, "is-unicode-supported": { "version": "0.1.0", "resolved": "https://registry.npmjs.org/is-unicode-supported/-/is-unicode-supported-0.1.0.tgz", @@ -10819,6 +10994,12 @@ "call-bind": "^1.0.2" } }, + "is-windows": { + "version": "1.0.2", + "resolved": "https://registry.npmjs.org/is-windows/-/is-windows-1.0.2.tgz", + "integrity": "sha512-eXK1UInq2bPmjyX6e3VHIzMLobc4J94i4AWn+Hpq3OU5KkrRC96OAcR3PRJ/pGu6m8TRnBHP9dkXQVsT/COVIA==", + "dev": true + }, "is-yarn-global": { "version": "0.3.0", "resolved": "https://registry.npmjs.org/is-yarn-global/-/is-yarn-global-0.3.0.tgz", @@ -11795,6 +11976,25 @@ } } }, + "readdir-enhanced": { + "version": "1.5.2", + "resolved": "https://registry.npmjs.org/readdir-enhanced/-/readdir-enhanced-1.5.2.tgz", + "integrity": "sha1-YUYwSGkKxqRVt1ti+nioj43IXlM=", + "dev": true, + "requires": { + "call-me-maybe": "^1.0.1", + "es6-promise": "^4.1.0", + "glob-to-regexp": "^0.3.0" + }, + "dependencies": { + "glob-to-regexp": { + "version": "0.3.0", + "resolved": "https://registry.npmjs.org/glob-to-regexp/-/glob-to-regexp-0.3.0.tgz", + "integrity": "sha1-jFoUlNIGbFcMw7/kSWF1rMTVAqs=", + "dev": true + } + } + }, "readdirp": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz", @@ -12142,6 +12342,27 @@ "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==", "dev": true }, + "sync-directory": { + "version": "4.0.12", + "resolved": "https://registry.npmjs.org/sync-directory/-/sync-directory-4.0.12.tgz", + "integrity": "sha512-ZpBeMXL60flA1jbuhNJoCE6EjzInjTuoh6f7cArruYwQ9V8DEZNzN9FI3Q9CkWz4VzF4IvcKG38xLO1rZaysow==", + "dev": true, + "requires": { + "chokidar": "^3.3.1", + "commander": "^6.2.0", + "fs-extra": "^7.0.1", + "is-absolute": "^1.0.0", + "readdir-enhanced": "^1.5.2" + }, + "dependencies": { + "commander": { + "version": "6.2.1", + "resolved": "https://registry.npmjs.org/commander/-/commander-6.2.1.tgz", + "integrity": "sha512-U7VdrJFnJgo4xjrHpTzu0yrHPGImdsmD95ZlgYSEajAn2JKzDhDTPG9kBTefmObL2w/ngeZnilk+OV9CG3d7UA==", + "dev": true + } + } + }, "tapable": { "version": "2.2.1", "resolved": "https://registry.npmjs.org/tapable/-/tapable-2.2.1.tgz", @@ -12271,6 +12492,12 @@ "which-boxed-primitive": "^1.0.2" } }, + "unc-path-regex": { + "version": "0.1.2", + "resolved": "https://registry.npmjs.org/unc-path-regex/-/unc-path-regex-0.1.2.tgz", + "integrity": "sha1-5z3T17DXxe2G+6xrCufYxqadUPo=", + "dev": true + }, "undefsafe": { "version": "2.0.5", "resolved": "https://registry.npmjs.org/undefsafe/-/undefsafe-2.0.5.tgz", diff --git a/package.json b/package.json index b9b3614..75624c9 100644 --- a/package.json +++ b/package.json @@ -10,12 +10,12 @@ "build:dev": "webpack --config webpack.dev.cjs", "test:junit": "mocha tests/** --reporter mocha-junit-reporter --reporter-options mochaFile=./junit/test_results.xml", "test:console": "mocha tests/**", - "watch:docs": "nodemon --watch dist/ --watch tutorials/ --watch src/ --exec 'npm run docs' -e js,mjs,json,htm,html,xml,markdown,md,xhtml", + "watch:docs": "nodemon --watch dist/ --watch tutorials/ --watch src/ --exec 'npm run docs' -e js,mjs,json,htm,html,xml,markdown,md,xhtml --ignore ./tutorials/assets/js/audioshowkit.js", "watch:lib": "webpack --watch --config webpack.dev.cjs", "build": "npm run build:dev", "test": "npm run test:console", "watch": "npm run watch:lib", - "docs": "jsdoc -c jsdoc.config.json" + "docs": "jsdoc -c jsdoc.config.json && cp ./dist/audioshowkit.js ./tutorials/assets/js/audioshowkit.js && syncdir ./tutorials/assets/ ./docs/assets/" }, "author": "", "license": "ISC", @@ -36,6 +36,7 @@ "mocha-junit-reporter": "^2.0.2", "nodemon": "^2.0.15", "style-loader": "^3.3.1", + "sync-directory": "^4.0.12", "webpack": "^5.72.0", "webpack-cli": "^4.9.2", "webpack-merge": "^5.8.0" diff --git a/src/mapping/numeric.js b/src/mapping/numeric.js index 0dd72ac..338b3bf 100644 --- a/src/mapping/numeric.js +++ b/src/mapping/numeric.js @@ -1,6 +1,4 @@ -// TODO: add for width -// TODO: add for height -// TODO: add for background +import VisualizerUpdateManager from "../visualization/VisualizerUpdateManager.js"; /** * @@ -13,6 +11,8 @@ * @param {number} value The new numerical value to update to. */ +// FIXME: Warnings about interpolator not being defined are broken. Docs generate fine. + /** * Maps the average of a range of bins to numerical value defined by a getter and a setter. * @@ -42,7 +42,7 @@ function mapRangedAvgNumerical({ minVal, maxVal, lowerBin = undefined, upperBin, } average /= normalBins.length; const range = maxVal - minVal; - let interpolated = interpolator(getter(), average, timeDelta); // TODO: May be optimized by allowing for transferring of previous style state. + let interpolated = interpolator(getter(), average, timeDelta); if (reversed) interpolated = 1 - interpolated; setter(minVal + range * interpolated); } diff --git a/src/player/PlaylistSong.js b/src/player/Music.js similarity index 74% rename from src/player/PlaylistSong.js rename to src/player/Music.js index 0269b05..48b810c 100644 --- a/src/player/PlaylistSong.js +++ b/src/player/Music.js @@ -8,10 +8,10 @@ import Visualizer from "../visualization/Visualizer.js"; */ /** - * A song with metadata that can be used as part of a {@link SongPlaylist}. + * A music with metadata that can be used as part of a {@link MusicPlaylist}. * */ -export default class PlayListSong { +export default class Music { #displayName; #author; #url; @@ -20,11 +20,11 @@ export default class PlayListSong { #ready = false; /** - * Constructs a song for a {@link SongPlaylist}. + * Constructs a music for a {@link MusicPlaylist}. * - * @param {string} url the url to fetch the song from. - * @param {string} name the name of the song. - * @param {string} author the author of the song. + * @param {string} url the url to fetch the music from. + * @param {string} name the name of the music. + * @param {string} author the author of the music. */ constructor(url, name, author) { this.#displayName = name; @@ -35,14 +35,12 @@ export default class PlayListSong { /** * - * @param {AudioEventCallback} [onReady] called when the song is ready, including right away if the song is already ready. - * @returns {HTMLAudioElement} The audio element that represents this song. + * @param {AudioEventCallback} [onReady] called when the music is ready, including right away if the music is already ready. + * @returns {HTMLAudioElement} The audio element that represents this music. */ fetchAudio(onReady) { - console.log("Fetching audio..."); if (this.#audio) { if (this.#ready) { - console.log("Already ready."); if (onReady) onReady(this.#audio); } else if (onReady) { this.#audio.addEventListener("canplaythrough", () => { @@ -52,35 +50,31 @@ export default class PlayListSong { return this.#audio; } this.#audio = new Audio(this.#url); - console.log("Fetching from url: " + this.#url); this.#audio.addEventListener("canplaythrough", () => { this.#ready = true; - console.log("attempting to invoke onReady."); - console.log(onReady); if (onReady && this.isAudioInstantiated()) { onReady(this.#audio); - console.log("onReady invoked."); } }, { once: true }); return this.#audio; } /** - * @returns {string} The name of the song to be displayed. + * @returns {string} The name of the music to be displayed. */ get displayName() { return this.#displayName; } /** - * @returns {string} The author of the song. + * @returns {string} The author of the music. */ get author() { return this.#author; } /** - * @returns {string} The url at which the file for this song can be found. + * @returns {string} The url at which the file for this music can be found. */ get url() { return this.#url; @@ -96,7 +90,7 @@ export default class PlayListSong { /** * Begins audio playback as soon as possible. * - * This function uses the {@link PlaylistSong#getAudio} function and specifically, the {@link AudioEventCallback}. + * This function uses the {@link PlaylistMusic#getAudio} function and specifically, the {@link AudioEventCallback}. */ play() { this.fetchAudio((audio) => { @@ -131,7 +125,7 @@ export default class PlayListSong { } /** * - * @returns {number} The number of seconds into the song. + * @returns {number} The number of seconds into the music. */ get currentTime() { return this.fetchAudio().currentTime; @@ -139,7 +133,7 @@ export default class PlayListSong { /** * - * The time position in the song to jump to in seconds. + * The time position in the music to jump to in seconds. */ set currentTime(currentTime) { this.fetchAudio().currentTime = currentTime; @@ -147,7 +141,7 @@ export default class PlayListSong { /** * - * @returns {number} The duration of the song. + * @returns {number} The duration of the music. */ get duration() { return this.fetchAudio().duration; @@ -157,7 +151,7 @@ export default class PlayListSong { * Unloads the audio data. * Makes sure the audio is paused. * - * Also calls {@link PlaylistSong#unloadVisualizer}. + * Also calls {@link PlaylistMusic#unloadVisualizer}. */ unloadAudio() { if (!this.isAudioInstantiated()) return; diff --git a/src/player/SongPlayer.js b/src/player/MusicPlayer.js similarity index 66% rename from src/player/SongPlayer.js rename to src/player/MusicPlayer.js index c5dc1b5..b1fa02a 100644 --- a/src/player/SongPlayer.js +++ b/src/player/MusicPlayer.js @@ -1,6 +1,5 @@ -import "../styles/songPlayer.css"; -import PlayListSong from "./PlaylistSong.js"; -import SongPlaylist from "./SongPlaylist.js"; +import "../styles/musicPlayer.css"; +import MusicPlaylist from "./MusicPlaylist.js"; /**@module */ @@ -11,22 +10,20 @@ import SongPlaylist from "./SongPlaylist.js"; */ /** - * A player that keeps track of global properties for playback as well as traversing a playlist. Additionally keeps track of song audio data and attempts to reduce memory usage when possible. + * A player that keeps track of global properties for playback as well as traversing a playlist. Additionally keeps track of music audio data and attempts to reduce memory usage when possible. */ -export default class SongPlayer { +export default class MusicPlayer { #playlist; - #current = 0; #volume = 1; #autoplay = false; #wasPlaying = false; #playlistChangeListeners = []; - #currentSongChangeListeners = []; #volumeChangeListeners = []; #allAudioEventListeners = {}; /** * - * @param {SongPlaylist} playlist the playlist of songs that this player is in charge of. + * @param {MusicPlaylist} playlist the playlist of musics that this player is in charge of. */ constructor(playlist) { this.playlist = playlist; @@ -36,15 +33,15 @@ export default class SongPlayer { } /** - * The new playlist of songs that this player is in charge of. + * The new playlist of musics that this player is in charge of. */ set playlist(playlist) { - this.#playlist?.unloadAllAudio(); + if (this.playlist) this.unloadAllAudio(); + if (this.playlist) this.playlist.removePositionChangeListener(this.#onMusicChange); const old = this.#playlist; this.#playlist = playlist; - if (!this.changeCurrentSongIndex(0)) { - throw new Error("The provided playlist has no songs."); - } + this.#playlist.addPositionChangeListener(this.#onMusicChange); + this.#playlist.currentPosition = 0; // Updates the listener. this.#playlistChangeListeners.forEach(playlistChangeListener => { playlistChangeListener(old, this.getPlaylist()); }); @@ -52,9 +49,9 @@ export default class SongPlayer { } /** - * The current playlist of songs that this player is in charge of. + * The current playlist of musics that this player is in charge of. * - * @returns {SongPlaylist} The song playlist this player is currently using. + * @returns {MusicPlaylist} The music playlist this player is currently using. */ get playlist() { return this.#playlist; @@ -62,65 +59,51 @@ export default class SongPlayer { /** * - * @returns {boolean} true if and only if successful in going to the next song. + * @returns {boolean} true if and only if successful in going to the next music. */ next() { - return this.changeCurrentSongIndex(this.currentSongIndex + 1); + return this.playlist.currentPosition += 1; } /** - * attempts to go to the previous song. + * attempts to go to the previous music. * - * @returns {boolean} true if and only if successful in going to the previous song. + * @returns {boolean} true if and only if successful in going to the previous music. */ previous() { - return this.changeCurrentSongIndex(this.currentSongIndex - 1); + return this.playlist.currentPosition -= 1; } /** - * - * @param {number} index the index of the song to jump to. + * @param {number} old the index of the previous music. + * @param {number} current the index of the current music. * @returns {boolean} true if and only if successful jumping to the given index. */ - changeCurrentSongIndex(index) { - console.log("Changing current song to " + index); - if (index >= this.#playlist.total()) return false; - if (index < 0) return false; + #onMusicChange = (old, current) => { // Anonymous to avoid overriding "this". + const oldMusic = this.playlist.musicAtIndex(old); + const currentMusic = this.playlist.musicAtIndex(current); Object.keys(this.#allAudioEventListeners).forEach(key => { const listeners = this.#allAudioEventListeners[key]; listeners.forEach(listener => { - this.currentSong.fetchAudio().removeEventListener(key, listener); + oldMusic.fetchAudio().removeEventListener(key, listener); }); }); - this.currentSong.unloadAudio(); - const old = this.currentSong; - this.#current = index; - this.#currentSongChangeListeners.forEach(currentChangeListener => { - currentChangeListener(old, this.currentSong); - }); + oldMusic.unloadAudio(); + Object.keys(this.#allAudioEventListeners).forEach(key => { const listeners = this.#allAudioEventListeners[key]; listeners.forEach(listener => { - this.currentSong.fetchAudio().addEventListener(key, listener); + currentMusic.fetchAudio().addEventListener(key, listener); }); }); if (this.#wasPlaying || this.#autoplay) { this.playCurrent(); } return true; - } - - /** - * - * @returns {number} The current song's index in the playlist. - */ - get currentSongIndex() { - return this.#current; - } + }; playCurrent() { - console.log("playing " + this.#current); - this.currentSong.fetchAudio((audio) => { + this.playlist.currentMusic.fetchAudio((audio) => { audio.volume = this.volume; audio.play(); }); @@ -128,15 +111,14 @@ export default class SongPlayer { } /** - * Pauses the current playing song (if there is one playing). + * Pauses the current playing music (if there is one playing). */ pauseCurrent() { - console.log("Pausing."); - this.currentSong.fetchAudio().pause(); + this.playlist.currentMusic.fetchAudio().pause(); } /** - * Toggles whether or not the current song is playing. + * Toggles whether or not the current music is playing. */ togglePlay() { if (this.playing) { @@ -153,7 +135,7 @@ export default class SongPlayer { if (volume > 1) volume = 1; if (volume < 0) volume = 0; this.#volume = volume; - this.currentSong.fetchAudio().volume = this.#volume; + this.playlist.currentMusic.fetchAudio().volume = this.#volume; } /** @@ -171,8 +153,8 @@ export default class SongPlayer { * @returns {boolean} true if and only if the position to seek to is within the duration of the track. This also means that if the track has not finished loading the duration data, than this will always return false. */ seek(position) { - if (position > this.currentSong.fetchAudio().duration || position < 0) return false; - this.currentSong.fetchAudio(audio => { + if (position > this.playlist.playlist.currentMusic.fetchAudio().duration || position < 0) return false; + this.playlist.playlist.currentMusic.fetchAudio(audio => { audio.currentTime = position; }); return true; @@ -183,15 +165,15 @@ export default class SongPlayer { * @returns {number} How many seconds into the audio track has been played in seconds. */ get currentPosition() { - return this.currentSong.fetchAudio().currentTime; + return this.playlist.playlist.currentMusic.fetchAudio().currentTime; } /** * - * @returns {number} The total length of the song, or NaN if this information has not loaded yet. + * @returns {number} The total length of the music, or NaN if this information has not loaded yet. */ get currentDuration() { - return this.currentSong.fetchAudio().duration; + return this.playlist.playlist.currentMusic.fetchAudio().duration; } /** @@ -208,7 +190,6 @@ export default class SongPlayer { if (this.playing) playButton.classList.add("pause"); playButton.addEventListener("click", () => { - console.log("Generated play button has been pressed."); this.togglePlay(); }); @@ -241,37 +222,6 @@ export default class SongPlayer { return previousButton; } - /** - * @returns {PlayListSong} The current playlist being used. - */ - get currentSong() { - return this.#playlist.songAtIndex(this.#current); - } - - /** - * - * @param {changeListener} listener the listener to receive the updates. - * @returns {boolean} true if and only if successfully added the listener. - */ - addCurrentSongChangeListener(listener) { - if (this.#currentSongChangeListeners.includes(listener)) return false; - this.#currentSongChangeListeners.push(listener); - return true; - } - - /** - * - * @param {changeListener} listener the song change listener to remove. - * @returns {boolean} true if and only if the song change listener given was successfully removed. - */ - removeCurrentSongChangeListener(listener) { - const removeIndex = this.#currentSongChangeListeners.indexOf(listener); - if (removeIndex < 0) return false; - - this.#currentSongChangeListeners.splice(removeIndex, 1); - return true; - } - /** * * @param {changeListener} listener the playlist change listener to add to the callback list. @@ -336,7 +286,7 @@ export default class SongPlayer { } if (typeListeners.includes(eventListener)) return false; typeListeners.push(eventListener); - this.currentSong.fetchAudio().addEventListener(type, eventListener); + this.playlist.currentMusic.fetchAudio().addEventListener(type, eventListener); return true; } @@ -352,14 +302,23 @@ export default class SongPlayer { const removeIndex = typeListeners.indexOf(eventListener); if (removeIndex < 0) return false; typeListeners.splice(removeIndex, 1); - this.currentSong.fetchAudio().removeEventListener(type, eventListener); + this.playlist.playlist.currentMusic.fetchAudio().removeEventListener(type, eventListener); return true; } /** - * @returns {boolean} If the player is playing any songs right now. + * @returns {boolean} If the player is playing any musics right now. */ get playing() { - return this.currentSong.isAudioInstantiated() && !this.currentSong.fetchAudio().paused; + return this.playlist.currentMusic.isAudioInstantiated() && !this.playlist.currentMusic.fetchAudio().paused; + } + + /** + * Unloads the audio data of all musics in this playlist. + */ + unloadAllAudio() { + this.playlist.forEach(music => { + music.unloadAudio(); + }); } } \ No newline at end of file diff --git a/src/player/MusicPlaylist.js b/src/player/MusicPlaylist.js new file mode 100644 index 0000000..6abe16e --- /dev/null +++ b/src/player/MusicPlaylist.js @@ -0,0 +1,250 @@ +import Music from "./Music.js"; + +/**@module */ + +/** + * @callback positionChangeListener + * @param {number} old the previous position. + * @param {number} current the the current (new) position. + */ + +/** + * @callback lengthChangeListener + * @param {number} changed The index of the music that has changed. + * @param {boolean} added True if and only if the change was caused by adding music. + * @param {Music} music The music that was added or removed. + */ + +/** + * A playlist that holds a multitude of musics in the form of {@link Music}. + */ +export default class MusicPlaylist { + #list = []; + #name; + #current = 0; + #positionChangeListeners = []; + #lengthChangeListeners = []; + + /** + * Instantiates a playlist for musics. + * + * @param {string} name The name of the playlist. + */ + constructor(name) { + this.#name = name; + } + + /** + * @returns {string} The name of the playlist. + */ + get name() { + return this.#name; + } + + /** + * + * @param {number} index the index of the music to retrieve. + * @returns {Music} the music at the given index. + */ + musicAtIndex(index) { + if (index >= this.total) { + return null; + } + return this.#list[index]; + } + + /** + * Automatically creates and adds a {@link Music} to this playlist. + * + * @param {string} url where the audio data can be found. + * @param {string} name the name of the music. + * @param {string} author the author(s) of the music. + */ + add(url, name, author) { + const music = new Music(url, name, author, this); + this.#list.push(music); + this.#onAdd(this.total - 1, music); + } + + /** + * Adds an existing {@link Music} object to this playlist. Alternatively, use {@link MusicPlaylist#add} to automatically instantiate a new one. + * + * @param {Music} music The existing music to add. + */ + addExisting(music) { + this.#list.push(music); + this.#onAdd(this.total - 1, music); + } + + /** + * Calls all playlist length change listeners. + * + * @param {number} index The index at which the new music was added. + * @param {Music} music The music that was added. + */ + #onAdd(index, music) { + this.#lengthChangeListeners.forEach(listener => { + listener(index, true, music); + }); + } + + /** + * Removes a {@link Music} from this playlist. + * + * @param {number} index the index of the music to be removed. + * @returns {Music} the music that was removed, or null if the index of was invalid. + */ + remove(index) { + if (index >= this.#list.length) { + return null; + } + let removed = this.#list.splice(index, 1)[0]; + this.#lengthChangeListeners.forEach(listener => { + listener(index, false, removed); + }); + if (removed.length > 0) { + return removed[0]; + } + } + + /** + * Attempts to find a {@link Music} given a name. Returns the first one found. + * + * @param {string} name the name of the music to be found. + * @returns {number} the index of the music found, or -1 if it was not found. + */ + findMusicIndex(name) { + return this.#list.findIndex((item) => item.getDisplayName() == name); + } + + /** + * The number of {@link Music} in this playlist. + * + * @returns {number} total number of musics in this playlist. + */ + get total() { + return this.#list.length; + } + + /** + * @returns {number} The current position in the playlist. + */ + get currentPosition() { + return this.#current; + } + + /** + * The current position in the playlist. + */ + set currentPosition(position) { + if (typeof position !== "number") throw new Error("Given position is invalid."); + if (position >= this.#list.length || position < 0) return; + const old = this.#current; + this.#current = position; + this.#positionChangeListeners.forEach(positionChangeListener => { + positionChangeListener(old, position); + }); + } + + /** + * @returns {Music} The current music in the playlist being used. + */ + get currentMusic() { + return this.musicAtIndex(this.currentPosition); + } + + /** + * + * @param {positionChangeListener} listener the listener to receive the updates. + * @returns {boolean} true if and only if successfully added the listener. + */ + addPositionChangeListener(listener) { + if (this.#positionChangeListeners.includes(listener)) return false; + this.#positionChangeListeners.push(listener); + return true; + } + + /** + * + * @param {positionChangeListener} listener the music change listener to remove. + * @returns {boolean} true if and only if the music change listener given was successfully removed. + */ + removePositionChangeListener(listener) { + const removeIndex = this.#positionChangeListeners.indexOf(listener); + if (removeIndex < 0) return false; + + this.#positionChangeListeners.splice(removeIndex, 1); + return true; + } + + /** + * Adds a listener for whether music was added or removed to the playlist. + * + * @param {lengthChangeListener} listener The listener to be added. + * @returns {boolean} True if and only if the listener was successfully added. + */ + addLengthChangeListener(listener) { + if (this.#lengthChangeListeners.includes(listener)) return false; + this.#lengthChangeListeners.push(listener); + return true; + } + + /** + * Removes a listener for whether music is added or removed to the playlist. + * + * @param {lengthChangeListener} listener The listener to be removed. + * @returns {boolean} True if and only if the provided listener was successfully removed. + */ + removeLengthChangeListener(listener) { + const removeIndex = this.#lengthChangeListeners.indexOf(listener); + if (removeIndex < 0) return false; + this.#lengthChangeListeners.splice(removeIndex, 1); + return true; + } + + generatePlaylistElement() { + const element = document.createElement("table"); + element.createTHead().textContent = "Playlist"; + const headers = element.insertRow(); + const musicHeader = document.createElement("th"); + musicHeader.textContent = "Music"; + headers.appendChild(musicHeader); + const authorHeader = document.createElement("th"); + authorHeader.textContent = "Author"; + headers.appendChild(authorHeader); + + const insertMusic = (music, musicPos = undefined) => { + if (musicPos) musicPos++; + const row = element.insertRow(musicPos); + row.insertCell().textContent = music.displayName; + row.insertCell().textContent = music.author; + }; + for (const music of this) { + insertMusic(music); + } + + this.addLengthChangeListener((changed, added, music) => { + if (added) { + insertMusic(music, changed); + } else { + element.deleteRow(changed); + } + }); + + element.rows[this.currentPosition + 1].classList.add("active"); + + this.addPositionChangeListener((old, current) => { + element.rows[old + 1].classList.remove("active"); + element.rows[current + 1].classList.add("active"); + }); + return element; + } + + [Symbol.iterator]() { + let index = -1; + const data = this.#list; + return { + next: () => ({ value: data[++index], done: index >= data.length }) // Brackets so not read as a body of anonymous function. + }; + } +} \ No newline at end of file diff --git a/src/player/SongPlaylist.js b/src/player/SongPlaylist.js deleted file mode 100644 index c99ea9f..0000000 --- a/src/player/SongPlaylist.js +++ /dev/null @@ -1,93 +0,0 @@ -import PlaylistSong from "./PlaylistSong.js"; - -/**@module */ - -/** - * A playlist that holds a multitude of songs in the form of {@link PlaylistSong}. - */ -export default class SongPlaylist { - #list = []; - #name; - - /** - * Instantiates a playlist for songs. - * - * @param {string} name The name of the playlist. - */ - constructor(name) { - this.#name = name; - } - - /** - * @returns {string} The name of the playlist. - */ - get name() { - return this.#name; - } - - /** - * - * @param {number} index the index of the song to retrieve. - * @returns {PlaylistSong} the song at the given index. - */ - songAtIndex(index) { - if (index >= this.#list.length) { - return null; - } - return this.#list[index]; - } - - /** - * Automatically creates and adds a {@see PlaylistSong} to this playlist. - * - * @param {string} url where the audio data can be found. - * @param {string} name the name of the song. - * @param {string} author the author(s) of the song. - */ - add(url, name, author) { - this.#list.push(new PlaylistSong(url, name, author, this)); - } - - /** - * Removes a {@see playlistSong} from this playlist. - * - * @param {number} index the index of the song to be removed. - * @returns {PlaylistSong} the song that was removed, or null if the index of was invalid. - */ - remove(index) { - if (index >= this.#list.length) { - return null; - } - let removed = this.#list.splice(index, 1)[0]; - if (removed.length > 0) { - return removed[0]; - } - } - - /** - * Attempts to find a {@link PlaylistSong} given a name. - * - * @param {string} name the name of the song to be found. - * @returns {number} the index of the song found, or -1 if it was not found. - */ - findSongIndex(name) { - return this.#list.findIndex((item) => item.getDisplayName() == name); - } - - /** - * - * @returns {number} total number of songs in this playlist. - */ - total() { - return this.#list.length; - } - - /** - * Unloads the audio data of all songs in this playlist. - */ - unloadAllAudio() { - this.#list.forEach(playlistSong => { - playlistSong.unloadAudio(); - }); - } -} \ No newline at end of file diff --git a/src/player/VisualizedSongPlayer.js b/src/player/VisualizedMusicPlayer.js similarity index 69% rename from src/player/VisualizedSongPlayer.js rename to src/player/VisualizedMusicPlayer.js index ee4040d..5104022 100644 --- a/src/player/VisualizedSongPlayer.js +++ b/src/player/VisualizedMusicPlayer.js @@ -1,30 +1,30 @@ import Visualizer from "../visualization/Visualizer.js"; import VisualizerUpdateManager from "../visualization/VisualizerUpdateManager.js"; -import SongPlayer from "./SongPlayer.js"; -import SongPlaylist from "./SongPlaylist.js"; +import MusicPlayer from "./MusicPlayer.js"; +import MusicPlaylist from "./MusicPlaylist.js"; /**@module */ /** - * A song player that provides easier access to the current songs visualizer data. - * Additionally, automatically re-binds all the visualizer update listeners for song changes. + * A music player that provides easier access to the current musics visualizer data. + * Additionally, automatically re-binds all the visualizer update listeners for music changes. * - * @augments SongPlayer + * @augments MusicPlayer */ -export default class VisualizedSongPlayer extends SongPlayer { +export default class VisualizedMusicPlayer extends MusicPlayer { #fftSize = 1024; #visualizerUpdateManager; /** - * Instantiates a song player with visualization features. + * Instantiates a music player with visualization features. * - * @param {SongPlaylist} playlist the playlist this player manages. + * @param {MusicPlaylist} playlist the playlist this player manages. * @param {number} [fftSize=1024] the size of the fft window for analysis. */ constructor(playlist, fftSize = 1024) { super(playlist); this.#fftSize = fftSize; - this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentSongVisualizer()); + this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentMusicVisualizer()); } /** @@ -32,44 +32,44 @@ export default class VisualizedSongPlayer extends SongPlayer { */ set playlist(playlist) { super.playlist = playlist; - this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentSongVisualizer()); + this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentMusicVisualizer()); } /** * - * @returns {boolean} true if and only if successful in changing to the next song. + * @returns {boolean} true if and only if successful in changing to the next music. */ next() { const updateListeners = this.#visualizerUpdateManager.getBinnedListeners(); if (!super.next()) return false; - this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentSongVisualizer()); + this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentMusicVisualizer()); this.#visualizerUpdateManager.setBinnedListeners(updateListeners); return true; } /** - * Jumps to the previous song if possible. + * Jumps to the previous music if possible. * - * @returns {boolean} true if and only if successful in switching to the previous song in the playlist. + * @returns {boolean} true if and only if successful in switching to the previous music in the playlist. */ previous() { const updateListeners = this.#visualizerUpdateManager.getBinnedListeners(); if (!super.previous()) return false; - this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentSongVisualizer()); + this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentMusicVisualizer()); this.#visualizerUpdateManager.setBinnedListeners(updateListeners); return true; } /** * - * @param {number} index the index of the song to change to. + * @param {number} index the index of the music to change to. * @returns {boolean} true if and only if successful in jumping to the given index. */ changeCurrent(index) { const updateListeners = this.VisualizerUpdateManager.getBinnedListeners(); if (!super.changeCurrent(index)) return false; - this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentSongVisualizer()); + this.#visualizerUpdateManager = new VisualizerUpdateManager(this.getCurrentMusicVisualizer()); this.#visualizerUpdateManager.setBinnedListeners = updateListeners(); return true; } @@ -82,9 +82,9 @@ export default class VisualizedSongPlayer extends SongPlayer { } /** - * @returns {Visualizer} The current song's visualizer. + * @returns {Visualizer} The current music's visualizer. */ - get currentSongVisualizer() { - return this.getCurrentSong().getVisualizer(); + get currentMusicVisualizer() { + return this.getCurrentMusic().getVisualizer(); } } \ No newline at end of file diff --git a/src/player/player.js b/src/player/player.js index 0f7cfb5..e97cfb5 100644 --- a/src/player/player.js +++ b/src/player/player.js @@ -1,7 +1,7 @@ -import SongPlayer from "./SongPlayer.js"; -import SongPlaylist from "./SongPlaylist.js"; -import VisualizedSongPlayer from "./VisualizedSongPlayer.js"; -import PlayListSong from "./PlaylistSong.js"; +import MusicPlayer from "./MusicPlayer.js"; +import MusicPlaylist from "./MusicPlaylist.js"; +import VisualizedMusicPlayer from "./VisualizedMusicPlayer.js"; +import Music from "./Music.js"; /**@module */ -export { SongPlayer, SongPlaylist, VisualizedSongPlayer, PlayListSong }; \ No newline at end of file +export { MusicPlayer, MusicPlaylist, VisualizedMusicPlayer, Music }; \ No newline at end of file diff --git a/src/styles/songPlayer.css b/src/styles/musicPlayer.css similarity index 100% rename from src/styles/songPlayer.css rename to src/styles/musicPlayer.css diff --git a/tutorials/MusicPlayer.html b/tutorials/MusicPlayer.html new file mode 100644 index 0000000..8eae4f2 --- /dev/null +++ b/tutorials/MusicPlayer.html @@ -0,0 +1,81 @@ + + + + + + + + + + +
+

Creating a Music Player

+

One of the most basic principles behind AudioShowKit is the {@link module:player/MusicPlayer}. The song player acts as an easy way for developers to set up a list of songs with simple generated controls while exposing more complex events if needed.

+
+
+

Instantiating a Music Player

+

Instantiating a music player requires a {@link module:player/MusicPlaylist}. See the [MusicPlaylist tutorial]{@tutorial MusicPlaylist} for information on how to get one of those! Once you have one, you can proceed with instantiating a Music Player.

+

+            const ask = window.audioshowkit; // Get a reference to the audioshowkit stuff.
+            const playlist = previousPlaylist; // We are assuming you have a playlist ready.
+            const player = new ask.player.MusicPlayer(playlist) // Creates a new music player with the playlist.
+        
+
+
+

Display a Play Button

+

We need some user interactive element to start playing music. You can set this up yourself, or use a generated one by us! We'll show you how to do the latter, but feel free to read the documentation on all the methods available that can be used in event listeners to see how to do the former.

+

+            // Still using the same context from the instantiation example...
+            const playElement = player.generatePlayElement();
+            document.getElementById("musicplayer-playbtn-demo").appendChild(playElement); // Assuming this element exists.
+        
+

The play button

+

Go ahead and hit the play button!

+
+
+
+
+ +
+
+

Similarly, Next and Previous Buttons

+

We can also traverse the playlist by the player via a previous and next button. These buttons can be generated as well as programmed.

+

+            // Still using the same context from the instantiation example...
+            const nextElement = player.generateNextElement();
+            document.getElementById("musicplayer-nextbtn-demo").appendChild(nextElement);
+
+            const prevElem = player.generatePreviousElement();
+            document.getElementById("musicplayer-prevbtn-demo").appendChild(prevElem);    
+        
+

The next button

+
+
+
+
+

The previous button

+
+
+
+
+
+ + + + \ No newline at end of file diff --git a/tutorials/MusicPlaylist.html b/tutorials/MusicPlaylist.html new file mode 100644 index 0000000..49dfb8d --- /dev/null +++ b/tutorials/MusicPlaylist.html @@ -0,0 +1,79 @@ + + + + + + + + + + +
+

Creating a Music Playlist

+

+ A music playlist is a list of {@link module:player/Music} which maintains metadata about the playlist, such as the length, and the name of the playlist. It also keeps track of an index for a position in the playlist. +

+
+
+

Instantiation

+

+            // Get the entry point to all of AudioShowKit's glory.
+            const ask = window.audioshowkit;
+
+            // The only parameter is the name of the playlist.
+            const playlist = new ask.player.MusicPlaylist("Awesome Music"); 
+        
+
+
+

Adding Music

+

A music playlist is no good if it doesn't have music. Lets add some.

+

+            // Still using the same context from the instantiation example...
+
+            playlist.add("/assets/audio/moments.mp3", "Moments", "Lost Identities x Robbie Rosen"); // It's that easy.
+            playlist.add("/assets/audio/XXI.mp3", "XXI", "QR"); // Let's add another one.
+        
+ +

If for whatever reason, you decided to instantiate your own {@link module:player/Music}, that's okay too!

+

+            // Still using the same context from the instantiation example...
+
+            // Either instantiate, or otherwise have this Music object.
+            const pathetiqueMusic = new ask.player.Music("/assets/audio/pathetique.mp3", "Sonata No. 8", "Beethoven");
+            playlist.addExisting(pathetiqueMusic); // Wow, a special function!
+        
+
+
+

Display

+

In the case you want to show the playlist without coding your own display, we provide a simple generator for you!

+

+            // Still using the same context from the instantiation example...
+
+            const playlistElem = playlist.generatePlaylistElement(); // Returns an HTMLElement!
+
+            // Assuming we do have an acceptable element that can take a child element.
+            document.getElementById("musicplaylist-demo").appendChild(playlistElem); 
+        
+
+

Result

+
+
+
+ + +
+ + + \ No newline at end of file diff --git a/tutorials/SongPlayer.html b/tutorials/SongPlayer.html deleted file mode 100644 index 3c059cc..0000000 --- a/tutorials/SongPlayer.html +++ /dev/null @@ -1,18 +0,0 @@ - - - - - - - - Creating a Song Player - - - -
-

Creating a Song Player

-

One of the most basic principles behind AudioShowKit is the {@link module:player/SongPlayer}. The song player acts as an easy way for developers to set up a list of songs and play them back with simple controls.

-
- - - \ No newline at end of file diff --git a/tutorials/SongPlaylist.html b/tutorials/SongPlaylist.html deleted file mode 100644 index 702886a..0000000 --- a/tutorials/SongPlaylist.html +++ /dev/null @@ -1,14 +0,0 @@ - - - - - - - - - - - - - - \ No newline at end of file diff --git a/tutorials/assets/audio/pathetique.mp3 b/tutorials/assets/audio/pathetique.mp3 new file mode 100644 index 0000000..fb11c6c Binary files /dev/null and b/tutorials/assets/audio/pathetique.mp3 differ diff --git a/tutorials/assets/css/prism.css b/tutorials/assets/css/prism.css new file mode 100644 index 0000000..f8300da --- /dev/null +++ b/tutorials/assets/css/prism.css @@ -0,0 +1,3 @@ +/* PrismJS 1.28.0 +https://prismjs.com/download.html#themes=prism-tomorrow&languages=markup+css+clike+javascript */ +code[class*=language-],pre[class*=language-]{color:#ccc;background:0 0;font-family:Consolas,Monaco,'Andale Mono','Ubuntu Mono',monospace;font-size:1em;text-align:left;white-space:pre;word-spacing:normal;word-break:normal;word-wrap:normal;line-height:1.5;-moz-tab-size:4;-o-tab-size:4;tab-size:4;-webkit-hyphens:none;-moz-hyphens:none;-ms-hyphens:none;hyphens:none}pre[class*=language-]{padding:1em;margin:.5em 0;overflow:auto}:not(pre)>code[class*=language-],pre[class*=language-]{background:#2d2d2d}:not(pre)>code[class*=language-]{padding:.1em;border-radius:.3em;white-space:normal}.token.block-comment,.token.cdata,.token.comment,.token.doctype,.token.prolog{color:#999}.token.punctuation{color:#ccc}.token.attr-name,.token.deleted,.token.namespace,.token.tag{color:#e2777a}.token.function-name{color:#6196cc}.token.boolean,.token.function,.token.number{color:#f08d49}.token.class-name,.token.constant,.token.property,.token.symbol{color:#f8c555}.token.atrule,.token.builtin,.token.important,.token.keyword,.token.selector{color:#cc99cd}.token.attr-value,.token.char,.token.regex,.token.string,.token.variable{color:#7ec699}.token.entity,.token.operator,.token.url{color:#67cdcc}.token.bold,.token.important{font-weight:700}.token.italic{font-style:italic}.token.entity{cursor:help}.token.inserted{color:green} diff --git a/tutorials/assets/js/audioshowkit.js b/tutorials/assets/js/audioshowkit.js new file mode 100644 index 0000000..fbf0354 --- /dev/null +++ b/tutorials/assets/js/audioshowkit.js @@ -0,0 +1,340 @@ +/* + * ATTENTION: An "eval-source-map" devtool has been used. + * This devtool is neither made for production nor for readable output files. + * It uses "eval()" calls to create a separate source file with attached SourceMaps in the browser devtools. + * If you are trying to read the output file, select a different devtool (https://webpack.js.org/configuration/devtool/) + * or disable the default devtool with "devtool: false". + * If you are looking for production-ready output files, see mode: "production" (https://webpack.js.org/configuration/mode/). + */ +var audioshowkit; +/******/ (() => { // webpackBootstrap +/******/ "use strict"; +/******/ var __webpack_modules__ = ({ + +/***/ "./node_modules/css-loader/dist/cjs.js!./src/styles/musicPlayer.css": +/*!**************************************************************************!*\ + !*** ./node_modules/css-loader/dist/cjs.js!./src/styles/musicPlayer.css ***! + \**************************************************************************/ +/***/ ((module, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../../node_modules/css-loader/dist/runtime/sourceMaps.js */ \"./node_modules/css-loader/dist/runtime/sourceMaps.js\");\n/* harmony import */ var _node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../../node_modules/css-loader/dist/runtime/api.js */ \"./node_modules/css-loader/dist/runtime/api.js\");\n/* harmony import */ var _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1__);\n// Imports\n\n\nvar ___CSS_LOADER_EXPORT___ = _node_modules_css_loader_dist_runtime_api_js__WEBPACK_IMPORTED_MODULE_1___default()((_node_modules_css_loader_dist_runtime_sourceMaps_js__WEBPACK_IMPORTED_MODULE_0___default()));\n// Module\n___CSS_LOADER_EXPORT___.push([module.id, \".player.play-btn {\\n box-sizing: border-box;\\n height: 48px;\\n width: 48;\\n transition: 80ms all ease;\\n will-change: border-width;\\n cursor: pointer;\\n border-color: transparent transparent transparent black;\\n\\n border-style: solid;\\n border-width: 24px 0px 24px 48px;\\n}\\n\\n.player.play-btn.pause {\\n border-style: double;\\n border-width: 0px 0px 0px 60px;\\n}\\n\\n\\n.player.next {\\n border-style: none;\\n width: 48px;\\n height: 48px;\\n}\\n\\n.player.previous {\\n border-style: none;\\n width: 48px;\\n height: 48px;\\n}\\n\\ntable.player {\\n border-collapse: collapse;\\n}\", \"\",{\"version\":3,\"sources\":[\"webpack://./src/styles/musicPlayer.css\"],\"names\":[],\"mappings\":\"AAAA;IACI,sBAAsB;IACtB,YAAY;IACZ,SAAS;IACT,yBAAyB;IACzB,yBAAyB;IACzB,eAAe;IACf,uDAAuD;;IAEvD,mBAAmB;IACnB,gCAAgC;AACpC;;AAEA;IACI,oBAAoB;IACpB,8BAA8B;AAClC;;;AAGA;IACI,kBAAkB;IAClB,WAAW;IACX,YAAY;AAChB;;AAEA;IACI,kBAAkB;IAClB,WAAW;IACX,YAAY;AAChB;;AAEA;IACI,yBAAyB;AAC7B\",\"sourcesContent\":[\".player.play-btn {\\n box-sizing: border-box;\\n height: 48px;\\n width: 48;\\n transition: 80ms all ease;\\n will-change: border-width;\\n cursor: pointer;\\n border-color: transparent transparent transparent black;\\n\\n border-style: solid;\\n border-width: 24px 0px 24px 48px;\\n}\\n\\n.player.play-btn.pause {\\n border-style: double;\\n border-width: 0px 0px 0px 60px;\\n}\\n\\n\\n.player.next {\\n border-style: none;\\n width: 48px;\\n height: 48px;\\n}\\n\\n.player.previous {\\n border-style: none;\\n width: 48px;\\n height: 48px;\\n}\\n\\ntable.player {\\n border-collapse: collapse;\\n}\"],\"sourceRoot\":\"\"}]);\n// Exports\n/* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (___CSS_LOADER_EXPORT___);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L2Nqcy5qcyEuL3NyYy9zdHlsZXMvbXVzaWNQbGF5ZXIuY3NzLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7O0FBQUE7QUFDNkc7QUFDakI7QUFDNUYsOEJBQThCLG1GQUEyQixDQUFDLDRGQUFxQztBQUMvRjtBQUNBLDREQUE0RCw2QkFBNkIsbUJBQW1CLGdCQUFnQixnQ0FBZ0MsZ0NBQWdDLHNCQUFzQiw4REFBOEQsNEJBQTRCLHVDQUF1QyxHQUFHLDRCQUE0QiwyQkFBMkIscUNBQXFDLEdBQUcsb0JBQW9CLHlCQUF5QixrQkFBa0IsbUJBQW1CLEdBQUcsc0JBQXNCLHlCQUF5QixrQkFBa0IsbUJBQW1CLEdBQUcsa0JBQWtCLGdDQUFnQyxHQUFHLE9BQU8sNkZBQTZGLFlBQVksV0FBVyxVQUFVLFlBQVksYUFBYSxXQUFXLGFBQWEsYUFBYSxhQUFhLE9BQU8sS0FBSyxZQUFZLGFBQWEsUUFBUSxLQUFLLFlBQVksV0FBVyxVQUFVLE9BQU8sS0FBSyxZQUFZLFdBQVcsVUFBVSxPQUFPLEtBQUssWUFBWSw0Q0FBNEMsNkJBQTZCLG1CQUFtQixnQkFBZ0IsZ0NBQWdDLGdDQUFnQyxzQkFBc0IsOERBQThELDRCQUE0Qix1Q0FBdUMsR0FBRyw0QkFBNEIsMkJBQTJCLHFDQUFxQyxHQUFHLG9CQUFvQix5QkFBeUIsa0JBQWtCLG1CQUFtQixHQUFHLHNCQUFzQix5QkFBeUIsa0JBQWtCLG1CQUFtQixHQUFHLGtCQUFrQixnQ0FBZ0MsR0FBRyxtQkFBbUI7QUFDdnBEO0FBQ0EsaUVBQWUsdUJBQXVCLEVBQUMiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9hdWRpb3Nob3draXQvLi9zcmMvc3R5bGVzL211c2ljUGxheWVyLmNzcz82MzUzIl0sInNvdXJjZXNDb250ZW50IjpbIi8vIEltcG9ydHNcbmltcG9ydCBfX19DU1NfTE9BREVSX0FQSV9TT1VSQ0VNQVBfSU1QT1JUX19fIGZyb20gXCIuLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvc291cmNlTWFwcy5qc1wiO1xuaW1wb3J0IF9fX0NTU19MT0FERVJfQVBJX0lNUE9SVF9fXyBmcm9tIFwiLi4vLi4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL2FwaS5qc1wiO1xudmFyIF9fX0NTU19MT0FERVJfRVhQT1JUX19fID0gX19fQ1NTX0xPQURFUl9BUElfSU1QT1JUX19fKF9fX0NTU19MT0FERVJfQVBJX1NPVVJDRU1BUF9JTVBPUlRfX18pO1xuLy8gTW9kdWxlXG5fX19DU1NfTE9BREVSX0VYUE9SVF9fXy5wdXNoKFttb2R1bGUuaWQsIFwiLnBsYXllci5wbGF5LWJ0biB7XFxuICAgIGJveC1zaXppbmc6IGJvcmRlci1ib3g7XFxuICAgIGhlaWdodDogNDhweDtcXG4gICAgd2lkdGg6IDQ4O1xcbiAgICB0cmFuc2l0aW9uOiA4MG1zIGFsbCBlYXNlO1xcbiAgICB3aWxsLWNoYW5nZTogYm9yZGVyLXdpZHRoO1xcbiAgICBjdXJzb3I6IHBvaW50ZXI7XFxuICAgIGJvcmRlci1jb2xvcjogdHJhbnNwYXJlbnQgdHJhbnNwYXJlbnQgdHJhbnNwYXJlbnQgYmxhY2s7XFxuXFxuICAgIGJvcmRlci1zdHlsZTogc29saWQ7XFxuICAgIGJvcmRlci13aWR0aDogMjRweCAwcHggMjRweCA0OHB4O1xcbn1cXG5cXG4ucGxheWVyLnBsYXktYnRuLnBhdXNlIHtcXG4gICAgYm9yZGVyLXN0eWxlOiBkb3VibGU7XFxuICAgIGJvcmRlci13aWR0aDogMHB4IDBweCAwcHggNjBweDtcXG59XFxuXFxuXFxuLnBsYXllci5uZXh0IHtcXG4gICAgYm9yZGVyLXN0eWxlOiBub25lO1xcbiAgICB3aWR0aDogNDhweDtcXG4gICAgaGVpZ2h0OiA0OHB4O1xcbn1cXG5cXG4ucGxheWVyLnByZXZpb3VzIHtcXG4gICAgYm9yZGVyLXN0eWxlOiBub25lO1xcbiAgICB3aWR0aDogNDhweDtcXG4gICAgaGVpZ2h0OiA0OHB4O1xcbn1cXG5cXG50YWJsZS5wbGF5ZXIge1xcbiAgICBib3JkZXItY29sbGFwc2U6IGNvbGxhcHNlO1xcbn1cIiwgXCJcIix7XCJ2ZXJzaW9uXCI6MyxcInNvdXJjZXNcIjpbXCJ3ZWJwYWNrOi8vLi9zcmMvc3R5bGVzL211c2ljUGxheWVyLmNzc1wiXSxcIm5hbWVzXCI6W10sXCJtYXBwaW5nc1wiOlwiQUFBQTtJQUNJLHNCQUFzQjtJQUN0QixZQUFZO0lBQ1osU0FBUztJQUNULHlCQUF5QjtJQUN6Qix5QkFBeUI7SUFDekIsZUFBZTtJQUNmLHVEQUF1RDs7SUFFdkQsbUJBQW1CO0lBQ25CLGdDQUFnQztBQUNwQzs7QUFFQTtJQUNJLG9CQUFvQjtJQUNwQiw4QkFBOEI7QUFDbEM7OztBQUdBO0lBQ0ksa0JBQWtCO0lBQ2xCLFdBQVc7SUFDWCxZQUFZO0FBQ2hCOztBQUVBO0lBQ0ksa0JBQWtCO0lBQ2xCLFdBQVc7SUFDWCxZQUFZO0FBQ2hCOztBQUVBO0lBQ0kseUJBQXlCO0FBQzdCXCIsXCJzb3VyY2VzQ29udGVudFwiOltcIi5wbGF5ZXIucGxheS1idG4ge1xcbiAgICBib3gtc2l6aW5nOiBib3JkZXItYm94O1xcbiAgICBoZWlnaHQ6IDQ4cHg7XFxuICAgIHdpZHRoOiA0ODtcXG4gICAgdHJhbnNpdGlvbjogODBtcyBhbGwgZWFzZTtcXG4gICAgd2lsbC1jaGFuZ2U6IGJvcmRlci13aWR0aDtcXG4gICAgY3Vyc29yOiBwb2ludGVyO1xcbiAgICBib3JkZXItY29sb3I6IHRyYW5zcGFyZW50IHRyYW5zcGFyZW50IHRyYW5zcGFyZW50IGJsYWNrO1xcblxcbiAgICBib3JkZXItc3R5bGU6IHNvbGlkO1xcbiAgICBib3JkZXItd2lkdGg6IDI0cHggMHB4IDI0cHggNDhweDtcXG59XFxuXFxuLnBsYXllci5wbGF5LWJ0bi5wYXVzZSB7XFxuICAgIGJvcmRlci1zdHlsZTogZG91YmxlO1xcbiAgICBib3JkZXItd2lkdGg6IDBweCAwcHggMHB4IDYwcHg7XFxufVxcblxcblxcbi5wbGF5ZXIubmV4dCB7XFxuICAgIGJvcmRlci1zdHlsZTogbm9uZTtcXG4gICAgd2lkdGg6IDQ4cHg7XFxuICAgIGhlaWdodDogNDhweDtcXG59XFxuXFxuLnBsYXllci5wcmV2aW91cyB7XFxuICAgIGJvcmRlci1zdHlsZTogbm9uZTtcXG4gICAgd2lkdGg6IDQ4cHg7XFxuICAgIGhlaWdodDogNDhweDtcXG59XFxuXFxudGFibGUucGxheWVyIHtcXG4gICAgYm9yZGVyLWNvbGxhcHNlOiBjb2xsYXBzZTtcXG59XCJdLFwic291cmNlUm9vdFwiOlwiXCJ9XSk7XG4vLyBFeHBvcnRzXG5leHBvcnQgZGVmYXVsdCBfX19DU1NfTE9BREVSX0VYUE9SVF9fXztcbiJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./node_modules/css-loader/dist/cjs.js!./src/styles/musicPlayer.css\n"); + +/***/ }), + +/***/ "./node_modules/css-loader/dist/runtime/api.js": +/*!*****************************************************!*\ + !*** ./node_modules/css-loader/dist/runtime/api.js ***! + \*****************************************************/ +/***/ ((module) => { + +eval("\n\n/*\n MIT License http://www.opensource.org/licenses/mit-license.php\n Author Tobias Koppers @sokra\n*/\nmodule.exports = function (cssWithMappingToString) {\n var list = []; // return the list of modules as css string\n\n list.toString = function toString() {\n return this.map(function (item) {\n var content = \"\";\n var needLayer = typeof item[5] !== \"undefined\";\n\n if (item[4]) {\n content += \"@supports (\".concat(item[4], \") {\");\n }\n\n if (item[2]) {\n content += \"@media \".concat(item[2], \" {\");\n }\n\n if (needLayer) {\n content += \"@layer\".concat(item[5].length > 0 ? \" \".concat(item[5]) : \"\", \" {\");\n }\n\n content += cssWithMappingToString(item);\n\n if (needLayer) {\n content += \"}\";\n }\n\n if (item[2]) {\n content += \"}\";\n }\n\n if (item[4]) {\n content += \"}\";\n }\n\n return content;\n }).join(\"\");\n }; // import a list of modules into the list\n\n\n list.i = function i(modules, media, dedupe, supports, layer) {\n if (typeof modules === \"string\") {\n modules = [[null, modules, undefined]];\n }\n\n var alreadyImportedModules = {};\n\n if (dedupe) {\n for (var k = 0; k < this.length; k++) {\n var id = this[k][0];\n\n if (id != null) {\n alreadyImportedModules[id] = true;\n }\n }\n }\n\n for (var _k = 0; _k < modules.length; _k++) {\n var item = [].concat(modules[_k]);\n\n if (dedupe && alreadyImportedModules[item[0]]) {\n continue;\n }\n\n if (typeof layer !== \"undefined\") {\n if (typeof item[5] === \"undefined\") {\n item[5] = layer;\n } else {\n item[1] = \"@layer\".concat(item[5].length > 0 ? \" \".concat(item[5]) : \"\", \" {\").concat(item[1], \"}\");\n item[5] = layer;\n }\n }\n\n if (media) {\n if (!item[2]) {\n item[2] = media;\n } else {\n item[1] = \"@media \".concat(item[2], \" {\").concat(item[1], \"}\");\n item[2] = media;\n }\n }\n\n if (supports) {\n if (!item[4]) {\n item[4] = \"\".concat(supports);\n } else {\n item[1] = \"@supports (\".concat(item[4], \") {\").concat(item[1], \"}\");\n item[4] = supports;\n }\n }\n\n list.push(item);\n }\n };\n\n return list;\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvYXBpLmpzLmpzIiwibWFwcGluZ3MiOiJBQUFhOztBQUViO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxpQkFBaUI7O0FBRWpCO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0EscURBQXFEO0FBQ3JEOztBQUVBO0FBQ0EsZ0RBQWdEO0FBQ2hEOztBQUVBO0FBQ0EscUZBQXFGO0FBQ3JGOztBQUVBOztBQUVBO0FBQ0EscUJBQXFCO0FBQ3JCOztBQUVBO0FBQ0EscUJBQXFCO0FBQ3JCOztBQUVBO0FBQ0EscUJBQXFCO0FBQ3JCOztBQUVBO0FBQ0EsS0FBSztBQUNMLEtBQUs7OztBQUdMO0FBQ0E7QUFDQTtBQUNBOztBQUVBOztBQUVBO0FBQ0Esc0JBQXNCLGlCQUFpQjtBQUN2Qzs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBLHFCQUFxQixxQkFBcUI7QUFDMUM7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTtBQUNBLFVBQVU7QUFDVixzRkFBc0YscUJBQXFCO0FBQzNHO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQSxVQUFVO0FBQ1YsaURBQWlELHFCQUFxQjtBQUN0RTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0EsVUFBVTtBQUNWLHNEQUFzRCxxQkFBcUI7QUFDM0U7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL2FwaS5qcz8yNGZiIl0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG4vKlxuICBNSVQgTGljZW5zZSBodHRwOi8vd3d3Lm9wZW5zb3VyY2Uub3JnL2xpY2Vuc2VzL21pdC1saWNlbnNlLnBocFxuICBBdXRob3IgVG9iaWFzIEtvcHBlcnMgQHNva3JhXG4qL1xubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAoY3NzV2l0aE1hcHBpbmdUb1N0cmluZykge1xuICB2YXIgbGlzdCA9IFtdOyAvLyByZXR1cm4gdGhlIGxpc3Qgb2YgbW9kdWxlcyBhcyBjc3Mgc3RyaW5nXG5cbiAgbGlzdC50b1N0cmluZyA9IGZ1bmN0aW9uIHRvU3RyaW5nKCkge1xuICAgIHJldHVybiB0aGlzLm1hcChmdW5jdGlvbiAoaXRlbSkge1xuICAgICAgdmFyIGNvbnRlbnQgPSBcIlwiO1xuICAgICAgdmFyIG5lZWRMYXllciA9IHR5cGVvZiBpdGVtWzVdICE9PSBcInVuZGVmaW5lZFwiO1xuXG4gICAgICBpZiAoaXRlbVs0XSkge1xuICAgICAgICBjb250ZW50ICs9IFwiQHN1cHBvcnRzIChcIi5jb25jYXQoaXRlbVs0XSwgXCIpIHtcIik7XG4gICAgICB9XG5cbiAgICAgIGlmIChpdGVtWzJdKSB7XG4gICAgICAgIGNvbnRlbnQgKz0gXCJAbWVkaWEgXCIuY29uY2F0KGl0ZW1bMl0sIFwiIHtcIik7XG4gICAgICB9XG5cbiAgICAgIGlmIChuZWVkTGF5ZXIpIHtcbiAgICAgICAgY29udGVudCArPSBcIkBsYXllclwiLmNvbmNhdChpdGVtWzVdLmxlbmd0aCA+IDAgPyBcIiBcIi5jb25jYXQoaXRlbVs1XSkgOiBcIlwiLCBcIiB7XCIpO1xuICAgICAgfVxuXG4gICAgICBjb250ZW50ICs9IGNzc1dpdGhNYXBwaW5nVG9TdHJpbmcoaXRlbSk7XG5cbiAgICAgIGlmIChuZWVkTGF5ZXIpIHtcbiAgICAgICAgY29udGVudCArPSBcIn1cIjtcbiAgICAgIH1cblxuICAgICAgaWYgKGl0ZW1bMl0pIHtcbiAgICAgICAgY29udGVudCArPSBcIn1cIjtcbiAgICAgIH1cblxuICAgICAgaWYgKGl0ZW1bNF0pIHtcbiAgICAgICAgY29udGVudCArPSBcIn1cIjtcbiAgICAgIH1cblxuICAgICAgcmV0dXJuIGNvbnRlbnQ7XG4gICAgfSkuam9pbihcIlwiKTtcbiAgfTsgLy8gaW1wb3J0IGEgbGlzdCBvZiBtb2R1bGVzIGludG8gdGhlIGxpc3RcblxuXG4gIGxpc3QuaSA9IGZ1bmN0aW9uIGkobW9kdWxlcywgbWVkaWEsIGRlZHVwZSwgc3VwcG9ydHMsIGxheWVyKSB7XG4gICAgaWYgKHR5cGVvZiBtb2R1bGVzID09PSBcInN0cmluZ1wiKSB7XG4gICAgICBtb2R1bGVzID0gW1tudWxsLCBtb2R1bGVzLCB1bmRlZmluZWRdXTtcbiAgICB9XG5cbiAgICB2YXIgYWxyZWFkeUltcG9ydGVkTW9kdWxlcyA9IHt9O1xuXG4gICAgaWYgKGRlZHVwZSkge1xuICAgICAgZm9yICh2YXIgayA9IDA7IGsgPCB0aGlzLmxlbmd0aDsgaysrKSB7XG4gICAgICAgIHZhciBpZCA9IHRoaXNba11bMF07XG5cbiAgICAgICAgaWYgKGlkICE9IG51bGwpIHtcbiAgICAgICAgICBhbHJlYWR5SW1wb3J0ZWRNb2R1bGVzW2lkXSA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgIH1cbiAgICB9XG5cbiAgICBmb3IgKHZhciBfayA9IDA7IF9rIDwgbW9kdWxlcy5sZW5ndGg7IF9rKyspIHtcbiAgICAgIHZhciBpdGVtID0gW10uY29uY2F0KG1vZHVsZXNbX2tdKTtcblxuICAgICAgaWYgKGRlZHVwZSAmJiBhbHJlYWR5SW1wb3J0ZWRNb2R1bGVzW2l0ZW1bMF1dKSB7XG4gICAgICAgIGNvbnRpbnVlO1xuICAgICAgfVxuXG4gICAgICBpZiAodHlwZW9mIGxheWVyICE9PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgICAgIGlmICh0eXBlb2YgaXRlbVs1XSA9PT0gXCJ1bmRlZmluZWRcIikge1xuICAgICAgICAgIGl0ZW1bNV0gPSBsYXllcjtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpdGVtWzFdID0gXCJAbGF5ZXJcIi5jb25jYXQoaXRlbVs1XS5sZW5ndGggPiAwID8gXCIgXCIuY29uY2F0KGl0ZW1bNV0pIDogXCJcIiwgXCIge1wiKS5jb25jYXQoaXRlbVsxXSwgXCJ9XCIpO1xuICAgICAgICAgIGl0ZW1bNV0gPSBsYXllcjtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAobWVkaWEpIHtcbiAgICAgICAgaWYgKCFpdGVtWzJdKSB7XG4gICAgICAgICAgaXRlbVsyXSA9IG1lZGlhO1xuICAgICAgICB9IGVsc2Uge1xuICAgICAgICAgIGl0ZW1bMV0gPSBcIkBtZWRpYSBcIi5jb25jYXQoaXRlbVsyXSwgXCIge1wiKS5jb25jYXQoaXRlbVsxXSwgXCJ9XCIpO1xuICAgICAgICAgIGl0ZW1bMl0gPSBtZWRpYTtcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBpZiAoc3VwcG9ydHMpIHtcbiAgICAgICAgaWYgKCFpdGVtWzRdKSB7XG4gICAgICAgICAgaXRlbVs0XSA9IFwiXCIuY29uY2F0KHN1cHBvcnRzKTtcbiAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICBpdGVtWzFdID0gXCJAc3VwcG9ydHMgKFwiLmNvbmNhdChpdGVtWzRdLCBcIikge1wiKS5jb25jYXQoaXRlbVsxXSwgXCJ9XCIpO1xuICAgICAgICAgIGl0ZW1bNF0gPSBzdXBwb3J0cztcbiAgICAgICAgfVxuICAgICAgfVxuXG4gICAgICBsaXN0LnB1c2goaXRlbSk7XG4gICAgfVxuICB9O1xuXG4gIHJldHVybiBsaXN0O1xufTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./node_modules/css-loader/dist/runtime/api.js\n"); + +/***/ }), + +/***/ "./node_modules/css-loader/dist/runtime/sourceMaps.js": +/*!************************************************************!*\ + !*** ./node_modules/css-loader/dist/runtime/sourceMaps.js ***! + \************************************************************/ +/***/ ((module) => { + +eval("\n\nmodule.exports = function (item) {\n var content = item[1];\n var cssMapping = item[3];\n\n if (!cssMapping) {\n return content;\n }\n\n if (typeof btoa === \"function\") {\n var base64 = btoa(unescape(encodeURIComponent(JSON.stringify(cssMapping))));\n var data = \"sourceMappingURL=data:application/json;charset=utf-8;base64,\".concat(base64);\n var sourceMapping = \"/*# \".concat(data, \" */\");\n var sourceURLs = cssMapping.sources.map(function (source) {\n return \"/*# sourceURL=\".concat(cssMapping.sourceRoot || \"\").concat(source, \" */\");\n });\n return [content].concat(sourceURLs).concat([sourceMapping]).join(\"\\n\");\n }\n\n return [content].join(\"\\n\");\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L3J1bnRpbWUvc291cmNlTWFwcy5qcy5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYjtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQSx1REFBdUQsY0FBYztBQUNyRTtBQUNBO0FBQ0E7QUFDQSxLQUFLO0FBQ0w7QUFDQTs7QUFFQTtBQUNBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vbm9kZV9tb2R1bGVzL2Nzcy1sb2FkZXIvZGlzdC9ydW50aW1lL3NvdXJjZU1hcHMuanM/YWYxMiJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxubW9kdWxlLmV4cG9ydHMgPSBmdW5jdGlvbiAoaXRlbSkge1xuICB2YXIgY29udGVudCA9IGl0ZW1bMV07XG4gIHZhciBjc3NNYXBwaW5nID0gaXRlbVszXTtcblxuICBpZiAoIWNzc01hcHBpbmcpIHtcbiAgICByZXR1cm4gY29udGVudDtcbiAgfVxuXG4gIGlmICh0eXBlb2YgYnRvYSA9PT0gXCJmdW5jdGlvblwiKSB7XG4gICAgdmFyIGJhc2U2NCA9IGJ0b2EodW5lc2NhcGUoZW5jb2RlVVJJQ29tcG9uZW50KEpTT04uc3RyaW5naWZ5KGNzc01hcHBpbmcpKSkpO1xuICAgIHZhciBkYXRhID0gXCJzb3VyY2VNYXBwaW5nVVJMPWRhdGE6YXBwbGljYXRpb24vanNvbjtjaGFyc2V0PXV0Zi04O2Jhc2U2NCxcIi5jb25jYXQoYmFzZTY0KTtcbiAgICB2YXIgc291cmNlTWFwcGluZyA9IFwiLyojIFwiLmNvbmNhdChkYXRhLCBcIiAqL1wiKTtcbiAgICB2YXIgc291cmNlVVJMcyA9IGNzc01hcHBpbmcuc291cmNlcy5tYXAoZnVuY3Rpb24gKHNvdXJjZSkge1xuICAgICAgcmV0dXJuIFwiLyojIHNvdXJjZVVSTD1cIi5jb25jYXQoY3NzTWFwcGluZy5zb3VyY2VSb290IHx8IFwiXCIpLmNvbmNhdChzb3VyY2UsIFwiICovXCIpO1xuICAgIH0pO1xuICAgIHJldHVybiBbY29udGVudF0uY29uY2F0KHNvdXJjZVVSTHMpLmNvbmNhdChbc291cmNlTWFwcGluZ10pLmpvaW4oXCJcXG5cIik7XG4gIH1cblxuICByZXR1cm4gW2NvbnRlbnRdLmpvaW4oXCJcXG5cIik7XG59OyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./node_modules/css-loader/dist/runtime/sourceMaps.js\n"); + +/***/ }), + +/***/ "./src/styles/musicPlayer.css": +/*!************************************!*\ + !*** ./src/styles/musicPlayer.css ***! + \************************************/ +/***/ ((__unused_webpack_module, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (__WEBPACK_DEFAULT_EXPORT__)\n/* harmony export */ });\n/* harmony import */ var _node_modules_style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js */ \"./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0__);\n/* harmony import */ var _node_modules_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/styleDomAPI.js */ \"./node_modules/style-loader/dist/runtime/styleDomAPI.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1__);\n/* harmony import */ var _node_modules_style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/insertBySelector.js */ \"./node_modules/style-loader/dist/runtime/insertBySelector.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2__);\n/* harmony import */ var _node_modules_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js */ \"./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3__);\n/* harmony import */ var _node_modules_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/insertStyleElement.js */ \"./node_modules/style-loader/dist/runtime/insertStyleElement.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4__);\n/* harmony import */ var _node_modules_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5__ = __webpack_require__(/*! !../../node_modules/style-loader/dist/runtime/styleTagTransform.js */ \"./node_modules/style-loader/dist/runtime/styleTagTransform.js\");\n/* harmony import */ var _node_modules_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5___default = /*#__PURE__*/__webpack_require__.n(_node_modules_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5__);\n/* harmony import */ var _node_modules_css_loader_dist_cjs_js_musicPlayer_css__WEBPACK_IMPORTED_MODULE_6__ = __webpack_require__(/*! !!../../node_modules/css-loader/dist/cjs.js!./musicPlayer.css */ \"./node_modules/css-loader/dist/cjs.js!./src/styles/musicPlayer.css\");\n\n \n \n \n \n \n \n \n \n \n\nvar options = {};\n\noptions.styleTagTransform = (_node_modules_style_loader_dist_runtime_styleTagTransform_js__WEBPACK_IMPORTED_MODULE_5___default());\noptions.setAttributes = (_node_modules_style_loader_dist_runtime_setAttributesWithoutAttributes_js__WEBPACK_IMPORTED_MODULE_3___default());\n\n options.insert = _node_modules_style_loader_dist_runtime_insertBySelector_js__WEBPACK_IMPORTED_MODULE_2___default().bind(null, \"head\");\n \noptions.domAPI = (_node_modules_style_loader_dist_runtime_styleDomAPI_js__WEBPACK_IMPORTED_MODULE_1___default());\noptions.insertStyleElement = (_node_modules_style_loader_dist_runtime_insertStyleElement_js__WEBPACK_IMPORTED_MODULE_4___default());\n\nvar update = _node_modules_style_loader_dist_runtime_injectStylesIntoStyleTag_js__WEBPACK_IMPORTED_MODULE_0___default()(_node_modules_css_loader_dist_cjs_js_musicPlayer_css__WEBPACK_IMPORTED_MODULE_6__[\"default\"], options);\n\n\n\n\n /* harmony default export */ const __WEBPACK_DEFAULT_EXPORT__ = (_node_modules_css_loader_dist_cjs_js_musicPlayer_css__WEBPACK_IMPORTED_MODULE_6__[\"default\"] && _node_modules_css_loader_dist_cjs_js_musicPlayer_css__WEBPACK_IMPORTED_MODULE_6__[\"default\"].locals ? _node_modules_css_loader_dist_cjs_js_musicPlayer_css__WEBPACK_IMPORTED_MODULE_6__[\"default\"].locals : undefined);\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvc3R5bGVzL211c2ljUGxheWVyLmNzcy5qcyIsIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFDQSxNQUFrRztBQUNsRyxNQUF3RjtBQUN4RixNQUErRjtBQUMvRixNQUFrSDtBQUNsSCxNQUEyRztBQUMzRyxNQUEyRztBQUMzRyxNQUE0RztBQUM1RztBQUNBOztBQUVBOztBQUVBLDRCQUE0QixxR0FBbUI7QUFDL0Msd0JBQXdCLGtIQUFhOztBQUVyQyx1QkFBdUIsdUdBQWE7QUFDcEM7QUFDQSxpQkFBaUIsK0ZBQU07QUFDdkIsNkJBQTZCLHNHQUFrQjs7QUFFL0MsYUFBYSwwR0FBRyxDQUFDLDRGQUFPOzs7O0FBSXNEO0FBQzlFLE9BQU8saUVBQWUsNEZBQU8sSUFBSSxtR0FBYyxHQUFHLG1HQUFjLFlBQVksRUFBQyIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL3NyYy9zdHlsZXMvbXVzaWNQbGF5ZXIuY3NzP2YzNWYiXSwic291cmNlc0NvbnRlbnQiOlsiXG4gICAgICBpbXBvcnQgQVBJIGZyb20gXCIhLi4vLi4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvaW5qZWN0U3R5bGVzSW50b1N0eWxlVGFnLmpzXCI7XG4gICAgICBpbXBvcnQgZG9tQVBJIGZyb20gXCIhLi4vLi4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc3R5bGVEb21BUEkuanNcIjtcbiAgICAgIGltcG9ydCBpbnNlcnRGbiBmcm9tIFwiIS4uLy4uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luc2VydEJ5U2VsZWN0b3IuanNcIjtcbiAgICAgIGltcG9ydCBzZXRBdHRyaWJ1dGVzIGZyb20gXCIhLi4vLi4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc2V0QXR0cmlidXRlc1dpdGhvdXRBdHRyaWJ1dGVzLmpzXCI7XG4gICAgICBpbXBvcnQgaW5zZXJ0U3R5bGVFbGVtZW50IGZyb20gXCIhLi4vLi4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvaW5zZXJ0U3R5bGVFbGVtZW50LmpzXCI7XG4gICAgICBpbXBvcnQgc3R5bGVUYWdUcmFuc2Zvcm1GbiBmcm9tIFwiIS4uLy4uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL3N0eWxlVGFnVHJhbnNmb3JtLmpzXCI7XG4gICAgICBpbXBvcnQgY29udGVudCwgKiBhcyBuYW1lZEV4cG9ydCBmcm9tIFwiISEuLi8uLi9ub2RlX21vZHVsZXMvY3NzLWxvYWRlci9kaXN0L2Nqcy5qcyEuL211c2ljUGxheWVyLmNzc1wiO1xuICAgICAgXG4gICAgICBcblxudmFyIG9wdGlvbnMgPSB7fTtcblxub3B0aW9ucy5zdHlsZVRhZ1RyYW5zZm9ybSA9IHN0eWxlVGFnVHJhbnNmb3JtRm47XG5vcHRpb25zLnNldEF0dHJpYnV0ZXMgPSBzZXRBdHRyaWJ1dGVzO1xuXG4gICAgICBvcHRpb25zLmluc2VydCA9IGluc2VydEZuLmJpbmQobnVsbCwgXCJoZWFkXCIpO1xuICAgIFxub3B0aW9ucy5kb21BUEkgPSBkb21BUEk7XG5vcHRpb25zLmluc2VydFN0eWxlRWxlbWVudCA9IGluc2VydFN0eWxlRWxlbWVudDtcblxudmFyIHVwZGF0ZSA9IEFQSShjb250ZW50LCBvcHRpb25zKTtcblxuXG5cbmV4cG9ydCAqIGZyb20gXCIhIS4uLy4uL25vZGVfbW9kdWxlcy9jc3MtbG9hZGVyL2Rpc3QvY2pzLmpzIS4vbXVzaWNQbGF5ZXIuY3NzXCI7XG4gICAgICAgZXhwb3J0IGRlZmF1bHQgY29udGVudCAmJiBjb250ZW50LmxvY2FscyA/IGNvbnRlbnQubG9jYWxzIDogdW5kZWZpbmVkO1xuIl0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/styles/musicPlayer.css\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js": +/*!****************************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js ***! + \****************************************************************************/ +/***/ ((module) => { + +eval("\n\nvar stylesInDOM = [];\n\nfunction getIndexByIdentifier(identifier) {\n var result = -1;\n\n for (var i = 0; i < stylesInDOM.length; i++) {\n if (stylesInDOM[i].identifier === identifier) {\n result = i;\n break;\n }\n }\n\n return result;\n}\n\nfunction modulesToDom(list, options) {\n var idCountMap = {};\n var identifiers = [];\n\n for (var i = 0; i < list.length; i++) {\n var item = list[i];\n var id = options.base ? item[0] + options.base : item[0];\n var count = idCountMap[id] || 0;\n var identifier = \"\".concat(id, \" \").concat(count);\n idCountMap[id] = count + 1;\n var indexByIdentifier = getIndexByIdentifier(identifier);\n var obj = {\n css: item[1],\n media: item[2],\n sourceMap: item[3],\n supports: item[4],\n layer: item[5]\n };\n\n if (indexByIdentifier !== -1) {\n stylesInDOM[indexByIdentifier].references++;\n stylesInDOM[indexByIdentifier].updater(obj);\n } else {\n var updater = addElementStyle(obj, options);\n options.byIndex = i;\n stylesInDOM.splice(i, 0, {\n identifier: identifier,\n updater: updater,\n references: 1\n });\n }\n\n identifiers.push(identifier);\n }\n\n return identifiers;\n}\n\nfunction addElementStyle(obj, options) {\n var api = options.domAPI(options);\n api.update(obj);\n\n var updater = function updater(newObj) {\n if (newObj) {\n if (newObj.css === obj.css && newObj.media === obj.media && newObj.sourceMap === obj.sourceMap && newObj.supports === obj.supports && newObj.layer === obj.layer) {\n return;\n }\n\n api.update(obj = newObj);\n } else {\n api.remove();\n }\n };\n\n return updater;\n}\n\nmodule.exports = function (list, options) {\n options = options || {};\n list = list || [];\n var lastIdentifiers = modulesToDom(list, options);\n return function update(newList) {\n newList = newList || [];\n\n for (var i = 0; i < lastIdentifiers.length; i++) {\n var identifier = lastIdentifiers[i];\n var index = getIndexByIdentifier(identifier);\n stylesInDOM[index].references--;\n }\n\n var newLastIdentifiers = modulesToDom(newList, options);\n\n for (var _i = 0; _i < lastIdentifiers.length; _i++) {\n var _identifier = lastIdentifiers[_i];\n\n var _index = getIndexByIdentifier(_identifier);\n\n if (stylesInDOM[_index].references === 0) {\n stylesInDOM[_index].updater();\n\n stylesInDOM.splice(_index, 1);\n }\n }\n\n lastIdentifiers = newLastIdentifiers;\n };\n};//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9pbmplY3RTdHlsZXNJbnRvU3R5bGVUYWcuanMuanMiLCJtYXBwaW5ncyI6IkFBQWE7O0FBRWI7O0FBRUE7QUFDQTs7QUFFQSxrQkFBa0Isd0JBQXdCO0FBQzFDO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUEsa0JBQWtCLGlCQUFpQjtBQUNuQztBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQSxNQUFNO0FBQ047QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EsT0FBTztBQUNQOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBLE1BQU07QUFDTjtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUEsb0JBQW9CLDRCQUE0QjtBQUNoRDtBQUNBO0FBQ0E7QUFDQTs7QUFFQTs7QUFFQSxxQkFBcUIsNkJBQTZCO0FBQ2xEOztBQUVBOztBQUVBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luamVjdFN0eWxlc0ludG9TdHlsZVRhZy5qcz8yZGJhIl0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG52YXIgc3R5bGVzSW5ET00gPSBbXTtcblxuZnVuY3Rpb24gZ2V0SW5kZXhCeUlkZW50aWZpZXIoaWRlbnRpZmllcikge1xuICB2YXIgcmVzdWx0ID0gLTE7XG5cbiAgZm9yICh2YXIgaSA9IDA7IGkgPCBzdHlsZXNJbkRPTS5sZW5ndGg7IGkrKykge1xuICAgIGlmIChzdHlsZXNJbkRPTVtpXS5pZGVudGlmaWVyID09PSBpZGVudGlmaWVyKSB7XG4gICAgICByZXN1bHQgPSBpO1xuICAgICAgYnJlYWs7XG4gICAgfVxuICB9XG5cbiAgcmV0dXJuIHJlc3VsdDtcbn1cblxuZnVuY3Rpb24gbW9kdWxlc1RvRG9tKGxpc3QsIG9wdGlvbnMpIHtcbiAgdmFyIGlkQ291bnRNYXAgPSB7fTtcbiAgdmFyIGlkZW50aWZpZXJzID0gW107XG5cbiAgZm9yICh2YXIgaSA9IDA7IGkgPCBsaXN0Lmxlbmd0aDsgaSsrKSB7XG4gICAgdmFyIGl0ZW0gPSBsaXN0W2ldO1xuICAgIHZhciBpZCA9IG9wdGlvbnMuYmFzZSA/IGl0ZW1bMF0gKyBvcHRpb25zLmJhc2UgOiBpdGVtWzBdO1xuICAgIHZhciBjb3VudCA9IGlkQ291bnRNYXBbaWRdIHx8IDA7XG4gICAgdmFyIGlkZW50aWZpZXIgPSBcIlwiLmNvbmNhdChpZCwgXCIgXCIpLmNvbmNhdChjb3VudCk7XG4gICAgaWRDb3VudE1hcFtpZF0gPSBjb3VudCArIDE7XG4gICAgdmFyIGluZGV4QnlJZGVudGlmaWVyID0gZ2V0SW5kZXhCeUlkZW50aWZpZXIoaWRlbnRpZmllcik7XG4gICAgdmFyIG9iaiA9IHtcbiAgICAgIGNzczogaXRlbVsxXSxcbiAgICAgIG1lZGlhOiBpdGVtWzJdLFxuICAgICAgc291cmNlTWFwOiBpdGVtWzNdLFxuICAgICAgc3VwcG9ydHM6IGl0ZW1bNF0sXG4gICAgICBsYXllcjogaXRlbVs1XVxuICAgIH07XG5cbiAgICBpZiAoaW5kZXhCeUlkZW50aWZpZXIgIT09IC0xKSB7XG4gICAgICBzdHlsZXNJbkRPTVtpbmRleEJ5SWRlbnRpZmllcl0ucmVmZXJlbmNlcysrO1xuICAgICAgc3R5bGVzSW5ET01baW5kZXhCeUlkZW50aWZpZXJdLnVwZGF0ZXIob2JqKTtcbiAgICB9IGVsc2Uge1xuICAgICAgdmFyIHVwZGF0ZXIgPSBhZGRFbGVtZW50U3R5bGUob2JqLCBvcHRpb25zKTtcbiAgICAgIG9wdGlvbnMuYnlJbmRleCA9IGk7XG4gICAgICBzdHlsZXNJbkRPTS5zcGxpY2UoaSwgMCwge1xuICAgICAgICBpZGVudGlmaWVyOiBpZGVudGlmaWVyLFxuICAgICAgICB1cGRhdGVyOiB1cGRhdGVyLFxuICAgICAgICByZWZlcmVuY2VzOiAxXG4gICAgICB9KTtcbiAgICB9XG5cbiAgICBpZGVudGlmaWVycy5wdXNoKGlkZW50aWZpZXIpO1xuICB9XG5cbiAgcmV0dXJuIGlkZW50aWZpZXJzO1xufVxuXG5mdW5jdGlvbiBhZGRFbGVtZW50U3R5bGUob2JqLCBvcHRpb25zKSB7XG4gIHZhciBhcGkgPSBvcHRpb25zLmRvbUFQSShvcHRpb25zKTtcbiAgYXBpLnVwZGF0ZShvYmopO1xuXG4gIHZhciB1cGRhdGVyID0gZnVuY3Rpb24gdXBkYXRlcihuZXdPYmopIHtcbiAgICBpZiAobmV3T2JqKSB7XG4gICAgICBpZiAobmV3T2JqLmNzcyA9PT0gb2JqLmNzcyAmJiBuZXdPYmoubWVkaWEgPT09IG9iai5tZWRpYSAmJiBuZXdPYmouc291cmNlTWFwID09PSBvYmouc291cmNlTWFwICYmIG5ld09iai5zdXBwb3J0cyA9PT0gb2JqLnN1cHBvcnRzICYmIG5ld09iai5sYXllciA9PT0gb2JqLmxheWVyKSB7XG4gICAgICAgIHJldHVybjtcbiAgICAgIH1cblxuICAgICAgYXBpLnVwZGF0ZShvYmogPSBuZXdPYmopO1xuICAgIH0gZWxzZSB7XG4gICAgICBhcGkucmVtb3ZlKCk7XG4gICAgfVxuICB9O1xuXG4gIHJldHVybiB1cGRhdGVyO1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IGZ1bmN0aW9uIChsaXN0LCBvcHRpb25zKSB7XG4gIG9wdGlvbnMgPSBvcHRpb25zIHx8IHt9O1xuICBsaXN0ID0gbGlzdCB8fCBbXTtcbiAgdmFyIGxhc3RJZGVudGlmaWVycyA9IG1vZHVsZXNUb0RvbShsaXN0LCBvcHRpb25zKTtcbiAgcmV0dXJuIGZ1bmN0aW9uIHVwZGF0ZShuZXdMaXN0KSB7XG4gICAgbmV3TGlzdCA9IG5ld0xpc3QgfHwgW107XG5cbiAgICBmb3IgKHZhciBpID0gMDsgaSA8IGxhc3RJZGVudGlmaWVycy5sZW5ndGg7IGkrKykge1xuICAgICAgdmFyIGlkZW50aWZpZXIgPSBsYXN0SWRlbnRpZmllcnNbaV07XG4gICAgICB2YXIgaW5kZXggPSBnZXRJbmRleEJ5SWRlbnRpZmllcihpZGVudGlmaWVyKTtcbiAgICAgIHN0eWxlc0luRE9NW2luZGV4XS5yZWZlcmVuY2VzLS07XG4gICAgfVxuXG4gICAgdmFyIG5ld0xhc3RJZGVudGlmaWVycyA9IG1vZHVsZXNUb0RvbShuZXdMaXN0LCBvcHRpb25zKTtcblxuICAgIGZvciAodmFyIF9pID0gMDsgX2kgPCBsYXN0SWRlbnRpZmllcnMubGVuZ3RoOyBfaSsrKSB7XG4gICAgICB2YXIgX2lkZW50aWZpZXIgPSBsYXN0SWRlbnRpZmllcnNbX2ldO1xuXG4gICAgICB2YXIgX2luZGV4ID0gZ2V0SW5kZXhCeUlkZW50aWZpZXIoX2lkZW50aWZpZXIpO1xuXG4gICAgICBpZiAoc3R5bGVzSW5ET01bX2luZGV4XS5yZWZlcmVuY2VzID09PSAwKSB7XG4gICAgICAgIHN0eWxlc0luRE9NW19pbmRleF0udXBkYXRlcigpO1xuXG4gICAgICAgIHN0eWxlc0luRE9NLnNwbGljZShfaW5kZXgsIDEpO1xuICAgICAgfVxuICAgIH1cblxuICAgIGxhc3RJZGVudGlmaWVycyA9IG5ld0xhc3RJZGVudGlmaWVycztcbiAgfTtcbn07Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/injectStylesIntoStyleTag.js\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/insertBySelector.js": +/*!********************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/insertBySelector.js ***! + \********************************************************************/ +/***/ ((module) => { + +eval("\n\nvar memo = {};\n/* istanbul ignore next */\n\nfunction getTarget(target) {\n if (typeof memo[target] === \"undefined\") {\n var styleTarget = document.querySelector(target); // Special case to return head of iframe instead of iframe itself\n\n if (window.HTMLIFrameElement && styleTarget instanceof window.HTMLIFrameElement) {\n try {\n // This will throw an exception if access to iframe is blocked\n // due to cross-origin restrictions\n styleTarget = styleTarget.contentDocument.head;\n } catch (e) {\n // istanbul ignore next\n styleTarget = null;\n }\n }\n\n memo[target] = styleTarget;\n }\n\n return memo[target];\n}\n/* istanbul ignore next */\n\n\nfunction insertBySelector(insert, style) {\n var target = getTarget(insert);\n\n if (!target) {\n throw new Error(\"Couldn't find a style target. This probably means that the value for the 'insert' parameter is invalid.\");\n }\n\n target.appendChild(style);\n}\n\nmodule.exports = insertBySelector;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9pbnNlcnRCeVNlbGVjdG9yLmpzLmpzIiwibWFwcGluZ3MiOiJBQUFhOztBQUViO0FBQ0E7O0FBRUE7QUFDQTtBQUNBLHNEQUFzRDs7QUFFdEQ7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLFFBQVE7QUFDUjtBQUNBO0FBQ0E7QUFDQTs7QUFFQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7O0FBR0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7O0FBRUE7QUFDQTs7QUFFQSIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL2luc2VydEJ5U2VsZWN0b3IuanM/YjIxNCJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxudmFyIG1lbW8gPSB7fTtcbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuXG5mdW5jdGlvbiBnZXRUYXJnZXQodGFyZ2V0KSB7XG4gIGlmICh0eXBlb2YgbWVtb1t0YXJnZXRdID09PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgdmFyIHN0eWxlVGFyZ2V0ID0gZG9jdW1lbnQucXVlcnlTZWxlY3Rvcih0YXJnZXQpOyAvLyBTcGVjaWFsIGNhc2UgdG8gcmV0dXJuIGhlYWQgb2YgaWZyYW1lIGluc3RlYWQgb2YgaWZyYW1lIGl0c2VsZlxuXG4gICAgaWYgKHdpbmRvdy5IVE1MSUZyYW1lRWxlbWVudCAmJiBzdHlsZVRhcmdldCBpbnN0YW5jZW9mIHdpbmRvdy5IVE1MSUZyYW1lRWxlbWVudCkge1xuICAgICAgdHJ5IHtcbiAgICAgICAgLy8gVGhpcyB3aWxsIHRocm93IGFuIGV4Y2VwdGlvbiBpZiBhY2Nlc3MgdG8gaWZyYW1lIGlzIGJsb2NrZWRcbiAgICAgICAgLy8gZHVlIHRvIGNyb3NzLW9yaWdpbiByZXN0cmljdGlvbnNcbiAgICAgICAgc3R5bGVUYXJnZXQgPSBzdHlsZVRhcmdldC5jb250ZW50RG9jdW1lbnQuaGVhZDtcbiAgICAgIH0gY2F0Y2ggKGUpIHtcbiAgICAgICAgLy8gaXN0YW5idWwgaWdub3JlIG5leHRcbiAgICAgICAgc3R5bGVUYXJnZXQgPSBudWxsO1xuICAgICAgfVxuICAgIH1cblxuICAgIG1lbW9bdGFyZ2V0XSA9IHN0eWxlVGFyZ2V0O1xuICB9XG5cbiAgcmV0dXJuIG1lbW9bdGFyZ2V0XTtcbn1cbi8qIGlzdGFuYnVsIGlnbm9yZSBuZXh0ICAqL1xuXG5cbmZ1bmN0aW9uIGluc2VydEJ5U2VsZWN0b3IoaW5zZXJ0LCBzdHlsZSkge1xuICB2YXIgdGFyZ2V0ID0gZ2V0VGFyZ2V0KGluc2VydCk7XG5cbiAgaWYgKCF0YXJnZXQpIHtcbiAgICB0aHJvdyBuZXcgRXJyb3IoXCJDb3VsZG4ndCBmaW5kIGEgc3R5bGUgdGFyZ2V0LiBUaGlzIHByb2JhYmx5IG1lYW5zIHRoYXQgdGhlIHZhbHVlIGZvciB0aGUgJ2luc2VydCcgcGFyYW1ldGVyIGlzIGludmFsaWQuXCIpO1xuICB9XG5cbiAgdGFyZ2V0LmFwcGVuZENoaWxkKHN0eWxlKTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSBpbnNlcnRCeVNlbGVjdG9yOyJdLCJuYW1lcyI6W10sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/insertBySelector.js\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/insertStyleElement.js": +/*!**********************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/insertStyleElement.js ***! + \**********************************************************************/ +/***/ ((module) => { + +eval("\n\n/* istanbul ignore next */\nfunction insertStyleElement(options) {\n var element = document.createElement(\"style\");\n options.setAttributes(element, options.attributes);\n options.insert(element, options.options);\n return element;\n}\n\nmodule.exports = insertStyleElement;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9pbnNlcnRTdHlsZUVsZW1lbnQuanMuanMiLCJtYXBwaW5ncyI6IkFBQWE7O0FBRWI7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FBRUEiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9hdWRpb3Nob3draXQvLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9pbnNlcnRTdHlsZUVsZW1lbnQuanM/ZGU2YyJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxuLyogaXN0YW5idWwgaWdub3JlIG5leHQgICovXG5mdW5jdGlvbiBpbnNlcnRTdHlsZUVsZW1lbnQob3B0aW9ucykge1xuICB2YXIgZWxlbWVudCA9IGRvY3VtZW50LmNyZWF0ZUVsZW1lbnQoXCJzdHlsZVwiKTtcbiAgb3B0aW9ucy5zZXRBdHRyaWJ1dGVzKGVsZW1lbnQsIG9wdGlvbnMuYXR0cmlidXRlcyk7XG4gIG9wdGlvbnMuaW5zZXJ0KGVsZW1lbnQsIG9wdGlvbnMub3B0aW9ucyk7XG4gIHJldHVybiBlbGVtZW50O1xufVxuXG5tb2R1bGUuZXhwb3J0cyA9IGluc2VydFN0eWxlRWxlbWVudDsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/insertStyleElement.js\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js": +/*!**********************************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js ***! + \**********************************************************************************/ +/***/ ((module, __unused_webpack_exports, __webpack_require__) => { + +eval("\n\n/* istanbul ignore next */\nfunction setAttributesWithoutAttributes(styleElement) {\n var nonce = true ? __webpack_require__.nc : 0;\n\n if (nonce) {\n styleElement.setAttribute(\"nonce\", nonce);\n }\n}\n\nmodule.exports = setAttributesWithoutAttributes;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9zZXRBdHRyaWJ1dGVzV2l0aG91dEF0dHJpYnV0ZXMuanMuanMiLCJtYXBwaW5ncyI6IkFBQWE7O0FBRWI7QUFDQTtBQUNBLGNBQWMsS0FBd0MsR0FBRyxzQkFBaUIsR0FBRyxDQUFJOztBQUVqRjtBQUNBO0FBQ0E7QUFDQTs7QUFFQSIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL25vZGVfbW9kdWxlcy9zdHlsZS1sb2FkZXIvZGlzdC9ydW50aW1lL3NldEF0dHJpYnV0ZXNXaXRob3V0QXR0cmlidXRlcy5qcz9kZGNlIl0sInNvdXJjZXNDb250ZW50IjpbIlwidXNlIHN0cmljdFwiO1xuXG4vKiBpc3RhbmJ1bCBpZ25vcmUgbmV4dCAgKi9cbmZ1bmN0aW9uIHNldEF0dHJpYnV0ZXNXaXRob3V0QXR0cmlidXRlcyhzdHlsZUVsZW1lbnQpIHtcbiAgdmFyIG5vbmNlID0gdHlwZW9mIF9fd2VicGFja19ub25jZV9fICE9PSBcInVuZGVmaW5lZFwiID8gX193ZWJwYWNrX25vbmNlX18gOiBudWxsO1xuXG4gIGlmIChub25jZSkge1xuICAgIHN0eWxlRWxlbWVudC5zZXRBdHRyaWJ1dGUoXCJub25jZVwiLCBub25jZSk7XG4gIH1cbn1cblxubW9kdWxlLmV4cG9ydHMgPSBzZXRBdHRyaWJ1dGVzV2l0aG91dEF0dHJpYnV0ZXM7Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/setAttributesWithoutAttributes.js\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/styleDomAPI.js": +/*!***************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/styleDomAPI.js ***! + \***************************************************************/ +/***/ ((module) => { + +eval("\n\n/* istanbul ignore next */\nfunction apply(styleElement, options, obj) {\n var css = \"\";\n\n if (obj.supports) {\n css += \"@supports (\".concat(obj.supports, \") {\");\n }\n\n if (obj.media) {\n css += \"@media \".concat(obj.media, \" {\");\n }\n\n var needLayer = typeof obj.layer !== \"undefined\";\n\n if (needLayer) {\n css += \"@layer\".concat(obj.layer.length > 0 ? \" \".concat(obj.layer) : \"\", \" {\");\n }\n\n css += obj.css;\n\n if (needLayer) {\n css += \"}\";\n }\n\n if (obj.media) {\n css += \"}\";\n }\n\n if (obj.supports) {\n css += \"}\";\n }\n\n var sourceMap = obj.sourceMap;\n\n if (sourceMap && typeof btoa !== \"undefined\") {\n css += \"\\n/*# sourceMappingURL=data:application/json;base64,\".concat(btoa(unescape(encodeURIComponent(JSON.stringify(sourceMap)))), \" */\");\n } // For old IE\n\n /* istanbul ignore if */\n\n\n options.styleTagTransform(css, styleElement, options.options);\n}\n\nfunction removeStyleElement(styleElement) {\n // istanbul ignore if\n if (styleElement.parentNode === null) {\n return false;\n }\n\n styleElement.parentNode.removeChild(styleElement);\n}\n/* istanbul ignore next */\n\n\nfunction domAPI(options) {\n var styleElement = options.insertStyleElement(options);\n return {\n update: function update(obj) {\n apply(styleElement, options, obj);\n },\n remove: function remove() {\n removeStyleElement(styleElement);\n }\n };\n}\n\nmodule.exports = domAPI;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9zdHlsZURvbUFQSS5qcy5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYjtBQUNBO0FBQ0E7O0FBRUE7QUFDQSxrREFBa0Q7QUFDbEQ7O0FBRUE7QUFDQSwwQ0FBMEM7QUFDMUM7O0FBRUE7O0FBRUE7QUFDQSxpRkFBaUY7QUFDakY7O0FBRUE7O0FBRUE7QUFDQSxhQUFhO0FBQ2I7O0FBRUE7QUFDQSxhQUFhO0FBQ2I7O0FBRUE7QUFDQSxhQUFhO0FBQ2I7O0FBRUE7O0FBRUE7QUFDQSx5REFBeUQ7QUFDekQsSUFBSTs7QUFFSjs7O0FBR0E7QUFDQTs7QUFFQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBO0FBQ0E7QUFDQTs7O0FBR0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBLEtBQUs7QUFDTDtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQUVBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc3R5bGVEb21BUEkuanM/ZTQ3OSJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxuLyogaXN0YW5idWwgaWdub3JlIG5leHQgICovXG5mdW5jdGlvbiBhcHBseShzdHlsZUVsZW1lbnQsIG9wdGlvbnMsIG9iaikge1xuICB2YXIgY3NzID0gXCJcIjtcblxuICBpZiAob2JqLnN1cHBvcnRzKSB7XG4gICAgY3NzICs9IFwiQHN1cHBvcnRzIChcIi5jb25jYXQob2JqLnN1cHBvcnRzLCBcIikge1wiKTtcbiAgfVxuXG4gIGlmIChvYmoubWVkaWEpIHtcbiAgICBjc3MgKz0gXCJAbWVkaWEgXCIuY29uY2F0KG9iai5tZWRpYSwgXCIge1wiKTtcbiAgfVxuXG4gIHZhciBuZWVkTGF5ZXIgPSB0eXBlb2Ygb2JqLmxheWVyICE9PSBcInVuZGVmaW5lZFwiO1xuXG4gIGlmIChuZWVkTGF5ZXIpIHtcbiAgICBjc3MgKz0gXCJAbGF5ZXJcIi5jb25jYXQob2JqLmxheWVyLmxlbmd0aCA+IDAgPyBcIiBcIi5jb25jYXQob2JqLmxheWVyKSA6IFwiXCIsIFwiIHtcIik7XG4gIH1cblxuICBjc3MgKz0gb2JqLmNzcztcblxuICBpZiAobmVlZExheWVyKSB7XG4gICAgY3NzICs9IFwifVwiO1xuICB9XG5cbiAgaWYgKG9iai5tZWRpYSkge1xuICAgIGNzcyArPSBcIn1cIjtcbiAgfVxuXG4gIGlmIChvYmouc3VwcG9ydHMpIHtcbiAgICBjc3MgKz0gXCJ9XCI7XG4gIH1cblxuICB2YXIgc291cmNlTWFwID0gb2JqLnNvdXJjZU1hcDtcblxuICBpZiAoc291cmNlTWFwICYmIHR5cGVvZiBidG9hICE9PSBcInVuZGVmaW5lZFwiKSB7XG4gICAgY3NzICs9IFwiXFxuLyojIHNvdXJjZU1hcHBpbmdVUkw9ZGF0YTphcHBsaWNhdGlvbi9qc29uO2Jhc2U2NCxcIi5jb25jYXQoYnRvYSh1bmVzY2FwZShlbmNvZGVVUklDb21wb25lbnQoSlNPTi5zdHJpbmdpZnkoc291cmNlTWFwKSkpKSwgXCIgKi9cIik7XG4gIH0gLy8gRm9yIG9sZCBJRVxuXG4gIC8qIGlzdGFuYnVsIGlnbm9yZSBpZiAgKi9cblxuXG4gIG9wdGlvbnMuc3R5bGVUYWdUcmFuc2Zvcm0oY3NzLCBzdHlsZUVsZW1lbnQsIG9wdGlvbnMub3B0aW9ucyk7XG59XG5cbmZ1bmN0aW9uIHJlbW92ZVN0eWxlRWxlbWVudChzdHlsZUVsZW1lbnQpIHtcbiAgLy8gaXN0YW5idWwgaWdub3JlIGlmXG4gIGlmIChzdHlsZUVsZW1lbnQucGFyZW50Tm9kZSA9PT0gbnVsbCkge1xuICAgIHJldHVybiBmYWxzZTtcbiAgfVxuXG4gIHN0eWxlRWxlbWVudC5wYXJlbnROb2RlLnJlbW92ZUNoaWxkKHN0eWxlRWxlbWVudCk7XG59XG4vKiBpc3RhbmJ1bCBpZ25vcmUgbmV4dCAgKi9cblxuXG5mdW5jdGlvbiBkb21BUEkob3B0aW9ucykge1xuICB2YXIgc3R5bGVFbGVtZW50ID0gb3B0aW9ucy5pbnNlcnRTdHlsZUVsZW1lbnQob3B0aW9ucyk7XG4gIHJldHVybiB7XG4gICAgdXBkYXRlOiBmdW5jdGlvbiB1cGRhdGUob2JqKSB7XG4gICAgICBhcHBseShzdHlsZUVsZW1lbnQsIG9wdGlvbnMsIG9iaik7XG4gICAgfSxcbiAgICByZW1vdmU6IGZ1bmN0aW9uIHJlbW92ZSgpIHtcbiAgICAgIHJlbW92ZVN0eWxlRWxlbWVudChzdHlsZUVsZW1lbnQpO1xuICAgIH1cbiAgfTtcbn1cblxubW9kdWxlLmV4cG9ydHMgPSBkb21BUEk7Il0sIm5hbWVzIjpbXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/styleDomAPI.js\n"); + +/***/ }), + +/***/ "./node_modules/style-loader/dist/runtime/styleTagTransform.js": +/*!*********************************************************************!*\ + !*** ./node_modules/style-loader/dist/runtime/styleTagTransform.js ***! + \*********************************************************************/ +/***/ ((module) => { + +eval("\n\n/* istanbul ignore next */\nfunction styleTagTransform(css, styleElement) {\n if (styleElement.styleSheet) {\n styleElement.styleSheet.cssText = css;\n } else {\n while (styleElement.firstChild) {\n styleElement.removeChild(styleElement.firstChild);\n }\n\n styleElement.appendChild(document.createTextNode(css));\n }\n}\n\nmodule.exports = styleTagTransform;//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9ub2RlX21vZHVsZXMvc3R5bGUtbG9hZGVyL2Rpc3QvcnVudGltZS9zdHlsZVRhZ1RyYW5zZm9ybS5qcy5qcyIsIm1hcHBpbmdzIjoiQUFBYTs7QUFFYjtBQUNBO0FBQ0E7QUFDQTtBQUNBLElBQUk7QUFDSjtBQUNBO0FBQ0E7O0FBRUE7QUFDQTtBQUNBOztBQUVBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vbm9kZV9tb2R1bGVzL3N0eWxlLWxvYWRlci9kaXN0L3J1bnRpbWUvc3R5bGVUYWdUcmFuc2Zvcm0uanM/MWRkZSJdLCJzb3VyY2VzQ29udGVudCI6WyJcInVzZSBzdHJpY3RcIjtcblxuLyogaXN0YW5idWwgaWdub3JlIG5leHQgICovXG5mdW5jdGlvbiBzdHlsZVRhZ1RyYW5zZm9ybShjc3MsIHN0eWxlRWxlbWVudCkge1xuICBpZiAoc3R5bGVFbGVtZW50LnN0eWxlU2hlZXQpIHtcbiAgICBzdHlsZUVsZW1lbnQuc3R5bGVTaGVldC5jc3NUZXh0ID0gY3NzO1xuICB9IGVsc2Uge1xuICAgIHdoaWxlIChzdHlsZUVsZW1lbnQuZmlyc3RDaGlsZCkge1xuICAgICAgc3R5bGVFbGVtZW50LnJlbW92ZUNoaWxkKHN0eWxlRWxlbWVudC5maXJzdENoaWxkKTtcbiAgICB9XG5cbiAgICBzdHlsZUVsZW1lbnQuYXBwZW5kQ2hpbGQoZG9jdW1lbnQuY3JlYXRlVGV4dE5vZGUoY3NzKSk7XG4gIH1cbn1cblxubW9kdWxlLmV4cG9ydHMgPSBzdHlsZVRhZ1RyYW5zZm9ybTsiXSwibmFtZXMiOltdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./node_modules/style-loader/dist/runtime/styleTagTransform.js\n"); + +/***/ }), + +/***/ "./src/audioshowkitlib.js": +/*!********************************!*\ + !*** ./src/audioshowkitlib.js ***! + \********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"mappings\": () => (/* reexport module object */ _mapping_mappings_js__WEBPACK_IMPORTED_MODULE_0__),\n/* harmony export */ \"player\": () => (/* reexport module object */ _player_player_js__WEBPACK_IMPORTED_MODULE_1__),\n/* harmony export */ \"support\": () => (/* reexport module object */ _support_support_js__WEBPACK_IMPORTED_MODULE_3__),\n/* harmony export */ \"visualization\": () => (/* reexport module object */ _visualization_visualization_js__WEBPACK_IMPORTED_MODULE_2__)\n/* harmony export */ });\n/* harmony import */ var _mapping_mappings_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./mapping/mappings.js */ \"./src/mapping/mappings.js\");\n/* harmony import */ var _player_player_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./player/player.js */ \"./src/player/player.js\");\n/* harmony import */ var _visualization_visualization_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./visualization/visualization.js */ \"./src/visualization/visualization.js\");\n/* harmony import */ var _support_support_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./support/support.js */ \"./src/support/support.js\");\n// TODO: basic playlist display.\n// TODO: Demo functions.\n// TODO: Detect annotated elements.\n// TODO: Create getting started.\n\n\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvYXVkaW9zaG93a2l0bGliLmpzLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7O0FBQUE7QUFDQTtBQUNBO0FBQ0E7QUFJQTtBQUNBO0FBQ0E7QUFDQSIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL3NyYy9hdWRpb3Nob3draXRsaWIuanM/MjgxMCJdLCJzb3VyY2VzQ29udGVudCI6WyIvLyBUT0RPOiBiYXNpYyBwbGF5bGlzdCBkaXNwbGF5LlxuLy8gVE9ETzogRGVtbyBmdW5jdGlvbnMuXG4vLyBUT0RPOiBEZXRlY3QgYW5ub3RhdGVkIGVsZW1lbnRzLlxuLy8gVE9ETzogQ3JlYXRlIGdldHRpbmcgc3RhcnRlZC5cblxuXG5cbmltcG9ydCAqIGFzIG1hcHBpbmdzIGZyb20gXCIuL21hcHBpbmcvbWFwcGluZ3MuanNcIjtcbmltcG9ydCAqIGFzIHBsYXllciBmcm9tIFwiLi9wbGF5ZXIvcGxheWVyLmpzXCI7XG5pbXBvcnQgKiBhcyB2aXN1YWxpemF0aW9uIGZyb20gXCIuL3Zpc3VhbGl6YXRpb24vdmlzdWFsaXphdGlvbi5qc1wiO1xuaW1wb3J0ICogYXMgc3VwcG9ydCBmcm9tIFwiLi9zdXBwb3J0L3N1cHBvcnQuanNcIjtcblxuZXhwb3J0IHsgbWFwcGluZ3MsIHBsYXllciwgdmlzdWFsaXphdGlvbiwgc3VwcG9ydCB9OyJdLCJuYW1lcyI6WyJtYXBwaW5ncyIsInBsYXllciIsInZpc3VhbGl6YXRpb24iLCJzdXBwb3J0Il0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/audioshowkitlib.js\n"); + +/***/ }), + +/***/ "./src/mapping/dimensions.js": +/*!***********************************!*\ + !*** ./src/mapping/dimensions.js ***! + \***********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"mapHeight\": () => (/* binding */ mapHeight),\n/* harmony export */ \"mapWidth\": () => (/* binding */ mapWidth)\n/* harmony export */ });\n/* harmony import */ var _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../visualization/VisualizerUpdateManager.js */ \"./src/visualization/VisualizerUpdateManager.js\");\n/* harmony import */ var _numeric_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./numeric.js */ \"./src/mapping/numeric.js\");\n\n\n/**\n * Maps the width of an element to frequency bin(s).\n * \n * @param {object} conf The configuration for mapping amplitudes to a width CSS property.\n * @param {HTMLElement} conf.element The html element to map the width to.\n * @param {number} conf.growLower The lower limit of the width.\n * @param {number} conf.growUpper The upper limit of the width.\n * @param {string} conf.unit The unit the upper and lower bounds are measured in.\n * @param {number} conf.lowerBin The lower boundary of bins to be mapped (or the single bin to be mapped if the upper bin is undefined).\n * @param {VisualizerUpdateManager} conf.visUpdateManager The visualizer update manager to be mapped to.\n * @param {interpolator} conf.interpolator The interpolation function to be used to transition from one value to the next.\n * @param {number} [conf.upperBin] The upper bin or undefined, which results in mapping to a single bin.\n * @param {boolean} [conf.reversed=false] If true, then high amplitudes are mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}|{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} The listener that was added (ranged if an upper bound was provided, binned otherwise).\n */\n\nfunction mapWidth(_ref) {\n var element = _ref.element,\n growLower = _ref.growLower,\n growUpper = _ref.growUpper,\n unit = _ref.unit,\n lowerBin = _ref.lowerBin,\n visUpdateManager = _ref.visUpdateManager,\n interpolator = _ref.interpolator,\n _ref$upperBin = _ref.upperBin,\n upperBin = _ref$upperBin === void 0 ? undefined : _ref$upperBin,\n _ref$reversed = _ref.reversed,\n reversed = _ref$reversed === void 0 ? false : _ref$reversed;\n\n var getter = function getter() {\n return element.style.width;\n };\n\n var setter = function setter(value) {\n return element.style.width = value + unit;\n };\n\n var conf = {\n minVal: growLower,\n maxVal: growUpper,\n bin: lowerBin,\n getter: getter,\n setter: setter,\n interpolator: interpolator,\n visUpdateManager: visUpdateManager,\n reversed: reversed\n };\n\n if (upperBin) {\n conf.bin = undefined;\n conf.lowerBin = lowerBin;\n conf.upperBin = upperBin;\n return (0,_numeric_js__WEBPACK_IMPORTED_MODULE_1__.mapRangedAvgNumerical)(conf);\n }\n\n return (0,_numeric_js__WEBPACK_IMPORTED_MODULE_1__.mapBinNumerical)(conf);\n}\n/**\n * Maps the height of an element to frequency bin(s).\n * \n * @param {object} conf The configuration for mapping amplitudes to a height CSS property.\n * @param {HTMLElement} conf.element The html element to map the height to.\n * @param {number} conf.growLower The lower limit of the height.\n * @param {number} conf.growUpper The upper limit of the height.\n * @param {string} conf.unit The unit the upper and lower bounds are measured in.\n * @param {number} conf.lowerBin The lower boundary of bins to be mapped (or the single bin to be mapped if the upper bin is undefined).\n * @param {VisualizerUpdateManager} conf.visUpdateManager The visualizer update manager to be mapped to.\n * @param {interpolator} conf.interpolator The interpolation function to be used to transition from one value to the next.\n * @param {number} [conf.upperBin] The upper bin or undefined, which results in mapping to a single bin.\n * @param {boolean} [conf.reversed=false] If true, then high amplitudes are mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}|{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} The listener that was added (ranged if an upper bound was provided, binned otherwise).\n */\n\n\nfunction mapHeight(_ref2) {\n var element = _ref2.element,\n growLower = _ref2.growLower,\n growUpper = _ref2.growUpper,\n unit = _ref2.unit,\n lowerBin = _ref2.lowerBin,\n visUpdateManager = _ref2.visUpdateManager,\n interpolator = _ref2.interpolator,\n _ref2$upperBin = _ref2.upperBin,\n upperBin = _ref2$upperBin === void 0 ? undefined : _ref2$upperBin,\n _ref2$reversed = _ref2.reversed,\n reversed = _ref2$reversed === void 0 ? false : _ref2$reversed;\n\n var getter = function getter() {\n return element.style.height;\n };\n\n var setter = function setter(value) {\n return element.style.height = value + unit;\n };\n\n var conf = {\n minVal: growLower,\n maxVal: growUpper,\n bin: lowerBin,\n getter: getter,\n setter: setter,\n interpolator: interpolator,\n visUpdateManager: visUpdateManager,\n reversed: reversed\n };\n\n if (upperBin) {\n conf.bin = undefined;\n conf.lowerBin = lowerBin;\n conf.upperBin = upperBin;\n return (0,_numeric_js__WEBPACK_IMPORTED_MODULE_1__.mapRangedAvgNumerical)(conf);\n }\n\n return (0,_numeric_js__WEBPACK_IMPORTED_MODULE_1__.mapBinNumerical)(conf);\n}\n/**@module */\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/mapping/dimensions.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/mapping/dimensions.js?7773"],"sourcesContent":["import VisualizerUpdateManager from \"../visualization/VisualizerUpdateManager.js\";\nimport { mapBinNumerical, mapRangedAvgNumerical } from \"./numeric.js\";\n\n\n/**\n * Maps the width of an element to frequency bin(s).\n * \n * @param {object} conf The configuration for mapping amplitudes to a width CSS property.\n * @param {HTMLElement} conf.element The html element to map the width to.\n * @param {number} conf.growLower The lower limit of the width.\n * @param {number} conf.growUpper The upper limit of the width.\n * @param {string} conf.unit The unit the upper and lower bounds are measured in.\n * @param {number} conf.lowerBin The lower boundary of bins to be mapped (or the single bin to be mapped if the upper bin is undefined).\n * @param {VisualizerUpdateManager} conf.visUpdateManager The visualizer update manager to be mapped to.\n * @param {interpolator} conf.interpolator The interpolation function to be used to transition from one value to the next.\n * @param {number} [conf.upperBin] The upper bin or undefined, which results in mapping to a single bin.\n * @param {boolean} [conf.reversed=false] If true, then high amplitudes are mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}|{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} The listener that was added (ranged if an upper bound was provided, binned otherwise).\n */\nfunction mapWidth({ element, growLower, growUpper, unit, lowerBin, visUpdateManager, interpolator, upperBin = undefined, reversed = false }) {\n    const getter = () => element.style.width;\n    const setter = (value) => element.style.width = value + unit;\n    const conf = {\n        minVal: growLower,\n        maxVal: growUpper,\n        bin: lowerBin,\n        getter: getter,\n        setter: setter,\n        interpolator: interpolator,\n        visUpdateManager: visUpdateManager,\n        reversed: reversed\n    };\n\n    if (upperBin) {\n        conf.bin = undefined;\n        conf.lowerBin = lowerBin;\n        conf.upperBin = upperBin;\n        return mapRangedAvgNumerical(conf);\n    }\n    return mapBinNumerical(conf);\n}\n\n/**\n * Maps the height of an element to frequency bin(s).\n * \n * @param {object} conf The configuration for mapping amplitudes to a height CSS property.\n * @param {HTMLElement} conf.element The html element to map the height to.\n * @param {number} conf.growLower The lower limit of the height.\n * @param {number} conf.growUpper The upper limit of the height.\n * @param {string} conf.unit The unit the upper and lower bounds are measured in.\n * @param {number} conf.lowerBin The lower boundary of bins to be mapped (or the single bin to be mapped if the upper bin is undefined).\n * @param {VisualizerUpdateManager} conf.visUpdateManager The visualizer update manager to be mapped to.\n * @param {interpolator} conf.interpolator The interpolation function to be used to transition from one value to the next.\n * @param {number} [conf.upperBin] The upper bin or undefined, which results in mapping to a single bin.\n * @param {boolean} [conf.reversed=false] If true, then high amplitudes are mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}|{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} The listener that was added (ranged if an upper bound was provided, binned otherwise).\n */\nfunction mapHeight({ element, growLower, growUpper, unit, lowerBin, visUpdateManager, interpolator, upperBin = undefined, reversed = false }) {\n    const getter = () => element.style.height;\n    const setter = (value) => element.style.height = value + unit;\n    const conf = {\n        minVal: growLower,\n        maxVal: growUpper,\n        bin: lowerBin,\n        getter: getter,\n        setter: setter,\n        interpolator: interpolator,\n        visUpdateManager: visUpdateManager,\n        reversed: reversed\n    };\n\n    if (upperBin) {\n        conf.bin = undefined;\n        conf.lowerBin = lowerBin;\n        conf.upperBin = upperBin;\n        return mapRangedAvgNumerical(conf);\n    }\n    return mapBinNumerical(conf);\n}\n\n/**@module */\nexport { mapWidth, mapHeight };"],"names":["VisualizerUpdateManager","mapBinNumerical","mapRangedAvgNumerical","mapWidth","element","growLower","growUpper","unit","lowerBin","visUpdateManager","interpolator","upperBin","undefined","reversed","getter","style","width","setter","value","conf","minVal","maxVal","bin","mapHeight","height"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/mapping/dimensions.js\n"); + +/***/ }), + +/***/ "./src/mapping/mappings.js": +/*!*********************************!*\ + !*** ./src/mapping/mappings.js ***! + \*********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"dimensions\": () => (/* reexport module object */ _dimensions_js__WEBPACK_IMPORTED_MODULE_0__),\n/* harmony export */ \"numeric\": () => (/* reexport module object */ _numeric_js__WEBPACK_IMPORTED_MODULE_1__)\n/* harmony export */ });\n/* harmony import */ var _dimensions_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./dimensions.js */ \"./src/mapping/dimensions.js\");\n/* harmony import */ var _numeric_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./numeric.js */ \"./src/mapping/numeric.js\");\n\n\n/**@module */\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvbWFwcGluZy9tYXBwaW5ncy5qcy5qcyIsIm1hcHBpbmdzIjoiOzs7Ozs7O0FBQUE7QUFDQTtBQUVBIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vc3JjL21hcHBpbmcvbWFwcGluZ3MuanM/NTdkZSJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgKiBhcyBkaW1lbnNpb25zIGZyb20gXCIuL2RpbWVuc2lvbnMuanNcIjtcbmltcG9ydCAqIGFzIG51bWVyaWMgZnJvbSBcIi4vbnVtZXJpYy5qc1wiO1xuXG4vKipAbW9kdWxlICovXG5leHBvcnQgeyBkaW1lbnNpb25zLCBudW1lcmljIH07Il0sIm5hbWVzIjpbImRpbWVuc2lvbnMiLCJudW1lcmljIl0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/mapping/mappings.js\n"); + +/***/ }), + +/***/ "./src/mapping/numeric.js": +/*!********************************!*\ + !*** ./src/mapping/numeric.js ***! + \********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"mapBinNumerical\": () => (/* binding */ mapBinNumerical),\n/* harmony export */ \"mapRangedAvgNumerical\": () => (/* binding */ mapRangedAvgNumerical)\n/* harmony export */ });\n/* harmony import */ var _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../visualization/VisualizerUpdateManager.js */ \"./src/visualization/VisualizerUpdateManager.js\");\nfunction _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }\n\nfunction _nonIterableSpread() { throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\"); }\n\nfunction _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === \"string\") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === \"Object\" && o.constructor) n = o.constructor.name; if (n === \"Map\" || n === \"Set\") return Array.from(o); if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }\n\nfunction _iterableToArray(iter) { if (typeof Symbol !== \"undefined\" && iter[Symbol.iterator] != null || iter[\"@@iterator\"] != null) return Array.from(iter); }\n\nfunction _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }\n\nfunction _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }\n\n\n/**\n * \n * @callback numericalGetter\n * @returns {number} The number this value currently represents.\n */\n\n/**\n * @callback numericalSetter\n * @param {number} value The new numerical value to update to.\n */\n// FIXME: Warnings about interpolator not being defined are broken. Docs generate fine.\n\n/**\n * Maps the average of a range of bins to numerical value defined by a getter and a setter.\n * \n * @param {object} conf A configuration for how to map a numerical value.\n * @param {number} conf.minVal The minimum value of the numerical value.\n * @param {number} conf.maxVal The maximum value of the numerical value.\n * @param {number} conf.lowerBin The lower bin of the range of bins this value is to be mapped to.\n * @param {number} conf.upperBin The upper bin of the range of bins this value si to be mapped to.\n * @param {numericalGetter} conf.getter The getter callback to be used to get the current number.\n * @param {numericalSetter} conf.setter The setter callback to be used to set the new number.\n * @param {interpolator} conf.interpolator The interpolation function to use.\n * @param {VisualizerUpdateManager} conf.visUpdateManager the visualizer update manager this mapping corresponds with.\n * @param {boolean} [conf.reversed = false] If true, then high amplitudes will be mapped to low values and vice versa.\n * @returns {{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} An object containing the lower and upper bounds for the range of a listener, which is also in the object.\n */\n\nfunction mapRangedAvgNumerical(_ref) {\n var minVal = _ref.minVal,\n maxVal = _ref.maxVal,\n _ref$lowerBin = _ref.lowerBin,\n lowerBin = _ref$lowerBin === void 0 ? undefined : _ref$lowerBin,\n upperBin = _ref.upperBin,\n getter = _ref.getter,\n setter = _ref.setter,\n interpolator = _ref.interpolator,\n visUpdateManager = _ref.visUpdateManager,\n _ref$reversed = _ref.reversed,\n reversed = _ref$reversed === void 0 ? false : _ref$reversed;\n var rangedListener = {\n lower: lowerBin,\n upper: upperBin,\n listener: function listener(timeDelta, bins) {\n var normalBins = _toConsumableArray(bins); // TODO: Future: add weighting / distribution system?\n\n\n var average = 0;\n\n for (var i = 0; i < normalBins.length; i++) {\n normalBins[i] = normalBins[i] / 255.0;\n average += normalBins[i];\n }\n\n average /= normalBins.length;\n var range = maxVal - minVal;\n var interpolated = interpolator(getter(), average, timeDelta);\n if (reversed) interpolated = 1 - interpolated;\n setter(minVal + range * interpolated);\n }\n };\n\n if (visUpdateManager.addVisualizerRangedUpdateListener(rangedListener)) {\n return rangedListener;\n }\n\n return null; // Technically doesn't occur since the functions are anonymous?\n}\n/**\n * Maps a single bin of frequency amplitudes to a numerical value defined by a getter and a setter.\n * \n * @param {object} conf The configuration for mapping a single bin to a numerical value.\n * @param {number} conf.minVal The minimum value the mapping can produce.\n * @param {number} conf.maxVal The maximum value the mapping can produce.\n * @param {number} conf.bin The bin to map this number to.\n * @param {numericalGetter} conf.getter The callback to be used to get the current number.\n * @param {numericalSetter} conf.setter The callback to be used to set the new number.\n * @param {interpolator} conf.interpolator The interpolation function to use.\n * @param {VisualizerUpdateManager} conf.visUpdateManager The update manager to map to.\n * @param {boolean} [conf.reversed] If true, then high amplitudes will be mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}} The bin listener that was added.\n */\n\n\nfunction mapBinNumerical(_ref2) {\n var minVal = _ref2.minVal,\n maxVal = _ref2.maxVal,\n bin = _ref2.bin,\n getter = _ref2.getter,\n setter = _ref2.setter,\n interpolator = _ref2.interpolator,\n visUpdateManager = _ref2.visUpdateManager,\n _ref2$reversed = _ref2.reversed,\n reversed = _ref2$reversed === void 0 ? false : _ref2$reversed;\n var listener = {\n bin: bin,\n listener: function listener(timeDelta, amp) {\n var range = maxVal - minVal;\n var interpolated = interpolator(getter(), amp, timeDelta);\n if (reversed) interpolated = 1 - interpolated;\n setter(minVal + range * interpolated);\n }\n };\n\n if (visUpdateManager.AddVisualizerBinUpdateListener(listener)) {\n return listener;\n }\n\n return null; // Technically doesn't occur since the functions are anonymous?\n}\n/**@module */\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/mapping/numeric.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/mapping/numeric.js?4fa0"],"sourcesContent":["import VisualizerUpdateManager from \"../visualization/VisualizerUpdateManager.js\";\n\n/**\n * \n * @callback numericalGetter\n * @returns {number} The number this value currently represents.\n */\n\n/**\n * @callback numericalSetter\n * @param {number} value The new numerical value to update to.\n */\n\n// FIXME: Warnings about interpolator not being defined are broken. Docs generate fine.\n\n/**\n * Maps the average of a range of bins to numerical value defined by a getter and a setter.\n * \n * @param {object} conf A configuration for how to map a numerical value.\n * @param {number} conf.minVal The minimum value of the numerical value.\n * @param {number} conf.maxVal The maximum value of the numerical value.\n * @param {number} conf.lowerBin The lower bin of the range of bins this value is to be mapped to.\n * @param {number} conf.upperBin The upper bin of the range of bins this value si to be mapped to.\n * @param {numericalGetter} conf.getter The getter callback to be used to get the current number.\n * @param {numericalSetter} conf.setter The setter callback to be used to set the new number.\n * @param {interpolator} conf.interpolator The interpolation function to use.\n * @param {VisualizerUpdateManager} conf.visUpdateManager the visualizer update manager this mapping corresponds with.\n * @param {boolean} [conf.reversed = false] If true, then high amplitudes will be mapped to low values and vice versa.\n * @returns {{lower: number, upper: number, listener: VisualizerUpdateManager.visualizerRangedUpdateListener}} An object containing the lower and upper bounds for the range of a listener, which is also in the object.\n */\nfunction mapRangedAvgNumerical({ minVal, maxVal, lowerBin = undefined, upperBin, getter, setter, interpolator, visUpdateManager, reversed = false }) {\n    const rangedListener = {\n        lower: lowerBin,\n        upper: upperBin,\n        listener: (timeDelta, bins) => {\n            const normalBins = [...bins];\n            // TODO: Future: add weighting / distribution system?\n            let average = 0;\n            for (let i = 0; i < normalBins.length; i++) {\n                normalBins[i] = normalBins[i] / 255.0;\n                average += normalBins[i];\n            }\n            average /= normalBins.length;\n            const range = maxVal - minVal;\n            let interpolated = interpolator(getter(), average, timeDelta);\n            if (reversed) interpolated = 1 - interpolated;\n            setter(minVal + range * interpolated);\n        }\n    };\n    if (visUpdateManager.addVisualizerRangedUpdateListener(rangedListener)) {\n        return rangedListener;\n    }\n    return null; // Technically doesn't occur since the functions are anonymous?\n}\n\n/**\n * Maps a single bin of frequency amplitudes to a numerical value defined by a getter and a setter.\n * \n * @param {object} conf The configuration for mapping a single bin to a numerical value.\n * @param {number} conf.minVal The minimum value the mapping can produce.\n * @param {number} conf.maxVal The maximum value the mapping can produce.\n * @param {number} conf.bin The bin to map this number to.\n * @param {numericalGetter} conf.getter The callback to be used to get the current number.\n * @param {numericalSetter} conf.setter The callback to be used to set the new number.\n * @param {interpolator} conf.interpolator The interpolation function to use.\n * @param {VisualizerUpdateManager} conf.visUpdateManager The update manager to map to.\n * @param {boolean} [conf.reversed] If true, then high amplitudes will be mapped to lower values and vice versa.\n * @returns {{bin: number, listener: VisualizerUpdateManager.visualizerBinUpdateListener}} The bin listener that was added.\n */\nfunction mapBinNumerical({ minVal, maxVal, bin, getter, setter, interpolator, visUpdateManager, reversed = false }) {\n    const listener = {\n        bin: bin,\n        listener: (timeDelta, amp) => {\n            const range = maxVal - minVal;\n            let interpolated = interpolator(getter(), amp, timeDelta);\n            if (reversed) interpolated = 1 - interpolated;\n            setter(minVal + range * interpolated);\n        }\n    };\n    if (visUpdateManager.AddVisualizerBinUpdateListener(listener)) {\n        return listener;\n    }\n    return null; // Technically doesn't occur since the functions are anonymous?\n}\n\n/**@module */\nexport { mapRangedAvgNumerical, mapBinNumerical };"],"names":["VisualizerUpdateManager","mapRangedAvgNumerical","minVal","maxVal","lowerBin","undefined","upperBin","getter","setter","interpolator","visUpdateManager","reversed","rangedListener","lower","upper","listener","timeDelta","bins","normalBins","average","i","length","range","interpolated","addVisualizerRangedUpdateListener","mapBinNumerical","bin","amp","AddVisualizerBinUpdateListener"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/mapping/numeric.js\n"); + +/***/ }), + +/***/ "./src/player/Music.js": +/*!*****************************!*\ + !*** ./src/player/Music.js ***! + \*****************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ Music)\n/* harmony export */ });\n/* harmony import */ var _visualization_Visualizer_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../visualization/Visualizer.js */ \"./src/visualization/Visualizer.js\");\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\n\n/**@module */\n\n/**\n * @callback AudioEventCallback\n * @param {HTMLAudioElement} audio\n */\n\n/**\n * A music with metadata that can be used as part of a {@link MusicPlaylist}.\n * \n */\n\nvar _displayName = /*#__PURE__*/new WeakMap();\n\nvar _author = /*#__PURE__*/new WeakMap();\n\nvar _url = /*#__PURE__*/new WeakMap();\n\nvar _audio = /*#__PURE__*/new WeakMap();\n\nvar _visualizer = /*#__PURE__*/new WeakMap();\n\nvar _ready = /*#__PURE__*/new WeakMap();\n\nvar Music = /*#__PURE__*/function () {\n /**\n * Constructs a music for a {@link MusicPlaylist}.\n * \n * @param {string} url the url to fetch the music from.\n * @param {string} name the name of the music.\n * @param {string} author the author of the music.\n */\n function Music(url, name, author) {\n _classCallCheck(this, Music);\n\n _classPrivateFieldInitSpec(this, _displayName, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _author, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _url, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _audio, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _visualizer, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _ready, {\n writable: true,\n value: false\n });\n\n _classPrivateFieldSet(this, _displayName, name);\n\n _classPrivateFieldSet(this, _author, author);\n\n _classPrivateFieldSet(this, _url, url);\n }\n /**\n * \n * @param {AudioEventCallback} [onReady] called when the music is ready, including right away if the music is already ready.\n * @returns {HTMLAudioElement} The audio element that represents this music.\n */\n\n\n _createClass(Music, [{\n key: \"fetchAudio\",\n value: function fetchAudio(onReady) {\n var _this = this;\n\n if (_classPrivateFieldGet(this, _audio)) {\n if (_classPrivateFieldGet(this, _ready)) {\n if (onReady) onReady(_classPrivateFieldGet(this, _audio));\n } else if (onReady) {\n _classPrivateFieldGet(this, _audio).addEventListener(\"canplaythrough\", function () {\n onReady(_classPrivateFieldGet(_this, _audio));\n }, {\n once: true\n });\n }\n\n return _classPrivateFieldGet(this, _audio);\n }\n\n _classPrivateFieldSet(this, _audio, new Audio(_classPrivateFieldGet(this, _url)));\n\n _classPrivateFieldGet(this, _audio).addEventListener(\"canplaythrough\", function () {\n _classPrivateFieldSet(_this, _ready, true);\n\n if (onReady && _this.isAudioInstantiated()) {\n onReady(_classPrivateFieldGet(_this, _audio));\n }\n }, {\n once: true\n });\n\n return _classPrivateFieldGet(this, _audio);\n }\n /**\n * @returns {string} The name of the music to be displayed.\n */\n\n }, {\n key: \"displayName\",\n get: function get() {\n return _classPrivateFieldGet(this, _displayName);\n }\n /**\n * @returns {string} The author of the music.\n */\n\n }, {\n key: \"author\",\n get: function get() {\n return _classPrivateFieldGet(this, _author);\n }\n /**\n * @returns {string} The url at which the file for this music can be found.\n */\n\n }, {\n key: \"url\",\n get: function get() {\n return _classPrivateFieldGet(this, _url);\n }\n /**\n * @returns {boolean} true if and only if there is audio data that is either already loaded or is being loaded.\n */\n\n }, {\n key: \"isAudioInstantiated\",\n value: function isAudioInstantiated() {\n return _classPrivateFieldGet(this, _audio) ? true : false;\n }\n /**\n * Begins audio playback as soon as possible.\n * \n * This function uses the {@link PlaylistMusic#getAudio} function and specifically, the {@link AudioEventCallback}.\n */\n\n }, {\n key: \"play\",\n value: function play() {\n this.fetchAudio(function (audio) {\n audio.play();\n });\n }\n /**\n * Pauses the audio playback, unless the audio data has yet to be instantiated.\n */\n\n }, {\n key: \"pause\",\n value: function pause() {\n if (!this.isAudioInstantiated()) return;\n this.fetchAudio(function (audio) {\n audio.pause();\n });\n }\n /**\n * \n * @returns {number} The volume on a scale of 0 to 1.\n */\n\n }, {\n key: \"volume\",\n get: function get() {\n return this.fetchAudio().volume;\n }\n /**\n * \n * The normalized volume on a scale of 0 to 1.\n */\n ,\n set: function set(volume) {\n this.fetchAudio().volume = volume;\n }\n /**\n * \n * @returns {number} The number of seconds into the music.\n */\n\n }, {\n key: \"currentTime\",\n get: function get() {\n return this.fetchAudio().currentTime;\n }\n /**\n * \n * The time position in the music to jump to in seconds.\n */\n ,\n set: function set(currentTime) {\n this.fetchAudio().currentTime = currentTime;\n }\n /**\n * \n * @returns {number} The duration of the music.\n */\n\n }, {\n key: \"duration\",\n get: function get() {\n return this.fetchAudio().duration;\n }\n /**\n * Unloads the audio data.\n * Makes sure the audio is paused.\n * \n * Also calls {@link PlaylistMusic#unloadVisualizer}.\n */\n\n }, {\n key: \"unloadAudio\",\n value: function unloadAudio() {\n if (!this.isAudioInstantiated()) return;\n\n _classPrivateFieldGet(this, _audio).pause();\n\n this.unloadVisualizer();\n\n _classPrivateFieldSet(this, _audio, null);\n\n _classPrivateFieldSet(this, _ready, false);\n }\n /**\n * \n * @param {number} [fftSize=1024] the size of the FFT window.\n * @returns {Visualizer} returns the visualizer.\n */\n\n }, {\n key: \"getVisualizer\",\n value: function getVisualizer() {\n var fftSize = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : 1024;\n if (_classPrivateFieldGet(this, _visualizer) && _classPrivateFieldGet(this, _visualizer).getFftSize() === fftSize) return _classPrivateFieldGet(this, _visualizer);\n\n _classPrivateFieldSet(this, _visualizer, new _visualization_Visualizer_js__WEBPACK_IMPORTED_MODULE_0__[\"default\"](this.fetchAudio(), fftSize));\n\n return _classPrivateFieldGet(this, _visualizer);\n }\n /**\n * \n * @returns {boolean} returns true if and only if the visualizer is instantiated.\n */\n\n }, {\n key: \"isVisualizerInstantiated\",\n value: function isVisualizerInstantiated() {\n return _classPrivateFieldGet(this, _visualizer) ? true : false;\n }\n /**\n * Unloads the visualizer.\n * Stops the visualizer.\n */\n\n }, {\n key: \"unloadVisualizer\",\n value: function unloadVisualizer() {\n if (_classPrivateFieldGet(this, _visualizer)) {\n _classPrivateFieldGet(this, _visualizer).stop();\n\n _classPrivateFieldSet(this, _visualizer, null);\n }\n }\n }]);\n\n return Music;\n}();\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/player/Music.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/player/Music.js?365f"],"sourcesContent":["import Visualizer from \"../visualization/Visualizer.js\";\n\n/**@module */\n\n/**\n * @callback AudioEventCallback\n * @param {HTMLAudioElement} audio\n */\n\n/**\n * A music with metadata that can be used as part of a {@link MusicPlaylist}.\n * \n */\nexport default class Music {\n    #displayName;\n    #author;\n    #url;\n    #audio;\n    #visualizer;\n    #ready = false;\n\n    /**\n     * Constructs a music for a {@link MusicPlaylist}.\n     * \n     * @param {string} url the url to fetch the music from.\n     * @param {string} name the name of the music.\n     * @param {string} author the author of the music.\n     */\n    constructor(url, name, author) {\n        this.#displayName = name;\n        this.#author = author;\n        this.#url = url;\n    }\n\n\n    /**\n     * \n     * @param {AudioEventCallback} [onReady] called when the music is ready, including right away if the music is already ready.\n     * @returns {HTMLAudioElement} The audio element that represents this music.\n     */\n    fetchAudio(onReady) {\n        if (this.#audio) {\n            if (this.#ready) {\n                if (onReady) onReady(this.#audio);\n            } else if (onReady) {\n                this.#audio.addEventListener(\"canplaythrough\", () => {\n                    onReady(this.#audio);\n                }, { once: true });\n            }\n            return this.#audio;\n        }\n        this.#audio = new Audio(this.#url);\n        this.#audio.addEventListener(\"canplaythrough\", () => {\n            this.#ready = true;\n            if (onReady && this.isAudioInstantiated()) {\n                onReady(this.#audio);\n            }\n        }, { once: true });\n        return this.#audio;\n    }\n\n    /**\n     * @returns {string} The name of the music to be displayed.\n     */\n    get displayName() {\n        return this.#displayName;\n    }\n\n    /**\n     * @returns {string} The author of the music.\n     */\n    get author() {\n        return this.#author;\n    }\n\n    /**\n     * @returns {string} The url at which the file for this music can be found.\n     */\n    get url() {\n        return this.#url;\n    }\n\n    /**\n     * @returns {boolean} true if and only if there is audio data that is either already loaded or is being loaded.\n     */\n    isAudioInstantiated() {\n        return this.#audio ? true : false;\n    }\n\n    /**\n     * Begins audio playback as soon as possible.\n     * \n     * This function uses the {@link PlaylistMusic#getAudio} function and specifically, the {@link AudioEventCallback}.\n     */\n    play() {\n        this.fetchAudio((audio) => {\n            audio.play();\n        });\n    }\n\n    /**\n     * Pauses the audio playback, unless the audio data has yet to be instantiated.\n     */\n    pause() {\n        if (!this.isAudioInstantiated()) return;\n        this.fetchAudio((audio) => {\n            audio.pause();\n        });\n    }\n\n    /**\n     * \n     * @returns {number} The volume on a scale of 0 to 1.\n     */\n    get volume() {\n        return this.fetchAudio().volume;\n    }\n\n    /**\n     * \n     * The normalized volume on a scale of 0 to 1.\n     */\n    set volume(volume) {\n        this.fetchAudio().volume = volume;\n    }\n    /**\n     * \n     * @returns {number} The number of seconds into the music.\n     */\n    get currentTime() {\n        return this.fetchAudio().currentTime;\n    }\n\n    /**\n     * \n     * The time position in the music to jump to in seconds.\n     */\n    set currentTime(currentTime) {\n        this.fetchAudio().currentTime = currentTime;\n    }\n\n    /**\n     * \n     * @returns {number} The duration of the music.\n     */\n    get duration() {\n        return this.fetchAudio().duration;\n    }\n\n    /**\n     * Unloads the audio data.\n     * Makes sure the audio is paused.\n     * \n     * Also calls {@link PlaylistMusic#unloadVisualizer}.\n     */\n    unloadAudio() {\n        if (!this.isAudioInstantiated()) return;\n        this.#audio.pause();\n        this.unloadVisualizer();\n        this.#audio = null;\n        this.#ready = false;\n    }\n\n    /**\n     * \n     * @param {number} [fftSize=1024] the size of the FFT window.\n     * @returns {Visualizer} returns the visualizer.\n     */\n    getVisualizer(fftSize = 1024) {\n        if (this.#visualizer && this.#visualizer.getFftSize() === fftSize) return this.#visualizer;\n        this.#visualizer = new Visualizer(this.fetchAudio(), fftSize);\n        return this.#visualizer;\n    }\n\n    /**\n     * \n     * @returns {boolean} returns true if and only if the visualizer is instantiated.\n     */\n    isVisualizerInstantiated() {\n        return this.#visualizer ? true : false;\n    }\n\n    /**\n     * Unloads the visualizer.\n     * Stops the visualizer.\n     */\n    unloadVisualizer() {\n        if (this.#visualizer) {\n            this.#visualizer.stop();\n            this.#visualizer = null;\n        }\n    }\n}"],"names":["Visualizer","Music","url","name","author","onReady","addEventListener","once","Audio","isAudioInstantiated","fetchAudio","audio","play","pause","volume","currentTime","duration","unloadVisualizer","fftSize","getFftSize","stop"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/player/Music.js\n"); + +/***/ }), + +/***/ "./src/player/MusicPlayer.js": +/*!***********************************!*\ + !*** ./src/player/MusicPlayer.js ***! + \***********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ MusicPlayer)\n/* harmony export */ });\n/* harmony import */ var _styles_musicPlayer_css__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../styles/musicPlayer.css */ \"./src/styles/musicPlayer.css\");\n/* harmony import */ var _MusicPlaylist_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./MusicPlaylist.js */ \"./src/player/MusicPlaylist.js\");\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\n\n\n/**@module */\n\n/**\n * @callback changeListener\n * @param {*} old the previous value.\n * @param {*} current the the current (new) value.\n */\n\n/**\n * A player that keeps track of global properties for playback as well as traversing a playlist. Additionally keeps track of music audio data and attempts to reduce memory usage when possible.\n */\n\nvar _playlist = /*#__PURE__*/new WeakMap();\n\nvar _volume = /*#__PURE__*/new WeakMap();\n\nvar _autoplay = /*#__PURE__*/new WeakMap();\n\nvar _wasPlaying = /*#__PURE__*/new WeakMap();\n\nvar _playlistChangeListeners = /*#__PURE__*/new WeakMap();\n\nvar _volumeChangeListeners = /*#__PURE__*/new WeakMap();\n\nvar _allAudioEventListeners = /*#__PURE__*/new WeakMap();\n\nvar _onMusicChange = /*#__PURE__*/new WeakMap();\n\nvar MusicPlayer = /*#__PURE__*/function () {\n /**\n * \n * @param {MusicPlaylist} playlist the playlist of musics that this player is in charge of.\n */\n function MusicPlayer(playlist) {\n var _this = this;\n\n _classCallCheck(this, MusicPlayer);\n\n _classPrivateFieldInitSpec(this, _playlist, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _volume, {\n writable: true,\n value: 1\n });\n\n _classPrivateFieldInitSpec(this, _autoplay, {\n writable: true,\n value: false\n });\n\n _classPrivateFieldInitSpec(this, _wasPlaying, {\n writable: true,\n value: false\n });\n\n _classPrivateFieldInitSpec(this, _playlistChangeListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _volumeChangeListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _allAudioEventListeners, {\n writable: true,\n value: {}\n });\n\n _classPrivateFieldInitSpec(this, _onMusicChange, {\n writable: true,\n value: function value(old, current) {\n // Anonymous to avoid overriding \"this\".\n var oldMusic = _this.playlist.musicAtIndex(old);\n\n var currentMusic = _this.playlist.musicAtIndex(current);\n\n Object.keys(_classPrivateFieldGet(_this, _allAudioEventListeners)).forEach(function (key) {\n var listeners = _classPrivateFieldGet(_this, _allAudioEventListeners)[key];\n\n listeners.forEach(function (listener) {\n oldMusic.fetchAudio().removeEventListener(key, listener);\n });\n });\n oldMusic.unloadAudio();\n Object.keys(_classPrivateFieldGet(_this, _allAudioEventListeners)).forEach(function (key) {\n var listeners = _classPrivateFieldGet(_this, _allAudioEventListeners)[key];\n\n listeners.forEach(function (listener) {\n currentMusic.fetchAudio().addEventListener(key, listener);\n });\n });\n\n if (_classPrivateFieldGet(_this, _wasPlaying) || _classPrivateFieldGet(_this, _autoplay)) {\n _this.playCurrent();\n }\n\n return true;\n }\n });\n\n this.playlist = playlist;\n this.addEventListenerToCurrentAudio(\"ended\", function () {\n _this.next();\n });\n }\n /**\n * The new playlist of musics that this player is in charge of.\n */\n\n\n _createClass(MusicPlayer, [{\n key: \"playlist\",\n get:\n /**\n * The current playlist of musics that this player is in charge of.\n * \n * @returns {MusicPlaylist} The music playlist this player is currently using.\n */\n function get() {\n return _classPrivateFieldGet(this, _playlist);\n }\n /**\n * \n * @returns {boolean} true if and only if successful in going to the next music.\n */\n ,\n set: function set(playlist) {\n var _this2 = this;\n\n if (this.playlist) this.unloadAllAudio();\n if (this.playlist) this.playlist.removePositionChangeListener(_classPrivateFieldGet(this, _onMusicChange));\n\n var old = _classPrivateFieldGet(this, _playlist);\n\n _classPrivateFieldSet(this, _playlist, playlist);\n\n _classPrivateFieldGet(this, _playlist).addPositionChangeListener(_classPrivateFieldGet(this, _onMusicChange));\n\n _classPrivateFieldGet(this, _playlist).currentPosition = 0; // Updates the listener.\n\n _classPrivateFieldGet(this, _playlistChangeListeners).forEach(function (playlistChangeListener) {\n playlistChangeListener(old, _this2.getPlaylist());\n });\n }\n }, {\n key: \"next\",\n value: function next() {\n return this.playlist.currentPosition += 1;\n }\n /**\n * attempts to go to the previous music.\n * \n * @returns {boolean} true if and only if successful in going to the previous music.\n */\n\n }, {\n key: \"previous\",\n value: function previous() {\n return this.playlist.currentPosition -= 1;\n }\n /**\n * @param {number} old the index of the previous music.\n * @param {number} current the index of the current music.\n * @returns {boolean} true if and only if successful jumping to the given index.\n */\n\n }, {\n key: \"playCurrent\",\n value: function playCurrent() {\n var _this3 = this;\n\n this.playlist.currentMusic.fetchAudio(function (audio) {\n audio.volume = _this3.volume;\n audio.play();\n });\n\n _classPrivateFieldSet(this, _wasPlaying, true);\n }\n /**\n * Pauses the current playing music (if there is one playing).\n */\n\n }, {\n key: \"pauseCurrent\",\n value: function pauseCurrent() {\n this.playlist.currentMusic.fetchAudio().pause();\n }\n /**\n * Toggles whether or not the current music is playing.\n */\n\n }, {\n key: \"togglePlay\",\n value: function togglePlay() {\n if (this.playing) {\n this.pauseCurrent();\n } else {\n this.playCurrent();\n }\n }\n /**\n * A number between 0 to 1 inclusive representing the volume of the player. Values out of these bounds will be rounded to the nearest bound.\n */\n\n }, {\n key: \"volume\",\n get:\n /**\n * \n * @returns {number} The current volume of the player represented by a number between 0 and 1 inclusive.\n */\n function get() {\n return _classPrivateFieldGet(this, _volume);\n }\n /**\n * Attempts to seek to the given position.\n * \n * @param {number} position the position to seek to.\n * @returns {boolean} true if and only if the position to seek to is within the duration of the track. This also means that if the track has not finished loading the duration data, than this will always return false.\n */\n ,\n set: function set(volume) {\n if (volume > 1) volume = 1;\n if (volume < 0) volume = 0;\n\n _classPrivateFieldSet(this, _volume, volume);\n\n this.playlist.currentMusic.fetchAudio().volume = _classPrivateFieldGet(this, _volume);\n }\n }, {\n key: \"seek\",\n value: function seek(position) {\n if (position > this.playlist.playlist.currentMusic.fetchAudio().duration || position < 0) return false;\n this.playlist.playlist.currentMusic.fetchAudio(function (audio) {\n audio.currentTime = position;\n });\n return true;\n }\n /**\n * \n * @returns {number} How many seconds into the audio track has been played in seconds.\n */\n\n }, {\n key: \"currentPosition\",\n get: function get() {\n return this.playlist.playlist.currentMusic.fetchAudio().currentTime;\n }\n /**\n * \n * @returns {number} The total length of the music, or NaN if this information has not loaded yet.\n */\n\n }, {\n key: \"currentDuration\",\n get: function get() {\n return this.playlist.playlist.currentMusic.fetchAudio().duration;\n }\n /**\n * Convenience method for creating a play button complete with CSS for transitioning to a paused button.\n * \n * Automatically hooked up to the player's state change updates and play and pause functions.\n * \n * @returns {HTMLElement} the play button element that can be added to a DOM.\n */\n\n }, {\n key: \"generatePlayElement\",\n value: function generatePlayElement() {\n var _this4 = this;\n\n var playButton = document.createElement(\"button\");\n playButton.classList.add(\"player\");\n playButton.classList.add(\"play-btn\");\n if (this.playing) playButton.classList.add(\"pause\");\n playButton.addEventListener(\"click\", function () {\n _this4.togglePlay();\n });\n this.addEventListenerToCurrentAudio(\"play\", function () {\n playButton.classList.add(\"pause\");\n });\n this.addEventListenerToCurrentAudio(\"pause\", function () {\n playButton.classList.remove(\"pause\");\n });\n return playButton;\n }\n }, {\n key: \"generateNextElement\",\n value: function generateNextElement() {\n var _this5 = this;\n\n var nextButton = document.createElement(\"button\");\n nextButton.classList.add(\"player\");\n nextButton.classList.add(\"next\");\n nextButton.addEventListener(\"click\", function () {\n _this5.next();\n });\n return nextButton;\n }\n }, {\n key: \"generatePreviousElement\",\n value: function generatePreviousElement() {\n var _this6 = this;\n\n var previousButton = document.createElement(\"button\");\n previousButton.classList.add(\"player\");\n previousButton.classList.add(\"previous\");\n previousButton.addEventListener(\"click\", function () {\n _this6.previous();\n });\n return previousButton;\n }\n /**\n * \n * @param {changeListener} listener the playlist change listener to add to the callback list.\n * @returns {boolean} true if and only if the given listener was successfully registered.\n */\n\n }, {\n key: \"addPlaylistChangeListener\",\n value: function addPlaylistChangeListener(listener) {\n if (_classPrivateFieldGet(this, _playlistChangeListeners).includes(listener)) return false;\n\n _classPrivateFieldGet(this, _playlistChangeListeners).push(listener);\n\n return true;\n }\n /**\n * \n * @param {changeListener} listener the playlist change listener to remove.\n * @returns {boolean} true if and only if a listener was successfully removed from the callback list.\n */\n\n }, {\n key: \"removePlaylistChangeListener\",\n value: function removePlaylistChangeListener(listener) {\n var removeIndex = _classPrivateFieldGet(this, _playlistChangeListeners).indexOf(listener);\n\n if (removeIndex < 0) return false;\n this.playlistChangeListener.splice(removeIndex, 1);\n return true;\n }\n /**\n * \n * @param {changeListener} listener the listener that is called when the player's volume is changed.\n * @returns {boolean} true if and only if the listener was successfully added.\n */\n\n }, {\n key: \"addVolumeChangeListener\",\n value: function addVolumeChangeListener(listener) {\n if (_classPrivateFieldGet(this, _volumeChangeListeners).includes(listener)) return false;\n\n _classPrivateFieldGet(this, _volumeChangeListeners).push(listener);\n\n return true;\n }\n /**\n * \n * @param {changeListener} listener the volume change listener to remove.\n * @returns {boolean} true if and only if a listener was successfully removed from the callback list.\n */\n\n }, {\n key: \"removeVolumeChangeListener\",\n value: function removeVolumeChangeListener(listener) {\n var removeIndex = _classPrivateFieldGet(this, _volumeChangeListeners).indexOf(listener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _volumeChangeListeners).splice(removeIndex, 1);\n\n return true;\n }\n /**\n * \n * @param {string} type the type of the listener on the {@link HTMLAudioElement}.\n * @param {Function} eventListener the event listener.\n * @returns {boolean} true if and only if successfully registered event listener.\n */\n\n }, {\n key: \"addEventListenerToCurrentAudio\",\n value: function addEventListenerToCurrentAudio(type, eventListener) {\n var typeListeners = _classPrivateFieldGet(this, _allAudioEventListeners)[type];\n\n if (!typeListeners) {\n typeListeners = [];\n _classPrivateFieldGet(this, _allAudioEventListeners)[type] = typeListeners;\n }\n\n if (typeListeners.includes(eventListener)) return false;\n typeListeners.push(eventListener);\n this.playlist.currentMusic.fetchAudio().addEventListener(type, eventListener);\n return true;\n }\n /**\n * \n * @param {string} type the type of the listener on the {@link HTMLAudioElement}. \n * @param {Function} eventListener the event listener. \n * @returns {boolean} true if and only if the event listener was successfully added.\n */\n\n }, {\n key: \"removeEventListenerFromCurrentAudio\",\n value: function removeEventListenerFromCurrentAudio(type, eventListener) {\n var typeListeners = _classPrivateFieldGet(this, _allAudioEventListeners)[type];\n\n if (!typeListeners) return false;\n var removeIndex = typeListeners.indexOf(eventListener);\n if (removeIndex < 0) return false;\n typeListeners.splice(removeIndex, 1);\n this.playlist.playlist.currentMusic.fetchAudio().removeEventListener(type, eventListener);\n return true;\n }\n /**\n * @returns {boolean} If the player is playing any musics right now.\n */\n\n }, {\n key: \"playing\",\n get: function get() {\n return this.playlist.currentMusic.isAudioInstantiated() && !this.playlist.currentMusic.fetchAudio().paused;\n }\n /**\n * Unloads the audio data of all musics in this playlist.\n */\n\n }, {\n key: \"unloadAllAudio\",\n value: function unloadAllAudio() {\n this.playlist.forEach(function (music) {\n music.unloadAudio();\n });\n }\n }]);\n\n return MusicPlayer;\n}();\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/player/MusicPlayer.js.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;AACA;AAEA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;;;;;;;;;;;;;;;;;;IACqBC;AASjB;AACJ;AACA;AACA;AACI,uBAAYC,QAAZ,EAAsB;AAAA;;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA,aAXZ;AAWY;;AAAA;AAAA;AAAA,aAVV;AAUU;;AAAA;AAAA;AAAA,aATR;AASQ;;AAAA;AAAA;AAAA,aARK;AAQL;;AAAA;AAAA;AAAA,aAPG;AAOH;;AAAA;AAAA;AAAA,aANI;AAMJ;;AAAA;AAAA;AAAA,aAsDL,eAACC,GAAD,EAAMC,OAAN,EAAkB;AAAE;AACjC,YAAMC,QAAQ,GAAG,KAAI,CAACH,QAAL,CAAcI,YAAd,CAA2BH,GAA3B,CAAjB;;AACA,YAAMI,YAAY,GAAG,KAAI,CAACL,QAAL,CAAcI,YAAd,CAA2BF,OAA3B,CAArB;;AACAI,QAAAA,MAAM,CAACC,IAAP,uBAAY,KAAZ,4BAA0CC,OAA1C,CAAkD,UAAAC,GAAG,EAAI;AACrD,cAAMC,SAAS,GAAG,2BAAI,0BAAJ,CAA6BD,GAA7B,CAAlB;;AACAC,UAAAA,SAAS,CAACF,OAAV,CAAkB,UAAAG,QAAQ,EAAI;AAC1BR,YAAAA,QAAQ,CAACS,UAAT,GAAsBC,mBAAtB,CAA0CJ,GAA1C,EAA+CE,QAA/C;AACH,WAFD;AAGH,SALD;AAMAR,QAAAA,QAAQ,CAACW,WAAT;AAEAR,QAAAA,MAAM,CAACC,IAAP,uBAAY,KAAZ,4BAA0CC,OAA1C,CAAkD,UAAAC,GAAG,EAAI;AACrD,cAAMC,SAAS,GAAG,2BAAI,0BAAJ,CAA6BD,GAA7B,CAAlB;;AACAC,UAAAA,SAAS,CAACF,OAAV,CAAkB,UAAAG,QAAQ,EAAI;AAC1BN,YAAAA,YAAY,CAACO,UAAb,GAA0BG,gBAA1B,CAA2CN,GAA3C,EAAgDE,QAAhD;AACH,WAFD;AAGH,SALD;;AAMA,YAAI,2BAAI,cAAJ,0BAAoB,KAApB,YAAJ,EAAwC;AACpC,eAAI,CAACK,WAAL;AACH;;AACD,eAAO,IAAP;AACH;AA3EqB;;AAClB,SAAKhB,QAAL,GAAgBA,QAAhB;AACA,SAAKiB,8BAAL,CAAoC,OAApC,EAA6C,YAAM;AAC/C,WAAI,CAACC,IAAL;AACH,KAFD;AAGH;AAED;AACJ;AACA;;;;;;AAcI;AACJ;AACA;AACA;AACA;AACI,mBAAe;AACX,mCAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;;SAzBI,aAAalB,QAAb,EAAuB;AAAA;;AACnB,UAAI,KAAKA,QAAT,EAAmB,KAAKmB,cAAL;AACnB,UAAI,KAAKnB,QAAT,EAAmB,KAAKA,QAAL,CAAcoB,4BAAd,uBAA2C,IAA3C;;AACnB,UAAMnB,GAAG,yBAAG,IAAH,YAAT;;AACA,6CAAiBD,QAAjB;;AACA,6CAAeqB,yBAAf,uBAAyC,IAAzC;;AACA,6CAAeC,eAAf,GAAiC,CAAjC,CANmB,CAMiB;;AACpC,4DAA8Bd,OAA9B,CAAsC,UAAAe,sBAAsB,EAAI;AAC5DA,QAAAA,sBAAsB,CAACtB,GAAD,EAAM,MAAI,CAACuB,WAAL,EAAN,CAAtB;AACH,OAFD;AAIH;;;WAeD,gBAAO;AACH,aAAO,KAAKxB,QAAL,CAAcsB,eAAd,IAAiC,CAAxC;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,oBAAW;AACP,aAAO,KAAKtB,QAAL,CAAcsB,eAAd,IAAiC,CAAxC;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WAwBI,uBAAc;AAAA;;AACV,WAAKtB,QAAL,CAAcK,YAAd,CAA2BO,UAA3B,CAAsC,UAACa,KAAD,EAAW;AAC7CA,QAAAA,KAAK,CAACC,MAAN,GAAe,MAAI,CAACA,MAApB;AACAD,QAAAA,KAAK,CAACE,IAAN;AACH,OAHD;;AAIA,+CAAmB,IAAnB;AACH;AAED;AACJ;AACA;;;;WACI,wBAAe;AACX,WAAK3B,QAAL,CAAcK,YAAd,CAA2BO,UAA3B,GAAwCgB,KAAxC;AACH;AAED;AACJ;AACA;;;;WACI,sBAAa;AACT,UAAI,KAAKC,OAAT,EAAkB;AACd,aAAKC,YAAL;AACH,OAFD,MAEO;AACH,aAAKd,WAAL;AACH;AACJ;AAED;AACJ;AACA;;;;;AAQI;AACJ;AACA;AACA;AACI,mBAAa;AACT,mCAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;AACA;AACA;;SApBI,aAAWU,MAAX,EAAmB;AACf,UAAIA,MAAM,GAAG,CAAb,EAAgBA,MAAM,GAAG,CAAT;AAChB,UAAIA,MAAM,GAAG,CAAb,EAAgBA,MAAM,GAAG,CAAT;;AAChB,2CAAeA,MAAf;;AACA,WAAK1B,QAAL,CAAcK,YAAd,CAA2BO,UAA3B,GAAwCc,MAAxC,yBAAiD,IAAjD;AACH;;;WAgBD,cAAKK,QAAL,EAAe;AACX,UAAIA,QAAQ,GAAG,KAAK/B,QAAL,CAAcA,QAAd,CAAuBK,YAAvB,CAAoCO,UAApC,GAAiDoB,QAA5D,IAAwED,QAAQ,GAAG,CAAvF,EAA0F,OAAO,KAAP;AAC1F,WAAK/B,QAAL,CAAcA,QAAd,CAAuBK,YAAvB,CAAoCO,UAApC,CAA+C,UAAAa,KAAK,EAAI;AACpDA,QAAAA,KAAK,CAACQ,WAAN,GAAoBF,QAApB;AACH,OAFD;AAGA,aAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;;;;SACI,eAAsB;AAClB,aAAO,KAAK/B,QAAL,CAAcA,QAAd,CAAuBK,YAAvB,CAAoCO,UAApC,GAAiDqB,WAAxD;AACH;AAED;AACJ;AACA;AACA;;;;SACI,eAAsB;AAClB,aAAO,KAAKjC,QAAL,CAAcA,QAAd,CAAuBK,YAAvB,CAAoCO,UAApC,GAAiDoB,QAAxD;AACH;AAED;AACJ;AACA;AACA;AACA;AACA;AACA;;;;WACI,+BAAsB;AAAA;;AAClB,UAAME,UAAU,GAAGC,QAAQ,CAACC,aAAT,CAAuB,QAAvB,CAAnB;AACAF,MAAAA,UAAU,CAACG,SAAX,CAAqBC,GAArB,CAAyB,QAAzB;AACAJ,MAAAA,UAAU,CAACG,SAAX,CAAqBC,GAArB,CAAyB,UAAzB;AACA,UAAI,KAAKT,OAAT,EAAkBK,UAAU,CAACG,SAAX,CAAqBC,GAArB,CAAyB,OAAzB;AAElBJ,MAAAA,UAAU,CAACnB,gBAAX,CAA4B,OAA5B,EAAqC,YAAM;AACvC,cAAI,CAACwB,UAAL;AACH,OAFD;AAIA,WAAKtB,8BAAL,CAAoC,MAApC,EAA4C,YAAM;AAC9CiB,QAAAA,UAAU,CAACG,SAAX,CAAqBC,GAArB,CAAyB,OAAzB;AACH,OAFD;AAGA,WAAKrB,8BAAL,CAAoC,OAApC,EAA6C,YAAM;AAC/CiB,QAAAA,UAAU,CAACG,SAAX,CAAqBG,MAArB,CAA4B,OAA5B;AACH,OAFD;AAGA,aAAON,UAAP;AACH;;;WAED,+BAAsB;AAAA;;AAClB,UAAMO,UAAU,GAAGN,QAAQ,CAACC,aAAT,CAAuB,QAAvB,CAAnB;AACAK,MAAAA,UAAU,CAACJ,SAAX,CAAqBC,GAArB,CAAyB,QAAzB;AACAG,MAAAA,UAAU,CAACJ,SAAX,CAAqBC,GAArB,CAAyB,MAAzB;AACAG,MAAAA,UAAU,CAAC1B,gBAAX,CAA4B,OAA5B,EAAqC,YAAM;AACvC,cAAI,CAACG,IAAL;AACH,OAFD;AAGA,aAAOuB,UAAP;AACH;;;WAED,mCAA0B;AAAA;;AACtB,UAAMC,cAAc,GAAGP,QAAQ,CAACC,aAAT,CAAuB,QAAvB,CAAvB;AACAM,MAAAA,cAAc,CAACL,SAAf,CAAyBC,GAAzB,CAA6B,QAA7B;AACAI,MAAAA,cAAc,CAACL,SAAf,CAAyBC,GAAzB,CAA6B,UAA7B;AACAI,MAAAA,cAAc,CAAC3B,gBAAf,CAAgC,OAAhC,EAAyC,YAAM;AAC3C,cAAI,CAAC4B,QAAL;AACH,OAFD;AAGA,aAAOD,cAAP;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,mCAA0B/B,QAA1B,EAAoC;AAChC,UAAI,sDAA8BiC,QAA9B,CAAuCjC,QAAvC,CAAJ,EAAsD,OAAO,KAAP;;AACtD,4DAA8BkC,IAA9B,CAAmClC,QAAnC;;AACA,aAAO,IAAP;AAEH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,sCAA6BA,QAA7B,EAAuC;AACnC,UAAMmC,WAAW,GAAG,sDAA8BC,OAA9B,CAAsCpC,QAAtC,CAApB;;AACA,UAAImC,WAAW,GAAG,CAAlB,EAAqB,OAAO,KAAP;AAErB,WAAKvB,sBAAL,CAA4ByB,MAA5B,CAAmCF,WAAnC,EAAgD,CAAhD;AACA,aAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,iCAAwBnC,QAAxB,EAAkC;AAC9B,UAAI,oDAA4BiC,QAA5B,CAAqCjC,QAArC,CAAJ,EAAoD,OAAO,KAAP;;AACpD,0DAA4BkC,IAA5B,CAAiClC,QAAjC;;AACA,aAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,oCAA2BA,QAA3B,EAAqC;AACjC,UAAMmC,WAAW,GAAG,oDAA4BC,OAA5B,CAAoCpC,QAApC,CAApB;;AACA,UAAImC,WAAW,GAAG,CAAlB,EAAqB,OAAO,KAAP;;AAErB,0DAA4BE,MAA5B,CAAmCF,WAAnC,EAAgD,CAAhD;;AACA,aAAO,IAAP;AACH;AAGD;AACJ;AACA;AACA;AACA;AACA;;;;WACI,wCAA+BG,IAA/B,EAAqCC,aAArC,EAAoD;AAChD,UAAIC,aAAa,GAAG,qDAA6BF,IAA7B,CAApB;;AACA,UAAI,CAACE,aAAL,EAAoB;AAChBA,QAAAA,aAAa,GAAG,EAAhB;AACA,6DAA6BF,IAA7B,IAAqCE,aAArC;AACH;;AACD,UAAIA,aAAa,CAACP,QAAd,CAAuBM,aAAvB,CAAJ,EAA2C,OAAO,KAAP;AAC3CC,MAAAA,aAAa,CAACN,IAAd,CAAmBK,aAAnB;AACA,WAAKlD,QAAL,CAAcK,YAAd,CAA2BO,UAA3B,GAAwCG,gBAAxC,CAAyDkC,IAAzD,EAA+DC,aAA/D;AACA,aAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;AACA;AACA;;;;WACI,6CAAoCD,IAApC,EAA0CC,aAA1C,EAAyD;AACrD,UAAIC,aAAa,GAAG,qDAA6BF,IAA7B,CAApB;;AACA,UAAI,CAACE,aAAL,EAAoB,OAAO,KAAP;AACpB,UAAML,WAAW,GAAGK,aAAa,CAACJ,OAAd,CAAsBG,aAAtB,CAApB;AACA,UAAIJ,WAAW,GAAG,CAAlB,EAAqB,OAAO,KAAP;AACrBK,MAAAA,aAAa,CAACH,MAAd,CAAqBF,WAArB,EAAkC,CAAlC;AACA,WAAK9C,QAAL,CAAcA,QAAd,CAAuBK,YAAvB,CAAoCO,UAApC,GAAiDC,mBAAjD,CAAqEoC,IAArE,EAA2EC,aAA3E;AACA,aAAO,IAAP;AACH;AAED;AACJ;AACA;;;;SACI,eAAc;AACV,aAAO,KAAKlD,QAAL,CAAcK,YAAd,CAA2B+C,mBAA3B,MAAoD,CAAC,KAAKpD,QAAL,CAAcK,YAAd,CAA2BO,UAA3B,GAAwCyC,MAApG;AACH;AAED;AACJ;AACA;;;;WACI,0BAAiB;AACb,WAAKrD,QAAL,CAAcQ,OAAd,CAAsB,UAAA8C,KAAK,EAAI;AAC3BA,QAAAA,KAAK,CAACxC,WAAN;AACH,OAFD;AAGH","sources":["webpack://audioshowkit/./src/player/MusicPlayer.js?9282"],"sourcesContent":["import \"../styles/musicPlayer.css\";\nimport MusicPlaylist from \"./MusicPlaylist.js\";\n\n/**@module */\n\n/**\n * @callback changeListener\n * @param {*} old the previous value.\n * @param {*} current the the current (new) value.\n */\n\n/**\n * A player that keeps track of global properties for playback as well as traversing a playlist. Additionally keeps track of music audio data and attempts to reduce memory usage when possible.\n */\nexport default class MusicPlayer {\n    #playlist;\n    #volume = 1;\n    #autoplay = false;\n    #wasPlaying = false;\n    #playlistChangeListeners = [];\n    #volumeChangeListeners = [];\n    #allAudioEventListeners = {};\n\n    /**\n     * \n     * @param {MusicPlaylist} playlist the playlist of musics that this player is in charge of.\n     */\n    constructor(playlist) {\n        this.playlist = playlist;\n        this.addEventListenerToCurrentAudio(\"ended\", () => {\n            this.next();\n        });\n    }\n\n    /**\n     * The new playlist of musics that this player is in charge of.\n     */\n    set playlist(playlist) {\n        if (this.playlist) this.unloadAllAudio();\n        if (this.playlist) this.playlist.removePositionChangeListener(this.#onMusicChange);\n        const old = this.#playlist;\n        this.#playlist = playlist;\n        this.#playlist.addPositionChangeListener(this.#onMusicChange);\n        this.#playlist.currentPosition = 0; // Updates the listener.\n        this.#playlistChangeListeners.forEach(playlistChangeListener => {\n            playlistChangeListener(old, this.getPlaylist());\n        });\n\n    }\n\n    /**\n     * The current playlist of musics that this player is in charge of.\n     * \n     * @returns {MusicPlaylist} The music playlist this player is currently using.\n     */\n    get playlist() {\n        return this.#playlist;\n    }\n\n    /**\n     * \n     * @returns {boolean} true if and only if successful in going to the next music.\n     */\n    next() {\n        return this.playlist.currentPosition += 1;\n    }\n\n    /**\n     * attempts to go to the previous music.\n     * \n     * @returns {boolean} true if and only if successful in going to the previous music.\n     */\n    previous() {\n        return this.playlist.currentPosition -= 1;\n    }\n\n    /**\n     * @param {number} old the index of the previous music.\n     * @param {number} current the index of the current music.\n     * @returns {boolean} true if and only if successful jumping to the given index.\n     */\n    #onMusicChange = (old, current) => { // Anonymous to avoid overriding \"this\".\n        const oldMusic = this.playlist.musicAtIndex(old);\n        const currentMusic = this.playlist.musicAtIndex(current);\n        Object.keys(this.#allAudioEventListeners).forEach(key => {\n            const listeners = this.#allAudioEventListeners[key];\n            listeners.forEach(listener => {\n                oldMusic.fetchAudio().removeEventListener(key, listener);\n            });\n        });\n        oldMusic.unloadAudio();\n\n        Object.keys(this.#allAudioEventListeners).forEach(key => {\n            const listeners = this.#allAudioEventListeners[key];\n            listeners.forEach(listener => {\n                currentMusic.fetchAudio().addEventListener(key, listener);\n            });\n        });\n        if (this.#wasPlaying || this.#autoplay) {\n            this.playCurrent();\n        }\n        return true;\n    };\n\n    playCurrent() {\n        this.playlist.currentMusic.fetchAudio((audio) => {\n            audio.volume = this.volume;\n            audio.play();\n        });\n        this.#wasPlaying = true;\n    }\n\n    /**\n     * Pauses the current playing music (if there is one playing).\n     */\n    pauseCurrent() {\n        this.playlist.currentMusic.fetchAudio().pause();\n    }\n\n    /**\n     * Toggles whether or not the current music is playing.\n     */\n    togglePlay() {\n        if (this.playing) {\n            this.pauseCurrent();\n        } else {\n            this.playCurrent();\n        }\n    }\n\n    /**\n     * A number between 0 to 1 inclusive representing the volume of the player. Values out of these bounds will be rounded to the nearest bound.\n     */\n    set volume(volume) {\n        if (volume > 1) volume = 1;\n        if (volume < 0) volume = 0;\n        this.#volume = volume;\n        this.playlist.currentMusic.fetchAudio().volume = this.#volume;\n    }\n\n    /**\n     * \n     * @returns {number} The current volume of the player represented by a number between 0 and 1 inclusive.\n     */\n    get volume() {\n        return this.#volume;\n    }\n\n    /**\n     * Attempts to seek to the given position.\n     * \n     * @param {number} position the position to seek to.\n     * @returns {boolean} true if and only if the position to seek to is within the duration of the track. This also means that if the track has not finished loading the duration data, than this will always return false.\n     */\n    seek(position) {\n        if (position > this.playlist.playlist.currentMusic.fetchAudio().duration || position < 0) return false;\n        this.playlist.playlist.currentMusic.fetchAudio(audio => {\n            audio.currentTime = position;\n        });\n        return true;\n    }\n\n    /**\n     * \n     * @returns {number} How many seconds into the audio track has been played in seconds.\n     */\n    get currentPosition() {\n        return this.playlist.playlist.currentMusic.fetchAudio().currentTime;\n    }\n\n    /**\n     * \n     * @returns {number} The total length of the music, or NaN if this information has not loaded yet.\n     */\n    get currentDuration() {\n        return this.playlist.playlist.currentMusic.fetchAudio().duration;\n    }\n\n    /**\n     * Convenience method for creating a play button complete with CSS for transitioning to a paused button.\n     * \n     * Automatically hooked up to the player's state change updates and play and pause functions.\n     * \n     * @returns {HTMLElement} the play button element that can be added to a DOM.\n     */\n    generatePlayElement() {\n        const playButton = document.createElement(\"button\");\n        playButton.classList.add(\"player\");\n        playButton.classList.add(\"play-btn\");\n        if (this.playing) playButton.classList.add(\"pause\");\n\n        playButton.addEventListener(\"click\", () => {\n            this.togglePlay();\n        });\n\n        this.addEventListenerToCurrentAudio(\"play\", () => {\n            playButton.classList.add(\"pause\");\n        });\n        this.addEventListenerToCurrentAudio(\"pause\", () => {\n            playButton.classList.remove(\"pause\");\n        });\n        return playButton;\n    }\n\n    generateNextElement() {\n        const nextButton = document.createElement(\"button\");\n        nextButton.classList.add(\"player\");\n        nextButton.classList.add(\"next\");\n        nextButton.addEventListener(\"click\", () => {\n            this.next();\n        });\n        return nextButton;\n    }\n\n    generatePreviousElement() {\n        const previousButton = document.createElement(\"button\");\n        previousButton.classList.add(\"player\");\n        previousButton.classList.add(\"previous\");\n        previousButton.addEventListener(\"click\", () => {\n            this.previous();\n        });\n        return previousButton;\n    }\n\n    /**\n     * \n     * @param {changeListener} listener the playlist change listener to add to the callback list.\n     * @returns {boolean} true if and only if the given listener was successfully registered.\n     */\n    addPlaylistChangeListener(listener) {\n        if (this.#playlistChangeListeners.includes(listener)) return false;\n        this.#playlistChangeListeners.push(listener);\n        return true;\n\n    }\n\n    /**\n     * \n     * @param {changeListener} listener the playlist change listener to remove.\n     * @returns {boolean} true if and only if a listener was successfully removed from the callback list.\n     */\n    removePlaylistChangeListener(listener) {\n        const removeIndex = this.#playlistChangeListeners.indexOf(listener);\n        if (removeIndex < 0) return false;\n\n        this.playlistChangeListener.splice(removeIndex, 1);\n        return true;\n    }\n\n    /**\n     * \n     * @param {changeListener} listener the listener that is called when the player's volume is changed.\n     * @returns {boolean} true if and only if the listener was successfully added.\n     */\n    addVolumeChangeListener(listener) {\n        if (this.#volumeChangeListeners.includes(listener)) return false;\n        this.#volumeChangeListeners.push(listener);\n        return true;\n    }\n\n    /**\n     * \n     * @param {changeListener} listener the volume change listener to remove.\n     * @returns {boolean} true if and only if a listener was successfully removed from the callback list.\n     */\n    removeVolumeChangeListener(listener) {\n        const removeIndex = this.#volumeChangeListeners.indexOf(listener);\n        if (removeIndex < 0) return false;\n\n        this.#volumeChangeListeners.splice(removeIndex, 1);\n        return true;\n    }\n\n\n    /**\n     * \n     * @param {string} type the type of the listener on the {@link HTMLAudioElement}.\n     * @param {Function} eventListener the event listener.\n     * @returns {boolean} true if and only if successfully registered event listener.\n     */\n    addEventListenerToCurrentAudio(type, eventListener) {\n        let typeListeners = this.#allAudioEventListeners[type];\n        if (!typeListeners) {\n            typeListeners = [];\n            this.#allAudioEventListeners[type] = typeListeners;\n        }\n        if (typeListeners.includes(eventListener)) return false;\n        typeListeners.push(eventListener);\n        this.playlist.currentMusic.fetchAudio().addEventListener(type, eventListener);\n        return true;\n    }\n\n    /**\n     * \n     * @param {string} type the type of the listener on the {@link HTMLAudioElement}.  \n     * @param {Function} eventListener the event listener. \n     * @returns {boolean} true if and only if the event listener was successfully added.\n     */\n    removeEventListenerFromCurrentAudio(type, eventListener) {\n        let typeListeners = this.#allAudioEventListeners[type];\n        if (!typeListeners) return false;\n        const removeIndex = typeListeners.indexOf(eventListener);\n        if (removeIndex < 0) return false;\n        typeListeners.splice(removeIndex, 1);\n        this.playlist.playlist.currentMusic.fetchAudio().removeEventListener(type, eventListener);\n        return true;\n    }\n\n    /**\n     * @returns {boolean} If the player is playing any musics right now.\n     */\n    get playing() {\n        return this.playlist.currentMusic.isAudioInstantiated() && !this.playlist.currentMusic.fetchAudio().paused;\n    }\n\n    /**\n     * Unloads the audio data of all musics in this playlist.\n     */\n    unloadAllAudio() {\n        this.playlist.forEach(music => {\n            music.unloadAudio();\n        });\n    }\n}"],"names":["MusicPlaylist","MusicPlayer","playlist","old","current","oldMusic","musicAtIndex","currentMusic","Object","keys","forEach","key","listeners","listener","fetchAudio","removeEventListener","unloadAudio","addEventListener","playCurrent","addEventListenerToCurrentAudio","next","unloadAllAudio","removePositionChangeListener","addPositionChangeListener","currentPosition","playlistChangeListener","getPlaylist","audio","volume","play","pause","playing","pauseCurrent","position","duration","currentTime","playButton","document","createElement","classList","add","togglePlay","remove","nextButton","previousButton","previous","includes","push","removeIndex","indexOf","splice","type","eventListener","typeListeners","isAudioInstantiated","paused","music"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/player/MusicPlayer.js\n"); + +/***/ }), + +/***/ "./src/player/MusicPlaylist.js": +/*!*************************************!*\ + !*** ./src/player/MusicPlaylist.js ***! + \*************************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ MusicPlaylist)\n/* harmony export */ });\n/* harmony import */ var _Music_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Music.js */ \"./src/player/Music.js\");\nvar _Symbol$iterator;\n\nfunction _createForOfIteratorHelper(o, allowArrayLike) { var it = typeof Symbol !== \"undefined\" && o[Symbol.iterator] || o[\"@@iterator\"]; if (!it) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === \"number\") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError(\"Invalid attempt to iterate non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\"); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = it.call(o); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it[\"return\"] != null) it[\"return\"](); } finally { if (didErr) throw err; } } }; }\n\nfunction _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === \"string\") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === \"Object\" && o.constructor) n = o.constructor.name; if (n === \"Map\" || n === \"Set\") return Array.from(o); if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }\n\nfunction _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _classPrivateMethodInitSpec(obj, privateSet) { _checkPrivateRedeclaration(obj, privateSet); privateSet.add(obj); }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateMethodGet(receiver, privateSet, fn) { if (!privateSet.has(receiver)) { throw new TypeError(\"attempted to get private field on non-instance\"); } return fn; }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\n\n/**@module */\n\n/**\n * @callback positionChangeListener\n * @param {number} old the previous position.\n * @param {number} current the the current (new) position.\n */\n\n/**\n * @callback lengthChangeListener\n * @param {number} changed The index of the music that has changed.\n * @param {boolean} added True if and only if the change was caused by adding music.\n * @param {Music} music The music that was added or removed.\n */\n\n/**\n * A playlist that holds a multitude of musics in the form of {@link Music}.\n */\n\nvar _list = /*#__PURE__*/new WeakMap();\n\nvar _name = /*#__PURE__*/new WeakMap();\n\nvar _current = /*#__PURE__*/new WeakMap();\n\nvar _positionChangeListeners = /*#__PURE__*/new WeakMap();\n\nvar _lengthChangeListeners = /*#__PURE__*/new WeakMap();\n\nvar _onAdd = /*#__PURE__*/new WeakSet();\n\n_Symbol$iterator = Symbol.iterator;\n\nvar MusicPlaylist = /*#__PURE__*/function () {\n /**\n * Instantiates a playlist for musics.\n * \n * @param {string} name The name of the playlist.\n */\n function MusicPlaylist(name) {\n _classCallCheck(this, MusicPlaylist);\n\n _classPrivateMethodInitSpec(this, _onAdd);\n\n _classPrivateFieldInitSpec(this, _list, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _name, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _current, {\n writable: true,\n value: 0\n });\n\n _classPrivateFieldInitSpec(this, _positionChangeListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _lengthChangeListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldSet(this, _name, name);\n }\n /**\n * @returns {string} The name of the playlist. \n */\n\n\n _createClass(MusicPlaylist, [{\n key: \"name\",\n get: function get() {\n return _classPrivateFieldGet(this, _name);\n }\n /**\n * \n * @param {number} index the index of the music to retrieve.\n * @returns {Music} the music at the given index.\n */\n\n }, {\n key: \"musicAtIndex\",\n value: function musicAtIndex(index) {\n if (index >= this.total) {\n return null;\n }\n\n return _classPrivateFieldGet(this, _list)[index];\n }\n /**\n * Automatically creates and adds a {@link Music} to this playlist.\n * \n * @param {string} url where the audio data can be found.\n * @param {string} name the name of the music.\n * @param {string} author the author(s) of the music.\n */\n\n }, {\n key: \"add\",\n value: function add(url, name, author) {\n var music = new _Music_js__WEBPACK_IMPORTED_MODULE_0__[\"default\"](url, name, author, this);\n\n _classPrivateFieldGet(this, _list).push(music);\n\n _classPrivateMethodGet(this, _onAdd, _onAdd2).call(this, this.total - 1, music);\n }\n /**\n * Adds an existing {@link Music} object to this playlist. Alternatively, use {@link MusicPlaylist#add} to automatically instantiate a new one.\n * \n * @param {Music} music The existing music to add.\n */\n\n }, {\n key: \"addExisting\",\n value: function addExisting(music) {\n _classPrivateFieldGet(this, _list).push(music);\n\n _classPrivateMethodGet(this, _onAdd, _onAdd2).call(this, this.total - 1, music);\n }\n /**\n * Calls all playlist length change listeners.\n * \n * @param {number} index The index at which the new music was added.\n * @param {Music} music The music that was added.\n */\n\n }, {\n key: \"remove\",\n value:\n /**\n * Removes a {@link Music} from this playlist.\n * \n * @param {number} index the index of the music to be removed.\n * @returns {Music} the music that was removed, or null if the index of was invalid.\n */\n function remove(index) {\n if (index >= _classPrivateFieldGet(this, _list).length) {\n return null;\n }\n\n var removed = _classPrivateFieldGet(this, _list).splice(index, 1)[0];\n\n _classPrivateFieldGet(this, _lengthChangeListeners).forEach(function (listener) {\n listener(index, false, removed);\n });\n\n if (removed.length > 0) {\n return removed[0];\n }\n }\n /**\n * Attempts to find a {@link Music} given a name. Returns the first one found.\n * \n * @param {string} name the name of the music to be found.\n * @returns {number} the index of the music found, or -1 if it was not found.\n */\n\n }, {\n key: \"findMusicIndex\",\n value: function findMusicIndex(name) {\n return _classPrivateFieldGet(this, _list).findIndex(function (item) {\n return item.getDisplayName() == name;\n });\n }\n /**\n * The number of {@link Music} in this playlist.\n * \n * @returns {number} total number of musics in this playlist.\n */\n\n }, {\n key: \"total\",\n get: function get() {\n return _classPrivateFieldGet(this, _list).length;\n }\n /**\n * @returns {number} The current position in the playlist.\n */\n\n }, {\n key: \"currentPosition\",\n get: function get() {\n return _classPrivateFieldGet(this, _current);\n }\n /**\n * The current position in the playlist.\n */\n ,\n set: function set(position) {\n if (typeof position !== \"number\") throw new Error(\"Given position is invalid.\");\n if (position >= _classPrivateFieldGet(this, _list).length || position < 0) return;\n\n var old = _classPrivateFieldGet(this, _current);\n\n _classPrivateFieldSet(this, _current, position);\n\n _classPrivateFieldGet(this, _positionChangeListeners).forEach(function (positionChangeListener) {\n positionChangeListener(old, position);\n });\n }\n /**\n * @returns {Music} The current music in the playlist being used.\n */\n\n }, {\n key: \"currentMusic\",\n get: function get() {\n return this.musicAtIndex(this.currentPosition);\n }\n /**\n * \n * @param {positionChangeListener} listener the listener to receive the updates.\n * @returns {boolean} true if and only if successfully added the listener.\n */\n\n }, {\n key: \"addPositionChangeListener\",\n value: function addPositionChangeListener(listener) {\n if (_classPrivateFieldGet(this, _positionChangeListeners).includes(listener)) return false;\n\n _classPrivateFieldGet(this, _positionChangeListeners).push(listener);\n\n return true;\n }\n /**\n * \n * @param {positionChangeListener} listener the music change listener to remove.\n * @returns {boolean} true if and only if the music change listener given was successfully removed.\n */\n\n }, {\n key: \"removePositionChangeListener\",\n value: function removePositionChangeListener(listener) {\n var removeIndex = _classPrivateFieldGet(this, _positionChangeListeners).indexOf(listener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _positionChangeListeners).splice(removeIndex, 1);\n\n return true;\n }\n /**\n * Adds a listener for whether music was added or removed to the playlist.\n * \n * @param {lengthChangeListener} listener The listener to be added.\n * @returns {boolean} True if and only if the listener was successfully added.\n */\n\n }, {\n key: \"addLengthChangeListener\",\n value: function addLengthChangeListener(listener) {\n if (_classPrivateFieldGet(this, _lengthChangeListeners).includes(listener)) return false;\n\n _classPrivateFieldGet(this, _lengthChangeListeners).push(listener);\n\n return true;\n }\n /**\n * Removes a listener for whether music is added or removed to the playlist.\n * \n * @param {lengthChangeListener} listener The listener to be removed.\n * @returns {boolean} True if and only if the provided listener was successfully removed.\n */\n\n }, {\n key: \"removeLengthChangeListener\",\n value: function removeLengthChangeListener(listener) {\n var removeIndex = _classPrivateFieldGet(this, _lengthChangeListeners).indexOf(listener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _lengthChangeListeners).splice(removeIndex, 1);\n\n return true;\n }\n }, {\n key: \"generatePlaylistElement\",\n value: function generatePlaylistElement() {\n var element = document.createElement(\"table\");\n element.createTHead().textContent = \"Playlist\";\n var headers = element.insertRow();\n var musicHeader = document.createElement(\"th\");\n musicHeader.textContent = \"Music\";\n headers.appendChild(musicHeader);\n var authorHeader = document.createElement(\"th\");\n authorHeader.textContent = \"Author\";\n headers.appendChild(authorHeader);\n\n var insertMusic = function insertMusic(music) {\n var musicPos = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : undefined;\n if (musicPos) musicPos++;\n var row = element.insertRow(musicPos);\n row.insertCell().textContent = music.displayName;\n row.insertCell().textContent = music.author;\n };\n\n var _iterator = _createForOfIteratorHelper(this),\n _step;\n\n try {\n for (_iterator.s(); !(_step = _iterator.n()).done;) {\n var music = _step.value;\n insertMusic(music);\n }\n } catch (err) {\n _iterator.e(err);\n } finally {\n _iterator.f();\n }\n\n this.addLengthChangeListener(function (changed, added, music) {\n if (added) {\n insertMusic(music, changed);\n } else {\n element.deleteRow(changed);\n }\n });\n element.rows[this.currentPosition + 1].classList.add(\"active\");\n this.addPositionChangeListener(function (old, current) {\n element.rows[old + 1].classList.remove(\"active\");\n element.rows[current + 1].classList.add(\"active\");\n });\n return element;\n }\n }, {\n key: _Symbol$iterator,\n value: function value() {\n var index = -1;\n\n var data = _classPrivateFieldGet(this, _list);\n\n return {\n next: function next() {\n return {\n value: data[++index],\n done: index >= data.length\n };\n } // Brackets so not read as a body of anonymous function.\n\n };\n }\n }]);\n\n return MusicPlaylist;\n}();\n\nfunction _onAdd2(index, music) {\n _classPrivateFieldGet(this, _lengthChangeListeners).forEach(function (listener) {\n listener(index, true, music);\n });\n}\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/player/MusicPlaylist.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/player/MusicPlaylist.js?ba35"],"sourcesContent":["import Music from \"./Music.js\";\n\n/**@module */\n\n/**\n * @callback positionChangeListener\n * @param {number} old the previous position.\n * @param {number} current the the current (new) position.\n */\n\n/**\n * @callback lengthChangeListener\n * @param {number} changed The index of the music that has changed.\n * @param {boolean} added True if and only if the change was caused by adding music.\n * @param {Music} music The music that was added or removed.\n */\n\n/**\n * A playlist that holds a multitude of musics in the form of {@link Music}.\n */\nexport default class MusicPlaylist {\n    #list = [];\n    #name;\n    #current = 0;\n    #positionChangeListeners = [];\n    #lengthChangeListeners = [];\n\n    /**\n     * Instantiates a playlist for musics.\n     * \n     * @param {string} name The name of the playlist.\n     */\n    constructor(name) {\n        this.#name = name;\n    }\n\n    /**\n     * @returns {string} The name of the playlist. \n     */\n    get name() {\n        return this.#name;\n    }\n\n    /**\n     * \n     * @param {number} index the index of the music to retrieve.\n     * @returns {Music} the music at the given index.\n     */\n    musicAtIndex(index) {\n        if (index >= this.total) {\n            return null;\n        }\n        return this.#list[index];\n    }\n\n    /**\n     * Automatically creates and adds a {@link Music} to this playlist.\n     * \n     * @param {string} url where the audio data can be found.\n     * @param {string} name the name of the music.\n     * @param {string} author the author(s) of the music.\n     */\n    add(url, name, author) {\n        const music = new Music(url, name, author, this);\n        this.#list.push(music);\n        this.#onAdd(this.total - 1, music);\n    }\n\n    /**\n     * Adds an existing {@link Music} object to this playlist. Alternatively, use {@link MusicPlaylist#add} to automatically instantiate a new one.\n     * \n     * @param {Music} music The existing music to add.\n     */\n    addExisting(music) {\n        this.#list.push(music);\n        this.#onAdd(this.total - 1, music);\n    }\n\n    /**\n     * Calls all playlist length change listeners.\n     * \n     * @param {number} index The index at which the new music was added.\n     * @param {Music} music The music that was added.\n     */\n    #onAdd(index, music) {\n        this.#lengthChangeListeners.forEach(listener => {\n            listener(index, true, music);\n        });\n    }\n\n    /**\n     * Removes a {@link Music} from this playlist.\n     * \n     * @param {number} index the index of the music to be removed.\n     * @returns {Music} the music that was removed, or null if the index of was invalid.\n     */\n    remove(index) {\n        if (index >= this.#list.length) {\n            return null;\n        }\n        let removed = this.#list.splice(index, 1)[0];\n        this.#lengthChangeListeners.forEach(listener => {\n            listener(index, false, removed);\n        });\n        if (removed.length > 0) {\n            return removed[0];\n        }\n    }\n\n    /**\n     * Attempts to find a {@link Music} given a name. Returns the first one found.\n     * \n     * @param {string} name the name of the music to be found.\n     * @returns {number} the index of the music found, or -1 if it was not found.\n     */\n    findMusicIndex(name) {\n        return this.#list.findIndex((item) => item.getDisplayName() == name);\n    }\n\n    /**\n     * The number of {@link Music} in this playlist.\n     * \n     * @returns {number} total number of musics in this playlist.\n     */\n    get total() {\n        return this.#list.length;\n    }\n\n    /**\n     * @returns {number} The current position in the playlist.\n     */\n    get currentPosition() {\n        return this.#current;\n    }\n\n    /**\n     * The current position in the playlist.\n     */\n    set currentPosition(position) {\n        if (typeof position !== \"number\") throw new Error(\"Given position is invalid.\");\n        if (position >= this.#list.length || position < 0) return;\n        const old = this.#current;\n        this.#current = position;\n        this.#positionChangeListeners.forEach(positionChangeListener => {\n            positionChangeListener(old, position);\n        });\n    }\n\n    /**\n     * @returns {Music} The current music in the playlist being used.\n     */\n    get currentMusic() {\n        return this.musicAtIndex(this.currentPosition);\n    }\n\n    /**\n     * \n     * @param {positionChangeListener} listener the listener to receive the updates.\n     * @returns {boolean} true if and only if successfully added the listener.\n     */\n    addPositionChangeListener(listener) {\n        if (this.#positionChangeListeners.includes(listener)) return false;\n        this.#positionChangeListeners.push(listener);\n        return true;\n    }\n\n    /**\n     * \n     * @param {positionChangeListener} listener the music change listener to remove.\n     * @returns {boolean} true if and only if the music change listener given was successfully removed.\n     */\n    removePositionChangeListener(listener) {\n        const removeIndex = this.#positionChangeListeners.indexOf(listener);\n        if (removeIndex < 0) return false;\n\n        this.#positionChangeListeners.splice(removeIndex, 1);\n        return true;\n    }\n\n    /**\n     * Adds a listener for whether music was added or removed to the playlist.\n     * \n     * @param {lengthChangeListener} listener The listener to be added.\n     * @returns {boolean} True if and only if the listener was successfully added.\n     */\n    addLengthChangeListener(listener) {\n        if (this.#lengthChangeListeners.includes(listener)) return false;\n        this.#lengthChangeListeners.push(listener);\n        return true;\n    }\n\n    /**\n     * Removes a listener for whether music is added or removed to the playlist.\n     * \n     * @param {lengthChangeListener} listener The listener to be removed.\n     * @returns {boolean} True if and only if the provided listener was successfully removed.\n     */\n    removeLengthChangeListener(listener) {\n        const removeIndex = this.#lengthChangeListeners.indexOf(listener);\n        if (removeIndex < 0) return false;\n        this.#lengthChangeListeners.splice(removeIndex, 1);\n        return true;\n    }\n\n    generatePlaylistElement() {\n        const element = document.createElement(\"table\");\n        element.createTHead().textContent = \"Playlist\";\n        const headers = element.insertRow();\n        const musicHeader = document.createElement(\"th\");\n        musicHeader.textContent = \"Music\";\n        headers.appendChild(musicHeader);\n        const authorHeader = document.createElement(\"th\");\n        authorHeader.textContent = \"Author\";\n        headers.appendChild(authorHeader);\n\n        const insertMusic = (music, musicPos = undefined) => {\n            if (musicPos) musicPos++;\n            const row = element.insertRow(musicPos);\n            row.insertCell().textContent = music.displayName;\n            row.insertCell().textContent = music.author;\n        };\n        for (const music of this) {\n            insertMusic(music);\n        }\n\n        this.addLengthChangeListener((changed, added, music) => {\n            if (added) {\n                insertMusic(music, changed);\n            } else {\n                element.deleteRow(changed);\n            }\n        });\n\n        element.rows[this.currentPosition + 1].classList.add(\"active\");\n\n        this.addPositionChangeListener((old, current) => {\n            element.rows[old + 1].classList.remove(\"active\");\n            element.rows[current + 1].classList.add(\"active\");\n        });\n        return element;\n    }\n\n    [Symbol.iterator]() {\n        let index = -1;\n        const data = this.#list;\n        return {\n            next: () => ({ value: data[++index], done: index >= data.length }) // Brackets so not read as a body of anonymous function.\n        };\n    }\n}"],"names":["Music","Symbol","iterator","MusicPlaylist","name","index","total","url","author","music","push","length","removed","splice","forEach","listener","findIndex","item","getDisplayName","position","Error","old","positionChangeListener","musicAtIndex","currentPosition","includes","removeIndex","indexOf","element","document","createElement","createTHead","textContent","headers","insertRow","musicHeader","appendChild","authorHeader","insertMusic","musicPos","undefined","row","insertCell","displayName","addLengthChangeListener","changed","added","deleteRow","rows","classList","add","addPositionChangeListener","current","remove","data","next","value","done"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/player/MusicPlaylist.js\n"); + +/***/ }), + +/***/ "./src/player/VisualizedMusicPlayer.js": +/*!*********************************************!*\ + !*** ./src/player/VisualizedMusicPlayer.js ***! + \*********************************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ VisualizedMusicPlayer)\n/* harmony export */ });\n/* harmony import */ var _visualization_Visualizer_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ../visualization/Visualizer.js */ \"./src/visualization/Visualizer.js\");\n/* harmony import */ var _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ../visualization/VisualizerUpdateManager.js */ \"./src/visualization/VisualizerUpdateManager.js\");\n/* harmony import */ var _MusicPlayer_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./MusicPlayer.js */ \"./src/player/MusicPlayer.js\");\n/* harmony import */ var _MusicPlaylist_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./MusicPlaylist.js */ \"./src/player/MusicPlaylist.js\");\nfunction _typeof(obj) { \"@babel/helpers - typeof\"; return _typeof = \"function\" == typeof Symbol && \"symbol\" == typeof Symbol.iterator ? function (obj) { return typeof obj; } : function (obj) { return obj && \"function\" == typeof Symbol && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; }, _typeof(obj); }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _get() { if (typeof Reflect !== \"undefined\" && Reflect.get) { _get = Reflect.get; } else { _get = function _get(target, property, receiver) { var base = _superPropBase(target, property); if (!base) return; var desc = Object.getOwnPropertyDescriptor(base, property); if (desc.get) { return desc.get.call(arguments.length < 3 ? target : receiver); } return desc.value; }; } return _get.apply(this, arguments); }\n\nfunction set(target, property, value, receiver) { if (typeof Reflect !== \"undefined\" && Reflect.set) { set = Reflect.set; } else { set = function set(target, property, value, receiver) { var base = _superPropBase(target, property); var desc; if (base) { desc = Object.getOwnPropertyDescriptor(base, property); if (desc.set) { desc.set.call(receiver, value); return true; } else if (!desc.writable) { return false; } } desc = Object.getOwnPropertyDescriptor(receiver, property); if (desc) { if (!desc.writable) { return false; } desc.value = value; Object.defineProperty(receiver, property, desc); } else { _defineProperty(receiver, property, value); } return true; }; } return set(target, property, value, receiver); }\n\nfunction _set(target, property, value, receiver, isStrict) { var s = set(target, property, value, receiver || target); if (!s && isStrict) { throw new Error('failed to set property'); } return value; }\n\nfunction _defineProperty(obj, key, value) { if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }\n\nfunction _superPropBase(object, property) { while (!Object.prototype.hasOwnProperty.call(object, property)) { object = _getPrototypeOf(object); if (object === null) break; } return object; }\n\nfunction _inherits(subClass, superClass) { if (typeof superClass !== \"function\" && superClass !== null) { throw new TypeError(\"Super expression must either be null or a function\"); } subClass.prototype = Object.create(superClass && superClass.prototype, { constructor: { value: subClass, writable: true, configurable: true } }); Object.defineProperty(subClass, \"prototype\", { writable: false }); if (superClass) _setPrototypeOf(subClass, superClass); }\n\nfunction _setPrototypeOf(o, p) { _setPrototypeOf = Object.setPrototypeOf || function _setPrototypeOf(o, p) { o.__proto__ = p; return o; }; return _setPrototypeOf(o, p); }\n\nfunction _createSuper(Derived) { var hasNativeReflectConstruct = _isNativeReflectConstruct(); return function _createSuperInternal() { var Super = _getPrototypeOf(Derived), result; if (hasNativeReflectConstruct) { var NewTarget = _getPrototypeOf(this).constructor; result = Reflect.construct(Super, arguments, NewTarget); } else { result = Super.apply(this, arguments); } return _possibleConstructorReturn(this, result); }; }\n\nfunction _possibleConstructorReturn(self, call) { if (call && (_typeof(call) === \"object\" || typeof call === \"function\")) { return call; } else if (call !== void 0) { throw new TypeError(\"Derived constructors may only return object or undefined\"); } return _assertThisInitialized(self); }\n\nfunction _assertThisInitialized(self) { if (self === void 0) { throw new ReferenceError(\"this hasn't been initialised - super() hasn't been called\"); } return self; }\n\nfunction _isNativeReflectConstruct() { if (typeof Reflect === \"undefined\" || !Reflect.construct) return false; if (Reflect.construct.sham) return false; if (typeof Proxy === \"function\") return true; try { Boolean.prototype.valueOf.call(Reflect.construct(Boolean, [], function () {})); return true; } catch (e) { return false; } }\n\nfunction _getPrototypeOf(o) { _getPrototypeOf = Object.setPrototypeOf ? Object.getPrototypeOf : function _getPrototypeOf(o) { return o.__proto__ || Object.getPrototypeOf(o); }; return _getPrototypeOf(o); }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\n\n\n\n\n/**@module */\n\n/**\n * A music player that provides easier access to the current musics visualizer data.\n * Additionally, automatically re-binds all the visualizer update listeners for music changes.\n * \n * @augments MusicPlayer\n */\n\nvar _fftSize = /*#__PURE__*/new WeakMap();\n\nvar _visualizerUpdateManager = /*#__PURE__*/new WeakMap();\n\nvar VisualizedMusicPlayer = /*#__PURE__*/function (_MusicPlayer) {\n _inherits(VisualizedMusicPlayer, _MusicPlayer);\n\n var _super = _createSuper(VisualizedMusicPlayer);\n\n /**\n * Instantiates a music player with visualization features.\n * \n * @param {MusicPlaylist} playlist the playlist this player manages.\n * @param {number} [fftSize=1024] the size of the fft window for analysis.\n */\n function VisualizedMusicPlayer(playlist) {\n var _this;\n\n var fftSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 1024;\n\n _classCallCheck(this, VisualizedMusicPlayer);\n\n _this = _super.call(this, playlist);\n\n _classPrivateFieldInitSpec(_assertThisInitialized(_this), _fftSize, {\n writable: true,\n value: 1024\n });\n\n _classPrivateFieldInitSpec(_assertThisInitialized(_this), _visualizerUpdateManager, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldSet(_assertThisInitialized(_this), _fftSize, fftSize);\n\n _classPrivateFieldSet(_assertThisInitialized(_this), _visualizerUpdateManager, new _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"](_this.getCurrentMusicVisualizer()));\n\n return _this;\n }\n /**\n * Sets the the playlist.\n */\n\n\n _createClass(VisualizedMusicPlayer, [{\n key: \"playlist\",\n set: function set(playlist) {\n _set(_getPrototypeOf(VisualizedMusicPlayer.prototype), \"playlist\", playlist, this, true);\n\n _classPrivateFieldSet(this, _visualizerUpdateManager, new _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"](this.getCurrentMusicVisualizer()));\n }\n /**\n * \n * @returns {boolean} true if and only if successful in changing to the next music.\n */\n\n }, {\n key: \"next\",\n value: function next() {\n var updateListeners = _classPrivateFieldGet(this, _visualizerUpdateManager).getBinnedListeners();\n\n if (!_get(_getPrototypeOf(VisualizedMusicPlayer.prototype), \"next\", this).call(this)) return false;\n\n _classPrivateFieldSet(this, _visualizerUpdateManager, new _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"](this.getCurrentMusicVisualizer()));\n\n _classPrivateFieldGet(this, _visualizerUpdateManager).setBinnedListeners(updateListeners);\n\n return true;\n }\n /**\n * Jumps to the previous music if possible.\n * \n * @returns {boolean} true if and only if successful in switching to the previous music in the playlist.\n */\n\n }, {\n key: \"previous\",\n value: function previous() {\n var updateListeners = _classPrivateFieldGet(this, _visualizerUpdateManager).getBinnedListeners();\n\n if (!_get(_getPrototypeOf(VisualizedMusicPlayer.prototype), \"previous\", this).call(this)) return false;\n\n _classPrivateFieldSet(this, _visualizerUpdateManager, new _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"](this.getCurrentMusicVisualizer()));\n\n _classPrivateFieldGet(this, _visualizerUpdateManager).setBinnedListeners(updateListeners);\n\n return true;\n }\n /**\n * \n * @param {number} index the index of the music to change to.\n * @returns {boolean} true if and only if successful in jumping to the given index.\n */\n\n }, {\n key: \"changeCurrent\",\n value: function changeCurrent(index) {\n var updateListeners = this.VisualizerUpdateManager.getBinnedListeners();\n if (!_get(_getPrototypeOf(VisualizedMusicPlayer.prototype), \"changeCurrent\", this).call(this, index)) return false;\n\n _classPrivateFieldSet(this, _visualizerUpdateManager, new _visualization_VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"](this.getCurrentMusicVisualizer()));\n\n _classPrivateFieldGet(this, _visualizerUpdateManager).setBinnedListeners = updateListeners();\n return true;\n }\n /**\n * @returns {VisualizerUpdateManager} The current visualizer update manager.\n */\n\n }, {\n key: \"currentVisualizerUpdateManager\",\n get: function get() {\n return _classPrivateFieldGet(this, _visualizerUpdateManager).call(this);\n }\n /**\n * @returns {Visualizer} The current music's visualizer.\n */\n\n }, {\n key: \"currentMusicVisualizer\",\n get: function get() {\n return this.getCurrentMusic().getVisualizer();\n }\n }]);\n\n return VisualizedMusicPlayer;\n}(_MusicPlayer_js__WEBPACK_IMPORTED_MODULE_2__[\"default\"]);\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvcGxheWVyL1Zpc3VhbGl6ZWRNdXNpY1BsYXllci5qcy5qcyIsIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7QUFBQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7Ozs7O0lBQ3FCSTs7Ozs7QUFJakI7QUFDSjtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0ksaUNBQVlDLFFBQVosRUFBc0M7QUFBQTs7QUFBQSxRQUFoQkMsT0FBZ0IsdUVBQU4sSUFBTTs7QUFBQTs7QUFDbEMsOEJBQU1ELFFBQU47O0FBRGtDO0FBQUE7QUFBQSxhQVQzQjtBQVMyQjs7QUFBQTtBQUFBO0FBQUE7QUFBQTs7QUFFbEMsbUVBQWdCQyxPQUFoQjs7QUFDQSxtRkFBZ0MsSUFBSUwsaUZBQUosQ0FBNEIsTUFBS00seUJBQUwsRUFBNUIsQ0FBaEM7O0FBSGtDO0FBSXJDO0FBRUQ7QUFDSjtBQUNBOzs7OztTQUNJLGFBQWFGLFFBQWIsRUFBdUI7QUFDbkIseUVBQWlCQSxRQUFqQjs7QUFDQSw0REFBZ0MsSUFBSUosaUZBQUosQ0FBNEIsS0FBS00seUJBQUwsRUFBNUIsQ0FBaEM7QUFDSDtBQUVEO0FBQ0o7QUFDQTtBQUNBOzs7O1dBQ0ksZ0JBQU87QUFDSCxVQUFNQyxlQUFlLEdBQUcsc0RBQThCQyxrQkFBOUIsRUFBeEI7O0FBQ0EsVUFBSSxnRkFBSixFQUFtQixPQUFPLEtBQVA7O0FBQ25CLDREQUFnQyxJQUFJUixpRkFBSixDQUE0QixLQUFLTSx5QkFBTCxFQUE1QixDQUFoQzs7QUFDQSw0REFBOEJHLGtCQUE5QixDQUFpREYsZUFBakQ7O0FBQ0EsYUFBTyxJQUFQO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7QUFDQTtBQUNBOzs7O1dBQ0ksb0JBQVc7QUFDUCxVQUFNQSxlQUFlLEdBQUcsc0RBQThCQyxrQkFBOUIsRUFBeEI7O0FBQ0EsVUFBSSxvRkFBSixFQUF1QixPQUFPLEtBQVA7O0FBRXZCLDREQUFnQyxJQUFJUixpRkFBSixDQUE0QixLQUFLTSx5QkFBTCxFQUE1QixDQUFoQzs7QUFDQSw0REFBOEJHLGtCQUE5QixDQUFpREYsZUFBakQ7O0FBQ0EsYUFBTyxJQUFQO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7QUFDQTtBQUNBOzs7O1dBQ0ksdUJBQWNHLEtBQWQsRUFBcUI7QUFDakIsVUFBTUgsZUFBZSxHQUFHLEtBQUtQLHVCQUFMLENBQTZCUSxrQkFBN0IsRUFBeEI7QUFDQSxVQUFJLDBGQUFxQkUsS0FBckIsQ0FBSixFQUFpQyxPQUFPLEtBQVA7O0FBQ2pDLDREQUFnQyxJQUFJVixpRkFBSixDQUE0QixLQUFLTSx5QkFBTCxFQUE1QixDQUFoQzs7QUFDQSw0REFBOEJHLGtCQUE5QixHQUFtREYsZUFBZSxFQUFsRTtBQUNBLGFBQU8sSUFBUDtBQUNIO0FBRUQ7QUFDSjtBQUNBOzs7O1NBQ0ksZUFBcUM7QUFDakMsbUNBQU8sSUFBUCxpQ0FBTyxJQUFQO0FBQ0g7QUFFRDtBQUNKO0FBQ0E7Ozs7U0FDSSxlQUE2QjtBQUN6QixhQUFPLEtBQUtJLGVBQUwsR0FBdUJDLGFBQXZCLEVBQVA7QUFDSDs7OztFQTNFOENYIiwic291cmNlcyI6WyJ3ZWJwYWNrOi8vYXVkaW9zaG93a2l0Ly4vc3JjL3BsYXllci9WaXN1YWxpemVkTXVzaWNQbGF5ZXIuanM/NWM1YSJdLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgVmlzdWFsaXplciBmcm9tIFwiLi4vdmlzdWFsaXphdGlvbi9WaXN1YWxpemVyLmpzXCI7XG5pbXBvcnQgVmlzdWFsaXplclVwZGF0ZU1hbmFnZXIgZnJvbSBcIi4uL3Zpc3VhbGl6YXRpb24vVmlzdWFsaXplclVwZGF0ZU1hbmFnZXIuanNcIjtcbmltcG9ydCBNdXNpY1BsYXllciBmcm9tIFwiLi9NdXNpY1BsYXllci5qc1wiO1xuaW1wb3J0IE11c2ljUGxheWxpc3QgZnJvbSBcIi4vTXVzaWNQbGF5bGlzdC5qc1wiO1xuXG4vKipAbW9kdWxlICovXG5cbi8qKlxuICogQSBtdXNpYyBwbGF5ZXIgdGhhdCBwcm92aWRlcyBlYXNpZXIgYWNjZXNzIHRvIHRoZSBjdXJyZW50IG11c2ljcyB2aXN1YWxpemVyIGRhdGEuXG4gKiBBZGRpdGlvbmFsbHksIGF1dG9tYXRpY2FsbHkgcmUtYmluZHMgYWxsIHRoZSB2aXN1YWxpemVyIHVwZGF0ZSBsaXN0ZW5lcnMgZm9yIG11c2ljIGNoYW5nZXMuXG4gKiBcbiAqIEBhdWdtZW50cyBNdXNpY1BsYXllclxuICovXG5leHBvcnQgZGVmYXVsdCBjbGFzcyBWaXN1YWxpemVkTXVzaWNQbGF5ZXIgZXh0ZW5kcyBNdXNpY1BsYXllciB7XG4gICAgI2ZmdFNpemUgPSAxMDI0O1xuICAgICN2aXN1YWxpemVyVXBkYXRlTWFuYWdlcjtcblxuICAgIC8qKlxuICAgICAqIEluc3RhbnRpYXRlcyBhIG11c2ljIHBsYXllciB3aXRoIHZpc3VhbGl6YXRpb24gZmVhdHVyZXMuXG4gICAgICogXG4gICAgICogQHBhcmFtIHtNdXNpY1BsYXlsaXN0fSBwbGF5bGlzdCB0aGUgcGxheWxpc3QgdGhpcyBwbGF5ZXIgbWFuYWdlcy5cbiAgICAgKiBAcGFyYW0ge251bWJlcn0gW2ZmdFNpemU9MTAyNF0gdGhlIHNpemUgb2YgdGhlIGZmdCB3aW5kb3cgZm9yIGFuYWx5c2lzLlxuICAgICAqL1xuICAgIGNvbnN0cnVjdG9yKHBsYXlsaXN0LCBmZnRTaXplID0gMTAyNCkge1xuICAgICAgICBzdXBlcihwbGF5bGlzdCk7XG4gICAgICAgIHRoaXMuI2ZmdFNpemUgPSBmZnRTaXplO1xuICAgICAgICB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlciA9IG5ldyBWaXN1YWxpemVyVXBkYXRlTWFuYWdlcih0aGlzLmdldEN1cnJlbnRNdXNpY1Zpc3VhbGl6ZXIoKSk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogU2V0cyB0aGUgdGhlIHBsYXlsaXN0LlxuICAgICAqL1xuICAgIHNldCBwbGF5bGlzdChwbGF5bGlzdCkge1xuICAgICAgICBzdXBlci5wbGF5bGlzdCA9IHBsYXlsaXN0O1xuICAgICAgICB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlciA9IG5ldyBWaXN1YWxpemVyVXBkYXRlTWFuYWdlcih0aGlzLmdldEN1cnJlbnRNdXNpY1Zpc3VhbGl6ZXIoKSk7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogXG4gICAgICogQHJldHVybnMge2Jvb2xlYW59IHRydWUgaWYgYW5kIG9ubHkgaWYgc3VjY2Vzc2Z1bCBpbiBjaGFuZ2luZyB0byB0aGUgbmV4dCBtdXNpYy5cbiAgICAgKi9cbiAgICBuZXh0KCkge1xuICAgICAgICBjb25zdCB1cGRhdGVMaXN0ZW5lcnMgPSB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlci5nZXRCaW5uZWRMaXN0ZW5lcnMoKTtcbiAgICAgICAgaWYgKCFzdXBlci5uZXh0KCkpIHJldHVybiBmYWxzZTtcbiAgICAgICAgdGhpcy4jdmlzdWFsaXplclVwZGF0ZU1hbmFnZXIgPSBuZXcgVmlzdWFsaXplclVwZGF0ZU1hbmFnZXIodGhpcy5nZXRDdXJyZW50TXVzaWNWaXN1YWxpemVyKCkpO1xuICAgICAgICB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlci5zZXRCaW5uZWRMaXN0ZW5lcnModXBkYXRlTGlzdGVuZXJzKTtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogSnVtcHMgdG8gdGhlIHByZXZpb3VzIG11c2ljIGlmIHBvc3NpYmxlLlxuICAgICAqIFxuICAgICAqIEByZXR1cm5zIHtib29sZWFufSB0cnVlIGlmIGFuZCBvbmx5IGlmIHN1Y2Nlc3NmdWwgaW4gc3dpdGNoaW5nIHRvIHRoZSBwcmV2aW91cyBtdXNpYyBpbiB0aGUgcGxheWxpc3QuXG4gICAgICovXG4gICAgcHJldmlvdXMoKSB7XG4gICAgICAgIGNvbnN0IHVwZGF0ZUxpc3RlbmVycyA9IHRoaXMuI3Zpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyLmdldEJpbm5lZExpc3RlbmVycygpO1xuICAgICAgICBpZiAoIXN1cGVyLnByZXZpb3VzKCkpIHJldHVybiBmYWxzZTtcblxuICAgICAgICB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlciA9IG5ldyBWaXN1YWxpemVyVXBkYXRlTWFuYWdlcih0aGlzLmdldEN1cnJlbnRNdXNpY1Zpc3VhbGl6ZXIoKSk7XG4gICAgICAgIHRoaXMuI3Zpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyLnNldEJpbm5lZExpc3RlbmVycyh1cGRhdGVMaXN0ZW5lcnMpO1xuICAgICAgICByZXR1cm4gdHJ1ZTtcbiAgICB9XG5cbiAgICAvKipcbiAgICAgKiBcbiAgICAgKiBAcGFyYW0ge251bWJlcn0gaW5kZXggdGhlIGluZGV4IG9mIHRoZSBtdXNpYyB0byBjaGFuZ2UgdG8uXG4gICAgICogQHJldHVybnMge2Jvb2xlYW59IHRydWUgaWYgYW5kIG9ubHkgaWYgc3VjY2Vzc2Z1bCBpbiBqdW1waW5nIHRvIHRoZSBnaXZlbiBpbmRleC5cbiAgICAgKi9cbiAgICBjaGFuZ2VDdXJyZW50KGluZGV4KSB7XG4gICAgICAgIGNvbnN0IHVwZGF0ZUxpc3RlbmVycyA9IHRoaXMuVmlzdWFsaXplclVwZGF0ZU1hbmFnZXIuZ2V0QmlubmVkTGlzdGVuZXJzKCk7XG4gICAgICAgIGlmICghc3VwZXIuY2hhbmdlQ3VycmVudChpbmRleCkpIHJldHVybiBmYWxzZTtcbiAgICAgICAgdGhpcy4jdmlzdWFsaXplclVwZGF0ZU1hbmFnZXIgPSBuZXcgVmlzdWFsaXplclVwZGF0ZU1hbmFnZXIodGhpcy5nZXRDdXJyZW50TXVzaWNWaXN1YWxpemVyKCkpO1xuICAgICAgICB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlci5zZXRCaW5uZWRMaXN0ZW5lcnMgPSB1cGRhdGVMaXN0ZW5lcnMoKTtcbiAgICAgICAgcmV0dXJuIHRydWU7XG4gICAgfVxuXG4gICAgLyoqXG4gICAgICogQHJldHVybnMge1Zpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyfSBUaGUgY3VycmVudCB2aXN1YWxpemVyIHVwZGF0ZSBtYW5hZ2VyLlxuICAgICAqL1xuICAgIGdldCBjdXJyZW50VmlzdWFsaXplclVwZGF0ZU1hbmFnZXIoKSB7XG4gICAgICAgIHJldHVybiB0aGlzLiN2aXN1YWxpemVyVXBkYXRlTWFuYWdlcigpO1xuICAgIH1cblxuICAgIC8qKlxuICAgICAqIEByZXR1cm5zIHtWaXN1YWxpemVyfSBUaGUgY3VycmVudCBtdXNpYydzIHZpc3VhbGl6ZXIuXG4gICAgICovXG4gICAgZ2V0IGN1cnJlbnRNdXNpY1Zpc3VhbGl6ZXIoKSB7XG4gICAgICAgIHJldHVybiB0aGlzLmdldEN1cnJlbnRNdXNpYygpLmdldFZpc3VhbGl6ZXIoKTtcbiAgICB9XG59Il0sIm5hbWVzIjpbIlZpc3VhbGl6ZXIiLCJWaXN1YWxpemVyVXBkYXRlTWFuYWdlciIsIk11c2ljUGxheWVyIiwiTXVzaWNQbGF5bGlzdCIsIlZpc3VhbGl6ZWRNdXNpY1BsYXllciIsInBsYXlsaXN0IiwiZmZ0U2l6ZSIsImdldEN1cnJlbnRNdXNpY1Zpc3VhbGl6ZXIiLCJ1cGRhdGVMaXN0ZW5lcnMiLCJnZXRCaW5uZWRMaXN0ZW5lcnMiLCJzZXRCaW5uZWRMaXN0ZW5lcnMiLCJpbmRleCIsImdldEN1cnJlbnRNdXNpYyIsImdldFZpc3VhbGl6ZXIiXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/player/VisualizedMusicPlayer.js\n"); + +/***/ }), + +/***/ "./src/player/player.js": +/*!******************************!*\ + !*** ./src/player/player.js ***! + \******************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"Music\": () => (/* reexport safe */ _Music_js__WEBPACK_IMPORTED_MODULE_3__[\"default\"]),\n/* harmony export */ \"MusicPlayer\": () => (/* reexport safe */ _MusicPlayer_js__WEBPACK_IMPORTED_MODULE_0__[\"default\"]),\n/* harmony export */ \"MusicPlaylist\": () => (/* reexport safe */ _MusicPlaylist_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"]),\n/* harmony export */ \"VisualizedMusicPlayer\": () => (/* reexport safe */ _VisualizedMusicPlayer_js__WEBPACK_IMPORTED_MODULE_2__[\"default\"])\n/* harmony export */ });\n/* harmony import */ var _MusicPlayer_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./MusicPlayer.js */ \"./src/player/MusicPlayer.js\");\n/* harmony import */ var _MusicPlaylist_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./MusicPlaylist.js */ \"./src/player/MusicPlaylist.js\");\n/* harmony import */ var _VisualizedMusicPlayer_js__WEBPACK_IMPORTED_MODULE_2__ = __webpack_require__(/*! ./VisualizedMusicPlayer.js */ \"./src/player/VisualizedMusicPlayer.js\");\n/* harmony import */ var _Music_js__WEBPACK_IMPORTED_MODULE_3__ = __webpack_require__(/*! ./Music.js */ \"./src/player/Music.js\");\n\n\n\n\n/**@module */\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvcGxheWVyL3BsYXllci5qcy5qcyIsIm1hcHBpbmdzIjoiOzs7Ozs7Ozs7OztBQUFBO0FBQ0E7QUFDQTtBQUNBO0FBRUEiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9hdWRpb3Nob3draXQvLi9zcmMvcGxheWVyL3BsYXllci5qcz80YmIwIl0sInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBNdXNpY1BsYXllciBmcm9tIFwiLi9NdXNpY1BsYXllci5qc1wiO1xuaW1wb3J0IE11c2ljUGxheWxpc3QgZnJvbSBcIi4vTXVzaWNQbGF5bGlzdC5qc1wiO1xuaW1wb3J0IFZpc3VhbGl6ZWRNdXNpY1BsYXllciBmcm9tIFwiLi9WaXN1YWxpemVkTXVzaWNQbGF5ZXIuanNcIjtcbmltcG9ydCBNdXNpYyBmcm9tIFwiLi9NdXNpYy5qc1wiO1xuXG4vKipAbW9kdWxlICovXG5leHBvcnQgeyBNdXNpY1BsYXllciwgTXVzaWNQbGF5bGlzdCwgVmlzdWFsaXplZE11c2ljUGxheWVyLCBNdXNpYyB9OyJdLCJuYW1lcyI6WyJNdXNpY1BsYXllciIsIk11c2ljUGxheWxpc3QiLCJWaXN1YWxpemVkTXVzaWNQbGF5ZXIiLCJNdXNpYyJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/player/player.js\n"); + +/***/ }), + +/***/ "./src/support/colors.js": +/*!*******************************!*\ + !*** ./src/support/colors.js ***! + \*******************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"cssRgbToRgba\": () => (/* binding */ cssRgbToRgba),\n/* harmony export */ \"cssRgbaToRgba\": () => (/* binding */ cssRgbaToRgba),\n/* harmony export */ \"parseColor\": () => (/* binding */ parseColor),\n/* harmony export */ \"rgbHexToRgba\": () => (/* binding */ rgbHexToRgba),\n/* harmony export */ \"rgbaHexToRgba\": () => (/* binding */ rgbaHexToRgba),\n/* harmony export */ \"rgbaToHexRgba\": () => (/* binding */ rgbaToHexRgba)\n/* harmony export */ });\n/**\n * Converts from rgba hex value to an array of four numbers representing r, g, b, and a respectively.\n * \n * @param {string} hex the hex value.\n * @returns {number[]} the resulting components of the hex value.\n */\nfunction rgbaHexToRgba(hex) {\n if (hex.startsWith(\"#\")) {\n hex = hex.substring(1);\n\n if (hex.length > 8) {\n throw new Error(\"Invalid hex syntax (length).\");\n }\n } else {\n throw new Error(\"Invalid hex syntax (missing pound).\");\n }\n\n for (var i = hex.length; i < 8; i++) {\n hex = \"0\" + hex;\n }\n\n var remaining = hex.length;\n var result = [0, 0, 0, 0];\n result[3] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n result[2] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n result[1] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n result[0] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n return result;\n}\n/**\n * \n * @param {string} hex the hex value.\n * @returns {number[]} the resulting r, g, and b components.\n */\n\n\nfunction rgbHexToRgba(hex) {\n var result = rgbaHexToRgba(hex + \"FF\");\n return result;\n}\n/**\n * Converts from the CSS rgba call to an array of four numbers representing r, g, b and a.\n * \n * @param {string} rgba The CSS rgba(r,g,b,a) call.\n * @returns {number[]} the rgba components.\n */\n\n\nfunction cssRgbaToRgba(rgba) {\n var cssRgbaRegex = /rgba\\((\\d+),(\\d+),(\\d+),(\\d+)\\)/;\n\n try {\n var matches = rgba.match(cssRgbaRegex);\n return [parseInt(matches[1]), parseInt(matches[2]), parseInt(matches[3]), parseInt(matches[4])];\n } catch (error) {\n throw new Error(\"Could not parse the given css rgba function call: \" + error.message);\n }\n}\n/**\n * Converts from the CSS rgb call to an array of four numbers representing r, g, b and a.\n * \n * @param {string} rgb The CSS rgb(r,g,b) call.\n * @returns {number[]} the rgba components.\n */\n\n\nfunction cssRgbToRgba(rgb) {\n var cssRgbRegex = /rgb\\((\\d+),(\\d+),(\\d+)\\)/;\n\n try {\n var matches = rgb.match(cssRgbRegex);\n return [parseInt(matches[1]), parseInt(matches[2]), parseInt(matches[3]), 255];\n } catch (error) {\n throw new Error(\"Could not parse the given css rgb function call: \" + error.message);\n }\n}\n/**\n * Converts a given array of r, g, b, and a components into a hex string.\n * \n * @param {number[]} rgba an array with red, green, blue, and alpha components in that order.\n * @returns {string} The resulting hex value.\n */\n\n\nfunction rgbaToHexRgba(rgba) {\n var filler = function filler(hex) {\n return hex.length < 2 ? \"0\" + hex : hex;\n };\n\n return \"#\" + filler(rgba[0].toString(16)) + filler(rgba[1].toString(16)) + filler(rgba[2].toString(16)) + filler(rgba[3].toString(16));\n}\n/**\n * Converts a css rgb, rgba, hex, or rgba hex to a rgba array.\n * \n * For hex, we assume there is no alpha channel unless the hex value is not minimized.\n * \n * @param {string} color The string that contains the color information.\n * @returns {number[]} an array that contains the r, g, b and a components.\n */\n\n\nfunction parseColor(color) {\n if (color.startsWith(\"rgba(\")) {\n return cssRgbaToRgba(color);\n } else if (color.startsWith(\"rgb(\")) {\n return cssRgbToRgba(color);\n } else if (color.startsWith(\"#\")) {\n if (color.length === 9) {\n return rgbaHexToRgba(color);\n } else {\n return rgbHexToRgba(color);\n }\n }\n\n throw new Error(\"Could not parse to an rgba value.\");\n}\n/**@module */\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/support/colors.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/support/colors.js?3d5d"],"sourcesContent":["\n/**\n * Converts from rgba hex value to an array of four numbers representing r, g, b, and a respectively.\n * \n * @param {string} hex the hex value.\n * @returns {number[]} the resulting components of the hex value.\n */\nfunction rgbaHexToRgba(hex) {\n    if (hex.startsWith(\"#\")) {\n        hex = hex.substring(1);\n        if (hex.length > 8) {\n            throw new Error(\"Invalid hex syntax (length).\");\n        }\n    } else {\n        throw new Error(\"Invalid hex syntax (missing pound).\");\n    }\n    for (let i = hex.length; i < 8; i++) {\n        hex = \"0\" + hex;\n    }\n    let remaining = hex.length;\n    let result = [0, 0, 0, 0];\n    result[3] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n    result[2] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n    result[1] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n    result[0] = parseInt(hex.substring(remaining -= 2, remaining + 2), 16);\n    return result;\n}\n\n/**\n * \n * @param {string} hex the hex value.\n * @returns {number[]} the resulting r, g, and b components.\n */\nfunction rgbHexToRgba(hex) {\n    const result = rgbaHexToRgba(hex + \"FF\");\n    return result;\n}\n\n/**\n * Converts from the CSS rgba call to an array of four numbers representing r, g, b and a.\n * \n * @param {string} rgba The CSS rgba(r,g,b,a) call.\n * @returns {number[]} the rgba components.\n */\nfunction cssRgbaToRgba(rgba) {\n    const cssRgbaRegex = /rgba\\((\\d+),(\\d+),(\\d+),(\\d+)\\)/;\n    try {\n        const matches = rgba.match(cssRgbaRegex);\n        return [parseInt(matches[1]), parseInt(matches[2]), parseInt(matches[3]), parseInt(matches[4])];\n    } catch (error) {\n        throw new Error(\"Could not parse the given css rgba function call: \" + error.message);\n    }\n}\n\n/**\n * Converts from the CSS rgb call to an array of four numbers representing r, g, b and a.\n * \n * @param {string} rgb The CSS rgb(r,g,b) call.\n * @returns {number[]} the rgba components.\n */\nfunction cssRgbToRgba(rgb) {\n    const cssRgbRegex = /rgb\\((\\d+),(\\d+),(\\d+)\\)/;\n    try {\n        const matches = rgb.match(cssRgbRegex);\n        return [parseInt(matches[1]), parseInt(matches[2]), parseInt(matches[3]), 255];\n    } catch (error) {\n        throw new Error(\"Could not parse the given css rgb function call: \" + error.message);\n    }\n}\n\n/**\n * Converts a given array of r, g, b, and a components into a hex string.\n * \n * @param {number[]} rgba an array with red, green, blue, and alpha components in that order.\n * @returns {string} The resulting hex value.\n */\nfunction rgbaToHexRgba(rgba) {\n    const filler = (hex) => hex.length < 2 ? \"0\" + hex : hex;\n    return \"#\" + filler(rgba[0].toString(16)) + filler(rgba[1].toString(16)) + filler(rgba[2].toString(16)) + filler(rgba[3].toString(16));\n}\n\n\n/**\n * Converts a css rgb, rgba, hex, or rgba hex to a rgba array.\n * \n * For hex, we assume there is no alpha channel unless the hex value is not minimized.\n * \n * @param {string} color The string that contains the color information.\n * @returns {number[]} an array that contains the r, g, b and a components.\n */\nfunction parseColor(color) {\n    if (color.startsWith(\"rgba(\")) {\n        return cssRgbaToRgba(color);\n    } else if (color.startsWith(\"rgb(\")) {\n        return cssRgbToRgba(color);\n    } else if (color.startsWith(\"#\")) {\n        if (color.length === 9) {\n            return rgbaHexToRgba(color);\n        } else {\n            return rgbHexToRgba(color);\n        }\n    }\n    throw new Error(\"Could not parse to an rgba value.\");\n}\n\n/**@module */\nexport { rgbaHexToRgba, rgbHexToRgba, cssRgbaToRgba, cssRgbToRgba, rgbaToHexRgba, parseColor };"],"names":["rgbaHexToRgba","hex","startsWith","substring","length","Error","i","remaining","result","parseInt","rgbHexToRgba","cssRgbaToRgba","rgba","cssRgbaRegex","matches","match","error","message","cssRgbToRgba","rgb","cssRgbRegex","rgbaToHexRgba","filler","toString","parseColor","color"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/support/colors.js\n"); + +/***/ }), + +/***/ "./src/support/easings.js": +/*!********************************!*\ + !*** ./src/support/easings.js ***! + \********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"createEaseLinear\": () => (/* binding */ createEaseLinear)\n/* harmony export */ });\n/**\n * @callback interpolator\n * @param {number} current the current value.\n * @param {number} dest the destination value.\n * @param {number} frameDelta the time elapsed since the last update.\n * @returns {number} the new current value.\n */\n// TODO: Add some interpolation functions.\n\n/**\n * \n * @param {number} rate the number of frequency values to shift by per second.\n * @returns {interpolator} the interpolation function with the given rate.\n */\nfunction createEaseLinear(rate) {\n return function (current, dest, frameDelta) {\n var direction = 1;\n\n if (dest < current) {\n direction = -1;\n }\n\n return current + Math.min(rate * frameDelta, dest) * direction;\n };\n}\n/**@module */\n\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvc3VwcG9ydC9lYXNpbmdzLmpzLmpzIiwibWFwcGluZ3MiOiI7Ozs7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUVBOztBQUVBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQSxTQUFTQSxnQkFBVCxDQUEwQkMsSUFBMUIsRUFBZ0M7QUFDNUIsU0FBTyxVQUFDQyxPQUFELEVBQVVDLElBQVYsRUFBZ0JDLFVBQWhCLEVBQStCO0FBQ2xDLFFBQUlDLFNBQVMsR0FBRyxDQUFoQjs7QUFDQSxRQUFJRixJQUFJLEdBQUdELE9BQVgsRUFBb0I7QUFDaEJHLE1BQUFBLFNBQVMsR0FBRyxDQUFDLENBQWI7QUFDSDs7QUFDRCxXQUFPSCxPQUFPLEdBQUlJLElBQUksQ0FBQ0MsR0FBTCxDQUFTTixJQUFJLEdBQUdHLFVBQWhCLEVBQTRCRCxJQUE1QixJQUFvQ0UsU0FBdEQ7QUFDSCxHQU5EO0FBT0g7QUFFRCIsInNvdXJjZXMiOlsid2VicGFjazovL2F1ZGlvc2hvd2tpdC8uL3NyYy9zdXBwb3J0L2Vhc2luZ3MuanM/MTFkMyJdLCJzb3VyY2VzQ29udGVudCI6WyJcbi8qKlxuICogQGNhbGxiYWNrIGludGVycG9sYXRvclxuICogQHBhcmFtIHtudW1iZXJ9IGN1cnJlbnQgdGhlIGN1cnJlbnQgdmFsdWUuXG4gKiBAcGFyYW0ge251bWJlcn0gZGVzdCB0aGUgZGVzdGluYXRpb24gdmFsdWUuXG4gKiBAcGFyYW0ge251bWJlcn0gZnJhbWVEZWx0YSB0aGUgdGltZSBlbGFwc2VkIHNpbmNlIHRoZSBsYXN0IHVwZGF0ZS5cbiAqIEByZXR1cm5zIHtudW1iZXJ9IHRoZSBuZXcgY3VycmVudCB2YWx1ZS5cbiAqL1xuXG4vLyBUT0RPOiBBZGQgc29tZSBpbnRlcnBvbGF0aW9uIGZ1bmN0aW9ucy5cblxuLyoqXG4gKiBcbiAqIEBwYXJhbSB7bnVtYmVyfSByYXRlIHRoZSBudW1iZXIgb2YgZnJlcXVlbmN5IHZhbHVlcyB0byBzaGlmdCBieSBwZXIgc2Vjb25kLlxuICogQHJldHVybnMge2ludGVycG9sYXRvcn0gdGhlIGludGVycG9sYXRpb24gZnVuY3Rpb24gd2l0aCB0aGUgZ2l2ZW4gcmF0ZS5cbiAqL1xuZnVuY3Rpb24gY3JlYXRlRWFzZUxpbmVhcihyYXRlKSB7XG4gICAgcmV0dXJuIChjdXJyZW50LCBkZXN0LCBmcmFtZURlbHRhKSA9PiB7XG4gICAgICAgIGxldCBkaXJlY3Rpb24gPSAxO1xuICAgICAgICBpZiAoZGVzdCA8IGN1cnJlbnQpIHtcbiAgICAgICAgICAgIGRpcmVjdGlvbiA9IC0xO1xuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjdXJyZW50ICsgKE1hdGgubWluKHJhdGUgKiBmcmFtZURlbHRhLCBkZXN0KSAqIGRpcmVjdGlvbik7XG4gICAgfTtcbn1cblxuLyoqQG1vZHVsZSAqL1xuZXhwb3J0IHsgY3JlYXRlRWFzZUxpbmVhciB9OyJdLCJuYW1lcyI6WyJjcmVhdGVFYXNlTGluZWFyIiwicmF0ZSIsImN1cnJlbnQiLCJkZXN0IiwiZnJhbWVEZWx0YSIsImRpcmVjdGlvbiIsIk1hdGgiLCJtaW4iXSwic291cmNlUm9vdCI6IiJ9\n//# sourceURL=webpack-internal:///./src/support/easings.js\n"); + +/***/ }), + +/***/ "./src/support/support.js": +/*!********************************!*\ + !*** ./src/support/support.js ***! + \********************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"colors\": () => (/* reexport module object */ _colors_js__WEBPACK_IMPORTED_MODULE_0__),\n/* harmony export */ \"easings\": () => (/* reexport module object */ _easings_js__WEBPACK_IMPORTED_MODULE_1__)\n/* harmony export */ });\n/* harmony import */ var _colors_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./colors.js */ \"./src/support/colors.js\");\n/* harmony import */ var _easings_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./easings.js */ \"./src/support/easings.js\");\n\n\n/**@module */\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvc3VwcG9ydC9zdXBwb3J0LmpzLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBRUEiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9hdWRpb3Nob3draXQvLi9zcmMvc3VwcG9ydC9zdXBwb3J0LmpzPzgzMjYiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0ICogYXMgY29sb3JzIGZyb20gXCIuL2NvbG9ycy5qc1wiO1xuaW1wb3J0ICogYXMgZWFzaW5ncyBmcm9tIFwiLi9lYXNpbmdzLmpzXCI7XG5cbi8qKkBtb2R1bGUgKi9cbmV4cG9ydCB7IGNvbG9ycywgZWFzaW5ncyB9OyJdLCJuYW1lcyI6WyJjb2xvcnMiLCJlYXNpbmdzIl0sInNvdXJjZVJvb3QiOiIifQ==\n//# sourceURL=webpack-internal:///./src/support/support.js\n"); + +/***/ }), + +/***/ "./src/visualization/Visualizer.js": +/*!*****************************************!*\ + !*** ./src/visualization/Visualizer.js ***! + \*****************************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ Visualizer)\n/* harmony export */ });\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\nvar _stream = /*#__PURE__*/new WeakMap();\n\nvar _analyzing = /*#__PURE__*/new WeakMap();\n\nvar _updateListeners = /*#__PURE__*/new WeakMap();\n\nvar _audioCtx = /*#__PURE__*/new WeakMap();\n\nvar _source = /*#__PURE__*/new WeakMap();\n\nvar _analyzer = /*#__PURE__*/new WeakMap();\n\nvar _buffer = /*#__PURE__*/new WeakMap();\n\nvar _lastUpdate = /*#__PURE__*/new WeakMap();\n\n/**@module */\n\n/**\n * @callback visualizerUpdateListener\n * @param {number} delta elapsed time since last update.\n * @param {Uint8Array} bins the bins with varying frequency values. \n */\n\n/**\n * Provides a simplified access point to the frequency bins in the form of a visualization update listener.\n */\nvar Visualizer = /*#__PURE__*/function () {\n /**\n * \n * @param {MediaSource|HTMLMediaElement} mediaSource a media source to analyze.\n * @param {number} [fftSize = 1024] the size of the fft window.\n */\n function Visualizer(mediaSource) {\n var fftSize = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : 1024;\n\n _classCallCheck(this, Visualizer);\n\n _classPrivateFieldInitSpec(this, _stream, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _analyzing, {\n writable: true,\n value: false\n });\n\n _classPrivateFieldInitSpec(this, _updateListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _audioCtx, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _source, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _analyzer, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _buffer, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _lastUpdate, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldSet(this, _stream, mediaSource);\n\n _classPrivateFieldSet(this, _analyzing, false);\n\n _classPrivateFieldSet(this, _updateListeners, []);\n\n _classPrivateFieldSet(this, _audioCtx, new window.AudioContext());\n\n if (mediaSource instanceof HTMLMediaElement) {\n _classPrivateFieldSet(this, _source, _classPrivateFieldGet(this, _audioCtx).createMediaElementSource(_classPrivateFieldGet(this, _stream)));\n } else {\n _classPrivateFieldSet(this, _source, _classPrivateFieldGet(this, _audioCtx).createMediaStreamSource(_classPrivateFieldGet(this, _stream)));\n }\n\n _classPrivateFieldSet(this, _analyzer, _classPrivateFieldGet(this, _audioCtx).createAnalyser());\n\n _classPrivateFieldGet(this, _analyzer).fftSize = fftSize;\n\n _classPrivateFieldSet(this, _buffer, new Uint8Array(_classPrivateFieldGet(this, _analyzer).frequencyBinCount));\n\n _classPrivateFieldGet(this, _source).connect(_classPrivateFieldGet(this, _analyzer));\n\n _classPrivateFieldGet(this, _analyzer).connect(_classPrivateFieldGet(this, _audioCtx).destination);\n\n _classPrivateFieldSet(this, _lastUpdate, null);\n }\n /**\n * Begins analyzing and sending out update pings.\n */\n\n\n _createClass(Visualizer, [{\n key: \"analyze\",\n value: function analyze() {\n if (_classPrivateFieldGet(this, _analyzing)) {\n return;\n }\n\n _classPrivateFieldSet(this, _analyzing, true);\n\n var self = this; // since calling from requestAnimationFrame means \"this\" is no longer set to produced object.\n\n var update = function update(timestamp) {\n if (!_classPrivateFieldGet(self, _analyzing)) return;\n\n if (!_classPrivateFieldGet(self, _lastUpdate)) {\n _classPrivateFieldSet(self, _lastUpdate, timestamp);\n }\n\n var delta = timestamp - _classPrivateFieldGet(self, _lastUpdate);\n\n _classPrivateFieldGet(self, _analyzer).getByteFrequencyData(_classPrivateFieldGet(self, _buffer));\n\n _classPrivateFieldGet(self, _updateListeners).forEach(function (listener) {\n listener(delta, _classPrivateFieldGet(self, _buffer));\n });\n\n requestAnimationFrame(update);\n };\n\n requestAnimationFrame(update);\n }\n /**\n * Stops the analysis. Listeners will stop receiving bins.\n */\n\n }, {\n key: \"stop\",\n value: function stop() {\n _classPrivateFieldSet(this, _analyzing, false);\n }\n /**\n * \n * @param {visualizerUpdateListener} listener the visualizer update listener to be registered.\n * @returns {boolean} true if and only if the listener was successfully added.\n */\n\n }, {\n key: \"addUpdateListener\",\n value: function addUpdateListener(listener) {\n if (_classPrivateFieldGet(this, _updateListeners).includes(listener)) ;\n\n _classPrivateFieldGet(this, _updateListeners).push(listener);\n\n return true;\n }\n /**\n * \n * @param {visualizerUpdateListener} listener the visualizer update listener to remove.\n * @returns {boolean} true if and only if the removal of the listener was a success.\n */\n\n }, {\n key: \"removeUpdateListener\",\n value: function removeUpdateListener(listener) {\n var removeIndex = _classPrivateFieldGet(this, _updateListeners).indexOf(listener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _updateListeners).splice(removeIndex, 1);\n\n return true;\n }\n /**\n * @returns {number} The number of bins based on the size of the FFT window.\n */\n\n }, {\n key: \"numberOfBins\",\n get: function get() {\n return _classPrivateFieldGet(this, _buffer).length;\n }\n /**\n * \n * @returns {number} The fft window size.\n */\n\n }, {\n key: \"fftSize\",\n get: function get() {\n return _classPrivateFieldGet(this, _analyzer).fftSize;\n }\n }]);\n\n return Visualizer;\n}();\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/visualization/Visualizer.js.js","mappings":";;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAAA;;AAEA;AACA;AACA;AACA;AACA;;AAEA;AACA;AACA;IACqBA;AAUjB;AACJ;AACA;AACA;AACA;AACI,sBAAYC,WAAZ,EAAyC;AAAA,QAAhBC,OAAgB,uEAAN,IAAM;;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA,aAb5B;AAa4B;;AAAA;AAAA;AAAA,aAZtB;AAYsB;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AAAA;AAAA;AAAA;AAAA;;AACrC,yCAAeD,WAAf;;AACA,4CAAkB,KAAlB;;AACA,kDAAwB,EAAxB;;AACA,2CAAiB,IAAIE,MAAM,CAACC,YAAX,EAAjB;;AACA,QAAIH,WAAW,YAAYI,gBAA3B,EAA6C;AACzC,2CAAe,uCAAeC,wBAAf,uBAAwC,IAAxC,WAAf;AACH,KAFD,MAEO;AACH,2CAAe,uCAAeC,uBAAf,uBAAuC,IAAvC,WAAf;AACH;;AACD,2CAAiB,uCAAeC,cAAf,EAAjB;;AACA,2CAAeN,OAAf,GAAyBA,OAAzB;;AACA,yCAAe,IAAIO,UAAJ,CAAe,uCAAeC,iBAA9B,CAAf;;AACA,yCAAaC,OAAb,uBAAqB,IAArB;;AACA,2CAAeA,OAAf,CAAuB,uCAAeC,WAAtC;;AAEA,6CAAmB,IAAnB;AACH;AAGD;AACJ;AACA;;;;;WACI,mBAAU;AACN,gCAAI,IAAJ,eAAqB;AACjB;AACH;;AACD,8CAAkB,IAAlB;;AACA,UAAIC,IAAI,GAAG,IAAX,CALM,CAKW;;AACjB,UAAMC,MAAM,GAAG,SAATA,MAAS,CAACC,SAAD,EAAe;AAC1B,YAAI,uBAACF,IAAD,aAAJ,EAAsB;;AAEtB,YAAI,uBAACA,IAAD,cAAJ,EAAuB;AACnB,gCAAAA,IAAI,eAAeE,SAAf,CAAJ;AACH;;AACD,YAAIC,KAAK,GAAGD,SAAS,yBAAGF,IAAH,cAArB;;AACA,8BAAAA,IAAI,YAAJ,CAAeI,oBAAf,uBAAoCJ,IAApC;;AAEA,8BAAAA,IAAI,mBAAJ,CAAsBK,OAAtB,CAA8B,UAAAC,QAAQ,EAAI;AACtCA,UAAAA,QAAQ,CAACH,KAAD,wBAAQH,IAAR,WAAR;AACH,SAFD;;AAGAO,QAAAA,qBAAqB,CAACN,MAAD,CAArB;AACH,OAbD;;AAcAM,MAAAA,qBAAqB,CAACN,MAAD,CAArB;AACH;AAED;AACJ;AACA;;;;WACI,gBAAO;AACH,8CAAkB,KAAlB;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,2BAAkBK,QAAlB,EAA4B;AACxB,UAAI,8CAAsBE,QAAtB,CAA+BF,QAA/B,CAAJ,EAA6C;;AAC7C,oDAAsBG,IAAtB,CAA2BH,QAA3B;;AACA,aAAO,IAAP;AACH;AAED;AACJ;AACA;AACA;AACA;;;;WACI,8BAAqBA,QAArB,EAA+B;AAC3B,UAAMI,WAAW,GAAG,8CAAsBC,OAAtB,CAA8BL,QAA9B,CAApB;;AACA,UAAII,WAAW,GAAG,CAAlB,EAAqB,OAAO,KAAP;;AAErB,oDAAsBE,MAAtB,CAA6BF,WAA7B,EAA0C,CAA1C;;AACA,aAAO,IAAP;AACH;AAGD;AACJ;AACA;;;;SACI,eAAmB;AACf,aAAO,qCAAaG,MAApB;AACH;AAED;AACJ;AACA;AACA;;;;SACI,eAAc;AACV,aAAO,uCAAexB,OAAtB;AACH","sources":["webpack://audioshowkit/./src/visualization/Visualizer.js?da4c"],"sourcesContent":["/**@module */\n\n/**\n * @callback visualizerUpdateListener\n * @param {number} delta elapsed time since last update.\n * @param {Uint8Array} bins the bins with varying frequency values. \n */\n\n/**\n * Provides a simplified access point to the frequency bins in the form of a visualization update listener.\n */\nexport default class Visualizer {\n    #stream;\n    #analyzing = false;\n    #updateListeners = [];\n    #audioCtx;\n    #source;\n    #analyzer;\n    #buffer;\n    #lastUpdate;\n\n    /**\n     * \n     * @param {MediaSource|HTMLMediaElement} mediaSource a media source to analyze.\n     * @param {number} [fftSize = 1024] the size of the fft window.\n     */\n    constructor(mediaSource, fftSize = 1024) {\n        this.#stream = mediaSource;\n        this.#analyzing = false;\n        this.#updateListeners = [];\n        this.#audioCtx = new window.AudioContext();\n        if (mediaSource instanceof HTMLMediaElement) {\n            this.#source = this.#audioCtx.createMediaElementSource(this.#stream);\n        } else {\n            this.#source = this.#audioCtx.createMediaStreamSource(this.#stream);\n        }\n        this.#analyzer = this.#audioCtx.createAnalyser();\n        this.#analyzer.fftSize = fftSize;\n        this.#buffer = new Uint8Array(this.#analyzer.frequencyBinCount);\n        this.#source.connect(this.#analyzer);\n        this.#analyzer.connect(this.#audioCtx.destination);\n\n        this.#lastUpdate = null;\n    }\n\n\n    /**\n     * Begins analyzing and sending out update pings.\n     */\n    analyze() {\n        if (this.#analyzing) {\n            return;\n        }\n        this.#analyzing = true;\n        let self = this; // since calling from requestAnimationFrame means \"this\" is no longer set to produced object.\n        const update = (timestamp) => {\n            if (!self.#analyzing) return;\n\n            if (!self.#lastUpdate) {\n                self.#lastUpdate = timestamp;\n            }\n            let delta = timestamp - self.#lastUpdate;\n            self.#analyzer.getByteFrequencyData(self.#buffer);\n\n            self.#updateListeners.forEach(listener => {\n                listener(delta, self.#buffer);\n            });\n            requestAnimationFrame(update);\n        };\n        requestAnimationFrame(update);\n    }\n\n    /**\n     * Stops the analysis. Listeners will stop receiving bins.\n     */\n    stop() {\n        this.#analyzing = false;\n    }\n\n    /**\n     * \n     * @param {visualizerUpdateListener} listener the visualizer update listener to be registered.\n     * @returns {boolean} true if and only if the listener was successfully added.\n     */\n    addUpdateListener(listener) {\n        if (this.#updateListeners.includes(listener));\n        this.#updateListeners.push(listener);\n        return true;\n    }\n\n    /**\n     * \n     * @param {visualizerUpdateListener} listener the visualizer update listener to remove.\n     * @returns {boolean} true if and only if the removal of the listener was a success.\n     */\n    removeUpdateListener(listener) {\n        const removeIndex = this.#updateListeners.indexOf(listener);\n        if (removeIndex < 0) return false;\n\n        this.#updateListeners.splice(removeIndex, 1);\n        return true;\n    }\n\n\n    /**\n     * @returns {number} The number of bins based on the size of the FFT window.\n     */\n    get numberOfBins() {\n        return this.#buffer.length;\n    }\n\n    /**\n     * \n     * @returns {number} The fft window size.\n     */\n    get fftSize() {\n        return this.#analyzer.fftSize;\n    }\n}"],"names":["Visualizer","mediaSource","fftSize","window","AudioContext","HTMLMediaElement","createMediaElementSource","createMediaStreamSource","createAnalyser","Uint8Array","frequencyBinCount","connect","destination","self","update","timestamp","delta","getByteFrequencyData","forEach","listener","requestAnimationFrame","includes","push","removeIndex","indexOf","splice","length"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/visualization/Visualizer.js\n"); + +/***/ }), + +/***/ "./src/visualization/VisualizerUpdateManager.js": +/*!******************************************************!*\ + !*** ./src/visualization/VisualizerUpdateManager.js ***! + \******************************************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"default\": () => (/* binding */ VisualizerUpdateManager)\n/* harmony export */ });\n/* harmony import */ var _Visualizer_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Visualizer.js */ \"./src/visualization/Visualizer.js\");\nfunction _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }\n\nfunction _nonIterableSpread() { throw new TypeError(\"Invalid attempt to spread non-iterable instance.\\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method.\"); }\n\nfunction _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === \"string\") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === \"Object\" && o.constructor) n = o.constructor.name; if (n === \"Map\" || n === \"Set\") return Array.from(o); if (n === \"Arguments\" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }\n\nfunction _iterableToArray(iter) { if (typeof Symbol !== \"undefined\" && iter[Symbol.iterator] != null || iter[\"@@iterator\"] != null) return Array.from(iter); }\n\nfunction _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }\n\nfunction _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) { arr2[i] = arr[i]; } return arr2; }\n\nfunction _classCallCheck(instance, Constructor) { if (!(instance instanceof Constructor)) { throw new TypeError(\"Cannot call a class as a function\"); } }\n\nfunction _defineProperties(target, props) { for (var i = 0; i < props.length; i++) { var descriptor = props[i]; descriptor.enumerable = descriptor.enumerable || false; descriptor.configurable = true; if (\"value\" in descriptor) descriptor.writable = true; Object.defineProperty(target, descriptor.key, descriptor); } }\n\nfunction _createClass(Constructor, protoProps, staticProps) { if (protoProps) _defineProperties(Constructor.prototype, protoProps); if (staticProps) _defineProperties(Constructor, staticProps); Object.defineProperty(Constructor, \"prototype\", { writable: false }); return Constructor; }\n\nfunction _classPrivateFieldInitSpec(obj, privateMap, value) { _checkPrivateRedeclaration(obj, privateMap); privateMap.set(obj, value); }\n\nfunction _checkPrivateRedeclaration(obj, privateCollection) { if (privateCollection.has(obj)) { throw new TypeError(\"Cannot initialize the same private elements twice on an object\"); } }\n\nfunction _classPrivateFieldGet(receiver, privateMap) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"get\"); return _classApplyDescriptorGet(receiver, descriptor); }\n\nfunction _classApplyDescriptorGet(receiver, descriptor) { if (descriptor.get) { return descriptor.get.call(receiver); } return descriptor.value; }\n\nfunction _classPrivateFieldSet(receiver, privateMap, value) { var descriptor = _classExtractFieldDescriptor(receiver, privateMap, \"set\"); _classApplyDescriptorSet(receiver, descriptor, value); return value; }\n\nfunction _classExtractFieldDescriptor(receiver, privateMap, action) { if (!privateMap.has(receiver)) { throw new TypeError(\"attempted to \" + action + \" private field on non-instance\"); } return privateMap.get(receiver); }\n\nfunction _classApplyDescriptorSet(receiver, descriptor, value) { if (descriptor.set) { descriptor.set.call(receiver, value); } else { if (!descriptor.writable) { throw new TypeError(\"attempted to set read only private field\"); } descriptor.value = value; } }\n\n\n/**@module */\n\n/**\n * @callback visualizerBinUpdateListener\n * @param {number} timeDelta elapsed time since last update.\n * @param {number} amplitude The amplitude of the associated bin.\n * @param {number} ampDelta change in amplitude of the frequency bin.\n */\n\n/**\n * @callback visualizerRangedUpdateListener\n * @param {number} timeDelta elapsed time since last update.\n * @param {number} bins the bins of the range.\n */\n\n/**\n * A visualizer update manager offers an extra layer of abstraction on top of the {@link Visualizer}'s update listener.\n * \n * Specifically, the update manager handles updates directly from the {@link Visualizer} and checks for changes in the individual bins. These changes are then broadcasted to the individual bin listeners.\n * In the rare event that a bin has not changed, then it will not receive an update call.\n */\n\nvar _binnedListeners = /*#__PURE__*/new WeakMap();\n\nvar _rangedListeners = /*#__PURE__*/new WeakMap();\n\nvar _lastBins = /*#__PURE__*/new WeakMap();\n\nvar _visualizer = /*#__PURE__*/new WeakMap();\n\nvar _visualizerListener = /*#__PURE__*/new WeakMap();\n\nvar VisualizerUpdateManager = /*#__PURE__*/function () {\n /**\n * \n * @param {Visualizer} visualizer the visualizer this manager obtains data from.\n */\n function VisualizerUpdateManager(visualizer) {\n var _this = this;\n\n _classCallCheck(this, VisualizerUpdateManager);\n\n _classPrivateFieldInitSpec(this, _binnedListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _rangedListeners, {\n writable: true,\n value: []\n });\n\n _classPrivateFieldInitSpec(this, _lastBins, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _visualizer, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldInitSpec(this, _visualizerListener, {\n writable: true,\n value: void 0\n });\n\n _classPrivateFieldSet(this, _binnedListeners, []);\n\n _classPrivateFieldSet(this, _rangedListeners, []);\n\n for (var i = 0; i < visualizer.getNumberOfBins(); i++) {\n _classPrivateFieldGet(this, _binnedListeners).push([]);\n }\n\n _classPrivateFieldSet(this, _lastBins, new Uint8Array(_classPrivateFieldGet(this, _binnedListeners).length));\n\n _classPrivateFieldSet(this, _visualizer, visualizer);\n\n _classPrivateFieldSet(this, _visualizerListener, function (delta, bins) {\n var sortedCopyOfRangedListeners = _toConsumableArray(_classPrivateFieldGet(_this, _rangedListeners)); // We assume this is sorted properly. A priority queue could be better.\n\n\n var _loop = function _loop(binInd) {\n var lastBin = _classPrivateFieldGet(_this, _lastBins)[binInd];\n\n if (lastBin !== bins[binInd]) {\n _classPrivateFieldGet(_this, _binnedListeners)[binInd].forEach(function (listener) {\n listener(delta, bins[binInd], bins[binInd] - lastBin);\n });\n\n for (var rangedInd = 0; rangedInd < sortedCopyOfRangedListeners.length; rangedInd++) {\n // Could switch to a while loop.\n var _sortedCopyOfRangedLi = sortedCopyOfRangedListeners[rangedInd],\n lower = _sortedCopyOfRangedLi.lower,\n upper = _sortedCopyOfRangedLi.upper,\n listener = _sortedCopyOfRangedLi.listener;\n if (lower > binInd) break; // Don't need to check the rest if the current lowest minimum is greater than the current bin index.\n\n if (binInd <= upper) {\n listener(delta, bins.slice(lower, upper));\n sortedCopyOfRangedListeners.shift();\n rangedInd--;\n }\n }\n\n _classPrivateFieldGet(_this, _lastBins)[binInd] = bins[binInd];\n }\n };\n\n for (var binInd = 0; binInd < _classPrivateFieldGet(_this, _lastBins).length; binInd++) {\n _loop(binInd);\n }\n });\n\n visualizer.addUpdateListener(_classPrivateFieldGet(this, _visualizerListener));\n }\n /**\n * Adds a listener to a specific frequency bin.\n * \n * @param {object} freqBinListener the listener for a specific frequency bin.\n * @param {number} freqBinListener.freqBin the frequency bin this update listener should listen to.\n * @param {visualizerBinUpdateListener} freqBinListener.listener the listener itself that will be called upon the bin updating.\n * @returns {boolean} true if and only if the updater was added successfully.\n */\n\n\n _createClass(VisualizerUpdateManager, [{\n key: \"AddVisualizerBinUpdateListener\",\n value: function AddVisualizerBinUpdateListener(_ref) {\n var freqBin = _ref.freqBin,\n listener = _ref.listener;\n if (_classPrivateFieldGet(this, _binnedListeners)[freqBin].includes(listener)) return false;\n\n _classPrivateFieldGet(this, _binnedListeners)[freqBin].push(listener);\n\n return true;\n }\n /**\n * Similar to {@link VisualizerUpdateManager#AddVisualizerBinUpdateListener}, this method adds a listener for to a range of bins.\n * \n * @param {object} rangedUpdateListener The ranged update listener to add.\n * @param {number} rangedUpdateListener.lower The lower bound of the bins to listen to (inclusive).\n * @param {number} rangedUpdateListener.upper The upper bound of the bins to listen to (inclusive).\n * @param {visualizerRangedUpdateListener} rangedUpdateListener.listener The listener to register to the range. \n * @returns {boolean} True if and only if the ranged listener was added successfully.\n */\n\n }, {\n key: \"addVisualizerRangedUpdateListener\",\n value: function addVisualizerRangedUpdateListener(_ref2) {\n var lower = _ref2.lower,\n upper = _ref2.upper,\n listener = _ref2.listener;\n var rangedListener = {\n lower: lower,\n upper: upper,\n listener: listener\n };\n if (_classPrivateFieldGet(this, _rangedListeners).includes(rangedListener)) return false;\n\n _classPrivateFieldGet(this, _rangedListeners).push(rangedListener);\n\n _classPrivateFieldGet(this, _rangedListeners).sort(function (a, b) {\n return a.lower - b.lower;\n });\n\n return true;\n }\n /**\n * \n * @param {object} binFreqListener The bin frequency listener to remove.\n * @param {number} binFreqListener.freqBin the frequency bin the update listener to be removed from is in.\n * @param {visualizerBinUpdateListener} binFreqListener.listener the listener that is to be removed.\n * @returns {boolean} true if and only if the listener was successfully removed.\n */\n\n }, {\n key: \"removeVisualizerBinUpdateListener\",\n value: function removeVisualizerBinUpdateListener(_ref3) {\n var freqBin = _ref3.freqBin,\n listener = _ref3.listener;\n\n var removeIndex = _classPrivateFieldGet(this, _binnedListeners)[freqBin].indexOf(listener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _binnedListeners)[freqBin].splice(removeIndex, 1);\n\n return true;\n }\n /**\n * Similar to {@link removeVisualizerBinUpdateListener}, this method removes the given listener from a range of bins.\n * \n * @param {object} rangedListener The ranged listener to remove.\n * @param {number} rangedListener.lower The lower bound of bins to remove the listener from (inclusive).\n * @param {number} rangedListener.upper The upper bound of bin to remove the listener from (inclusive.)\n * @param {visualizerRangedUpdateListener} rangedListener.listener The update listener to remove from the bins.\n * @returns {boolean} True if and only if the given listener was removed.\n */\n\n }, {\n key: \"removeVisualizerRangedUpdateListener\",\n value: function removeVisualizerRangedUpdateListener(_ref4) {\n var lower = _ref4.lower,\n upper = _ref4.upper,\n listener = _ref4.listener;\n var rangedListener = {\n lower: lower,\n upper: upper,\n listener: listener\n };\n\n var removeIndex = _classPrivateFieldGet(this, _rangedListeners).indexOf(rangedListener);\n\n if (removeIndex < 0) return false;\n\n _classPrivateFieldGet(this, _binnedListeners).splice(removeIndex, 1);\n\n return true;\n }\n /**\n * @param {object} listeners The listeners that this visualizer update manager should be set to use.\n * @param {visualizerBinUpdateListener[][]} listeners.binned The bin listeners.\n * @param {object[]} listeners.ranged The array of ranged listeners.\n * @param {number} listeners.ranged[].lower The lower bound of the listener.\n * @param {number} listeners.ranged[].upper The upper bound of the listener.\n * @param {visualizerRangedUpdateListener} listeners.ranged[].listener The listener for the previously mentioned ranges.\n * @returns {boolean} true if and only if successfully loaded the new listeners.\n */\n\n }, {\n key: \"loadListeners\",\n value: function loadListeners(listeners) {\n if (listeners.binned.length !== _classPrivateFieldGet(this, _binnedListeners).length) return false;\n\n _classPrivateFieldSet(this, _binnedListeners, listeners.binned);\n\n _classPrivateFieldSet(this, _rangedListeners, listeners.ranged.sort(function (a, b) {\n return a.lower - b.lower;\n }));\n\n return true;\n }\n /**\n * \n * @returns {object} All the listeners, both for binned, and ranged listeners. See {@link VisualizerUpdateManager#loadListeners} to see the structure of the returned object.\n */\n\n }, {\n key: \"retrieveListeners\",\n value: function retrieveListeners() {\n return {\n binned: _classPrivateFieldGet(this, _binnedListeners),\n ranged: _classPrivateFieldGet(this, _rangedListeners)\n };\n }\n /**\n * Clears this manager of all listeners.\n */\n\n }, {\n key: \"clearBinnedListeners\",\n value: function clearBinnedListeners() {\n _classPrivateFieldGet(this, _binnedListeners).forEach(function (bin) {\n bin.length = 0;\n });\n }\n /**\n * Unbinds this update manager from the initial visualizer. Effectively meaning this manager will no longer be used.\n */\n\n }, {\n key: \"unbindVisualizer\",\n value: function unbindVisualizer() {\n _classPrivateFieldGet(this, _visualizer).removeUpdateListener(_classPrivateFieldGet(this, _visualizerListener));\n }\n }]);\n\n return VisualizerUpdateManager;\n}();\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"file":"./src/visualization/VisualizerUpdateManager.js.js","mappings":"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","sources":["webpack://audioshowkit/./src/visualization/VisualizerUpdateManager.js?78e2"],"sourcesContent":["import Visualizer from \"./Visualizer.js\";\n\n/**@module */\n\n/**\n * @callback visualizerBinUpdateListener\n * @param {number} timeDelta elapsed time since last update.\n * @param {number} amplitude The amplitude of the associated bin.\n * @param {number} ampDelta change in amplitude of the frequency bin.\n */\n\n/**\n * @callback visualizerRangedUpdateListener\n * @param {number} timeDelta elapsed time since last update.\n * @param {number} bins the bins of the range.\n */\n\n\n/**\n * A visualizer update manager offers an extra layer of abstraction on top of the {@link Visualizer}'s update listener.\n * \n * Specifically, the update manager handles updates directly from the {@link Visualizer} and checks for changes in the individual bins. These changes are then broadcasted to the individual bin listeners.\n * In the rare event that a bin has not changed, then it will not receive an update call.\n */\nexport default class VisualizerUpdateManager {\n    #binnedListeners = [];\n    #rangedListeners = [];\n    #lastBins;\n    #visualizer;\n    #visualizerListener;\n\n\n    /**\n     * \n     * @param {Visualizer} visualizer the visualizer this manager obtains data from.\n     */\n    constructor(visualizer) {\n        this.#binnedListeners = [];\n        this.#rangedListeners = [];\n        for (let i = 0; i < visualizer.getNumberOfBins(); i++) {\n            this.#binnedListeners.push([]);\n        }\n        this.#lastBins = new Uint8Array(this.#binnedListeners.length);\n        this.#visualizer = visualizer;\n        this.#visualizerListener = (delta, bins) => {\n            const sortedCopyOfRangedListeners = [... this.#rangedListeners]; // We assume this is sorted properly. A priority queue could be better.\n            for (let binInd = 0; binInd < this.#lastBins.length; binInd++) {\n                const lastBin = this.#lastBins[binInd];\n                if (lastBin !== bins[binInd]) {\n                    this.#binnedListeners[binInd].forEach(listener => {\n                        listener(delta, bins[binInd], bins[binInd] - lastBin);\n                    });\n                    for (let rangedInd = 0; rangedInd < sortedCopyOfRangedListeners.length; rangedInd++) { // Could switch to a while loop.\n                        const { lower, upper, listener } = sortedCopyOfRangedListeners[rangedInd];\n                        if (lower > binInd) break; // Don't need to check the rest if the current lowest minimum is greater than the current bin index.\n                        if (binInd <= upper) {\n                            listener(delta, bins.slice(lower, upper));\n                            sortedCopyOfRangedListeners.shift();\n                            rangedInd--;\n                        }\n                    }\n                    this.#lastBins[binInd] = bins[binInd];\n                }\n            }\n        };\n        visualizer.addUpdateListener(this.#visualizerListener);\n    }\n\n\n    /**\n     * Adds a listener to a specific frequency bin.\n     * \n     * @param {object} freqBinListener the listener for a specific frequency bin.\n     * @param {number} freqBinListener.freqBin the frequency bin this update listener should listen to.\n     * @param {visualizerBinUpdateListener} freqBinListener.listener the listener itself that will be called upon the bin updating.\n     * @returns {boolean} true if and only if the updater was added successfully.\n     */\n    AddVisualizerBinUpdateListener({ freqBin, listener }) {\n        if (this.#binnedListeners[freqBin].includes(listener)) return false;\n        this.#binnedListeners[freqBin].push(listener);\n        return true;\n    }\n\n    /**\n     * Similar to {@link VisualizerUpdateManager#AddVisualizerBinUpdateListener}, this method adds a listener for to a range of bins.\n     * \n     * @param {object} rangedUpdateListener The ranged update listener to add.\n     * @param {number} rangedUpdateListener.lower The lower bound of the bins to listen to (inclusive).\n     * @param {number} rangedUpdateListener.upper The upper bound of the bins to listen to (inclusive).\n     * @param {visualizerRangedUpdateListener} rangedUpdateListener.listener The listener to register to the range. \n     * @returns {boolean} True if and only if the ranged listener was added successfully.\n     */\n    addVisualizerRangedUpdateListener({ lower, upper, listener }) {\n        const rangedListener = {\n            lower: lower,\n            upper: upper,\n            listener: listener\n        };\n        if (this.#rangedListeners.includes(rangedListener)) return false;\n        this.#rangedListeners.push(rangedListener);\n        this.#rangedListeners.sort((a, b) => a.lower - b.lower);\n        return true;\n    }\n\n    /**\n     * \n     * @param {object} binFreqListener The bin frequency listener to remove.\n     * @param {number} binFreqListener.freqBin the frequency bin the update listener to be removed from is in.\n     * @param {visualizerBinUpdateListener} binFreqListener.listener the listener that is to be removed.\n     * @returns {boolean} true if and only if the listener was successfully removed.\n     */\n    removeVisualizerBinUpdateListener({ freqBin, listener }) {\n        const removeIndex = this.#binnedListeners[freqBin].indexOf(listener);\n        if (removeIndex < 0) return false;\n        this.#binnedListeners[freqBin].splice(removeIndex, 1);\n        return true;\n    }\n\n    /**\n     * Similar to {@link removeVisualizerBinUpdateListener}, this method removes the given listener from a range of bins.\n     * \n     * @param {object} rangedListener The ranged listener to remove.\n     * @param {number} rangedListener.lower The lower bound of bins to remove the listener from (inclusive).\n     * @param {number} rangedListener.upper The upper bound of bin to remove the listener from (inclusive.)\n     * @param {visualizerRangedUpdateListener} rangedListener.listener The update listener to remove from the bins.\n     * @returns {boolean} True if and only if the given listener was removed.\n     */\n    removeVisualizerRangedUpdateListener({ lower, upper, listener }) {\n        const rangedListener = {\n            lower: lower,\n            upper: upper,\n            listener: listener\n        };\n        const removeIndex = this.#rangedListeners.indexOf(rangedListener);\n        if (removeIndex < 0) return false;\n        this.#binnedListeners.splice(removeIndex, 1);\n        return true;\n    }\n\n    /**\n     * @param {object} listeners The listeners that this visualizer update manager should be set to use.\n     * @param {visualizerBinUpdateListener[][]} listeners.binned The bin listeners.\n     * @param {object[]} listeners.ranged The array of ranged listeners.\n     * @param {number} listeners.ranged[].lower The lower bound of the listener.\n     * @param {number} listeners.ranged[].upper The upper bound of the listener.\n     * @param {visualizerRangedUpdateListener} listeners.ranged[].listener The listener for the previously mentioned ranges.\n     * @returns {boolean} true if and only if successfully loaded the new listeners.\n     */\n    loadListeners(listeners) {\n        if (listeners.binned.length !== this.#binnedListeners.length) return false;\n        this.#binnedListeners = listeners.binned;\n        this.#rangedListeners = listeners.ranged.sort((a, b) => a.lower - b.lower);\n        return true;\n    }\n\n    /**\n     * \n     * @returns {object} All the listeners, both for binned, and ranged listeners. See {@link VisualizerUpdateManager#loadListeners} to see the structure of the returned object.\n     */\n    retrieveListeners() {\n        return {\n            binned: this.#binnedListeners,\n            ranged: this.#rangedListeners\n        };\n    }\n\n    /**\n     * Clears this manager of all listeners.\n     */\n    clearBinnedListeners() {\n        this.#binnedListeners.forEach(bin => {\n            bin.length = 0;\n        });\n    }\n\n    /**\n     * Unbinds this update manager from the initial visualizer. Effectively meaning this manager will no longer be used.\n     */\n    unbindVisualizer() {\n        this.#visualizer.removeUpdateListener(this.#visualizerListener);\n    }\n}"],"names":["Visualizer","VisualizerUpdateManager","visualizer","i","getNumberOfBins","push","Uint8Array","length","delta","bins","sortedCopyOfRangedListeners","binInd","lastBin","forEach","listener","rangedInd","lower","upper","slice","shift","addUpdateListener","freqBin","includes","rangedListener","sort","a","b","removeIndex","indexOf","splice","listeners","binned","ranged","bin","removeUpdateListener"],"sourceRoot":""}\n//# sourceURL=webpack-internal:///./src/visualization/VisualizerUpdateManager.js\n"); + +/***/ }), + +/***/ "./src/visualization/visualization.js": +/*!********************************************!*\ + !*** ./src/visualization/visualization.js ***! + \********************************************/ +/***/ ((__unused_webpack___webpack_module__, __webpack_exports__, __webpack_require__) => { + +eval("__webpack_require__.r(__webpack_exports__);\n/* harmony export */ __webpack_require__.d(__webpack_exports__, {\n/* harmony export */ \"Visualizer\": () => (/* reexport safe */ _Visualizer_js__WEBPACK_IMPORTED_MODULE_0__[\"default\"]),\n/* harmony export */ \"VisualizerUpdateManager\": () => (/* reexport safe */ _VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__[\"default\"])\n/* harmony export */ });\n/* harmony import */ var _Visualizer_js__WEBPACK_IMPORTED_MODULE_0__ = __webpack_require__(/*! ./Visualizer.js */ \"./src/visualization/Visualizer.js\");\n/* harmony import */ var _VisualizerUpdateManager_js__WEBPACK_IMPORTED_MODULE_1__ = __webpack_require__(/*! ./VisualizerUpdateManager.js */ \"./src/visualization/VisualizerUpdateManager.js\");\n\n\n/**@module */\n\n//# sourceURL=[module]\n//# sourceMappingURL=data:application/json;charset=utf-8;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiLi9zcmMvdmlzdWFsaXphdGlvbi92aXN1YWxpemF0aW9uLmpzLmpzIiwibWFwcGluZ3MiOiI7Ozs7Ozs7QUFBQTtBQUNBO0FBRUEiLCJzb3VyY2VzIjpbIndlYnBhY2s6Ly9hdWRpb3Nob3draXQvLi9zcmMvdmlzdWFsaXphdGlvbi92aXN1YWxpemF0aW9uLmpzPzQxODIiXSwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IFZpc3VhbGl6ZXIgZnJvbSBcIi4vVmlzdWFsaXplci5qc1wiO1xuaW1wb3J0IFZpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyIGZyb20gXCIuL1Zpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyLmpzXCI7XG5cbi8qKkBtb2R1bGUgKi9cbmV4cG9ydCB7IFZpc3VhbGl6ZXIsIFZpc3VhbGl6ZXJVcGRhdGVNYW5hZ2VyIH07Il0sIm5hbWVzIjpbIlZpc3VhbGl6ZXIiLCJWaXN1YWxpemVyVXBkYXRlTWFuYWdlciJdLCJzb3VyY2VSb290IjoiIn0=\n//# sourceURL=webpack-internal:///./src/visualization/visualization.js\n"); + +/***/ }) + +/******/ }); +/************************************************************************/ +/******/ // The module cache +/******/ var __webpack_module_cache__ = {}; +/******/ +/******/ // The require function +/******/ function __webpack_require__(moduleId) { +/******/ // Check if module is in cache +/******/ var cachedModule = __webpack_module_cache__[moduleId]; +/******/ if (cachedModule !== undefined) { +/******/ return cachedModule.exports; +/******/ } +/******/ // Create a new module (and put it into the cache) +/******/ var module = __webpack_module_cache__[moduleId] = { +/******/ id: moduleId, +/******/ // no module.loaded needed +/******/ exports: {} +/******/ }; +/******/ +/******/ // Execute the module function +/******/ __webpack_modules__[moduleId](module, module.exports, __webpack_require__); +/******/ +/******/ // Return the exports of the module +/******/ return module.exports; +/******/ } +/******/ +/************************************************************************/ +/******/ /* webpack/runtime/compat get default export */ +/******/ (() => { +/******/ // getDefaultExport function for compatibility with non-harmony modules +/******/ __webpack_require__.n = (module) => { +/******/ var getter = module && module.__esModule ? +/******/ () => (module['default']) : +/******/ () => (module); +/******/ __webpack_require__.d(getter, { a: getter }); +/******/ return getter; +/******/ }; +/******/ })(); +/******/ +/******/ /* webpack/runtime/define property getters */ +/******/ (() => { +/******/ // define getter functions for harmony exports +/******/ __webpack_require__.d = (exports, definition) => { +/******/ for(var key in definition) { +/******/ if(__webpack_require__.o(definition, key) && !__webpack_require__.o(exports, key)) { +/******/ Object.defineProperty(exports, key, { enumerable: true, get: definition[key] }); +/******/ } +/******/ } +/******/ }; +/******/ })(); +/******/ +/******/ /* webpack/runtime/hasOwnProperty shorthand */ +/******/ (() => { +/******/ __webpack_require__.o = (obj, prop) => (Object.prototype.hasOwnProperty.call(obj, prop)) +/******/ })(); +/******/ +/******/ /* webpack/runtime/make namespace object */ +/******/ (() => { +/******/ // define __esModule on exports +/******/ __webpack_require__.r = (exports) => { +/******/ if(typeof Symbol !== 'undefined' && Symbol.toStringTag) { +/******/ Object.defineProperty(exports, Symbol.toStringTag, { value: 'Module' }); +/******/ } +/******/ Object.defineProperty(exports, '__esModule', { value: true }); +/******/ }; +/******/ })(); +/******/ +/************************************************************************/ +/******/ +/******/ // startup +/******/ // Load entry module and return exports +/******/ // This entry module can't be inlined because the eval-source-map devtool is used. +/******/ var __webpack_exports__ = __webpack_require__("./src/audioshowkitlib.js"); +/******/ audioshowkit = __webpack_exports__; +/******/ +/******/ })() +; \ No newline at end of file diff --git a/tutorials/assets/js/prism.js b/tutorials/assets/js/prism.js new file mode 100644 index 0000000..3c58e2a --- /dev/null +++ b/tutorials/assets/js/prism.js @@ -0,0 +1,7 @@ +/* PrismJS 1.28.0 +https://prismjs.com/download.html#themes=prism-tomorrow&languages=markup+css+clike+javascript */ +var _self="undefined"!=typeof window?window:"undefined"!=typeof WorkerGlobalScope&&self instanceof WorkerGlobalScope?self:{},Prism=function(e){var n=/(?:^|\s)lang(?:uage)?-([\w-]+)(?=\s|$)/i,t=0,r={},a={manual:e.Prism&&e.Prism.manual,disableWorkerMessageHandler:e.Prism&&e.Prism.disableWorkerMessageHandler,util:{encode:function e(n){return n instanceof i?new i(n.type,e(n.content),n.alias):Array.isArray(n)?n.map(e):n.replace(/&/g,"&").replace(/=g.reach);A+=w.value.length,w=w.next){var E=w.value;if(n.length>e.length)return;if(!(E instanceof i)){var P,L=1;if(y){if(!(P=l(b,A,e,m))||P.index>=e.length)break;var S=P.index,O=P.index+P[0].length,j=A;for(j+=w.value.length;S>=j;)j+=(w=w.next).value.length;if(A=j-=w.value.length,w.value instanceof i)continue;for(var C=w;C!==n.tail&&(jg.reach&&(g.reach=W);var z=w.prev;if(_&&(z=u(n,z,_),A+=_.length),c(n,z,L),w=u(n,z,new i(f,p?a.tokenize(N,p):N,k,N)),M&&u(n,w,M),L>1){var I={cause:f+","+d,reach:W};o(e,n,t,w.prev,A,I),g&&I.reach>g.reach&&(g.reach=I.reach)}}}}}}function s(){var e={value:null,prev:null,next:null},n={value:null,prev:e,next:null};e.next=n,this.head=e,this.tail=n,this.length=0}function u(e,n,t){var r=n.next,a={value:t,prev:n,next:r};return n.next=a,r.prev=a,e.length++,a}function c(e,n,t){for(var r=n.next,a=0;a"+i.content+""},!e.document)return e.addEventListener?(a.disableWorkerMessageHandler||e.addEventListener("message",(function(n){var t=JSON.parse(n.data),r=t.language,i=t.code,l=t.immediateClose;e.postMessage(a.highlight(i,a.languages[r],r)),l&&e.close()}),!1),a):a;var g=a.util.currentScript();function f(){a.manual||a.highlightAll()}if(g&&(a.filename=g.src,g.hasAttribute("data-manual")&&(a.manual=!0)),!a.manual){var h=document.readyState;"loading"===h||"interactive"===h&&g&&g.defer?document.addEventListener("DOMContentLoaded",f):window.requestAnimationFrame?window.requestAnimationFrame(f):window.setTimeout(f,16)}return a}(_self);"undefined"!=typeof module&&module.exports&&(module.exports=Prism),"undefined"!=typeof global&&(global.Prism=Prism); +Prism.languages.markup={comment:{pattern://,greedy:!0},prolog:{pattern:/<\?[\s\S]+?\?>/,greedy:!0},doctype:{pattern:/"'[\]]|"[^"]*"|'[^']*')+(?:\[(?:[^<"'\]]|"[^"]*"|'[^']*'|<(?!!--)|)*\]\s*)?>/i,greedy:!0,inside:{"internal-subset":{pattern:/(^[^\[]*\[)[\s\S]+(?=\]>$)/,lookbehind:!0,greedy:!0,inside:null},string:{pattern:/"[^"]*"|'[^']*'/,greedy:!0},punctuation:/^$|[[\]]/,"doctype-tag":/^DOCTYPE/i,name:/[^\s<>'"]+/}},cdata:{pattern://i,greedy:!0},tag:{pattern:/<\/?(?!\d)[^\s>\/=$<%]+(?:\s(?:\s*[^\s>\/=]+(?:\s*=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+(?=[\s>]))|(?=[\s/>])))+)?\s*\/?>/,greedy:!0,inside:{tag:{pattern:/^<\/?[^\s>\/]+/,inside:{punctuation:/^<\/?/,namespace:/^[^\s>\/:]+:/}},"special-attr":[],"attr-value":{pattern:/=\s*(?:"[^"]*"|'[^']*'|[^\s'">=]+)/,inside:{punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}},punctuation:/\/?>/,"attr-name":{pattern:/[^\s>\/]+/,inside:{namespace:/^[^\s>\/:]+:/}}}},entity:[{pattern:/&[\da-z]{1,8};/i,alias:"named-entity"},/&#x?[\da-f]{1,8};/i]},Prism.languages.markup.tag.inside["attr-value"].inside.entity=Prism.languages.markup.entity,Prism.languages.markup.doctype.inside["internal-subset"].inside=Prism.languages.markup,Prism.hooks.add("wrap",(function(a){"entity"===a.type&&(a.attributes.title=a.content.replace(/&/,"&"))})),Object.defineProperty(Prism.languages.markup.tag,"addInlined",{value:function(a,e){var s={};s["language-"+e]={pattern:/(^$)/i,lookbehind:!0,inside:Prism.languages[e]},s.cdata=/^$/i;var t={"included-cdata":{pattern://i,inside:s}};t["language-"+e]={pattern:/[\s\S]+/,inside:Prism.languages[e]};var n={};n[a]={pattern:RegExp("(<__[^>]*>)(?:))*\\]\\]>|(?!)".replace(/__/g,(function(){return a})),"i"),lookbehind:!0,greedy:!0,inside:t},Prism.languages.insertBefore("markup","cdata",n)}}),Object.defineProperty(Prism.languages.markup.tag,"addAttribute",{value:function(a,e){Prism.languages.markup.tag.inside["special-attr"].push({pattern:RegExp("(^|[\"'\\s])(?:"+a+")\\s*=\\s*(?:\"[^\"]*\"|'[^']*'|[^\\s'\">=]+(?=[\\s>]))","i"),lookbehind:!0,inside:{"attr-name":/^[^\s=]+/,"attr-value":{pattern:/=[\s\S]+/,inside:{value:{pattern:/(^=\s*(["']|(?!["'])))\S[\s\S]*(?=\2$)/,lookbehind:!0,alias:[e,"language-"+e],inside:Prism.languages[e]},punctuation:[{pattern:/^=/,alias:"attr-equals"},/"|'/]}}}})}}),Prism.languages.html=Prism.languages.markup,Prism.languages.mathml=Prism.languages.markup,Prism.languages.svg=Prism.languages.markup,Prism.languages.xml=Prism.languages.extend("markup",{}),Prism.languages.ssml=Prism.languages.xml,Prism.languages.atom=Prism.languages.xml,Prism.languages.rss=Prism.languages.xml; +!function(s){var e=/(?:"(?:\\(?:\r\n|[\s\S])|[^"\\\r\n])*"|'(?:\\(?:\r\n|[\s\S])|[^'\\\r\n])*')/;s.languages.css={comment:/\/\*[\s\S]*?\*\//,atrule:{pattern:/@[\w-](?:[^;{\s]|\s+(?![\s{]))*(?:;|(?=\s*\{))/,inside:{rule:/^@[\w-]+/,"selector-function-argument":{pattern:/(\bselector\s*\(\s*(?![\s)]))(?:[^()\s]|\s+(?![\s)])|\((?:[^()]|\([^()]*\))*\))+(?=\s*\))/,lookbehind:!0,alias:"selector"},keyword:{pattern:/(^|[^\w-])(?:and|not|only|or)(?![\w-])/,lookbehind:!0}}},url:{pattern:RegExp("\\burl\\((?:"+e.source+"|(?:[^\\\\\r\n()\"']|\\\\[^])*)\\)","i"),greedy:!0,inside:{function:/^url/i,punctuation:/^\(|\)$/,string:{pattern:RegExp("^"+e.source+"$"),alias:"url"}}},selector:{pattern:RegExp("(^|[{}\\s])[^{}\\s](?:[^{};\"'\\s]|\\s+(?![\\s{])|"+e.source+")*(?=\\s*\\{)"),lookbehind:!0},string:{pattern:e,greedy:!0},property:{pattern:/(^|[^-\w\xA0-\uFFFF])(?!\s)[-_a-z\xA0-\uFFFF](?:(?!\s)[-\w\xA0-\uFFFF])*(?=\s*:)/i,lookbehind:!0},important:/!important\b/i,function:{pattern:/(^|[^-a-z0-9])[-a-z0-9]+(?=\()/i,lookbehind:!0},punctuation:/[(){};:,]/},s.languages.css.atrule.inside.rest=s.languages.css;var t=s.languages.markup;t&&(t.tag.addInlined("style","css"),t.tag.addAttribute("style","css"))}(Prism); +Prism.languages.clike={comment:[{pattern:/(^|[^\\])\/\*[\s\S]*?(?:\*\/|$)/,lookbehind:!0,greedy:!0},{pattern:/(^|[^\\:])\/\/.*/,lookbehind:!0,greedy:!0}],string:{pattern:/(["'])(?:\\(?:\r\n|[\s\S])|(?!\1)[^\\\r\n])*\1/,greedy:!0},"class-name":{pattern:/(\b(?:class|extends|implements|instanceof|interface|new|trait)\s+|\bcatch\s+\()[\w.\\]+/i,lookbehind:!0,inside:{punctuation:/[.\\]/}},keyword:/\b(?:break|catch|continue|do|else|finally|for|function|if|in|instanceof|new|null|return|throw|try|while)\b/,boolean:/\b(?:false|true)\b/,function:/\b\w+(?=\()/,number:/\b0x[\da-f]+\b|(?:\b\d+(?:\.\d*)?|\B\.\d+)(?:e[+-]?\d+)?/i,operator:/[<>]=?|[!=]=?=?|--?|\+\+?|&&?|\|\|?|[?*/~^%]/,punctuation:/[{}[\];(),.:]/}; +Prism.languages.javascript=Prism.languages.extend("clike",{"class-name":[Prism.languages.clike["class-name"],{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$A-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\.(?:constructor|prototype))/,lookbehind:!0}],keyword:[{pattern:/((?:^|\})\s*)catch\b/,lookbehind:!0},{pattern:/(^|[^.]|\.\.\.\s*)\b(?:as|assert(?=\s*\{)|async(?=\s*(?:function\b|\(|[$\w\xA0-\uFFFF]|$))|await|break|case|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally(?=\s*(?:\{|$))|for|from(?=\s*(?:['"]|$))|function|(?:get|set)(?=\s*(?:[#\[$\w\xA0-\uFFFF]|$))|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)\b/,lookbehind:!0}],function:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*(?:\.\s*(?:apply|bind|call)\s*)?\()/,number:{pattern:RegExp("(^|[^\\w$])(?:NaN|Infinity|0[bB][01]+(?:_[01]+)*n?|0[oO][0-7]+(?:_[0-7]+)*n?|0[xX][\\dA-Fa-f]+(?:_[\\dA-Fa-f]+)*n?|\\d+(?:_\\d+)*n|(?:\\d+(?:_\\d+)*(?:\\.(?:\\d+(?:_\\d+)*)?)?|\\.\\d+(?:_\\d+)*)(?:[Ee][+-]?\\d+(?:_\\d+)*)?)(?![\\w$])"),lookbehind:!0},operator:/--|\+\+|\*\*=?|=>|&&=?|\|\|=?|[!=]==|<<=?|>>>?=?|[-+*/%&|^!=<>]=?|\.{3}|\?\?=?|\?\.?|[~:]/}),Prism.languages.javascript["class-name"][0].pattern=/(\b(?:class|extends|implements|instanceof|interface|new)\s+)[\w.\\]+/,Prism.languages.insertBefore("javascript","keyword",{regex:{pattern:RegExp("((?:^|[^$\\w\\xA0-\\uFFFF.\"'\\])\\s]|\\b(?:return|yield))\\s*)/(?:(?:\\[(?:[^\\]\\\\\r\n]|\\\\.)*\\]|\\\\.|[^/\\\\\\[\r\n])+/[dgimyus]{0,7}|(?:\\[(?:[^[\\]\\\\\r\n]|\\\\.|\\[(?:[^[\\]\\\\\r\n]|\\\\.|\\[(?:[^[\\]\\\\\r\n]|\\\\.)*\\])*\\])*\\]|\\\\.|[^/\\\\\\[\r\n])+/[dgimyus]{0,7}v[dgimyus]{0,7})(?=(?:\\s|/\\*(?:[^*]|\\*(?!/))*\\*/)*(?:$|[\r\n,.;:})\\]]|//))"),lookbehind:!0,greedy:!0,inside:{"regex-source":{pattern:/^(\/)[\s\S]+(?=\/[a-z]*$)/,lookbehind:!0,alias:"language-regex",inside:Prism.languages.regex},"regex-delimiter":/^\/|\/$/,"regex-flags":/^[a-z]+$/}},"function-variable":{pattern:/#?(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*[=:]\s*(?:async\s*)?(?:\bfunction\b|(?:\((?:[^()]|\([^()]*\))*\)|(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)\s*=>))/,alias:"function"},parameter:[{pattern:/(function(?:\s+(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*)?\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\))/,lookbehind:!0,inside:Prism.languages.javascript},{pattern:/(^|[^$\w\xA0-\uFFFF])(?!\s)[_$a-z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*=>)/i,lookbehind:!0,inside:Prism.languages.javascript},{pattern:/(\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*=>)/,lookbehind:!0,inside:Prism.languages.javascript},{pattern:/((?:\b|\s|^)(?!(?:as|async|await|break|case|catch|class|const|continue|debugger|default|delete|do|else|enum|export|extends|finally|for|from|function|get|if|implements|import|in|instanceof|interface|let|new|null|of|package|private|protected|public|return|set|static|super|switch|this|throw|try|typeof|undefined|var|void|while|with|yield)(?![$\w\xA0-\uFFFF]))(?:(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*\s*)\(\s*|\]\s*\(\s*)(?!\s)(?:[^()\s]|\s+(?![\s)])|\([^()]*\))+(?=\s*\)\s*\{)/,lookbehind:!0,inside:Prism.languages.javascript}],constant:/\b[A-Z](?:[A-Z_]|\dx?)*\b/}),Prism.languages.insertBefore("javascript","string",{hashbang:{pattern:/^#!.*/,greedy:!0,alias:"comment"},"template-string":{pattern:/`(?:\\[\s\S]|\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}|(?!\$\{)[^\\`])*`/,greedy:!0,inside:{"template-punctuation":{pattern:/^`|`$/,alias:"string"},interpolation:{pattern:/((?:^|[^\\])(?:\\{2})*)\$\{(?:[^{}]|\{(?:[^{}]|\{[^}]*\})*\})+\}/,lookbehind:!0,inside:{"interpolation-punctuation":{pattern:/^\$\{|\}$/,alias:"punctuation"},rest:Prism.languages.javascript}},string:/[\s\S]+/}},"string-property":{pattern:/((?:^|[,{])[ \t]*)(["'])(?:\\(?:\r\n|[\s\S])|(?!\2)[^\\\r\n])*\2(?=\s*:)/m,lookbehind:!0,greedy:!0,alias:"property"}}),Prism.languages.insertBefore("javascript","operator",{"literal-property":{pattern:/((?:^|[,{])[ \t]*)(?!\s)[_$a-zA-Z\xA0-\uFFFF](?:(?!\s)[$\w\xA0-\uFFFF])*(?=\s*:)/m,lookbehind:!0,alias:"property"}}),Prism.languages.markup&&(Prism.languages.markup.tag.addInlined("script","javascript"),Prism.languages.markup.tag.addAttribute("on(?:abort|blur|change|click|composition(?:end|start|update)|dblclick|error|focus(?:in|out)?|key(?:down|up)|load|mouse(?:down|enter|leave|move|out|over|up)|reset|resize|scroll|select|slotchange|submit|unload|wheel)","javascript")),Prism.languages.js=Prism.languages.javascript; diff --git a/webpack.dev.cjs b/webpack.dev.cjs index 6c542f7..f46422f 100644 --- a/webpack.dev.cjs +++ b/webpack.dev.cjs @@ -6,6 +6,9 @@ const webpackCommon = require("./webpack.common.cjs"); const devConfig = { mode: "development", devtool: "eval-source-map", + watchOptions: { + ignored: "./tutorials/**" + } }; module.exports = merge(webpackCommon, devConfig); \ No newline at end of file