diff options
-rw-r--r-- | resources/media-controls.css | 9 | ||||
-rw-r--r-- | resources/media-controls.js | 75 |
2 files changed, 63 insertions, 21 deletions
diff --git a/resources/media-controls.css b/resources/media-controls.css index cbea462157c..62c39164747 100644 --- a/resources/media-controls.css +++ b/resources/media-controls.css @@ -16,7 +16,7 @@ button { display: block; position: relative; min-height: 40px; - min-width: 200px; + min-width: 230px; } .controls { @@ -52,3 +52,10 @@ button { background: url("") no-repeat; } +.fullscreen { + background: url('') no-repeat; +} + +.fullscreen.fullscreen-active { + background: url('') no-repeat; +} diff --git a/resources/media-controls.js b/resources/media-controls.js index 383b025ee0e..b8f100c6f05 100644 --- a/resources/media-controls.js +++ b/resources/media-controls.js @@ -5,19 +5,6 @@ (() => { "use strict"; - const MARKUP = ` - <div class="controls"> - <button id="play-pause-button"></button> - <input id="progress" type="range" value="0" min="0" max="100" step="1"></input> - <span id="position-duration-box" class="hidden"> - <span id="position-text">#1</span> - <span id="duration"> / #2</span> - </span> - <button id="volume-switch"></button> - <input id="volume-level" type="range" value="100" min="0" max="100" step="1"></input> - </div> - `; - // States. const BUFFERING = "buffering"; const ENDED = "ended"; @@ -50,6 +37,22 @@ } }; + function generateMarkup(isAudioOnly) { + return ` + <div class="controls"> + <button id="play-pause-button"></button> + <input id="progress" type="range" value="0" min="0" max="100" step="1"></input> + <span id="position-duration-box" class="hidden"> + <span id="position-text">#1</span> + <span id="duration"> / #2</span> + </span> + <button id="volume-switch"></button> + <input id="volume-level" type="range" value="100" min="0" max="100" step="1"></input> + ${isAudioOnly ? "" : '<button id="fullscreen-switch" class="fullscreen"></button>'} + </div> + `; + } + function camelCase(str) { const rdashes = /-(.)/g; return str.replace(rdashes, (str, p1) => { @@ -89,15 +92,16 @@ attributeFilter: ["controls"] }); + this.isAudioOnly = this.media.localName == "audio"; + // Create root element and load markup. this.root = document.createElement("div"); this.root.classList.add("root"); - this.root.innerHTML = MARKUP; + this.root.innerHTML = generateMarkup(this.isAudioOnly); this.controls.appendChild(this.root); - // Import elements. - this.elements = {}; - [ + + const elementNames = [ "duration", "play-pause-button", "position-duration-box", @@ -105,7 +109,15 @@ "progress", "volume-switch", "volume-level" - ].forEach(id => { + ]; + + if (!this.isAudioOnly) { + elementNames.push("fullscreen-switch"); + } + + // Import elements. + this.elements = {}; + elementNames.forEach(id => { this.elements[camelCase(id)] = this.controls.getElementById(id); }); @@ -182,6 +194,11 @@ { el: this.elements.volumeSwitch, type: "click" }, { el: this.elements.volumeLevel, type: "input" } ]; + + if (!this.isAudioOnly) { + this.controlEvents.push({ el: this.elements.fullscreenSwitch, type: "click" }); + } + this.controlEvents.forEach(({ el, type }) => { el.addEventListener(type, this); }); @@ -238,8 +255,7 @@ } render(from = this.state) { - const isAudioOnly = this.media.localName == "audio"; - if (!isAudioOnly) { + if (!this.isAudioOnly) { // XXX This should ideally use clientHeight/clientWidth, // but for some reason I couldn't figure out yet, // using it breaks layout. @@ -312,6 +328,9 @@ case this.elements.volumeSwitch: this.toggleMuted(); break; + case this.elements.fullscreenSwitch: + this.toggleFullscreen(); + break; } break; case "input": @@ -368,6 +387,22 @@ this.media.muted = !this.media.muted; } + toggleFullscreen() { + const { fullscreenEnabled, fullscreenElement } = document; + + const isElementFullscreen = fullscreenElement && fullscreenElement === this.media; + + if (fullscreenEnabled && isElementFullscreen) { + document.exitFullscreen().then(() => { + this.elements.fullscreenSwitch.classList.remove("fullscreen-active"); + }); + } else { + this.media.requestFullscreen().then(() => { + this.elements.fullscreenSwitch.classList.add("fullscreen-active"); + }); + } + } + changeVolume() { const volume = parseInt(this.elements.volumeLevel.value); if (!isNaN(volume)) { |