Compare commits
36 Commits
Author | SHA1 | Date | |
---|---|---|---|
b5fc21239b | |||
93cc9edd9a | |||
654e9cd623 | |||
73c3888309 | |||
4f0633fdc1 | |||
f41854ebe7 | |||
f398266206 | |||
4c17f98520 | |||
df84ce6e90 | |||
7161378da1 | |||
224b612ae7 | |||
19d7522722 | |||
ceace2a678 | |||
d627454b2a | |||
7ccbfad6ad | |||
91f8a158d2 | |||
8925dcde01 | |||
530abac3a7 | |||
e702d9a881 | |||
b0aeccb793 | |||
43d8d748ce | |||
b43ea5c7d3 | |||
f56759222d | |||
c8b7bb570c | |||
d536b29550 | |||
ecbda018c5 | |||
5187311ff0 | |||
d9a94ac7b0 | |||
3526e322ef | |||
99cabd545d | |||
1f7f7b10de | |||
b2421b592a | |||
5322f4c62f | |||
7ab8647fc8 | |||
50c76f3d7e | |||
2f4c56176d |
43
changelog.md
43
changelog.md
@ -1,5 +1,46 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## v1.1.13
|
||||||
|
- Added icon prefix option for when using default controls
|
||||||
|
|
||||||
|
## v1.1.13
|
||||||
|
- Logic tweaks for hiding controls in fullscreen
|
||||||
|
|
||||||
|
## v1.1.12
|
||||||
|
- Bug fix for Chrome Canary
|
||||||
|
|
||||||
|
## v1.1.11
|
||||||
|
- Bug fix
|
||||||
|
|
||||||
|
## v1.1.10
|
||||||
|
- Bug fix
|
||||||
|
|
||||||
|
## v1.1.9
|
||||||
|
- Bug fix for 1.1.8
|
||||||
|
|
||||||
|
## v1.1.8
|
||||||
|
- setVolume API method improvements (Fixes #83)
|
||||||
|
|
||||||
|
## v1.1.7
|
||||||
|
- Restore classname on destroy()
|
||||||
|
|
||||||
|
## v1.1.6
|
||||||
|
- New API methods (fixes #77), Fix for non strict mode (fixes #78)
|
||||||
|
|
||||||
|
## v1.1.5
|
||||||
|
- Fix for incorrect `isFullscreen()` return value in Mozilla (Fixes #38)
|
||||||
|
|
||||||
|
## v1.1.4
|
||||||
|
- Minor bug fixes
|
||||||
|
|
||||||
|
## v1.1.3
|
||||||
|
- Fixes for random id used in controls with multiple instances and one call to setup
|
||||||
|
- Audio player UI improvements
|
||||||
|
|
||||||
|
## v1.1.2
|
||||||
|
- Added an onSetup callback option
|
||||||
|
- Added fullscreen API methods `toggleFullscreen()` (must be user iniated), and `isFullscreen()`
|
||||||
|
|
||||||
## v1.1.1
|
## v1.1.1
|
||||||
- Fix for unsupported browser handling
|
- Fix for unsupported browser handling
|
||||||
- Fix for config.controls having no effect
|
- Fix for config.controls having no effect
|
||||||
@ -8,7 +49,7 @@
|
|||||||
- Added config option to set which controls are shown (if using the default controls html) and better handling of missing controls
|
- Added config option to set which controls are shown (if using the default controls html) and better handling of missing controls
|
||||||
|
|
||||||
## v1.0.31
|
## v1.0.31
|
||||||
- Display duration on metadataloaded
|
- Display duration on `metadataloaded`
|
||||||
|
|
||||||
## v1.0.30
|
## v1.0.30
|
||||||
- Fixed bug with media longer than 60 minutes (Fixes #69)
|
- Fixed bug with media longer than 60 minutes (Fixes #69)
|
||||||
|
2
dist/plyr.css
vendored
2
dist/plyr.css
vendored
File diff suppressed because one or more lines are too long
2
dist/plyr.js
vendored
2
dist/plyr.js
vendored
File diff suppressed because one or more lines are too long
2
dist/sprite.svg
vendored
2
dist/sprite.svg
vendored
File diff suppressed because one or more lines are too long
Before Width: | Height: | Size: 5.3 KiB After Width: | Height: | Size: 5.3 KiB |
2
docs/dist/docs.css
vendored
2
docs/dist/docs.css
vendored
@ -1 +1 @@
|
|||||||
/*! normalize.css v2.1.3 | MIT License | git.io/normalize */article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block}audio,canvas,video{display:inline-block}audio:not([controls]){display:none;height:0}[hidden],template{display:none}html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}a{background:0 0}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}hr{box-sizing:content-box;height:0}mark{background:#ff0;color:#000}code,kbd,pre,samp{font-family:monospace,serif;font-size:1em}pre{white-space:pre-wrap}q{quotes:"\201C" "\201D" "\2018" "\2019"}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:0}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}button,input,select,textarea{font-family:inherit;font-size:100%;margin:0}button,input{line-height:normal}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=search]{-webkit-appearance:textfield;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top}table{border-collapse:collapse;border-spacing:0}@font-face{font-family:Avenir;src:url(//cdn.plyr.io/fonts/avenir-medium.woff2) format("woff2"),url(//cdn.plyr.io/fonts/avenir-medium.woff) format("woff"),url(//cdn.plyr.io/fonts/avenir-medium.ttf) format("truetype");font-style:normal;font-weight:400}@font-face{font-family:Avenir;src:url(//cdn.plyr.io/fonts/avenir-bold.woff2) format("woff2"),url(//cdn.plyr.io/fonts/avenir-bold.woff) format("woff"),url(//cdn.plyr.io/fonts/avenir-bold.ttf) format("truetype");font-style:normal;font-weight:600}*,::after,::before{box-sizing:border-box}body{font-family:Avenir,"Helvetica Neue",Helvetica,Arial,sans-serif;background:#fff;line-height:1.5;text-align:center;color:#6D797F}.error body,html.error{height:100%}.error body{width:100%;display:table;table-layout:fixed}.error main{display:table-cell;width:100%;vertical-align:middle}h1,h2{letter-spacing:-.025em;color:#2E3C44;margin:0 0 10px;line-height:1.2;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased}h1{font-size:64px;font-size:4rem;color:#3498DB}p,small{margin:0 0 20px}small{display:block;padding:0 10px;font-size:14px;font-size:.9rem}header{padding:20px;margin-bottom:20px}header p{font-size:18px;font-size:1.1rem}@media (min-width:560px){header{padding-top:60px;padding-bottom:60px}}section{padding-bottom:20px}@media (min-width:560px){section{padding-bottom:40px}}a{text-decoration:none;color:#3498db;border-bottom:1px solid currentColor;transition:all .3s ease}a:focus,a:hover{color:#000}a:focus{outline:#343f4a dotted thin;outline-offset:1px}a.logo{border:0}.btn{display:inline-block;padding:10px 30px;background:#3498db;border:0;color:#fff;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-weight:600;border-radius:3px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.btn:focus,.btn:hover{color:#fff;background:#258cd1}.example-audio .player{max-width:520px}.example-video .player{max-width:1200px}.example-audio .player,.example-video .player{margin:0 auto 20px}.example-audio .player-fullscreen,.example-audio .player.fullscreen-active,.example-video .player-fullscreen,.example-video .player.fullscreen-active{max-width:none}footer{margin-bottom:20px}footer p{margin-bottom:10px}
|
/*! normalize.css v2.1.3 | MIT License | git.io/normalize */article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block}audio,canvas,video{display:inline-block}audio:not([controls]){display:none;height:0}[hidden],template{display:none}html{font-family:sans-serif;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}a{background:0 0}a:active,a:hover{outline:0}abbr[title]{border-bottom:1px dotted}b,strong{font-weight:700}dfn{font-style:italic}hr{box-sizing:content-box;height:0}mark{background:#ff0;color:#000}code,kbd,pre,samp{font-family:monospace,serif;font-size:1em}pre{white-space:pre-wrap}q{quotes:"\201C" "\201D" "\2018" "\2019"}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sup{top:-.5em}sub{bottom:-.25em}img{border:0}svg:not(:root){overflow:hidden}figure{margin:0}fieldset{border:1px solid silver;margin:0 2px;padding:.35em .625em .75em}legend{border:0;padding:0}button,input,select,textarea{font-family:inherit;font-size:100%;margin:0}button,input{line-height:normal}button,select{text-transform:none}button,html input[type=button],input[type=reset],input[type=submit]{-webkit-appearance:button;cursor:pointer}button[disabled],html input[disabled]{cursor:default}input[type=checkbox],input[type=radio]{box-sizing:border-box;padding:0}input[type=search]{-webkit-appearance:textfield;box-sizing:content-box}input[type=search]::-webkit-search-cancel-button,input[type=search]::-webkit-search-decoration{-webkit-appearance:none}button::-moz-focus-inner,input::-moz-focus-inner{border:0;padding:0}textarea{overflow:auto;vertical-align:top}table{border-collapse:collapse;border-spacing:0}@font-face{font-family:Avenir;src:url(//cdn.plyr.io/fonts/avenir-medium.woff2) format("woff2"),url(//cdn.plyr.io/fonts/avenir-medium.woff) format("woff"),url(//cdn.plyr.io/fonts/avenir-medium.ttf) format("truetype");font-style:normal;font-weight:400}@font-face{font-family:Avenir;src:url(//cdn.plyr.io/fonts/avenir-bold.woff2) format("woff2"),url(//cdn.plyr.io/fonts/avenir-bold.woff) format("woff"),url(//cdn.plyr.io/fonts/avenir-bold.ttf) format("truetype");font-style:normal;font-weight:600}*,::after,::before{box-sizing:border-box}body{font-family:Avenir,"Helvetica Neue",Helvetica,Arial,sans-serif;background:#f2f5f7;line-height:1.5;text-align:center;color:#565d64}.error body,html.error{height:100%}.error body{width:100%;display:table;table-layout:fixed}.error main{display:table-cell;width:100%;vertical-align:middle}h1,h2{letter-spacing:-.025em;color:#2E3C44;margin:0 0 10px;line-height:1.2;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased}h1{font-size:64px;font-size:4rem;color:#3498DB}p,small{margin:0 0 20px}small{display:block;padding:0 10px;font-size:14px;font-size:.9rem}header{padding:20px;margin-bottom:20px}header p{font-size:18px;font-size:1.1rem}@media (min-width:560px){header{padding-top:60px;padding-bottom:60px}}section{padding-bottom:20px}@media (min-width:560px){section{padding-bottom:40px}}.actions{list-style:none;margin:0;padding:0;font-size:0}.actions li{display:inline-block;margin-top:10px;font-size:16px;font-size:1rem;white-space:nowrap}.actions li+li{margin-left:20px}a{text-decoration:none;color:#3498db;border-bottom:1px solid currentColor;transition:all .3s ease}a:focus,a:hover{color:#000}a:focus{outline:#343f4a dotted thin;outline-offset:1px}a.logo{border:0}.btn,.btn-count{display:inline-block;vertical-align:middle;border-radius:3px;-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-weight:600;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none}.btn{padding:10px 20px;background:#3498db;border:0;color:#fff}.btn:focus,.btn:hover{color:#fff;background:#258cd1}.btn-twitter{background:#8799A2}.btn-twitter:focus,.btn-twitter:hover{background:#798d97}.btn-count{position:relative;margin-left:6px;padding:9px;background:#fff;border:1px solid #cbd0d3}.btn-count::before{content:"";position:absolute;display:block;width:8px;height:8px;left:1px;top:50%;margin-top:-4px;background:inherit;border:inherit;border-width:1px 0 0 1px;-webkit-transform:rotate(-45deg) translate(-50%,-50%);transform:rotate(-45deg) translate(-50%,-50%)}.example-audio .player,.example-video .player{margin:0 auto 20px}.example-audio .player-controls,.example-video .player-controls{border-radius:0 0 4px 4px}.example-audio .player{max-width:520px}.example-audio .player-controls{border-radius:4px}.example-audio .player-progress{border-radius:4px 4px 0 0;overflow:hidden}.example-video .player{max-width:1200px}.example-video .player video{border-radius:4px 4px 0 0}.example-video .player-fullscreen,.example-video .player.fullscreen-active{max-width:none}footer{margin-bottom:20px}footer p{margin-bottom:10px}
|
2
docs/dist/docs.js
vendored
2
docs/dist/docs.js
vendored
File diff suppressed because one or more lines are too long
@ -6,7 +6,7 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
|
||||||
<!-- Docs styles -->
|
<!-- Docs styles -->
|
||||||
<link rel="stylesheet" href="//cdn.plyr.io/1.1.1/docs.css">
|
<link rel="stylesheet" href="//cdn.plyr.io/1.1.14/docs.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main>
|
<main>
|
||||||
|
@ -8,33 +8,40 @@
|
|||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
<meta name="viewport" content="width=device-width, initial-scale=1">
|
||||||
|
|
||||||
<!-- Styles -->
|
<!-- Styles -->
|
||||||
<link rel="stylesheet" href="//cdn.plyr.io/1.1.1/plyr.css">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.1.14/plyr.css?1">
|
||||||
|
|
||||||
<!-- Docs styles -->
|
<!-- Docs styles -->
|
||||||
<link rel="stylesheet" href="//cdn.plyr.io/1.1.1/docs.css">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.1.14/docs.css?2">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<header>
|
<header>
|
||||||
<h1>Plyr</h1>
|
<h1>Plyr</h1>
|
||||||
<p>A simple HTML5 media player with custom controls and WebVTT captions.</p>
|
<p>A simple HTML5 media player with custom controls and WebVTT captions by <a href="https://twitter.com/sam_potts" target="_blank">@sam_potts</a></p>
|
||||||
<a href="https://github.com/selz/plyr" target="_blank" class="btn">Download on GitHub</a>
|
<ul class="actions">
|
||||||
|
<li>
|
||||||
|
<a href="https://github.com/selz/plyr" target="_blank" class="btn btn-download">Download on GitHub</a>
|
||||||
|
<span class="btn-count js-stargazers-count">…</span>
|
||||||
|
</li>
|
||||||
|
<li>
|
||||||
|
<a href="https://twitter.com/intent/tweet?text=A+simple+HTML5+media+player+with+custom+controls+and+WebVTT+captions.&url=http%3A%2F%2Fplyr.io&via=Sam_Potts" target="_blank" class="btn btn-twitter js-popup" data-window-height="250" data-window-width="500">Tweet</a>
|
||||||
|
<span class="btn-count js-tweet-count">…</span>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main>
|
<main>
|
||||||
<section class="example-video">
|
<section class="example-video">
|
||||||
<div class="player">
|
<div class="player">
|
||||||
<video poster="//cdn.selz.com/plyr/1.0/poster.jpg" controls crossorigin>
|
<video poster="https://cdn.plyr.io/static/poster.jpg" controls crossorigin>
|
||||||
<!-- Video files -->
|
<!-- Video files -->
|
||||||
<source src="//cdn.selz.com/plyr/1.0/movie.mp4" type="video/mp4">
|
<source src="https://cdn.selz.com/plyr/1.0/movie.mp4" type="video/mp4">
|
||||||
<source src="//cdn.selz.com/plyr/1.0/movie.webm" type="video/webm">
|
<source src="https://cdn.selz.com/plyr/1.0/movie.webm" type="video/webm">
|
||||||
|
|
||||||
<!-- Text track file -->
|
<!-- Text track file -->
|
||||||
<track kind="captions" label="English" srclang="en" src="//cdn.selz.com/plyr/1.0/en.vtt" default>
|
<track kind="captions" label="English" srclang="en" src="https://cdn.selz.com/plyr/1.0/en.vtt" default>
|
||||||
|
|
||||||
<!-- Fallback for browsers that don't support the <video> element -->
|
<!-- Fallback for browsers that don't support the <video> element -->
|
||||||
<div>
|
<a href="https://cdn.selz.com/plyr/1.0/movie.mp4">Download</a>
|
||||||
<a href="//cdn.selz.com/plyr/1.0/movie.mp4">Download</a>
|
|
||||||
</div>
|
|
||||||
</video>
|
</video>
|
||||||
</div>
|
</div>
|
||||||
<small>Big Buck Bunny. More info can be found at <a href="https://peach.blender.org" target="_blank">peach.blender.org</a>.</small>
|
<small>Big Buck Bunny. More info can be found at <a href="https://peach.blender.org" target="_blank">peach.blender.org</a>.</small>
|
||||||
@ -44,13 +51,11 @@
|
|||||||
<div class="player">
|
<div class="player">
|
||||||
<audio controls>
|
<audio controls>
|
||||||
<!-- Audio files -->
|
<!-- Audio files -->
|
||||||
<source src="//cdn.selz.com/plyr/1.0/logistics-96-sample.mp3" type="audio/mp3">
|
<source src="https://cdn.selz.com/plyr/1.0/logistics-96-sample.mp3" type="audio/mp3">
|
||||||
<source src="//cdn.selz.com/plyr/1.0/logistics-96-sample.ogg" type="audio/ogg">
|
<source src="https://cdn.selz.com/plyr/1.0/logistics-96-sample.ogg" type="audio/ogg">
|
||||||
|
|
||||||
<!-- Fallback for browsers that don't support the <audio> element -->
|
<!-- Fallback for browsers that don't support the <audio> element -->
|
||||||
<div>
|
<a href="https://cdn.selz.com/plyr/1.0/logistics-96-sample.mp3">Download</a>
|
||||||
<a href="//cdn.selz.com/plyr/1.0/logistics-96-sample.mp3">Download</a>
|
|
||||||
</div>
|
|
||||||
</audio>
|
</audio>
|
||||||
</div>
|
</div>
|
||||||
<small>"96" by Logistics, which can be purchased from <a href="https://www.hospitalrecords.com/shop/artist/logistics" target="_blank">Hospital Records</a>.</small>
|
<small>"96" by Logistics, which can be purchased from <a href="https://www.hospitalrecords.com/shop/artist/logistics" target="_blank">Hospital Records</a>.</small>
|
||||||
@ -83,13 +88,13 @@
|
|||||||
b.insertBefore(c, b.childNodes[0]);
|
b.insertBefore(c, b.childNodes[0]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})(document, "https://cdn.plyr.io/1.1.1/sprite.svg");
|
})(document, "https://cdn.plyr.io/1.1.14/sprite.svg");
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<!-- Plyr core script -->
|
<!-- Plyr core script -->
|
||||||
<script src="//cdn.plyr.io/1.1.1/plyr.js"></script>
|
<script src="https://cdn.plyr.io/1.1.14/plyr.js?1"></script>
|
||||||
|
|
||||||
<!-- Docs script -->
|
<!-- Docs script -->
|
||||||
<script src="//cdn.plyr.io/1.1.1/docs.js"></script>
|
<script src="https://cdn.plyr.io/1.1.14/docs.js?1"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
@ -7,13 +7,136 @@
|
|||||||
// Setup the player
|
// Setup the player
|
||||||
plyr.setup({
|
plyr.setup({
|
||||||
debug: true,
|
debug: true,
|
||||||
|
volume: 9,
|
||||||
title: "Video demo",
|
title: "Video demo",
|
||||||
html: templates.controls.render({}),
|
html: templates.controls.render({}),
|
||||||
captions: {
|
captions: {
|
||||||
defaultActive: true
|
defaultActive: true
|
||||||
|
},
|
||||||
|
onSetup: function() {
|
||||||
|
var player = this,
|
||||||
|
type = player.media.tagName.toLowerCase(),
|
||||||
|
toggle = document.querySelector("[data-toggle='fullscreen']");
|
||||||
|
|
||||||
|
console.log("✓ Setup done for <" + type + ">");
|
||||||
|
|
||||||
|
if(type === "video" && toggle) {
|
||||||
|
toggle.addEventListener("click", player.toggleFullscreen, false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// Popup
|
||||||
|
function popup(event) {
|
||||||
|
// Prevent the link opening
|
||||||
|
if(event.target.nodeName.toLowerCase() == "a") {
|
||||||
|
if(event.preventDefault) {
|
||||||
|
event.preventDefault();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
event.returnValue = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
var link = event.target,
|
||||||
|
url = link.href,
|
||||||
|
width = link.getAttribute("data-window-width") || 600,
|
||||||
|
height = link.getAttribute("data-window-height") || 600,
|
||||||
|
name = link.getAttribute("data-window-name") || "popup";
|
||||||
|
|
||||||
|
// If window exists, just focus it
|
||||||
|
if(window["window-"+name] && !window["window-"+name].closed) {
|
||||||
|
window["window-"+name].focus();
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Get position
|
||||||
|
var left = window.screenLeft !== undefined ? window.screenLeft : screen.left;
|
||||||
|
var top = window.screenTop !== undefined ? window.screenTop : screen.top;
|
||||||
|
|
||||||
|
// Open in the centre of the screen
|
||||||
|
var x = (screen.width / 2) - (width / 2) + left,
|
||||||
|
y = (screen.height / 2) - (height / 2) + top;
|
||||||
|
|
||||||
|
// Open that window
|
||||||
|
window["window-"+name] = window.open(url, name, "top=" + y +",left="+ x +",width=" + width + ",height=" + height);
|
||||||
|
|
||||||
|
// Focus new window
|
||||||
|
window["window-"+name].focus();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trigger popups
|
||||||
|
document.querySelector(".js-popup").addEventListener("click", popup);
|
||||||
|
|
||||||
|
// Get JSONP
|
||||||
|
function getJSONP(url, callback) {
|
||||||
|
var name = "jsonp_callback_" + Math.round(100000 * Math.random());
|
||||||
|
|
||||||
|
// Cleanup to prevent memory leaks and hit original callback
|
||||||
|
window[name] = function(data) {
|
||||||
|
delete window[name];
|
||||||
|
document.body.removeChild(script);
|
||||||
|
callback(data);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Create a faux script
|
||||||
|
var script = document.createElement("script");
|
||||||
|
script.setAttribute("src", url + (url.indexOf("?") >= 0 ? "&" : "?") + "callback=" + name);
|
||||||
|
|
||||||
|
// Inject to the body
|
||||||
|
document.body.appendChild(script);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get star count
|
||||||
|
var storageSupported = ("sessionStorage" in window),
|
||||||
|
selectors = {
|
||||||
|
github: ".js-stargazers-count",
|
||||||
|
twitter: ".js-tweet-count"
|
||||||
|
};
|
||||||
|
|
||||||
|
// Display the count next to the button
|
||||||
|
function displayCount(selector, count) {
|
||||||
|
document.querySelector(selector).innerHTML = count;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add star
|
||||||
|
function formatGitHubCount(count) {
|
||||||
|
return "★ " + count;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check if it's in session storage first
|
||||||
|
if(storageSupported && "github_stargazers" in window.sessionStorage) {
|
||||||
|
displayCount(selectors.github, formatGitHubCount(window.sessionStorage.github_stargazers));
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
getJSONP("https://api.github.com/repos/selz/plyr?access_token=a46ac653210ba6a6be44260c29c333470c3fbbf5", function (json) {
|
||||||
|
if (json && typeof json.data.stargazers_count !== "undefined") {
|
||||||
|
// Update UI
|
||||||
|
displayCount(selectors.github, formatGitHubCount(json.data.stargazers_count));
|
||||||
|
|
||||||
|
// Store in session storage
|
||||||
|
window.sessionStorage.github_stargazers = json.data.stargazers_count;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get tweet count
|
||||||
|
if(storageSupported && "tweets" in window.sessionStorage) {
|
||||||
|
displayCount(selectors.twitter, window.sessionStorage.tweets);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
getJSONP("https://cdn.api.twitter.com/1/urls/count.json?url=plyr.io", function (json) {
|
||||||
|
if (json && typeof json.count !== "undefined") {
|
||||||
|
// Update UI
|
||||||
|
displayCount(selectors.twitter, json.count);
|
||||||
|
|
||||||
|
// Store in session storage
|
||||||
|
window.sessionStorage.tweets = json.count;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
// Google analytics
|
// Google analytics
|
||||||
// For demo site (http://[www.]plyr.io) only
|
// For demo site (http://[www.]plyr.io) only
|
||||||
if(document.domain.indexOf("plyr.io") > -1) {
|
if(document.domain.indexOf("plyr.io") > -1) {
|
||||||
|
@ -12,14 +12,16 @@
|
|||||||
// Variables
|
// Variables
|
||||||
// ---------------------------------------
|
// ---------------------------------------
|
||||||
// Colors
|
// Colors
|
||||||
@blue: #3498DB;
|
@blue: #3498db;
|
||||||
@gray-dark: #343f4a;
|
@gray-dark: #343f4a;
|
||||||
@gray: #565d64;
|
@gray: #565d64;
|
||||||
@gray-light: #cbd0d3;
|
@gray-light: #cbd0d3;
|
||||||
|
@off-white: #f2f5f7;
|
||||||
|
|
||||||
// Elements
|
// Elements
|
||||||
@link-color: @blue;
|
@link-color: @blue;
|
||||||
@padding-base: 20px;
|
@padding-base: 20px;
|
||||||
|
@arrow-size: 8px;
|
||||||
|
|
||||||
// Breakpoints
|
// Breakpoints
|
||||||
@screen-md: 768px;
|
@screen-md: 768px;
|
||||||
@ -33,10 +35,10 @@
|
|||||||
// Base
|
// Base
|
||||||
body {
|
body {
|
||||||
font-family: "Avenir", "Helvetica Neue", Helvetica, Arial, sans-serif;
|
font-family: "Avenir", "Helvetica Neue", Helvetica, Arial, sans-serif;
|
||||||
background: #fff;
|
background: @off-white;
|
||||||
line-height: 1.5;
|
line-height: 1.5;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
color: #6D797F;
|
color: @gray;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Error page
|
// Error page
|
||||||
@ -102,6 +104,22 @@ section {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Links & Buttons
|
// Links & Buttons
|
||||||
|
.actions {
|
||||||
|
list-style: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 0;
|
||||||
|
font-size: 0;
|
||||||
|
|
||||||
|
li {
|
||||||
|
display: inline-block;
|
||||||
|
margin-top: (@padding-base / 2);
|
||||||
|
.font-size();
|
||||||
|
white-space: nowrap;
|
||||||
|
}
|
||||||
|
li + li {
|
||||||
|
margin-left: @padding-base;
|
||||||
|
}
|
||||||
|
}
|
||||||
a {
|
a {
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
color: @link-color;
|
color: @link-color;
|
||||||
@ -119,41 +137,93 @@ a {
|
|||||||
border: 0;
|
border: 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.btn {
|
.btn,
|
||||||
|
.btn-count {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
padding: (@padding-base / 2) (@padding-base * 1.5);
|
vertical-align: middle;
|
||||||
|
border-radius: 3px;
|
||||||
|
.font-smoothing(on);
|
||||||
|
font-weight: 600;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
.btn {
|
||||||
|
padding: (@padding-base / 2) @padding-base;
|
||||||
background: @link-color;
|
background: @link-color;
|
||||||
border: 0;
|
border: 0;
|
||||||
color: #fff;
|
color: #fff;
|
||||||
.font-smoothing(on);
|
|
||||||
font-weight: 600;
|
|
||||||
border-radius: 3px;
|
|
||||||
user-select: none;
|
|
||||||
|
|
||||||
&:hover,
|
&:hover,
|
||||||
&:focus {
|
&:focus {
|
||||||
color: #fff;
|
color: #fff;
|
||||||
background: darken(@link-color, 5%);
|
background: darken(@link-color, 5%);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
&-twitter {
|
||||||
|
background: #8799A2;
|
||||||
|
|
||||||
|
&:hover,
|
||||||
|
&:focus {
|
||||||
|
background: darken(#8799A2, 5%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.btn-count {
|
||||||
|
position: relative;
|
||||||
|
margin-left: 6px;
|
||||||
|
padding: ((@padding-base / 2) - 1px);
|
||||||
|
background: #fff;
|
||||||
|
border: 1px solid @gray-light;
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
display: block;
|
||||||
|
width: @arrow-size;
|
||||||
|
height: @arrow-size;
|
||||||
|
left: 1px;
|
||||||
|
top: 50%;
|
||||||
|
margin-top: -(@arrow-size / 2);
|
||||||
|
|
||||||
|
background: inherit;
|
||||||
|
border: inherit;
|
||||||
|
border-width: 1px 0 0 1px;
|
||||||
|
transform: rotate(-45deg) translate(-50%, -50%);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Players
|
// Example players
|
||||||
.example-audio .player {
|
|
||||||
max-width: 520px;
|
|
||||||
}
|
|
||||||
.example-video .player {
|
|
||||||
max-width: 1200px;
|
|
||||||
}
|
|
||||||
.example-audio .player,
|
.example-audio .player,
|
||||||
.example-video .player {
|
.example-video .player {
|
||||||
margin: 0 auto @padding-base;
|
margin: 0 auto @padding-base;
|
||||||
|
|
||||||
|
&-controls {
|
||||||
|
border-radius: 0 0 4px 4px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.example-audio .player {
|
||||||
|
max-width: 520px;
|
||||||
|
|
||||||
|
&-controls {
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
&-progress {
|
||||||
|
border-radius: 4px 4px 0 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.example-video .player {
|
||||||
|
max-width: 1200px;
|
||||||
|
|
||||||
|
video {
|
||||||
|
border-radius: 4px 4px 0 0;
|
||||||
|
}
|
||||||
&-fullscreen,
|
&-fullscreen,
|
||||||
&.fullscreen-active {
|
&.fullscreen-active {
|
||||||
max-width: none;
|
max-width: none;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// Footer
|
// Footer
|
||||||
footer {
|
footer {
|
||||||
margin-bottom: @padding-base;
|
margin-bottom: @padding-base;
|
||||||
|
@ -185,12 +185,12 @@ gulp.task("watch", function () {
|
|||||||
// Plyr core
|
// Plyr core
|
||||||
gulp.watch(paths.plyr.src.js, tasks.js);
|
gulp.watch(paths.plyr.src.js, tasks.js);
|
||||||
gulp.watch(paths.plyr.src.less, tasks.less);
|
gulp.watch(paths.plyr.src.less, tasks.less);
|
||||||
gulp.watch(paths.plyr.src.sprite, "sprite");
|
gulp.watch(paths.plyr.src.sprite, ["sprite"]);
|
||||||
|
|
||||||
// Docs
|
// Docs
|
||||||
gulp.watch(paths.docs.src.js, tasks.js);
|
gulp.watch(paths.docs.src.js, tasks.js);
|
||||||
gulp.watch(paths.docs.src.less, tasks.less);
|
gulp.watch(paths.docs.src.less, tasks.less);
|
||||||
gulp.watch(paths.docs.src.templates, "js");
|
gulp.watch(paths.docs.src.templates, ["js"]);
|
||||||
});
|
});
|
||||||
|
|
||||||
// Publish a version to CDN and docs
|
// Publish a version to CDN and docs
|
||||||
@ -272,5 +272,5 @@ gulp.task("open", function () {
|
|||||||
|
|
||||||
// Do everything
|
// Do everything
|
||||||
gulp.task("publish", function () {
|
gulp.task("publish", function () {
|
||||||
run("templates", tasks.js, tasks.less, "sprite", "cdn", "docs", "open");
|
run("templates", tasks.js, tasks.less, "sprite", "cdn", "docs");
|
||||||
});
|
});
|
14
package.json
14
package.json
@ -1,18 +1,18 @@
|
|||||||
{
|
{
|
||||||
"name": "plyr",
|
"name": "plyr",
|
||||||
"version": "1.1.1",
|
"version": "1.1.14",
|
||||||
"description": "A simple HTML5 media player using custom controls",
|
"description": "A simple HTML5 media player using custom controls",
|
||||||
"homepage": "http://plyr.io",
|
"homepage": "http://plyr.io",
|
||||||
"main": "gulpfile.js",
|
"main": "gulpfile.js",
|
||||||
"dependencies": {},
|
"dependencies": {},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"gulp": "~3.8.6",
|
"gulp": "^3.8.6",
|
||||||
"gulp-autoprefixer": "^0.0.8",
|
"gulp-autoprefixer": "^0.0.8",
|
||||||
"gulp-concat": "~2.3.3",
|
"gulp-concat": "^2.3.3",
|
||||||
"gulp-gzip": "^1.0.0",
|
"gulp-gzip": "^1.0.0",
|
||||||
"gulp-hogan-compile": "^0.4.1",
|
"gulp-hogan-compile": "^0.4.1",
|
||||||
"gulp-less": "~1.3.1",
|
"gulp-less": "^1.3.1",
|
||||||
"gulp-minify-css": "~0.3.6",
|
"gulp-minify-css": "^0.3.6",
|
||||||
"gulp-open": "^0.3.2",
|
"gulp-open": "^0.3.2",
|
||||||
"gulp-rename": "^1.2.0",
|
"gulp-rename": "^1.2.0",
|
||||||
"gulp-replace": "^0.5.3",
|
"gulp-replace": "^0.5.3",
|
||||||
@ -21,8 +21,8 @@
|
|||||||
"gulp-size": "^1.2.1",
|
"gulp-size": "^1.2.1",
|
||||||
"gulp-svgmin": "^1.0.0",
|
"gulp-svgmin": "^1.0.0",
|
||||||
"gulp-svgstore": "^5.0.0",
|
"gulp-svgstore": "^5.0.0",
|
||||||
"gulp-uglify": "~0.3.1",
|
"gulp-uglify": "^0.3.1",
|
||||||
"gulp-util": "~2.2.20",
|
"gulp-util": "^2.2.20",
|
||||||
"run-sequence": "^0.3.6"
|
"run-sequence": "^0.3.6"
|
||||||
},
|
},
|
||||||
"scripts": {
|
"scripts": {
|
||||||
|
103
readme.md
103
readme.md
@ -3,14 +3,14 @@ A simple, accessible HTML5 media player.
|
|||||||
|
|
||||||
[Checkout the demo](http://plyr.io)
|
[Checkout the demo](http://plyr.io)
|
||||||
|
|
||||||
[](http://plyr.io)
|
[](http://plyr.io)
|
||||||
|
|
||||||
## Why?
|
## Why?
|
||||||
We wanted a lightweight, accessible and customisable media player that just supports *modern* browsers. Sure, there are many other players out there but we wanted to keep things simple, using the right elements for the job.
|
We wanted a lightweight, accessible and customisable media player that just supports *modern* browsers. Sure, there are many other players out there but we wanted to keep things simple, using the right elements for the job.
|
||||||
|
|
||||||
## Features
|
## Features
|
||||||
- **Accessible** - full support for captions and screen readers.
|
- **Accessible** - full support for captions and screen readers.
|
||||||
- **Lightweight** - just 6KB minified and gzipped.
|
- **Lightweight** - just 6.4KB minified and gzipped.
|
||||||
- **Customisable** - make the player look how you want with the markup you want.
|
- **Customisable** - make the player look how you want with the markup you want.
|
||||||
- **Semantic** - uses the *right* elements. `<input type="range">` for volume and `<progress>` for progress and well, `<button>`s for buttons. There's no `<span>` or `<a href="#">` button hacks.
|
- **Semantic** - uses the *right* elements. `<input type="range">` for volume and `<progress>` for progress and well, `<button>`s for buttons. There's no `<span>` or `<a href="#">` button hacks.
|
||||||
- **Responsive** - as you'd expect these days.
|
- **Responsive** - as you'd expect these days.
|
||||||
@ -25,11 +25,10 @@ Oh and yes, it works with Bootstrap.
|
|||||||
Check out [the changelog](changelog.md)
|
Check out [the changelog](changelog.md)
|
||||||
|
|
||||||
## Planned development
|
## Planned development
|
||||||
- Accept a string selector, a node, or a nodelist for the `container` property of `selectors`.
|
|
||||||
- Multiple language captions (with selection)
|
|
||||||
- Playlists (audio and video)
|
- Playlists (audio and video)
|
||||||
- Set source by API
|
- YouTube and Vimeo support
|
||||||
- Tooltip option (for seeking and controls)
|
- Playback speed
|
||||||
|
- Multiple language captions (with selection)
|
||||||
... and whatever else has been raised in [issues](https://github.com/Selz/plyr/issues)
|
... and whatever else has been raised in [issues](https://github.com/Selz/plyr/issues)
|
||||||
|
|
||||||
If you have any cool ideas or features, please let me know by [creating an issue](https://github.com/Selz/plyr/issues/new) or of course, forking and sending a pull request.
|
If you have any cool ideas or features, please let me know by [creating an issue](https://github.com/Selz/plyr/issues/new) or of course, forking and sending a pull request.
|
||||||
@ -38,7 +37,7 @@ If you have any cool ideas or features, please let me know by [creating an issue
|
|||||||
|
|
||||||
Check `docs/index.html` and `docs/dist/docs.js` for an example setup.
|
Check `docs/index.html` and `docs/dist/docs.js` for an example setup.
|
||||||
|
|
||||||
**Heads up**, the example `index.html` file needs to be served from a webserver (such as Apache, Nginx, IIS or similar) unless you change the file sources to include http or https. e.g. change `//cdn.plyr.io/1.1.1/plyr.js` to `https://cdn.plyr.io/1.1.1/plyr.js`
|
**Heads up**, the example `index.html` file needs to be served from a webserver (such as Apache, Nginx, IIS or similar) unless you change the file sources to include http or https. e.g. change `//cdn.plyr.io/1.1.14/plyr.js` to `https://cdn.plyr.io/1.1.14/plyr.js`
|
||||||
|
|
||||||
### Bower
|
### Bower
|
||||||
If bower is your thang, you can grab Plyr using:
|
If bower is your thang, you can grab Plyr using:
|
||||||
@ -47,15 +46,22 @@ bower install plyr
|
|||||||
```
|
```
|
||||||
More info on setting up dependencies can be found in the [Bower Docs](http://bower.io/docs/creating-packages/#maintaining-dependencies)
|
More info on setting up dependencies can be found in the [Bower Docs](http://bower.io/docs/creating-packages/#maintaining-dependencies)
|
||||||
|
|
||||||
|
### Ember
|
||||||
|
The awesome [@louisrudner](https://twitter.com/louisrudner) has created an ember component, available by running:
|
||||||
|
```
|
||||||
|
ember addon:install ember-cli-plyr
|
||||||
|
```
|
||||||
|
More info is on [npm](https://www.npmjs.com/package/ember-cli-plyr) and [GitHub](https://github.com/louisrudner/ember-cli-plyr)
|
||||||
|
|
||||||
### CDN
|
### CDN
|
||||||
If you want to use our CDN, you can use the following. HTTPS (SSL) is supported.
|
If you want to use our CDN, you can use the following:
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<link rel="stylesheet" href="//cdn.plyr.io/1.1.1/plyr.css">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.1.14/plyr.css">
|
||||||
<script src="//cdn.plyr.io/1.1.1/plyr.js"></script>
|
<script src="https://cdn.plyr.io/1.1.14/plyr.js"></script>
|
||||||
```
|
```
|
||||||
|
|
||||||
You can also access the `sprite.svg` file at `//cdn.plyr.io/1.1.1/sprite.svg`.
|
You can also access the `sprite.svg` file at `https://cdn.plyr.io/1.1.14/sprite.svg`.
|
||||||
|
|
||||||
### CSS
|
### CSS
|
||||||
If you want to use the default css, add the `plyr.css` file from /dist into your head, or even better use `plyr.less` or `plyr.sass` file included in `/src` in your build to save a request.
|
If you want to use the default css, add the `plyr.css` file from /dist into your head, or even better use `plyr.less` or `plyr.sass` file included in `/src` in your build to save a request.
|
||||||
@ -92,18 +98,16 @@ and the AJAX technique here:
|
|||||||
The only extra markup that's needed to use plyr is a `<div>` wrapper. Replace the source, poster and captions with urls for your media.
|
The only extra markup that's needed to use plyr is a `<div>` wrapper. Replace the source, poster and captions with urls for your media.
|
||||||
```html
|
```html
|
||||||
<div class="player">
|
<div class="player">
|
||||||
<video poster="//cdn.selz.com/plyr/1.0/poster.jpg" controls crossorigin>
|
<video poster="https://cdn.selz.com/plyr/1.0/poster.jpg" controls crossorigin>
|
||||||
<!-- Video files -->
|
<!-- Video files -->
|
||||||
<source src="//cdn.selz.com/plyr/1.0/movie.mp4" type="video/mp4">
|
<source src="https://cdn.selz.com/plyr/1.0/movie.mp4" type="video/mp4">
|
||||||
<source src="//cdn.selz.com/plyr/1.0/movie.webm" type="video/webm">
|
<source src="https://cdn.selz.com/plyr/1.0/movie.webm" type="video/webm">
|
||||||
|
|
||||||
<!-- Text track file -->
|
<!-- Text track file -->
|
||||||
<track kind="captions" label="English captions" src="//cdn.selz.com/plyr/1.0/movie_captions_en.vtt" srclang="en" default>
|
<track kind="captions" label="English captions" src="https://cdn.selz.com/plyr/1.0/movie_captions_en.vtt" srclang="en" default>
|
||||||
|
|
||||||
<!-- Fallback for browsers that don't support the <video> element -->
|
<!-- Fallback for browsers that don't support the <video> element -->
|
||||||
<div>
|
<a href="https://cdn.selz.com/plyr/1.0/movie.mp4">Download</a>
|
||||||
<a href="//cdn.selz.com/plyr/1.0/movie.mp4">Download</a>
|
|
||||||
</div>
|
|
||||||
</video>
|
</video>
|
||||||
</div>
|
</div>
|
||||||
```
|
```
|
||||||
@ -113,13 +117,11 @@ And the same for `<audio>`
|
|||||||
<div class="player">
|
<div class="player">
|
||||||
<audio controls>
|
<audio controls>
|
||||||
<!-- Audio files -->
|
<!-- Audio files -->
|
||||||
<source src="//cdn.selz.com/plyr/1.0/logistics-96-sample.mp3" type="audio/mp3">
|
<source src="https://cdn.selz.com/plyr/1.0/logistics-96-sample.mp3" type="audio/mp3">
|
||||||
<source src="//cdn.selz.com/plyr/1.0/logistics-96-sample.ogg" type="audio/ogg">
|
<source src="https://cdn.selz.com/plyr/1.0/logistics-96-sample.ogg" type="audio/ogg">
|
||||||
|
|
||||||
<!-- Fallback for browsers that don't support the <audio> element -->
|
<!-- Fallback for browsers that don't support the <audio> element -->
|
||||||
<div>
|
<a href="https://cdn.selz.com/plyr/1.0/logistics-96-sample.mp3">Download</a>
|
||||||
<a href="//cdn.selz.com/plyr/1.0/logistics-96-sample.mp3">Download</a>
|
|
||||||
</div>
|
|
||||||
</audio>
|
</audio>
|
||||||
</div>
|
</div>
|
||||||
```
|
```
|
||||||
@ -131,20 +133,16 @@ More info on CORS here:
|
|||||||
[https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS)
|
[https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS](https://developer.mozilla.org/en-US/docs/Web/HTTP/Access_control_CORS)
|
||||||
|
|
||||||
### JavaScript
|
### JavaScript
|
||||||
Much of the behaviour of the player is configurable when initialising the library. Below is an example of a default instance.
|
Much of the behaviour of the player is configurable when initialising the library. Here's an example of a default setup:
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<script src="dist/plyr.js"></script>
|
<script src="dist/plyr.js"></script>
|
||||||
<script>
|
<script>plyr.setup();</script>
|
||||||
plyr.setup({
|
|
||||||
*options*
|
|
||||||
});
|
|
||||||
</script>
|
|
||||||
```
|
```
|
||||||
|
|
||||||
#### Options
|
#### Options
|
||||||
|
|
||||||
You can pass the following options to the setup method.
|
You can pass the following options to the setup method using `plyr.setup({...})`.
|
||||||
|
|
||||||
<table class="table" width="100%">
|
<table class="table" width="100%">
|
||||||
<thead>
|
<thead>
|
||||||
@ -174,6 +172,12 @@ You can pass the following options to the setup method.
|
|||||||
<td><code>["restart", "rewind", "play", "fast-forward", "current-time", "duration", "mute", "volume", "captions", "fullscreen"]</code></td>
|
<td><code>["restart", "rewind", "play", "fast-forward", "current-time", "duration", "mute", "volume", "captions", "fullscreen"]</code></td>
|
||||||
<td>Toggle which control elements you would like to display when using the default controls html. If you specify a <code>html</code> option, this is redundant. The default value is to display everything.</td>
|
<td>Toggle which control elements you would like to display when using the default controls html. If you specify a <code>html</code> option, this is redundant. The default value is to display everything.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>iconPrefix</code></td>
|
||||||
|
<td>String</td>
|
||||||
|
<td><code>icon</code></td>
|
||||||
|
<td>Specify the id prefix for the icons used in the default controls (e.g. "icon-play" would be "icon"). This is to prevent clashes if you're using your own SVG defs file but with the default controls. Most people can ignore this option.</td>
|
||||||
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>debug</code></td>
|
<td><code>debug</code></td>
|
||||||
<td>Boolean</td>
|
<td>Boolean</td>
|
||||||
@ -240,6 +244,12 @@ You can pass the following options to the setup method.
|
|||||||
<td>—</td>
|
<td>—</td>
|
||||||
<td>Two properties; <code>enabled</code> which toggles if local storage should be enabled (if the browser supports it). The default value is `true`. This enables storing user settings, currently it only stores volume but more will be added later. The second property <code>key</code> is the key used for the local storage. The default is <code>plyr_volume</code> until more settings are stored.</td>
|
<td>Two properties; <code>enabled</code> which toggles if local storage should be enabled (if the browser supports it). The default value is `true`. This enables storing user settings, currently it only stores volume but more will be added later. The second property <code>key</code> is the key used for the local storage. The default is <code>plyr_volume</code> until more settings are stored.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>onSetup</code></td>
|
||||||
|
<td>Function</td>
|
||||||
|
<td>—</td>
|
||||||
|
<td>This callback function is called on every new plyr instance created. The context (<code>this</code>) is the plyr instance itself.</td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@ -297,6 +307,11 @@ Here's a list of the methods supported:
|
|||||||
<td>Number</td>
|
<td>Number</td>
|
||||||
<td>Sets the player volume to the provided parameter. The value should be between 0 (muted) and 10 (loudest). If no parameter is provided, the default volume is used (5). Values over 10 are ignored.</td>
|
<td>Sets the player volume to the provided parameter. The value should be between 0 (muted) and 10 (loudest). If no parameter is provided, the default volume is used (5). Values over 10 are ignored.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>togglePlay()</code></td>
|
||||||
|
<td>Boolean</td>
|
||||||
|
<td>Toggles playback for the player based on either the boolean argument or it's current state.</td>
|
||||||
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>toggleMute()</code></td>
|
<td><code>toggleMute()</code></td>
|
||||||
<td>—</td>
|
<td>—</td>
|
||||||
@ -307,6 +322,16 @@ Here's a list of the methods supported:
|
|||||||
<td>—</td>
|
<td>—</td>
|
||||||
<td>Toggles whether captions are enabled.</td>
|
<td>Toggles whether captions are enabled.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>toggleFullscreen()</code></td>
|
||||||
|
<td>Event</td>
|
||||||
|
<td>Toggles fullscreen. This can only be initiated by a user gesture due to browser security, i.e. a user event such as click.</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>isFullscreen()</code></td>
|
||||||
|
<td>—</td>
|
||||||
|
<td>Boolean returned if the player is in fullscreen.</td>
|
||||||
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>support(...)</code></td>
|
<td><code>support(...)</code></td>
|
||||||
<td>String</td>
|
<td>String</td>
|
||||||
@ -332,6 +357,16 @@ Here's a list of the methods supported:
|
|||||||
<td>String</td>
|
<td>String</td>
|
||||||
<td>Set the poster url. This is supported for the <code>video</code> element only.</td>
|
<td>Set the poster url. This is supported for the <code>video</code> element only.</td>
|
||||||
</tr>
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>destroy()</code></td>
|
||||||
|
<td>—</td>
|
||||||
|
<td>Destroys the plyr UI and any media event listeners, effectively restoring to the previous state before <code>setup()</code> was called.</td>
|
||||||
|
</tr>
|
||||||
|
<tr>
|
||||||
|
<td><code>restore()</code></td>
|
||||||
|
<td>—</td>
|
||||||
|
<td>Reverses the effects of the <code>destroy()</code> method, restoring the UI and listeners.</td>
|
||||||
|
</tr>
|
||||||
</tbody>
|
</tbody>
|
||||||
</table>
|
</table>
|
||||||
|
|
||||||
@ -350,11 +385,12 @@ A complete list of events can be found here:
|
|||||||
[Media Events - W3.org](http://www.w3.org/2010/05/video/mediaevents.html)
|
[Media Events - W3.org](http://www.w3.org/2010/05/video/mediaevents.html)
|
||||||
|
|
||||||
## Fullscreen
|
## Fullscreen
|
||||||
|
|
||||||
Fullscreen in Plyr is supported for all browsers that [currently support it](http://caniuse.com/#feat=fullscreen). If you're using the default CSS, you can also use a "full browser" mode which will use the full browser window by adding the `player-fullscreen` class to your container.
|
Fullscreen in Plyr is supported for all browsers that [currently support it](http://caniuse.com/#feat=fullscreen). If you're using the default CSS, you can also use a "full browser" mode which will use the full browser window by adding the `player-fullscreen` class to your container.
|
||||||
|
|
||||||
## Browser support
|
## Browser support
|
||||||
|
|
||||||
<table width="100%" style="text-align: center;">
|
<table width="100%" style="text-align: center">
|
||||||
<thead>
|
<thead>
|
||||||
<tr>
|
<tr>
|
||||||
<td>Safari</td>
|
<td>Safari</td>
|
||||||
@ -392,11 +428,14 @@ If a User Agent is disabled but supports `<video>` and `<audio>` natively, it wi
|
|||||||
|
|
||||||
Any unsupported browsers will display links to download the media if the correct html is used.
|
Any unsupported browsers will display links to download the media if the correct html is used.
|
||||||
|
|
||||||
|
### Checking for support
|
||||||
|
There's an API method for checking support. You can call `plyr.supported()` and optionally pass a type to it, e.g. `plyr.supported("video")`. It will return an object with two keys; `basic` meaning there's basic support for that media type (or both if no type is passed) and `full` meaning there's full support for plyr.
|
||||||
|
|
||||||
## Issues
|
## Issues
|
||||||
If you find anything weird with Plyr, please let us know using the GitHub issues tracker.
|
If you find anything weird with Plyr, please let us know using the GitHub issues tracker.
|
||||||
|
|
||||||
## Author
|
## Author
|
||||||
Plyr is developed by Sam Potts ([@sam_potts](https://twitter.com/sam_potts)) ([sampotts.me](http://sampotts.me))
|
Plyr is developed by [@sam_potts](https://twitter.com/sam_potts) / [sampotts.me](http://sampotts.me) with help from the awesome [contributors](https://github.com/Selz/plyr/graphs/contributors)
|
||||||
|
|
||||||
## Mentions
|
## Mentions
|
||||||
- [The Changelog](http://thechangelog.com/plyr-simple-html5-media-player-custom-controls-webvtt-captions/)
|
- [The Changelog](http://thechangelog.com/plyr-simple-html5-media-player-custom-controls-webvtt-captions/)
|
||||||
|
335
src/js/plyr.js
335
src/js/plyr.js
@ -1,6 +1,6 @@
|
|||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
// Plyr
|
// Plyr
|
||||||
// plyr.js v1.1.1
|
// plyr.js v1.1.14
|
||||||
// https://github.com/selz/plyr
|
// https://github.com/selz/plyr
|
||||||
// License: The MIT License (MIT)
|
// License: The MIT License (MIT)
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
@ -22,6 +22,7 @@
|
|||||||
click: true,
|
click: true,
|
||||||
tooltips: false,
|
tooltips: false,
|
||||||
displayDuration: true,
|
displayDuration: true,
|
||||||
|
iconPrefix: "icon",
|
||||||
selectors: {
|
selectors: {
|
||||||
container: ".player",
|
container: ".player",
|
||||||
controls: ".player-controls",
|
controls: ".player-controls",
|
||||||
@ -80,7 +81,8 @@
|
|||||||
enabled: true,
|
enabled: true,
|
||||||
key: "plyr_volume"
|
key: "plyr_volume"
|
||||||
},
|
},
|
||||||
controls: ["restart", "rewind", "play", "fast-forward", "current-time", "duration", "mute", "volume", "captions", "fullscreen"]
|
controls: ["restart", "rewind", "play", "fast-forward", "current-time", "duration", "mute", "volume", "captions", "fullscreen"],
|
||||||
|
onSetup: function() {},
|
||||||
};
|
};
|
||||||
|
|
||||||
// Build the default HTML
|
// Build the default HTML
|
||||||
@ -104,7 +106,7 @@
|
|||||||
if(_inArray(config.controls, "restart")) {
|
if(_inArray(config.controls, "restart")) {
|
||||||
html.push(
|
html.push(
|
||||||
"<button type='button' data-player='restart'>",
|
"<button type='button' data-player='restart'>",
|
||||||
"<svg><use xlink:href='#icon-restart'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-restart'></use></svg>",
|
||||||
"<span class='sr-only'>Restart</span>",
|
"<span class='sr-only'>Restart</span>",
|
||||||
"</button>"
|
"</button>"
|
||||||
);
|
);
|
||||||
@ -114,7 +116,7 @@
|
|||||||
if(_inArray(config.controls, "rewind")) {
|
if(_inArray(config.controls, "rewind")) {
|
||||||
html.push(
|
html.push(
|
||||||
"<button type='button' data-player='rewind'>",
|
"<button type='button' data-player='rewind'>",
|
||||||
"<svg><use xlink:href='#icon-rewind'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-rewind'></use></svg>",
|
||||||
"<span class='sr-only'>Rewind {seektime} secs</span>",
|
"<span class='sr-only'>Rewind {seektime} secs</span>",
|
||||||
"</button>"
|
"</button>"
|
||||||
);
|
);
|
||||||
@ -124,11 +126,11 @@
|
|||||||
if(_inArray(config.controls, "play")) {
|
if(_inArray(config.controls, "play")) {
|
||||||
html.push(
|
html.push(
|
||||||
"<button type='button' data-player='play'>",
|
"<button type='button' data-player='play'>",
|
||||||
"<svg><use xlink:href='#icon-play'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-play'></use></svg>",
|
||||||
"<span class='sr-only'>Play</span>",
|
"<span class='sr-only'>Play</span>",
|
||||||
"</button>",
|
"</button>",
|
||||||
"<button type='button' data-player='pause'>",
|
"<button type='button' data-player='pause'>",
|
||||||
"<svg><use xlink:href='#icon-pause'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-pause'></use></svg>",
|
||||||
"<span class='sr-only'>Pause</span>",
|
"<span class='sr-only'>Pause</span>",
|
||||||
"</button>"
|
"</button>"
|
||||||
);
|
);
|
||||||
@ -138,7 +140,7 @@
|
|||||||
if(_inArray(config.controls, "fast-forward")) {
|
if(_inArray(config.controls, "fast-forward")) {
|
||||||
html.push(
|
html.push(
|
||||||
"<button type='button' data-player='fast-forward'>",
|
"<button type='button' data-player='fast-forward'>",
|
||||||
"<svg><use xlink:href='#icon-fast-forward'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-fast-forward'></use></svg>",
|
||||||
"<span class='sr-only'>Forward {seektime} secs</span>",
|
"<span class='sr-only'>Forward {seektime} secs</span>",
|
||||||
"</button>"
|
"</button>"
|
||||||
);
|
);
|
||||||
@ -175,8 +177,8 @@
|
|||||||
html.push(
|
html.push(
|
||||||
"<input class='inverted sr-only' id='mute{id}' type='checkbox' data-player='mute'>",
|
"<input class='inverted sr-only' id='mute{id}' type='checkbox' data-player='mute'>",
|
||||||
"<label id='mute{id}' for='mute{id}'>",
|
"<label id='mute{id}' for='mute{id}'>",
|
||||||
"<svg class='icon-muted'><use xlink:href='#icon-muted'></use></svg>",
|
"<svg class='icon-muted'><use xlink:href='#" + config.iconPrefix + "-muted'></use></svg>",
|
||||||
"<svg><use xlink:href='#icon-volume'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-volume'></use></svg>",
|
||||||
"<span class='sr-only'>Toggle Mute</span>",
|
"<span class='sr-only'>Toggle Mute</span>",
|
||||||
"</label>"
|
"</label>"
|
||||||
);
|
);
|
||||||
@ -195,8 +197,8 @@
|
|||||||
html.push(
|
html.push(
|
||||||
"<input class='sr-only' id='captions{id}' type='checkbox' data-player='captions'>",
|
"<input class='sr-only' id='captions{id}' type='checkbox' data-player='captions'>",
|
||||||
"<label for='captions{id}'>",
|
"<label for='captions{id}'>",
|
||||||
"<svg class='icon-captions-on'><use xlink:href='#icon-captions-on'></use></svg>",
|
"<svg class='icon-captions-on'><use xlink:href='#" + config.iconPrefix + "-captions-on'></use></svg>",
|
||||||
"<svg><use xlink:href='#icon-captions-off'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-captions-off'></use></svg>",
|
||||||
"<span class='sr-only'>Toggle Captions</span>",
|
"<span class='sr-only'>Toggle Captions</span>",
|
||||||
"</label>"
|
"</label>"
|
||||||
);
|
);
|
||||||
@ -206,8 +208,8 @@
|
|||||||
if(_inArray(config.controls, "fullscreen")) {
|
if(_inArray(config.controls, "fullscreen")) {
|
||||||
html.push(
|
html.push(
|
||||||
"<button type='button' data-player='fullscreen'>",
|
"<button type='button' data-player='fullscreen'>",
|
||||||
"<svg class='icon-exit-fullscreen'><use xlink:href='#icon-exit-fullscreen'></use></svg>",
|
"<svg class='icon-exit-fullscreen'><use xlink:href='#" + config.iconPrefix + "-exit-fullscreen'></use></svg>",
|
||||||
"<svg><use xlink:href='#icon-enter-fullscreen'></use></svg>",
|
"<svg><use xlink:href='#" + config.iconPrefix + "-enter-fullscreen'></use></svg>",
|
||||||
"<span class='sr-only'>Toggle Fullscreen</span>",
|
"<span class='sr-only'>Toggle Fullscreen</span>",
|
||||||
"</button>"
|
"</button>"
|
||||||
);
|
);
|
||||||
@ -272,6 +274,7 @@
|
|||||||
else if ((nameOffset=nAgt.lastIndexOf(" ") + 1) < (verOffset=nAgt.lastIndexOf("/"))) {
|
else if ((nameOffset=nAgt.lastIndexOf(" ") + 1) < (verOffset=nAgt.lastIndexOf("/"))) {
|
||||||
name = nAgt.substring(nameOffset,verOffset);
|
name = nAgt.substring(nameOffset,verOffset);
|
||||||
fullVersion = nAgt.substring(verOffset + 1);
|
fullVersion = nAgt.substring(verOffset + 1);
|
||||||
|
|
||||||
if (name.toLowerCase() == name.toUpperCase()) {
|
if (name.toLowerCase() == name.toUpperCase()) {
|
||||||
name = navigator.appName;
|
name = navigator.appName;
|
||||||
}
|
}
|
||||||
@ -340,7 +343,7 @@
|
|||||||
|
|
||||||
// Wrap an element
|
// Wrap an element
|
||||||
function _wrap(elements, wrapper) {
|
function _wrap(elements, wrapper) {
|
||||||
// Convert `elms` to an array, if necessary.
|
// Convert `elements` to an array, if necessary.
|
||||||
if (!elements.length) {
|
if (!elements.length) {
|
||||||
elements = [elements];
|
elements = [elements];
|
||||||
}
|
}
|
||||||
@ -349,27 +352,43 @@
|
|||||||
// first element (see `child` below).
|
// first element (see `child` below).
|
||||||
for (var i = elements.length - 1; i >= 0; i--) {
|
for (var i = elements.length - 1; i >= 0; i--) {
|
||||||
var child = (i > 0) ? wrapper.cloneNode(true) : wrapper;
|
var child = (i > 0) ? wrapper.cloneNode(true) : wrapper;
|
||||||
var el = elements[i];
|
var element = elements[i];
|
||||||
|
|
||||||
// Cache the current parent and sibling.
|
// Cache the current parent and sibling.
|
||||||
var parent = el.parentNode;
|
var parent = element.parentNode;
|
||||||
var sibling = el.nextSibling;
|
var sibling = element.nextSibling;
|
||||||
|
|
||||||
// Wrap the element (is automatically removed from its current
|
// Wrap the element (is automatically removed from its current
|
||||||
// parent).
|
// parent).
|
||||||
child.appendChild(el);
|
child.appendChild(element);
|
||||||
|
|
||||||
// If the element had a sibling, insert the wrapper before
|
// If the element had a sibling, insert the wrapper before
|
||||||
// the sibling to maintain the HTML structure; otherwise, just
|
// the sibling to maintain the HTML structure; otherwise, just
|
||||||
// append it to the parent.
|
// append it to the parent.
|
||||||
if (sibling) {
|
if (sibling) {
|
||||||
parent.insertBefore(child, sibling);
|
parent.insertBefore(child, sibling);
|
||||||
} else {
|
}
|
||||||
|
else {
|
||||||
parent.appendChild(child);
|
parent.appendChild(child);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Unwrap an element
|
||||||
|
// http://plainjs.com/javascript/manipulation/unwrap-a-dom-element-35/
|
||||||
|
function _unwrap(wrapper) {
|
||||||
|
// Get the element's parent node
|
||||||
|
var parent = wrapper.parentNode;
|
||||||
|
|
||||||
|
// Move all children out of the element
|
||||||
|
while (wrapper.firstChild) {
|
||||||
|
parent.insertBefore(wrapper.firstChild, wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Remove the empty element
|
||||||
|
parent.removeChild(wrapper);
|
||||||
|
}
|
||||||
|
|
||||||
// Remove an element
|
// Remove an element
|
||||||
function _remove(element) {
|
function _remove(element) {
|
||||||
element.parentNode.removeChild(element);
|
element.parentNode.removeChild(element);
|
||||||
@ -402,7 +421,7 @@
|
|||||||
|
|
||||||
// Toggle event
|
// Toggle event
|
||||||
function _toggleHandler(element, events, callback, toggle) {
|
function _toggleHandler(element, events, callback, toggle) {
|
||||||
events = events.split(" ");
|
var eventList = events.split(" ");
|
||||||
|
|
||||||
// If a nodelist is passed, call itself on each node
|
// If a nodelist is passed, call itself on each node
|
||||||
if(element instanceof NodeList) {
|
if(element instanceof NodeList) {
|
||||||
@ -415,8 +434,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// If a single node is passed, bind the event listener
|
// If a single node is passed, bind the event listener
|
||||||
for (var i = 0; i < events.length; i++) {
|
for (var i = 0; i < eventList.length; i++) {
|
||||||
element[toggle ? "addEventListener" : "removeEventListener"](events[i], callback, false);
|
element[toggle ? "addEventListener" : "removeEventListener"](eventList[i], callback, false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -519,7 +538,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Safari doesn't support the ALLOW_KEYBOARD_INPUT flag so set it to not supported
|
// Safari doesn't support the ALLOW_KEYBOARD_INPUT flag (for security) so set it to not supported
|
||||||
// https://bugs.webkit.org/show_bug.cgi?id=121496
|
// https://bugs.webkit.org/show_bug.cgi?id=121496
|
||||||
if(fullscreen.prefix === "webkit" && !!navigator.userAgent.match(/Version\/[\d\.]+.*Safari/)) {
|
if(fullscreen.prefix === "webkit" && !!navigator.userAgent.match(/Version\/[\d\.]+.*Safari/)) {
|
||||||
fullscreen.supportsFullScreen = false;
|
fullscreen.supportsFullScreen = false;
|
||||||
@ -531,19 +550,18 @@
|
|||||||
// Sometimes the prefix is "ms", sometimes "MS" to keep you on your toes
|
// Sometimes the prefix is "ms", sometimes "MS" to keep you on your toes
|
||||||
fullscreen.fullScreenEventName = (fullscreen.prefix == "ms" ? "MSFullscreenChange" : fullscreen.prefix + "fullscreenchange");
|
fullscreen.fullScreenEventName = (fullscreen.prefix == "ms" ? "MSFullscreenChange" : fullscreen.prefix + "fullscreenchange");
|
||||||
|
|
||||||
fullscreen.isFullScreen = function() {
|
fullscreen.isFullScreen = function(element) {
|
||||||
|
if(typeof element == "undefined") {
|
||||||
|
element = document;
|
||||||
|
}
|
||||||
|
|
||||||
switch (this.prefix) {
|
switch (this.prefix) {
|
||||||
case "":
|
case "":
|
||||||
return document.fullScreen;
|
return document.fullscreenElement == element;
|
||||||
case "webkit":
|
case "moz":
|
||||||
return document.webkitIsFullScreen;
|
return document.mozFullScreenElement == element;
|
||||||
case "ms":
|
|
||||||
// Docs say document.msFullScreenElement returns undefined
|
|
||||||
// if no element is full screem but it returns null, cheers
|
|
||||||
// https://msdn.microsoft.com/en-us/library/ie/dn265028%28v=vs.85%29.aspx
|
|
||||||
return (document.msFullscreenElement !== null);
|
|
||||||
default:
|
default:
|
||||||
return document[this.prefix + "FullScreen"];
|
return document[this.prefix + "FullscreenElement"] == element;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
fullscreen.requestFullScreen = function(element) {
|
fullscreen.requestFullScreen = function(element) {
|
||||||
@ -681,22 +699,25 @@
|
|||||||
|
|
||||||
// Insert controls
|
// Insert controls
|
||||||
function _injectControls() {
|
function _injectControls() {
|
||||||
|
// Make a copy of the html
|
||||||
|
var html = config.html;
|
||||||
|
|
||||||
// Insert custom video controls
|
// Insert custom video controls
|
||||||
_log("Injecting custom controls.");
|
_log("Injecting custom controls.");
|
||||||
|
|
||||||
// If no controls are specified, create default
|
// If no controls are specified, create default
|
||||||
if(!config.html) {
|
if(!html) {
|
||||||
config.html = _buildControls();
|
html = _buildControls();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Replace seek time instances
|
// Replace seek time instances
|
||||||
config.html = _replaceAll(config.html, "{seektime}", config.seekTime);
|
html = _replaceAll(html, "{seektime}", config.seekTime);
|
||||||
|
|
||||||
// Replace all id references
|
// Replace all id references with random numbers
|
||||||
config.html = _replaceAll(config.html, "{id}", player.random);
|
html = _replaceAll(html, "{id}", Math.floor(Math.random() * (10000)));
|
||||||
|
|
||||||
// Inject into the container
|
// Inject into the container
|
||||||
player.container.insertAdjacentHTML("beforeend", config.html);
|
player.container.insertAdjacentHTML("beforeend", html);
|
||||||
|
|
||||||
// Setup tooltips
|
// Setup tooltips
|
||||||
if(config.tooltips) {
|
if(config.tooltips) {
|
||||||
@ -881,13 +902,13 @@
|
|||||||
// Enable UI
|
// Enable UI
|
||||||
_showCaptions(player);
|
_showCaptions(player);
|
||||||
|
|
||||||
// If IE 10/11 or Firefox 31+ or Safari 7+, don"t use native captioning (still doesn"t work although they claim it"s now supported)
|
// Disable unsupported browsers than report false positive
|
||||||
if ((player.browser.name === "IE" && player.browser.version === 10) ||
|
if ((player.browser.name === "IE" && player.browser.version >= 10) ||
|
||||||
(player.browser.name === "IE" && player.browser.version === 11) ||
|
|
||||||
(player.browser.name === "Firefox" && player.browser.version >= 31) ||
|
(player.browser.name === "Firefox" && player.browser.version >= 31) ||
|
||||||
|
(player.browser.name === "Chrome" && player.browser.version >= 43) ||
|
||||||
(player.browser.name === "Safari" && player.browser.version >= 7)) {
|
(player.browser.name === "Safari" && player.browser.version >= 7)) {
|
||||||
// Debugging
|
// Debugging
|
||||||
_log("Detected IE 10/11 or Firefox 31+ or Safari 7+.");
|
_log("Detected unsupported browser for HTML5 captions. Using fallback.");
|
||||||
|
|
||||||
// Set to false so skips to "manual" captioning
|
// Set to false so skips to "manual" captioning
|
||||||
player.usingTextTracks = false;
|
player.usingTextTracks = false;
|
||||||
@ -1007,6 +1028,22 @@
|
|||||||
player.media.pause();
|
player.media.pause();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Toggle playback
|
||||||
|
function _togglePlay(toggle) {
|
||||||
|
// Play
|
||||||
|
if(toggle === true) {
|
||||||
|
_play();
|
||||||
|
}
|
||||||
|
// Pause
|
||||||
|
else if(toggle === false) {
|
||||||
|
_pause();
|
||||||
|
}
|
||||||
|
// True toggle
|
||||||
|
else {
|
||||||
|
player.media[player.media.paused ? "play" : "pause"]();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Rewind
|
// Rewind
|
||||||
function _rewind(seekTime) {
|
function _rewind(seekTime) {
|
||||||
// Use default if needed
|
// Use default if needed
|
||||||
@ -1035,7 +1072,7 @@
|
|||||||
targetTime = input;
|
targetTime = input;
|
||||||
}
|
}
|
||||||
// Event
|
// Event
|
||||||
else if (typeof input === "object" && (input.type === "change" || input.type === "input")) {
|
else if (typeof input === "object" && (input.type === "input" || input.type === "change")) {
|
||||||
// It's the seek slider
|
// It's the seek slider
|
||||||
// Seek to the selected time
|
// Seek to the selected time
|
||||||
targetTime = ((input.target.value / input.target.max) * player.media.duration);
|
targetTime = ((input.target.value / input.target.max) * player.media.duration);
|
||||||
@ -1076,12 +1113,12 @@
|
|||||||
|
|
||||||
// If it's a fullscreen change event, it's probably a native close
|
// If it's a fullscreen change event, it's probably a native close
|
||||||
if(event && event.type === fullscreen.fullScreenEventName) {
|
if(event && event.type === fullscreen.fullScreenEventName) {
|
||||||
config.fullscreen.active = fullscreen.isFullScreen();
|
player.isFullscreen = fullscreen.isFullScreen(player.container);
|
||||||
}
|
}
|
||||||
// If there's native support, use it
|
// If there's native support, use it
|
||||||
else if(nativeSupport) {
|
else if(nativeSupport) {
|
||||||
// Request fullscreen
|
// Request fullscreen
|
||||||
if(!fullscreen.isFullScreen()) {
|
if(!fullscreen.isFullScreen(player.container)) {
|
||||||
fullscreen.requestFullScreen(player.container);
|
fullscreen.requestFullScreen(player.container);
|
||||||
}
|
}
|
||||||
// Bail from fullscreen
|
// Bail from fullscreen
|
||||||
@ -1090,14 +1127,14 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Check if we're actually full screen (it could fail)
|
// Check if we're actually full screen (it could fail)
|
||||||
config.fullscreen.active = fullscreen.isFullScreen();
|
player.isFullscreen = fullscreen.isFullScreen(player.container);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
// Otherwise, it's a simple toggle
|
// Otherwise, it's a simple toggle
|
||||||
config.fullscreen.active = !config.fullscreen.active;
|
player.isFullscreen = !player.isFullscreen;
|
||||||
|
|
||||||
// Bind/unbind escape key
|
// Bind/unbind escape key
|
||||||
if(config.fullscreen.active) {
|
if(player.isFullscreen) {
|
||||||
_on(document, "keyup", _handleEscapeFullscreen);
|
_on(document, "keyup", _handleEscapeFullscreen);
|
||||||
document.body.style.overflow = "hidden";
|
document.body.style.overflow = "hidden";
|
||||||
}
|
}
|
||||||
@ -1108,25 +1145,55 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Set class hook
|
// Set class hook
|
||||||
_toggleClass(player.container, config.classes.fullscreen.active, config.fullscreen.active);
|
_toggleClass(player.container, config.classes.fullscreen.active, player.isFullscreen);
|
||||||
|
|
||||||
|
// Toggle controls visibility based on mouse movement and location
|
||||||
|
var hoverTimer, isMouseOver = false;
|
||||||
|
|
||||||
|
// Show the player controls
|
||||||
|
function _showControls() {
|
||||||
|
// Set shown class
|
||||||
|
_toggleClass(player.controls, config.classes.hover, true);
|
||||||
|
|
||||||
|
// Clear timer every movement
|
||||||
|
window.clearTimeout(hoverTimer);
|
||||||
|
|
||||||
|
// If the mouse is not over the controls, set a timeout to hide them
|
||||||
|
if(!isMouseOver) {
|
||||||
|
hoverTimer = window.setTimeout(function() {
|
||||||
|
_toggleClass(player.controls, config.classes.hover, false);
|
||||||
|
}, 2000);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check mouse is over the controls
|
||||||
|
function _setMouseOver (event) {
|
||||||
|
isMouseOver = (event.type === "mouseenter");
|
||||||
|
}
|
||||||
|
|
||||||
|
if(config.fullscreen.hideControls) {
|
||||||
|
// Hide on entering full screen
|
||||||
|
_toggleClass(player.controls, config.classes.hover, false);
|
||||||
|
|
||||||
|
// Keep an eye on the mouse location in relation to controls
|
||||||
|
_toggleHandler(player.controls, "mouseenter mouseleave", _setMouseOver, player.isFullscreen);
|
||||||
|
|
||||||
|
// Show the controls on mouse move
|
||||||
|
_toggleHandler(player.container, "mousemove", _showControls, player.isFullscreen);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bail from faux-fullscreen
|
// Bail from faux-fullscreen
|
||||||
function _handleEscapeFullscreen(event) {
|
function _handleEscapeFullscreen(event) {
|
||||||
// If it's a keypress and not escape, bail
|
// If it's a keypress and not escape, bail
|
||||||
if((event.which || event.charCode || event.keyCode) === 27 && config.fullscreen.active) {
|
if((event.which || event.charCode || event.keyCode) === 27 && player.isFullscreen) {
|
||||||
_toggleFullscreen();
|
_toggleFullscreen();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set volume
|
// Set volume
|
||||||
function _setVolume(volume) {
|
function _setVolume(volume) {
|
||||||
// Bail if there's no volume element
|
// Use default if no value specified
|
||||||
if(!player.volume) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Use default if needed
|
|
||||||
if(typeof volume === "undefined") {
|
if(typeof volume === "undefined") {
|
||||||
if(config.storage.enabled && _storage().supported) {
|
if(config.storage.enabled && _storage().supported) {
|
||||||
volume = window.localStorage[config.storage.key] || config.volume;
|
volume = window.localStorage[config.storage.key] || config.volume;
|
||||||
@ -1135,25 +1202,22 @@
|
|||||||
volume = config.volume;
|
volume = config.volume;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Maximum is 10
|
// Maximum is 10
|
||||||
if(volume > 10) {
|
if(volume > 10) {
|
||||||
volume = 10;
|
volume = 10;
|
||||||
}
|
}
|
||||||
|
// Minimum is 0
|
||||||
// If the controls are there
|
if(volume < 0) {
|
||||||
if(player.supported.full) {
|
volume = 0;
|
||||||
player.volume.value = volume;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Set the player volume
|
// Set the player volume
|
||||||
player.media.volume = parseFloat(volume / 10);
|
player.media.volume = parseFloat(volume / 10);
|
||||||
|
|
||||||
// Update the UI
|
// Toggle muted state
|
||||||
_checkMute();
|
if(player.media.muted && volume > 0) {
|
||||||
|
_toggleMute();
|
||||||
// Store the volume in storage
|
|
||||||
if(config.storage.enabled && _storage().supported) {
|
|
||||||
window.localStorage.plyr_volume = volume;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1164,16 +1228,32 @@
|
|||||||
muted = !player.media.muted;
|
muted = !player.media.muted;
|
||||||
}
|
}
|
||||||
|
|
||||||
// If the controls are there
|
|
||||||
if(player.supported.full) {
|
|
||||||
player.buttons.mute.checked = muted;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Set mute on the player
|
// Set mute on the player
|
||||||
player.media.muted = muted;
|
player.media.muted = muted;
|
||||||
|
}
|
||||||
|
|
||||||
// Update UI
|
// Update volume UI and storage
|
||||||
_checkMute();
|
function _updateVolume() {
|
||||||
|
// Get the current volume
|
||||||
|
var volume = player.media.muted ? 0 : (player.media.volume * 10);
|
||||||
|
|
||||||
|
// Update the <input type="range"> if present
|
||||||
|
if(player.supported.full && player.volume) {
|
||||||
|
player.volume.value = volume;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Store the volume in storage
|
||||||
|
if(config.storage.enabled && _storage().supported) {
|
||||||
|
window.localStorage.setItem(config.storage.key, volume);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Toggle class if muted
|
||||||
|
_toggleClass(player.container, config.classes.muted, (volume === 0));
|
||||||
|
|
||||||
|
// Update checkbox for mute state
|
||||||
|
if(player.supported.full && player.buttons.mute) {
|
||||||
|
player.buttons.mute.checked = (volume === 0);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Toggle captions
|
// Toggle captions
|
||||||
@ -1192,11 +1272,6 @@
|
|||||||
_toggleClass(player.container, config.classes.captions.active, show);
|
_toggleClass(player.container, config.classes.captions.active, show);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check mute state
|
|
||||||
function _checkMute() {
|
|
||||||
_toggleClass(player.container, config.classes.muted, (player.media.volume === 0 || player.media.muted));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if media is loading
|
// Check if media is loading
|
||||||
function _checkLoading(event) {
|
function _checkLoading(event) {
|
||||||
var loading = (event.type === "waiting");
|
var loading = (event.type === "waiting");
|
||||||
@ -1391,6 +1466,9 @@
|
|||||||
|
|
||||||
// Listen for events
|
// Listen for events
|
||||||
function _listeners() {
|
function _listeners() {
|
||||||
|
// IE doesn't support input event, so we fallback to change
|
||||||
|
var inputEvent = (player.browser.name == "IE" ? "change" : "input");
|
||||||
|
|
||||||
// Play
|
// Play
|
||||||
_on(player.buttons.play, "click", function() {
|
_on(player.buttons.play, "click", function() {
|
||||||
_play();
|
_play();
|
||||||
@ -1412,9 +1490,11 @@
|
|||||||
// Fast forward
|
// Fast forward
|
||||||
_on(player.buttons.forward, "click", _forward);
|
_on(player.buttons.forward, "click", _forward);
|
||||||
|
|
||||||
// Get the HTML5 range input element and append audio volume adjustment on change/input
|
// Seek
|
||||||
// IE10 doesn't support the "input" event so they have to wait for change
|
_on(player.buttons.seek, inputEvent, _seek);
|
||||||
_on(player.volume, "change input", function() {
|
|
||||||
|
// Set volume
|
||||||
|
_on(player.volume, inputEvent, function() {
|
||||||
_setVolume(this.value);
|
_setVolume(this.value);
|
||||||
});
|
});
|
||||||
|
|
||||||
@ -1440,9 +1520,6 @@
|
|||||||
// Display duration
|
// Display duration
|
||||||
_on(player.media, "loadedmetadata", _displayDuration);
|
_on(player.media, "loadedmetadata", _displayDuration);
|
||||||
|
|
||||||
// Seek
|
|
||||||
_on(player.buttons.seek, "change input", _seek);
|
|
||||||
|
|
||||||
// Captions
|
// Captions
|
||||||
_on(player.buttons.captions, "change", function() {
|
_on(player.buttons.captions, "change", function() {
|
||||||
_toggleCaptions(this.checked);
|
_toggleCaptions(this.checked);
|
||||||
@ -1466,7 +1543,7 @@
|
|||||||
_on(player.media, "playing", _updateProgress);
|
_on(player.media, "playing", _updateProgress);
|
||||||
|
|
||||||
// Handle native mute
|
// Handle native mute
|
||||||
_on(player.media, "volumechange", _checkMute);
|
_on(player.media, "volumechange", _updateVolume);
|
||||||
|
|
||||||
// Handle native play/pause
|
// Handle native play/pause
|
||||||
_on(player.media, "play pause", _checkPlaying);
|
_on(player.media, "play pause", _checkPlaying);
|
||||||
@ -1481,27 +1558,63 @@
|
|||||||
if(player.type === "video" && config.click) {
|
if(player.type === "video" && config.click) {
|
||||||
_on(player.videoContainer, "click", function() {
|
_on(player.videoContainer, "click", function() {
|
||||||
if(player.media.paused) {
|
if(player.media.paused) {
|
||||||
_play();
|
_triggerEvent(player.buttons.play, "click");
|
||||||
}
|
}
|
||||||
else if(player.media.ended) {
|
else if(player.media.ended) {
|
||||||
_seek();
|
_seek();
|
||||||
_play();
|
_triggerEvent(player.buttons.play, "click");
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
_pause();
|
_triggerEvent(player.buttons.pause, "click");
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
// Bind to mouse hover
|
|
||||||
if(config.fullscreen.hideControls) {
|
|
||||||
_on(player.controls, "mouseenter mouseleave", function(event) {
|
|
||||||
_toggleClass(player.controls, config.classes.hover, (event.type === "mouseenter"));
|
|
||||||
})
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Destroy an instance
|
||||||
|
function _destroy() {
|
||||||
|
// Bail if the element is not initialized
|
||||||
|
if(!player.init) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reset container classname
|
||||||
|
player.container.setAttribute("class", config.selectors.container.replace(".", ""));
|
||||||
|
|
||||||
|
// Event listeners are removed when elements are removed
|
||||||
|
// http://stackoverflow.com/questions/12528049/if-a-dom-element-is-removed-are-its-listeners-also-removed-from-memory
|
||||||
|
|
||||||
|
// Remove controls
|
||||||
|
_remove(_getElement(config.selectors.controls));
|
||||||
|
|
||||||
|
// If video, we need to remove some more
|
||||||
|
if(player.type === "video") {
|
||||||
|
// Remove captions
|
||||||
|
_remove(_getElement(config.selectors.captions));
|
||||||
|
|
||||||
|
// Remove video wrapper
|
||||||
|
_unwrap(player.videoContainer);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Restore native video controls
|
||||||
|
player.media.setAttribute("controls", "");
|
||||||
|
|
||||||
|
// Clone the media element to remove listeners
|
||||||
|
// http://stackoverflow.com/questions/19469881/javascript-remove-all-event-listeners-of-specific-type
|
||||||
|
var clone = player.media.cloneNode(true);
|
||||||
|
player.media.parentNode.replaceChild(clone, player.media);
|
||||||
|
|
||||||
|
// Remove init flag
|
||||||
|
player.init = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup a player
|
||||||
function _init() {
|
function _init() {
|
||||||
|
// Bail if the element is initialized
|
||||||
|
if(player.init) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
// Setup the fullscreen api
|
// Setup the fullscreen api
|
||||||
fullscreen = _fullscreen();
|
fullscreen = _fullscreen();
|
||||||
|
|
||||||
@ -1528,9 +1641,6 @@
|
|||||||
// Setup media
|
// Setup media
|
||||||
_setupMedia();
|
_setupMedia();
|
||||||
|
|
||||||
// Generate random number for id/for attribute values for controls
|
|
||||||
player.random = Math.floor(Math.random() * (10000));
|
|
||||||
|
|
||||||
// If there's full support
|
// If there's full support
|
||||||
if(player.supported.full) {
|
if(player.supported.full) {
|
||||||
// Inject custom controls
|
// Inject custom controls
|
||||||
@ -1554,6 +1664,7 @@
|
|||||||
|
|
||||||
// Set volume
|
// Set volume
|
||||||
_setVolume();
|
_setVolume();
|
||||||
|
_updateVolume();
|
||||||
|
|
||||||
// Setup fullscreen
|
// Setup fullscreen
|
||||||
_setupFullscreen();
|
_setupFullscreen();
|
||||||
@ -1563,10 +1674,14 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Successful setup
|
// Successful setup
|
||||||
return true;
|
player.init = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!_init()) {
|
// Initialize instance
|
||||||
|
_init();
|
||||||
|
|
||||||
|
// If init failed, return an empty object
|
||||||
|
if(!player.init) {
|
||||||
return {};
|
return {};
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1578,12 +1693,17 @@
|
|||||||
rewind: _rewind,
|
rewind: _rewind,
|
||||||
forward: _forward,
|
forward: _forward,
|
||||||
seek: _seek,
|
seek: _seek,
|
||||||
setVolume: _setVolume,
|
|
||||||
toggleMute: _toggleMute,
|
|
||||||
toggleCaptions: _toggleCaptions,
|
|
||||||
source: _parseSource,
|
source: _parseSource,
|
||||||
poster: _updatePoster,
|
poster: _updatePoster,
|
||||||
support: function(mimeType) { return _supportMime(player, mimeType); }
|
setVolume: _setVolume,
|
||||||
|
togglePlay: _togglePlay,
|
||||||
|
toggleMute: _toggleMute,
|
||||||
|
toggleCaptions: _toggleCaptions,
|
||||||
|
toggleFullscreen: _toggleFullscreen,
|
||||||
|
isFullscreen: function() { return player.isFullscreen || false; },
|
||||||
|
support: function(mimeType) { return _supportMime(player, mimeType); },
|
||||||
|
destroy: _destroy,
|
||||||
|
restore: _init
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1646,9 +1766,12 @@
|
|||||||
|
|
||||||
// Set plyr to false if setup failed
|
// Set plyr to false if setup failed
|
||||||
element.plyr = (Object.keys(instance).length ? instance : false);
|
element.plyr = (Object.keys(instance).length ? instance : false);
|
||||||
|
|
||||||
|
// Callback
|
||||||
|
config.onSetup.apply(element.plyr);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Add to return array
|
// Add to return array even if it's already setup
|
||||||
players.push(element.plyr);
|
players.push(element.plyr);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -9,6 +9,7 @@
|
|||||||
@gray-dark: #343f4a;
|
@gray-dark: #343f4a;
|
||||||
@gray: #565d64;
|
@gray: #565d64;
|
||||||
@gray-light: #cbd0d3;
|
@gray-light: #cbd0d3;
|
||||||
|
@off-white: #d6dadd;
|
||||||
|
|
||||||
// Font sizes
|
// Font sizes
|
||||||
@font-size-small: 14px;
|
@font-size-small: 14px;
|
||||||
@ -462,7 +463,8 @@
|
|||||||
margin: 0 @control-spacing 0 0;
|
margin: 0 @control-spacing 0 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background: none;
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
|
||||||
// Webkit
|
// Webkit
|
||||||
&::-webkit-slider-runnable-track {
|
&::-webkit-slider-runnable-track {
|
||||||
@ -527,6 +529,17 @@
|
|||||||
float: none;
|
float: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Audio specific styles
|
||||||
|
// Position the progress within the container
|
||||||
|
&-audio .player-controls {
|
||||||
|
padding-top: (@control-spacing * 2);
|
||||||
|
}
|
||||||
|
&-audio .player-progress {
|
||||||
|
bottom: auto;
|
||||||
|
top: 0;
|
||||||
|
background: @off-white;
|
||||||
|
}
|
||||||
|
|
||||||
// Full screen mode
|
// Full screen mode
|
||||||
&-fullscreen,
|
&-fullscreen,
|
||||||
&.fullscreen-active {
|
&.fullscreen-active {
|
||||||
@ -551,12 +564,9 @@
|
|||||||
top: auto;
|
top: auto;
|
||||||
bottom: 90px;
|
bottom: 90px;
|
||||||
|
|
||||||
@media (min-width: @bp-control-split) and (max-width: (@bp-captions-large - 1)) {
|
@media (min-width: @bp-control-split) {
|
||||||
bottom: 60px;
|
bottom: 60px;
|
||||||
}
|
}
|
||||||
@media (min-width: @bp-captions-large) {
|
|
||||||
bottom: 80px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.player-controls {
|
.player-controls {
|
||||||
@ -569,11 +579,10 @@
|
|||||||
// Hide controls when playing in full screen
|
// Hide controls when playing in full screen
|
||||||
&.fullscreen-hide-controls.playing .player-controls {
|
&.fullscreen-hide-controls.playing .player-controls {
|
||||||
transform: translateY(100%) translateY(@control-spacing / 2);
|
transform: translateY(100%) translateY(@control-spacing / 2);
|
||||||
transition: transform .3s 1s ease;
|
transition: transform .3s .2s ease;
|
||||||
|
|
||||||
&.hover {
|
&.hover {
|
||||||
transform: translateY(0);
|
transform: translateY(0);
|
||||||
transition-delay: 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -5,49 +5,50 @@
|
|||||||
// Variables
|
// Variables
|
||||||
// -------------------------------
|
// -------------------------------
|
||||||
// Colors
|
// Colors
|
||||||
$blue: #3498DB;
|
$blue: #3498DB !default;
|
||||||
$gray-dark: #343f4a;
|
$gray-dark: #343f4a !default;
|
||||||
$gray: #565d64;
|
$gray: #565d64 !default;
|
||||||
$gray-light: #cbd0d3;
|
$gray-light: #cbd0d3 !default;
|
||||||
|
$off-white: #d6dadd !default;
|
||||||
|
|
||||||
// Font sizes
|
// Font sizes
|
||||||
$font-size-small: 14px;
|
$font-size-small: 14px !default;
|
||||||
$font-size-base: 16px;
|
$font-size-base: 16px !default;
|
||||||
$font-size-large: ceil(($font-size-base * 1.5));
|
$font-size-large: ceil(($font-size-base * 1.5)) !default;
|
||||||
|
|
||||||
// Controls
|
// Controls
|
||||||
$control-spacing: 10px;
|
$control-spacing: 10px !default;
|
||||||
$controls-bg: $gray-dark;
|
$controls-bg: $gray-dark !default;
|
||||||
$control-bg-hover: $blue;
|
$control-bg-hover: $blue !default;
|
||||||
$control-color: $gray-light;
|
$control-color: $gray-light !default;
|
||||||
$control-color-inactive: $gray;
|
$control-color-inactive: $gray !default;
|
||||||
$control-color-hover: #fff;
|
$control-color-hover: #fff !default;
|
||||||
|
|
||||||
// Tooltips
|
// Tooltips
|
||||||
$tooltip-bg: $controls-bg;
|
$tooltip-bg: $controls-bg !default;
|
||||||
$tooltip-color: #fff;
|
$tooltip-color: #fff !default;
|
||||||
$tooltip-padding: $control-spacing;
|
$tooltip-padding: $control-spacing !default;
|
||||||
$tooltip-arrow-size: 5px;
|
$tooltip-arrow-size: 5px !default;
|
||||||
$tooltip-radius: 3px;
|
$tooltip-radius: 3px !default;
|
||||||
|
|
||||||
// Progress
|
// Progress
|
||||||
$progress-bg: rgba(red($gray), green($gray), blue($gray), .2);
|
$progress-bg: rgba(red($gray), green($gray), blue($gray), .2) !default;
|
||||||
$progress-playing-bg: $blue;
|
$progress-playing-bg: $blue !default;
|
||||||
$progress-buffered-bg: rgba(red($gray), green($gray), blue($gray), .25);
|
$progress-buffered-bg: rgba(red($gray), green($gray), blue($gray), .25) !default;
|
||||||
$progress-loading-size: 40px;
|
$progress-loading-size: 40px !default;
|
||||||
$progress-loading-bg: rgba(0,0,0, .15);
|
$progress-loading-bg: rgba(0,0,0, .15) !default;
|
||||||
|
|
||||||
// Volume
|
// Volume
|
||||||
$volume-track-height: 6px;
|
$volume-track-height: 6px !default;
|
||||||
$volume-track-bg: $gray;
|
$volume-track-bg: $gray !default;
|
||||||
$volume-thumb-height: ($volume-track-height * 2);
|
$volume-thumb-height: ($volume-track-height * 2) !default;
|
||||||
$volume-thumb-width: ($volume-track-height * 2);
|
$volume-thumb-width: ($volume-track-height * 2) !default;
|
||||||
$volume-thumb-bg: $control-color;
|
$volume-thumb-bg: $control-color !default;
|
||||||
$volume-thumb-bg-focus: $control-bg-hover;
|
$volume-thumb-bg-focus: $control-bg-hover !default;
|
||||||
|
|
||||||
// Breakpoints
|
// Breakpoints
|
||||||
$bp-control-split: 560px; // When controls split into left/right
|
$bp-control-split: 560px !default; // When controls split into left/right
|
||||||
$bp-captions-large: 768px; // When captions jump to the larger font size
|
$bp-captions-large: 768px !default; // When captions jump to the larger font size
|
||||||
|
|
||||||
// Utility classes & mixins
|
// Utility classes & mixins
|
||||||
// -------------------------------
|
// -------------------------------
|
||||||
@ -470,7 +471,8 @@ $bp-captions-large: 768px; // When captions jump to the larger font size
|
|||||||
margin: 0 $control-spacing 0 0;
|
margin: 0 $control-spacing 0 0;
|
||||||
padding: 0;
|
padding: 0;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
background: none;
|
background: transparent;
|
||||||
|
border: none;
|
||||||
|
|
||||||
// Webkit
|
// Webkit
|
||||||
&::-webkit-slider-runnable-track {
|
&::-webkit-slider-runnable-track {
|
||||||
@ -535,6 +537,17 @@ $bp-captions-large: 768px; // When captions jump to the larger font size
|
|||||||
float: none;
|
float: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Audio specific styles
|
||||||
|
// Position the progress within the container
|
||||||
|
&-audio .player-controls {
|
||||||
|
padding-top: ($control-spacing * 2);
|
||||||
|
}
|
||||||
|
&-audio .player-progress {
|
||||||
|
bottom: auto;
|
||||||
|
top: 0;
|
||||||
|
background: $off-white;
|
||||||
|
}
|
||||||
|
|
||||||
// Full screen mode
|
// Full screen mode
|
||||||
&-fullscreen,
|
&-fullscreen,
|
||||||
&.fullscreen-active {
|
&.fullscreen-active {
|
||||||
@ -559,12 +572,9 @@ $bp-captions-large: 768px; // When captions jump to the larger font size
|
|||||||
top: auto;
|
top: auto;
|
||||||
bottom: 90px;
|
bottom: 90px;
|
||||||
|
|
||||||
@media (min-width: $bp-control-split) and (max-width: ($bp-captions-large - 1)) {
|
@media (min-width: $bp-control-split) {
|
||||||
bottom: 60px;
|
bottom: 60px;
|
||||||
}
|
}
|
||||||
@media (min-width: $bp-captions-large) {
|
|
||||||
bottom: 80px;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
.player-controls {
|
.player-controls {
|
||||||
@ -577,11 +587,10 @@ $bp-captions-large: 768px; // When captions jump to the larger font size
|
|||||||
// Hide controls when playing in full screen
|
// Hide controls when playing in full screen
|
||||||
&.fullscreen-hide-controls.playing .player-controls {
|
&.fullscreen-hide-controls.playing .player-controls {
|
||||||
transform: translateY(100%) translateY($control-spacing / 2);
|
transform: translateY(100%) translateY($control-spacing / 2);
|
||||||
transition: transform .3s 1s ease;
|
transition: transform .3s .2s ease;
|
||||||
|
|
||||||
&.hover {
|
&.hover {
|
||||||
transform: translateY(0);
|
transform: translateY(0);
|
||||||
transition-delay: 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user