From c626fb2855ea27e38c92e01759afe0e99daddff2 Mon Sep 17 00:00:00 2001 From: Cat Date: Wed, 14 Dec 2016 10:53:36 +0100 Subject: [PATCH] Embed: smaller design for controls --- embed/core.js | 95 ++++++++++++++++++++++++++++++-------- embed/mist.css | 96 ++++++++++++++++++++++++++++++++++----- embed/test.html | 6 +-- embed/wrappers/videojs.js | 12 +++-- 4 files changed, 174 insertions(+), 35 deletions(-) diff --git a/embed/core.js b/embed/core.js index d8aaeabf..c3ea7f00 100644 --- a/embed/core.js +++ b/embed/core.js @@ -52,6 +52,7 @@ MistPlayer.prototype.onready = function(dothis){ }; MistPlayer.prototype.play = false; MistPlayer.prototype.pause = false; +MistPlayer.prototype.paused = false; MistPlayer.prototype.volume = false; MistPlayer.prototype.loop = false; MistPlayer.prototype.fullscreen = false; @@ -99,7 +100,27 @@ MistPlayer.prototype.setTracks = function(usetracks){ } var time = this.element.currentTime; - this.updateSrc(urlAddParam(this.options.src,usetracks)); + var newurl; + if (this.options.source.type == 'html5/application/vnd.apple.mpegurl') { //for HLS, use a different format for track selection + newurl = this.options.src.split('/'); + var m3u8 = newurl.pop(); //take this off now, it will be added back later + for (var i in usetracks) { + //for audio or video tracks, just add the tracknumber between slashes + switch (i) { + case 'audio': + case 'video': + if (usetracks[i] == 0) { continue; } + newurl.push(usetracks[i]); + break; + } + } + newurl.push(m3u8); //put back index.m3u8 + newurl = newurl.join('/'); + } + else { + newurl = urlAddParam(this.options.src,usetracks); + } + this.updateSrc(newurl); if (this.element.readyState) { this.element.load(); } @@ -175,7 +196,10 @@ MistPlayer.prototype.buildMistControls = function(){ var zoom = options.width/480; if (zoom < 1) { zoom = Math.max(zoom,0.5); - controls.style.zoom = zoom; + if ('zoom' in controls.style) { controls.style.zoom = zoom; } + else { + controls.className += ' smaller'; //if css doesn't support zoom, apply smaller class to use smaller controls + } } else { zoom = 1; } ele.style['min-width'] = zoom*400+'px'; @@ -206,7 +230,7 @@ MistPlayer.prototype.buildMistControls = function(){ progress.getPos = function(e){ if (!isFinite(ele.duration)) { return 0; } var style = progress.currentStyle || window.getComputedStyle(progress, null); - var zoom = Number(controls.style.zoom == '' ? 1 : controls.style.zoom); + var zoom = Number(!('zoom' in controls.style) || controls.style.zoom == '' ? 1 : controls.style.zoom); var pos0 = progress.getBoundingClientRect().left - parseInt(style.borderLeftWidth,10); var perc = (e.clientX - pos0 * zoom) / progress.offsetWidth / zoom; @@ -268,7 +292,13 @@ MistPlayer.prototype.buildMistControls = function(){ sound.appendChild(volume); sound.getPos = function(ypos){ var style = this.currentStyle || window.getComputedStyle(this, null); - return 1 - Math.min(1,Math.max(0,(ypos - sound.getBoundingClientRect().top - parseInt(style.borderTopWidth,10)) / sound.offsetHeight)); + + var zoom = Number(!('zoom' in controls.style) || controls.style.zoom == '' ? 1 : controls.style.zoom); + + var pos0 = sound.getBoundingClientRect().top - parseInt(style.borderTopWidth,10); + var perc = (ypos - pos0 * zoom) / sound.offsetHeight / zoom; + var secs = Math.max(0,perc) * ele.duration; + return 1 - Math.min(1,Math.max(0,perc)); } volume.className = 'volume'; sound.title = 'Volume'; @@ -328,14 +358,15 @@ MistPlayer.prototype.buildMistControls = function(){ buttons.className = 'column'; controls.appendChild(buttons); - if ((this.setTracks(false)) && ((this.tracks.audio.length) || (this.tracks.video.length) || (this.tracks.subtitle.length)) && ((this.options.source.type != 'html5/application/vnd.apple.mpegurl') && (this.options.source.type != 'html5/video/ogg'))) { + if ((this.setTracks(false)) && ((this.tracks.audio.length) || (this.tracks.video.length) || (this.tracks.subtitle.length)) && (this.options.source.type != 'html5/video/ogg')) { /* - the player supports setting tracks; - there is something to choose - - it's not HLS, which would have to use a different way of switching tracks than this script currently uses + - it's not OGG, which doesn't have track selection yet */ + //prepare the html stuff var tracks = this.tracks; var tracksc = document.createElement('div'); tracksc.innerHTML = 'Tracks'; @@ -347,7 +378,7 @@ MistPlayer.prototype.buildMistControls = function(){ settings.className = 'settings'; me.trackselects = {}; - for (var i in tracks) { + for (var i in tracks) { //for each track type (video, audio, subtitle..) if (tracks[i].length) { var l = document.createElement('label'); settings.appendChild(l); @@ -359,10 +390,12 @@ MistPlayer.prototype.buildMistControls = function(){ l.appendChild(s); me.trackselects[i] = s; s.setAttribute('data-type',i); - for (var j in tracks[i]) { + for (var j in tracks[i]) { //for each track var o = document.createElement('option'); s.appendChild(o); o.value = tracks[i][j].trackid; + + //make up something logical for the track name var name; if ('name' in tracks[i][j]) { name = tracks[i][j].name; @@ -789,7 +822,7 @@ function mistPlay(streamName,options) { continue; } - if (mistplayers[p_shortname].isBrowserSupported(mime,loop[s],options,streaminfo)) { + if (mistplayers[p_shortname].isBrowserSupported(mime,loop[s],options,streaminfo,embedLog)) { embedLog('Found a working combo: '+mistplayers[p_shortname].name+' with '+mime+' @ '+loop[s].url); streaminfo.working[p_shortname].push(mime); if (!source) { @@ -903,7 +936,6 @@ function mistPlay(streamName,options) { if (player.setTracks(false)) { //gather track info - //tracks var tracks = { video: [], audio: [], @@ -982,34 +1014,61 @@ function mistPlay(streamName,options) { if (player.setTracks(false)) { player.onready(function(){ - //player.setTracks(usetracks); if ('setTracks' in options) { player.setTracks(options.setTracks); } }); } //monitor for errors - player.checkPlayingTimeout = false; + player.checkStalledTimeout = false; + player.checkProgressTimeout = false; element.addEventListener('error',function(e){ player.askNextCombo(); },true); var stalled = function(e){ - if (player.checkPlayingTimeout) { return; } - player.checkPlayingTimeout = setTimeout(function(){ - if (player.element.readyState >= 2) { return; } + if (player.checkStalledTimeout) { return; } + player.checkStalledTimeout = setTimeout(function(){ + if (player.paused) { return; } player.askNextCombo(); },5e3); }; element.addEventListener('stalled',stalled,true); element.addEventListener('waiting',stalled,true); var progress = function(e){ - if (player.checkPlayingTimeout) { - clearTimeout(player.checkPlayingTimeout); - player.checkPlayingTimeout = false; + if (player.checkStalledTimeout) { + clearTimeout(player.checkStalledTimeout); + player.checkStalledTimeout = false; player.cancelAskNextCombo(); } }; element.addEventListener('progress',progress,true); element.addEventListener('playing',progress,true); + element.addEventListener('play',function(){ + player.paused = false; + if ((!player.checkProgressTimeout) && (player.element) && ('currentTime' in player.element)) { + //check if the progress made is equal to the time spent + var lasttime = player.element.currentTime; + player.checkProgressTimeout = setInterval(function(){ + var newtime = player.element.currentTime; + var progress = newtime - lasttime; + lasttime = newtime; + if (progress == 0) { + player.addlog('There should be playback but nothing was played'); + player.askNextCombo(); + return; + } + if (progress < 4.9) { + player.addlog('It seems playback is \lagging ('+Math.round(100 - progress/0.5)/10+'%)'); + } + },5e3); + } + },true); + element.addEventListener('pause',function(){ + player.paused = true; + if (player.checkProgressTimeout) { + clearInterval(player.checkStalledTimeout); + player.checkStalledTimeout = false; + } + },true); if (player.resize) { //monitor for resizes and fire if needed diff --git a/embed/mist.css b/embed/mist.css index f123c353..35988a56 100644 --- a/embed/mist.css +++ b/embed/mist.css @@ -63,11 +63,16 @@ background-color: black; opacity: 0.6; position: absolute; - left: 1px; - right: 1px; + left: 0px; + right: 0px; bottom: -75px; display: flex; align-items: center; + text-shadow: none; +} +.mistplayer .controls.smaller { + height: 30px; + bottom: -30px; } .mistplayer:hover:not([data-hide]) .controls { bottom: 0; @@ -88,14 +93,13 @@ .mistplayer .controls .row { display: flex; flex-flow: row nowrap; + flex-shrink: 1; } .mistplayer .controls .column { display: flex; flex-flow: column nowrap; align-items: center; -} -.mistplayer .controls .row .button { - + flex-shrink: 1; } .mistplayer .controls .row .button:not(:first-child) { margin-left: 0; @@ -107,11 +111,29 @@ font-size: 16px; position: relative; background: transparent center none no-repeat; + background-size: contain; +} +.mistplayer .controls.smaller .button { + width: 15px; + line-height: 15px; + font-size: 8px; } .mistplayer .controls .button.play { height: 45px; margin-left: 15px; } +.mistplayer .controls.minimal .button.play, +.mistplayer .controls.minimal .button.fullscreen, +.mistplayer .controls.minimal .button.tracks, +.mistplayer .controls.minimal .button.loop, +.mistplayer .controls.minimal .button.timestamp, +.mistplayer .controls.minimal .button.sound { + display: none; +} +.mistplayer .controls.smaller .button.play { + height: 15px; + margin-left: 5px; +} .mistplayer .controls .button.play[data-state=playing] { background-image: url(""); } @@ -123,6 +145,10 @@ position: relative; margin: 15px; } +.mistplayer .controls.smaller .progress_container { + margin: 5px; + min-width: 20px; +} .mistplayer .controls .button.progress { height: 15px; border: 1px solid white; @@ -130,6 +156,9 @@ width: auto; margin: 0; } +.mistplayer .controls.smaller .button.progress { + height: 10px; +} .mistplayer .controls .button.progress .bar { background-color: white; position: absolute; @@ -137,6 +166,7 @@ top: 0; bottom: 0; left: 0; + min-width: 5px; } .mistplayer .controls .button.progress .buffer { background-color: white; @@ -156,6 +186,11 @@ display: none; font-size: 12px; } +.mistplayer .controls.smaller .progress_container .hint { + bottom: 17px; + font-size: 6px; + border-radius: 3px; +} .mistplayer .controls .progress_container .hint:after { content: ''; display: block; @@ -165,6 +200,10 @@ border-left-color: white; bottom: -5px; } +.mistplayer .controls.smaller .progress_container .hint:after { + border-width: 3px; + bottom: -3px; +} .mistplayer .controls .button.timestamp { width: auto; cursor: default; @@ -175,10 +214,17 @@ } .mistplayer .controls .button.sound { height: 65px; - width: 30px; - margin-left: 20px; + width: 43px; position: relative; background: url("") no-repeat; + background-position: center right; + border: 1px solid black; /*helps with white border*/ + flex-shrink: 0; +} +.mistplayer .controls.smaller .button.sound { + height: 19.5px; + width: 14px; + background-size: contain; } .mistplayer .controls .button.sound .speaker { width: 25px; @@ -186,27 +232,42 @@ margin: 0; background-image: url(""); position: absolute; - left: -15px; top: 30px; background-color: white; + box-shadow: inset 1px 1px black, inset -1px -1px black; /* helps with white shadow*/ } .mistplayer .controls .button.sound .speaker[data-muted] { background-color: transparent; } +.mistplayer .controls.smaller .button.sound .speaker { + width: 8px; + height: 8px; + top: 10px; + box-shadow: none; +} .mistplayer .controls .button.sound .volume { position: absolute; - bottom: 0; - left: 1px; + bottom: 0.5px; right: 1px; + width: 23.5px; background-color: white; opacity: 0.6; height: 100%; z-index: -1; + border-left: 1px solid transparent; /* these help with white showing through */ + border-right: 1px solid transparent; +} +.mistplayer .controls.smaller .button.sound .volume { + width: 6px; } .mistplayer .controls .button.loop { min-height: 45px; background-color: transparent; background-image: url(""); + box-shadow: inset 1px 1px black, inset -1px -1px black; /* helps with white shadow*/ +} +.mistplayer .controls.smaller .button.loop { + min-height: 15px; } .mistplayer .controls .button.loop[data-on] { background-color: rgba(255,255,255,0.6); @@ -215,18 +276,24 @@ background-image: url(""); height: 45px; } +.mistplayer .controls.smaller .button.fullscreen { + height: 15px; +} .mistplayer .controls .button.tracks { line-height: 25px; width: 100%; margin: 0; height: 25px; - padding: 0 15px; box-sizing: border-box; -webkit-user-select: none; -moz-user-select: none; -ms-user-select: none; user-select: none; } +.mistplayer .controls.smaller .button.tracks { + height: 10px; + line-height: 10px; +} .mistplayer .controls .tracks .settings { position: absolute; background-color: black; @@ -234,6 +301,10 @@ right: -1000px; bottom: 27px; } +.mistplayer .controls.smaller .tracks .settings { + padding: 2px 3px; + bottom: 12.5px; +} .mistplayer .controls .tracks:hover .settings { right: 0; } @@ -261,9 +332,12 @@ color: white; border: none; outline: none; + font-size: inherit; } .mistplayer .controls .tracks .settings label option { color: black; + background-color: white; + font-size: inherit; } @keyframes spin { diff --git a/embed/test.html b/embed/test.html index 8fc95132..8e7b2f27 100644 --- a/embed/test.html +++ b/embed/test.html @@ -25,8 +25,8 @@ - + @@ -83,11 +83,11 @@ //tryplayers.push('flash_strobe'); //tryplayers.push('silverlight'); streams = []; - //streams.push('live'); + streams.push('live'); //streams.push('subtel'); //streams.push('ogg'); //streams.push('vids+mist.mp4'); - streams.push('vids+hahalol.mp3'); + //streams.push('vids+hahalol.mp3'); //streams.push('lama'); //streams.push('bunny'); diff --git a/embed/wrappers/videojs.js b/embed/wrappers/videojs.js index 5bf34816..b4b0d239 100644 --- a/embed/wrappers/videojs.js +++ b/embed/wrappers/videojs.js @@ -5,10 +5,16 @@ mistplayers.videojs = { isMimeSupported: function (mimetype) { return (this.mimes.indexOf(mimetype) == -1 ? false : true); }, - isBrowserSupported: function (mimetype,source,options) { - if ((options.host.substr(0,7) == 'http://') && (source.url.substr(0,8) == 'https://')) { return false; } + isBrowserSupported: function (mimetype,source,options,streaminfo,logfunc) { + if ((options.host.substr(0,7) == 'http://') && (source.url.substr(0,8) == 'https://')) { + if (logfunc) { logfunc('HTTP/HTTPS mismatch for this source'); } + return false; + } var support = true; - if ((location.protocol == 'file:') && (mimetype == 'html5/application/vnd.apple.mpegurl')) { return false; } + if ((location.protocol == 'file:') && (mimetype == 'html5/application/vnd.apple.mpegurl')) { + if (logfunc) { logfunc('This source ('+mimetype+') won\'t work if the page is run via file://'); } + return false; + } return ('MediaSource' in window); }, player: function(){},