Compare commits
18 Commits
Author | SHA1 | Date | |
---|---|---|---|
e49c417e54 | |||
b39961ec49 | |||
8894b4c7b9 | |||
cdf3deb458 | |||
b5fc21239b | |||
93cc9edd9a | |||
dcd9ca3a93 | |||
c202cc1ffb | |||
093af22942 | |||
9d966e41b1 | |||
240aa7aa5f | |||
654e9cd623 | |||
73c3888309 | |||
398815857f | |||
4c5020a396 | |||
3d1a586314 | |||
d04b278802 | |||
7345f656c1 |
30
changelog.md
30
changelog.md
@ -1,5 +1,35 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
|
## v1.2.0
|
||||||
|
- Added YouTube support.
|
||||||
|
|
||||||
|
## 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
|
## v1.1.5
|
||||||
- Fix for incorrect `isFullscreen()` return value in Mozilla (Fixes #38)
|
- Fix for incorrect `isFullscreen()` return value in Mozilla (Fixes #38)
|
||||||
|
|
||||||
|
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
docs/dist/docs.css
vendored
2
docs/dist/docs.css
vendored
File diff suppressed because one or more lines are too long
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.13/docs.css">
|
<link rel="stylesheet" href="//cdn.plyr.io/1.2.0/docs.css">
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<main>
|
<main>
|
||||||
|
@ -8,31 +8,41 @@
|
|||||||
<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="https://cdn.plyr.io/1.1.13/plyr.css?1">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.2.0/plyr.css?1">
|
||||||
|
|
||||||
<!-- Docs styles -->
|
<!-- Docs styles -->
|
||||||
<link rel="stylesheet" href="https://cdn.plyr.io/1.1.13/docs.css?1">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.2.0/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 by <a href="https://twitter.com/sam_potts" target="_blank">@sam_potts</a></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>
|
||||||
<ul class="actions">
|
<nav>
|
||||||
|
<ul>
|
||||||
<li>
|
<li>
|
||||||
<a href="https://github.com/selz/plyr" target="_blank" class="btn btn-download">Download on GitHub</a>
|
<a href="https://github.com/selz/plyr" target="_blank" class="btn btn-primary">Download on GitHub</a>
|
||||||
<span class="btn-count js-stargazers-count">…</span>
|
<span class="btn-count js-stargazers-count">…</span>
|
||||||
</li>
|
</li>
|
||||||
<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>
|
<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 js-popup" data-window-height="250" data-window-width="500">Tweet</a>
|
||||||
<span class="btn-count js-tweet-count">…</span>
|
<span class="btn-count js-tweet-count">…</span>
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
|
</nav>
|
||||||
</header>
|
</header>
|
||||||
|
|
||||||
<main>
|
<main role="main" id="main">
|
||||||
<section class="example-video">
|
<nav class="btn-bar nav-panel">
|
||||||
|
<ul>
|
||||||
|
<li><a href="#video" class="btn active btn-small">Video</a></li>
|
||||||
|
<li><a href="#youtube" class="btn btn-small">YouTube</a></li>
|
||||||
|
<li><a href="#audio" class="btn btn-small">Audio</a></li>
|
||||||
|
</ul>
|
||||||
|
</nav>
|
||||||
|
<div class="panels">
|
||||||
|
<section class="panel example-video active" id="video">
|
||||||
<div class="player">
|
<div class="player">
|
||||||
<video poster="https://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="https://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="https://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">
|
||||||
@ -46,29 +56,28 @@
|
|||||||
</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>
|
||||||
</section>
|
</section>
|
||||||
|
<section class="panel example-video" id="youtube">
|
||||||
<section class="example-audio">
|
<div class="player">
|
||||||
|
<div data-video-id="Au87oAJ2jeE" data-type="youtube"></div>
|
||||||
|
</div>
|
||||||
|
<small>Envato's "Made By" interview of <a href="https://www.youtube.com/watch?v=Au87oAJ2jeE" target="_blank">Dan Cederholm</a> from <a href="https://dribbble.com" target="_blank">Dribbble</a>.</small>
|
||||||
|
</section>
|
||||||
|
<section class="panel example-audio" id="audio">
|
||||||
<div class="player">
|
<div class="player">
|
||||||
<audio controls>
|
<audio controls>
|
||||||
<!-- Audio files -->
|
<!-- Audio files -->
|
||||||
<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.mp3" type="audio/mp3">
|
||||||
<source src="https://cdn.selz.com/plyr/1.0/logistics-96-sample.ogg" type="audio/ogg">
|
<source src="//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 -->
|
||||||
<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>
|
||||||
</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>
|
||||||
</section>
|
</section>
|
||||||
|
</div>
|
||||||
</main>
|
</main>
|
||||||
|
|
||||||
<footer>
|
|
||||||
<p>Used by …</p>
|
|
||||||
<a href="https://selz.com" target="_blank" class="logo">
|
|
||||||
<img src="https://d33i624pw6jj68.cloudfront.net/static/img/logos/selz.svg" alt="Selz">
|
|
||||||
</a>
|
|
||||||
</footer>
|
|
||||||
|
|
||||||
<!-- Load SVG defs -->
|
<!-- Load SVG defs -->
|
||||||
<!-- You should bundle all SVG/Icons into one file using a build tool such as gulp and svg store -->
|
<!-- You should bundle all SVG/Icons into one file using a build tool such as gulp and svg store -->
|
||||||
<script>
|
<script>
|
||||||
@ -88,13 +97,13 @@
|
|||||||
b.insertBefore(c, b.childNodes[0]);
|
b.insertBefore(c, b.childNodes[0]);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})(document, "https://cdn.plyr.io/1.1.13/sprite.svg");
|
})(document, "https://cdn.plyr.io/1.2.0/sprite.svg");
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<!-- Plyr core script -->
|
<!-- Plyr core script -->
|
||||||
<script src="https://cdn.plyr.io/1.1.13/plyr.js?1"></script>
|
<script src="https://cdn.plyr.io/1.2.0/plyr.js?1"></script>
|
||||||
|
|
||||||
<!-- Docs script -->
|
<!-- Docs script -->
|
||||||
<script src="https://cdn.plyr.io/1.1.13/docs.js?1"></script>
|
<script src="https://cdn.plyr.io/1.2.0/docs.js?1"></script>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
@ -14,6 +14,10 @@ plyr.setup({
|
|||||||
defaultActive: true
|
defaultActive: true
|
||||||
},
|
},
|
||||||
onSetup: function() {
|
onSetup: function() {
|
||||||
|
if(!("media" in this)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
var player = this,
|
var player = this,
|
||||||
type = player.media.tagName.toLowerCase(),
|
type = player.media.tagName.toLowerCase(),
|
||||||
toggle = document.querySelector("[data-toggle='fullscreen']");
|
toggle = document.querySelector("[data-toggle='fullscreen']");
|
||||||
@ -26,6 +30,8 @@ plyr.setup({
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// General functions
|
||||||
|
(function() {
|
||||||
// Popup
|
// Popup
|
||||||
function popup(event) {
|
function popup(event) {
|
||||||
// Prevent the link opening
|
// Prevent the link opening
|
||||||
@ -136,6 +142,33 @@ else {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Tabs
|
||||||
|
var tabs = document.querySelectorAll(".nav-panel a"),
|
||||||
|
panels = document.querySelectorAll(".panels > .panel"),
|
||||||
|
activeClass = "active";
|
||||||
|
|
||||||
|
for (var i = tabs.length - 1; i >= 0; i--) {
|
||||||
|
tabs[i].addEventListener("click", togglePanel);
|
||||||
|
}
|
||||||
|
|
||||||
|
function togglePanel(event) {
|
||||||
|
event.preventDefault();
|
||||||
|
|
||||||
|
var tab = event.target,
|
||||||
|
panel = document.querySelector(tab.getAttribute("href"));
|
||||||
|
|
||||||
|
for (var i = panels.length - 1; i >= 0; i--) {
|
||||||
|
panels[i].classList.remove(activeClass);
|
||||||
|
}
|
||||||
|
|
||||||
|
for (var x = tabs.length - 1; x >= 0; x--) {
|
||||||
|
tabs[x].classList.remove(activeClass);
|
||||||
|
}
|
||||||
|
|
||||||
|
panel.classList.add(activeClass);
|
||||||
|
event.target.classList.add(activeClass);
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
|
||||||
// Google analytics
|
// Google analytics
|
||||||
// For demo site (http://[www.]plyr.io) only
|
// For demo site (http://[www.]plyr.io) only
|
||||||
|
45
docs/src/less/components/base.less
Normal file
45
docs/src/less/components/base.less
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Base layout
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// BORDER-BOX ALL THE THINGS!
|
||||||
|
// http://paulirish.com/2012/box-sizing-border-box-ftw/
|
||||||
|
*, *::after, *::before {
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Base
|
||||||
|
html {
|
||||||
|
font-size: 100%;
|
||||||
|
}
|
||||||
|
body {
|
||||||
|
font-family: "Avenir", "Helvetica Neue", Helvetica, Arial, sans-serif;
|
||||||
|
background: @body-background;
|
||||||
|
line-height: 1.5;
|
||||||
|
text-align: center;
|
||||||
|
color: @gray;
|
||||||
|
.font-smoothing(on);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Header
|
||||||
|
header {
|
||||||
|
padding: @padding-base;
|
||||||
|
margin-bottom: @padding-base;
|
||||||
|
|
||||||
|
p {
|
||||||
|
.font-size(18);
|
||||||
|
}
|
||||||
|
@media (min-width: @screen-sm) {
|
||||||
|
padding-top: (@padding-base * 3);
|
||||||
|
padding-bottom: (@padding-base * 3);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Sections
|
||||||
|
section {
|
||||||
|
padding-bottom: @padding-base;
|
||||||
|
|
||||||
|
@media (min-width: @screen-sm) {
|
||||||
|
padding-bottom: (@padding-base * 2);
|
||||||
|
}
|
||||||
|
}
|
144
docs/src/less/components/buttons.less
Normal file
144
docs/src/less/components/buttons.less
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Buttons
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
nav {
|
||||||
|
ul {
|
||||||
|
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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Tabs
|
||||||
|
.btn-bar {
|
||||||
|
position: relative;
|
||||||
|
margin: 0 auto @padding-base;
|
||||||
|
max-width: @example-width-video;
|
||||||
|
|
||||||
|
&::before {
|
||||||
|
content: "";
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
height: 1px;
|
||||||
|
background: @gray-lighter;
|
||||||
|
}
|
||||||
|
|
||||||
|
ul {
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
display: inline-block;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
li {
|
||||||
|
margin: 0;
|
||||||
|
|
||||||
|
&:first-child .btn {
|
||||||
|
border-radius: 4px 0 0 4px;
|
||||||
|
}
|
||||||
|
&:last-child .btn {
|
||||||
|
border-radius: 0 4px 4px 0;
|
||||||
|
}
|
||||||
|
& + li .btn {
|
||||||
|
margin-left: -1px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.btn {
|
||||||
|
display: block;
|
||||||
|
border-radius: 0;
|
||||||
|
}
|
||||||
|
.active {
|
||||||
|
&:extend(.btn-primary);
|
||||||
|
}
|
||||||
|
.btn.active {
|
||||||
|
box-shadow: inset 0 1px 1px rgba(0,0,0, .2);
|
||||||
|
position: relative;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (min-width: 560px) {
|
||||||
|
margin-bottom: (@padding-base * 2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shared
|
||||||
|
.btn,
|
||||||
|
.btn-count {
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
border-radius: @border-radius-base;
|
||||||
|
font-weight: 600;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Buttons
|
||||||
|
.btn {
|
||||||
|
padding: (@padding-base / 2) @padding-base;
|
||||||
|
background: @body-background;
|
||||||
|
border: 1px solid @gray-light;
|
||||||
|
box-shadow: inset 0 1px 0 #fff, 0 1px 1px rgba(0,0,0, .05);
|
||||||
|
text-shadow: 0 1px 1px #fff;
|
||||||
|
color: @gray;
|
||||||
|
|
||||||
|
&:hover,
|
||||||
|
&:focus {
|
||||||
|
background-color: #fff;
|
||||||
|
border-color: darken(@gray-light, 5%);
|
||||||
|
color: @link-color;
|
||||||
|
outline: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.btn-primary {
|
||||||
|
background: linear-gradient(@link-color, darken(@link-color, 3%));
|
||||||
|
border-color: darken(@link-color, 10%);
|
||||||
|
box-shadow: 0 1px 1px rgba(0,0,0, .15);
|
||||||
|
text-shadow: 0 1px 1px rgba(0,0,0, .1);
|
||||||
|
color: #fff;
|
||||||
|
|
||||||
|
&:hover,
|
||||||
|
&:focus {
|
||||||
|
color: #fff;
|
||||||
|
border-color: darken(@link-color, 20%);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.btn-small {
|
||||||
|
padding-top: ceil(@padding-base / 3);
|
||||||
|
padding-bottom: ceil(@padding-base / 3);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Count bubble
|
||||||
|
.btn-count {
|
||||||
|
position: relative;
|
||||||
|
margin-left: 6px;
|
||||||
|
padding: ((@padding-base / 2) - 1px);
|
||||||
|
background: @body-background;
|
||||||
|
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%);
|
||||||
|
}
|
||||||
|
}
|
19
docs/src/less/components/error.less
Normal file
19
docs/src/less/components/error.less
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Errors (AWS pages)
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Error page
|
||||||
|
html.error,
|
||||||
|
.error body {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.error body {
|
||||||
|
width: 100%;
|
||||||
|
display: table;
|
||||||
|
table-layout: fixed;
|
||||||
|
}
|
||||||
|
.error main {
|
||||||
|
display: table-cell;
|
||||||
|
width: 100%;
|
||||||
|
vertical-align: middle;
|
||||||
|
}
|
39
docs/src/less/components/examples.less
Normal file
39
docs/src/less/components/examples.less
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Examples
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Example players
|
||||||
|
.example-audio .player,
|
||||||
|
.example-video .player {
|
||||||
|
margin: 0 auto @padding-base;
|
||||||
|
|
||||||
|
&-controls {
|
||||||
|
border-radius: 0 0 @border-radius-base @border-radius-base;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.example-audio .player {
|
||||||
|
max-width: @example-width-audio;
|
||||||
|
|
||||||
|
&-controls {
|
||||||
|
border-radius: @border-radius-base;
|
||||||
|
}
|
||||||
|
&-progress {
|
||||||
|
border-radius: @border-radius-base @border-radius-base 0 0;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
.example-video .player {
|
||||||
|
max-width: @example-width-video;
|
||||||
|
|
||||||
|
video,
|
||||||
|
iframe {
|
||||||
|
border-radius: @border-radius-base @border-radius-base 0 0;
|
||||||
|
}
|
||||||
|
iframe {
|
||||||
|
-webkit-mask-image: url(data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAIAAACQd1PeAAAAGXRFWHRTb2Z0d2FyZQBBZG9iZSBJbWFnZVJlYWR5ccllPAAAAA5JREFUeNpiYGBgAAgwAAAEAAGbA+oJAAAAAElFTkSuQmCC);
|
||||||
|
}
|
||||||
|
&-fullscreen,
|
||||||
|
&.fullscreen-active {
|
||||||
|
max-width: none;
|
||||||
|
}
|
||||||
|
}
|
13
docs/src/less/components/panels.less
Normal file
13
docs/src/less/components/panels.less
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Panels
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Panels
|
||||||
|
.panel {
|
||||||
|
display: none;
|
||||||
|
|
||||||
|
&.active {
|
||||||
|
display: block;
|
||||||
|
animation: fade-in .3s ease;
|
||||||
|
}
|
||||||
|
}
|
47
docs/src/less/components/type.less
Normal file
47
docs/src/less/components/type.less
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Typography
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Headings
|
||||||
|
h1,
|
||||||
|
h2 {
|
||||||
|
letter-spacing: -.025em;
|
||||||
|
color: #2E3C44;
|
||||||
|
margin: 0 0 (@padding-base / 2);
|
||||||
|
line-height: 1.2;
|
||||||
|
.font-smoothing();
|
||||||
|
}
|
||||||
|
h1 {
|
||||||
|
.font-size(64);
|
||||||
|
color: #3498DB;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Paragraph and small
|
||||||
|
p,
|
||||||
|
small {
|
||||||
|
margin: 0 0 @padding-base;
|
||||||
|
}
|
||||||
|
small {
|
||||||
|
display: block;
|
||||||
|
padding: 0 (@padding-base / 2);
|
||||||
|
.font-size(14);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Links
|
||||||
|
a {
|
||||||
|
text-decoration: none;
|
||||||
|
color: @link-color;
|
||||||
|
border-bottom: 1px solid currentColor;
|
||||||
|
transition: background .3s ease, color .3s ease;
|
||||||
|
|
||||||
|
&:hover,
|
||||||
|
&:focus {
|
||||||
|
color: @gray-dark;
|
||||||
|
}
|
||||||
|
&:focus {
|
||||||
|
.tab-focus();
|
||||||
|
}
|
||||||
|
&.logo {
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
}
|
@ -2,232 +2,33 @@
|
|||||||
// HTML5 Video Player Demo Page
|
// HTML5 Video Player Demo Page
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
|
|
||||||
// Reset
|
// CSS Reset
|
||||||
@import "lib/normalize.less";
|
@import "lib/normalize.less";
|
||||||
|
|
||||||
// Mixins
|
// Mixins
|
||||||
@import "lib/mixins.less";
|
@import "lib/mixins.less";
|
||||||
// Fonts - docs only!
|
|
||||||
@import "lib/fontface.less";
|
|
||||||
|
|
||||||
// Variables
|
// Variables
|
||||||
// ---------------------------------------
|
@import "variables.less";
|
||||||
// Colors
|
|
||||||
@blue: #3498DB;
|
|
||||||
@gray-dark: #343f4a;
|
|
||||||
@gray: #565d64;
|
|
||||||
@gray-light: #cbd0d3;
|
|
||||||
|
|
||||||
// Elements
|
// Animation
|
||||||
@link-color: @blue;
|
@import "lib/animation.less";
|
||||||
@padding-base: 20px;
|
|
||||||
@arrow-size: 8px;
|
|
||||||
|
|
||||||
// Breakpoints
|
// Base layout
|
||||||
@screen-md: 768px;
|
@import "components/base.less";
|
||||||
|
|
||||||
// BORDER-BOX ALL THE THINGS!
|
|
||||||
// http://paulirish.com/2012/box-sizing-border-box-ftw/
|
|
||||||
*, *::after, *::before {
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Base
|
|
||||||
body {
|
|
||||||
font-family: "Avenir", "Helvetica Neue", Helvetica, Arial, sans-serif;
|
|
||||||
background: #fff;
|
|
||||||
line-height: 1.5;
|
|
||||||
text-align: center;
|
|
||||||
color: #6D797F;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Error page
|
|
||||||
html.error,
|
|
||||||
.error body {
|
|
||||||
height: 100%;
|
|
||||||
}
|
|
||||||
.error body {
|
|
||||||
width: 100%;
|
|
||||||
display: table;
|
|
||||||
table-layout: fixed;
|
|
||||||
}
|
|
||||||
.error main {
|
|
||||||
display: table-cell;
|
|
||||||
width: 100%;
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Type
|
// Type
|
||||||
h1,
|
@import "lib/fontface.less";
|
||||||
h2 {
|
@import "components/type.less";
|
||||||
letter-spacing: -.025em;
|
|
||||||
color: #2E3C44;
|
|
||||||
margin: 0 0 (@padding-base / 2);
|
|
||||||
line-height: 1.2;
|
|
||||||
.font-smoothing();
|
|
||||||
}
|
|
||||||
h1 {
|
|
||||||
.font-size(64);
|
|
||||||
color: #3498DB;
|
|
||||||
}
|
|
||||||
p,
|
|
||||||
small {
|
|
||||||
margin: 0 0 @padding-base;
|
|
||||||
}
|
|
||||||
small {
|
|
||||||
display: block;
|
|
||||||
padding: 0 (@padding-base / 2);
|
|
||||||
.font-size(14);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Header
|
// Buttons
|
||||||
header {
|
@import "components/buttons.less";
|
||||||
padding: @padding-base;
|
|
||||||
margin-bottom: @padding-base;
|
|
||||||
|
|
||||||
p {
|
// Panels
|
||||||
.font-size(18);
|
@import "components/panels.less";
|
||||||
}
|
|
||||||
@media (min-width: 560px) {
|
|
||||||
padding-top: (@padding-base * 3);
|
|
||||||
padding-bottom: (@padding-base * 3);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sections
|
// Error
|
||||||
section {
|
@import "components/error.less";
|
||||||
padding-bottom: @padding-base;
|
|
||||||
|
|
||||||
@media (min-width: 560px) {
|
// Examples
|
||||||
padding-bottom: (@padding-base * 2);
|
@import "components/examples.less";
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// 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 {
|
|
||||||
text-decoration: none;
|
|
||||||
color: @link-color;
|
|
||||||
border-bottom: 1px solid currentColor;
|
|
||||||
transition: all .3s ease;
|
|
||||||
|
|
||||||
&:hover,
|
|
||||||
&:focus {
|
|
||||||
color: #000;
|
|
||||||
}
|
|
||||||
&:focus {
|
|
||||||
.tab-focus();
|
|
||||||
}
|
|
||||||
&.logo {
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
.btn,
|
|
||||||
.btn-count {
|
|
||||||
display: inline-block;
|
|
||||||
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;
|
|
||||||
border: 0;
|
|
||||||
color: #fff;
|
|
||||||
|
|
||||||
&:hover,
|
|
||||||
&:focus {
|
|
||||||
color: #fff;
|
|
||||||
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%);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Example players
|
|
||||||
.example-audio .player,
|
|
||||||
.example-video .player {
|
|
||||||
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-active {
|
|
||||||
max-width: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Footer
|
|
||||||
footer {
|
|
||||||
margin-bottom: @padding-base;
|
|
||||||
|
|
||||||
p {
|
|
||||||
margin-bottom: (@padding-base / 2);
|
|
||||||
}
|
|
||||||
}
|
|
9
docs/src/less/lib/animation.less
Normal file
9
docs/src/less/lib/animation.less
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Animations
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Fade
|
||||||
|
@keyframes fade-in {
|
||||||
|
0% { opacity: 0 }
|
||||||
|
100% { opacity: 1 }
|
||||||
|
}
|
@ -1,16 +1,18 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Fonts
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "Avenir";
|
font-family: "Avenir";
|
||||||
src: url("//cdn.plyr.io/fonts/avenir-medium.woff2") format("woff2"),
|
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.woff") format("woff");
|
||||||
url("//cdn.plyr.io/fonts/avenir-medium.ttf") format("truetype");
|
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-weight: 400;
|
font-weight: 400;
|
||||||
}
|
}
|
||||||
@font-face {
|
@font-face {
|
||||||
font-family: "Avenir";
|
font-family: "Avenir";
|
||||||
src: url("//cdn.plyr.io/fonts/avenir-bold.woff2") format("woff2"),
|
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.woff") format("woff");
|
||||||
url("//cdn.plyr.io/fonts/avenir-bold.ttf") format("truetype");
|
|
||||||
font-style: normal;
|
font-style: normal;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
}
|
}
|
@ -17,7 +17,6 @@
|
|||||||
// Default
|
// Default
|
||||||
outline: thin dotted @gray-dark;
|
outline: thin dotted @gray-dark;
|
||||||
// Webkit
|
// Webkit
|
||||||
//outline: 5px auto -webkit-focus-ring-color;
|
|
||||||
outline-offset: 1px;
|
outline-offset: 1px;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -25,7 +24,7 @@
|
|||||||
// Leave <body> at 100%/16px
|
// Leave <body> at 100%/16px
|
||||||
// ---------------------------------------
|
// ---------------------------------------
|
||||||
.font-size(@font-size: 16){
|
.font-size(@font-size: 16){
|
||||||
@rem: round((@font-size / 16), 1);
|
@rem: round((@font-size / 16), 3);
|
||||||
font-size: (@font-size * 1px);
|
font-size: (@font-size * 1px);
|
||||||
font-size: ~"@{rem}rem";
|
font-size: ~"@{rem}rem";
|
||||||
}
|
}
|
||||||
|
30
docs/src/less/variables.less
Normal file
30
docs/src/less/variables.less
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
// ==========================================================================
|
||||||
|
// Variables
|
||||||
|
// ==========================================================================
|
||||||
|
|
||||||
|
// Colors
|
||||||
|
@blue: #3498db;
|
||||||
|
@gray-dark: #343f4a;
|
||||||
|
@gray: #55646b;
|
||||||
|
@gray-light: #cbd0d3;
|
||||||
|
@gray-lighter: #dbe3e8;
|
||||||
|
@off-white: #f2f5f7;
|
||||||
|
|
||||||
|
// Base
|
||||||
|
@body-background: @off-white;
|
||||||
|
|
||||||
|
// Elements
|
||||||
|
@link-color: @blue;
|
||||||
|
@padding-base: 20px;
|
||||||
|
@arrow-size: 8px;
|
||||||
|
|
||||||
|
// Breakpoints
|
||||||
|
@screen-sm: 480px;
|
||||||
|
@screen-md: 768px;
|
||||||
|
|
||||||
|
// Radii
|
||||||
|
@border-radius-base: 4px;
|
||||||
|
|
||||||
|
// Examples
|
||||||
|
@example-width-audio: 520px;
|
||||||
|
@example-width-video: 1200px;
|
@ -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
|
||||||
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "plyr",
|
"name": "plyr",
|
||||||
"version": "1.1.13",
|
"version": "1.2.0",
|
||||||
"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",
|
||||||
|
60
readme.md
60
readme.md
@ -11,12 +11,13 @@ We wanted a lightweight, accessible and customisable media player that just supp
|
|||||||
## Features
|
## Features
|
||||||
- **Accessible** - full support for captions and screen readers.
|
- **Accessible** - full support for captions and screen readers.
|
||||||
- **Lightweight** - just 6.4KB minified and gzipped.
|
- **Lightweight** - just 6.4KB minified and gzipped.
|
||||||
- **Customisable** - make the player look how you want with the markup you want.
|
- **[Customisable](#html)** - 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.
|
||||||
- **Audio & Video** - support for both formats.
|
- **Audio & Video** - support for both formats.
|
||||||
- **API** - toggle playback, volume, seeking, and more.
|
- **[Embed](#embed)** - support for YouTube (Vimeo soon).
|
||||||
- **Fullscreen** - supports native fullscreen with fallback to "full window" modes.
|
- **[API](#api)** - toggle playback, volume, seeking, and more.
|
||||||
|
- **[Fullscreen](#fullscreen)** - supports native fullscreen with fallback to "full window" modes.
|
||||||
- **No dependencies** - written in vanilla JavaScript, no jQuery required.
|
- **No dependencies** - written in vanilla JavaScript, no jQuery required.
|
||||||
|
|
||||||
Oh and yes, it works with Bootstrap.
|
Oh and yes, it works with Bootstrap.
|
||||||
@ -25,10 +26,11 @@ Oh and yes, it works with Bootstrap.
|
|||||||
Check out [the changelog](changelog.md)
|
Check out [the changelog](changelog.md)
|
||||||
|
|
||||||
## Planned development
|
## Planned development
|
||||||
- Playlists (audio and video)
|
- Playlists
|
||||||
- YouTube and Vimeo support
|
- ~~YouTube~~ and Vimeo support
|
||||||
- Playback speed
|
- Playback speed
|
||||||
- Multiple language captions (with selection)
|
- Multiple language captions (with selection)
|
||||||
|
- Audio captions
|
||||||
... 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.
|
||||||
@ -37,7 +39,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.13/plyr.js` to `https://cdn.plyr.io/1.1.13/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.2.0/plyr.js` to `https://cdn.plyr.io/1.2.0/plyr.js`
|
||||||
|
|
||||||
### Bower
|
### Bower
|
||||||
If bower is your thang, you can grab Plyr using:
|
If bower is your thang, you can grab Plyr using:
|
||||||
@ -57,11 +59,11 @@ More info is on [npm](https://www.npmjs.com/package/ember-cli-plyr) and [GitHub]
|
|||||||
If you want to use our CDN, you can use the following:
|
If you want to use our CDN, you can use the following:
|
||||||
|
|
||||||
```html
|
```html
|
||||||
<link rel="stylesheet" href="https://cdn.plyr.io/1.1.13/plyr.css">
|
<link rel="stylesheet" href="https://cdn.plyr.io/1.2.0/plyr.css">
|
||||||
<script src="https://cdn.plyr.io/1.1.13/plyr.js"></script>
|
<script src="https://cdn.plyr.io/1.2.0/plyr.js"></script>
|
||||||
```
|
```
|
||||||
|
|
||||||
You can also access the `sprite.svg` file at `https://cdn.plyr.io/1.1.13/sprite.svg`.
|
You can also access the `sprite.svg` file at `https://cdn.plyr.io/1.2.0/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.
|
||||||
@ -78,13 +80,13 @@ The SVG sprite for the controls icons is loaded in by AJAX to help with performa
|
|||||||
(function(d, p){
|
(function(d, p){
|
||||||
var a = new XMLHttpRequest(),
|
var a = new XMLHttpRequest(),
|
||||||
b = d.body;
|
b = d.body;
|
||||||
a.open("GET",p,!0);
|
a.open("GET", p, true);
|
||||||
a.send();
|
a.send();
|
||||||
a.onload = function(){
|
a.onload = function(){
|
||||||
var c = d.createElement("div");
|
var c = d.createElement("div");
|
||||||
c.style.display = "none";
|
c.style.display = "none";
|
||||||
c.innerHTML = a.responseText;
|
c.innerHTML = a.responseText;
|
||||||
b.insertBefore(c,b.childNodes[0])
|
b.insertBefore(c, b.childNodes[0]);
|
||||||
}
|
}
|
||||||
})(document, "dist/sprite.svg");
|
})(document, "dist/sprite.svg");
|
||||||
</script>
|
</script>
|
||||||
@ -126,6 +128,14 @@ And the same for `<audio>`
|
|||||||
</div>
|
</div>
|
||||||
```
|
```
|
||||||
|
|
||||||
|
For YouTube, Plyr uses the standard YouTube API markup (an empty `<div>`):
|
||||||
|
|
||||||
|
```html
|
||||||
|
<div class="player">
|
||||||
|
<div data-video-id="L1h9xxCU20g" data-type="youtube"></div>
|
||||||
|
</div>
|
||||||
|
```
|
||||||
|
|
||||||
#### Cross Origin (CORS)
|
#### Cross Origin (CORS)
|
||||||
You'll notice the `crossorigin` attribute on the example `<video>` and `<audio>` elements. This is because the media is loaded from another domain. If your media is hosted on another domain, you may need to add this attribute.
|
You'll notice the `crossorigin` attribute on the example `<video>` and `<audio>` elements. This is because the media is loaded from another domain. If your media is hosted on another domain, you may need to add this attribute.
|
||||||
|
|
||||||
@ -172,6 +182,12 @@ You can pass the following options to the setup method using `plyr.setup({...})`
|
|||||||
<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>
|
||||||
@ -329,7 +345,7 @@ Here's a list of the methods supported:
|
|||||||
<tr>
|
<tr>
|
||||||
<td><code>support(...)</code></td>
|
<td><code>support(...)</code></td>
|
||||||
<td>String</td>
|
<td>String</td>
|
||||||
<td>Determine if a player supports a certain MIME type.</td>
|
<td>Determine if a player supports a certain MIME type. This is not supported for embedded content (YouTube).</td>
|
||||||
</tr>
|
</tr>
|
||||||
<tr>
|
<tr>
|
||||||
<td><code>source(...)</code></td>
|
<td><code>source(...)</code></td>
|
||||||
@ -378,6 +394,26 @@ media.addEventListener("playing", function() {
|
|||||||
A complete list of events can be found here:
|
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)
|
||||||
|
|
||||||
|
## Embeds
|
||||||
|
|
||||||
|
Currently only YouTube is supported. Vimeo will be coming soon. Some HTML5 media events are triggered on the `media` property of the `plyr` object:
|
||||||
|
- `play`
|
||||||
|
- `pause`
|
||||||
|
- `timeupdate`
|
||||||
|
- `progress`
|
||||||
|
|
||||||
|
Due to the way the YouTube API works, the `timeupdate` and `progress` events are triggered by polling every 200ms so the event may trigger without an actual value change. Buffering progress is `media.buffered` in the `plyr` object. It is a a number between 0 and 1 that specifies the percentage of the video that the player shows as buffered.
|
||||||
|
|
||||||
|
```javascript
|
||||||
|
document.querySelector(".player").plyr.media.addEventListener("play", function() {
|
||||||
|
console.log("play");
|
||||||
|
});
|
||||||
|
```
|
||||||
|
|
||||||
|
The `.source()` API method can also be used but the video id must be passed as the argument.
|
||||||
|
|
||||||
|
Currently caption control is not supported but I will work on this.
|
||||||
|
|
||||||
## 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.
|
||||||
|
317
src/js/plyr.js
317
src/js/plyr.js
@ -1,6 +1,6 @@
|
|||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
// Plyr
|
// Plyr
|
||||||
// plyr.js v1.1.13
|
// plyr.js v1.2.0
|
||||||
// https://github.com/selz/plyr
|
// https://github.com/selz/plyr
|
||||||
// License: The MIT License (MIT)
|
// License: The MIT License (MIT)
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
@ -9,6 +9,7 @@
|
|||||||
|
|
||||||
(function (api) {
|
(function (api) {
|
||||||
"use strict";
|
"use strict";
|
||||||
|
/*global YT*/
|
||||||
|
|
||||||
// Globals
|
// Globals
|
||||||
var fullscreen, config;
|
var fullscreen, config;
|
||||||
@ -22,6 +23,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",
|
||||||
@ -48,9 +50,9 @@
|
|||||||
duration: ".player-duration"
|
duration: ".player-duration"
|
||||||
},
|
},
|
||||||
classes: {
|
classes: {
|
||||||
video: "player-video",
|
|
||||||
videoWrapper: "player-video-wrapper",
|
videoWrapper: "player-video-wrapper",
|
||||||
audio: "player-audio",
|
embedWrapper: "player-video-embed",
|
||||||
|
type: "player-{0}",
|
||||||
stopped: "stopped",
|
stopped: "stopped",
|
||||||
playing: "playing",
|
playing: "playing",
|
||||||
muted: "muted",
|
muted: "muted",
|
||||||
@ -81,7 +83,7 @@
|
|||||||
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() {},
|
onSetup: function() {}
|
||||||
};
|
};
|
||||||
|
|
||||||
// Build the default HTML
|
// Build the default HTML
|
||||||
@ -105,7 +107,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>"
|
||||||
);
|
);
|
||||||
@ -115,7 +117,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>"
|
||||||
);
|
);
|
||||||
@ -125,11 +127,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>"
|
||||||
);
|
);
|
||||||
@ -139,7 +141,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>"
|
||||||
);
|
);
|
||||||
@ -176,8 +178,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>"
|
||||||
);
|
);
|
||||||
@ -196,8 +198,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>"
|
||||||
);
|
);
|
||||||
@ -207,8 +209,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>"
|
||||||
);
|
);
|
||||||
@ -330,6 +332,18 @@
|
|||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Inject a script
|
||||||
|
function _injectScript(source) {
|
||||||
|
if(document.querySelectorAll("script[src='" + source + "']").length) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var tag = document.createElement("script");
|
||||||
|
tag.src = source;
|
||||||
|
var firstScriptTag = document.getElementsByTagName("script")[0];
|
||||||
|
firstScriptTag.parentNode.insertBefore(tag, firstScriptTag);
|
||||||
|
}
|
||||||
|
|
||||||
// Element exists in an array
|
// Element exists in an array
|
||||||
function _inArray(haystack, needle) {
|
function _inArray(haystack, needle) {
|
||||||
return Array.prototype.indexOf && (haystack.indexOf(needle) != -1);
|
return Array.prototype.indexOf && (haystack.indexOf(needle) != -1);
|
||||||
@ -816,7 +830,7 @@
|
|||||||
player.media.removeAttribute("controls");
|
player.media.removeAttribute("controls");
|
||||||
|
|
||||||
// Add type class
|
// Add type class
|
||||||
_toggleClass(player.container, config.classes[player.type], true);
|
_toggleClass(player.container, config.classes.type.replace("{0}", player.type), true);
|
||||||
|
|
||||||
// If there's no autoplay attribute, assume the video is stopped and add state class
|
// If there's no autoplay attribute, assume the video is stopped and add state class
|
||||||
_toggleClass(player.container, config.classes.stopped, (player.media.getAttribute("autoplay") === null));
|
_toggleClass(player.container, config.classes.stopped, (player.media.getAttribute("autoplay") === null));
|
||||||
@ -838,6 +852,11 @@
|
|||||||
// Cache the container
|
// Cache the container
|
||||||
player.videoContainer = wrapper;
|
player.videoContainer = wrapper;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
if(player.type == "youtube") {
|
||||||
|
_setupYouTube(player.media.getAttribute("data-video-id"));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Autoplay
|
// Autoplay
|
||||||
@ -846,6 +865,149 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Setup YouTube
|
||||||
|
function _setupYouTube(id) {
|
||||||
|
// Remove old containers
|
||||||
|
var containers = _getElements("[id^='youtube']");
|
||||||
|
for (var i = containers.length - 1; i >= 0; i--) {
|
||||||
|
_remove(containers[i]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Create the YouTube container
|
||||||
|
var container = document.createElement("div");
|
||||||
|
container.setAttribute("id", "youtube-" + Math.floor(Math.random() * (10000)));
|
||||||
|
player.media.appendChild(container);
|
||||||
|
|
||||||
|
// Add embed class for responsive
|
||||||
|
_toggleClass(player.media, config.classes.videoWrapper, true);
|
||||||
|
_toggleClass(player.media, config.classes.embedWrapper, true);
|
||||||
|
|
||||||
|
if(typeof YT === "object") {
|
||||||
|
_YTReady(id, container);
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
// Load the API
|
||||||
|
_injectScript("https://www.youtube.com/iframe_api");
|
||||||
|
|
||||||
|
// Setup callback for the API
|
||||||
|
window.onYouTubeIframeAPIReady = function () { _YTReady(id, container); }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle API ready
|
||||||
|
function _YTReady(id, container) {
|
||||||
|
_log("YouTube API Ready");
|
||||||
|
|
||||||
|
// Setup timers object
|
||||||
|
// We have to poll YouTube for updates
|
||||||
|
if(!("timer" in player)) {
|
||||||
|
player.timer = {};
|
||||||
|
}
|
||||||
|
|
||||||
|
// Setup instance
|
||||||
|
// https://developers.google.com/youtube/iframe_api_reference
|
||||||
|
player.embed = new YT.Player(container.id, {
|
||||||
|
videoId: id,
|
||||||
|
playerVars: {
|
||||||
|
autoplay: 0,
|
||||||
|
controls: 0,
|
||||||
|
vq: "hd720",
|
||||||
|
rel: 0,
|
||||||
|
showinfo: 0,
|
||||||
|
iv_load_policy: 3,
|
||||||
|
cc_lang_pref: "en",
|
||||||
|
wmode: "transparent",
|
||||||
|
modestbranding: 1
|
||||||
|
},
|
||||||
|
events: {
|
||||||
|
onReady: function(event) {
|
||||||
|
// Get the instance
|
||||||
|
var instance = event.target;
|
||||||
|
|
||||||
|
// Create a faux HTML5 API using the YouTube API
|
||||||
|
player.media.play = function() { instance.playVideo(); };
|
||||||
|
player.media.pause = function() { instance.pauseVideo(); };
|
||||||
|
player.media.stop = function() { instance.stopVideo(); };
|
||||||
|
player.media.duration = instance.getDuration();
|
||||||
|
player.media.paused = (instance.getPlayerState() == 2);
|
||||||
|
player.media.currentTime = instance.getCurrentTime();
|
||||||
|
player.media.muted = instance.isMuted();
|
||||||
|
|
||||||
|
// Trigger timeupdate
|
||||||
|
_triggerEvent(player.media, "timeupdate");
|
||||||
|
|
||||||
|
// Reset timer
|
||||||
|
window.clearInterval(player.timer.buffering);
|
||||||
|
|
||||||
|
// Setup buffering
|
||||||
|
player.timer.buffering = window.setInterval(function() {
|
||||||
|
// Get loaded % from YouTube
|
||||||
|
player.media.buffered = instance.getVideoLoadedFraction();
|
||||||
|
|
||||||
|
// Trigger progress
|
||||||
|
_triggerEvent(player.media, "progress");
|
||||||
|
|
||||||
|
// Bail if we're at 100%
|
||||||
|
if(player.media.buffered === 1) {
|
||||||
|
window.clearInterval(player.timer.buffering);
|
||||||
|
}
|
||||||
|
}, 200);
|
||||||
|
|
||||||
|
// Only setup controls once
|
||||||
|
if(!player.container.querySelectorAll(config.selectors.controls).length) {
|
||||||
|
_setupInterface();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Display duration if available
|
||||||
|
if(config.displayDuration) {
|
||||||
|
_displayDuration();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
onStateChange: function(event) {
|
||||||
|
// Get the instance
|
||||||
|
var instance = event.target;
|
||||||
|
|
||||||
|
// Reset timer
|
||||||
|
window.clearInterval(player.timer.playing);
|
||||||
|
|
||||||
|
// Handle events
|
||||||
|
// -1 Unstarted
|
||||||
|
// 0 Ended
|
||||||
|
// 1 Playing
|
||||||
|
// 2 Paused
|
||||||
|
// 3 Buffering
|
||||||
|
// 5 Video cued
|
||||||
|
switch(event.data) {
|
||||||
|
case 0:
|
||||||
|
player.media.paused = true;
|
||||||
|
_triggerEvent(player.media, "ended");
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 1:
|
||||||
|
player.media.paused = false;
|
||||||
|
_triggerEvent(player.media, "play");
|
||||||
|
|
||||||
|
// Poll to get playback progress
|
||||||
|
player.timer.playing = window.setInterval(function() {
|
||||||
|
// Set the current time
|
||||||
|
player.media.currentTime = instance.getCurrentTime();
|
||||||
|
|
||||||
|
// Trigger timeupdate
|
||||||
|
_triggerEvent(player.media, "timeupdate");
|
||||||
|
}, 200);
|
||||||
|
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 2:
|
||||||
|
player.media.paused = true;
|
||||||
|
_triggerEvent(player.media, "pause");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
// Setup captions
|
// Setup captions
|
||||||
function _setupCaptions() {
|
function _setupCaptions() {
|
||||||
if(player.type === "video") {
|
if(player.type === "video") {
|
||||||
@ -996,7 +1158,7 @@
|
|||||||
|
|
||||||
// Setup fullscreen
|
// Setup fullscreen
|
||||||
function _setupFullscreen() {
|
function _setupFullscreen() {
|
||||||
if(player.type === "video" && config.fullscreen.enabled) {
|
if(player.type != "audio" && config.fullscreen.enabled) {
|
||||||
// Check for native support
|
// Check for native support
|
||||||
var nativeSupport = fullscreen.supportsFullScreen;
|
var nativeSupport = fullscreen.supportsFullScreen;
|
||||||
|
|
||||||
@ -1092,6 +1254,14 @@
|
|||||||
}
|
}
|
||||||
catch(e) {}
|
catch(e) {}
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
if(player.type == "youtube") {
|
||||||
|
player.embed.seekTo(player.media.currentTime);
|
||||||
|
|
||||||
|
// Trigger timeupdate
|
||||||
|
_triggerEvent(player.media, "timeupdate");
|
||||||
|
}
|
||||||
|
|
||||||
// Logging
|
// Logging
|
||||||
_log("Seeking to " + player.media.currentTime + " seconds");
|
_log("Seeking to " + player.media.currentTime + " seconds");
|
||||||
|
|
||||||
@ -1202,8 +1372,6 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_log(volume);
|
|
||||||
|
|
||||||
// Maximum is 10
|
// Maximum is 10
|
||||||
if(volume > 10) {
|
if(volume > 10) {
|
||||||
volume = 10;
|
volume = 10;
|
||||||
@ -1216,6 +1384,14 @@
|
|||||||
// Set the player volume
|
// Set the player volume
|
||||||
player.media.volume = parseFloat(volume / 10);
|
player.media.volume = parseFloat(volume / 10);
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
if(player.type == "youtube") {
|
||||||
|
player.embed.setVolume(player.media.volume * 100);
|
||||||
|
|
||||||
|
// Trigger timeupdate
|
||||||
|
_triggerEvent(player.media, "volumechange");
|
||||||
|
}
|
||||||
|
|
||||||
// Toggle muted state
|
// Toggle muted state
|
||||||
if(player.media.muted && volume > 0) {
|
if(player.media.muted && volume > 0) {
|
||||||
_toggleMute();
|
_toggleMute();
|
||||||
@ -1231,6 +1407,14 @@
|
|||||||
|
|
||||||
// Set mute on the player
|
// Set mute on the player
|
||||||
player.media.muted = muted;
|
player.media.muted = muted;
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
if(player.type === "youtube") {
|
||||||
|
player.embed[player.media.muted ? "mute" : "unMute"]();
|
||||||
|
|
||||||
|
// Trigger timeupdate
|
||||||
|
_triggerEvent(player.media, "volumechange");
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update volume UI and storage
|
// Update volume UI and storage
|
||||||
@ -1321,9 +1505,14 @@
|
|||||||
value = (function() {
|
value = (function() {
|
||||||
var buffered = player.media.buffered;
|
var buffered = player.media.buffered;
|
||||||
|
|
||||||
if(buffered.length) {
|
// HTML5
|
||||||
|
if(buffered && buffered.length) {
|
||||||
return _getPercentage(buffered.end(0), player.media.duration);
|
return _getPercentage(buffered.end(0), player.media.duration);
|
||||||
}
|
}
|
||||||
|
// YouTube returns between 0 and 1
|
||||||
|
else if(typeof buffered == "number") {
|
||||||
|
return (buffered * 100);
|
||||||
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
})();
|
})();
|
||||||
@ -1417,6 +1606,22 @@
|
|||||||
// Update source
|
// Update source
|
||||||
// Sources are not checked for support so be careful
|
// Sources are not checked for support so be careful
|
||||||
function _parseSource(sources) {
|
function _parseSource(sources) {
|
||||||
|
// YouTube
|
||||||
|
if(player.type === "youtube" && typeof sources === "string") {
|
||||||
|
// Destroy YouTube instance
|
||||||
|
player.embed.destroy();
|
||||||
|
|
||||||
|
// Re-setup YouTube
|
||||||
|
// We don't use loadVideoBy[x] here since it has issues
|
||||||
|
_setupYouTube(sources);
|
||||||
|
|
||||||
|
// Update times
|
||||||
|
_timeUpdate();
|
||||||
|
|
||||||
|
// Bail
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// Pause playback (webkit freaks out)
|
// Pause playback (webkit freaks out)
|
||||||
_pause();
|
_pause();
|
||||||
|
|
||||||
@ -1538,10 +1743,7 @@
|
|||||||
});
|
});
|
||||||
|
|
||||||
// Check for buffer progress
|
// Check for buffer progress
|
||||||
_on(player.media, "progress", _updateProgress);
|
_on(player.media, "progress playing", _updateProgress);
|
||||||
|
|
||||||
// Also check on start of playing
|
|
||||||
_on(player.media, "playing", _updateProgress);
|
|
||||||
|
|
||||||
// Handle native mute
|
// Handle native mute
|
||||||
_on(player.media, "volumechange", _updateVolume);
|
_on(player.media, "volumechange", _updateVolume);
|
||||||
@ -1573,6 +1775,8 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Destroy an instance
|
// Destroy an instance
|
||||||
|
// 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
|
||||||
function _destroy() {
|
function _destroy() {
|
||||||
// Bail if the element is not initialized
|
// Bail if the element is not initialized
|
||||||
if(!player.init) {
|
if(!player.init) {
|
||||||
@ -1582,12 +1786,18 @@
|
|||||||
// Reset container classname
|
// Reset container classname
|
||||||
player.container.setAttribute("class", config.selectors.container.replace(".", ""));
|
player.container.setAttribute("class", config.selectors.container.replace(".", ""));
|
||||||
|
|
||||||
// Event listeners are removed when elements are removed
|
// Remove init flag
|
||||||
// http://stackoverflow.com/questions/12528049/if-a-dom-element-is-removed-are-its-listeners-also-removed-from-memory
|
player.init = false;
|
||||||
|
|
||||||
// Remove controls
|
// Remove controls
|
||||||
_remove(_getElement(config.selectors.controls));
|
_remove(_getElement(config.selectors.controls));
|
||||||
|
|
||||||
|
// YouTube
|
||||||
|
if(player.type === "youtube") {
|
||||||
|
player.embed.destroy();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
// If video, we need to remove some more
|
// If video, we need to remove some more
|
||||||
if(player.type === "video") {
|
if(player.type === "video") {
|
||||||
// Remove captions
|
// Remove captions
|
||||||
@ -1604,9 +1814,6 @@
|
|||||||
// http://stackoverflow.com/questions/19469881/javascript-remove-all-event-listeners-of-specific-type
|
// http://stackoverflow.com/questions/19469881/javascript-remove-all-event-listeners-of-specific-type
|
||||||
var clone = player.media.cloneNode(true);
|
var clone = player.media.cloneNode(true);
|
||||||
player.media.parentNode.replaceChild(clone, player.media);
|
player.media.parentNode.replaceChild(clone, player.media);
|
||||||
|
|
||||||
// Remove init flag
|
|
||||||
player.init = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Setup a player
|
// Setup a player
|
||||||
@ -1623,10 +1830,19 @@
|
|||||||
player.browser = _browserSniff();
|
player.browser = _browserSniff();
|
||||||
|
|
||||||
// Get the media element
|
// Get the media element
|
||||||
player.media = player.container.querySelectorAll("audio, video")[0];
|
player.media = player.container.querySelectorAll("audio, video, div")[0];
|
||||||
|
|
||||||
// Set media type
|
// Set media type
|
||||||
player.type = player.media.tagName.toLowerCase();
|
var tagName = player.media.tagName.toLowerCase();
|
||||||
|
switch(tagName) {
|
||||||
|
case "div":
|
||||||
|
player.type = player.media.getAttribute("data-type");
|
||||||
|
break;
|
||||||
|
|
||||||
|
default:
|
||||||
|
player.type = tagName;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
// Check for full support
|
// Check for full support
|
||||||
player.supported = api.supported(player.type);
|
player.supported = api.supported(player.type);
|
||||||
@ -1642,16 +1858,16 @@
|
|||||||
// Setup media
|
// Setup media
|
||||||
_setupMedia();
|
_setupMedia();
|
||||||
|
|
||||||
// If there's full support
|
// Setup interface
|
||||||
if(player.supported.full) {
|
if(player.type == "video" || player.type == "audio") {
|
||||||
// Inject custom controls
|
// Bail if no support
|
||||||
_injectControls();
|
if(!player.supported.full) {
|
||||||
|
return;
|
||||||
// Find the elements
|
|
||||||
if(!_findElements()) {
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Setup UI
|
||||||
|
_setupInterface();
|
||||||
|
|
||||||
// Display duration if available
|
// Display duration if available
|
||||||
if(config.displayDuration) {
|
if(config.displayDuration) {
|
||||||
_displayDuration();
|
_displayDuration();
|
||||||
@ -1659,6 +1875,20 @@
|
|||||||
|
|
||||||
// Set up aria-label for Play button with the title option
|
// Set up aria-label for Play button with the title option
|
||||||
_setupAria();
|
_setupAria();
|
||||||
|
}
|
||||||
|
|
||||||
|
// Successful setup
|
||||||
|
player.init = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
function _setupInterface() {
|
||||||
|
// Inject custom controls
|
||||||
|
_injectControls();
|
||||||
|
|
||||||
|
// Find the elements
|
||||||
|
if(!_findElements()) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
// Captions
|
// Captions
|
||||||
_setupCaptions();
|
_setupCaptions();
|
||||||
@ -1674,10 +1904,6 @@
|
|||||||
_listeners();
|
_listeners();
|
||||||
}
|
}
|
||||||
|
|
||||||
// Successful setup
|
|
||||||
player.init = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Initialize instance
|
// Initialize instance
|
||||||
_init();
|
_init();
|
||||||
|
|
||||||
@ -1728,6 +1954,11 @@
|
|||||||
full = (basic && !oldIE);
|
full = (basic && !oldIE);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case "youtube":
|
||||||
|
basic = true;
|
||||||
|
full = !oldIE;
|
||||||
|
break;
|
||||||
|
|
||||||
default:
|
default:
|
||||||
basic = (audio && video);
|
basic = (audio && video);
|
||||||
full = (basic && !oldIE);
|
full = (basic && !oldIE);
|
||||||
|
@ -1,15 +1,18 @@
|
|||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
// HTML5 Media Player
|
// Plyr styles
|
||||||
|
// https://github.com/selz/plyr
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
|
|
||||||
// Variables
|
// Variables
|
||||||
// -------------------------------
|
// -------------------------------
|
||||||
|
|
||||||
// Colors
|
// Colors
|
||||||
@blue: #3498DB;
|
@blue: #3498DB;
|
||||||
@gray-dark: #343f4a;
|
@gray-dark: #343F4A;
|
||||||
@gray: #565d64;
|
@gray: #565D64;
|
||||||
@gray-light: #cbd0d3;
|
@gray-light: #6B7D86;
|
||||||
@off-white: #d6dadd;
|
@gray-lighter: #CBD0D3;
|
||||||
|
@off-white: #D6DADD;
|
||||||
|
|
||||||
// Font sizes
|
// Font sizes
|
||||||
@font-size-small: 14px;
|
@font-size-small: 14px;
|
||||||
@ -18,11 +21,10 @@
|
|||||||
|
|
||||||
// Controls
|
// Controls
|
||||||
@control-spacing: 10px;
|
@control-spacing: 10px;
|
||||||
@controls-bg: @gray-dark;
|
@controls-bg: #fff;
|
||||||
@control-bg-hover: @blue;
|
@control-bg-hover: @blue;
|
||||||
@control-color: @gray-light;
|
.contrast-control-color(@controls-bg);
|
||||||
@control-color-inactive: @gray;
|
.contrast-control-color-hover(@control-bg-hover);
|
||||||
@control-color-hover: #fff;
|
|
||||||
|
|
||||||
// Tooltips
|
// Tooltips
|
||||||
@tooltip-bg: @controls-bg;
|
@tooltip-bg: @controls-bg;
|
||||||
@ -40,7 +42,7 @@
|
|||||||
|
|
||||||
// Volume
|
// Volume
|
||||||
@volume-track-height: 6px;
|
@volume-track-height: 6px;
|
||||||
@volume-track-bg: @gray;
|
@volume-track-bg: darken(@controls-bg, 10%);
|
||||||
@volume-thumb-height: (@volume-track-height * 2);
|
@volume-thumb-height: (@volume-track-height * 2);
|
||||||
@volume-thumb-width: (@volume-track-height * 2);
|
@volume-thumb-width: (@volume-track-height * 2);
|
||||||
@volume-thumb-bg: @control-color;
|
@volume-thumb-bg: @control-color;
|
||||||
@ -50,18 +52,40 @@
|
|||||||
@bp-control-split: 560px; // When controls split into left/right
|
@bp-control-split: 560px; // When controls split into left/right
|
||||||
@bp-captions-large: 768px; // When captions jump to the larger font size
|
@bp-captions-large: 768px; // When captions jump to the larger font size
|
||||||
|
|
||||||
// Utility classes & mixins
|
// Animation
|
||||||
// -------------------------------
|
// ---------------------------------------
|
||||||
// Screen reader only
|
|
||||||
.sr-only {
|
@keyframes progress {
|
||||||
position: absolute !important;
|
to { background-position: @progress-loading-size 0; }
|
||||||
clip: rect(1px, 1px, 1px, 1px);
|
|
||||||
padding: 0 !important;
|
|
||||||
border: 0 !important;
|
|
||||||
height: 1px !important;
|
|
||||||
width: 1px !important;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mixins
|
||||||
|
// -------------------------------
|
||||||
|
|
||||||
|
// Contrast
|
||||||
|
.contrast-control-color(@color: "") when (lightness(@color) >= 65%) {
|
||||||
|
@control-color: @gray-light;
|
||||||
|
}
|
||||||
|
.contrast-control-color(@color: "") when (lightness(@color) < 65%) {
|
||||||
|
@control-color: @gray-lighter;
|
||||||
|
}
|
||||||
|
.contrast-control-color-hover(@color: "") when (lightness(@color) >= 65%) {
|
||||||
|
@control-color-hover: @gray;
|
||||||
|
}
|
||||||
|
.contrast-control-color-hover(@color: "") when (lightness(@color) < 65%) {
|
||||||
|
@control-color-hover: #fff;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Font smoothing
|
||||||
|
.font-smoothing(@mode: on) when (@mode = on) {
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
||||||
|
.font-smoothing(@mode: on) when (@mode = off) {
|
||||||
|
-moz-osx-font-smoothing: auto;
|
||||||
|
-webkit-font-smoothing: subpixel-antialiased;
|
||||||
|
}
|
||||||
|
|
||||||
// Contain floats: nicolasgallagher.com/micro-clearfix-hack/
|
// Contain floats: nicolasgallagher.com/micro-clearfix-hack/
|
||||||
.clearfix() {
|
.clearfix() {
|
||||||
zoom: 1;
|
zoom: 1;
|
||||||
@ -75,14 +99,7 @@
|
|||||||
outline-offset: 0;
|
outline-offset: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Animation
|
|
||||||
// ---------------------------------------
|
|
||||||
@keyframes progress {
|
|
||||||
to { background-position: @progress-loading-size 0; }
|
|
||||||
}
|
|
||||||
|
|
||||||
// <input type="range"> styling
|
// <input type="range"> styling
|
||||||
// ---------------------------------------
|
|
||||||
.volume-thumb() {
|
.volume-thumb() {
|
||||||
height: @volume-thumb-height;
|
height: @volume-thumb-height;
|
||||||
width: @volume-thumb-width;
|
width: @volume-thumb-width;
|
||||||
@ -109,15 +126,16 @@
|
|||||||
border: 0;
|
border: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Font smoothing
|
// Screen reader only
|
||||||
// ---------------------------------------
|
// -------------------------------
|
||||||
.font-smoothing(@mode: on) when (@mode = on) {
|
.sr-only {
|
||||||
-moz-osx-font-smoothing: grayscale;
|
position: absolute !important;
|
||||||
-webkit-font-smoothing: antialiased;
|
clip: rect(1px, 1px, 1px, 1px);
|
||||||
}
|
padding: 0 !important;
|
||||||
.font-smoothing(@mode: on) when (@mode = off) {
|
border: 0 !important;
|
||||||
-moz-osx-font-smoothing: auto;
|
height: 1px !important;
|
||||||
-webkit-font-smoothing: subpixel-antialiased;
|
width: 1px !important;
|
||||||
|
overflow: hidden;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Styles
|
// Styles
|
||||||
@ -141,12 +159,28 @@
|
|||||||
&-video-wrapper {
|
&-video-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
video {
|
video,
|
||||||
|
audio {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: auto;
|
height: auto;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// For embeds
|
||||||
|
&-video-embed {
|
||||||
|
padding-bottom: 56.25%; /* 16:9 */
|
||||||
|
height: 0;
|
||||||
|
|
||||||
|
iframe {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Captions
|
// Captions
|
||||||
&-captions {
|
&-captions {
|
||||||
display: none;
|
display: none;
|
||||||
@ -184,6 +218,7 @@
|
|||||||
background: @controls-bg;
|
background: @controls-bg;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
box-shadow: 0 1px 1px rgba(red(@gray-dark), green(@gray-dark), blue(@gray-dark), .2);
|
||||||
|
|
||||||
// Layout
|
// Layout
|
||||||
&-right {
|
&-right {
|
||||||
@ -207,7 +242,7 @@
|
|||||||
margin: 0 2px;
|
margin: 0 2px;
|
||||||
padding: (@control-spacing / 2) @control-spacing;
|
padding: (@control-spacing / 2) @control-spacing;
|
||||||
|
|
||||||
transition: background .3s ease;
|
transition: background .3s ease, color .3s ease, opacity .3s ease;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
||||||
@ -221,12 +256,13 @@
|
|||||||
}
|
}
|
||||||
input + label,
|
input + label,
|
||||||
.inverted:checked + label {
|
.inverted:checked + label {
|
||||||
color: @control-color-inactive;
|
opacity: .5;
|
||||||
}
|
}
|
||||||
button,
|
button,
|
||||||
.inverted + label,
|
.inverted + label,
|
||||||
input:checked + label {
|
input:checked + label {
|
||||||
color: @control-color;
|
color: @control-color;
|
||||||
|
opacity: 1;
|
||||||
}
|
}
|
||||||
button {
|
button {
|
||||||
border: 0;
|
border: 0;
|
||||||
@ -241,6 +277,7 @@
|
|||||||
[type="checkbox"] + label:hover {
|
[type="checkbox"] + label:hover {
|
||||||
background: @control-bg-hover;
|
background: @control-bg-hover;
|
||||||
color: @control-color-hover;
|
color: @control-color-hover;
|
||||||
|
opacity: 1;
|
||||||
}
|
}
|
||||||
button:focus,
|
button:focus,
|
||||||
input:focus + label {
|
input:focus + label {
|
||||||
@ -273,7 +310,6 @@
|
|||||||
&::before {
|
&::before {
|
||||||
content: "\2044";
|
content: "\2044";
|
||||||
margin-right: @control-spacing;
|
margin-right: @control-spacing;
|
||||||
color: darken(@control-color, 30%);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,15 +1,18 @@
|
|||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
// HTML5 Media Player
|
// Plyr styles
|
||||||
|
// https://github.com/selz/plyr
|
||||||
// ==========================================================================
|
// ==========================================================================
|
||||||
|
|
||||||
// Variables
|
// Variables
|
||||||
// -------------------------------
|
// -------------------------------
|
||||||
|
|
||||||
// Colors
|
// Colors
|
||||||
$blue: #3498DB !default;
|
$blue: #3498DB !default;
|
||||||
$gray-dark: #343f4a !default;
|
$gray-dark: #343F4A !default;
|
||||||
$gray: #565d64 !default;
|
$gray: #565D64 !default;
|
||||||
$gray-light: #cbd0d3 !default;
|
$gray-light: #6B7D86 !default;
|
||||||
$off-white: #d6dadd !default;
|
$gray-lighter: #CBD0D3 !default;
|
||||||
|
$off-white: #D6DADD !default;
|
||||||
|
|
||||||
// Font sizes
|
// Font sizes
|
||||||
$font-size-small: 14px !default;
|
$font-size-small: 14px !default;
|
||||||
@ -18,11 +21,10 @@ $font-size-large: ceil(($font-size-base * 1.5)) !default;
|
|||||||
|
|
||||||
// Controls
|
// Controls
|
||||||
$control-spacing: 10px !default;
|
$control-spacing: 10px !default;
|
||||||
$controls-bg: $gray-dark !default;
|
$controls-bg: #fff !default;
|
||||||
$control-bg-hover: $blue !default;
|
$control-bg-hover: @blue !default;
|
||||||
$control-color: $gray-light !default;
|
.contrast-control-color($controls-bg);
|
||||||
$control-color-inactive: $gray !default;
|
.contrast-control-color-hover($control-bg-hover);
|
||||||
$control-color-hover: #fff !default;
|
|
||||||
|
|
||||||
// Tooltips
|
// Tooltips
|
||||||
$tooltip-bg: $controls-bg !default;
|
$tooltip-bg: $controls-bg !default;
|
||||||
@ -40,7 +42,7 @@ $progress-loading-bg: rgba(0,0,0, .15) !default;
|
|||||||
|
|
||||||
// Volume
|
// Volume
|
||||||
$volume-track-height: 6px !default;
|
$volume-track-height: 6px !default;
|
||||||
$volume-track-bg: $gray !default;
|
$volume-track-bg: darken($controls-bg, 10%) !default;
|
||||||
$volume-thumb-height: ($volume-track-height * 2) !default;
|
$volume-thumb-height: ($volume-track-height * 2) !default;
|
||||||
$volume-thumb-width: ($volume-track-height * 2) !default;
|
$volume-thumb-width: ($volume-track-height * 2) !default;
|
||||||
$volume-thumb-bg: $control-color !default;
|
$volume-thumb-bg: $control-color !default;
|
||||||
@ -50,18 +52,47 @@ $volume-thumb-bg-focus: $control-bg-hover !default;
|
|||||||
$bp-control-split: 560px !default; // When controls split into left/right
|
$bp-control-split: 560px !default; // When controls split into left/right
|
||||||
$bp-captions-large: 768px !default; // When captions jump to the larger font size
|
$bp-captions-large: 768px !default; // When captions jump to the larger font size
|
||||||
|
|
||||||
// Utility classes & mixins
|
// Animation
|
||||||
// -------------------------------
|
// ---------------------------------------
|
||||||
// Screen reader only
|
|
||||||
.sr-only {
|
@keyframes progress {
|
||||||
position: absolute !important;
|
to { background-position: $progress-loading-size 0; }
|
||||||
clip: rect(1px, 1px, 1px, 1px);
|
|
||||||
padding: 0 !important;
|
|
||||||
border: 0 !important;
|
|
||||||
height: 1px !important;
|
|
||||||
width: 1px !important;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Mixins
|
||||||
|
// -------------------------------
|
||||||
|
|
||||||
|
// Contrast
|
||||||
|
@mixin contrast-control-color($color: "") {
|
||||||
|
@if (lightness($color) >= 65%) {
|
||||||
|
$control-color: $gray-light;
|
||||||
|
}
|
||||||
|
@else if(lightness(@color) < 65%) {
|
||||||
|
$control-color: $gray-lighter;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@mixin contrast-control-color-hover($color: "") {
|
||||||
|
@if (lightness($color) >= 65%) {
|
||||||
|
$control-color-hover: $gray;
|
||||||
|
}
|
||||||
|
@else if (lightness($color) < 65%) {
|
||||||
|
$control-color-hover: #fff;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Font smoothing
|
||||||
|
@mixin font-smoothing($mode: on)
|
||||||
|
{
|
||||||
|
@if ($mode == 'on') {
|
||||||
|
-moz-osx-font-smoothing: grayscale;
|
||||||
|
-webkit-font-smoothing: antialiased;
|
||||||
|
}
|
||||||
|
@else if ($mode == 'off') {
|
||||||
|
-moz-osx-font-smoothing: auto;
|
||||||
|
-webkit-font-smoothing: subpixel-antialiased;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Contain floats: nicolasgallagher.com/micro-clearfix-hack/
|
// Contain floats: nicolasgallagher.com/micro-clearfix-hack/
|
||||||
@mixin clearfix()
|
@mixin clearfix()
|
||||||
{
|
{
|
||||||
@ -77,14 +108,7 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
outline-offset: 0;
|
outline-offset: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Animation
|
// Range mixins
|
||||||
// ---------------------------------------
|
|
||||||
@keyframes progress {
|
|
||||||
to { background-position: $progress-loading-size 0; }
|
|
||||||
}
|
|
||||||
|
|
||||||
// <input type="range"> styling
|
|
||||||
// ---------------------------------------
|
|
||||||
@mixin volume-thumb()
|
@mixin volume-thumb()
|
||||||
{
|
{
|
||||||
height: $volume-thumb-height;
|
height: $volume-thumb-height;
|
||||||
@ -115,17 +139,16 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
border: 0;
|
border: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Font smoothing
|
// Screen reader only
|
||||||
// ---------------------------------------
|
// -------------------------------
|
||||||
@mixin font-smoothing($mode: on)
|
.sr-only {
|
||||||
{
|
position: absolute !important;
|
||||||
@if $mode == 'on' {
|
clip: rect(1px, 1px, 1px, 1px);
|
||||||
-moz-osx-font-smoothing: grayscale;
|
padding: 0 !important;
|
||||||
-webkit-font-smoothing: antialiased;
|
border: 0 !important;
|
||||||
} @else if $mode == 'off' {
|
height: 1px !important;
|
||||||
-moz-osx-font-smoothing: auto;
|
width: 1px !important;
|
||||||
-webkit-font-smoothing: subpixel-antialiased;
|
overflow: hidden;
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Styles
|
// Styles
|
||||||
@ -149,7 +172,8 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
&-video-wrapper {
|
&-video-wrapper {
|
||||||
position: relative;
|
position: relative;
|
||||||
}
|
}
|
||||||
video {
|
video,
|
||||||
|
audio {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: auto;
|
height: auto;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
@ -192,6 +216,7 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
background: $controls-bg;
|
background: $controls-bg;
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
text-align: center;
|
text-align: center;
|
||||||
|
box-shadow: 0 1px 1px rgba(red($gray-dark), green($gray-dark), blue($gray-dark), .2);
|
||||||
|
|
||||||
// Layout
|
// Layout
|
||||||
&-right {
|
&-right {
|
||||||
@ -215,7 +240,7 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
margin: 0 2px;
|
margin: 0 2px;
|
||||||
padding: ($control-spacing / 2) $control-spacing;
|
padding: ($control-spacing / 2) $control-spacing;
|
||||||
|
|
||||||
transition: background .3s ease;
|
background .3s ease, color .3s ease, opacity .3s ease;
|
||||||
border-radius: 3px;
|
border-radius: 3px;
|
||||||
cursor: pointer;
|
cursor: pointer;
|
||||||
|
|
||||||
@ -229,12 +254,13 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
}
|
}
|
||||||
input + label,
|
input + label,
|
||||||
.inverted:checked + label {
|
.inverted:checked + label {
|
||||||
color: $control-color-inactive;
|
opacity: .5;
|
||||||
}
|
}
|
||||||
button,
|
button,
|
||||||
.inverted + label,
|
.inverted + label,
|
||||||
input:checked + label {
|
input:checked + label {
|
||||||
color: $control-color;
|
color: $control-color;
|
||||||
|
opacity: 1;
|
||||||
}
|
}
|
||||||
button {
|
button {
|
||||||
border: 0;
|
border: 0;
|
||||||
@ -249,6 +275,7 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
[type="checkbox"] + label:hover {
|
[type="checkbox"] + label:hover {
|
||||||
background: $control-bg-hover;
|
background: $control-bg-hover;
|
||||||
color: $control-color-hover;
|
color: $control-color-hover;
|
||||||
|
opacity: 1;
|
||||||
}
|
}
|
||||||
button:focus,
|
button:focus,
|
||||||
input:focus + label {
|
input:focus + label {
|
||||||
@ -281,7 +308,6 @@ $bp-captions-large: 768px !default; // When captions jump to the larger fo
|
|||||||
&::before {
|
&::before {
|
||||||
content: "\2044";
|
content: "\2044";
|
||||||
margin-right: $control-spacing;
|
margin-right: $control-spacing;
|
||||||
color: darken($control-color, 30%);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Reference in New Issue
Block a user