diff options
author | Sam Potts <me@sampotts.me> | 2018-01-12 19:35:46 +1100 |
---|---|---|
committer | Sam Potts <me@sampotts.me> | 2018-01-12 19:35:46 +1100 |
commit | d9ec1d1b8e251cf30509e88a76132c0e04f8c00d (patch) | |
tree | 7b342f44dce8855680776b904678ce03c03a11ce | |
parent | 2e5d06ad8529f99c607e3f2e4c678f07e6973274 (diff) | |
download | plyr-d9ec1d1b8e251cf30509e88a76132c0e04f8c00d.tar.lz plyr-d9ec1d1b8e251cf30509e88a76132c0e04f8c00d.tar.xz plyr-d9ec1d1b8e251cf30509e88a76132c0e04f8c00d.zip |
Progressively enhance <iframe> embeds
-rw-r--r-- | changelog.md | 29 | ||||
-rw-r--r-- | demo/dist/demo.css | 2 | ||||
-rw-r--r-- | dist/plyr.css | 2 | ||||
-rw-r--r-- | dist/plyr.js | 2 | ||||
-rw-r--r-- | dist/plyr.js.map | 2 | ||||
-rw-r--r-- | package.json | 14 | ||||
-rw-r--r-- | readme.md | 21 | ||||
-rw-r--r-- | src/js/console.js | 4 | ||||
-rw-r--r-- | src/js/media.js | 2 | ||||
-rw-r--r-- | src/js/plugins/vimeo.js | 19 | ||||
-rw-r--r-- | src/js/plugins/youtube.js | 40 | ||||
-rw-r--r-- | src/js/plyr.js | 94 | ||||
-rw-r--r-- | src/js/source.js | 7 | ||||
-rw-r--r-- | src/js/utils.js | 75 | ||||
-rw-r--r-- | src/sass/components/embed.scss | 1 |
15 files changed, 204 insertions, 110 deletions
diff --git a/changelog.md b/changelog.md index 7451288b..158d9683 100644 --- a/changelog.md +++ b/changelog.md @@ -13,6 +13,7 @@ This is a massive release. A _mostly_ complete rewrite in ES6. What started out * Added AirPlay support (again, Safari only) * Added `playsinline` support for iOS 10+ * Soundcloud removed until I can work on a plugin framework +* Embedded players are now progressively enhanced - no more empty `<div>`s! ### Other stuff @@ -68,27 +69,33 @@ You gotta break eggs to make an omelette. Sadly, there's quite a few breaking ch Because we're using the fancy new ES6 syntax, you will need to polyfill for vintage browsers if you want to use Plyr and still support them. Luckily there's a decent service for this that makes it painless, [https://polyfill.io](polyfill.io). ## v2.0.18 -- Fix for YouTube .getVideoData() issue (fixes #709) + +* Fix for YouTube .getVideoData() issue (fixes #709) ## v2.0.17 -- Vimeo controls fix (fixes #697) -- SVG4everybody compatibility fix -- Allow Plyr.setup event listeners to be set up as separate event listeners (https://github.com/sampotts/plyr/pull/703) -- Added title to the layer html template (for custom controls) (https://github.com/sampotts/plyr/pull/649) -- Target is null bug fix (https://github.com/sampotts/plyr/pull/617) -- fix #684 memory leaks issues after destroy (https://github.com/sampotts/plyr/pull/700) + +* Vimeo controls fix (fixes #697) +* SVG4everybody compatibility fix +* Allow Plyr.setup event listeners to be set up as separate event listeners (https://github.com/sampotts/plyr/pull/703) +* Added title to the layer html template (for custom controls) (https://github.com/sampotts/plyr/pull/649) +* Target is null bug fix (https://github.com/sampotts/plyr/pull/617) +* fix #684 memory leaks issues after destroy (https://github.com/sampotts/plyr/pull/700) ## v2.0.16 -- Fullscreen bug fix (fixes #664) + +* Fullscreen bug fix (fixes #664) ## v2.0.15 -- Demo fix + +* Demo fix ## v2.0.14 -- CDN URL updates. Sorry, still working on V3 as hard as I can... + +* CDN URL updates. Sorry, still working on V3 as hard as I can... ## v2.0.13 -- Repo moved and Vimeo demo fix + +* Repo moved and Vimeo demo fix ## v2.0.12 diff --git a/demo/dist/demo.css b/demo/dist/demo.css index a805d7f1..3e9a5831 100644 --- a/demo/dist/demo.css +++ b/demo/dist/demo.css @@ -1 +1 @@ -@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:300;src:url(https://cdn.plyr.io/static/fonts/gordita-light.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-light.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:400;src:url(https://cdn.plyr.io/static/fonts/gordita-regular.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-regular.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:500;src:url(https://cdn.plyr.io/static/fonts/gordita-medium.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-medium.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:600;src:url(https://cdn.plyr.io/static/fonts/gordita-bold.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-bold.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:900;src:url(https://cdn.plyr.io/static/fonts/gordita-black.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-black.woff) format("woff")}@-webkit-keyframes fadein{0%{opacity:0}100%{opacity:1}}@keyframes fadein{0%{opacity:0}100%{opacity:1}}/*! normalize.css v7.0.0 | MIT License | github.com/necolas/normalize.css */html{line-height:1.15;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,footer,header,nav,section{display:block}h1{font-size:2em;margin:.67em 0}figcaption,figure,main{display:block}figure{margin:1em 40px}hr{box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a,button.faux-link{background-color:transparent;-webkit-text-decoration-skip:objects}abbr[title]{border-bottom:none;text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}b,strong{font-weight:inherit}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}dfn{font-style:italic}mark{background-color:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}audio,video{display:inline-block}audio:not([controls]){display:none;height:0}img{border-style:none}svg:not(:root){overflow:hidden}button,input,optgroup,select,textarea{font-family:sans-serif;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=reset],[type=submit],button,html [type=button]{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:1px dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{display:inline-block;vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-cancel-button,[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}details,menu{display:block}summary{display:list-item}canvas{display:inline-block}template{display:none}[hidden]{display:none}*,::after,::before{box-sizing:border-box}body,html{display:-webkit-box;display:flex;width:100%}html{background:linear-gradient(to left top,#4dc1ff,#0074b3);background-attachment:fixed;height:100%}body{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;flex-direction:column;min-height:100%}.grid{-webkit-box-flex:1;flex:1;overflow:auto}main{margin:auto;text-align:center}aside{-webkit-box-align:center;align-items:center;background:#fff;color:#55646b;display:-webkit-box;display:flex;flex-shrink:0;-webkit-box-pack:center;justify-content:center;padding:15px;position:relative;text-align:center;text-shadow:none;width:100%}aside .icon{fill:#4baaf4;margin-right:10px}aside p{margin:0}aside a,aside button.faux-link{color:#4baaf4}aside a.tab-focus,aside button.tab-focus.faux-link{box-shadow:0 0 0 3px rgba(75,170,244,.35);outline:0}.grid{margin:0 auto;padding:20px}@media only screen and (min-width:768px){.grid{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;max-width:1280px;width:100%}.grid>*{-webkit-box-flex:1;flex:1}}html{font-size:100%}body{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:15px;font-size:.9375rem;color:#fff;font-family:Gordita,Avenir,"Helvetica Neue",sans-serif;font-weight:500;line-height:1.75;text-shadow:0 1px 1px rgba(0,0,0,.15)}button,input,select,textarea{font:inherit}p,small{margin:0 0 20px}small{font-size:13px;font-size:.8125rem;display:block}h1{font-size:64px;font-size:4rem;font-weight:600;letter-spacing:-.025em;margin:0 0 10px}.button,.button__count{-webkit-box-align:center;align-items:center;background:#fff;border:0;border-radius:4px;box-shadow:0 1px 1px rgba(0,0,0,.1);color:#55646b;display:-webkit-inline-box;display:inline-flex;padding:15px;position:relative;text-shadow:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;vertical-align:middle}.button{font-weight:600;padding-left:20px;padding-right:20px;transition:all .2s ease}.button:focus,.button:hover{color:#343f4a}.button:focus::after,.button:hover::after{display:none}.button:hover{box-shadow:0 2px 2px rgba(0,0,0,.1);-webkit-transform:translateY(-1px);transform:translateY(-1px)}.button:focus{outline:0}.button.tab-focus{box-shadow:0 0 0 3px rgba(255,255,255,.35);outline:0}.button:active{-webkit-transform:translateY(1px);transform:translateY(1px)}.button--with-count{display:-webkit-inline-box;display:inline-flex}.button--with-count .button .icon{flex-shrink:0}.button__count{-webkit-animation:fadein .2s ease;animation:fadein .2s ease;margin-left:10px}.button__count::before{border:5px solid transparent;border-left-width:0;border-right-color:#fff;content:'';height:0;position:absolute;right:100%;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);width:0}header{padding-bottom:20px;text-align:center}header .call-to-action{margin-top:30px}@media only screen and (min-width:768px){header{margin-right:60px;max-width:360px;padding-bottom:40px;text-align:left}}.icon{fill:currentColor;height:16px;vertical-align:-3px;width:16px}a svg,button svg,button.faux-link svg,label svg{pointer-events:none}.btn .icon,a .icon,button.faux-link .icon{margin-right:6px}button.faux-link{background:0 0;border:0;border-radius:0;cursor:pointer;font:inherit;line-height:1.75;margin:0;padding:0;position:relative;text-align:inherit;text-shadow:inherit;-moz-user-select:text;vertical-align:baseline;width:auto}a,button.faux-link{border-bottom:1px dotted currentColor;color:#fff;font-weight:600;position:relative;text-decoration:none;transition:all .2s ease}a::after,button.faux-link::after{background:currentColor;content:'';height:1px;left:50%;position:absolute;top:100%;-webkit-transform:translateX(-50%);transform:translateX(-50%);transition:width .2s ease;width:0}a:focus,a:hover,button.faux-link:focus,button.faux-link:hover{border-bottom-color:transparent;outline:0}a:focus::after,a:hover::after,button.faux-link:focus::after,button.faux-link:hover::after{width:100%}a.tab-focus,button.tab-focus.faux-link{box-shadow:0 0 0 3px rgba(255,255,255,.35);outline:0}li,ul{list-style:none;margin:0;padding:0}nav{display:-webkit-box;display:flex;-webkit-box-pack:center;justify-content:center;margin-bottom:20px}video{max-width:100%;vertical-align:middle}.plyr{border-radius:4px;box-shadow:0 2px 5px rgba(0,0,0,.2);margin:20px auto}.plyr.plyr--audio{max-width:480px}.plyr__video-wrapper::after{border:1px solid rgba(0,0,0,.15);border-radius:inherit;bottom:0;content:'';left:0;pointer-events:none;position:absolute;right:0;top:0}.plyr__cite{display:none;margin-top:20px}.plyr__cite .icon{margin-right:4px}.plyr--audio~ul .plyr__cite--audio,.plyr--video:not(.plyr--youtube):not(.plyr--vimeo)~ul .plyr__cite--video,.plyr--vimeo~ul .plyr__cite--vimeo,.plyr--youtube~ul .plyr__cite--youtube{display:block}@-webkit-keyframes plyr-progress{to{background-position:25px 0}}@keyframes plyr-progress{to{background-position:25px 0}}@-webkit-keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}@keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}.plyr{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;direction:ltr;font-family:inherit;font-weight:500;line-height:1.7;max-width:100%;min-width:200px;position:relative;text-shadow:none;transition:box-shadow .3s ease}.plyr audio,.plyr video{border-radius:inherit;height:auto;vertical-align:middle;width:100%}.plyr:focus{outline:0}.plyr--full-ui{box-sizing:border-box}.plyr--full-ui *,.plyr--full-ui ::after,.plyr--full-ui ::before{box-sizing:inherit}.plyr--full-ui a,.plyr--full-ui button,.plyr--full-ui button.faux-link,.plyr--full-ui input,.plyr--full-ui label{touch-action:manipulation}.plyr__badge{background:#4f5b5f;border-radius:2px;color:rgba(255,255,255,.9);font-size:9px;line-height:1;padding:3px 4px}.plyr--full-ui ::-webkit-media-text-track-container{display:none}.plyr__captions{-webkit-animation:plyr-fade-in .3s ease;animation:plyr-fade-in .3s ease;bottom:0;color:#fff;display:none;font-size:12px;left:0;padding:10px;position:absolute;text-align:center;-webkit-transform:translateY(-40px);transform:translateY(-40px);transition:-webkit-transform .4s ease-in-out;transition:transform .4s ease-in-out;transition:transform .4s ease-in-out,-webkit-transform .4s ease-in-out;width:100%}.plyr__captions span{background:rgba(0,0,0,.8);border-radius:2px;-webkit-box-decoration-break:clone;box-decoration-break:clone;line-height:185%;padding:.2em .5em;white-space:pre-wrap}.plyr__captions span div{display:inline}.plyr__captions span:empty{display:none}@media (min-width:480px){.plyr__captions{font-size:13px;padding:20px}}@media (min-width:768px){.plyr__captions{font-size:18px}}.plyr--captions-active .plyr__captions{display:block}.plyr--hide-controls .plyr__captions{-webkit-transform:translateY(-15px);transform:translateY(-15px)}.plyr__control{background:0 0;border:0;border-radius:3px;color:inherit;cursor:pointer;flex-shrink:0;overflow:visible;padding:7px;position:relative;transition:all .3s ease}.plyr__control svg{display:block;fill:currentColor;height:18px;pointer-events:none;width:18px}.plyr__control:focus{outline:0}.plyr__control.plyr__tab-focus{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__control[aria-pressed=false] .icon--pressed,.plyr__control[aria-pressed=false] .label--pressed,.plyr__control[aria-pressed=true] .icon--not-pressed,.plyr__control[aria-pressed=true] .label--not-pressed{display:none}.plyr--audio .plyr__control.plyr__tab-focus,.plyr--audio .plyr__control:hover,.plyr--audio .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr__control--overlaid{background:rgba(26,175,255,.8);border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15);color:#fff;display:none;left:50%;padding:15px;position:absolute;top:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);z-index:2}.plyr__control--overlaid svg{height:20px;left:2px;position:relative;width:20px}.plyr__control--overlaid:focus,.plyr__control--overlaid:hover{background:#1aafff}.plyr--playing .plyr__control--overlaid{opacity:0;visibility:hidden}.plyr--full-ui.plyr--video .plyr__control--overlaid{display:block}.plyr--full-ui ::-webkit-media-controls{display:none}.plyr__controls{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;text-align:center}.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:5px}.plyr__controls .plyr__menu:first-child,.plyr__controls .plyr__menu:first-child+[data-plyr=pause],.plyr__controls .plyr__progress:first-child,.plyr__controls .plyr__progress:first-child+[data-plyr=pause],.plyr__controls .plyr__time:first-child,.plyr__controls .plyr__time:first-child+[data-plyr=pause],.plyr__controls>.plyr__control:first-child,.plyr__controls>.plyr__control:first-child+[data-plyr=pause]{margin-left:0}.plyr__controls .plyr__volume{margin-left:5px}@media (min-width:480px){.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:10px}.plyr__controls .plyr__menu+.plyr__control,.plyr__controls>.plyr__control+.plyr__control,.plyr__controls>.plyr__control+.plyr__menu{margin-left:5px}}.plyr--video .plyr__controls{background:linear-gradient(transparent,rgba(0,0,0,.7));border-bottom-left-radius:inherit;border-bottom-right-radius:inherit;bottom:0;color:#fff;left:0;padding:35px 10px 10px;position:absolute;right:0;transition:opacity .4s ease-in-out,-webkit-transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out,-webkit-transform .4s ease-in-out;z-index:2}.plyr--video .plyr__controls .plyr__control svg{-webkit-filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15));filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15))}.plyr--video .plyr__controls .plyr__control.plyr__tab-focus,.plyr--video .plyr__controls .plyr__control:hover,.plyr--video .plyr__controls .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr--audio .plyr__controls{background:#fff;border-radius:inherit;color:#4f5b5f;padding:10px}.plyr--video.plyr--hide-controls .plyr__controls{opacity:0;pointer-events:none;-webkit-transform:translateY(100%);transform:translateY(100%)}.plyr [data-plyr=airplay],.plyr [data-plyr=captions],.plyr [data-plyr=fullscreen],.plyr [data-plyr=pip]{display:none}.plyr--airplay-supported [data-plyr=airplay],.plyr--captions-enabled [data-plyr=captions],.plyr--fullscreen-enabled [data-plyr=fullscreen],.plyr--pip-supported [data-plyr=pip]{display:inline-block}.plyr__video-embed{height:0;padding-bottom:56.25%}.plyr__video-embed iframe{border:0;height:100%;left:0;position:absolute;top:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__video-embed>div{padding-bottom:200%;position:relative;-webkit-transform:translateY(-35.9375%);transform:translateY(-35.9375%)}.plyr--full-ui .plyr__video-embed iframe{pointer-events:none}.plyr__menu{display:-webkit-box;display:flex;position:relative}.plyr__menu .plyr__control svg{transition:-webkit-transform .3s ease;transition:transform .3s ease;transition:transform .3s ease,-webkit-transform .3s ease}.plyr__menu .plyr__control[aria-expanded=true] svg{-webkit-transform:rotate(90deg);transform:rotate(90deg)}.plyr__menu .plyr__control[aria-expanded=true] .plyr__tooltip{display:none}.plyr__menu__container{-webkit-animation:plyr-popup .2s ease;animation:plyr-popup .2s ease;background:rgba(255,255,255,.9);border-radius:4px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:13px;margin-bottom:10px;position:absolute;right:-3px;text-align:left;white-space:nowrap;z-index:1}.plyr__menu__container>div{overflow:hidden;transition:height .35s cubic-bezier(.4,0,.2,1),width .35s cubic-bezier(.4,0,.2,1)}.plyr__menu__container::after{border:4px solid transparent;border-top-color:rgba(255,255,255,.9);content:'';height:0;position:absolute;right:15px;top:100%;width:0}.plyr__menu__container ul{list-style:none;margin:0;overflow:hidden;padding:7px}.plyr__menu__container .plyr__control{-webkit-box-align:center;align-items:center;color:#4f5b5f;display:-webkit-box;display:flex;padding:4px 14px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__menu__container .plyr__control::after{border:4px solid transparent;content:'';position:absolute;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);transition:border-color .2s ease}.plyr__menu__container .plyr__control--forward{padding-right:28px}.plyr__menu__container .plyr__control--forward::after{border-left-color:rgba(79,91,95,.8);right:5px}.plyr__menu__container .plyr__control--forward.plyr__tab-focus::after,.plyr__menu__container .plyr__control--forward:hover::after{border-left-color:currentColor}.plyr__menu__container .plyr__control--back{font-weight:500;margin:7px;margin-bottom:3px;padding-left:28px;position:relative;width:calc(100% - 14px)}.plyr__menu__container .plyr__control--back::after{border-right-color:rgba(79,91,95,.8);left:7px}.plyr__menu__container .plyr__control--back::before{background:#b7c5cd;box-shadow:0 1px 0 #fff;content:'';height:1px;left:0;margin-top:4px;overflow:hidden;position:absolute;right:0;top:100%}.plyr__menu__container .plyr__control--back.plyr__tab-focus::after,.plyr__menu__container .plyr__control--back:hover::after{border-right-color:currentColor}.plyr__menu__container label.plyr__control{padding-left:7px}.plyr__menu__container label.plyr__control input[type=radio]+span{background:rgba(0,0,0,.1);border-radius:100%;display:block;flex-shrink:0;height:16px;margin-right:10px;position:relative;transition:all .3s ease;width:16px}.plyr__menu__container label.plyr__control input[type=radio]+span::after{background:#fff;border-radius:100%;content:'';height:6px;left:5px;opacity:0;position:absolute;top:5px;-webkit-transform:scale(0);transform:scale(0);transition:opacity .3s ease,-webkit-transform .3s ease;transition:transform .3s ease,opacity .3s ease;transition:transform .3s ease,opacity .3s ease,-webkit-transform .3s ease;width:6px}.plyr__menu__container label.plyr__control input[type=radio]:checked+span{background:#1aafff}.plyr__menu__container label.plyr__control input[type=radio]:checked+span::after{opacity:1;-webkit-transform:scale(1);transform:scale(1)}.plyr__menu__container label.plyr__control input[type=radio]:focus+span{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__menu__container label.plyr__control.plyr__tab-focus input[type=radio]+span,.plyr__menu__container label.plyr__control:hover input[type=radio]+span{background:rgba(0,0,0,.1)}.plyr__menu__container .plyr__menu__value{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;margin-left:auto;margin-right:-7px;overflow:hidden;padding-left:25px;pointer-events:none}.plyr__progress{display:-webkit-box;display:flex;-webkit-box-flex:1;flex:1;position:relative}.plyr__progress input[type=range]{position:relative;z-index:2}.plyr__progress .plyr__tooltip{font-size:11px;left:0}.plyr__progress--buffer{-webkit-appearance:none;background:0 0;border:0;border-radius:100px;height:6px;left:0;margin:-3px 0 0;padding:0;position:absolute;top:50%;width:100%}.plyr__progress--buffer::-webkit-progress-bar{background:0 0;transition:width .2s ease}.plyr__progress--buffer::-webkit-progress-value{background:currentColor;border-radius:100px;min-width:6px}.plyr__progress--buffer::-moz-progress-bar{background:currentColor;border-radius:100px;min-width:6px;transition:width .2s ease}.plyr__progress--buffer::-ms-fill{border-radius:100px;transition:width .2s ease}.plyr--video .plyr__progress--buffer{box-shadow:0 1px 1px rgba(0,0,0,.15);color:rgba(255,255,255,.25)}.plyr--audio .plyr__progress--buffer{color:rgba(183,197,205,.66)}.plyr--loading .plyr__progress--buffer{-webkit-animation:plyr-progress 1s linear infinite;animation:plyr-progress 1s linear infinite;background-image:linear-gradient(-45deg,rgba(47,52,61,.6) 25%,transparent 25%,transparent 50%,rgba(47,52,61,.6) 50%,rgba(47,52,61,.6) 75%,transparent 75%,transparent);background-repeat:repeat-x;background-size:25px 25px;color:transparent}.plyr--video.plyr--loading .plyr__progress--buffer{background-color:rgba(255,255,255,.25)}.plyr--audio.plyr--loading .plyr__progress--buffer{background-color:rgba(183,197,205,.66)}.plyr--full-ui input[type=range]{-webkit-appearance:none;background:0 0;border:0;border-radius:28px;color:#1aafff;display:block;height:20px;margin:0;padding:0;transition:box-shadow .3s ease;width:100%}.plyr--full-ui input[type=range]::-webkit-slider-runnable-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-webkit-user-select:none;user-select:none;background-image:linear-gradient(to right,currentColor var(--value),transparent var(--value))}.plyr--full-ui input[type=range]::-webkit-slider-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;-webkit-appearance:none;margin-top:-4px}.plyr--full-ui input[type=range]::-moz-range-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-moz-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-moz-range-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px}.plyr--full-ui input[type=range]::-moz-range-progress{background:currentColor;border-radius:3px;height:6px}.plyr--full-ui input[type=range]::-ms-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;color:transparent}.plyr--full-ui input[type=range]::-ms-fill-upper{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-ms-fill-lower{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;background:currentColor}.plyr--full-ui input[type=range]::-ms-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;margin-top:0}.plyr--full-ui input[type=range]::-ms-tooltip{display:none}.plyr--full-ui input[type=range]:focus{outline:0}.plyr--full-ui input[type=range]::-moz-focus-outer{border:0}.plyr--full-ui input[type=range].plyr__tab-focus::-webkit-slider-runnable-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-moz-range-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-ms-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range]:active::-webkit-slider-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-moz-range-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-ms-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui.plyr--video input[type=range]::-webkit-slider-runnable-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-moz-range-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-ms-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--audio input[type=range]::-webkit-slider-runnable-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-moz-range-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-ms-track{background-color:rgba(183,197,205,.66)}.plyr__time{font-size:11px}.plyr__time+.plyr__time::before{content:'\2044';margin-right:10px}@media (max-width:767px){.plyr__time+.plyr__time{display:none}}.plyr--video .plyr__time{text-shadow:0 1px 1px rgba(0,0,0,.15)}.plyr__tooltip{background:rgba(255,255,255,.9);border-radius:3px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:12px;font-weight:500;line-height:1.3;margin-bottom:10px;opacity:0;padding:5px 7.5px;pointer-events:none;position:absolute;-webkit-transform:translate(-50%,10px) scale(.8);transform:translate(-50%,10px) scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;transition:opacity .2s .1s ease,-webkit-transform .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease,-webkit-transform .2s .1s ease;z-index:2}.plyr__tooltip::before{border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid rgba(255,255,255,.9);bottom:-4px;content:'';height:0;left:50%;position:absolute;-webkit-transform:translateX(-50%);transform:translateX(-50%);width:0;z-index:2}.plyr .plyr__control.plyr__tab-focus .plyr__tooltip,.plyr .plyr__control:hover .plyr__tooltip,.plyr__tooltip--visible{opacity:1;-webkit-transform:translate(-50%,0) scale(1);transform:translate(-50%,0) scale(1)}.plyr .plyr__control:hover .plyr__tooltip{z-index:3}.plyr__controls>.plyr__control:first-child .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip{left:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:0 100%;transform-origin:0 100%}.plyr__controls>.plyr__control:first-child .plyr__tooltip::before,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip::before{left:16px}.plyr__controls>.plyr__control:last-child .plyr__tooltip{right:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%}.plyr__controls>.plyr__control:last-child .plyr__tooltip::before{left:auto;right:16px;-webkit-transform:translateX(50%);transform:translateX(50%)}.plyr__controls>.plyr__control:first-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control:hover .plyr__tooltip,.plyr__controls>.plyr__control:first-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child:hover .plyr__tooltip,.plyr__controls>.plyr__control:last-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:last-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:last-child:hover .plyr__tooltip{-webkit-transform:translate(0,0) scale(1);transform:translate(0,0) scale(1)}.plyr--video{overflow:hidden}.plyr--video.plyr--menu-open{overflow:visible}.plyr__video-wrapper{background:#000;border-radius:inherit;overflow:hidden;position:relative;z-index:0}.plyr__volume{-webkit-box-flex:1;flex:1;position:relative}.plyr__volume input[type=range]{position:relative;z-index:2}@media (min-width:480px){.plyr__volume{max-width:50px}}@media (min-width:768px){.plyr__volume{max-width:80px}}.plyr--is-ios .plyr__volume,.plyr--is-ios [data-plyr=mute]{display:none!important}.plyr--has-error{pointer-events:none}.plyr--has-error::after{-webkit-box-align:center;align-items:center;background:#000;color:#fff;content:attr(data-plyr-error);display:-webkit-box;display:flex;font-size:13px;height:100%;-webkit-box-pack:center;justify-content:center;left:0;position:absolute;text-align:center;text-shadow:0 1px 1px #000;top:0;width:100%;z-index:10}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:fullscreen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:fullscreen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}.plyr:-moz-full-screen .plyr__captions{font-size:21px}.plyr:-ms-fullscreen .plyr__captions{font-size:21px}.plyr:fullscreen .plyr__captions{font-size:21px}}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-moz-full-screen .plyr__captions{font-size:21px}}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-ms-fullscreen .plyr__captions{font-size:21px}}.plyr--fullscreen-fallback{background:#000;border-radius:0!important;height:100%;margin:0;width:100%;bottom:0;left:0;position:fixed;right:0;top:0;z-index:10000000}.plyr--fullscreen-fallback video{height:100%}.plyr--fullscreen-fallback .plyr__video-wrapper{height:100%;width:100%}.plyr--fullscreen-fallback .plyr__video-embed{overflow:visible}.plyr--fullscreen-fallback.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen{display:block}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr--fullscreen-fallback .plyr__captions{font-size:21px}}.plyr--no-transition{transition:none!important}.plyr--full-ui [hidden]{display:none}.plyr--full-ui [aria-hidden=true]{display:none}.plyr__sr-only{clip:rect(1px,1px,1px,1px);overflow:hidden;border:0!important;height:1px!important;padding:0!important;position:absolute!important;width:1px!important}
\ No newline at end of file +@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:300;src:url(https://cdn.plyr.io/static/fonts/gordita-light.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-light.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:400;src:url(https://cdn.plyr.io/static/fonts/gordita-regular.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-regular.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:500;src:url(https://cdn.plyr.io/static/fonts/gordita-medium.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-medium.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:600;src:url(https://cdn.plyr.io/static/fonts/gordita-bold.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-bold.woff) format("woff")}@font-face{font-display:swap;font-family:Gordita;font-style:normal;font-weight:900;src:url(https://cdn.plyr.io/static/fonts/gordita-black.woff2) format("woff2"),url(https://cdn.plyr.io/static/fonts/gordita-black.woff) format("woff")}@-webkit-keyframes fadein{0%{opacity:0}100%{opacity:1}}@keyframes fadein{0%{opacity:0}100%{opacity:1}}/*! normalize.css v7.0.0 | MIT License | github.com/necolas/normalize.css */html{line-height:1.15;-ms-text-size-adjust:100%;-webkit-text-size-adjust:100%}body{margin:0}article,aside,footer,header,nav,section{display:block}h1{font-size:2em;margin:.67em 0}figcaption,figure,main{display:block}figure{margin:1em 40px}hr{box-sizing:content-box;height:0;overflow:visible}pre{font-family:monospace,monospace;font-size:1em}a,button.faux-link{background-color:transparent;-webkit-text-decoration-skip:objects}abbr[title]{border-bottom:none;text-decoration:underline;-webkit-text-decoration:underline dotted;text-decoration:underline dotted}b,strong{font-weight:inherit}b,strong{font-weight:bolder}code,kbd,samp{font-family:monospace,monospace;font-size:1em}dfn{font-style:italic}mark{background-color:#ff0;color:#000}small{font-size:80%}sub,sup{font-size:75%;line-height:0;position:relative;vertical-align:baseline}sub{bottom:-.25em}sup{top:-.5em}audio,video{display:inline-block}audio:not([controls]){display:none;height:0}img{border-style:none}svg:not(:root){overflow:hidden}button,input,optgroup,select,textarea{font-family:sans-serif;font-size:100%;line-height:1.15;margin:0}button,input{overflow:visible}button,select{text-transform:none}[type=reset],[type=submit],button,html [type=button]{-webkit-appearance:button}[type=button]::-moz-focus-inner,[type=reset]::-moz-focus-inner,[type=submit]::-moz-focus-inner,button::-moz-focus-inner{border-style:none;padding:0}[type=button]:-moz-focusring,[type=reset]:-moz-focusring,[type=submit]:-moz-focusring,button:-moz-focusring{outline:1px dotted ButtonText}fieldset{padding:.35em .75em .625em}legend{box-sizing:border-box;color:inherit;display:table;max-width:100%;padding:0;white-space:normal}progress{display:inline-block;vertical-align:baseline}textarea{overflow:auto}[type=checkbox],[type=radio]{box-sizing:border-box;padding:0}[type=number]::-webkit-inner-spin-button,[type=number]::-webkit-outer-spin-button{height:auto}[type=search]{-webkit-appearance:textfield;outline-offset:-2px}[type=search]::-webkit-search-cancel-button,[type=search]::-webkit-search-decoration{-webkit-appearance:none}::-webkit-file-upload-button{-webkit-appearance:button;font:inherit}details,menu{display:block}summary{display:list-item}canvas{display:inline-block}template{display:none}[hidden]{display:none}*,::after,::before{box-sizing:border-box}body,html{display:-webkit-box;display:flex;width:100%}html{background:linear-gradient(to left top,#4dc1ff,#0074b3);background-attachment:fixed;height:100%}body{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;-webkit-box-orient:vertical;-webkit-box-direction:normal;flex-direction:column;min-height:100%}.grid{-webkit-box-flex:1;flex:1;overflow:auto}main{margin:auto;text-align:center}aside{-webkit-box-align:center;align-items:center;background:#fff;color:#55646b;display:-webkit-box;display:flex;flex-shrink:0;-webkit-box-pack:center;justify-content:center;padding:15px;position:relative;text-align:center;text-shadow:none;width:100%}aside .icon{fill:#4baaf4;margin-right:10px}aside p{margin:0}aside a,aside button.faux-link{color:#4baaf4}aside a.tab-focus,aside button.tab-focus.faux-link{box-shadow:0 0 0 3px rgba(75,170,244,.35);outline:0}.grid{margin:0 auto;padding:20px}@media only screen and (min-width:768px){.grid{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;max-width:1280px;width:100%}.grid>*{-webkit-box-flex:1;flex:1}}html{font-size:100%}body{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;font-size:15px;font-size:.9375rem;color:#fff;font-family:Gordita,Avenir,"Helvetica Neue",sans-serif;font-weight:500;line-height:1.75;text-shadow:0 1px 1px rgba(0,0,0,.15)}button,input,select,textarea{font:inherit}p,small{margin:0 0 20px}small{font-size:13px;font-size:.8125rem;display:block}h1{font-size:64px;font-size:4rem;font-weight:600;letter-spacing:-.025em;margin:0 0 10px}.button,.button__count{-webkit-box-align:center;align-items:center;background:#fff;border:0;border-radius:4px;box-shadow:0 1px 1px rgba(0,0,0,.1);color:#55646b;display:-webkit-inline-box;display:inline-flex;padding:15px;position:relative;text-shadow:none;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;vertical-align:middle}.button{font-weight:600;padding-left:20px;padding-right:20px;transition:all .2s ease}.button:focus,.button:hover{color:#343f4a}.button:focus::after,.button:hover::after{display:none}.button:hover{box-shadow:0 2px 2px rgba(0,0,0,.1);-webkit-transform:translateY(-1px);transform:translateY(-1px)}.button:focus{outline:0}.button.tab-focus{box-shadow:0 0 0 3px rgba(255,255,255,.35);outline:0}.button:active{-webkit-transform:translateY(1px);transform:translateY(1px)}.button--with-count{display:-webkit-inline-box;display:inline-flex}.button--with-count .button .icon{flex-shrink:0}.button__count{-webkit-animation:fadein .2s ease;animation:fadein .2s ease;margin-left:10px}.button__count::before{border:5px solid transparent;border-left-width:0;border-right-color:#fff;content:'';height:0;position:absolute;right:100%;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);width:0}header{padding-bottom:20px;text-align:center}header .call-to-action{margin-top:30px}@media only screen and (min-width:768px){header{margin-right:60px;max-width:360px;padding-bottom:40px;text-align:left}}.icon{fill:currentColor;height:16px;vertical-align:-3px;width:16px}a svg,button svg,button.faux-link svg,label svg{pointer-events:none}.btn .icon,a .icon,button.faux-link .icon{margin-right:6px}button.faux-link{background:0 0;border:0;border-radius:0;cursor:pointer;font:inherit;line-height:1.75;margin:0;padding:0;position:relative;text-align:inherit;text-shadow:inherit;-moz-user-select:text;vertical-align:baseline;width:auto}a,button.faux-link{border-bottom:1px dotted currentColor;color:#fff;font-weight:600;position:relative;text-decoration:none;transition:all .2s ease}a::after,button.faux-link::after{background:currentColor;content:'';height:1px;left:50%;position:absolute;top:100%;-webkit-transform:translateX(-50%);transform:translateX(-50%);transition:width .2s ease;width:0}a:focus,a:hover,button.faux-link:focus,button.faux-link:hover{border-bottom-color:transparent;outline:0}a:focus::after,a:hover::after,button.faux-link:focus::after,button.faux-link:hover::after{width:100%}a.tab-focus,button.tab-focus.faux-link{box-shadow:0 0 0 3px rgba(255,255,255,.35);outline:0}li,ul{list-style:none;margin:0;padding:0}nav{display:-webkit-box;display:flex;-webkit-box-pack:center;justify-content:center;margin-bottom:20px}video{max-width:100%;vertical-align:middle}.plyr{border-radius:4px;box-shadow:0 2px 5px rgba(0,0,0,.2);margin:20px auto}.plyr.plyr--audio{max-width:480px}.plyr__video-wrapper::after{border:1px solid rgba(0,0,0,.15);border-radius:inherit;bottom:0;content:'';left:0;pointer-events:none;position:absolute;right:0;top:0}.plyr__cite{display:none;margin-top:20px}.plyr__cite .icon{margin-right:4px}.plyr--audio~ul .plyr__cite--audio,.plyr--video:not(.plyr--youtube):not(.plyr--vimeo)~ul .plyr__cite--video,.plyr--vimeo~ul .plyr__cite--vimeo,.plyr--youtube~ul .plyr__cite--youtube{display:block}@-webkit-keyframes plyr-progress{to{background-position:25px 0}}@keyframes plyr-progress{to{background-position:25px 0}}@-webkit-keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}@keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}.plyr{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;direction:ltr;font-family:inherit;font-weight:500;line-height:1.7;max-width:100%;min-width:200px;position:relative;text-shadow:none;transition:box-shadow .3s ease}.plyr audio,.plyr video{border-radius:inherit;height:auto;vertical-align:middle;width:100%}.plyr:focus{outline:0}.plyr--full-ui{box-sizing:border-box}.plyr--full-ui *,.plyr--full-ui ::after,.plyr--full-ui ::before{box-sizing:inherit}.plyr--full-ui a,.plyr--full-ui button,.plyr--full-ui button.faux-link,.plyr--full-ui input,.plyr--full-ui label{touch-action:manipulation}.plyr__badge{background:#4f5b5f;border-radius:2px;color:rgba(255,255,255,.9);font-size:9px;line-height:1;padding:3px 4px}.plyr--full-ui ::-webkit-media-text-track-container{display:none}.plyr__captions{-webkit-animation:plyr-fade-in .3s ease;animation:plyr-fade-in .3s ease;bottom:0;color:#fff;display:none;font-size:12px;left:0;padding:10px;position:absolute;text-align:center;-webkit-transform:translateY(-40px);transform:translateY(-40px);transition:-webkit-transform .4s ease-in-out;transition:transform .4s ease-in-out;transition:transform .4s ease-in-out,-webkit-transform .4s ease-in-out;width:100%}.plyr__captions span{background:rgba(0,0,0,.8);border-radius:2px;-webkit-box-decoration-break:clone;box-decoration-break:clone;line-height:185%;padding:.2em .5em;white-space:pre-wrap}.plyr__captions span div{display:inline}.plyr__captions span:empty{display:none}@media (min-width:480px){.plyr__captions{font-size:13px;padding:20px}}@media (min-width:768px){.plyr__captions{font-size:18px}}.plyr--captions-active .plyr__captions{display:block}.plyr--hide-controls .plyr__captions{-webkit-transform:translateY(-15px);transform:translateY(-15px)}.plyr__control{background:0 0;border:0;border-radius:3px;color:inherit;cursor:pointer;flex-shrink:0;overflow:visible;padding:7px;position:relative;transition:all .3s ease}.plyr__control svg{display:block;fill:currentColor;height:18px;pointer-events:none;width:18px}.plyr__control:focus{outline:0}.plyr__control.plyr__tab-focus{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__control[aria-pressed=false] .icon--pressed,.plyr__control[aria-pressed=false] .label--pressed,.plyr__control[aria-pressed=true] .icon--not-pressed,.plyr__control[aria-pressed=true] .label--not-pressed{display:none}.plyr--audio .plyr__control.plyr__tab-focus,.plyr--audio .plyr__control:hover,.plyr--audio .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr__control--overlaid{background:rgba(26,175,255,.8);border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15);color:#fff;display:none;left:50%;padding:15px;position:absolute;top:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);z-index:2}.plyr__control--overlaid svg{height:20px;left:2px;position:relative;width:20px}.plyr__control--overlaid:focus,.plyr__control--overlaid:hover{background:#1aafff}.plyr--playing .plyr__control--overlaid{opacity:0;visibility:hidden}.plyr--full-ui.plyr--video .plyr__control--overlaid{display:block}.plyr--full-ui ::-webkit-media-controls{display:none}.plyr__controls{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;text-align:center}.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:5px}.plyr__controls .plyr__menu:first-child,.plyr__controls .plyr__menu:first-child+[data-plyr=pause],.plyr__controls .plyr__progress:first-child,.plyr__controls .plyr__progress:first-child+[data-plyr=pause],.plyr__controls .plyr__time:first-child,.plyr__controls .plyr__time:first-child+[data-plyr=pause],.plyr__controls>.plyr__control:first-child,.plyr__controls>.plyr__control:first-child+[data-plyr=pause]{margin-left:0}.plyr__controls .plyr__volume{margin-left:5px}@media (min-width:480px){.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:10px}.plyr__controls .plyr__menu+.plyr__control,.plyr__controls>.plyr__control+.plyr__control,.plyr__controls>.plyr__control+.plyr__menu{margin-left:5px}}.plyr--video .plyr__controls{background:linear-gradient(transparent,rgba(0,0,0,.7));border-bottom-left-radius:inherit;border-bottom-right-radius:inherit;bottom:0;color:#fff;left:0;padding:35px 10px 10px;position:absolute;right:0;transition:opacity .4s ease-in-out,-webkit-transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out,-webkit-transform .4s ease-in-out;z-index:2}.plyr--video .plyr__controls .plyr__control svg{-webkit-filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15));filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15))}.plyr--video .plyr__controls .plyr__control.plyr__tab-focus,.plyr--video .plyr__controls .plyr__control:hover,.plyr--video .plyr__controls .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr--audio .plyr__controls{background:#fff;border-radius:inherit;color:#4f5b5f;padding:10px}.plyr--video.plyr--hide-controls .plyr__controls{opacity:0;pointer-events:none;-webkit-transform:translateY(100%);transform:translateY(100%)}.plyr [data-plyr=airplay],.plyr [data-plyr=captions],.plyr [data-plyr=fullscreen],.plyr [data-plyr=pip]{display:none}.plyr--airplay-supported [data-plyr=airplay],.plyr--captions-enabled [data-plyr=captions],.plyr--fullscreen-enabled [data-plyr=fullscreen],.plyr--pip-supported [data-plyr=pip]{display:inline-block}.plyr__video-embed{height:0;padding-bottom:56.25%;position:relative}.plyr__video-embed iframe{border:0;height:100%;left:0;position:absolute;top:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__video-embed>div{padding-bottom:200%;position:relative;-webkit-transform:translateY(-35.9375%);transform:translateY(-35.9375%)}.plyr--full-ui .plyr__video-embed iframe{pointer-events:none}.plyr__menu{display:-webkit-box;display:flex;position:relative}.plyr__menu .plyr__control svg{transition:-webkit-transform .3s ease;transition:transform .3s ease;transition:transform .3s ease,-webkit-transform .3s ease}.plyr__menu .plyr__control[aria-expanded=true] svg{-webkit-transform:rotate(90deg);transform:rotate(90deg)}.plyr__menu .plyr__control[aria-expanded=true] .plyr__tooltip{display:none}.plyr__menu__container{-webkit-animation:plyr-popup .2s ease;animation:plyr-popup .2s ease;background:rgba(255,255,255,.9);border-radius:4px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:13px;margin-bottom:10px;position:absolute;right:-3px;text-align:left;white-space:nowrap;z-index:1}.plyr__menu__container>div{overflow:hidden;transition:height .35s cubic-bezier(.4,0,.2,1),width .35s cubic-bezier(.4,0,.2,1)}.plyr__menu__container::after{border:4px solid transparent;border-top-color:rgba(255,255,255,.9);content:'';height:0;position:absolute;right:15px;top:100%;width:0}.plyr__menu__container ul{list-style:none;margin:0;overflow:hidden;padding:7px}.plyr__menu__container .plyr__control{-webkit-box-align:center;align-items:center;color:#4f5b5f;display:-webkit-box;display:flex;padding:4px 14px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__menu__container .plyr__control::after{border:4px solid transparent;content:'';position:absolute;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);transition:border-color .2s ease}.plyr__menu__container .plyr__control--forward{padding-right:28px}.plyr__menu__container .plyr__control--forward::after{border-left-color:rgba(79,91,95,.8);right:5px}.plyr__menu__container .plyr__control--forward.plyr__tab-focus::after,.plyr__menu__container .plyr__control--forward:hover::after{border-left-color:currentColor}.plyr__menu__container .plyr__control--back{font-weight:500;margin:7px;margin-bottom:3px;padding-left:28px;position:relative;width:calc(100% - 14px)}.plyr__menu__container .plyr__control--back::after{border-right-color:rgba(79,91,95,.8);left:7px}.plyr__menu__container .plyr__control--back::before{background:#b7c5cd;box-shadow:0 1px 0 #fff;content:'';height:1px;left:0;margin-top:4px;overflow:hidden;position:absolute;right:0;top:100%}.plyr__menu__container .plyr__control--back.plyr__tab-focus::after,.plyr__menu__container .plyr__control--back:hover::after{border-right-color:currentColor}.plyr__menu__container label.plyr__control{padding-left:7px}.plyr__menu__container label.plyr__control input[type=radio]+span{background:rgba(0,0,0,.1);border-radius:100%;display:block;flex-shrink:0;height:16px;margin-right:10px;position:relative;transition:all .3s ease;width:16px}.plyr__menu__container label.plyr__control input[type=radio]+span::after{background:#fff;border-radius:100%;content:'';height:6px;left:5px;opacity:0;position:absolute;top:5px;-webkit-transform:scale(0);transform:scale(0);transition:opacity .3s ease,-webkit-transform .3s ease;transition:transform .3s ease,opacity .3s ease;transition:transform .3s ease,opacity .3s ease,-webkit-transform .3s ease;width:6px}.plyr__menu__container label.plyr__control input[type=radio]:checked+span{background:#1aafff}.plyr__menu__container label.plyr__control input[type=radio]:checked+span::after{opacity:1;-webkit-transform:scale(1);transform:scale(1)}.plyr__menu__container label.plyr__control input[type=radio]:focus+span{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__menu__container label.plyr__control.plyr__tab-focus input[type=radio]+span,.plyr__menu__container label.plyr__control:hover input[type=radio]+span{background:rgba(0,0,0,.1)}.plyr__menu__container .plyr__menu__value{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;margin-left:auto;margin-right:-7px;overflow:hidden;padding-left:25px;pointer-events:none}.plyr__progress{display:-webkit-box;display:flex;-webkit-box-flex:1;flex:1;position:relative}.plyr__progress input[type=range]{position:relative;z-index:2}.plyr__progress .plyr__tooltip{font-size:11px;left:0}.plyr__progress--buffer{-webkit-appearance:none;background:0 0;border:0;border-radius:100px;height:6px;left:0;margin:-3px 0 0;padding:0;position:absolute;top:50%;width:100%}.plyr__progress--buffer::-webkit-progress-bar{background:0 0;transition:width .2s ease}.plyr__progress--buffer::-webkit-progress-value{background:currentColor;border-radius:100px;min-width:6px}.plyr__progress--buffer::-moz-progress-bar{background:currentColor;border-radius:100px;min-width:6px;transition:width .2s ease}.plyr__progress--buffer::-ms-fill{border-radius:100px;transition:width .2s ease}.plyr--video .plyr__progress--buffer{box-shadow:0 1px 1px rgba(0,0,0,.15);color:rgba(255,255,255,.25)}.plyr--audio .plyr__progress--buffer{color:rgba(183,197,205,.66)}.plyr--loading .plyr__progress--buffer{-webkit-animation:plyr-progress 1s linear infinite;animation:plyr-progress 1s linear infinite;background-image:linear-gradient(-45deg,rgba(47,52,61,.6) 25%,transparent 25%,transparent 50%,rgba(47,52,61,.6) 50%,rgba(47,52,61,.6) 75%,transparent 75%,transparent);background-repeat:repeat-x;background-size:25px 25px;color:transparent}.plyr--video.plyr--loading .plyr__progress--buffer{background-color:rgba(255,255,255,.25)}.plyr--audio.plyr--loading .plyr__progress--buffer{background-color:rgba(183,197,205,.66)}.plyr--full-ui input[type=range]{-webkit-appearance:none;background:0 0;border:0;border-radius:28px;color:#1aafff;display:block;height:20px;margin:0;padding:0;transition:box-shadow .3s ease;width:100%}.plyr--full-ui input[type=range]::-webkit-slider-runnable-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-webkit-user-select:none;user-select:none;background-image:linear-gradient(to right,currentColor var(--value),transparent var(--value))}.plyr--full-ui input[type=range]::-webkit-slider-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;-webkit-appearance:none;margin-top:-4px}.plyr--full-ui input[type=range]::-moz-range-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-moz-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-moz-range-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px}.plyr--full-ui input[type=range]::-moz-range-progress{background:currentColor;border-radius:3px;height:6px}.plyr--full-ui input[type=range]::-ms-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;color:transparent}.plyr--full-ui input[type=range]::-ms-fill-upper{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-ms-fill-lower{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;background:currentColor}.plyr--full-ui input[type=range]::-ms-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;margin-top:0}.plyr--full-ui input[type=range]::-ms-tooltip{display:none}.plyr--full-ui input[type=range]:focus{outline:0}.plyr--full-ui input[type=range]::-moz-focus-outer{border:0}.plyr--full-ui input[type=range].plyr__tab-focus::-webkit-slider-runnable-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-moz-range-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-ms-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range]:active::-webkit-slider-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-moz-range-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-ms-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui.plyr--video input[type=range]::-webkit-slider-runnable-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-moz-range-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-ms-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--audio input[type=range]::-webkit-slider-runnable-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-moz-range-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-ms-track{background-color:rgba(183,197,205,.66)}.plyr__time{font-size:11px}.plyr__time+.plyr__time::before{content:'\2044';margin-right:10px}@media (max-width:767px){.plyr__time+.plyr__time{display:none}}.plyr--video .plyr__time{text-shadow:0 1px 1px rgba(0,0,0,.15)}.plyr__tooltip{background:rgba(255,255,255,.9);border-radius:3px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:12px;font-weight:500;line-height:1.3;margin-bottom:10px;opacity:0;padding:5px 7.5px;pointer-events:none;position:absolute;-webkit-transform:translate(-50%,10px) scale(.8);transform:translate(-50%,10px) scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;transition:opacity .2s .1s ease,-webkit-transform .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease,-webkit-transform .2s .1s ease;z-index:2}.plyr__tooltip::before{border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid rgba(255,255,255,.9);bottom:-4px;content:'';height:0;left:50%;position:absolute;-webkit-transform:translateX(-50%);transform:translateX(-50%);width:0;z-index:2}.plyr .plyr__control.plyr__tab-focus .plyr__tooltip,.plyr .plyr__control:hover .plyr__tooltip,.plyr__tooltip--visible{opacity:1;-webkit-transform:translate(-50%,0) scale(1);transform:translate(-50%,0) scale(1)}.plyr .plyr__control:hover .plyr__tooltip{z-index:3}.plyr__controls>.plyr__control:first-child .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip{left:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:0 100%;transform-origin:0 100%}.plyr__controls>.plyr__control:first-child .plyr__tooltip::before,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip::before{left:16px}.plyr__controls>.plyr__control:last-child .plyr__tooltip{right:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%}.plyr__controls>.plyr__control:last-child .plyr__tooltip::before{left:auto;right:16px;-webkit-transform:translateX(50%);transform:translateX(50%)}.plyr__controls>.plyr__control:first-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control:hover .plyr__tooltip,.plyr__controls>.plyr__control:first-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child:hover .plyr__tooltip,.plyr__controls>.plyr__control:last-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:last-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:last-child:hover .plyr__tooltip{-webkit-transform:translate(0,0) scale(1);transform:translate(0,0) scale(1)}.plyr--video{overflow:hidden}.plyr--video.plyr--menu-open{overflow:visible}.plyr__video-wrapper{background:#000;border-radius:inherit;overflow:hidden;position:relative;z-index:0}.plyr__volume{-webkit-box-flex:1;flex:1;position:relative}.plyr__volume input[type=range]{position:relative;z-index:2}@media (min-width:480px){.plyr__volume{max-width:50px}}@media (min-width:768px){.plyr__volume{max-width:80px}}.plyr--is-ios .plyr__volume,.plyr--is-ios [data-plyr=mute]{display:none!important}.plyr--has-error{pointer-events:none}.plyr--has-error::after{-webkit-box-align:center;align-items:center;background:#000;color:#fff;content:attr(data-plyr-error);display:-webkit-box;display:flex;font-size:13px;height:100%;-webkit-box-pack:center;justify-content:center;left:0;position:absolute;text-align:center;text-shadow:0 1px 1px #000;top:0;width:100%;z-index:10}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:fullscreen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:fullscreen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}.plyr:-moz-full-screen .plyr__captions{font-size:21px}.plyr:-ms-fullscreen .plyr__captions{font-size:21px}.plyr:fullscreen .plyr__captions{font-size:21px}}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-moz-full-screen .plyr__captions{font-size:21px}}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-ms-fullscreen .plyr__captions{font-size:21px}}.plyr--fullscreen-fallback{background:#000;border-radius:0!important;height:100%;margin:0;width:100%;bottom:0;left:0;position:fixed;right:0;top:0;z-index:10000000}.plyr--fullscreen-fallback video{height:100%}.plyr--fullscreen-fallback .plyr__video-wrapper{height:100%;width:100%}.plyr--fullscreen-fallback .plyr__video-embed{overflow:visible}.plyr--fullscreen-fallback.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen{display:block}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr--fullscreen-fallback .plyr__captions{font-size:21px}}.plyr--no-transition{transition:none!important}.plyr--full-ui [hidden]{display:none}.plyr--full-ui [aria-hidden=true]{display:none}.plyr__sr-only{clip:rect(1px,1px,1px,1px);overflow:hidden;border:0!important;height:1px!important;padding:0!important;position:absolute!important;width:1px!important}
\ No newline at end of file diff --git a/dist/plyr.css b/dist/plyr.css index dc124c03..0a7e442c 100644 --- a/dist/plyr.css +++ b/dist/plyr.css @@ -1 +1 @@ -@-webkit-keyframes plyr-progress{to{background-position:25px 0}}@keyframes plyr-progress{to{background-position:25px 0}}@-webkit-keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}@keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}.plyr{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;direction:ltr;font-family:Avenir,"Avenir Next","Helvetica Neue","Segoe UI",Helvetica,Arial,sans-serif;font-weight:500;line-height:1.7;max-width:100%;min-width:200px;position:relative;text-shadow:none;transition:box-shadow .3s ease}.plyr audio,.plyr video{border-radius:inherit;height:auto;vertical-align:middle;width:100%}.plyr:focus{outline:0}.plyr--full-ui{box-sizing:border-box}.plyr--full-ui *,.plyr--full-ui ::after,.plyr--full-ui ::before{box-sizing:inherit}.plyr--full-ui a,.plyr--full-ui button,.plyr--full-ui input,.plyr--full-ui label{touch-action:manipulation}.plyr__badge{background:#4f5b5f;border-radius:2px;color:rgba(255,255,255,.9);font-size:9px;line-height:1;padding:3px 4px}.plyr--full-ui ::-webkit-media-text-track-container{display:none}.plyr__captions{-webkit-animation:plyr-fade-in .3s ease;animation:plyr-fade-in .3s ease;bottom:0;color:#fff;display:none;font-size:14px;left:0;padding:10px;position:absolute;text-align:center;-webkit-transform:translateY(-40px);transform:translateY(-40px);transition:-webkit-transform .4s ease-in-out;transition:transform .4s ease-in-out;transition:transform .4s ease-in-out,-webkit-transform .4s ease-in-out;width:100%}.plyr__captions span{background:rgba(0,0,0,.8);border-radius:2px;-webkit-box-decoration-break:clone;box-decoration-break:clone;line-height:185%;padding:.2em .5em;white-space:pre-wrap}.plyr__captions span div{display:inline}.plyr__captions span:empty{display:none}@media (min-width:480px){.plyr__captions{font-size:16px;padding:20px}}@media (min-width:768px){.plyr__captions{font-size:18px}}.plyr--captions-active .plyr__captions{display:block}.plyr--hide-controls .plyr__captions{-webkit-transform:translateY(-15px);transform:translateY(-15px)}.plyr__control{background:0 0;border:0;border-radius:3px;color:inherit;cursor:pointer;flex-shrink:0;overflow:visible;padding:7px;position:relative;transition:all .3s ease}.plyr__control svg{display:block;fill:currentColor;height:18px;pointer-events:none;width:18px}.plyr__control:focus{outline:0}.plyr__control.plyr__tab-focus{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__control[aria-pressed=false] .icon--pressed,.plyr__control[aria-pressed=false] .label--pressed,.plyr__control[aria-pressed=true] .icon--not-pressed,.plyr__control[aria-pressed=true] .label--not-pressed{display:none}.plyr--audio .plyr__control.plyr__tab-focus,.plyr--audio .plyr__control:hover,.plyr--audio .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr__control--overlaid{background:rgba(26,175,255,.8);border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15);color:#fff;display:none;left:50%;padding:15px;position:absolute;top:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);z-index:2}.plyr__control--overlaid svg{height:20px;left:2px;position:relative;width:20px}.plyr__control--overlaid:focus,.plyr__control--overlaid:hover{background:#1aafff}.plyr--playing .plyr__control--overlaid{opacity:0;visibility:hidden}.plyr--full-ui.plyr--video .plyr__control--overlaid{display:block}.plyr--full-ui ::-webkit-media-controls{display:none}.plyr__controls{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;text-align:center}.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:5px}.plyr__controls .plyr__menu:first-child,.plyr__controls .plyr__menu:first-child+[data-plyr=pause],.plyr__controls .plyr__progress:first-child,.plyr__controls .plyr__progress:first-child+[data-plyr=pause],.plyr__controls .plyr__time:first-child,.plyr__controls .plyr__time:first-child+[data-plyr=pause],.plyr__controls>.plyr__control:first-child,.plyr__controls>.plyr__control:first-child+[data-plyr=pause]{margin-left:0}.plyr__controls .plyr__volume{margin-left:5px}@media (min-width:480px){.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:10px}.plyr__controls .plyr__menu+.plyr__control,.plyr__controls>.plyr__control+.plyr__control,.plyr__controls>.plyr__control+.plyr__menu{margin-left:5px}}.plyr--video .plyr__controls{background:linear-gradient(transparent,rgba(0,0,0,.7));border-bottom-left-radius:inherit;border-bottom-right-radius:inherit;bottom:0;color:#fff;left:0;padding:35px 10px 10px;position:absolute;right:0;transition:opacity .4s ease-in-out,-webkit-transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out,-webkit-transform .4s ease-in-out;z-index:2}.plyr--video .plyr__controls .plyr__control svg{-webkit-filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15));filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15))}.plyr--video .plyr__controls .plyr__control.plyr__tab-focus,.plyr--video .plyr__controls .plyr__control:hover,.plyr--video .plyr__controls .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr--audio .plyr__controls{background:#fff;border-radius:inherit;color:#4f5b5f;padding:10px}.plyr--video.plyr--hide-controls .plyr__controls{opacity:0;pointer-events:none;-webkit-transform:translateY(100%);transform:translateY(100%)}.plyr [data-plyr=airplay],.plyr [data-plyr=captions],.plyr [data-plyr=fullscreen],.plyr [data-plyr=pip]{display:none}.plyr--airplay-supported [data-plyr=airplay],.plyr--captions-enabled [data-plyr=captions],.plyr--fullscreen-enabled [data-plyr=fullscreen],.plyr--pip-supported [data-plyr=pip]{display:inline-block}.plyr__video-embed{height:0;padding-bottom:56.25%}.plyr__video-embed iframe{border:0;height:100%;left:0;position:absolute;top:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__video-embed>div{padding-bottom:200%;position:relative;-webkit-transform:translateY(-35.9375%);transform:translateY(-35.9375%)}.plyr--full-ui .plyr__video-embed iframe{pointer-events:none}.plyr__menu{display:-webkit-box;display:flex;position:relative}.plyr__menu .plyr__control svg{transition:-webkit-transform .3s ease;transition:transform .3s ease;transition:transform .3s ease,-webkit-transform .3s ease}.plyr__menu .plyr__control[aria-expanded=true] svg{-webkit-transform:rotate(90deg);transform:rotate(90deg)}.plyr__menu .plyr__control[aria-expanded=true] .plyr__tooltip{display:none}.plyr__menu__container{-webkit-animation:plyr-popup .2s ease;animation:plyr-popup .2s ease;background:rgba(255,255,255,.9);border-radius:4px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:16px;margin-bottom:10px;position:absolute;right:-3px;text-align:left;white-space:nowrap;z-index:1}.plyr__menu__container>div{overflow:hidden;transition:height .35s cubic-bezier(.4,0,.2,1),width .35s cubic-bezier(.4,0,.2,1)}.plyr__menu__container::after{border:4px solid transparent;border-top-color:rgba(255,255,255,.9);content:'';height:0;position:absolute;right:15px;top:100%;width:0}.plyr__menu__container ul{list-style:none;margin:0;overflow:hidden;padding:7px}.plyr__menu__container .plyr__control{-webkit-box-align:center;align-items:center;color:#4f5b5f;display:-webkit-box;display:flex;padding:4px 14px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__menu__container .plyr__control::after{border:4px solid transparent;content:'';position:absolute;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);transition:border-color .2s ease}.plyr__menu__container .plyr__control--forward{padding-right:28px}.plyr__menu__container .plyr__control--forward::after{border-left-color:rgba(79,91,95,.8);right:5px}.plyr__menu__container .plyr__control--forward.plyr__tab-focus::after,.plyr__menu__container .plyr__control--forward:hover::after{border-left-color:currentColor}.plyr__menu__container .plyr__control--back{font-weight:500;margin:7px;margin-bottom:3px;padding-left:28px;position:relative;width:calc(100% - 14px)}.plyr__menu__container .plyr__control--back::after{border-right-color:rgba(79,91,95,.8);left:7px}.plyr__menu__container .plyr__control--back::before{background:#b7c5cd;box-shadow:0 1px 0 #fff;content:'';height:1px;left:0;margin-top:4px;overflow:hidden;position:absolute;right:0;top:100%}.plyr__menu__container .plyr__control--back.plyr__tab-focus::after,.plyr__menu__container .plyr__control--back:hover::after{border-right-color:currentColor}.plyr__menu__container label.plyr__control{padding-left:7px}.plyr__menu__container label.plyr__control input[type=radio]+span{background:rgba(0,0,0,.1);border-radius:100%;display:block;flex-shrink:0;height:16px;margin-right:10px;position:relative;transition:all .3s ease;width:16px}.plyr__menu__container label.plyr__control input[type=radio]+span::after{background:#fff;border-radius:100%;content:'';height:6px;left:5px;opacity:0;position:absolute;top:5px;-webkit-transform:scale(0);transform:scale(0);transition:opacity .3s ease,-webkit-transform .3s ease;transition:transform .3s ease,opacity .3s ease;transition:transform .3s ease,opacity .3s ease,-webkit-transform .3s ease;width:6px}.plyr__menu__container label.plyr__control input[type=radio]:checked+span{background:#1aafff}.plyr__menu__container label.plyr__control input[type=radio]:checked+span::after{opacity:1;-webkit-transform:scale(1);transform:scale(1)}.plyr__menu__container label.plyr__control input[type=radio]:focus+span{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__menu__container label.plyr__control.plyr__tab-focus input[type=radio]+span,.plyr__menu__container label.plyr__control:hover input[type=radio]+span{background:rgba(0,0,0,.1)}.plyr__menu__container .plyr__menu__value{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;margin-left:auto;margin-right:-7px;overflow:hidden;padding-left:25px;pointer-events:none}.plyr__progress{display:-webkit-box;display:flex;-webkit-box-flex:1;flex:1;position:relative}.plyr__progress input[type=range]{position:relative;z-index:2}.plyr__progress .plyr__tooltip{font-size:14px;left:0}.plyr__progress--buffer{-webkit-appearance:none;background:0 0;border:0;border-radius:100px;height:6px;left:0;margin:-3px 0 0;padding:0;position:absolute;top:50%;width:100%}.plyr__progress--buffer::-webkit-progress-bar{background:0 0;transition:width .2s ease}.plyr__progress--buffer::-webkit-progress-value{background:currentColor;border-radius:100px;min-width:6px}.plyr__progress--buffer::-moz-progress-bar{background:currentColor;border-radius:100px;min-width:6px;transition:width .2s ease}.plyr__progress--buffer::-ms-fill{border-radius:100px;transition:width .2s ease}.plyr--video .plyr__progress--buffer{box-shadow:0 1px 1px rgba(0,0,0,.15);color:rgba(255,255,255,.25)}.plyr--audio .plyr__progress--buffer{color:rgba(183,197,205,.66)}.plyr--loading .plyr__progress--buffer{-webkit-animation:plyr-progress 1s linear infinite;animation:plyr-progress 1s linear infinite;background-image:linear-gradient(-45deg,rgba(47,52,61,.6) 25%,transparent 25%,transparent 50%,rgba(47,52,61,.6) 50%,rgba(47,52,61,.6) 75%,transparent 75%,transparent);background-repeat:repeat-x;background-size:25px 25px;color:transparent}.plyr--video.plyr--loading .plyr__progress--buffer{background-color:rgba(255,255,255,.25)}.plyr--audio.plyr--loading .plyr__progress--buffer{background-color:rgba(183,197,205,.66)}.plyr--full-ui input[type=range]{-webkit-appearance:none;background:0 0;border:0;border-radius:28px;color:#1aafff;display:block;height:20px;margin:0;padding:0;transition:box-shadow .3s ease;width:100%}.plyr--full-ui input[type=range]::-webkit-slider-runnable-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-webkit-user-select:none;user-select:none;background-image:linear-gradient(to right,currentColor var(--value),transparent var(--value))}.plyr--full-ui input[type=range]::-webkit-slider-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;-webkit-appearance:none;margin-top:-4px}.plyr--full-ui input[type=range]::-moz-range-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-moz-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-moz-range-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px}.plyr--full-ui input[type=range]::-moz-range-progress{background:currentColor;border-radius:3px;height:6px}.plyr--full-ui input[type=range]::-ms-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;color:transparent}.plyr--full-ui input[type=range]::-ms-fill-upper{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-ms-fill-lower{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;background:currentColor}.plyr--full-ui input[type=range]::-ms-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;margin-top:0}.plyr--full-ui input[type=range]::-ms-tooltip{display:none}.plyr--full-ui input[type=range]:focus{outline:0}.plyr--full-ui input[type=range]::-moz-focus-outer{border:0}.plyr--full-ui input[type=range].plyr__tab-focus::-webkit-slider-runnable-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-moz-range-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-ms-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range]:active::-webkit-slider-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-moz-range-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-ms-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui.plyr--video input[type=range]::-webkit-slider-runnable-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-moz-range-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-ms-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--audio input[type=range]::-webkit-slider-runnable-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-moz-range-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-ms-track{background-color:rgba(183,197,205,.66)}.plyr__time{font-size:14px}.plyr__time+.plyr__time::before{content:'\2044';margin-right:10px}@media (max-width:767px){.plyr__time+.plyr__time{display:none}}.plyr--video .plyr__time{text-shadow:0 1px 1px rgba(0,0,0,.15)}.plyr__tooltip{background:rgba(255,255,255,.9);border-radius:3px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:14px;font-weight:500;line-height:1.3;margin-bottom:10px;opacity:0;padding:5px 7.5px;pointer-events:none;position:absolute;-webkit-transform:translate(-50%,10px) scale(.8);transform:translate(-50%,10px) scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;transition:opacity .2s .1s ease,-webkit-transform .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease,-webkit-transform .2s .1s ease;z-index:2}.plyr__tooltip::before{border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid rgba(255,255,255,.9);bottom:-4px;content:'';height:0;left:50%;position:absolute;-webkit-transform:translateX(-50%);transform:translateX(-50%);width:0;z-index:2}.plyr .plyr__control.plyr__tab-focus .plyr__tooltip,.plyr .plyr__control:hover .plyr__tooltip,.plyr__tooltip--visible{opacity:1;-webkit-transform:translate(-50%,0) scale(1);transform:translate(-50%,0) scale(1)}.plyr .plyr__control:hover .plyr__tooltip{z-index:3}.plyr__controls>.plyr__control:first-child .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip{left:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:0 100%;transform-origin:0 100%}.plyr__controls>.plyr__control:first-child .plyr__tooltip::before,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip::before{left:16px}.plyr__controls>.plyr__control:last-child .plyr__tooltip{right:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%}.plyr__controls>.plyr__control:last-child .plyr__tooltip::before{left:auto;right:16px;-webkit-transform:translateX(50%);transform:translateX(50%)}.plyr__controls>.plyr__control:first-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control:hover .plyr__tooltip,.plyr__controls>.plyr__control:first-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child:hover .plyr__tooltip,.plyr__controls>.plyr__control:last-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:last-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:last-child:hover .plyr__tooltip{-webkit-transform:translate(0,0) scale(1);transform:translate(0,0) scale(1)}.plyr--video{overflow:hidden}.plyr--video.plyr--menu-open{overflow:visible}.plyr__video-wrapper{background:#000;border-radius:inherit;overflow:hidden;position:relative;z-index:0}.plyr__volume{-webkit-box-flex:1;flex:1;position:relative}.plyr__volume input[type=range]{position:relative;z-index:2}@media (min-width:480px){.plyr__volume{max-width:50px}}@media (min-width:768px){.plyr__volume{max-width:80px}}.plyr--is-ios .plyr__volume,.plyr--is-ios [data-plyr=mute]{display:none!important}.plyr--has-error{pointer-events:none}.plyr--has-error::after{-webkit-box-align:center;align-items:center;background:#000;color:#fff;content:attr(data-plyr-error);display:-webkit-box;display:flex;font-size:16px;height:100%;-webkit-box-pack:center;justify-content:center;left:0;position:absolute;text-align:center;text-shadow:0 1px 1px #000;top:0;width:100%;z-index:10}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:fullscreen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:fullscreen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}.plyr:-moz-full-screen .plyr__captions{font-size:21px}.plyr:-ms-fullscreen .plyr__captions{font-size:21px}.plyr:fullscreen .plyr__captions{font-size:21px}}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-moz-full-screen .plyr__captions{font-size:21px}}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-ms-fullscreen .plyr__captions{font-size:21px}}.plyr--fullscreen-fallback{background:#000;border-radius:0!important;height:100%;margin:0;width:100%;bottom:0;left:0;position:fixed;right:0;top:0;z-index:10000000}.plyr--fullscreen-fallback video{height:100%}.plyr--fullscreen-fallback .plyr__video-wrapper{height:100%;width:100%}.plyr--fullscreen-fallback .plyr__video-embed{overflow:visible}.plyr--fullscreen-fallback.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen{display:block}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr--fullscreen-fallback .plyr__captions{font-size:21px}}.plyr--no-transition{transition:none!important}.plyr--full-ui [hidden]{display:none}.plyr--full-ui [aria-hidden=true]{display:none}.plyr__sr-only{clip:rect(1px,1px,1px,1px);overflow:hidden;border:0!important;height:1px!important;padding:0!important;position:absolute!important;width:1px!important}
\ No newline at end of file +@-webkit-keyframes plyr-progress{to{background-position:25px 0}}@keyframes plyr-progress{to{background-position:25px 0}}@-webkit-keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@keyframes plyr-popup{0%{opacity:.5;-webkit-transform:translateY(10px);transform:translateY(10px)}to{opacity:1;-webkit-transform:translateY(0);transform:translateY(0)}}@-webkit-keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}@keyframes plyr-fade-in{from{opacity:0}to{opacity:1}}.plyr{-moz-osx-font-smoothing:grayscale;-webkit-font-smoothing:antialiased;direction:ltr;font-family:Avenir,"Avenir Next","Helvetica Neue","Segoe UI",Helvetica,Arial,sans-serif;font-weight:500;line-height:1.7;max-width:100%;min-width:200px;position:relative;text-shadow:none;transition:box-shadow .3s ease}.plyr audio,.plyr video{border-radius:inherit;height:auto;vertical-align:middle;width:100%}.plyr:focus{outline:0}.plyr--full-ui{box-sizing:border-box}.plyr--full-ui *,.plyr--full-ui ::after,.plyr--full-ui ::before{box-sizing:inherit}.plyr--full-ui a,.plyr--full-ui button,.plyr--full-ui input,.plyr--full-ui label{touch-action:manipulation}.plyr__badge{background:#4f5b5f;border-radius:2px;color:rgba(255,255,255,.9);font-size:9px;line-height:1;padding:3px 4px}.plyr--full-ui ::-webkit-media-text-track-container{display:none}.plyr__captions{-webkit-animation:plyr-fade-in .3s ease;animation:plyr-fade-in .3s ease;bottom:0;color:#fff;display:none;font-size:14px;left:0;padding:10px;position:absolute;text-align:center;-webkit-transform:translateY(-40px);transform:translateY(-40px);transition:-webkit-transform .4s ease-in-out;transition:transform .4s ease-in-out;transition:transform .4s ease-in-out,-webkit-transform .4s ease-in-out;width:100%}.plyr__captions span{background:rgba(0,0,0,.8);border-radius:2px;-webkit-box-decoration-break:clone;box-decoration-break:clone;line-height:185%;padding:.2em .5em;white-space:pre-wrap}.plyr__captions span div{display:inline}.plyr__captions span:empty{display:none}@media (min-width:480px){.plyr__captions{font-size:16px;padding:20px}}@media (min-width:768px){.plyr__captions{font-size:18px}}.plyr--captions-active .plyr__captions{display:block}.plyr--hide-controls .plyr__captions{-webkit-transform:translateY(-15px);transform:translateY(-15px)}.plyr__control{background:0 0;border:0;border-radius:3px;color:inherit;cursor:pointer;flex-shrink:0;overflow:visible;padding:7px;position:relative;transition:all .3s ease}.plyr__control svg{display:block;fill:currentColor;height:18px;pointer-events:none;width:18px}.plyr__control:focus{outline:0}.plyr__control.plyr__tab-focus{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__control[aria-pressed=false] .icon--pressed,.plyr__control[aria-pressed=false] .label--pressed,.plyr__control[aria-pressed=true] .icon--not-pressed,.plyr__control[aria-pressed=true] .label--not-pressed{display:none}.plyr--audio .plyr__control.plyr__tab-focus,.plyr--audio .plyr__control:hover,.plyr--audio .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr__control--overlaid{background:rgba(26,175,255,.8);border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15);color:#fff;display:none;left:50%;padding:15px;position:absolute;top:50%;-webkit-transform:translate(-50%,-50%);transform:translate(-50%,-50%);z-index:2}.plyr__control--overlaid svg{height:20px;left:2px;position:relative;width:20px}.plyr__control--overlaid:focus,.plyr__control--overlaid:hover{background:#1aafff}.plyr--playing .plyr__control--overlaid{opacity:0;visibility:hidden}.plyr--full-ui.plyr--video .plyr__control--overlaid{display:block}.plyr--full-ui ::-webkit-media-controls{display:none}.plyr__controls{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;text-align:center}.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:5px}.plyr__controls .plyr__menu:first-child,.plyr__controls .plyr__menu:first-child+[data-plyr=pause],.plyr__controls .plyr__progress:first-child,.plyr__controls .plyr__progress:first-child+[data-plyr=pause],.plyr__controls .plyr__time:first-child,.plyr__controls .plyr__time:first-child+[data-plyr=pause],.plyr__controls>.plyr__control:first-child,.plyr__controls>.plyr__control:first-child+[data-plyr=pause]{margin-left:0}.plyr__controls .plyr__volume{margin-left:5px}@media (min-width:480px){.plyr__controls .plyr__menu,.plyr__controls .plyr__progress,.plyr__controls .plyr__time,.plyr__controls>.plyr__control{margin-left:10px}.plyr__controls .plyr__menu+.plyr__control,.plyr__controls>.plyr__control+.plyr__control,.plyr__controls>.plyr__control+.plyr__menu{margin-left:5px}}.plyr--video .plyr__controls{background:linear-gradient(transparent,rgba(0,0,0,.7));border-bottom-left-radius:inherit;border-bottom-right-radius:inherit;bottom:0;color:#fff;left:0;padding:35px 10px 10px;position:absolute;right:0;transition:opacity .4s ease-in-out,-webkit-transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out;transition:opacity .4s ease-in-out,transform .4s ease-in-out,-webkit-transform .4s ease-in-out;z-index:2}.plyr--video .plyr__controls .plyr__control svg{-webkit-filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15));filter:drop-shadow(0 1px 1px rgba(0, 0, 0, .15))}.plyr--video .plyr__controls .plyr__control.plyr__tab-focus,.plyr--video .plyr__controls .plyr__control:hover,.plyr--video .plyr__controls .plyr__control[aria-expanded=true]{background:#1aafff;color:#fff}.plyr--audio .plyr__controls{background:#fff;border-radius:inherit;color:#4f5b5f;padding:10px}.plyr--video.plyr--hide-controls .plyr__controls{opacity:0;pointer-events:none;-webkit-transform:translateY(100%);transform:translateY(100%)}.plyr [data-plyr=airplay],.plyr [data-plyr=captions],.plyr [data-plyr=fullscreen],.plyr [data-plyr=pip]{display:none}.plyr--airplay-supported [data-plyr=airplay],.plyr--captions-enabled [data-plyr=captions],.plyr--fullscreen-enabled [data-plyr=fullscreen],.plyr--pip-supported [data-plyr=pip]{display:inline-block}.plyr__video-embed{height:0;padding-bottom:56.25%;position:relative}.plyr__video-embed iframe{border:0;height:100%;left:0;position:absolute;top:0;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__video-embed>div{padding-bottom:200%;position:relative;-webkit-transform:translateY(-35.9375%);transform:translateY(-35.9375%)}.plyr--full-ui .plyr__video-embed iframe{pointer-events:none}.plyr__menu{display:-webkit-box;display:flex;position:relative}.plyr__menu .plyr__control svg{transition:-webkit-transform .3s ease;transition:transform .3s ease;transition:transform .3s ease,-webkit-transform .3s ease}.plyr__menu .plyr__control[aria-expanded=true] svg{-webkit-transform:rotate(90deg);transform:rotate(90deg)}.plyr__menu .plyr__control[aria-expanded=true] .plyr__tooltip{display:none}.plyr__menu__container{-webkit-animation:plyr-popup .2s ease;animation:plyr-popup .2s ease;background:rgba(255,255,255,.9);border-radius:4px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:16px;margin-bottom:10px;position:absolute;right:-3px;text-align:left;white-space:nowrap;z-index:1}.plyr__menu__container>div{overflow:hidden;transition:height .35s cubic-bezier(.4,0,.2,1),width .35s cubic-bezier(.4,0,.2,1)}.plyr__menu__container::after{border:4px solid transparent;border-top-color:rgba(255,255,255,.9);content:'';height:0;position:absolute;right:15px;top:100%;width:0}.plyr__menu__container ul{list-style:none;margin:0;overflow:hidden;padding:7px}.plyr__menu__container .plyr__control{-webkit-box-align:center;align-items:center;color:#4f5b5f;display:-webkit-box;display:flex;padding:4px 14px;-webkit-user-select:none;-moz-user-select:none;-ms-user-select:none;user-select:none;width:100%}.plyr__menu__container .plyr__control::after{border:4px solid transparent;content:'';position:absolute;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%);transition:border-color .2s ease}.plyr__menu__container .plyr__control--forward{padding-right:28px}.plyr__menu__container .plyr__control--forward::after{border-left-color:rgba(79,91,95,.8);right:5px}.plyr__menu__container .plyr__control--forward.plyr__tab-focus::after,.plyr__menu__container .plyr__control--forward:hover::after{border-left-color:currentColor}.plyr__menu__container .plyr__control--back{font-weight:500;margin:7px;margin-bottom:3px;padding-left:28px;position:relative;width:calc(100% - 14px)}.plyr__menu__container .plyr__control--back::after{border-right-color:rgba(79,91,95,.8);left:7px}.plyr__menu__container .plyr__control--back::before{background:#b7c5cd;box-shadow:0 1px 0 #fff;content:'';height:1px;left:0;margin-top:4px;overflow:hidden;position:absolute;right:0;top:100%}.plyr__menu__container .plyr__control--back.plyr__tab-focus::after,.plyr__menu__container .plyr__control--back:hover::after{border-right-color:currentColor}.plyr__menu__container label.plyr__control{padding-left:7px}.plyr__menu__container label.plyr__control input[type=radio]+span{background:rgba(0,0,0,.1);border-radius:100%;display:block;flex-shrink:0;height:16px;margin-right:10px;position:relative;transition:all .3s ease;width:16px}.plyr__menu__container label.plyr__control input[type=radio]+span::after{background:#fff;border-radius:100%;content:'';height:6px;left:5px;opacity:0;position:absolute;top:5px;-webkit-transform:scale(0);transform:scale(0);transition:opacity .3s ease,-webkit-transform .3s ease;transition:transform .3s ease,opacity .3s ease;transition:transform .3s ease,opacity .3s ease,-webkit-transform .3s ease;width:6px}.plyr__menu__container label.plyr__control input[type=radio]:checked+span{background:#1aafff}.plyr__menu__container label.plyr__control input[type=radio]:checked+span::after{opacity:1;-webkit-transform:scale(1);transform:scale(1)}.plyr__menu__container label.plyr__control input[type=radio]:focus+span{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr__menu__container label.plyr__control.plyr__tab-focus input[type=radio]+span,.plyr__menu__container label.plyr__control:hover input[type=radio]+span{background:rgba(0,0,0,.1)}.plyr__menu__container .plyr__menu__value{-webkit-box-align:center;align-items:center;display:-webkit-box;display:flex;margin-left:auto;margin-right:-7px;overflow:hidden;padding-left:25px;pointer-events:none}.plyr__progress{display:-webkit-box;display:flex;-webkit-box-flex:1;flex:1;position:relative}.plyr__progress input[type=range]{position:relative;z-index:2}.plyr__progress .plyr__tooltip{font-size:14px;left:0}.plyr__progress--buffer{-webkit-appearance:none;background:0 0;border:0;border-radius:100px;height:6px;left:0;margin:-3px 0 0;padding:0;position:absolute;top:50%;width:100%}.plyr__progress--buffer::-webkit-progress-bar{background:0 0;transition:width .2s ease}.plyr__progress--buffer::-webkit-progress-value{background:currentColor;border-radius:100px;min-width:6px}.plyr__progress--buffer::-moz-progress-bar{background:currentColor;border-radius:100px;min-width:6px;transition:width .2s ease}.plyr__progress--buffer::-ms-fill{border-radius:100px;transition:width .2s ease}.plyr--video .plyr__progress--buffer{box-shadow:0 1px 1px rgba(0,0,0,.15);color:rgba(255,255,255,.25)}.plyr--audio .plyr__progress--buffer{color:rgba(183,197,205,.66)}.plyr--loading .plyr__progress--buffer{-webkit-animation:plyr-progress 1s linear infinite;animation:plyr-progress 1s linear infinite;background-image:linear-gradient(-45deg,rgba(47,52,61,.6) 25%,transparent 25%,transparent 50%,rgba(47,52,61,.6) 50%,rgba(47,52,61,.6) 75%,transparent 75%,transparent);background-repeat:repeat-x;background-size:25px 25px;color:transparent}.plyr--video.plyr--loading .plyr__progress--buffer{background-color:rgba(255,255,255,.25)}.plyr--audio.plyr--loading .plyr__progress--buffer{background-color:rgba(183,197,205,.66)}.plyr--full-ui input[type=range]{-webkit-appearance:none;background:0 0;border:0;border-radius:28px;color:#1aafff;display:block;height:20px;margin:0;padding:0;transition:box-shadow .3s ease;width:100%}.plyr--full-ui input[type=range]::-webkit-slider-runnable-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-webkit-user-select:none;user-select:none;background-image:linear-gradient(to right,currentColor var(--value),transparent var(--value))}.plyr--full-ui input[type=range]::-webkit-slider-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;-webkit-appearance:none;margin-top:-4px}.plyr--full-ui input[type=range]::-moz-range-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-moz-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-moz-range-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px}.plyr--full-ui input[type=range]::-moz-range-progress{background:currentColor;border-radius:3px;height:6px}.plyr--full-ui input[type=range]::-ms-track{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;color:transparent}.plyr--full-ui input[type=range]::-ms-fill-upper{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none}.plyr--full-ui input[type=range]::-ms-fill-lower{background:0 0;border:0;border-radius:3px;height:6px;transition:all .3s ease;-ms-user-select:none;user-select:none;background:currentColor}.plyr--full-ui input[type=range]::-ms-thumb{background:#fff;border:0;border-radius:100%;box-shadow:0 1px 1px rgba(0,0,0,.15),0 0 0 1px rgba(47,52,61,.2);box-sizing:border-box;height:14px;position:relative;transition:all .2s ease;width:14px;margin-top:0}.plyr--full-ui input[type=range]::-ms-tooltip{display:none}.plyr--full-ui input[type=range]:focus{outline:0}.plyr--full-ui input[type=range]::-moz-focus-outer{border:0}.plyr--full-ui input[type=range].plyr__tab-focus::-webkit-slider-runnable-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-moz-range-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range].plyr__tab-focus::-ms-track{box-shadow:0 0 0 3px rgba(26,175,255,.35);outline:0}.plyr--full-ui input[type=range]:active::-webkit-slider-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-moz-range-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui input[type=range]:active::-ms-thumb{box-shadow:0 0 0 3px rgba(255,255,255,.5)}.plyr--full-ui.plyr--video input[type=range]::-webkit-slider-runnable-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-moz-range-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--video input[type=range]::-ms-track{background-color:rgba(255,255,255,.25)}.plyr--full-ui.plyr--audio input[type=range]::-webkit-slider-runnable-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-moz-range-track{background-color:rgba(183,197,205,.66)}.plyr--full-ui.plyr--audio input[type=range]::-ms-track{background-color:rgba(183,197,205,.66)}.plyr__time{font-size:14px}.plyr__time+.plyr__time::before{content:'\2044';margin-right:10px}@media (max-width:767px){.plyr__time+.plyr__time{display:none}}.plyr--video .plyr__time{text-shadow:0 1px 1px rgba(0,0,0,.15)}.plyr__tooltip{background:rgba(255,255,255,.9);border-radius:3px;bottom:100%;box-shadow:0 1px 2px rgba(0,0,0,.15);color:#4f5b5f;font-size:14px;font-weight:500;line-height:1.3;margin-bottom:10px;opacity:0;padding:5px 7.5px;pointer-events:none;position:absolute;-webkit-transform:translate(-50%,10px) scale(.8);transform:translate(-50%,10px) scale(.8);-webkit-transform-origin:50% 100%;transform-origin:50% 100%;transition:opacity .2s .1s ease,-webkit-transform .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease;transition:transform .2s .1s ease,opacity .2s .1s ease,-webkit-transform .2s .1s ease;z-index:2}.plyr__tooltip::before{border-left:4px solid transparent;border-right:4px solid transparent;border-top:4px solid rgba(255,255,255,.9);bottom:-4px;content:'';height:0;left:50%;position:absolute;-webkit-transform:translateX(-50%);transform:translateX(-50%);width:0;z-index:2}.plyr .plyr__control.plyr__tab-focus .plyr__tooltip,.plyr .plyr__control:hover .plyr__tooltip,.plyr__tooltip--visible{opacity:1;-webkit-transform:translate(-50%,0) scale(1);transform:translate(-50%,0) scale(1)}.plyr .plyr__control:hover .plyr__tooltip{z-index:3}.plyr__controls>.plyr__control:first-child .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip{left:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:0 100%;transform-origin:0 100%}.plyr__controls>.plyr__control:first-child .plyr__tooltip::before,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip::before{left:16px}.plyr__controls>.plyr__control:last-child .plyr__tooltip{right:0;-webkit-transform:translate(0,10px) scale(.8);transform:translate(0,10px) scale(.8);-webkit-transform-origin:100% 100%;transform-origin:100% 100%}.plyr__controls>.plyr__control:last-child .plyr__tooltip::before{left:auto;right:16px;-webkit-transform:translateX(50%);transform:translateX(50%)}.plyr__controls>.plyr__control:first-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control .plyr__tooltip--visible,.plyr__controls>.plyr__control:first-child+.plyr__control.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child+.plyr__control:hover .plyr__tooltip,.plyr__controls>.plyr__control:first-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:first-child:hover .plyr__tooltip,.plyr__controls>.plyr__control:last-child .plyr__tooltip--visible,.plyr__controls>.plyr__control:last-child.plyr__tab-focus .plyr__tooltip,.plyr__controls>.plyr__control:last-child:hover .plyr__tooltip{-webkit-transform:translate(0,0) scale(1);transform:translate(0,0) scale(1)}.plyr--video{overflow:hidden}.plyr--video.plyr--menu-open{overflow:visible}.plyr__video-wrapper{background:#000;border-radius:inherit;overflow:hidden;position:relative;z-index:0}.plyr__volume{-webkit-box-flex:1;flex:1;position:relative}.plyr__volume input[type=range]{position:relative;z-index:2}@media (min-width:480px){.plyr__volume{max-width:50px}}@media (min-width:768px){.plyr__volume{max-width:80px}}.plyr--is-ios .plyr__volume,.plyr--is-ios [data-plyr=mute]{display:none!important}.plyr--has-error{pointer-events:none}.plyr--has-error::after{-webkit-box-align:center;align-items:center;background:#000;color:#fff;content:attr(data-plyr-error);display:-webkit-box;display:flex;font-size:16px;height:100%;-webkit-box-pack:center;justify-content:center;left:0;position:absolute;text-align:center;text-shadow:0 1px 1px #000;top:0;width:100%;z-index:10}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:fullscreen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:fullscreen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}.plyr:fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}.plyr:-moz-full-screen .plyr__captions{font-size:21px}.plyr:-ms-fullscreen .plyr__captions{font-size:21px}.plyr:fullscreen .plyr__captions{font-size:21px}}.plyr:-webkit-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-webkit-full-screen video{height:100%}.plyr:-webkit-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-webkit-full-screen .plyr__video-embed{overflow:visible}.plyr:-webkit-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-webkit-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-webkit-full-screen .plyr__captions{font-size:21px}}.plyr:-moz-full-screen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-moz-full-screen video{height:100%}.plyr:-moz-full-screen .plyr__video-wrapper{height:100%;width:100%}.plyr:-moz-full-screen .plyr__video-embed{overflow:visible}.plyr:-moz-full-screen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-moz-full-screen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-moz-full-screen .plyr__captions{font-size:21px}}.plyr:-ms-fullscreen{background:#000;border-radius:0!important;height:100%;margin:0;width:100%}.plyr:-ms-fullscreen video{height:100%}.plyr:-ms-fullscreen .plyr__video-wrapper{height:100%;width:100%}.plyr:-ms-fullscreen .plyr__video-embed{overflow:visible}.plyr:-ms-fullscreen.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;transform:translateY(-50%)}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen{display:block}.plyr:-ms-fullscreen .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr:-ms-fullscreen .plyr__captions{font-size:21px}}.plyr--fullscreen-fallback{background:#000;border-radius:0!important;height:100%;margin:0;width:100%;bottom:0;left:0;position:fixed;right:0;top:0;z-index:10000000}.plyr--fullscreen-fallback video{height:100%}.plyr--fullscreen-fallback .plyr__video-wrapper{height:100%;width:100%}.plyr--fullscreen-fallback .plyr__video-embed{overflow:visible}.plyr--fullscreen-fallback.plyr--vimeo .plyr__video-wrapper{height:0;top:50%;-webkit-transform:translateY(-50%);transform:translateY(-50%)}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen{display:block}.plyr--fullscreen-fallback .plyr__control .icon--exit-fullscreen+svg{display:none}@media (min-width:1024px){.plyr--fullscreen-fallback .plyr__captions{font-size:21px}}.plyr--no-transition{transition:none!important}.plyr--full-ui [hidden]{display:none}.plyr--full-ui [aria-hidden=true]{display:none}.plyr__sr-only{clip:rect(1px,1px,1px,1px);overflow:hidden;border:0!important;height:1px!important;padding:0!important;position:absolute!important;width:1px!important}
\ No newline at end of file diff --git a/dist/plyr.js b/dist/plyr.js index 8575cdc9..9623bac1 100644 --- a/dist/plyr.js +++ b/dist/plyr.js @@ -1,3 +1,3 @@ -!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define("Plyr",t):e.Plyr=t()}(this,function(){"use strict";var e,t,i,n,s={html5:"html5",youtube:"youtube",vimeo:"vimeo"},a={audio:"audio",video:"video"},o={enabled:!0,title:"",debug:!1,autoplay:!1,autopause:!0,seekTime:10,volume:1,muted:!1,duration:null,displayDuration:!0,invertTime:!0,toggleInvert:!0,ratio:"16:9",clickToPlay:!0,hideControls:!0,showPosterOnEnd:!1,disableContextMenu:!0,loadSprite:!0,iconPrefix:"plyr",iconUrl:"https://cdn.plyr.io/2.0.10/plyr.svg",blankVideo:"https://cdn.plyr.io/static/blank.mp4",quality:{default:"default",options:["hd2160","hd1440","hd1080","hd720","large","medium","small","tiny","default"]},loop:{active:!1},speed:{selected:1,options:[.5,.75,1,1.25,1.5,1.75,2]},keyboard:{focused:!0,global:!1},tooltips:{controls:!1,seek:!0},captions:{active:!1,language:window.navigator.language.split("-")[0]},fullscreen:{enabled:!0,fallback:!0},storage:{enabled:!0,key:"plyr"},controls:["play-large","play","progress","current-time","mute","volume","captions","settings","pip","airplay","fullscreen"],settings:["captions","quality","speed"],i18n:{restart:"Restart",rewind:"Rewind {seektime} secs",play:"Play",pause:"Pause",forward:"Forward {seektime} secs",seek:"Seek",played:"Played",buffered:"Buffered",currentTime:"Current time",duration:"Duration",volume:"Volume",mute:"Mute",unmute:"Unmute",enableCaptions:"Enable captions",disableCaptions:"Disable captions",enterFullscreen:"Enter fullscreen",exitFullscreen:"Exit fullscreen",frameTitle:"Player for {title}",captions:"Captions",settings:"Settings",speed:"Speed",quality:"Quality",loop:"Loop",start:"Start",end:"End",all:"All",reset:"Reset",none:"None",disabled:"Disabled"},urls:{vimeo:{api:"https://player.vimeo.com/api/player.js"},youtube:{api:"https://www.youtube.com/iframe_api"}},listeners:{seek:null,play:null,pause:null,restart:null,rewind:null,forward:null,mute:null,volume:null,captions:null,fullscreen:null,pip:null,airplay:null,speed:null,quality:null,loop:null,language:null},events:["ended","progress","stalled","playing","waiting","canplay","canplaythrough","loadstart","loadeddata","loadedmetadata","timeupdate","volumechange","play","pause","error","seeking","seeked","emptied","ratechange","cuechange","enterfullscreen","exitfullscreen","captionsenabled","captionsdisabled","languagechange","controlshidden","controlsshown","ready","statechange","qualitychange","qualityrequested"],selectors:{editable:"input, textarea, select, [contenteditable]",container:".plyr",controls:{container:null,wrapper:".plyr__controls"},labels:"[data-plyr]",buttons:{play:'[data-plyr="play"]',pause:'[data-plyr="pause"]',restart:'[data-plyr="restart"]',rewind:'[data-plyr="rewind"]',forward:'[data-plyr="fast-forward"]',mute:'[data-plyr="mute"]',captions:'[data-plyr="captions"]',fullscreen:'[data-plyr="fullscreen"]',pip:'[data-plyr="pip"]',airplay:'[data-plyr="airplay"]',settings:'[data-plyr="settings"]',loop:'[data-plyr="loop"]'},inputs:{seek:'[data-plyr="seek"]',volume:'[data-plyr="volume"]',speed:'[data-plyr="speed"]',language:'[data-plyr="language"]',quality:'[data-plyr="quality"]'},display:{currentTime:".plyr__time--current",duration:".plyr__time--duration",buffer:".plyr__progress--buffer",played:".plyr__progress--played",loop:".plyr__progress--loop",volume:".plyr__volume--display"},progress:".plyr__progress",captions:".plyr__captions",menu:{quality:".js-plyr__menu__list--quality"}},classNames:{video:"plyr__video-wrapper",embed:"plyr__video-embed",control:"plyr__control",type:"plyr--{0}",provider:"plyr--{0}",stopped:"plyr--stopped",playing:"plyr--playing",loading:"plyr--loading",error:"plyr--has-error",hover:"plyr--hover",tooltip:"plyr__tooltip",hidden:"plyr__sr-only",hideControls:"plyr--hide-controls",isIos:"plyr--is-ios",isTouch:"plyr--is-touch",uiSupported:"plyr--full-ui",noTransition:"plyr--no-transition",menu:{value:"plyr__menu__value",badge:"plyr__badge",open:"plyr--menu-open"},captions:{enabled:"plyr--captions-enabled",active:"plyr--captions-active"},fullscreen:{enabled:"plyr--fullscreen-enabled",fallback:"plyr--fullscreen-fallback"},pip:{supported:"plyr--pip-supported",active:"plyr--pip-active"},airplay:{supported:"plyr--airplay-supported",active:"plyr--airplay-active"},tabFocus:"plyr__tab-focus"},keys:{google:null}},l=(function(){function e(e){this.value=e}function t(t){var i,n;function s(i,n){try{var o=t[i](n),l=o.value;l instanceof e?Promise.resolve(l.value).then(function(e){s("next",e)},function(e){s("throw",e)}):a(o.done?"return":"normal",o.value)}catch(e){a("throw",e)}}function a(e,t){switch(e){case"return":i.resolve({value:t,done:!0});break;case"throw":i.reject(t);break;default:i.resolve({value:t,done:!1})}(i=i.next)?s(i.key,i.arg):n=null}this._invoke=function(e,t){return new Promise(function(a,o){var l={key:e,arg:t,resolve:a,reject:o,next:null};n?n=n.next=l:(i=n=l,s(e,t))})},"function"!=typeof t.return&&(this.return=void 0)}"function"==typeof Symbol&&Symbol.asyncIterator&&(t.prototype[Symbol.asyncIterator]=function(){return this}),t.prototype.next=function(e){return this._invoke("next",e)},t.prototype.throw=function(e){return this._invoke("throw",e)},t.prototype.return=function(e){return this._invoke("return",e)}}(),function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}),r=function(){function e(e,t){for(var i=0;i<t.length;i++){var n=t[i];n.enumerable=n.enumerable||!1,n.configurable=!0,"value"in n&&(n.writable=!0),Object.defineProperty(e,n.key,n)}}return function(t,i,n){return i&&e(t.prototype,i),n&&e(t,n),t}}(),c=function(e,t,i){return t in e?Object.defineProperty(e,t,{value:i,enumerable:!0,configurable:!0,writable:!0}):e[t]=i,e},u={is:{plyr:function(e){return this.instanceof(e,Plyr)},object:function(e){return this.getConstructor(e)===Object},number:function(e){return this.getConstructor(e)===Number&&!Number.isNaN(e)},string:function(e){return this.getConstructor(e)===String},boolean:function(e){return this.getConstructor(e)===Boolean},function:function(e){return this.getConstructor(e)===Function},array:function(e){return!this.nullOrUndefined(e)&&Array.isArray(e)},weakMap:function(e){return this.instanceof(e,window.WeakMap)},nodeList:function(e){return this.instanceof(e,window.NodeList)},element:function(e){return this.instanceof(e,window.Element)},textNode:function(e){return this.getConstructor(e)===Text},event:function(e){return this.instanceof(e,window.Event)},cue:function(e){return this.instanceof(e,window.TextTrackCue)||this.instanceof(e,window.VTTCue)},track:function(e){return this.instanceof(e,TextTrack)||!this.nullOrUndefined(e)&&this.string(e.kind)},nullOrUndefined:function(e){return null===e||void 0===e},empty:function(e){return this.nullOrUndefined(e)||(this.string(e)||this.array(e)||this.nodeList(e))&&!e.length||this.object(e)&&!Object.keys(e).length},instanceof:function(e,t){return Boolean(e&&t&&e instanceof t)},getConstructor:function(e){return this.nullOrUndefined(e)?null:e.constructor}},getBrowser:function(){return{isIE:!!document.documentMode,isWebkit:"WebkitAppearance"in document.documentElement.style&&!/Edge/.test(navigator.userAgent),isIPhone:/(iPhone|iPod)/gi.test(navigator.platform),isIos:/(iPad|iPhone|iPod)/gi.test(navigator.platform)}},loadScript:function(e,t){var i=document.querySelector('script[src="'+e+'"]');if(null!==i)return i.callbacks=i.callbacks||[],void i.callbacks.push(t);var n=document.createElement("script");n.callbacks=n.callbacks||[],n.callbacks.push(t),u.is.function(t)&&n.addEventListener("load",function(e){n.callbacks.forEach(function(t){return t.call(null,e)}),n.callbacks=null},!1),n.src=e;var s=document.getElementsByTagName("script")[0];s.parentNode.insertBefore(n,s)},loadSprite:function(e,t){if(u.is.string(e)){var i=u.is.string(t);if(!i||!document.querySelectorAll("#"+t).length){var n=document.createElement("div");if(u.toggleHidden(n,!0),i&&n.setAttribute("id",t),d.storage){var s=window.localStorage.getItem("cache-"+t);if(null!==s){var a=JSON.parse(s);return void o.call(n,a.content)}}fetch(e).then(function(e){return e.ok?e.text():null}).then(function(e){null!==e&&(d.storage&&window.localStorage.setItem("cache-"+t,JSON.stringify({content:e})),o.call(n,e))}).catch(function(){})}}function o(e){this.innerHTML=e,document.body.insertBefore(this,document.body.childNodes[0])}},generateId:function(e){return e+"-"+Math.floor(1e4*Math.random())},inFrame:function(){try{return window.self!==window.top}catch(e){return!0}},wrap:function(e,t){var i=e.length?e:[e];Array.from(i).reverse().forEach(function(e,i){var n=i>0?t.cloneNode(!0):t,s=e.parentNode,a=e.nextSibling;n.appendChild(e),a?s.insertBefore(n,a):s.appendChild(n)})},createElement:function(e,t,i){var n=document.createElement(e);return u.is.object(t)&&u.setAttributes(n,t),u.is.string(i)&&(n.textContent=i),n},insertAfter:function(e,t){t.parentNode.insertBefore(e,t.nextSibling)},insertElement:function(e,t,i,n){t.appendChild(u.createElement(e,i,n))},removeElement:function(e){return u.is.element(e)&&u.is.element(e.parentNode)?(e.parentNode.removeChild(e),e):null},emptyElement:function(e){for(var t=e.childNodes.length;t>0;)e.removeChild(e.lastChild),t-=1},setAttributes:function(e,t){u.is.element(e)&&!u.is.empty(t)&&Object.keys(t).forEach(function(i){e.setAttribute(i,t[i])})},getAttributesFromSelector:function(e,t){if(!u.is.string(e)||u.is.empty(e))return{};var i={},n=t;return e.split(",").forEach(function(e){var t=e.trim(),s=t.replace(".",""),a=t.replace(/[[\]]/g,"").split("="),o=a[0],l=a.length>1?a[1].replace(/["']/g,""):"";switch(t.charAt(0)){case".":u.is.object(n)&&u.is.string(n.class)&&(n.class+=" "+s),i.class=s;break;case"#":i.id=t.replace("#","");break;case"[":i[o]=l}}),i},toggleClass:function(e,t,i){if(u.is.element(e)){var n=e.classList.contains(t);return e.classList[i?"add":"remove"](t),i&&!n||!i&&n}return null},hasClass:function(e,t){return u.is.element(e)&&e.classList.contains(t)},toggleHidden:function(e,t){u.is.element(e)&&(t?e.setAttribute("hidden",""):e.removeAttribute("hidden"))},matches:function(e,t){var i={Element:Element};var n=i.matches||i.webkitMatchesSelector||i.mozMatchesSelector||i.msMatchesSelector||function(){return Array.from(document.querySelectorAll(t)).includes(this)};return n.call(e,t)},getElements:function(e){return this.elements.container.querySelectorAll(e)},getElement:function(e){return this.elements.container.querySelector(e)},findElements:function(){try{return this.elements.controls=u.getElement.call(this,this.config.selectors.controls.wrapper),this.elements.buttons={play:u.getElements.call(this,this.config.selectors.buttons.play),pause:u.getElement.call(this,this.config.selectors.buttons.pause),restart:u.getElement.call(this,this.config.selectors.buttons.restart),rewind:u.getElement.call(this,this.config.selectors.buttons.rewind),forward:u.getElement.call(this,this.config.selectors.buttons.forward),mute:u.getElement.call(this,this.config.selectors.buttons.mute),pip:u.getElement.call(this,this.config.selectors.buttons.pip),airplay:u.getElement.call(this,this.config.selectors.buttons.airplay),settings:u.getElement.call(this,this.config.selectors.buttons.settings),captions:u.getElement.call(this,this.config.selectors.buttons.captions),fullscreen:u.getElement.call(this,this.config.selectors.buttons.fullscreen)},this.elements.progress=u.getElement.call(this,this.config.selectors.progress),this.elements.inputs={seek:u.getElement.call(this,this.config.selectors.inputs.seek),volume:u.getElement.call(this,this.config.selectors.inputs.volume)},this.elements.display={buffer:u.getElement.call(this,this.config.selectors.display.buffer),duration:u.getElement.call(this,this.config.selectors.display.duration),currentTime:u.getElement.call(this,this.config.selectors.display.currentTime)},u.is.element(this.elements.progress)&&(this.elements.display.seekTooltip=this.elements.progress.querySelector("."+this.config.classNames.tooltip)),!0}catch(e){return this.debug.warn("It looks like there is a problem with your custom controls HTML",e),this.toggleNativeControls(!0),!1}},getFocusElement:function(){var e=document.activeElement;return e=e&&e!==document.body?document.querySelector(":focus"):null},trapFocus:function(){var e=this,t=u.getElements.call(this,"button:not(:disabled), input:not(:disabled), [tabindex]"),i=t[0],n=t[t.length-1];u.on(this.elements.container,"keydown",function(t){if("Tab"===t.key&&9===t.keyCode&&e.fullscreen.active){var s=u.getFocusElement();s!==n||t.shiftKey?s===i&&t.shiftKey&&(n.focus(),t.preventDefault()):(i.focus(),t.preventDefault())}},!1)},toggleListener:function(e,t,i,n,s,a){if(!u.is.nullOrUndefined(e))if(u.is.nodeList(e))Array.from(e).forEach(function(e){e instanceof Node&&u.toggleListener.call(null,e,t,i,n,s,a)});else{var o=t.split(" "),l=!!u.is.boolean(a)&&a;d.passiveListeners&&(l={passive:!u.is.boolean(s)||s,capture:!!u.is.boolean(a)&&a}),o.forEach(function(t){e[n?"addEventListener":"removeEventListener"](t,i,l)})}},on:function(e,t,i,n,s){u.toggleListener(e,t,i,!0,n,s)},off:function(e,t,i,n,s){u.toggleListener(e,t,i,!1,n,s)},dispatchEvent:function(e,t,i,n){if(e&&t){var s=new CustomEvent(t,{bubbles:!!u.is.boolean(i)&&i,detail:Object.assign({},n,{plyr:this instanceof Plyr?this:null})});e.dispatchEvent(s)}},toggleState:function(e,t){if(u.is.element(e)){var i="true"===e.getAttribute("aria-pressed"),n=u.is.boolean(t)?t:!i;e.setAttribute("aria-pressed",n)}},getPercentage:function(e,t){return 0===e||0===t||Number.isNaN(e)||Number.isNaN(t)?0:(e/t*100).toFixed(2)},extend:function(){for(var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments.length,i=Array(t>1?t-1:0),n=1;n<t;n++)i[n-1]=arguments[n];if(!i.length)return e;var s=i.shift();return u.is.object(s)?(Object.keys(s).forEach(function(t){u.is.object(s[t])?(Object.keys(e).includes(t)||Object.assign(e,c({},t,{})),u.extend(e[t],s[t])):Object.assign(e,c({},t,s[t]))}),u.extend.apply(u,[e].concat(function(e){if(Array.isArray(e)){for(var t=0,i=Array(e.length);t<e.length;t++)i[t]=e[t];return i}return Array.from(e)}(i)))):e},parseYouTubeId:function(e){return e.match(/^.*(youtu.be\/|v\/|u\/\w\/|embed\/|watch\?v=|&v=)([^#&?]*).*/)?RegExp.$2:e},parseVimeoId:function(e){if(u.is.number(Number(e)))return e;return e.match(/^.*(vimeo.com\/|video\/)(\d+).*/)?RegExp.$2:e},buildUrlParameters:function(e){return u.is.object(e)?Object.keys(e).map(function(t){return encodeURIComponent(t)+"="+encodeURIComponent(e[t])}).join("&"):""},stripHTML:function(e){var t=document.createDocumentFragment(),i=document.createElement("div");return t.appendChild(i),i.innerHTML=e,t.firstChild.innerText},getAspectRatio:function(e,t){var i=function e(t,i){return 0===i?t:e(i,t%i)}(e,t);return e/i+":"+t/i},transitionEnd:(e=document.createElement("span"),t=Object.keys({WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"}).find(function(t){return void 0!==e.style[t]}),"string"==typeof t&&t),repaint:function(e){window.setTimeout(function(){e.setAttribute("hidden",""),e.offsetHeight,e.removeAttribute("hidden")},0)}},d={audio:"canPlayType"in document.createElement("audio"),video:"canPlayType"in document.createElement("video"),check:function(e,t,i){var n=!1,s=!1,a=u.getBrowser(),o=a.isIPhone&&i&&d.inline;switch(t+":"+e){case"html5:video":s=(n=d.video)&&d.rangeInput&&(!a.isIPhone||o);break;case"html5:audio":s=(n=d.audio)&&d.rangeInput;break;case"youtube:video":n=!0,s=d.rangeInput&&(!a.isIPhone||o);break;case"vimeo:video":n=!0,s=d.rangeInput&&!a.isIPhone;break;default:s=(n=d.audio&&d.video)&&d.rangeInput}return{api:n,ui:s}},pip:!u.getBrowser().isIPhone&&u.is.function(u.createElement("video").webkitSetPresentationMode),airplay:u.is.function(window.WebKitPlaybackTargetAvailabilityEvent),inline:"playsInline"in document.createElement("video"),mime:function(e){var t=this.media;try{if(!this.isHTML5||!u.is.function(t.canPlayType))return!1;if(this.isVideo)switch(e){case"video/webm":return t.canPlayType('video/webm; codecs="vp8, vorbis"').replace(/no/,"");case"video/mp4":return t.canPlayType('video/mp4; codecs="avc1.42E01E, mp4a.40.2"').replace(/no/,"");case"video/ogg":return t.canPlayType('video/ogg; codecs="theora"').replace(/no/,"");default:return!1}else if(this.isAudio)switch(e){case"audio/mpeg":return t.canPlayType("audio/mpeg;").replace(/no/,"");case"audio/ogg":return t.canPlayType('audio/ogg; codecs="vorbis"').replace(/no/,"");case"audio/wav":return t.canPlayType('audio/wav; codecs="1"').replace(/no/,"");default:return!1}}catch(e){return!1}return!1},textTracks:"textTracks"in document.createElement("video"),passiveListeners:function(){var e=!1;try{var t=Object.defineProperty({},"passive",{get:function(){return e=!0,null}});window.addEventListener("test",null,t)}catch(e){}return e}(),rangeInput:(i=document.createElement("input"),i.type="range","range"===i.type),touch:"ontouchstart"in document.documentElement,transitions:!1!==u.transitionEnd,reducedMotion:"matchMedia"in window&&window.matchMedia("(prefers-reduced-motion)").matches},p=function(){},h=function(){function e(t){l(this,e),this.enabled=window.console&&t.config.debug,this.enabled&&this.log("Debugging enabled")}return r(e,[{key:"log",get:function(){return this.enabled?Function.prototype.bind.call(console.log,console):p}},{key:"warn",get:function(){return this.enabled?Function.prototype.bind.call(console.warn,console):p}},{key:"error",get:function(){return this.enabled?Function.prototype.bind.call(console.error,console):p}}]),e}(),m=function(){function e(t){l(this,e),this.enabled=t.config.storage.enabled,this.key=t.config.storage.key}return r(e,[{key:"get",value:function(t){var i=window.localStorage.getItem(this.key);if(!e.supported||u.is.empty(i))return null;var n=JSON.parse(i);return u.is.string(t)&&t.length?n[t]:n}},{key:"set",value:function(t){if(e.supported&&this.enabled&&u.is.object(t)){var i=this.get();u.is.empty(i)&&(i={}),u.extend(i,t),window.localStorage.setItem(this.key,JSON.stringify(i))}}}],[{key:"supported",get:function(){if(!("localStorage"in window))return!1;try{return window.localStorage.setItem("___test","___test"),window.localStorage.removeItem("___test"),!0}catch(e){return!1}}}]),e}(),g=(n=!1,u.is.function(document.cancelFullScreen)?n="":["webkit","o","moz","ms","khtml"].some(function(e){return u.is.function(document[e+"CancelFullScreen"])?(n=e,!0):!(!u.is.function(document.msExitFullscreen)||!document.msFullscreenEnabled||(n="ms",0))}),n),f={prefix:g,enabled:document.fullscreenEnabled||document.webkitFullscreenEnabled||document.mozFullScreenEnabled||document.msFullscreenEnabled,eventType:"ms"===g?"MSFullscreenChange":g+"fullscreenchange",isFullScreen:function(e){if(!f.enabled)return!1;var t=u.is.nullOrUndefined(e)?document.body:e;switch(g){case"":return document.fullscreenElement===t;case"moz":return document.mozFullScreenElement===t;default:return document[g+"FullscreenElement"]===t}},requestFullScreen:function(e){if(!f.enabled)return!1;var t=u.is.nullOrUndefined(e)?document.body:e;return g.length?t[g+("ms"===g?"RequestFullscreen":"RequestFullScreen")]():t.requestFullScreen()},cancelFullScreen:function(){return!!f.enabled&&(g.length?document[g+("ms"===g?"ExitFullscreen":"CancelFullScreen")]():document.cancelFullScreen())},element:function(){return f.enabled?g.length?document[g+"FullscreenElement"]:document.fullscreenElement:null},setup:function(){if(this.supported.ui&&!this.isAudio&&this.config.fullscreen.enabled){var e=f.enabled;e||this.config.fullscreen.fallback&&!u.inFrame()?(this.debug.log((e?"Native":"Fallback")+" fullscreen enabled"),u.toggleClass(this.elements.container,this.config.classNames.fullscreen.enabled,!0)):this.debug.log("Fullscreen not supported and fallback disabled"),this.elements.buttons&&this.elements.buttons.fullscreen&&u.toggleState(this.elements.buttons.fullscreen,!1),u.trapFocus.call(this)}}},y=u.getBrowser(),b={global:function(){var e=this,t=null,i=function(i){var n,s=(n=i).keyCode?n.keyCode:n.which,a="keydown"===i.type,o=a&&s===t;if(!(i.altKey||i.ctrlKey||i.metaKey||i.shiftKey)&&u.is.number(s)){if(a){var l=u.getFocusElement();if(u.is.element(l)&&u.matches(l,e.config.selectors.editable))return;switch([48,49,50,51,52,53,54,56,57,32,75,38,40,77,39,37,70,67,73,76,79].includes(s)&&(i.preventDefault(),i.stopPropagation()),s){case 48:case 49:case 50:case 51:case 52:case 53:case 54:case 55:case 56:case 57:o||(e.currentTime=e.duration/10*(s-48));break;case 32:case 75:o||e.togglePlay();break;case 38:e.increaseVolume(.1);break;case 40:e.decreaseVolume(.1);break;case 77:o||(e.muted=!e.muted);break;case 39:e.forward();break;case 37:e.rewind();break;case 70:e.toggleFullscreen();break;case 67:o||e.toggleCaptions();break;case 76:e.loop=!e.loop}!f.enabled&&e.fullscreen.active&&27===s&&e.toggleFullscreen(),t=s}else t=null}};this.config.keyboard.global?u.on(window,"keydown keyup",i,!1):this.config.keyboard.focused&&u.on(this.elements.container,"keydown keyup",i,!1),u.on(this.elements.container,"focusout",function(t){u.toggleClass(t.target,e.config.classNames.tabFocus,!1)}),u.on(this.elements.container,"keydown",function(t){9===t.keyCode&&window.setTimeout(function(){u.toggleClass(u.getFocusElement(),e.config.classNames.tabFocus,!0)},0)}),this.config.hideControls&&u.on(this.elements.container,"mouseenter mouseleave mousemove touchstart touchend touchmove enterfullscreen exitfullscreen",function(t){e.toggleControls(t)}),f.enabled&&(u.on(document,f.eventType,function(t){e.toggleFullscreen(t)}),u.on(this.elements.container,"dblclick",function(t){e.toggleFullscreen(t)}))},media:function(){var e=this;if(u.on(this.media,"timeupdate seeking",function(t){return v.timeUpdate.call(e,t)}),u.on(this.media,"durationchange loadedmetadata",function(t){return v.durationUpdate.call(e,t)}),u.on(this.media,"loadeddata",function(){u.toggleHidden(e.elements.volume,!e.hasAudio),u.toggleHidden(e.elements.buttons.mute,!e.hasAudio)}),u.on(this.media,"ended",function(){e.isHTML5&&e.isVideo&&e.config.showPosterOnEnd&&(e.restart(),e.media.load())}),u.on(this.media,"progress playing",function(t){return v.updateProgress.call(e,t)}),u.on(this.media,"volumechange",function(t){return v.updateVolume.call(e,t)}),u.on(this.media,"playing play pause ended",function(t){return v.checkPlaying.call(e,t)}),u.on(this.media,"stalled waiting canplay seeked playing",function(t){return v.checkLoading.call(e,t)}),this.supported.ui&&this.config.clickToPlay&&!this.isAudio){var t=u.getElement.call(this,"."+this.config.classNames.video);if(!u.is.element(t))return;u.on(t,"click",function(){e.config.hideControls&&d.touch&&!e.paused||(e.paused?e.play():e.ended?(e.restart(),e.play()):e.pause())})}this.supported.ui&&this.config.disableContextMenu&&u.on(this.media,"contextmenu",function(e){e.preventDefault()},!1),u.on(this.media,"volumechange",function(){e.storage.set({volume:e.volume,muted:e.muted})}),u.on(this.media,"ratechange",function(){w.updateSetting.call(e,"speed"),e.storage.set({speed:e.speed})}),u.on(this.media,"qualitychange",function(){w.updateSetting.call(e,"quality"),e.storage.set({quality:e.quality})}),u.on(this.media,"languagechange",function(){w.updateSetting.call(e,"captions"),e.storage.set({language:e.language})}),u.on(this.media,"captionsenabled captionsdisabled",function(){w.updateSetting.call(e,"captions"),e.storage.set({captions:e.captions.active})}),u.on(this.media,this.config.events.concat(["keyup","keydown"]).join(" "),function(t){var i={};"error"===t.type&&(i=e.media.error),u.dispatchEvent.call(e,e.elements.container,t.type,!0,i)})},controls:function(){var e=this,t=y.isIE?"change":"input",i=function(t,i,n){var s=e.config.listeners[i];u.is.function(s)&&s.call(e,t),!t.defaultPrevented&&u.is.function(n)&&n.call(e,t)};u.on(this.elements.buttons.play,"click",function(t){return i(t,"play",function(){e.togglePlay()})}),u.on(this.elements.buttons.restart,"click",function(t){return i(t,"restart",function(){e.restart()})}),u.on(this.elements.buttons.rewind,"click",function(t){return i(t,"rewind",function(){e.rewind()})}),u.on(this.elements.buttons.forward,"click",function(t){return i(t,"forward",function(){e.forward()})}),u.on(this.elements.buttons.mute,"click",function(t){return i(t,"mute",function(){e.muted=!e.muted})}),u.on(this.elements.buttons.captions,"click",function(t){return i(t,"captions",function(){e.toggleCaptions()})}),u.on(this.elements.buttons.fullscreen,"click",function(t){return i(t,"fullscreen",function(){e.toggleFullscreen()})}),u.on(this.elements.buttons.pip,"click",function(t){return i(t,"pip",function(){e.pip="toggle"})}),u.on(this.elements.buttons.airplay,"click",function(t){return i(t,"airplay",function(){e.airplay()})}),u.on(this.elements.buttons.settings,"click",function(t){w.toggleMenu.call(e,t)}),u.on(document.documentElement,"click",function(t){w.toggleMenu.call(e,t)}),u.on(this.elements.settings.form,"click",function(t){t.stopPropagation(),u.matches(t.target,e.config.selectors.inputs.language)?i(t,"language",function(){e.language=t.target.value}):u.matches(t.target,e.config.selectors.inputs.quality)?i(t,"quality",function(){e.quality=t.target.value}):u.matches(t.target,e.config.selectors.inputs.speed)?i(t,"speed",function(){e.speed=parseFloat(t.target.value)}):w.showTab.call(e,t)}),u.on(this.elements.inputs.seek,t,function(t){return i(t,"seek",function(){e.currentTime=t.target.value/t.target.max*e.duration})}),this.config.toggleInvert&&!u.is.element(this.elements.display.duration)&&u.on(this.elements.display.currentTime,"click",function(){0!==e.currentTime&&(e.config.invertTime=!e.config.invertTime,v.timeUpdate.call(e))}),u.on(this.elements.inputs.volume,t,function(t){return i(t,"volume",function(){e.volume=t.target.value})}),y.isWebkit&&u.on(u.getElements.call(this,'input[type="range"]'),"input",function(t){w.updateRangeFill.call(e,t.target)}),u.on(this.elements.progress,"mouseenter mouseleave mousemove",function(t){return w.updateSeekTooltip.call(e,t)}),this.config.hideControls&&(u.on(this.elements.controls,"mouseenter mouseleave",function(t){e.elements.controls.hover="mouseenter"===t.type}),u.on(this.elements.controls,"mousedown mouseup touchstart touchend touchcancel",function(t){e.elements.controls.pressed=["mousedown","touchstart"].includes(t.type)}),u.on(this.elements.controls,"focusin focusout",function(t){e.toggleControls(t)})),u.on(this.elements.inputs.volume,"wheel",function(t){return i(t,"volume",function(){var i=t.webkitDirectionInvertedFromDevice,n=0;(t.deltaY<0||t.deltaX>0)&&(i?(e.decreaseVolume(.02),n=-1):(e.increaseVolume(.02),n=1)),(t.deltaY>0||t.deltaX<0)&&(i?(e.increaseVolume(.02),n=1):(e.decreaseVolume(.02),n=-1)),(1===n&&e.media.volume<1||-1===n&&e.media.volume>0)&&t.preventDefault()})},!1)}},v={addStyleHook:function(){u.toggleClass(this.elements.container,this.config.selectors.container.replace(".",""),!0),u.toggleClass(this.elements.container,this.config.classNames.uiSupported,this.supported.ui)},toggleNativeControls:function(){arguments.length>0&&void 0!==arguments[0]&&arguments[0]&&this.isHTML5?this.media.setAttribute("controls",""):this.media.removeAttribute("controls")},build:function(){if(b.media.call(this),!this.supported.ui)return this.debug.warn("Basic support only for "+this.provider+" "+this.type),u.removeElement.call(this,"controls"),u.removeElement.call(this,"buttons.play"),void v.toggleNativeControls.call(this,!0);u.is.element(this.elements.controls)||(w.inject.call(this),b.controls.call(this)),u.is.element(this.elements.controls)&&(v.toggleNativeControls.call(this),f.setup.call(this),E.setup.call(this),this.volume=null,this.muted=null,this.speed=null,this.loop=null,this.options.quality=[],v.timeUpdate.call(this),v.checkPlaying.call(this),this.ready=!0,u.dispatchEvent.call(this,this.media,"ready"),v.setTitle.call(this))},setTitle:function(){var e=this.config.i18n.play;if(u.is.string(this.config.title)&&!u.is.empty(this.config.title)&&(e+=", "+this.config.title,this.elements.container.setAttribute("aria-label",this.config.title)),u.is.nodeList(this.elements.buttons.play)&&Array.from(this.elements.buttons.play).forEach(function(t){t.setAttribute("aria-label",e)}),this.isEmbed){var t=u.getElement.call(this,"iframe");if(!u.is.element(t))return;var i=u.is.empty(this.config.title)?"video":this.config.title;t.setAttribute("title",this.config.i18n.frameTitle.replace("{title}",i))}},checkPlaying:function(){var e=this;u.toggleClass(this.elements.container,this.config.classNames.playing,this.playing),u.toggleClass(this.elements.container,this.config.classNames.stopped,this.paused),u.is.nodeList(this.elements.buttons.play)&&Array.from(this.elements.buttons.play).forEach(function(t){return u.toggleState(t,e.playing)}),this.toggleControls(!this.playing)},checkLoading:function(e){var t=this;this.loading=["stalled","waiting"].includes(e.type),clearTimeout(this.timers.loading),this.timers.loading=setTimeout(function(){u.toggleClass(t.elements.container,t.config.classNames.loading,t.loading),t.toggleControls(t.loading)},this.loading?250:0)},checkFailed:function(){var e=this;this.failed=3===this.media.networkState,this.failed&&(u.toggleClass(this.elements.container,this.config.classNames.loading,!1),u.toggleClass(this.elements.container,this.config.classNames.error,!0)),clearTimeout(this.timers.failed),this.timers.loading=setTimeout(function(){u.toggleClass(e.elements.container,e.config.classNames.loading,e.loading),e.toggleControls(e.loading)},this.loading?250:0)},updateVolume:function(){this.supported.ui&&(u.is.element(this.elements.inputs.volume)&&v.setRange.call(this,this.elements.inputs.volume,this.muted?0:this.volume),u.is.element(this.elements.buttons.mute)&&u.toggleState(this.elements.buttons.mute,this.muted||0===this.volume))},setRange:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:0;u.is.element(e)&&(e.value=t,w.updateRangeFill.call(this,e))},setProgress:function(e,t){var i=u.is.number(t)?t:0,n=u.is.element(e)?e:this.elements.display.buffer;if(u.is.element(n)){n.value=i;var s=n.getElementsByTagName("span")[0];u.is.element(s)&&(s.childNodes[0].nodeValue=i)}},updateProgress:function(e){var t=this;if(this.supported.ui&&u.is.event(e)){var i,n=0;if(e)switch(e.type){case"timeupdate":case"seeking":n=u.getPercentage(this.currentTime,this.duration),"timeupdate"===e.type&&v.setRange.call(this,this.elements.inputs.seek,n);break;case"playing":case"progress":n=(i=t.media.buffered)&&i.length?u.getPercentage(i.end(0),t.duration):u.is.number(i)?100*i:0,v.setProgress.call(this,this.elements.display.buffer,n)}}},updateTimeDisplay:function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:null,t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:0,i=arguments.length>2&&void 0!==arguments[2]&&arguments[2];if(u.is.element(e)&&u.is.number(t)){var n=function(e){return("0"+e).slice(-2)},s=function(e){return parseInt(e/60/60%60,10)},a=s(t),o=parseInt(t/60%60,10),l=parseInt(t%60,10);s(this.duration)>0?a+=":":a="",e.textContent=(i?"-":"")+a+n(o)+":"+n(l)}},timeUpdate:function(e){var t=!u.is.element(this.elements.display.duration)&&this.config.invertTime;v.updateTimeDisplay.call(this,this.elements.display.currentTime,t?this.duration-this.currentTime:this.currentTime,t),e&&"timeupdate"===e.type&&this.media.seeking||v.updateProgress.call(this,e)},durationUpdate:function(){this.supported.ui&&(!u.is.element(this.elements.display.duration)&&this.config.displayDuration&&this.paused&&v.updateTimeDisplay.call(this,this.elements.display.currentTime,this.duration),u.is.element(this.elements.display.duration)&&v.updateTimeDisplay.call(this,this.elements.display.duration,this.duration),w.updateSeekTooltip.call(this))}},k=u.getBrowser(),w={updateRangeFill:function(e){if(k.isWebkit){var t=u.is.event(e)?e.target:e;u.is.element(t)&&"range"===t.getAttribute("type")&&t.style.setProperty("--value",t.value/t.max*100+"%")}},getIconUrl:function(){return{url:this.config.iconUrl,absolute:0===this.config.iconUrl.indexOf("http")||k.isIE&&!window.svg4everybody}},createIcon:function(e,t){var i="http://www.w3.org/2000/svg",n=w.getIconUrl.call(this),s=(n.absolute?"":n.url)+"#"+this.config.iconPrefix,a=document.createElementNS(i,"svg");u.setAttributes(a,u.extend(t,{role:"presentation"}));var o=document.createElementNS(i,"use"),l=s+"-"+e;return"href"in o?o.setAttributeNS("http://www.w3.org/1999/xlink","href",l):o.setAttributeNS("http://www.w3.org/1999/xlink","xlink:href",l),a.appendChild(o),a},createLabel:function(e,t){var i=this.config.i18n[e],n=Object.assign({},t);switch(e){case"pip":i="PIP";break;case"airplay":i="AirPlay"}return"class"in n?n.class+=" "+this.config.classNames.hidden:n.class=this.config.classNames.hidden,u.createElement("span",n,i)},createBadge:function(e){if(u.is.empty(e))return null;var t=u.createElement("span",{class:this.config.classNames.menu.value});return t.appendChild(u.createElement("span",{class:this.config.classNames.menu.badge},e)),t},createButton:function(e,t){var i=u.createElement("button"),n=Object.assign({},t),s=e,a=!1,o=void 0,l=void 0,r=void 0,c=void 0;switch("type"in n||(n.type="button"),"class"in n?n.class.includes(this.config.classNames.control)&&(n.class+=" "+this.config.classNames.control):n.class=this.config.classNames.control,s){case"play":a=!0,o="play",r="pause",l="play",c="pause";break;case"mute":a=!0,o="mute",r="unmute",l="volume",c="muted";break;case"captions":a=!0,o="enableCaptions",r="disableCaptions",l="captions-off",c="captions-on";break;case"fullscreen":a=!0,o="enterFullscreen",r="exitFullscreen",l="enter-fullscreen",c="exit-fullscreen";break;case"play-large":n.class+=" "+this.config.classNames.control+"--overlaid",s="play",o="play",l="play";break;default:o=s,l=s}return a?(i.appendChild(w.createIcon.call(this,c,{class:"icon--pressed"})),i.appendChild(w.createIcon.call(this,l,{class:"icon--not-pressed"})),i.appendChild(w.createLabel.call(this,r,{class:"label--pressed"})),i.appendChild(w.createLabel.call(this,o,{class:"label--not-pressed"})),n["aria-pressed"]=!1,n["aria-label"]=this.config.i18n[o]):(i.appendChild(w.createIcon.call(this,l)),i.appendChild(w.createLabel.call(this,o))),u.extend(n,u.getAttributesFromSelector(this.config.selectors.buttons[s],n)),u.setAttributes(i,n),this.elements.buttons[s]=i,i},createRange:function(e,t){var i=u.createElement("label",{for:t.id,class:this.config.classNames.hidden},this.config.i18n[e]),n=u.createElement("input",u.extend(u.getAttributesFromSelector(this.config.selectors.inputs[e]),{type:"range",min:0,max:100,step:.01,value:0,autocomplete:"off"},t));return this.elements.inputs[e]=n,w.updateRangeFill.call(this,n),{label:i,input:n}},createProgress:function(e,t){var i=u.createElement("progress",u.extend(u.getAttributesFromSelector(this.config.selectors.display[e]),{min:0,max:100,value:0},t));if("volume"!==e){i.appendChild(u.createElement("span",null,"0"));var n="";switch(e){case"played":n=this.config.i18n.played;break;case"buffer":n=this.config.i18n.buffered}i.textContent="% "+n.toLowerCase()}return this.elements.display[e]=i,i},createTime:function(e){var t=u.createElement("div",{class:"plyr__time"});return t.appendChild(u.createElement("span",{class:this.config.classNames.hidden},this.config.i18n[e])),t.appendChild(u.createElement("span",u.getAttributesFromSelector(this.config.selectors.display[e]),"00:00")),this.elements.display[e]=t,t},createMenuItem:function(e,t,i,n){var s=arguments.length>4&&void 0!==arguments[4]?arguments[4]:null,a=arguments.length>5&&void 0!==arguments[5]&&arguments[5],o=u.createElement("li"),l=u.createElement("label",{class:this.config.classNames.control}),r=u.createElement("input",u.extend(u.getAttributesFromSelector(this.config.selectors.inputs[i]),{type:"radio",name:"plyr-"+i,value:e,checked:a,class:"plyr__sr-only"})),c=u.createElement("span",{"aria-hidden":!0});l.appendChild(r),l.appendChild(c),l.insertAdjacentHTML("beforeend",n),u.is.element(s)&&l.appendChild(s),o.appendChild(l),t.appendChild(o)},updateSeekTooltip:function(e){if(this.config.tooltips.seek&&u.is.element(this.elements.inputs.seek)&&u.is.element(this.elements.display.seekTooltip)&&0!==this.duration){var t=0,i=this.elements.inputs.seek.getBoundingClientRect(),n=this.config.classNames.tooltip+"--visible";if(u.is.event(e))t=100/i.width*(e.pageX-i.left);else{if(!u.hasClass(this.elements.display.seekTooltip,n))return;t=parseFloat(this.elements.display.seekTooltip.style.left,10)}t<0?t=0:t>100&&(t=100),v.updateTimeDisplay.call(this,this.elements.display.seekTooltip,this.duration/100*t),this.elements.display.seekTooltip.style.left=t+"%",u.is.event(e)&&["mouseenter","mouseleave"].includes(e.type)&&u.toggleClass(this.elements.display.seekTooltip,n,"mouseenter"===e.type)}},toggleTab:function(e,t){var i=this.elements.settings.tabs[e],n=this.elements.settings.panes[e];u.toggleHidden(i,!t),u.toggleHidden(n,!t)},setQualityMenu:function(e){var t=this,i=this.elements.settings.panes.quality.querySelector("ul");u.is.array(e)?this.options.quality=e.filter(function(e){return t.config.quality.options.includes(e)}):this.options.quality=this.config.quality.options;var n=!u.is.empty(this.options.quality)&&this.isYouTube;if(w.toggleTab.call(this,"quality",n),n){u.emptyElement(i);this.options.quality.forEach(function(e){return w.createMenuItem.call(t,e,i,"quality",w.getLabel.call(t,"quality",e),function(e){var i="";switch(e){case"hd2160":i="4K";break;case"hd1440":i="WQHD";break;case"hd1080":case"hd720":i="HD"}return i.length?w.createBadge.call(t,i):null}(e))}),w.updateSetting.call(this,"quality",i)}},getLabel:function(e,t){switch(e){case"speed":return 1===t?"Normal":t+"×";case"quality":switch(t){case"hd2160":return"2160P";case"hd1440":return"1440P";case"hd1080":return"1080P";case"hd720":return"720P";case"large":return"480P";case"medium":return"360P";case"small":return"240P";case"tiny":return"Tiny";case"default":return"Auto";default:return t}case"captions":return w.getLanguage.call(this);default:return null}},updateSetting:function(e,t){var i=this.elements.settings.panes[e],n=null,s=t;switch(e){case"captions":n=this.captions.active?this.captions.language:"";break;default:if(n=this[e],u.is.empty(n)&&(n=this.config[e].default),!this.options[e].includes(n))return void this.debug.warn("Unsupported value of '"+n+"' for "+e);if(!this.config[e].options.includes(n))return void this.debug.warn("Disabled value of '"+n+"' for "+e)}(u.is.element(s)||(s=i&&i.querySelector("ul")),u.is.empty(n))||(this.elements.settings.tabs[e].querySelector("."+this.config.classNames.menu.value).innerHTML=w.getLabel.call(this,e,n));var a=s&&s.querySelector('input[value="'+n+'"]');u.is.element(a)&&(a.checked=!0)},getLanguage:function(){if(!this.supported.ui)return null;if(!d.textTracks||!E.getTracks.call(this).length)return this.config.i18n.none;if(this.captions.active){var e=E.getCurrentTrack.call(this);if(u.is.track(e))return e.label}return this.config.i18n.disabled},setCaptionsMenu:function(){var e=this,t=this.elements.settings.panes.captions.querySelector("ul"),i=E.getTracks.call(this).length;if(w.toggleTab.call(this,"captions",i),u.emptyElement(t),i){var n=E.getTracks.call(this).map(function(e){return{language:e.language,label:u.is.empty(e.label)?e.language.toUpperCase():e.label}});n.unshift({language:"",label:this.config.i18n.none}),n.forEach(function(i){w.createMenuItem.call(e,i.language,t,"language",i.label||i.language,w.createBadge.call(e,i.language.toUpperCase()),i.language.toLowerCase()===e.captions.language.toLowerCase())}),w.updateSetting.call(this,"captions",t)}},setSpeedMenu:function(){var e=this;u.is.object(this.options.speed)&&Object.keys(this.options.speed).length||(this.options.speed=[.5,.75,1,1.25,1.5,1.75,2]),this.options.speed=this.options.speed.filter(function(t){return e.config.speed.options.includes(t)});var t=!u.is.empty(this.options.speed);if(w.toggleTab.call(this,"speed",t),t){var i=this.elements.settings.panes.speed.querySelector("ul");u.toggleHidden(this.elements.settings.tabs.speed,!1),u.toggleHidden(this.elements.settings.panes.speed,!1),u.emptyElement(i),this.options.speed.forEach(function(t){return w.createMenuItem.call(e,t,i,"speed",w.getLabel.call(e,"speed",t))}),w.updateSetting.call(this,"speed",i)}},toggleMenu:function(e){var t=this.elements.settings.form,i=this.elements.buttons.settings,n=u.is.boolean(e)?e:u.is.element(t)&&"true"===t.getAttribute("aria-hidden");if(u.is.event(e)){var s=u.is.element(t)&&t.contains(e.target),a=e.target===this.elements.buttons.settings;if(s||!s&&!a&&n)return;a&&e.stopPropagation()}u.is.element(i)&&i.setAttribute("aria-expanded",n),u.is.element(t)&&(t.setAttribute("aria-hidden",!n),u.toggleClass(this.elements.container,this.config.classNames.menu.open,n),n?t.removeAttribute("tabindex"):t.setAttribute("tabindex",-1))},getTabSize:function(e){var t=e.cloneNode(!0);t.style.position="absolute",t.style.opacity=0,t.setAttribute("aria-hidden",!1),Array.from(t.querySelectorAll("input[name]")).forEach(function(e){var t=e.getAttribute("name");e.setAttribute("name",t+"-clone")}),e.parentNode.appendChild(t);var i=t.scrollWidth,n=t.scrollHeight;return u.removeElement(t),{width:i,height:n}},showTab:function(e){var t=this.elements.settings.menu,i=e.target,n="false"===i.getAttribute("aria-expanded"),s=document.getElementById(i.getAttribute("aria-controls"));if(u.is.element(s)&&"tabpanel"===s.getAttribute("role")){var a=t.querySelector('[role="tabpanel"][aria-hidden="false"]'),o=a.parentNode;if(Array.from(t.querySelectorAll('[aria-controls="'+a.getAttribute("id")+'"]')).forEach(function(e){e.setAttribute("aria-expanded",!1)}),d.transitions&&!d.reducedMotion){o.style.width=a.scrollWidth+"px",o.style.height=a.scrollHeight+"px";var l=w.getTabSize.call(this,s);u.on(o,u.transitionEnd,function e(t){t.target===o&&["width","height"].includes(t.propertyName)&&(o.style.width="",o.style.height="",u.off(o,u.transitionEnd,e))}),o.style.width=l.width+"px",o.style.height=l.height+"px"}a.setAttribute("aria-hidden",!0),a.setAttribute("tabindex",-1),s.setAttribute("aria-hidden",!n),i.setAttribute("aria-expanded",n),s.removeAttribute("tabindex"),s.querySelectorAll("button:not(:disabled), input:not(:disabled), [tabindex]")[0].focus()}},create:function(e){var t=this;if(u.is.empty(this.config.controls))return null;var i=u.createElement("div",u.getAttributesFromSelector(this.config.selectors.controls.wrapper));if(this.config.controls.includes("restart")&&i.appendChild(w.createButton.call(this,"restart")),this.config.controls.includes("rewind")&&i.appendChild(w.createButton.call(this,"rewind")),this.config.controls.includes("play")&&i.appendChild(w.createButton.call(this,"play")),this.config.controls.includes("fast-forward")&&i.appendChild(w.createButton.call(this,"fast-forward")),this.config.controls.includes("progress")){var n=u.createElement("div",u.getAttributesFromSelector(this.config.selectors.progress)),s=w.createRange.call(this,"seek",{id:"plyr-seek-"+e.id});if(n.appendChild(s.label),n.appendChild(s.input),n.appendChild(w.createProgress.call(this,"buffer")),this.config.tooltips.seek){var a=u.createElement("span",{role:"tooltip",class:this.config.classNames.tooltip},"00:00");n.appendChild(a),this.elements.display.seekTooltip=a}this.elements.progress=n,i.appendChild(this.elements.progress)}if(this.config.controls.includes("current-time")&&i.appendChild(w.createTime.call(this,"currentTime")),this.config.controls.includes("duration")&&i.appendChild(w.createTime.call(this,"duration")),this.config.controls.includes("mute")&&i.appendChild(w.createButton.call(this,"mute")),this.config.controls.includes("volume")){var o=u.createElement("div",{class:"plyr__volume"}),l={max:1,step:.05,value:this.config.volume},r=w.createRange.call(this,"volume",u.extend(l,{id:"plyr-volume-"+e.id}));o.appendChild(r.label),o.appendChild(r.input),this.elements.volume=o,i.appendChild(o)}if(this.config.controls.includes("captions")&&i.appendChild(w.createButton.call(this,"captions")),this.config.controls.includes("settings")&&!u.is.empty(this.config.settings)){var c=u.createElement("div",{class:"plyr__menu"});c.appendChild(w.createButton.call(this,"settings",{id:"plyr-settings-toggle-"+e.id,"aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id,"aria-expanded":!1}));var p=u.createElement("form",{class:"plyr__menu__container",id:"plyr-settings-"+e.id,"aria-hidden":!0,"aria-labelled-by":"plyr-settings-toggle-"+e.id,role:"tablist",tabindex:-1}),h=u.createElement("div"),m=u.createElement("div",{id:"plyr-settings-"+e.id+"-home","aria-hidden":!1,"aria-labelled-by":"plyr-settings-toggle-"+e.id,role:"tabpanel"}),g=u.createElement("ul",{role:"tablist"});this.config.settings.forEach(function(i){var n=u.createElement("li",{role:"tab",hidden:""}),s=u.createElement("button",u.extend(u.getAttributesFromSelector(t.config.selectors.buttons.settings),{type:"button",class:t.config.classNames.control+" "+t.config.classNames.control+"--forward",id:"plyr-settings-"+e.id+"-"+i+"-tab","aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id+"-"+i,"aria-expanded":!1}),t.config.i18n[i]),a=u.createElement("span",{class:t.config.classNames.menu.value});a.innerHTML=e[i],s.appendChild(a),n.appendChild(s),g.appendChild(n),t.elements.settings.tabs[i]=n}),m.appendChild(g),h.appendChild(m),this.config.settings.forEach(function(i){var n=u.createElement("div",{id:"plyr-settings-"+e.id+"-"+i,"aria-hidden":!0,"aria-labelled-by":"plyr-settings-"+e.id+"-"+i+"-tab",role:"tabpanel",tabindex:-1,hidden:""}),s=u.createElement("button",{type:"button",class:t.config.classNames.control+" "+t.config.classNames.control+"--back","aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id+"-home","aria-expanded":!1},t.config.i18n[i]);n.appendChild(s);var a=u.createElement("ul");n.appendChild(a),h.appendChild(n),t.elements.settings.panes[i]=n}),p.appendChild(h),c.appendChild(p),i.appendChild(c),this.elements.settings.form=p,this.elements.settings.menu=c}return this.config.controls.includes("pip")&&d.pip&&i.appendChild(w.createButton.call(this,"pip")),this.config.controls.includes("airplay")&&d.airplay&&i.appendChild(w.createButton.call(this,"airplay")),this.config.controls.includes("fullscreen")&&i.appendChild(w.createButton.call(this,"fullscreen")),this.config.controls.includes("play-large")&&this.elements.container.appendChild(w.createButton.call(this,"play-large")),this.elements.controls=i,this.config.controls.includes("settings")&&this.config.settings.includes("speed")&&w.setSpeedMenu.call(this),i},inject:function(){var e=this;if(this.config.loadSprite){var t=w.getIconUrl.call(this);t.absolute&&u.loadSprite(t.url,"sprite-plyr")}this.id=Math.floor(1e4*Math.random());var i=null;i=u.is.string(this.config.controls)?this.config.controls:u.is.function(this.config.controls)?this.config.controls({id:this.id,seektime:this.config.seekTime,title:this.config.title}):w.create.call(this,{id:this.id,seektime:this.config.seekTime,speed:this.speed,quality:this.quality,captions:w.getLanguage.call(this)});var n=void 0;if(u.is.string(this.config.selectors.controls.container)&&(n=document.querySelector(this.config.selectors.controls.container)),u.is.element(n)||(n=this.elements.container),u.is.element(i)?n.appendChild(i):n.insertAdjacentHTML("beforeend",i),u.is.element(this.elements.controls)&&u.findElements.call(this),window.navigator.userAgent.includes("Edge")&&u.repaint(n),this.config.tooltips.controls){var s=u.getElements.call(this,[this.config.selectors.controls.wrapper," ",this.config.selectors.labels," .",this.config.classNames.hidden].join(""));Array.from(s).forEach(function(t){u.toggleClass(t,e.config.classNames.hidden,!1),u.toggleClass(t,e.config.classNames.tooltip,!0),t.setAttribute("role","tooltip")})}}},E={setup:function(){if(this.supported.ui){var e=this.storage.get("language");if(u.is.empty(e)||(this.captions.language=e),u.is.empty(this.captions.language)&&(this.captions.language=this.config.captions.language.toLowerCase()),!u.is.boolean(this.captions.active)){var t=this.storage.get("captions");u.is.boolean(t)?this.captions.active=t:this.captions.active=this.config.captions.active}!this.isVideo||this.isYouTube||this.isHTML5&&!d.textTracks?this.config.controls.includes("settings")&&this.config.settings.includes("captions")&&w.setCaptionsMenu.call(this):(u.is.element(this.elements.captions)||(this.elements.captions=u.createElement("div",u.getAttributesFromSelector(this.config.selectors.captions)),u.insertAfter(this.elements.captions,this.elements.wrapper)),u.toggleClass(this.elements.container,this.config.classNames.captions.enabled,!u.is.empty(E.getTracks.call(this))),u.is.empty(E.getTracks.call(this))||(E.setLanguage.call(this),E.show.call(this),this.config.controls.includes("settings")&&this.config.settings.includes("captions")&&w.setCaptionsMenu.call(this)))}},setLanguage:function(){var e=this;if(this.isHTML5&&this.isVideo){E.getTracks.call(this).forEach(function(t){u.on(t,"cuechange",function(t){return E.setCue.call(e,t)}),t.mode="hidden"});var t=E.getCurrentTrack.call(this);u.is.track(t)&&Array.from(t.activeCues||[]).length&&E.setCue.call(this,t)}else this.isVimeo&&this.captions.active&&this.embed.enableTextTrack(this.language)},getTracks:function(){return u.is.nullOrUndefined(this.media)?[]:Array.from(this.media.textTracks||[]).filter(function(e){return["captions","subtitles"].includes(e.kind)})},getCurrentTrack:function(){var e=this;return E.getTracks.call(this).find(function(t){return t.language.toLowerCase()===e.language})},setCue:function(e){var t=u.is.event(e)?e.target:e,i=t.activeCues[0];t===E.getCurrentTrack.call(this)&&(u.is.cue(i)?E.setText.call(this,i.getCueAsHTML()):E.setText.call(this,null),u.dispatchEvent.call(this,this.media,"cuechange"))},setText:function(e){if(this.supported.ui)if(u.is.element(this.elements.captions)){var t=u.createElement("span");u.emptyElement(this.elements.captions);var i=u.is.nullOrUndefined(e)?"":e;u.is.string(i)?t.textContent=i.trim():t.appendChild(i),this.elements.captions.appendChild(t)}else this.debug.warn("No captions element to render to")},show:function(){if(u.is.element(this.elements.buttons.captions)){var e=this.storage.get("captions");u.is.boolean(e)?this.captions.active=e:e=this.config.captions.active,e&&(u.toggleClass(this.elements.container,this.config.classNames.captions.active,!0),u.toggleState(this.elements.buttons.captions,!0))}}},T={setup:function(){var e=this,t=u.parseYouTubeId(this.embedId),i=u.getElements.call(this,'[id^="'+this.provider+'-"]');Array.from(i).forEach(u.removeElement),u.toggleClass(this.elements.wrapper,this.config.classNames.embed,!0),T.setAspectRatio.call(this),this.media.setAttribute("id",u.generateId(this.provider)),u.is.object(window.YT)?T.ready.call(this,t):(u.loadScript(this.config.urls.youtube.api),window.onYouTubeReadyCallbacks=window.onYouTubeReadyCallbacks||[],window.onYouTubeReadyCallbacks.push(function(){T.ready.call(e,t)}),window.onYouTubeIframeAPIReady=function(){window.onYouTubeReadyCallbacks.forEach(function(e){e()})})},getTitle:function(){var e=this;if(u.is.function(this.embed.getVideoData)){var t=this.embed.getVideoData().title;if(u.is.empty(t))return this.config.title=t,void v.setTitle.call(this)}var i=this.config.keys.google,n=u.parseYouTubeId(this.embedId);u.is.string(i)&&!u.is.empty(i)&&fetch("https://www.googleapis.com/youtube/v3/videos?id="+n+"&key="+i+"&fields=items(snippet(title))&part=snippet").then(function(e){return e.ok?e.json():null}).then(function(t){null!==t&&u.is.object(t)&&(e.config.title=t.items[0].snippet.title,v.setTitle.call(e))}).catch(function(){})},setAspectRatio:function(){var e=this.config.ratio.split(":");this.elements.wrapper.style.paddingBottom=100/e[0]*e[1]+"%"},ready:function(e){var t=this;t.embed=new window.YT.Player(t.media.id,{videoId:e,playerVars:{autoplay:t.config.autoplay?1:0,controls:t.supported.ui?0:1,rel:0,showinfo:0,iv_load_policy:3,modestbranding:1,disablekb:1,playsinline:1,origin:window&&window.location.hostname,widget_referrer:window&&window.location.href,cc_load_policy:this.captions.active?1:0,cc_lang_pref:this.config.captions.language},events:{onError:function(e){if(!u.is.object(t.media.error)){var i={code:e.data};switch(e.data){case 2:i.message="The request contains an invalid parameter value. For example, this error occurs if you specify a video ID that does not have 11 characters, or if the video ID contains invalid characters, such as exclamation points or asterisks.";break;case 5:i.message="The requested content cannot be played in an HTML5 player or another error related to the HTML5 player has occurred.";break;case 100:i.message="The video requested was not found. This error occurs when a video has been removed (for any reason) or has been marked as private.";break;case 101:case 150:i.message="The owner of the requested video does not allow it to be played in embedded players.";break;default:i.message="An unknown error occured"}t.media.error=i,u.dispatchEvent.call(t,t.media,"error")}},onPlaybackQualityChange:function(e){var i=e.target;t.media.quality=i.getPlaybackQuality(),u.dispatchEvent.call(t,t.media,"qualitychange")},onPlaybackRateChange:function(e){var i=e.target;t.media.playbackRate=i.getPlaybackRate(),u.dispatchEvent.call(t,t.media,"ratechange")},onReady:function(e){var i=e.target;T.getTitle.call(t),t.media.play=function(){i.playVideo(),t.media.paused=!1},t.media.pause=function(){i.pauseVideo(),t.media.paused=!0},t.media.stop=function(){i.stopVideo(),t.media.paused=!0},t.media.duration=i.getDuration(),t.media.paused=!0,t.media.currentTime=0,Object.defineProperty(t.media,"currentTime",{get:function(){return Number(i.getCurrentTime())},set:function(e){t.media.seeking=!0,u.dispatchEvent.call(t,t.media,"seeking"),i.seekTo(e)}}),Object.defineProperty(t.media,"playbackRate",{get:function(){return i.getPlaybackRate()},set:function(e){i.setPlaybackRate(e)}}),Object.defineProperty(t.media,"quality",{get:function(){return i.getPlaybackQuality()},set:function(e){u.dispatchEvent.call(t,t.media,"qualityrequested",!1,{quality:e}),i.setPlaybackQuality(e)}});var n=t.config.volume;Object.defineProperty(t.media,"volume",{get:function(){return n},set:function(e){n=e,i.setVolume(100*n),u.dispatchEvent.call(t,t.media,"volumechange")}});var s=t.config.muted;Object.defineProperty(t.media,"muted",{get:function(){return s},set:function(e){var n=u.is.boolean(e)?e:s;s=n,i[n?"mute":"unMute"](),u.dispatchEvent.call(t,t.media,"volumechange")}}),Object.defineProperty(t.media,"currentSrc",{get:function(){return i.getVideoUrl()}}),Object.defineProperty(t.media,"ended",{get:function(){return t.currentTime===t.duration}}),t.options.speed=i.getAvailablePlaybackRates(),t.supported.ui&&t.media.setAttribute("tabindex",-1),u.dispatchEvent.call(t,t.media,"timeupdate"),u.dispatchEvent.call(t,t.media,"durationchange"),window.clearInterval(t.timers.buffering),t.timers.buffering=window.setInterval(function(){t.media.buffered=i.getVideoLoadedFraction(),(null===t.media.lastBuffered||t.media.lastBuffered<t.media.buffered)&&u.dispatchEvent.call(t,t.media,"progress"),t.media.lastBuffered=t.media.buffered,1===t.media.buffered&&(window.clearInterval(t.timers.buffering),u.dispatchEvent.call(t,t.media,"canplaythrough"))},200),window.setTimeout(function(){return v.build.call(t)},50)},onStateChange:function(e){var i=e.target;switch(window.clearInterval(t.timers.playing),e.data){case 0:t.media.paused=!0,t.media.loop?(i.stopVideo(),i.playVideo()):u.dispatchEvent.call(t,t.media,"ended");break;case 1:t.media.seeking&&u.dispatchEvent.call(t,t.media,"seeked"),t.media.seeking=!1,t.media.paused&&u.dispatchEvent.call(t,t.media,"play"),t.media.paused=!1,u.dispatchEvent.call(t,t.media,"playing"),t.timers.playing=window.setInterval(function(){u.dispatchEvent.call(t,t.media,"timeupdate")},50),t.media.duration!==i.getDuration()&&(t.media.duration=i.getDuration(),u.dispatchEvent.call(t,t.media,"durationchange")),w.setQualityMenu.call(t,i.getAvailableQualityLevels());break;case 2:t.media.paused=!0,u.dispatchEvent.call(t,t.media,"pause")}u.dispatchEvent.call(t,t.elements.container,"statechange",!1,{code:e.data})}}})}},C={setup:function(){var e=this,t=u.getElements.call(this,'[id^="'+this.provider+'-"]');Array.from(t).forEach(u.removeElement),u.toggleClass(this.elements.wrapper,this.config.classNames.embed,!0),C.setAspectRatio.call(this),this.media.setAttribute("id",u.generateId(this.provider)),u.is.object(window.Vimeo)?C.ready.call(this):u.loadScript(this.config.urls.vimeo.api,function(){C.ready.call(e)})},setAspectRatio:function(e){var t=u.is.string(e)?e.split(":"):this.config.ratio.split(":"),i=100/t[0]*t[1],n=(200-i)/4;this.elements.wrapper.style.paddingBottom=i+"%",this.media.style.transform="translateY(-"+n+"%)"},ready:function(){var e=this,t=this,i={loop:t.config.loop.active,autoplay:t.autoplay,byline:!1,portrait:!1,title:!1,speed:!0,transparent:0,gesture:"media"},n=u.buildUrlParameters(i),s=u.parseVimeoId(t.embedId),a=u.createElement("iframe"),o="https://player.vimeo.com/video/"+s+"?"+n;a.setAttribute("src",o),a.setAttribute("allowfullscreen",""),t.media.appendChild(a),t.embed=new window.Vimeo.Player(a),t.media.paused=!0,t.media.currentTime=0,t.media.play=function(){t.embed.play().then(function(){t.media.paused=!1})},t.media.pause=function(){t.embed.pause().then(function(){t.media.paused=!0})},t.media.stop=function(){t.embed.stop().then(function(){t.media.paused=!0,t.currentTime=0})};var l=t.media.currentTime;Object.defineProperty(t.media,"currentTime",{get:function(){return l},set:function(e){var i=t.media.paused;t.media.seeking=!0,u.dispatchEvent.call(t,t.media,"seeking"),t.embed.setCurrentTime(e),i&&t.pause()}});var r=t.config.speed.selected;Object.defineProperty(t.media,"playbackRate",{get:function(){return r},set:function(e){t.embed.setPlaybackRate(e).then(function(){r=e,u.dispatchEvent.call(t,t.media,"ratechange")})}});var c=t.config.volume;Object.defineProperty(t.media,"volume",{get:function(){return c},set:function(e){t.embed.setVolume(e).then(function(){c=e,u.dispatchEvent.call(t,t.media,"volumechange")})}});var d=t.config.muted;Object.defineProperty(t.media,"muted",{get:function(){return d},set:function(e){var i=!!u.is.boolean(e)&&e;t.embed.setVolume(i?0:t.config.volume).then(function(){d=i,u.dispatchEvent.call(t,t.media,"volumechange")})}});var p=t.config.loop;Object.defineProperty(t.media,"loop",{get:function(){return p},set:function(e){var i=u.is.boolean(e)?e:t.config.loop.active;t.embed.setLoop(i).then(function(){p=i})}});var h=void 0;t.embed.getVideoUrl().then(function(e){h=e}),Object.defineProperty(t.media,"currentSrc",{get:function(){return h}}),Object.defineProperty(t.media,"ended",{get:function(){return t.currentTime===t.duration}}),Promise.all([t.embed.getVideoWidth(),t.embed.getVideoHeight()]).then(function(t){var i=u.getAspectRatio(t[0],t[1]);C.setAspectRatio.call(e,i)}),t.embed.setAutopause(t.config.autopause).then(function(e){t.config.autopause=e}),t.embed.getVideoTitle().then(function(i){t.config.title=i,v.setTitle.call(e)}),t.embed.getCurrentTime().then(function(e){l=e,u.dispatchEvent.call(t,t.media,"timeupdate")}),t.embed.getDuration().then(function(e){t.media.duration=e,u.dispatchEvent.call(t,t.media,"durationchange")}),t.embed.getTextTracks().then(function(e){t.media.textTracks=e,E.setup.call(t)}),t.embed.on("cuechange",function(e){var i=null;e.cues.length&&(i=u.stripHTML(e.cues[0].text)),E.setText.call(t,i)}),t.embed.on("loaded",function(){u.is.element(t.embed.element)&&t.supported.ui&&t.embed.element.setAttribute("tabindex",-1)}),t.embed.on("play",function(){t.media.paused&&u.dispatchEvent.call(t,t.media,"play"),t.media.paused=!1,u.dispatchEvent.call(t,t.media,"playing")}),t.embed.on("pause",function(){t.media.paused=!0,u.dispatchEvent.call(t,t.media,"pause")}),t.embed.on("timeupdate",function(e){t.media.seeking=!1,l=e.seconds,u.dispatchEvent.call(t,t.media,"timeupdate")}),t.embed.on("progress",function(e){t.media.buffered=e.percent,u.dispatchEvent.call(t,t.media,"progress"),1===parseInt(e.percent,10)&&u.dispatchEvent.call(t,t.media,"canplaythrough")}),t.embed.on("seeked",function(){t.media.seeking=!1,u.dispatchEvent.call(t,t.media,"seeked"),u.dispatchEvent.call(t,t.media,"play")}),t.embed.on("ended",function(){t.media.paused=!0,u.dispatchEvent.call(t,t.media,"ended")}),t.embed.on("error",function(e){t.media.error=e,u.dispatchEvent.call(t,t.media,"error")}),window.setTimeout(function(){return v.build.call(t)},0)}},A=u.getBrowser(),S={setup:function(){if(this.media)if(u.toggleClass(this.elements.container,this.config.classNames.type.replace("{0}",this.type),!0),u.toggleClass(this.elements.container,this.config.classNames.provider.replace("{0}",this.provider),!0),this.isEmbed&&u.toggleClass(this.elements.container,this.config.classNames.type.replace("{0}","video"),!0),this.supported.ui&&(u.toggleClass(this.elements.container,this.config.classNames.pip.supported,d.pip&&this.isHTML5&&this.isVideo),u.toggleClass(this.elements.container,this.config.classNames.airplay.supported,d.airplay&&this.isHTML5),u.toggleClass(this.elements.container,this.config.classNames.stopped,this.config.autoplay),u.toggleClass(this.elements.container,this.config.classNames.isIos,A.isIos),u.toggleClass(this.elements.container,this.config.classNames.isTouch,d.touch)),(this.isVideo||this.isYouTube||this.isVimeo)&&(this.elements.wrapper=u.createElement("div",{class:this.config.classNames.video}),u.wrap(this.media,this.elements.wrapper)),this.isEmbed)switch(this.provider){case"youtube":T.setup.call(this);break;case"vimeo":C.setup.call(this)}else this.isHTML5&&v.setTitle.call(this);else this.debug.warn("No media element found!")},cancelRequests:function(){this.isHTML5&&(Array.from(this.media.querySelectorAll("source")).forEach(u.removeElement),this.media.setAttribute("src",this.config.blankVideo),this.media.load(),this.debug.log("Cancelled network requests"))}},N={insertElements:function(e,t){var i=this;u.is.string(t)?u.insertElement(e,this.media,{src:t}):u.is.array(t)&&t.forEach(function(t){u.insertElement(e,i.media,t)})},change:function(e){var t=this;u.is.object(e)&&"sources"in e&&e.sources.length?(S.cancelRequests.call(this),this.destroy.call(this,function(){switch(u.removeElement(t.media),t.media=null,u.is.element(t.elements.container)&&t.elements.container.removeAttribute("class"),t.type=e.type,t.provider=u.is.empty(e.sources[0].provider)?s.html5:e.sources[0].provider,t.supported=d.check(t.type,t.provider,t.config.inline),t.provider+":"+t.type){case"html5:video":t.media=u.createElement("video");break;case"html5:audio":t.media=u.createElement("audio");break;case"youtube:video":case"vimeo:video":t.media=u.createElement("div"),t.embedId=e.sources[0].src}t.elements.container.appendChild(t.media),u.is.boolean(e.autoplay)&&(t.config.autoplay=e.autoplay),t.isHTML5&&(t.config.crossorigin&&t.media.setAttribute("crossorigin",""),t.config.autoplay&&t.media.setAttribute("autoplay",""),"poster"in e&&t.media.setAttribute("poster",e.poster),t.config.loop.active&&t.media.setAttribute("loop",""),t.config.muted&&t.media.setAttribute("muted",""),t.config.inline&&t.media.setAttribute("playsinline","")),v.addStyleHook.call(t),t.isHTML5&&N.insertElements.call(t,"source",e.sources),t.config.title=e.title,S.setup.call(t),t.isHTML5&&("tracks"in e&&N.insertElements.call(t,"track",e.tracks),t.media.load()),(t.isHTML5||t.isEmbed&&!t.supported.ui)&&v.build.call(t)},!0)):this.debug.warn("Invalid source format")}},P={x:0,y:0};return function(){function e(t,i){var n=this;if(l(this,e),this.timers={},this.ready=!1,this.loading=!1,this.failed=!1,this.media=t,u.is.string(this.media)&&(this.media=document.querySelectorAll(this.media)),(window.jQuery&&this.media instanceof jQuery||u.is.nodeList(this.media)||u.is.array(this.media))&&(this.media=this.media[0]),this.config=u.extend({},o,i,function(){try{return JSON.parse(n.media.getAttribute("data-plyr-config"))}catch(e){return{}}}()),this.elements={container:null,buttons:{},display:{},progress:{},inputs:{},settings:{menu:null,panes:{},tabs:{}},captions:null},this.captions={active:null,currentTrack:null},this.fullscreen={active:!1},this.options={speed:[],quality:[]},this.debug=new h(this),this.debug.log("Config",this.config),this.debug.log("Support",d),!u.is.nullOrUndefined(this.media)&&u.is.element(this.media))if(this.media.plyr)this.debug.warn("Target already setup");else if(this.config.enabled)if(d.check().api){this.elements.original=this.media.cloneNode(!0);var r=this.media.tagName.toLowerCase(),c="data-plyr-provider",p="data-plyr-embed-id";switch(r){case"div":if(this.type=a.video,this.provider=this.media.getAttribute(c),this.embedId=this.media.getAttribute(p),u.is.empty(this.provider)||!Object.keys(s).includes(this.provider))return void this.debug.error("Setup failed: Invalid provider");if(u.is.empty(this.embedId))return void this.debug.error("Setup failed: Embed ID or URL missing");this.media.removeAttribute(c),this.media.removeAttribute(p);break;case"video":case"audio":this.type=r,this.provider=s.html5,this.media.hasAttribute("crossorigin")&&(this.config.crossorigin=!0),this.media.hasAttribute("autoplay")&&(this.config.autoplay=!0),this.media.hasAttribute("playsinline")&&(this.config.inline=!0),this.media.hasAttribute("muted")&&(this.config.muted=!0),this.media.hasAttribute("loop")&&(this.config.loop.active=!0);break;default:return void this.debug.error("Setup failed: unsupported type")}this.storage=new m(this),this.supported=d.check(this.type,this.provider,this.config.inline),this.supported.api?(this.media.plyr=this,this.elements.container=u.createElement("div"),u.wrap(this.media,this.elements.container),this.elements.container.setAttribute("tabindex",0),b.global.call(this),v.addStyleHook.call(this),S.setup.call(this),this.config.debug&&u.on(this.elements.container,this.config.events.join(" "),function(e){n.debug.log("event: "+e.type)}),(this.isHTML5||this.isEmbed&&!this.supported.ui)&&v.build.call(this)):this.debug.error("Setup failed: no support")}else this.debug.error("Setup failed: no support");else this.debug.error("Setup failed: disabled by config");else this.debug.error("Setup failed: no suitable element passed")}return r(e,[{key:"play",value:function(){return this.media.play()}},{key:"pause",value:function(){this.playing&&this.media.pause()}},{key:"togglePlay",value:function(e){(u.is.boolean(e)?e:!this.playing)?this.play():this.pause()}},{key:"stop",value:function(){this.restart(),this.pause()}},{key:"restart",value:function(){this.currentTime=0}},{key:"rewind",value:function(e){this.currentTime=this.currentTime-(u.is.number(e)?e:this.config.seekTime)}},{key:"forward",value:function(e){this.currentTime=this.currentTime+(u.is.number(e)?e:this.config.seekTime)}},{key:"increaseVolume",value:function(e){var t=this.media.muted?0:this.volume;this.volume=t+(u.is.number(e)?e:1)}},{key:"decreaseVolume",value:function(e){var t=this.media.muted?0:this.volume;this.volume=t-(u.is.number(e)?e:1)}},{key:"toggleCaptions",value:function(e){if(this.supported.ui&&u.is.element(this.elements.buttons.captions)){var t=u.is.boolean(e)?e:-1===this.elements.container.className.indexOf(this.config.classNames.captions.active);this.captions.active!==t&&(this.captions.active=t,u.toggleState(this.elements.buttons.captions,this.captions.active),u.toggleClass(this.elements.container,this.config.classNames.captions.active,this.captions.active),u.dispatchEvent.call(this,this.media,this.captions.active?"captionsenabled":"captionsdisabled"))}}},{key:"toggleFullscreen",value:function(e){if(!this.isAudio){if(f.enabled){if(!u.is.event(e)||e.type!==f.eventType)return void(this.fullscreen.active?f.cancelFullScreen():f.requestFullScreen(this.elements.container));this.fullscreen.active=f.isFullScreen(this.elements.container)}else this.fullscreen.active=!this.fullscreen.active,u.toggleClass(this.elements.container,this.config.classNames.fullscreen.fallback,this.fullscreen.active),this.fullscreen.active?P={x:window.pageXOffset||0,y:window.pageYOffset||0}:window.scrollTo(P.x,P.y),document.body.style.overflow=this.fullscreen.active?"hidden":"";u.is.element(this.elements.buttons.fullscreen)&&u.toggleState(this.elements.buttons.fullscreen,this.fullscreen.active),u.dispatchEvent.call(this,this.media,this.fullscreen.active?"enterfullscreen":"exitfullscreen")}}},{key:"airplay",value:function(){d.airplay&&this.media.webkitShowPlaybackTargetPicker()}},{key:"toggleControls",value:function(e){var t=this;if(u.is.element(this.elements.controls)&&this.supported.ui&&!this.isAudio){var i=0,n=e,s=!1;if(u.is.boolean(e)||(u.is.event(e)?(s="enterfullscreen"===e.type,n=["mouseenter","mousemove","touchstart","touchmove","focusin"].includes(e.type),["mousemove","touchmove","touchend"].includes(e.type)&&(i=2e3),"focusin"===e.type&&(i=3e3,u.toggleClass(this.elements.controls,this.config.classNames.noTransition,!0))):n=u.hasClass(this.elements.container,this.config.classNames.hideControls)),window.clearTimeout(this.timers.controls),n||this.paused||this.loading){if(u.toggleClass(this.elements.container,this.config.classNames.hideControls,!1)&&u.dispatchEvent.call(this,this.media,"controlsshown"),this.paused||this.loading)return;d.touch&&(i=3e3)}n&&!this.playing||(this.timers.controls=window.setTimeout(function(){(!t.elements.controls.pressed&&!t.elements.controls.hover||s)&&(u.hasClass(t.elements.container,t.config.classNames.hideControls)||u.toggleClass(t.elements.controls,t.config.classNames.noTransition,!1),u.toggleClass(t.elements.container,t.config.classNames.hideControls,!0)&&(u.dispatchEvent.call(t,t.media,"controlshidden"),t.config.controls.includes("settings")&&!u.is.empty(t.config.settings)&&w.toggleMenu.call(t,!1)))},i))}}},{key:"on",value:function(e,t){u.on(this.elements.container,e,t)}},{key:"off",value:function(e,t){u.off(this.elements.container,e,t)}},{key:"destroy",value:function(e){var t=this,i=arguments.length>1&&void 0!==arguments[1]&&arguments[1],n=function(){if(document.body.style.overflow="",t.embed=null,t.embedId=null,i)Object.keys(t.elements).length&&(t.elements.buttons&&t.elements.buttons.play&&Array.from(t.elements.buttons.play).forEach(function(e){return u.removeElement(e)}),u.removeElement(t.elements.captions),u.removeElement(t.elements.controls),u.removeElement(t.elements.wrapper),t.elements.buttons.play=null,t.elements.captions=null,t.elements.controls=null,t.elements.wrapper=null),u.is.function(e)&&e();else{var n=t.elements.container.parentNode;u.is.element(n)&&n.replaceChild(t.elements.original,t.elements.container),u.dispatchEvent.call(t,t.elements.original,"destroyed",!0),u.is.function(e)&&e.call(t.elements.original),t.elements=null}};switch(this.provider+":"+this.type){case"html5:video":case"html5:audio":v.toggleNativeControls.call(this,!0),n();break;case"youtube:video":window.clearInterval(this.timers.buffering),window.clearInterval(this.timers.playing),this.embed.destroy(),n();break;case"vimeo:video":this.embed.unload().then(n),window.setTimeout(n,200)}}},{key:"supports",value:function(e){return d.mime.call(this,e)}},{key:"isHTML5",get:function(){return this.provider===s.html5}},{key:"isEmbed",get:function(){return this.isYouTube||this.isVimeo}},{key:"isYouTube",get:function(){return this.provider===s.youtube}},{key:"isVimeo",get:function(){return this.provider===s.vimeo}},{key:"isVideo",get:function(){return this.type===a.video}},{key:"isAudio",get:function(){return this.type===a.audio}},{key:"paused",get:function(){return this.media.paused}},{key:"playing",get:function(){return!this.paused&&!this.ended&&(!this.isHTML5||this.media.readyState>2)}},{key:"ended",get:function(){return this.media.ended}},{key:"currentTime",set:function(e){var t=0;u.is.number(e)&&(t=e),t<0?t=0:t>this.duration&&(t=this.duration),this.media.currentTime=t.toFixed(4),this.debug.log("Seeking to "+this.currentTime+" seconds")},get:function(){return Number(this.media.currentTime)}},{key:"seeking",get:function(){return this.media.seeking}},{key:"duration",get:function(){var e=parseInt(this.config.duration,10),t=Number(this.media.duration);return Number.isNaN(e)?t:e}},{key:"volume",set:function(e){var t=e;u.is.string(t)&&(t=Number(t)),u.is.number(t)||(t=this.storage.get("volume")),u.is.number(t)||(t=this.config.volume),t>1&&(t=1),t<0&&(t=0),this.config.volume=t,this.media.volume=t,this.muted&&t>0&&(this.muted=!1)},get:function(){return this.media.volume}},{key:"muted",set:function(e){var t=e;u.is.boolean(t)||(t=this.storage.get("muted")),u.is.boolean(t)||(t=this.config.muted),this.config.muted=t,this.media.muted=t},get:function(){return this.media.muted}},{key:"hasAudio",get:function(){return!this.isHTML5||(this.media.mozHasAudio||Boolean(this.media.webkitAudioDecodedByteCount)||Boolean(this.media.audioTracks&&this.media.audioTracks.length))}},{key:"speed",set:function(e){var t=null;u.is.number(e)&&(t=e),u.is.number(t)||(t=this.storage.get("speed")),u.is.number(t)||(t=this.config.speed.selected),t<.1&&(t=.1),t>2&&(t=2),this.config.speed.options.includes(t)?(this.config.speed.selected=t,this.media.playbackRate=t):this.debug.warn("Unsupported speed ("+t+")")},get:function(){return this.media.playbackRate}},{key:"quality",set:function(e){var t=null;u.is.string(e)&&(t=e),u.is.string(t)||(t=this.storage.get("quality")),u.is.string(t)||(t=this.config.quality.selected),this.options.quality.includes(t)?(this.config.quality.selected=t,this.media.quality=t):this.debug.warn("Unsupported quality option ("+t+")")},get:function(){return this.media.quality}},{key:"loop",set:function(e){var t=u.is.boolean(e)?e:this.config.loop.active;this.config.loop.active=t,this.media.loop=t},get:function(){return this.media.loop}},{key:"source",set:function(e){N.change.call(this,e)},get:function(){return this.media.currentSrc}},{key:"poster",set:function(e){this.isHTML5&&this.isVideo?u.is.string(e)&&this.media.setAttribute("poster",e):this.debug.warn("Poster can only be set on HTML5 video")},get:function(){return this.isHTML5&&this.isVideo?this.media.getAttribute("poster"):null}},{key:"autoplay",set:function(e){var t=u.is.boolean(e)?e:this.config.autoplay;this.config.autoplay=t},get:function(){return this.config.autoplay}},{key:"language",set:function(e){if(u.is.string(e)&&(this.toggleCaptions(!u.is.empty(e)),!u.is.empty(e))){var t=e.toLowerCase();this.language!==t&&(this.captions.language=t,E.setText.call(this,null),E.setLanguage.call(this),u.dispatchEvent.call(this,this.media,"languagechange"))}},get:function(){return this.captions.language}},{key:"pip",set:function(e){var t="picture-in-picture",i="inline";if(d.pip){var n=u.is.boolean(e)?e:this.pip===i;this.media.webkitSetPresentationMode(n?t:i)}},get:function(){return d.pip?this.media.webkitPresentationMode:null}}],[{key:"supported",value:function(e,t,i){return d.check(e,t,i)}},{key:"loadSprite",value:function(e,t){return u.loadSprite(e,t)}}]),e}()}); +!function(e,t){"object"==typeof exports&&"undefined"!=typeof module?module.exports=t():"function"==typeof define&&define.amd?define("Plyr",t):e.Plyr=t()}(this,function(){"use strict";var e,t,i,n,s={html5:"html5",youtube:"youtube",vimeo:"vimeo"},a={audio:"audio",video:"video"},l={enabled:!0,title:"",debug:!1,autoplay:!1,autopause:!0,seekTime:10,volume:1,muted:!1,duration:null,displayDuration:!0,invertTime:!0,toggleInvert:!0,ratio:"16:9",clickToPlay:!0,hideControls:!0,showPosterOnEnd:!1,disableContextMenu:!0,loadSprite:!0,iconPrefix:"plyr",iconUrl:"https://cdn.plyr.io/2.0.10/plyr.svg",blankVideo:"https://cdn.plyr.io/static/blank.mp4",quality:{default:"default",options:["hd2160","hd1440","hd1080","hd720","large","medium","small","tiny","default"]},loop:{active:!1},speed:{selected:1,options:[.5,.75,1,1.25,1.5,1.75,2]},keyboard:{focused:!0,global:!1},tooltips:{controls:!1,seek:!0},captions:{active:!1,language:window.navigator.language.split("-")[0]},fullscreen:{enabled:!0,fallback:!0},storage:{enabled:!0,key:"plyr"},controls:["play-large","play","progress","current-time","mute","volume","captions","settings","pip","airplay","fullscreen"],settings:["captions","quality","speed"],i18n:{restart:"Restart",rewind:"Rewind {seektime} secs",play:"Play",pause:"Pause",forward:"Forward {seektime} secs",seek:"Seek",played:"Played",buffered:"Buffered",currentTime:"Current time",duration:"Duration",volume:"Volume",mute:"Mute",unmute:"Unmute",enableCaptions:"Enable captions",disableCaptions:"Disable captions",enterFullscreen:"Enter fullscreen",exitFullscreen:"Exit fullscreen",frameTitle:"Player for {title}",captions:"Captions",settings:"Settings",speed:"Speed",quality:"Quality",loop:"Loop",start:"Start",end:"End",all:"All",reset:"Reset",none:"None",disabled:"Disabled"},urls:{vimeo:{api:"https://player.vimeo.com/api/player.js"},youtube:{api:"https://www.youtube.com/iframe_api"}},listeners:{seek:null,play:null,pause:null,restart:null,rewind:null,forward:null,mute:null,volume:null,captions:null,fullscreen:null,pip:null,airplay:null,speed:null,quality:null,loop:null,language:null},events:["ended","progress","stalled","playing","waiting","canplay","canplaythrough","loadstart","loadeddata","loadedmetadata","timeupdate","volumechange","play","pause","error","seeking","seeked","emptied","ratechange","cuechange","enterfullscreen","exitfullscreen","captionsenabled","captionsdisabled","languagechange","controlshidden","controlsshown","ready","statechange","qualitychange","qualityrequested"],selectors:{editable:"input, textarea, select, [contenteditable]",container:".plyr",controls:{container:null,wrapper:".plyr__controls"},labels:"[data-plyr]",buttons:{play:'[data-plyr="play"]',pause:'[data-plyr="pause"]',restart:'[data-plyr="restart"]',rewind:'[data-plyr="rewind"]',forward:'[data-plyr="fast-forward"]',mute:'[data-plyr="mute"]',captions:'[data-plyr="captions"]',fullscreen:'[data-plyr="fullscreen"]',pip:'[data-plyr="pip"]',airplay:'[data-plyr="airplay"]',settings:'[data-plyr="settings"]',loop:'[data-plyr="loop"]'},inputs:{seek:'[data-plyr="seek"]',volume:'[data-plyr="volume"]',speed:'[data-plyr="speed"]',language:'[data-plyr="language"]',quality:'[data-plyr="quality"]'},display:{currentTime:".plyr__time--current",duration:".plyr__time--duration",buffer:".plyr__progress--buffer",played:".plyr__progress--played",loop:".plyr__progress--loop",volume:".plyr__volume--display"},progress:".plyr__progress",captions:".plyr__captions",menu:{quality:".js-plyr__menu__list--quality"}},classNames:{video:"plyr__video-wrapper",embed:"plyr__video-embed",control:"plyr__control",type:"plyr--{0}",provider:"plyr--{0}",stopped:"plyr--stopped",playing:"plyr--playing",loading:"plyr--loading",error:"plyr--has-error",hover:"plyr--hover",tooltip:"plyr__tooltip",hidden:"plyr__sr-only",hideControls:"plyr--hide-controls",isIos:"plyr--is-ios",isTouch:"plyr--is-touch",uiSupported:"plyr--full-ui",noTransition:"plyr--no-transition",menu:{value:"plyr__menu__value",badge:"plyr__badge",open:"plyr--menu-open"},captions:{enabled:"plyr--captions-enabled",active:"plyr--captions-active"},fullscreen:{enabled:"plyr--fullscreen-enabled",fallback:"plyr--fullscreen-fallback"},pip:{supported:"plyr--pip-supported",active:"plyr--pip-active"},airplay:{supported:"plyr--airplay-supported",active:"plyr--airplay-active"},tabFocus:"plyr__tab-focus"},keys:{google:null}},o=(function(){function e(e){this.value=e}function t(t){var i,n;function s(i,n){try{var l=t[i](n),o=l.value;o instanceof e?Promise.resolve(o.value).then(function(e){s("next",e)},function(e){s("throw",e)}):a(l.done?"return":"normal",l.value)}catch(e){a("throw",e)}}function a(e,t){switch(e){case"return":i.resolve({value:t,done:!0});break;case"throw":i.reject(t);break;default:i.resolve({value:t,done:!1})}(i=i.next)?s(i.key,i.arg):n=null}this._invoke=function(e,t){return new Promise(function(a,l){var o={key:e,arg:t,resolve:a,reject:l,next:null};n?n=n.next=o:(i=n=o,s(e,t))})},"function"!=typeof t.return&&(this.return=void 0)}"function"==typeof Symbol&&Symbol.asyncIterator&&(t.prototype[Symbol.asyncIterator]=function(){return this}),t.prototype.next=function(e){return this._invoke("next",e)},t.prototype.throw=function(e){return this._invoke("throw",e)},t.prototype.return=function(e){return this._invoke("return",e)}}(),function(e,t){if(!(e instanceof t))throw new TypeError("Cannot call a class as a function")}),r=function(){function e(e,t){for(var i=0;i<t.length;i++){var n=t[i];n.enumerable=n.enumerable||!1,n.configurable=!0,"value"in n&&(n.writable=!0),Object.defineProperty(e,n.key,n)}}return function(t,i,n){return i&&e(t.prototype,i),n&&e(t,n),t}}(),c=function(e,t,i){return t in e?Object.defineProperty(e,t,{value:i,enumerable:!0,configurable:!0,writable:!0}):e[t]=i,e},u=function(){return function(e,t){if(Array.isArray(e))return e;if(Symbol.iterator in Object(e))return function(e,t){var i=[],n=!0,s=!1,a=void 0;try{for(var l,o=e[Symbol.iterator]();!(n=(l=o.next()).done)&&(i.push(l.value),!t||i.length!==t);n=!0);}catch(e){s=!0,a=e}finally{try{!n&&o.return&&o.return()}finally{if(s)throw a}}return i}(e,t);throw new TypeError("Invalid attempt to destructure non-iterable instance")}}(),d={is:{plyr:function(e){return this.instanceof(e,Plyr)},object:function(e){return this.getConstructor(e)===Object},number:function(e){return this.getConstructor(e)===Number&&!Number.isNaN(e)},string:function(e){return this.getConstructor(e)===String},boolean:function(e){return this.getConstructor(e)===Boolean},function:function(e){return this.getConstructor(e)===Function},array:function(e){return!this.nullOrUndefined(e)&&Array.isArray(e)},weakMap:function(e){return this.instanceof(e,window.WeakMap)},nodeList:function(e){return this.instanceof(e,window.NodeList)},element:function(e){return this.instanceof(e,window.Element)},textNode:function(e){return this.getConstructor(e)===Text},event:function(e){return this.instanceof(e,window.Event)},cue:function(e){return this.instanceof(e,window.TextTrackCue)||this.instanceof(e,window.VTTCue)},track:function(e){return this.instanceof(e,TextTrack)||!this.nullOrUndefined(e)&&this.string(e.kind)},nullOrUndefined:function(e){return null===e||void 0===e},empty:function(e){return this.nullOrUndefined(e)||(this.string(e)||this.array(e)||this.nodeList(e))&&!e.length||this.object(e)&&!Object.keys(e).length},instanceof:function(e,t){return Boolean(e&&t&&e instanceof t)},getConstructor:function(e){return this.nullOrUndefined(e)?null:e.constructor}},getBrowser:function(){return{isIE:!!document.documentMode,isWebkit:"WebkitAppearance"in document.documentElement.style&&!/Edge/.test(navigator.userAgent),isIPhone:/(iPhone|iPod)/gi.test(navigator.platform),isIos:/(iPad|iPhone|iPod)/gi.test(navigator.platform)}},loadScript:function(e,t){var i=document.querySelector('script[src="'+e+'"]');if(null!==i)return i.callbacks=i.callbacks||[],void i.callbacks.push(t);var n=document.createElement("script");n.callbacks=n.callbacks||[],n.callbacks.push(t),d.is.function(t)&&n.addEventListener("load",function(e){n.callbacks.forEach(function(t){return t.call(null,e)}),n.callbacks=null},!1),n.src=e;var s=document.getElementsByTagName("script")[0];s.parentNode.insertBefore(n,s)},loadSprite:function(e,t){if(d.is.string(e)){var i=d.is.string(t);if(!i||!document.querySelectorAll("#"+t).length){var n=document.createElement("div");if(d.toggleHidden(n,!0),i&&n.setAttribute("id",t),p.storage){var s=window.localStorage.getItem("cache-"+t);if(null!==s){var a=JSON.parse(s);return void l.call(n,a.content)}}fetch(e).then(function(e){return e.ok?e.text():null}).then(function(e){null!==e&&(p.storage&&window.localStorage.setItem("cache-"+t,JSON.stringify({content:e})),l.call(n,e))}).catch(function(){})}}function l(e){this.innerHTML=e,document.body.insertBefore(this,document.body.childNodes[0])}},generateId:function(e){return e+"-"+Math.floor(1e4*Math.random())},inFrame:function(){try{return window.self!==window.top}catch(e){return!0}},wrap:function(e,t){var i=e.length?e:[e];Array.from(i).reverse().forEach(function(e,i){var n=i>0?t.cloneNode(!0):t,s=e.parentNode,a=e.nextSibling;n.appendChild(e),a?s.insertBefore(n,a):s.appendChild(n)})},createElement:function(e,t,i){var n=document.createElement(e);return d.is.object(t)&&d.setAttributes(n,t),d.is.string(i)&&(n.textContent=i),n},insertAfter:function(e,t){t.parentNode.insertBefore(e,t.nextSibling)},insertElement:function(e,t,i,n){t.appendChild(d.createElement(e,i,n))},removeElement:function(e){return d.is.element(e)&&d.is.element(e.parentNode)?(e.parentNode.removeChild(e),e):null},emptyElement:function(e){for(var t=e.childNodes.length;t>0;)e.removeChild(e.lastChild),t-=1},replaceElement:function(e,t){return d.is.element(t)&&d.is.element(t.parentNode)&&d.is.element(e)?(t.parentNode.replaceChild(e,t),e):null},setAttributes:function(e,t){d.is.element(e)&&!d.is.empty(t)&&Object.keys(t).forEach(function(i){e.setAttribute(i,t[i])})},getAttributesFromSelector:function(e,t){if(!d.is.string(e)||d.is.empty(e))return{};var i={},n=t;return e.split(",").forEach(function(e){var t=e.trim(),s=t.replace(".",""),a=t.replace(/[[\]]/g,"").split("="),l=a[0],o=a.length>1?a[1].replace(/["']/g,""):"";switch(t.charAt(0)){case".":d.is.object(n)&&d.is.string(n.class)&&(n.class+=" "+s),i.class=s;break;case"#":i.id=t.replace("#","");break;case"[":i[l]=o}}),i},toggleClass:function(e,t,i){if(d.is.element(e)){var n=e.classList.contains(t);return e.classList[i?"add":"remove"](t),i&&!n||!i&&n}return null},hasClass:function(e,t){return d.is.element(e)&&e.classList.contains(t)},toggleHidden:function(e,t){d.is.element(e)&&(t?e.setAttribute("hidden",""):e.removeAttribute("hidden"))},matches:function(e,t){var i={Element:Element};var n=i.matches||i.webkitMatchesSelector||i.mozMatchesSelector||i.msMatchesSelector||function(){return Array.from(document.querySelectorAll(t)).includes(this)};return n.call(e,t)},getElements:function(e){return this.elements.container.querySelectorAll(e)},getElement:function(e){return this.elements.container.querySelector(e)},findElements:function(){try{return this.elements.controls=d.getElement.call(this,this.config.selectors.controls.wrapper),this.elements.buttons={play:d.getElements.call(this,this.config.selectors.buttons.play),pause:d.getElement.call(this,this.config.selectors.buttons.pause),restart:d.getElement.call(this,this.config.selectors.buttons.restart),rewind:d.getElement.call(this,this.config.selectors.buttons.rewind),forward:d.getElement.call(this,this.config.selectors.buttons.forward),mute:d.getElement.call(this,this.config.selectors.buttons.mute),pip:d.getElement.call(this,this.config.selectors.buttons.pip),airplay:d.getElement.call(this,this.config.selectors.buttons.airplay),settings:d.getElement.call(this,this.config.selectors.buttons.settings),captions:d.getElement.call(this,this.config.selectors.buttons.captions),fullscreen:d.getElement.call(this,this.config.selectors.buttons.fullscreen)},this.elements.progress=d.getElement.call(this,this.config.selectors.progress),this.elements.inputs={seek:d.getElement.call(this,this.config.selectors.inputs.seek),volume:d.getElement.call(this,this.config.selectors.inputs.volume)},this.elements.display={buffer:d.getElement.call(this,this.config.selectors.display.buffer),duration:d.getElement.call(this,this.config.selectors.display.duration),currentTime:d.getElement.call(this,this.config.selectors.display.currentTime)},d.is.element(this.elements.progress)&&(this.elements.display.seekTooltip=this.elements.progress.querySelector("."+this.config.classNames.tooltip)),!0}catch(e){return this.debug.warn("It looks like there is a problem with your custom controls HTML",e),this.toggleNativeControls(!0),!1}},getFocusElement:function(){var e=document.activeElement;return e=e&&e!==document.body?document.querySelector(":focus"):null},trapFocus:function(){var e=this,t=d.getElements.call(this,"button:not(:disabled), input:not(:disabled), [tabindex]"),i=t[0],n=t[t.length-1];d.on(this.elements.container,"keydown",function(t){if("Tab"===t.key&&9===t.keyCode&&e.fullscreen.active){var s=d.getFocusElement();s!==n||t.shiftKey?s===i&&t.shiftKey&&(n.focus(),t.preventDefault()):(i.focus(),t.preventDefault())}},!1)},toggleListener:function(e,t,i,n,s,a){if(!d.is.nullOrUndefined(e))if(d.is.nodeList(e))Array.from(e).forEach(function(e){e instanceof Node&&d.toggleListener.call(null,e,t,i,n,s,a)});else{var l=t.split(" "),o=!!d.is.boolean(a)&&a;p.passiveListeners&&(o={passive:!d.is.boolean(s)||s,capture:!!d.is.boolean(a)&&a}),l.forEach(function(t){e[n?"addEventListener":"removeEventListener"](t,i,o)})}},on:function(e,t,i,n,s){d.toggleListener(e,t,i,!0,n,s)},off:function(e,t,i,n,s){d.toggleListener(e,t,i,!1,n,s)},dispatchEvent:function(e,t,i,n){if(e&&t){var s=new CustomEvent(t,{bubbles:!!d.is.boolean(i)&&i,detail:Object.assign({},n,{plyr:this instanceof Plyr?this:null})});e.dispatchEvent(s)}},toggleState:function(e,t){if(d.is.element(e)){var i="true"===e.getAttribute("aria-pressed"),n=d.is.boolean(t)?t:!i;e.setAttribute("aria-pressed",n)}},getPercentage:function(e,t){return 0===e||0===t||Number.isNaN(e)||Number.isNaN(t)?0:(e/t*100).toFixed(2)},extend:function(){for(var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{},t=arguments.length,i=Array(t>1?t-1:0),n=1;n<t;n++)i[n-1]=arguments[n];if(!i.length)return e;var s=i.shift();return d.is.object(s)?(Object.keys(s).forEach(function(t){d.is.object(s[t])?(Object.keys(e).includes(t)||Object.assign(e,c({},t,{})),d.extend(e[t],s[t])):Object.assign(e,c({},t,s[t]))}),d.extend.apply(d,[e].concat(function(e){if(Array.isArray(e)){for(var t=0,i=Array(e.length);t<e.length;t++)i[t]=e[t];return i}return Array.from(e)}(i)))):e},getProviderByUrl:function(e){return/^(https?:\/\/)?(www\.)?(youtube\.com|youtu\.?be)\/.+$/.test(e)?s.youtube:/^https?:\/\/player.vimeo.com\/video\/\d{8,}(?=\b|\/)/.test(e)?s.vimeo:null},parseYouTubeId:function(e){if(d.is.empty(e))return null;return e.match(/^.*(youtu.be\/|v\/|u\/\w\/|embed\/|watch\?v=|&v=)([^#&?]*).*/)?RegExp.$2:e},parseVimeoId:function(e){if(d.is.empty(e))return null;if(d.is.number(Number(e)))return e;return e.match(/^.*(vimeo.com\/|video\/)(\d+).*/)?RegExp.$2:e},parseUrl:function(e){var t=document.createElement("a");return t.href=e,t},getUrlParams:function(e){var t=e;(e.startsWith("http://")||e.startsWith("https://"))&&(t=this.parseUrl(e).search);return this.is.empty(t)?null:t.slice(t.indexOf("?")+1).split("&").reduce(function(e,t){var i=t.split("="),n=u(i,2),s=n[0],a=n[1];return Object.assign(e,c({},s,decodeURIComponent(a)))},{})},buildUrlParams:function(e){return d.is.object(e)?Object.keys(e).map(function(t){return encodeURIComponent(t)+"="+encodeURIComponent(e[t])}).join("&"):""},stripHTML:function(e){var t=document.createDocumentFragment(),i=document.createElement("div");return t.appendChild(i),i.innerHTML=e,t.firstChild.innerText},getAspectRatio:function(e,t){var i=function e(t,i){return 0===i?t:e(i,t%i)}(e,t);return e/i+":"+t/i},transitionEnd:(e=document.createElement("span"),t=Object.keys({WebkitTransition:"webkitTransitionEnd",MozTransition:"transitionend",OTransition:"oTransitionEnd otransitionend",transition:"transitionend"}).find(function(t){return void 0!==e.style[t]}),"string"==typeof t&&t),repaint:function(e){window.setTimeout(function(){e.setAttribute("hidden",""),e.offsetHeight,e.removeAttribute("hidden")},0)}},p={audio:"canPlayType"in document.createElement("audio"),video:"canPlayType"in document.createElement("video"),check:function(e,t,i){var n=!1,s=!1,a=d.getBrowser(),l=a.isIPhone&&i&&p.inline;switch(t+":"+e){case"html5:video":s=(n=p.video)&&p.rangeInput&&(!a.isIPhone||l);break;case"html5:audio":s=(n=p.audio)&&p.rangeInput;break;case"youtube:video":n=!0,s=p.rangeInput&&(!a.isIPhone||l);break;case"vimeo:video":n=!0,s=p.rangeInput&&!a.isIPhone;break;default:s=(n=p.audio&&p.video)&&p.rangeInput}return{api:n,ui:s}},pip:!d.getBrowser().isIPhone&&d.is.function(d.createElement("video").webkitSetPresentationMode),airplay:d.is.function(window.WebKitPlaybackTargetAvailabilityEvent),inline:"playsInline"in document.createElement("video"),mime:function(e){var t=this.media;try{if(!this.isHTML5||!d.is.function(t.canPlayType))return!1;if(this.isVideo)switch(e){case"video/webm":return t.canPlayType('video/webm; codecs="vp8, vorbis"').replace(/no/,"");case"video/mp4":return t.canPlayType('video/mp4; codecs="avc1.42E01E, mp4a.40.2"').replace(/no/,"");case"video/ogg":return t.canPlayType('video/ogg; codecs="theora"').replace(/no/,"");default:return!1}else if(this.isAudio)switch(e){case"audio/mpeg":return t.canPlayType("audio/mpeg;").replace(/no/,"");case"audio/ogg":return t.canPlayType('audio/ogg; codecs="vorbis"').replace(/no/,"");case"audio/wav":return t.canPlayType('audio/wav; codecs="1"').replace(/no/,"");default:return!1}}catch(e){return!1}return!1},textTracks:"textTracks"in document.createElement("video"),passiveListeners:function(){var e=!1;try{var t=Object.defineProperty({},"passive",{get:function(){return e=!0,null}});window.addEventListener("test",null,t)}catch(e){}return e}(),rangeInput:(i=document.createElement("input"),i.type="range","range"===i.type),touch:"ontouchstart"in document.documentElement,transitions:!1!==d.transitionEnd,reducedMotion:"matchMedia"in window&&window.matchMedia("(prefers-reduced-motion)").matches},h=function(){},m=function(){function e(){var t=arguments.length>0&&void 0!==arguments[0]&&arguments[0];o(this,e),this.enabled=window.console&&t,this.enabled&&this.log("Debugging enabled")}return r(e,[{key:"log",get:function(){return this.enabled?Function.prototype.bind.call(console.log,console):h}},{key:"warn",get:function(){return this.enabled?Function.prototype.bind.call(console.warn,console):h}},{key:"error",get:function(){return this.enabled?Function.prototype.bind.call(console.error,console):h}}]),e}(),g=function(){function e(t){o(this,e),this.enabled=t.config.storage.enabled,this.key=t.config.storage.key}return r(e,[{key:"get",value:function(t){var i=window.localStorage.getItem(this.key);if(!e.supported||d.is.empty(i))return null;var n=JSON.parse(i);return d.is.string(t)&&t.length?n[t]:n}},{key:"set",value:function(t){if(e.supported&&this.enabled&&d.is.object(t)){var i=this.get();d.is.empty(i)&&(i={}),d.extend(i,t),window.localStorage.setItem(this.key,JSON.stringify(i))}}}],[{key:"supported",get:function(){if(!("localStorage"in window))return!1;try{return window.localStorage.setItem("___test","___test"),window.localStorage.removeItem("___test"),!0}catch(e){return!1}}}]),e}(),f=(n=!1,d.is.function(document.cancelFullScreen)?n="":["webkit","o","moz","ms","khtml"].some(function(e){return d.is.function(document[e+"CancelFullScreen"])?(n=e,!0):!(!d.is.function(document.msExitFullscreen)||!document.msFullscreenEnabled||(n="ms",0))}),n),y={prefix:f,enabled:document.fullscreenEnabled||document.webkitFullscreenEnabled||document.mozFullScreenEnabled||document.msFullscreenEnabled,eventType:"ms"===f?"MSFullscreenChange":f+"fullscreenchange",isFullScreen:function(e){if(!y.enabled)return!1;var t=d.is.nullOrUndefined(e)?document.body:e;switch(f){case"":return document.fullscreenElement===t;case"moz":return document.mozFullScreenElement===t;default:return document[f+"FullscreenElement"]===t}},requestFullScreen:function(e){if(!y.enabled)return!1;var t=d.is.nullOrUndefined(e)?document.body:e;return f.length?t[f+("ms"===f?"RequestFullscreen":"RequestFullScreen")]():t.requestFullScreen()},cancelFullScreen:function(){return!!y.enabled&&(f.length?document[f+("ms"===f?"ExitFullscreen":"CancelFullScreen")]():document.cancelFullScreen())},element:function(){return y.enabled?f.length?document[f+"FullscreenElement"]:document.fullscreenElement:null},setup:function(){if(this.supported.ui&&!this.isAudio&&this.config.fullscreen.enabled){var e=y.enabled;e||this.config.fullscreen.fallback&&!d.inFrame()?(this.debug.log((e?"Native":"Fallback")+" fullscreen enabled"),d.toggleClass(this.elements.container,this.config.classNames.fullscreen.enabled,!0)):this.debug.log("Fullscreen not supported and fallback disabled"),this.elements.buttons&&this.elements.buttons.fullscreen&&d.toggleState(this.elements.buttons.fullscreen,!1),d.trapFocus.call(this)}}},b=d.getBrowser(),v={global:function(){var e=this,t=null,i=function(i){var n,s=(n=i).keyCode?n.keyCode:n.which,a="keydown"===i.type,l=a&&s===t;if(!(i.altKey||i.ctrlKey||i.metaKey||i.shiftKey)&&d.is.number(s)){if(a){var o=d.getFocusElement();if(d.is.element(o)&&d.matches(o,e.config.selectors.editable))return;switch([48,49,50,51,52,53,54,56,57,32,75,38,40,77,39,37,70,67,73,76,79].includes(s)&&(i.preventDefault(),i.stopPropagation()),s){case 48:case 49:case 50:case 51:case 52:case 53:case 54:case 55:case 56:case 57:l||(e.currentTime=e.duration/10*(s-48));break;case 32:case 75:l||e.togglePlay();break;case 38:e.increaseVolume(.1);break;case 40:e.decreaseVolume(.1);break;case 77:l||(e.muted=!e.muted);break;case 39:e.forward();break;case 37:e.rewind();break;case 70:e.toggleFullscreen();break;case 67:l||e.toggleCaptions();break;case 76:e.loop=!e.loop}!y.enabled&&e.fullscreen.active&&27===s&&e.toggleFullscreen(),t=s}else t=null}};this.config.keyboard.global?d.on(window,"keydown keyup",i,!1):this.config.keyboard.focused&&d.on(this.elements.container,"keydown keyup",i,!1),d.on(this.elements.container,"focusout",function(t){d.toggleClass(t.target,e.config.classNames.tabFocus,!1)}),d.on(this.elements.container,"keydown",function(t){9===t.keyCode&&window.setTimeout(function(){d.toggleClass(d.getFocusElement(),e.config.classNames.tabFocus,!0)},0)}),this.config.hideControls&&d.on(this.elements.container,"mouseenter mouseleave mousemove touchstart touchend touchmove enterfullscreen exitfullscreen",function(t){e.toggleControls(t)}),y.enabled&&(d.on(document,y.eventType,function(t){e.toggleFullscreen(t)}),d.on(this.elements.container,"dblclick",function(t){e.toggleFullscreen(t)}))},media:function(){var e=this;if(d.on(this.media,"timeupdate seeking",function(t){return k.timeUpdate.call(e,t)}),d.on(this.media,"durationchange loadedmetadata",function(t){return k.durationUpdate.call(e,t)}),d.on(this.media,"loadeddata",function(){d.toggleHidden(e.elements.volume,!e.hasAudio),d.toggleHidden(e.elements.buttons.mute,!e.hasAudio)}),d.on(this.media,"ended",function(){e.isHTML5&&e.isVideo&&e.config.showPosterOnEnd&&(e.restart(),e.media.load())}),d.on(this.media,"progress playing",function(t){return k.updateProgress.call(e,t)}),d.on(this.media,"volumechange",function(t){return k.updateVolume.call(e,t)}),d.on(this.media,"playing play pause ended",function(t){return k.checkPlaying.call(e,t)}),d.on(this.media,"stalled waiting canplay seeked playing",function(t){return k.checkLoading.call(e,t)}),this.supported.ui&&this.config.clickToPlay&&!this.isAudio){var t=d.getElement.call(this,"."+this.config.classNames.video);if(!d.is.element(t))return;d.on(t,"click",function(){e.config.hideControls&&p.touch&&!e.paused||(e.paused?e.play():e.ended?(e.restart(),e.play()):e.pause())})}this.supported.ui&&this.config.disableContextMenu&&d.on(this.media,"contextmenu",function(e){e.preventDefault()},!1),d.on(this.media,"volumechange",function(){e.storage.set({volume:e.volume,muted:e.muted})}),d.on(this.media,"ratechange",function(){E.updateSetting.call(e,"speed"),e.storage.set({speed:e.speed})}),d.on(this.media,"qualitychange",function(){E.updateSetting.call(e,"quality"),e.storage.set({quality:e.quality})}),d.on(this.media,"languagechange",function(){E.updateSetting.call(e,"captions"),e.storage.set({language:e.language})}),d.on(this.media,"captionsenabled captionsdisabled",function(){E.updateSetting.call(e,"captions"),e.storage.set({captions:e.captions.active})}),d.on(this.media,this.config.events.concat(["keyup","keydown"]).join(" "),function(t){var i={};"error"===t.type&&(i=e.media.error),d.dispatchEvent.call(e,e.elements.container,t.type,!0,i)})},controls:function(){var e=this,t=b.isIE?"change":"input",i=function(t,i,n){var s=e.config.listeners[i];d.is.function(s)&&s.call(e,t),!t.defaultPrevented&&d.is.function(n)&&n.call(e,t)};d.on(this.elements.buttons.play,"click",function(t){return i(t,"play",function(){e.togglePlay()})}),d.on(this.elements.buttons.restart,"click",function(t){return i(t,"restart",function(){e.restart()})}),d.on(this.elements.buttons.rewind,"click",function(t){return i(t,"rewind",function(){e.rewind()})}),d.on(this.elements.buttons.forward,"click",function(t){return i(t,"forward",function(){e.forward()})}),d.on(this.elements.buttons.mute,"click",function(t){return i(t,"mute",function(){e.muted=!e.muted})}),d.on(this.elements.buttons.captions,"click",function(t){return i(t,"captions",function(){e.toggleCaptions()})}),d.on(this.elements.buttons.fullscreen,"click",function(t){return i(t,"fullscreen",function(){e.toggleFullscreen()})}),d.on(this.elements.buttons.pip,"click",function(t){return i(t,"pip",function(){e.pip="toggle"})}),d.on(this.elements.buttons.airplay,"click",function(t){return i(t,"airplay",function(){e.airplay()})}),d.on(this.elements.buttons.settings,"click",function(t){E.toggleMenu.call(e,t)}),d.on(document.documentElement,"click",function(t){E.toggleMenu.call(e,t)}),d.on(this.elements.settings.form,"click",function(t){t.stopPropagation(),d.matches(t.target,e.config.selectors.inputs.language)?i(t,"language",function(){e.language=t.target.value}):d.matches(t.target,e.config.selectors.inputs.quality)?i(t,"quality",function(){e.quality=t.target.value}):d.matches(t.target,e.config.selectors.inputs.speed)?i(t,"speed",function(){e.speed=parseFloat(t.target.value)}):E.showTab.call(e,t)}),d.on(this.elements.inputs.seek,t,function(t){return i(t,"seek",function(){e.currentTime=t.target.value/t.target.max*e.duration})}),this.config.toggleInvert&&!d.is.element(this.elements.display.duration)&&d.on(this.elements.display.currentTime,"click",function(){0!==e.currentTime&&(e.config.invertTime=!e.config.invertTime,k.timeUpdate.call(e))}),d.on(this.elements.inputs.volume,t,function(t){return i(t,"volume",function(){e.volume=t.target.value})}),b.isWebkit&&d.on(d.getElements.call(this,'input[type="range"]'),"input",function(t){E.updateRangeFill.call(e,t.target)}),d.on(this.elements.progress,"mouseenter mouseleave mousemove",function(t){return E.updateSeekTooltip.call(e,t)}),this.config.hideControls&&(d.on(this.elements.controls,"mouseenter mouseleave",function(t){e.elements.controls.hover="mouseenter"===t.type}),d.on(this.elements.controls,"mousedown mouseup touchstart touchend touchcancel",function(t){e.elements.controls.pressed=["mousedown","touchstart"].includes(t.type)}),d.on(this.elements.controls,"focusin focusout",function(t){e.toggleControls(t)})),d.on(this.elements.inputs.volume,"wheel",function(t){return i(t,"volume",function(){var i=t.webkitDirectionInvertedFromDevice,n=0;(t.deltaY<0||t.deltaX>0)&&(i?(e.decreaseVolume(.02),n=-1):(e.increaseVolume(.02),n=1)),(t.deltaY>0||t.deltaX<0)&&(i?(e.increaseVolume(.02),n=1):(e.decreaseVolume(.02),n=-1)),(1===n&&e.media.volume<1||-1===n&&e.media.volume>0)&&t.preventDefault()})},!1)}},k={addStyleHook:function(){d.toggleClass(this.elements.container,this.config.selectors.container.replace(".",""),!0),d.toggleClass(this.elements.container,this.config.classNames.uiSupported,this.supported.ui)},toggleNativeControls:function(){arguments.length>0&&void 0!==arguments[0]&&arguments[0]&&this.isHTML5?this.media.setAttribute("controls",""):this.media.removeAttribute("controls")},build:function(){if(v.media.call(this),!this.supported.ui)return this.debug.warn("Basic support only for "+this.provider+" "+this.type),d.removeElement.call(this,"controls"),d.removeElement.call(this,"buttons.play"),void k.toggleNativeControls.call(this,!0);d.is.element(this.elements.controls)||(E.inject.call(this),v.controls.call(this)),d.is.element(this.elements.controls)&&(k.toggleNativeControls.call(this),y.setup.call(this),T.setup.call(this),this.volume=null,this.muted=null,this.speed=null,this.loop=null,this.options.quality=[],k.timeUpdate.call(this),k.checkPlaying.call(this),this.ready=!0,d.dispatchEvent.call(this,this.media,"ready"),k.setTitle.call(this))},setTitle:function(){var e=this.config.i18n.play;if(d.is.string(this.config.title)&&!d.is.empty(this.config.title)&&(e+=", "+this.config.title,this.elements.container.setAttribute("aria-label",this.config.title)),d.is.nodeList(this.elements.buttons.play)&&Array.from(this.elements.buttons.play).forEach(function(t){t.setAttribute("aria-label",e)}),this.isEmbed){var t=d.getElement.call(this,"iframe");if(!d.is.element(t))return;var i=d.is.empty(this.config.title)?"video":this.config.title;t.setAttribute("title",this.config.i18n.frameTitle.replace("{title}",i))}},checkPlaying:function(){var e=this;d.toggleClass(this.elements.container,this.config.classNames.playing,this.playing),d.toggleClass(this.elements.container,this.config.classNames.stopped,this.paused),d.is.nodeList(this.elements.buttons.play)&&Array.from(this.elements.buttons.play).forEach(function(t){return d.toggleState(t,e.playing)}),this.toggleControls(!this.playing)},checkLoading:function(e){var t=this;this.loading=["stalled","waiting"].includes(e.type),clearTimeout(this.timers.loading),this.timers.loading=setTimeout(function(){d.toggleClass(t.elements.container,t.config.classNames.loading,t.loading),t.toggleControls(t.loading)},this.loading?250:0)},checkFailed:function(){var e=this;this.failed=3===this.media.networkState,this.failed&&(d.toggleClass(this.elements.container,this.config.classNames.loading,!1),d.toggleClass(this.elements.container,this.config.classNames.error,!0)),clearTimeout(this.timers.failed),this.timers.loading=setTimeout(function(){d.toggleClass(e.elements.container,e.config.classNames.loading,e.loading),e.toggleControls(e.loading)},this.loading?250:0)},updateVolume:function(){this.supported.ui&&(d.is.element(this.elements.inputs.volume)&&k.setRange.call(this,this.elements.inputs.volume,this.muted?0:this.volume),d.is.element(this.elements.buttons.mute)&&d.toggleState(this.elements.buttons.mute,this.muted||0===this.volume))},setRange:function(e){var t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:0;d.is.element(e)&&(e.value=t,E.updateRangeFill.call(this,e))},setProgress:function(e,t){var i=d.is.number(t)?t:0,n=d.is.element(e)?e:this.elements.display.buffer;if(d.is.element(n)){n.value=i;var s=n.getElementsByTagName("span")[0];d.is.element(s)&&(s.childNodes[0].nodeValue=i)}},updateProgress:function(e){var t=this;if(this.supported.ui&&d.is.event(e)){var i,n=0;if(e)switch(e.type){case"timeupdate":case"seeking":n=d.getPercentage(this.currentTime,this.duration),"timeupdate"===e.type&&k.setRange.call(this,this.elements.inputs.seek,n);break;case"playing":case"progress":n=(i=t.media.buffered)&&i.length?d.getPercentage(i.end(0),t.duration):d.is.number(i)?100*i:0,k.setProgress.call(this,this.elements.display.buffer,n)}}},updateTimeDisplay:function(){var e=arguments.length>0&&void 0!==arguments[0]?arguments[0]:null,t=arguments.length>1&&void 0!==arguments[1]?arguments[1]:0,i=arguments.length>2&&void 0!==arguments[2]&&arguments[2];if(d.is.element(e)&&d.is.number(t)){var n=function(e){return("0"+e).slice(-2)},s=function(e){return parseInt(e/60/60%60,10)},a=s(t),l=parseInt(t/60%60,10),o=parseInt(t%60,10);s(this.duration)>0?a+=":":a="",e.textContent=(i?"-":"")+a+n(l)+":"+n(o)}},timeUpdate:function(e){var t=!d.is.element(this.elements.display.duration)&&this.config.invertTime;k.updateTimeDisplay.call(this,this.elements.display.currentTime,t?this.duration-this.currentTime:this.currentTime,t),e&&"timeupdate"===e.type&&this.media.seeking||k.updateProgress.call(this,e)},durationUpdate:function(){this.supported.ui&&(!d.is.element(this.elements.display.duration)&&this.config.displayDuration&&this.paused&&k.updateTimeDisplay.call(this,this.elements.display.currentTime,this.duration),d.is.element(this.elements.display.duration)&&k.updateTimeDisplay.call(this,this.elements.display.duration,this.duration),E.updateSeekTooltip.call(this))}},w=d.getBrowser(),E={updateRangeFill:function(e){if(w.isWebkit){var t=d.is.event(e)?e.target:e;d.is.element(t)&&"range"===t.getAttribute("type")&&t.style.setProperty("--value",t.value/t.max*100+"%")}},getIconUrl:function(){return{url:this.config.iconUrl,absolute:0===this.config.iconUrl.indexOf("http")||w.isIE&&!window.svg4everybody}},createIcon:function(e,t){var i="http://www.w3.org/2000/svg",n=E.getIconUrl.call(this),s=(n.absolute?"":n.url)+"#"+this.config.iconPrefix,a=document.createElementNS(i,"svg");d.setAttributes(a,d.extend(t,{role:"presentation"}));var l=document.createElementNS(i,"use"),o=s+"-"+e;return"href"in l?l.setAttributeNS("http://www.w3.org/1999/xlink","href",o):l.setAttributeNS("http://www.w3.org/1999/xlink","xlink:href",o),a.appendChild(l),a},createLabel:function(e,t){var i=this.config.i18n[e],n=Object.assign({},t);switch(e){case"pip":i="PIP";break;case"airplay":i="AirPlay"}return"class"in n?n.class+=" "+this.config.classNames.hidden:n.class=this.config.classNames.hidden,d.createElement("span",n,i)},createBadge:function(e){if(d.is.empty(e))return null;var t=d.createElement("span",{class:this.config.classNames.menu.value});return t.appendChild(d.createElement("span",{class:this.config.classNames.menu.badge},e)),t},createButton:function(e,t){var i=d.createElement("button"),n=Object.assign({},t),s=e,a=!1,l=void 0,o=void 0,r=void 0,c=void 0;switch("type"in n||(n.type="button"),"class"in n?n.class.includes(this.config.classNames.control)&&(n.class+=" "+this.config.classNames.control):n.class=this.config.classNames.control,s){case"play":a=!0,l="play",r="pause",o="play",c="pause";break;case"mute":a=!0,l="mute",r="unmute",o="volume",c="muted";break;case"captions":a=!0,l="enableCaptions",r="disableCaptions",o="captions-off",c="captions-on";break;case"fullscreen":a=!0,l="enterFullscreen",r="exitFullscreen",o="enter-fullscreen",c="exit-fullscreen";break;case"play-large":n.class+=" "+this.config.classNames.control+"--overlaid",s="play",l="play",o="play";break;default:l=s,o=s}return a?(i.appendChild(E.createIcon.call(this,c,{class:"icon--pressed"})),i.appendChild(E.createIcon.call(this,o,{class:"icon--not-pressed"})),i.appendChild(E.createLabel.call(this,r,{class:"label--pressed"})),i.appendChild(E.createLabel.call(this,l,{class:"label--not-pressed"})),n["aria-pressed"]=!1,n["aria-label"]=this.config.i18n[l]):(i.appendChild(E.createIcon.call(this,o)),i.appendChild(E.createLabel.call(this,l))),d.extend(n,d.getAttributesFromSelector(this.config.selectors.buttons[s],n)),d.setAttributes(i,n),this.elements.buttons[s]=i,i},createRange:function(e,t){var i=d.createElement("label",{for:t.id,class:this.config.classNames.hidden},this.config.i18n[e]),n=d.createElement("input",d.extend(d.getAttributesFromSelector(this.config.selectors.inputs[e]),{type:"range",min:0,max:100,step:.01,value:0,autocomplete:"off"},t));return this.elements.inputs[e]=n,E.updateRangeFill.call(this,n),{label:i,input:n}},createProgress:function(e,t){var i=d.createElement("progress",d.extend(d.getAttributesFromSelector(this.config.selectors.display[e]),{min:0,max:100,value:0},t));if("volume"!==e){i.appendChild(d.createElement("span",null,"0"));var n="";switch(e){case"played":n=this.config.i18n.played;break;case"buffer":n=this.config.i18n.buffered}i.textContent="% "+n.toLowerCase()}return this.elements.display[e]=i,i},createTime:function(e){var t=d.createElement("div",{class:"plyr__time"});return t.appendChild(d.createElement("span",{class:this.config.classNames.hidden},this.config.i18n[e])),t.appendChild(d.createElement("span",d.getAttributesFromSelector(this.config.selectors.display[e]),"00:00")),this.elements.display[e]=t,t},createMenuItem:function(e,t,i,n){var s=arguments.length>4&&void 0!==arguments[4]?arguments[4]:null,a=arguments.length>5&&void 0!==arguments[5]&&arguments[5],l=d.createElement("li"),o=d.createElement("label",{class:this.config.classNames.control}),r=d.createElement("input",d.extend(d.getAttributesFromSelector(this.config.selectors.inputs[i]),{type:"radio",name:"plyr-"+i,value:e,checked:a,class:"plyr__sr-only"})),c=d.createElement("span",{"aria-hidden":!0});o.appendChild(r),o.appendChild(c),o.insertAdjacentHTML("beforeend",n),d.is.element(s)&&o.appendChild(s),l.appendChild(o),t.appendChild(l)},updateSeekTooltip:function(e){if(this.config.tooltips.seek&&d.is.element(this.elements.inputs.seek)&&d.is.element(this.elements.display.seekTooltip)&&0!==this.duration){var t=0,i=this.elements.inputs.seek.getBoundingClientRect(),n=this.config.classNames.tooltip+"--visible";if(d.is.event(e))t=100/i.width*(e.pageX-i.left);else{if(!d.hasClass(this.elements.display.seekTooltip,n))return;t=parseFloat(this.elements.display.seekTooltip.style.left,10)}t<0?t=0:t>100&&(t=100),k.updateTimeDisplay.call(this,this.elements.display.seekTooltip,this.duration/100*t),this.elements.display.seekTooltip.style.left=t+"%",d.is.event(e)&&["mouseenter","mouseleave"].includes(e.type)&&d.toggleClass(this.elements.display.seekTooltip,n,"mouseenter"===e.type)}},toggleTab:function(e,t){var i=this.elements.settings.tabs[e],n=this.elements.settings.panes[e];d.toggleHidden(i,!t),d.toggleHidden(n,!t)},setQualityMenu:function(e){var t=this,i=this.elements.settings.panes.quality.querySelector("ul");d.is.array(e)?this.options.quality=e.filter(function(e){return t.config.quality.options.includes(e)}):this.options.quality=this.config.quality.options;var n=!d.is.empty(this.options.quality)&&this.isYouTube;if(E.toggleTab.call(this,"quality",n),n){d.emptyElement(i);this.options.quality.forEach(function(e){return E.createMenuItem.call(t,e,i,"quality",E.getLabel.call(t,"quality",e),function(e){var i="";switch(e){case"hd2160":i="4K";break;case"hd1440":i="WQHD";break;case"hd1080":case"hd720":i="HD"}return i.length?E.createBadge.call(t,i):null}(e))}),E.updateSetting.call(this,"quality",i)}},getLabel:function(e,t){switch(e){case"speed":return 1===t?"Normal":t+"×";case"quality":switch(t){case"hd2160":return"2160P";case"hd1440":return"1440P";case"hd1080":return"1080P";case"hd720":return"720P";case"large":return"480P";case"medium":return"360P";case"small":return"240P";case"tiny":return"Tiny";case"default":return"Auto";default:return t}case"captions":return E.getLanguage.call(this);default:return null}},updateSetting:function(e,t){var i=this.elements.settings.panes[e],n=null,s=t;switch(e){case"captions":n=this.captions.active?this.captions.language:"";break;default:if(n=this[e],d.is.empty(n)&&(n=this.config[e].default),!this.options[e].includes(n))return void this.debug.warn("Unsupported value of '"+n+"' for "+e);if(!this.config[e].options.includes(n))return void this.debug.warn("Disabled value of '"+n+"' for "+e)}(d.is.element(s)||(s=i&&i.querySelector("ul")),d.is.empty(n))||(this.elements.settings.tabs[e].querySelector("."+this.config.classNames.menu.value).innerHTML=E.getLabel.call(this,e,n));var a=s&&s.querySelector('input[value="'+n+'"]');d.is.element(a)&&(a.checked=!0)},getLanguage:function(){if(!this.supported.ui)return null;if(!p.textTracks||!T.getTracks.call(this).length)return this.config.i18n.none;if(this.captions.active){var e=T.getCurrentTrack.call(this);if(d.is.track(e))return e.label}return this.config.i18n.disabled},setCaptionsMenu:function(){var e=this,t=this.elements.settings.panes.captions.querySelector("ul"),i=T.getTracks.call(this).length;if(E.toggleTab.call(this,"captions",i),d.emptyElement(t),i){var n=T.getTracks.call(this).map(function(e){return{language:e.language,label:d.is.empty(e.label)?e.language.toUpperCase():e.label}});n.unshift({language:"",label:this.config.i18n.none}),n.forEach(function(i){E.createMenuItem.call(e,i.language,t,"language",i.label||i.language,E.createBadge.call(e,i.language.toUpperCase()),i.language.toLowerCase()===e.captions.language.toLowerCase())}),E.updateSetting.call(this,"captions",t)}},setSpeedMenu:function(){var e=this;d.is.object(this.options.speed)&&Object.keys(this.options.speed).length||(this.options.speed=[.5,.75,1,1.25,1.5,1.75,2]),this.options.speed=this.options.speed.filter(function(t){return e.config.speed.options.includes(t)});var t=!d.is.empty(this.options.speed);if(E.toggleTab.call(this,"speed",t),t){var i=this.elements.settings.panes.speed.querySelector("ul");d.toggleHidden(this.elements.settings.tabs.speed,!1),d.toggleHidden(this.elements.settings.panes.speed,!1),d.emptyElement(i),this.options.speed.forEach(function(t){return E.createMenuItem.call(e,t,i,"speed",E.getLabel.call(e,"speed",t))}),E.updateSetting.call(this,"speed",i)}},toggleMenu:function(e){var t=this.elements.settings.form,i=this.elements.buttons.settings,n=d.is.boolean(e)?e:d.is.element(t)&&"true"===t.getAttribute("aria-hidden");if(d.is.event(e)){var s=d.is.element(t)&&t.contains(e.target),a=e.target===this.elements.buttons.settings;if(s||!s&&!a&&n)return;a&&e.stopPropagation()}d.is.element(i)&&i.setAttribute("aria-expanded",n),d.is.element(t)&&(t.setAttribute("aria-hidden",!n),d.toggleClass(this.elements.container,this.config.classNames.menu.open,n),n?t.removeAttribute("tabindex"):t.setAttribute("tabindex",-1))},getTabSize:function(e){var t=e.cloneNode(!0);t.style.position="absolute",t.style.opacity=0,t.setAttribute("aria-hidden",!1),Array.from(t.querySelectorAll("input[name]")).forEach(function(e){var t=e.getAttribute("name");e.setAttribute("name",t+"-clone")}),e.parentNode.appendChild(t);var i=t.scrollWidth,n=t.scrollHeight;return d.removeElement(t),{width:i,height:n}},showTab:function(e){var t=this.elements.settings.menu,i=e.target,n="false"===i.getAttribute("aria-expanded"),s=document.getElementById(i.getAttribute("aria-controls"));if(d.is.element(s)&&"tabpanel"===s.getAttribute("role")){var a=t.querySelector('[role="tabpanel"][aria-hidden="false"]'),l=a.parentNode;if(Array.from(t.querySelectorAll('[aria-controls="'+a.getAttribute("id")+'"]')).forEach(function(e){e.setAttribute("aria-expanded",!1)}),p.transitions&&!p.reducedMotion){l.style.width=a.scrollWidth+"px",l.style.height=a.scrollHeight+"px";var o=E.getTabSize.call(this,s);d.on(l,d.transitionEnd,function e(t){t.target===l&&["width","height"].includes(t.propertyName)&&(l.style.width="",l.style.height="",d.off(l,d.transitionEnd,e))}),l.style.width=o.width+"px",l.style.height=o.height+"px"}a.setAttribute("aria-hidden",!0),a.setAttribute("tabindex",-1),s.setAttribute("aria-hidden",!n),i.setAttribute("aria-expanded",n),s.removeAttribute("tabindex"),s.querySelectorAll("button:not(:disabled), input:not(:disabled), [tabindex]")[0].focus()}},create:function(e){var t=this;if(d.is.empty(this.config.controls))return null;var i=d.createElement("div",d.getAttributesFromSelector(this.config.selectors.controls.wrapper));if(this.config.controls.includes("restart")&&i.appendChild(E.createButton.call(this,"restart")),this.config.controls.includes("rewind")&&i.appendChild(E.createButton.call(this,"rewind")),this.config.controls.includes("play")&&i.appendChild(E.createButton.call(this,"play")),this.config.controls.includes("fast-forward")&&i.appendChild(E.createButton.call(this,"fast-forward")),this.config.controls.includes("progress")){var n=d.createElement("div",d.getAttributesFromSelector(this.config.selectors.progress)),s=E.createRange.call(this,"seek",{id:"plyr-seek-"+e.id});if(n.appendChild(s.label),n.appendChild(s.input),n.appendChild(E.createProgress.call(this,"buffer")),this.config.tooltips.seek){var a=d.createElement("span",{role:"tooltip",class:this.config.classNames.tooltip},"00:00");n.appendChild(a),this.elements.display.seekTooltip=a}this.elements.progress=n,i.appendChild(this.elements.progress)}if(this.config.controls.includes("current-time")&&i.appendChild(E.createTime.call(this,"currentTime")),this.config.controls.includes("duration")&&i.appendChild(E.createTime.call(this,"duration")),this.config.controls.includes("mute")&&i.appendChild(E.createButton.call(this,"mute")),this.config.controls.includes("volume")){var l=d.createElement("div",{class:"plyr__volume"}),o={max:1,step:.05,value:this.config.volume},r=E.createRange.call(this,"volume",d.extend(o,{id:"plyr-volume-"+e.id}));l.appendChild(r.label),l.appendChild(r.input),this.elements.volume=l,i.appendChild(l)}if(this.config.controls.includes("captions")&&i.appendChild(E.createButton.call(this,"captions")),this.config.controls.includes("settings")&&!d.is.empty(this.config.settings)){var c=d.createElement("div",{class:"plyr__menu"});c.appendChild(E.createButton.call(this,"settings",{id:"plyr-settings-toggle-"+e.id,"aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id,"aria-expanded":!1}));var u=d.createElement("form",{class:"plyr__menu__container",id:"plyr-settings-"+e.id,"aria-hidden":!0,"aria-labelled-by":"plyr-settings-toggle-"+e.id,role:"tablist",tabindex:-1}),h=d.createElement("div"),m=d.createElement("div",{id:"plyr-settings-"+e.id+"-home","aria-hidden":!1,"aria-labelled-by":"plyr-settings-toggle-"+e.id,role:"tabpanel"}),g=d.createElement("ul",{role:"tablist"});this.config.settings.forEach(function(i){var n=d.createElement("li",{role:"tab",hidden:""}),s=d.createElement("button",d.extend(d.getAttributesFromSelector(t.config.selectors.buttons.settings),{type:"button",class:t.config.classNames.control+" "+t.config.classNames.control+"--forward",id:"plyr-settings-"+e.id+"-"+i+"-tab","aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id+"-"+i,"aria-expanded":!1}),t.config.i18n[i]),a=d.createElement("span",{class:t.config.classNames.menu.value});a.innerHTML=e[i],s.appendChild(a),n.appendChild(s),g.appendChild(n),t.elements.settings.tabs[i]=n}),m.appendChild(g),h.appendChild(m),this.config.settings.forEach(function(i){var n=d.createElement("div",{id:"plyr-settings-"+e.id+"-"+i,"aria-hidden":!0,"aria-labelled-by":"plyr-settings-"+e.id+"-"+i+"-tab",role:"tabpanel",tabindex:-1,hidden:""}),s=d.createElement("button",{type:"button",class:t.config.classNames.control+" "+t.config.classNames.control+"--back","aria-haspopup":!0,"aria-controls":"plyr-settings-"+e.id+"-home","aria-expanded":!1},t.config.i18n[i]);n.appendChild(s);var a=d.createElement("ul");n.appendChild(a),h.appendChild(n),t.elements.settings.panes[i]=n}),u.appendChild(h),c.appendChild(u),i.appendChild(c),this.elements.settings.form=u,this.elements.settings.menu=c}return this.config.controls.includes("pip")&&p.pip&&i.appendChild(E.createButton.call(this,"pip")),this.config.controls.includes("airplay")&&p.airplay&&i.appendChild(E.createButton.call(this,"airplay")),this.config.controls.includes("fullscreen")&&i.appendChild(E.createButton.call(this,"fullscreen")),this.config.controls.includes("play-large")&&this.elements.container.appendChild(E.createButton.call(this,"play-large")),this.elements.controls=i,this.config.controls.includes("settings")&&this.config.settings.includes("speed")&&E.setSpeedMenu.call(this),i},inject:function(){var e=this;if(this.config.loadSprite){var t=E.getIconUrl.call(this);t.absolute&&d.loadSprite(t.url,"sprite-plyr")}this.id=Math.floor(1e4*Math.random());var i=null;i=d.is.string(this.config.controls)?this.config.controls:d.is.function(this.config.controls)?this.config.controls({id:this.id,seektime:this.config.seekTime,title:this.config.title}):E.create.call(this,{id:this.id,seektime:this.config.seekTime,speed:this.speed,quality:this.quality,captions:E.getLanguage.call(this)});var n=void 0;if(d.is.string(this.config.selectors.controls.container)&&(n=document.querySelector(this.config.selectors.controls.container)),d.is.element(n)||(n=this.elements.container),d.is.element(i)?n.appendChild(i):n.insertAdjacentHTML("beforeend",i),d.is.element(this.elements.controls)&&d.findElements.call(this),window.navigator.userAgent.includes("Edge")&&d.repaint(n),this.config.tooltips.controls){var s=d.getElements.call(this,[this.config.selectors.controls.wrapper," ",this.config.selectors.labels," .",this.config.classNames.hidden].join(""));Array.from(s).forEach(function(t){d.toggleClass(t,e.config.classNames.hidden,!1),d.toggleClass(t,e.config.classNames.tooltip,!0),t.setAttribute("role","tooltip")})}}},T={setup:function(){if(this.supported.ui){var e=this.storage.get("language");if(d.is.empty(e)||(this.captions.language=e),d.is.empty(this.captions.language)&&(this.captions.language=this.config.captions.language.toLowerCase()),!d.is.boolean(this.captions.active)){var t=this.storage.get("captions");d.is.boolean(t)?this.captions.active=t:this.captions.active=this.config.captions.active}!this.isVideo||this.isYouTube||this.isHTML5&&!p.textTracks?this.config.controls.includes("settings")&&this.config.settings.includes("captions")&&E.setCaptionsMenu.call(this):(d.is.element(this.elements.captions)||(this.elements.captions=d.createElement("div",d.getAttributesFromSelector(this.config.selectors.captions)),d.insertAfter(this.elements.captions,this.elements.wrapper)),d.toggleClass(this.elements.container,this.config.classNames.captions.enabled,!d.is.empty(T.getTracks.call(this))),d.is.empty(T.getTracks.call(this))||(T.setLanguage.call(this),T.show.call(this),this.config.controls.includes("settings")&&this.config.settings.includes("captions")&&E.setCaptionsMenu.call(this)))}},setLanguage:function(){var e=this;if(this.isHTML5&&this.isVideo){T.getTracks.call(this).forEach(function(t){d.on(t,"cuechange",function(t){return T.setCue.call(e,t)}),t.mode="hidden"});var t=T.getCurrentTrack.call(this);d.is.track(t)&&Array.from(t.activeCues||[]).length&&T.setCue.call(this,t)}else this.isVimeo&&this.captions.active&&this.embed.enableTextTrack(this.language)},getTracks:function(){return d.is.nullOrUndefined(this.media)?[]:Array.from(this.media.textTracks||[]).filter(function(e){return["captions","subtitles"].includes(e.kind)})},getCurrentTrack:function(){var e=this;return T.getTracks.call(this).find(function(t){return t.language.toLowerCase()===e.language})},setCue:function(e){var t=d.is.event(e)?e.target:e,i=t.activeCues[0];t===T.getCurrentTrack.call(this)&&(d.is.cue(i)?T.setText.call(this,i.getCueAsHTML()):T.setText.call(this,null),d.dispatchEvent.call(this,this.media,"cuechange"))},setText:function(e){if(this.supported.ui)if(d.is.element(this.elements.captions)){var t=d.createElement("span");d.emptyElement(this.elements.captions);var i=d.is.nullOrUndefined(e)?"":e;d.is.string(i)?t.textContent=i.trim():t.appendChild(i),this.elements.captions.appendChild(t)}else this.debug.warn("No captions element to render to")},show:function(){if(d.is.element(this.elements.buttons.captions)){var e=this.storage.get("captions");d.is.boolean(e)?this.captions.active=e:e=this.config.captions.active,e&&(d.toggleClass(this.elements.container,this.config.classNames.captions.active,!0),d.toggleState(this.elements.buttons.captions,!0))}}},C={setup:function(){var e=this;d.toggleClass(this.elements.wrapper,this.config.classNames.embed,!0),C.setAspectRatio.call(this),d.is.object(window.YT)&&d.is.function(window.YT.Player)?C.ready.call(this):(d.loadScript(this.config.urls.youtube.api),window.onYouTubeReadyCallbacks=window.onYouTubeReadyCallbacks||[],window.onYouTubeReadyCallbacks.push(function(){C.ready.call(e)}),window.onYouTubeIframeAPIReady=function(){window.onYouTubeReadyCallbacks.forEach(function(e){e()})})},getTitle:function(e){var t=this;if(d.is.function(this.embed.getVideoData)){var i=this.embed.getVideoData().title;if(d.is.empty(i))return this.config.title=i,void k.setTitle.call(this)}var n=this.config.keys.google;d.is.string(n)&&!d.is.empty(n)&&fetch("https://www.googleapis.com/youtube/v3/videos?id="+e+"&key="+n+"&fields=items(snippet(title))&part=snippet").then(function(e){return e.ok?e.json():null}).then(function(e){null!==e&&d.is.object(e)&&(t.config.title=e.items[0].snippet.title,k.setTitle.call(t))}).catch(function(){})},setAspectRatio:function(){var e=this.config.ratio.split(":");this.elements.wrapper.style.paddingBottom=100/e[0]*e[1]+"%"},ready:function(){var e=this,t=e.media.getAttribute("id");if(d.is.empty(t)||!t.startsWith("youtube-")){var i=d.parseYouTubeId(e.media.getAttribute("src")),n=d.generateId(e.provider),s=d.createElement("div",{id:n});e.media=d.replaceElement(s,e.media),e.embed=new window.YT.Player(n,{videoId:i,playerVars:{autoplay:e.config.autoplay?1:0,controls:e.supported.ui?0:1,rel:0,showinfo:0,iv_load_policy:3,modestbranding:1,disablekb:1,playsinline:1,origin:window&&window.location.hostname,widget_referrer:window&&window.location.href,cc_load_policy:e.captions.active?1:0,cc_lang_pref:e.config.captions.language},events:{onError:function(t){if(!d.is.object(e.media.error)){var i={code:t.data};switch(t.data){case 2:i.message="The request contains an invalid parameter value. For example, this error occurs if you specify a video ID that does not have 11 characters, or if the video ID contains invalid characters, such as exclamation points or asterisks.";break;case 5:i.message="The requested content cannot be played in an HTML5 player or another error related to the HTML5 player has occurred.";break;case 100:i.message="The video requested was not found. This error occurs when a video has been removed (for any reason) or has been marked as private.";break;case 101:case 150:i.message="The owner of the requested video does not allow it to be played in embedded players.";break;default:i.message="An unknown error occured"}e.media.error=i,d.dispatchEvent.call(e,e.media,"error")}},onPlaybackQualityChange:function(t){var i=t.target;e.media.quality=i.getPlaybackQuality(),d.dispatchEvent.call(e,e.media,"qualitychange")},onPlaybackRateChange:function(t){var i=t.target;e.media.playbackRate=i.getPlaybackRate(),d.dispatchEvent.call(e,e.media,"ratechange")},onReady:function(t){var n=t.target;C.getTitle.call(e,i),e.media.play=function(){n.playVideo(),e.media.paused=!1},e.media.pause=function(){n.pauseVideo(),e.media.paused=!0},e.media.stop=function(){n.stopVideo(),e.media.paused=!0},e.media.duration=n.getDuration(),e.media.paused=!0,e.media.currentTime=0,Object.defineProperty(e.media,"currentTime",{get:function(){return Number(n.getCurrentTime())},set:function(t){e.media.seeking=!0,d.dispatchEvent.call(e,e.media,"seeking"),n.seekTo(t)}}),Object.defineProperty(e.media,"playbackRate",{get:function(){return n.getPlaybackRate()},set:function(e){n.setPlaybackRate(e)}}),Object.defineProperty(e.media,"quality",{get:function(){return n.getPlaybackQuality()},set:function(t){d.dispatchEvent.call(e,e.media,"qualityrequested",!1,{quality:t}),n.setPlaybackQuality(t)}});var s=e.config.volume;Object.defineProperty(e.media,"volume",{get:function(){return s},set:function(t){s=t,n.setVolume(100*s),d.dispatchEvent.call(e,e.media,"volumechange")}});var a=e.config.muted;Object.defineProperty(e.media,"muted",{get:function(){return a},set:function(t){var i=d.is.boolean(t)?t:a;a=i,n[i?"mute":"unMute"](),d.dispatchEvent.call(e,e.media,"volumechange")}}),Object.defineProperty(e.media,"currentSrc",{get:function(){return n.getVideoUrl()}}),Object.defineProperty(e.media,"ended",{get:function(){return e.currentTime===e.duration}}),e.options.speed=n.getAvailablePlaybackRates(),e.supported.ui&&e.media.setAttribute("tabindex",-1),d.dispatchEvent.call(e,e.media,"timeupdate"),d.dispatchEvent.call(e,e.media,"durationchange"),window.clearInterval(e.timers.buffering),e.timers.buffering=window.setInterval(function(){e.media.buffered=n.getVideoLoadedFraction(),(null===e.media.lastBuffered||e.media.lastBuffered<e.media.buffered)&&d.dispatchEvent.call(e,e.media,"progress"),e.media.lastBuffered=e.media.buffered,1===e.media.buffered&&(window.clearInterval(e.timers.buffering),d.dispatchEvent.call(e,e.media,"canplaythrough"))},200),window.setTimeout(function(){return k.build.call(e)},50)},onStateChange:function(t){var i=t.target;switch(window.clearInterval(e.timers.playing),t.data){case 0:e.media.paused=!0,e.media.loop?(i.stopVideo(),i.playVideo()):d.dispatchEvent.call(e,e.media,"ended");break;case 1:e.media.seeking&&d.dispatchEvent.call(e,e.media,"seeked"),e.media.seeking=!1,e.media.paused&&d.dispatchEvent.call(e,e.media,"play"),e.media.paused=!1,d.dispatchEvent.call(e,e.media,"playing"),e.timers.playing=window.setInterval(function(){d.dispatchEvent.call(e,e.media,"timeupdate")},50),e.media.duration!==i.getDuration()&&(e.media.duration=i.getDuration(),d.dispatchEvent.call(e,e.media,"durationchange")),E.setQualityMenu.call(e,i.getAvailableQualityLevels());break;case 2:e.media.paused=!0,d.dispatchEvent.call(e,e.media,"pause")}d.dispatchEvent.call(e,e.elements.container,"statechange",!1,{code:t.data})}}})}}},A={setup:function(){var e=this;d.toggleClass(this.elements.wrapper,this.config.classNames.embed,!0),A.setAspectRatio.call(this),d.is.object(window.Vimeo)?A.ready.call(this):d.loadScript(this.config.urls.vimeo.api,function(){A.ready.call(e)})},setAspectRatio:function(e){var t=d.is.string(e)?e.split(":"):this.config.ratio.split(":"),i=100/t[0]*t[1],n=(200-i)/4;this.elements.wrapper.style.paddingBottom=i+"%",this.media.style.transform="translateY(-"+n+"%)"},ready:function(){var e=this,t=this,i={loop:t.config.loop.active,autoplay:t.autoplay,byline:!1,portrait:!1,title:!1,speed:!0,transparent:0,gesture:"media"},n=d.buildUrlParams(i),s=d.parseVimeoId(t.media.getAttribute("src")),a=d.createElement("iframe"),l="https://player.vimeo.com/video/"+s+"?"+n;a.setAttribute("src",l),a.setAttribute("allowfullscreen",""),a.setAttribute("allowtransparency",""),a.setAttribute("allow","autoplay");var o=d.createElement("div");o.appendChild(a),t.media=d.replaceElement(o,t.media),t.embed=new window.Vimeo.Player(a),t.media.paused=!0,t.media.currentTime=0,t.media.play=function(){t.embed.play().then(function(){t.media.paused=!1})},t.media.pause=function(){t.embed.pause().then(function(){t.media.paused=!0})},t.media.stop=function(){t.embed.stop().then(function(){t.media.paused=!0,t.currentTime=0})};var r=t.media.currentTime;Object.defineProperty(t.media,"currentTime",{get:function(){return r},set:function(e){var i=t.media.paused;t.media.seeking=!0,d.dispatchEvent.call(t,t.media,"seeking"),t.embed.setCurrentTime(e),i&&t.pause()}});var c=t.config.speed.selected;Object.defineProperty(t.media,"playbackRate",{get:function(){return c},set:function(e){t.embed.setPlaybackRate(e).then(function(){c=e,d.dispatchEvent.call(t,t.media,"ratechange")})}});var u=t.config.volume;Object.defineProperty(t.media,"volume",{get:function(){return u},set:function(e){t.embed.setVolume(e).then(function(){u=e,d.dispatchEvent.call(t,t.media,"volumechange")})}});var p=t.config.muted;Object.defineProperty(t.media,"muted",{get:function(){return p},set:function(e){var i=!!d.is.boolean(e)&&e;t.embed.setVolume(i?0:t.config.volume).then(function(){p=i,d.dispatchEvent.call(t,t.media,"volumechange")})}});var h=t.config.loop;Object.defineProperty(t.media,"loop",{get:function(){return h},set:function(e){var i=d.is.boolean(e)?e:t.config.loop.active;t.embed.setLoop(i).then(function(){h=i})}});var m=void 0;t.embed.getVideoUrl().then(function(e){m=e}),Object.defineProperty(t.media,"currentSrc",{get:function(){return m}}),Object.defineProperty(t.media,"ended",{get:function(){return t.currentTime===t.duration}}),Promise.all([t.embed.getVideoWidth(),t.embed.getVideoHeight()]).then(function(t){var i=d.getAspectRatio(t[0],t[1]);A.setAspectRatio.call(e,i)}),t.embed.setAutopause(t.config.autopause).then(function(e){t.config.autopause=e}),t.embed.getVideoTitle().then(function(i){t.config.title=i,k.setTitle.call(e)}),t.embed.getCurrentTime().then(function(e){r=e,d.dispatchEvent.call(t,t.media,"timeupdate")}),t.embed.getDuration().then(function(e){t.media.duration=e,d.dispatchEvent.call(t,t.media,"durationchange")}),t.embed.getTextTracks().then(function(e){t.media.textTracks=e,T.setup.call(t)}),t.embed.on("cuechange",function(e){var i=null;e.cues.length&&(i=d.stripHTML(e.cues[0].text)),T.setText.call(t,i)}),t.embed.on("loaded",function(){d.is.element(t.embed.element)&&t.supported.ui&&t.embed.element.setAttribute("tabindex",-1)}),t.embed.on("play",function(){t.media.paused&&d.dispatchEvent.call(t,t.media,"play"),t.media.paused=!1,d.dispatchEvent.call(t,t.media,"playing")}),t.embed.on("pause",function(){t.media.paused=!0,d.dispatchEvent.call(t,t.media,"pause")}),t.embed.on("timeupdate",function(e){t.media.seeking=!1,r=e.seconds,d.dispatchEvent.call(t,t.media,"timeupdate")}),t.embed.on("progress",function(e){t.media.buffered=e.percent,d.dispatchEvent.call(t,t.media,"progress"),1===parseInt(e.percent,10)&&d.dispatchEvent.call(t,t.media,"canplaythrough")}),t.embed.on("seeked",function(){t.media.seeking=!1,d.dispatchEvent.call(t,t.media,"seeked"),d.dispatchEvent.call(t,t.media,"play")}),t.embed.on("ended",function(){t.media.paused=!0,d.dispatchEvent.call(t,t.media,"ended")}),t.embed.on("error",function(e){t.media.error=e,d.dispatchEvent.call(t,t.media,"error")}),window.setTimeout(function(){return k.build.call(t)},0)}},S=d.getBrowser(),N={setup:function(){if(this.media)if(d.toggleClass(this.elements.container,this.config.classNames.type.replace("{0}",this.type),!0),d.toggleClass(this.elements.container,this.config.classNames.provider.replace("{0}",this.provider),!0),this.isEmbed&&d.toggleClass(this.elements.container,this.config.classNames.type.replace("{0}","video"),!0),this.supported.ui&&(d.toggleClass(this.elements.container,this.config.classNames.pip.supported,p.pip&&this.isHTML5&&this.isVideo),d.toggleClass(this.elements.container,this.config.classNames.airplay.supported,p.airplay&&this.isHTML5),d.toggleClass(this.elements.container,this.config.classNames.stopped,this.config.autoplay),d.toggleClass(this.elements.container,this.config.classNames.isIos,S.isIos),d.toggleClass(this.elements.container,this.config.classNames.isTouch,p.touch)),this.isVideo&&(this.elements.wrapper=d.createElement("div",{class:this.config.classNames.video}),d.wrap(this.media,this.elements.wrapper)),this.isEmbed)switch(this.provider){case"youtube":C.setup.call(this);break;case"vimeo":A.setup.call(this)}else this.isHTML5&&k.setTitle.call(this);else this.debug.warn("No media element found!")},cancelRequests:function(){this.isHTML5&&(Array.from(this.media.querySelectorAll("source")).forEach(d.removeElement),this.media.setAttribute("src",this.config.blankVideo),this.media.load(),this.debug.log("Cancelled network requests"))}},P={insertElements:function(e,t){var i=this;d.is.string(t)?d.insertElement(e,this.media,{src:t}):d.is.array(t)&&t.forEach(function(t){d.insertElement(e,i.media,t)})},change:function(e){var t=this;d.is.object(e)&&"sources"in e&&e.sources.length?(N.cancelRequests.call(this),this.destroy.call(this,function(){switch(d.removeElement(t.media),t.media=null,d.is.element(t.elements.container)&&t.elements.container.removeAttribute("class"),t.type=e.type,t.provider=d.is.empty(e.sources[0].provider)?s.html5:e.sources[0].provider,t.supported=p.check(t.type,t.provider,t.config.inline),t.provider+":"+t.type){case"html5:video":t.media=d.createElement("video");break;case"html5:audio":t.media=d.createElement("audio");break;case"youtube:video":case"vimeo:video":t.media=d.createElement("div",{src:e.sources[0].src})}t.elements.container.appendChild(t.media),d.is.boolean(e.autoplay)&&(t.config.autoplay=e.autoplay),t.isHTML5&&(t.config.crossorigin&&t.media.setAttribute("crossorigin",""),t.config.autoplay&&t.media.setAttribute("autoplay",""),"poster"in e&&t.media.setAttribute("poster",e.poster),t.config.loop.active&&t.media.setAttribute("loop",""),t.config.muted&&t.media.setAttribute("muted",""),t.config.inline&&t.media.setAttribute("playsinline","")),k.addStyleHook.call(t),t.isHTML5&&P.insertElements.call(t,"source",e.sources),t.config.title=e.title,N.setup.call(t),t.isHTML5&&("tracks"in e&&P.insertElements.call(t,"track",e.tracks),t.media.load()),(t.isHTML5||t.isEmbed&&!t.supported.ui)&&k.build.call(t)},!0)):this.debug.warn("Invalid source format")}},x={x:0,y:0};return function(){function e(t,i){var n=this;if(o(this,e),this.timers={},this.ready=!1,this.loading=!1,this.failed=!1,this.media=t,d.is.string(this.media)&&(this.media=document.querySelectorAll(this.media)),(window.jQuery&&this.media instanceof jQuery||d.is.nodeList(this.media)||d.is.array(this.media))&&(this.media=this.media[0]),this.config=d.extend({},l,i,function(){try{return JSON.parse(n.media.getAttribute("data-plyr-config"))}catch(e){return{}}}()),this.elements={container:null,buttons:{},display:{},progress:{},inputs:{},settings:{menu:null,panes:{},tabs:{}},captions:null},this.captions={active:null,currentTrack:null},this.fullscreen={active:!1},this.options={speed:[],quality:[]},this.debug=new m(this.config.debug),this.debug.log("Config",this.config),this.debug.log("Support",p),!d.is.nullOrUndefined(this.media)&&d.is.element(this.media))if(this.media.plyr)this.debug.warn("Target already setup");else if(this.config.enabled)if(p.check().api){this.elements.original=this.media.cloneNode(!0);var r=this.media.tagName.toLowerCase(),c=null,u=null,h=null;switch(r){case"div":if(c=this.media.querySelector("iframe"),!d.is.element(c))return void this.debug.error("Setup failed: <iframe> is missing");if(this.type=a.video,u=c.getAttribute("src"),this.provider=d.getProviderByUrl(u),h=d.getUrlParams(u),!d.is.empty(h)){var f=["1","true"];f.includes(h.autoplay)&&(this.config.autoplay=!0),f.includes(h.playsinline)&&(this.config.inline=!0),f.includes(h.loop)&&(this.config.loop.active=!0)}if(d.is.empty(this.provider)||!Object.keys(s).includes(this.provider))return void this.debug.error("Setup failed: Invalid provider");this.elements.container=this.media,this.media=c,this.elements.container.className="";break;case"video":case"audio":this.type=r,this.provider=s.html5,this.media.hasAttribute("crossorigin")&&(this.config.crossorigin=!0),this.media.hasAttribute("autoplay")&&(this.config.autoplay=!0),this.media.hasAttribute("playsinline")&&(this.config.inline=!0),this.media.hasAttribute("muted")&&(this.config.muted=!0),this.media.hasAttribute("loop")&&(this.config.loop.active=!0);break;default:return void this.debug.error("Setup failed: unsupported type")}this.storage=new g(this),this.supported=p.check(this.type,this.provider,this.config.inline),this.supported.api?(this.media.plyr=this,d.is.element(this.elements.container)||(this.elements.container=d.createElement("div"),d.wrap(this.media,this.elements.container)),this.elements.container.setAttribute("tabindex",0),v.global.call(this),k.addStyleHook.call(this),N.setup.call(this),this.config.debug&&d.on(this.elements.container,this.config.events.join(" "),function(e){n.debug.log("event: "+e.type)}),(this.isHTML5||this.isEmbed&&!this.supported.ui)&&k.build.call(this)):this.debug.error("Setup failed: no support")}else this.debug.error("Setup failed: no support");else this.debug.error("Setup failed: disabled by config");else this.debug.error("Setup failed: no suitable element passed")}return r(e,[{key:"play",value:function(){return this.media.play()}},{key:"pause",value:function(){this.playing&&this.media.pause()}},{key:"togglePlay",value:function(e){(d.is.boolean(e)?e:!this.playing)?this.play():this.pause()}},{key:"stop",value:function(){this.restart(),this.pause()}},{key:"restart",value:function(){this.currentTime=0}},{key:"rewind",value:function(e){this.currentTime=this.currentTime-(d.is.number(e)?e:this.config.seekTime)}},{key:"forward",value:function(e){this.currentTime=this.currentTime+(d.is.number(e)?e:this.config.seekTime)}},{key:"increaseVolume",value:function(e){var t=this.media.muted?0:this.volume;this.volume=t+(d.is.number(e)?e:1)}},{key:"decreaseVolume",value:function(e){var t=this.media.muted?0:this.volume;this.volume=t-(d.is.number(e)?e:1)}},{key:"toggleCaptions",value:function(e){if(this.supported.ui&&d.is.element(this.elements.buttons.captions)){var t=d.is.boolean(e)?e:-1===this.elements.container.className.indexOf(this.config.classNames.captions.active);this.captions.active!==t&&(this.captions.active=t,d.toggleState(this.elements.buttons.captions,this.captions.active),d.toggleClass(this.elements.container,this.config.classNames.captions.active,this.captions.active),d.dispatchEvent.call(this,this.media,this.captions.active?"captionsenabled":"captionsdisabled"))}}},{key:"toggleFullscreen",value:function(e){if(!this.isAudio){if(y.enabled){if(!d.is.event(e)||e.type!==y.eventType)return void(this.fullscreen.active?y.cancelFullScreen():y.requestFullScreen(this.elements.container));this.fullscreen.active=y.isFullScreen(this.elements.container)}else this.fullscreen.active=!this.fullscreen.active,d.toggleClass(this.elements.container,this.config.classNames.fullscreen.fallback,this.fullscreen.active),this.fullscreen.active?x={x:window.pageXOffset||0,y:window.pageYOffset||0}:window.scrollTo(x.x,x.y),document.body.style.overflow=this.fullscreen.active?"hidden":"";d.is.element(this.elements.buttons.fullscreen)&&d.toggleState(this.elements.buttons.fullscreen,this.fullscreen.active),d.dispatchEvent.call(this,this.media,this.fullscreen.active?"enterfullscreen":"exitfullscreen")}}},{key:"airplay",value:function(){p.airplay&&this.media.webkitShowPlaybackTargetPicker()}},{key:"toggleControls",value:function(e){var t=this;if(d.is.element(this.elements.controls)&&this.supported.ui&&!this.isAudio){var i=0,n=e,s=!1;if(d.is.boolean(e)||(d.is.event(e)?(s="enterfullscreen"===e.type,n=["mouseenter","mousemove","touchstart","touchmove","focusin"].includes(e.type),["mousemove","touchmove","touchend"].includes(e.type)&&(i=2e3),"focusin"===e.type&&(i=3e3,d.toggleClass(this.elements.controls,this.config.classNames.noTransition,!0))):n=d.hasClass(this.elements.container,this.config.classNames.hideControls)),window.clearTimeout(this.timers.controls),n||this.paused||this.loading){if(d.toggleClass(this.elements.container,this.config.classNames.hideControls,!1)&&d.dispatchEvent.call(this,this.media,"controlsshown"),this.paused||this.loading)return;p.touch&&(i=3e3)}n&&!this.playing||(this.timers.controls=window.setTimeout(function(){(!t.elements.controls.pressed&&!t.elements.controls.hover||s)&&(d.hasClass(t.elements.container,t.config.classNames.hideControls)||d.toggleClass(t.elements.controls,t.config.classNames.noTransition,!1),d.toggleClass(t.elements.container,t.config.classNames.hideControls,!0)&&(d.dispatchEvent.call(t,t.media,"controlshidden"),t.config.controls.includes("settings")&&!d.is.empty(t.config.settings)&&E.toggleMenu.call(t,!1)))},i))}}},{key:"on",value:function(e,t){d.on(this.elements.container,e,t)}},{key:"off",value:function(e,t){d.off(this.elements.container,e,t)}},{key:"destroy",value:function(e){var t=this,i=arguments.length>1&&void 0!==arguments[1]&&arguments[1],n=function(){document.body.style.overflow="",t.embed=null,i?(Object.keys(t.elements).length&&(t.elements.buttons&&t.elements.buttons.play&&Array.from(t.elements.buttons.play).forEach(function(e){return d.removeElement(e)}),d.removeElement(t.elements.captions),d.removeElement(t.elements.controls),d.removeElement(t.elements.wrapper),t.elements.buttons.play=null,t.elements.captions=null,t.elements.controls=null,t.elements.wrapper=null),d.is.function(e)&&e()):(d.replaceElement(t.elements.original,t.elements.container),d.dispatchEvent.call(t,t.elements.original,"destroyed",!0),d.is.function(e)&&e.call(t.elements.original),t.elements=null)};switch(this.provider+":"+this.type){case"html5:video":case"html5:audio":k.toggleNativeControls.call(this,!0),n();break;case"youtube:video":window.clearInterval(this.timers.buffering),window.clearInterval(this.timers.playing),null!==this.embed&&this.embed.destroy(),n();break;case"vimeo:video":null!==this.embed&&this.embed.unload().then(n),window.setTimeout(n,200)}}},{key:"supports",value:function(e){return p.mime.call(this,e)}},{key:"isHTML5",get:function(){return this.provider===s.html5}},{key:"isEmbed",get:function(){return this.isYouTube||this.isVimeo}},{key:"isYouTube",get:function(){return this.provider===s.youtube}},{key:"isVimeo",get:function(){return this.provider===s.vimeo}},{key:"isVideo",get:function(){return this.type===a.video}},{key:"isAudio",get:function(){return this.type===a.audio}},{key:"paused",get:function(){return this.media.paused}},{key:"playing",get:function(){return!this.paused&&!this.ended&&(!this.isHTML5||this.media.readyState>2)}},{key:"ended",get:function(){return this.media.ended}},{key:"currentTime",set:function(e){var t=0;d.is.number(e)&&(t=e),t<0?t=0:t>this.duration&&(t=this.duration),this.media.currentTime=t.toFixed(4),this.debug.log("Seeking to "+this.currentTime+" seconds")},get:function(){return Number(this.media.currentTime)}},{key:"seeking",get:function(){return this.media.seeking}},{key:"duration",get:function(){var e=parseInt(this.config.duration,10),t=Number(this.media.duration);return Number.isNaN(e)?t:e}},{key:"volume",set:function(e){var t=e;d.is.string(t)&&(t=Number(t)),d.is.number(t)||(t=this.storage.get("volume")),d.is.number(t)||(t=this.config.volume),t>1&&(t=1),t<0&&(t=0),this.config.volume=t,this.media.volume=t,this.muted&&t>0&&(this.muted=!1)},get:function(){return this.media.volume}},{key:"muted",set:function(e){var t=e;d.is.boolean(t)||(t=this.storage.get("muted")),d.is.boolean(t)||(t=this.config.muted),this.config.muted=t,this.media.muted=t},get:function(){return this.media.muted}},{key:"hasAudio",get:function(){return!this.isHTML5||(this.media.mozHasAudio||Boolean(this.media.webkitAudioDecodedByteCount)||Boolean(this.media.audioTracks&&this.media.audioTracks.length))}},{key:"speed",set:function(e){var t=null;d.is.number(e)&&(t=e),d.is.number(t)||(t=this.storage.get("speed")),d.is.number(t)||(t=this.config.speed.selected),t<.1&&(t=.1),t>2&&(t=2),this.config.speed.options.includes(t)?(this.config.speed.selected=t,this.media.playbackRate=t):this.debug.warn("Unsupported speed ("+t+")")},get:function(){return this.media.playbackRate}},{key:"quality",set:function(e){var t=null;d.is.string(e)&&(t=e),d.is.string(t)||(t=this.storage.get("quality")),d.is.string(t)||(t=this.config.quality.selected),this.options.quality.includes(t)?(this.config.quality.selected=t,this.media.quality=t):this.debug.warn("Unsupported quality option ("+t+")")},get:function(){return this.media.quality}},{key:"loop",set:function(e){var t=d.is.boolean(e)?e:this.config.loop.active;this.config.loop.active=t,this.media.loop=t},get:function(){return this.media.loop}},{key:"source",set:function(e){P.change.call(this,e)},get:function(){return this.media.currentSrc}},{key:"poster",set:function(e){this.isHTML5&&this.isVideo?d.is.string(e)&&this.media.setAttribute("poster",e):this.debug.warn("Poster can only be set on HTML5 video")},get:function(){return this.isHTML5&&this.isVideo?this.media.getAttribute("poster"):null}},{key:"autoplay",set:function(e){var t=d.is.boolean(e)?e:this.config.autoplay;this.config.autoplay=t},get:function(){return this.config.autoplay}},{key:"language",set:function(e){if(d.is.string(e)&&(this.toggleCaptions(!d.is.empty(e)),!d.is.empty(e))){var t=e.toLowerCase();this.language!==t&&(this.captions.language=t,T.setText.call(this,null),T.setLanguage.call(this),d.dispatchEvent.call(this,this.media,"languagechange"))}},get:function(){return this.captions.language}},{key:"pip",set:function(e){var t="picture-in-picture",i="inline";if(p.pip){var n=d.is.boolean(e)?e:this.pip===i;this.media.webkitSetPresentationMode(n?t:i)}},get:function(){return p.pip?this.media.webkitPresentationMode:null}}],[{key:"supported",value:function(e,t,i){return p.check(e,t,i)}},{key:"loadSprite",value:function(e,t){return d.loadSprite(e,t)}}]),e}()}); //# sourceMappingURL=plyr.js.map diff --git a/dist/plyr.js.map b/dist/plyr.js.map index bac0c648..567ccabf 100644 --- a/dist/plyr.js.map +++ b/dist/plyr.js.map @@ -1 +1 @@ -{"version":3,"file":"plyr.js","sources":["src/js/types.js","src/js/utils.js","src/js/support.js","src/js/fullscreen.js","src/js/defaults.js","src/js/console.js","src/js/storage.js","src/js/listeners.js","src/js/ui.js","src/js/controls.js","src/js/captions.js","src/js/plugins/youtube.js","src/js/plugins/vimeo.js","src/js/media.js","src/js/source.js","src/js/plyr.js"],"sourcesContent":["// ==========================================================================\n// Plyr supported types and providers\n// ==========================================================================\n\nexport const providers = {\n html5: 'html5',\n youtube: 'youtube',\n vimeo: 'vimeo',\n};\n\nexport const types = {\n audio: 'audio',\n video: 'video',\n};\n\nexport default { providers, types };\n","// ==========================================================================\n// Plyr utils\n// ==========================================================================\n\nimport support from './support';\n\nconst utils = {\n // Check variable types\n is: {\n plyr(input) {\n return this.instanceof(input, Plyr);\n },\n object(input) {\n return this.getConstructor(input) === Object;\n },\n number(input) {\n return this.getConstructor(input) === Number && !Number.isNaN(input);\n },\n string(input) {\n return this.getConstructor(input) === String;\n },\n boolean(input) {\n return this.getConstructor(input) === Boolean;\n },\n function(input) {\n return this.getConstructor(input) === Function;\n },\n array(input) {\n return !this.nullOrUndefined(input) && Array.isArray(input);\n },\n weakMap(input) {\n return this.instanceof(input, window.WeakMap);\n },\n nodeList(input) {\n return this.instanceof(input, window.NodeList);\n },\n element(input) {\n return this.instanceof(input, window.Element);\n },\n textNode(input) {\n return this.getConstructor(input) === Text;\n },\n event(input) {\n return this.instanceof(input, window.Event);\n },\n cue(input) {\n return this.instanceof(input, window.TextTrackCue) || this.instanceof(input, window.VTTCue);\n },\n track(input) {\n return this.instanceof(input, TextTrack) || (!this.nullOrUndefined(input) && this.string(input.kind));\n },\n nullOrUndefined(input) {\n return input === null || typeof input === 'undefined';\n },\n empty(input) {\n return (\n this.nullOrUndefined(input) ||\n ((this.string(input) || this.array(input) || this.nodeList(input)) && !input.length) ||\n (this.object(input) && !Object.keys(input).length)\n );\n },\n instanceof(input, constructor) {\n return Boolean(input && constructor && input instanceof constructor);\n },\n getConstructor(input) {\n return !this.nullOrUndefined(input) ? input.constructor : null;\n },\n },\n\n // Unfortunately, due to mixed support, UA sniffing is required\n getBrowser() {\n return {\n isIE: /* @cc_on!@ */ false || !!document.documentMode,\n isWebkit: 'WebkitAppearance' in document.documentElement.style && !/Edge/.test(navigator.userAgent),\n isIPhone: /(iPhone|iPod)/gi.test(navigator.platform),\n isIos: /(iPad|iPhone|iPod)/gi.test(navigator.platform),\n };\n },\n\n // Load an external script\n loadScript(url, callback) {\n const current = document.querySelector(`script[src=\"${url}\"]`);\n\n // Check script is not already referenced, if so wait for load\n if (current !== null) {\n current.callbacks = current.callbacks || [];\n current.callbacks.push(callback);\n return;\n }\n\n // Build the element\n const element = document.createElement('script');\n\n // Callback queue\n element.callbacks = element.callbacks || [];\n element.callbacks.push(callback);\n\n // Bind callback\n if (utils.is.function(callback)) {\n element.addEventListener(\n 'load',\n event => {\n element.callbacks.forEach(cb => cb.call(null, event));\n element.callbacks = null;\n },\n false\n );\n }\n\n // Set the URL after binding callback\n element.src = url;\n\n // Inject\n const first = document.getElementsByTagName('script')[0];\n first.parentNode.insertBefore(element, first);\n },\n\n // Load an external SVG sprite\n loadSprite(url, id) {\n if (!utils.is.string(url)) {\n return;\n }\n\n const prefix = 'cache-';\n const hasId = utils.is.string(id);\n let isCached = false;\n\n function updateSprite(data) {\n // Inject content\n this.innerHTML = data;\n\n // Inject the SVG to the body\n document.body.insertBefore(this, document.body.childNodes[0]);\n }\n\n // Only load once\n if (!hasId || !document.querySelectorAll(`#${id}`).length) {\n // Create container\n const container = document.createElement('div');\n utils.toggleHidden(container, true);\n\n if (hasId) {\n container.setAttribute('id', id);\n }\n\n // Check in cache\n if (support.storage) {\n const cached = window.localStorage.getItem(prefix + id);\n isCached = cached !== null;\n\n if (isCached) {\n const data = JSON.parse(cached);\n updateSprite.call(container, data.content);\n return;\n }\n }\n\n // Get the sprite\n fetch(url)\n .then(response => (response.ok ? response.text() : null))\n .then(text => {\n if (text === null) {\n return;\n }\n\n if (support.storage) {\n window.localStorage.setItem(\n prefix + id,\n JSON.stringify({\n content: text,\n })\n );\n }\n\n updateSprite.call(container, text);\n })\n .catch(() => {});\n }\n },\n\n // Generate a random ID\n generateId(prefix) {\n return `${prefix}-${Math.floor(Math.random() * 10000)}`;\n },\n\n // Determine if we're in an iframe\n inFrame() {\n try {\n return window.self !== window.top;\n } catch (e) {\n return true;\n }\n },\n\n // Wrap an element\n wrap(elements, wrapper) {\n // Convert `elements` to an array, if necessary.\n const targets = elements.length ? elements : [elements];\n\n // Loops backwards to prevent having to clone the wrapper on the\n // first element (see `child` below).\n Array.from(targets)\n .reverse()\n .forEach((element, index) => {\n const child = index > 0 ? wrapper.cloneNode(true) : wrapper;\n\n // Cache the current parent and sibling.\n const parent = element.parentNode;\n const sibling = element.nextSibling;\n\n // Wrap the element (is automatically removed from its current\n // parent).\n child.appendChild(element);\n\n // If the element had a sibling, insert the wrapper before\n // the sibling to maintain the HTML structure; otherwise, just\n // append it to the parent.\n if (sibling) {\n parent.insertBefore(child, sibling);\n } else {\n parent.appendChild(child);\n }\n });\n },\n\n // Create a DocumentFragment\n createElement(type, attributes, text) {\n // Create a new <element>\n const element = document.createElement(type);\n\n // Set all passed attributes\n if (utils.is.object(attributes)) {\n utils.setAttributes(element, attributes);\n }\n\n // Add text node\n if (utils.is.string(text)) {\n element.textContent = text;\n }\n\n // Return built element\n return element;\n },\n\n // Inaert an element after another\n insertAfter(element, target) {\n target.parentNode.insertBefore(element, target.nextSibling);\n },\n\n // Insert a DocumentFragment\n insertElement(type, parent, attributes, text) {\n // Inject the new <element>\n parent.appendChild(utils.createElement(type, attributes, text));\n },\n\n // Remove an element\n removeElement(element) {\n if (!utils.is.element(element) || !utils.is.element(element.parentNode)) {\n return null;\n }\n\n element.parentNode.removeChild(element);\n\n return element;\n },\n\n // Remove all child elements\n emptyElement(element) {\n let { length } = element.childNodes;\n\n while (length > 0) {\n element.removeChild(element.lastChild);\n length -= 1;\n }\n },\n\n // Set attributes\n setAttributes(element, attributes) {\n if (!utils.is.element(element) || utils.is.empty(attributes)) {\n return;\n }\n\n Object.keys(attributes).forEach(key => {\n element.setAttribute(key, attributes[key]);\n });\n },\n\n // Get an attribute object from a string selector\n getAttributesFromSelector(sel, existingAttributes) {\n // For example:\n // '.test' to { class: 'test' }\n // '#test' to { id: 'test' }\n // '[data-test=\"test\"]' to { 'data-test': 'test' }\n\n if (!utils.is.string(sel) || utils.is.empty(sel)) {\n return {};\n }\n\n const attributes = {};\n const existing = existingAttributes;\n\n sel.split(',').forEach(s => {\n // Remove whitespace\n const selector = s.trim();\n const className = selector.replace('.', '');\n const stripped = selector.replace(/[[\\]]/g, '');\n\n // Get the parts and value\n const parts = stripped.split('=');\n const key = parts[0];\n const value = parts.length > 1 ? parts[1].replace(/[\"']/g, '') : '';\n\n // Get the first character\n const start = selector.charAt(0);\n\n switch (start) {\n case '.':\n // Add to existing classname\n if (utils.is.object(existing) && utils.is.string(existing.class)) {\n existing.class += ` ${className}`;\n }\n\n attributes.class = className;\n break;\n\n case '#':\n // ID selector\n attributes.id = selector.replace('#', '');\n break;\n\n case '[':\n // Attribute selector\n attributes[key] = value;\n\n break;\n\n default:\n break;\n }\n });\n\n return attributes;\n },\n\n // Toggle class on an element\n toggleClass(element, className, toggle) {\n if (utils.is.element(element)) {\n const contains = element.classList.contains(className);\n\n element.classList[toggle ? 'add' : 'remove'](className);\n\n return (toggle && !contains) || (!toggle && contains);\n }\n\n return null;\n },\n\n // Has class name\n hasClass(element, className) {\n return utils.is.element(element) && element.classList.contains(className);\n },\n\n // Toggle hidden attribute on an element\n toggleHidden(element, toggle) {\n if (!utils.is.element(element)) {\n return;\n }\n\n if (toggle) {\n element.setAttribute('hidden', '');\n } else {\n element.removeAttribute('hidden');\n }\n },\n\n // Element matches selector\n matches(element, selector) {\n const prototype = { Element };\n\n function match() {\n return Array.from(document.querySelectorAll(selector)).includes(this);\n }\n\n const matches = prototype.matches || prototype.webkitMatchesSelector || prototype.mozMatchesSelector || prototype.msMatchesSelector || match;\n\n return matches.call(element, selector);\n },\n\n // Find all elements\n getElements(selector) {\n return this.elements.container.querySelectorAll(selector);\n },\n\n // Find a single element\n getElement(selector) {\n return this.elements.container.querySelector(selector);\n },\n\n // Find the UI controls and store references in custom controls\n // TODO: Allow settings menus with custom controls\n findElements() {\n try {\n this.elements.controls = utils.getElement.call(this, this.config.selectors.controls.wrapper);\n\n // Buttons\n this.elements.buttons = {\n play: utils.getElements.call(this, this.config.selectors.buttons.play),\n pause: utils.getElement.call(this, this.config.selectors.buttons.pause),\n restart: utils.getElement.call(this, this.config.selectors.buttons.restart),\n rewind: utils.getElement.call(this, this.config.selectors.buttons.rewind),\n forward: utils.getElement.call(this, this.config.selectors.buttons.forward),\n mute: utils.getElement.call(this, this.config.selectors.buttons.mute),\n pip: utils.getElement.call(this, this.config.selectors.buttons.pip),\n airplay: utils.getElement.call(this, this.config.selectors.buttons.airplay),\n settings: utils.getElement.call(this, this.config.selectors.buttons.settings),\n captions: utils.getElement.call(this, this.config.selectors.buttons.captions),\n fullscreen: utils.getElement.call(this, this.config.selectors.buttons.fullscreen),\n };\n\n // Progress\n this.elements.progress = utils.getElement.call(this, this.config.selectors.progress);\n\n // Inputs\n this.elements.inputs = {\n seek: utils.getElement.call(this, this.config.selectors.inputs.seek),\n volume: utils.getElement.call(this, this.config.selectors.inputs.volume),\n };\n\n // Display\n this.elements.display = {\n buffer: utils.getElement.call(this, this.config.selectors.display.buffer),\n duration: utils.getElement.call(this, this.config.selectors.display.duration),\n currentTime: utils.getElement.call(this, this.config.selectors.display.currentTime),\n };\n\n // Seek tooltip\n if (utils.is.element(this.elements.progress)) {\n this.elements.display.seekTooltip = this.elements.progress.querySelector(`.${this.config.classNames.tooltip}`);\n }\n\n return true;\n } catch (error) {\n // Log it\n this.debug.warn('It looks like there is a problem with your custom controls HTML', error);\n\n // Restore native video controls\n this.toggleNativeControls(true);\n\n return false;\n }\n },\n\n // Get the focused element\n getFocusElement() {\n let focused = document.activeElement;\n\n if (!focused || focused === document.body) {\n focused = null;\n } else {\n focused = document.querySelector(':focus');\n }\n\n return focused;\n },\n\n // Trap focus inside container\n trapFocus() {\n const focusable = utils.getElements.call(this, 'button:not(:disabled), input:not(:disabled), [tabindex]');\n const first = focusable[0];\n const last = focusable[focusable.length - 1];\n\n utils.on(\n this.elements.container,\n 'keydown',\n event => {\n // Bail if not tab key or not fullscreen\n if (event.key !== 'Tab' || event.keyCode !== 9 || !this.fullscreen.active) {\n return;\n }\n\n // Get the current focused element\n const focused = utils.getFocusElement();\n\n if (focused === last && !event.shiftKey) {\n // Move focus to first element that can be tabbed if Shift isn't used\n first.focus();\n event.preventDefault();\n } else if (focused === first && event.shiftKey) {\n // Move focus to last element that can be tabbed if Shift is used\n last.focus();\n event.preventDefault();\n }\n },\n false\n );\n },\n\n // Toggle event listener\n toggleListener(elements, event, callback, toggle, passive, capture) {\n // Bail if no elements\n if (utils.is.nullOrUndefined(elements)) {\n return;\n }\n\n // If a nodelist is passed, call itself on each node\n if (utils.is.nodeList(elements)) {\n // Create listener for each node\n Array.from(elements).forEach(element => {\n if (element instanceof Node) {\n utils.toggleListener.call(null, element, event, callback, toggle, passive, capture);\n }\n });\n\n return;\n }\n\n // Allow multiple events\n const events = event.split(' ');\n\n // Build options\n // Default to just capture boolean\n let options = utils.is.boolean(capture) ? capture : false;\n\n // If passive events listeners are supported\n if (support.passiveListeners) {\n options = {\n // Whether the listener can be passive (i.e. default never prevented)\n passive: utils.is.boolean(passive) ? passive : true,\n // Whether the listener is a capturing listener or not\n capture: utils.is.boolean(capture) ? capture : false,\n };\n }\n\n // If a single node is passed, bind the event listener\n events.forEach(type => {\n elements[toggle ? 'addEventListener' : 'removeEventListener'](type, callback, options);\n });\n },\n\n // Bind event handler\n on(element, events, callback, passive, capture) {\n utils.toggleListener(element, events, callback, true, passive, capture);\n },\n\n // Unbind event handler\n off(element, events, callback, passive, capture) {\n utils.toggleListener(element, events, callback, false, passive, capture);\n },\n\n // Trigger event\n dispatchEvent(element, type, bubbles, detail) {\n // Bail if no element\n if (!element || !type) {\n return;\n }\n\n // Create and dispatch the event\n const event = new CustomEvent(type, {\n bubbles: utils.is.boolean(bubbles) ? bubbles : false,\n detail: Object.assign({}, detail, {\n plyr: this instanceof Plyr ? this : null,\n }),\n });\n\n // Dispatch the event\n element.dispatchEvent(event);\n },\n\n // Toggle aria-pressed state on a toggle button\n // http://www.ssbbartgroup.com/blog/how-not-to-misuse-aria-states-properties-and-roles\n toggleState(element, input) {\n // Bail if no target\n if (!utils.is.element(element)) {\n return;\n }\n\n // Get state\n const pressed = element.getAttribute('aria-pressed') === 'true';\n const state = utils.is.boolean(input) ? input : !pressed;\n\n // Set the attribute on target\n element.setAttribute('aria-pressed', state);\n },\n\n // Get percentage\n getPercentage(current, max) {\n if (current === 0 || max === 0 || Number.isNaN(current) || Number.isNaN(max)) {\n return 0;\n }\n return (current / max * 100).toFixed(2);\n },\n\n // Deep extend destination object with N more objects\n extend(target = {}, ...sources) {\n if (!sources.length) {\n return target;\n }\n\n const source = sources.shift();\n\n if (!utils.is.object(source)) {\n return target;\n }\n\n Object.keys(source).forEach(key => {\n if (utils.is.object(source[key])) {\n if (!Object.keys(target).includes(key)) {\n Object.assign(target, { [key]: {} });\n }\n\n utils.extend(target[key], source[key]);\n } else {\n Object.assign(target, { [key]: source[key] });\n }\n });\n\n return utils.extend(target, ...sources);\n },\n\n // Parse YouTube ID from URL\n parseYouTubeId(url) {\n const regex = /^.*(youtu.be\\/|v\\/|u\\/\\w\\/|embed\\/|watch\\?v=|&v=)([^#&?]*).*/;\n return url.match(regex) ? RegExp.$2 : url;\n },\n\n // Parse Vimeo ID from URL\n parseVimeoId(url) {\n if (utils.is.number(Number(url))) {\n return url;\n }\n\n const regex = /^.*(vimeo.com\\/|video\\/)(\\d+).*/;\n return url.match(regex) ? RegExp.$2 : url;\n },\n\n // Convert object to URL parameters\n buildUrlParameters(input) {\n if (!utils.is.object(input)) {\n return '';\n }\n\n return Object.keys(input)\n .map(key => `${encodeURIComponent(key)}=${encodeURIComponent(input[key])}`)\n .join('&');\n },\n\n // Remove HTML from a string\n stripHTML(source) {\n const fragment = document.createDocumentFragment();\n const element = document.createElement('div');\n fragment.appendChild(element);\n element.innerHTML = source;\n return fragment.firstChild.innerText;\n },\n\n // Get aspect ratio for dimensions\n getAspectRatio(width, height) {\n const getRatio = (w, h) => (h === 0 ? w : getRatio(h, w % h));\n const ratio = getRatio(width, height);\n return `${width / ratio}:${height / ratio}`;\n },\n\n // Get the transition end event\n transitionEnd: (() => {\n const element = document.createElement('span');\n\n const events = {\n WebkitTransition: 'webkitTransitionEnd',\n MozTransition: 'transitionend',\n OTransition: 'oTransitionEnd otransitionend',\n transition: 'transitionend',\n };\n\n const type = Object.keys(events).find(event => element.style[event] !== undefined);\n\n return typeof type === 'string' ? type : false;\n })(),\n\n // Force repaint of element\n repaint(element) {\n window.setTimeout(() => {\n element.setAttribute('hidden', '');\n element.offsetHeight; // eslint-disable-line\n element.removeAttribute('hidden');\n }, 0);\n },\n};\n\nexport default utils;\n","// ==========================================================================\n// Plyr support checks\n// ==========================================================================\n\nimport utils from './utils';\n\n// Check for feature support\nconst support = {\n // Basic support\n audio: 'canPlayType' in document.createElement('audio'),\n video: 'canPlayType' in document.createElement('video'),\n\n // Check for support\n // Basic functionality vs full UI\n check(type, provider, inline) {\n let api = false;\n let ui = false;\n const browser = utils.getBrowser();\n const playsInline = browser.isIPhone && inline && support.inline;\n\n switch (`${provider}:${type}`) {\n case 'html5:video':\n api = support.video;\n ui = api && support.rangeInput && (!browser.isIPhone || playsInline);\n break;\n\n case 'html5:audio':\n api = support.audio;\n ui = api && support.rangeInput;\n break;\n\n case 'youtube:video':\n api = true;\n ui = support.rangeInput && (!browser.isIPhone || playsInline);\n break;\n\n case 'vimeo:video':\n api = true;\n ui = support.rangeInput && !browser.isIPhone;\n break;\n\n default:\n api = support.audio && support.video;\n ui = api && support.rangeInput;\n }\n\n return {\n api,\n ui,\n };\n },\n\n // Picture-in-picture support\n // Safari only currently\n pip: (() => {\n const browser = utils.getBrowser();\n return !browser.isIPhone && utils.is.function(utils.createElement('video').webkitSetPresentationMode);\n })(),\n\n // Airplay support\n // Safari only currently\n airplay: utils.is.function(window.WebKitPlaybackTargetAvailabilityEvent),\n\n // Inline playback support\n // https://webkit.org/blog/6784/new-video-policies-for-ios/\n inline: 'playsInline' in document.createElement('video'),\n\n // Check for mime type support against a player instance\n // Credits: http://diveintohtml5.info/everything.html\n // Related: http://www.leanbackplayer.com/test/h5mt.html\n mime(type) {\n const { media } = this;\n\n try {\n // Bail if no checking function\n if (!this.isHTML5 || !utils.is.function(media.canPlayType)) {\n return false;\n }\n\n // Type specific checks\n if (this.isVideo) {\n switch (type) {\n case 'video/webm':\n return media.canPlayType('video/webm; codecs=\"vp8, vorbis\"').replace(/no/, '');\n\n case 'video/mp4':\n return media.canPlayType('video/mp4; codecs=\"avc1.42E01E, mp4a.40.2\"').replace(/no/, '');\n\n case 'video/ogg':\n return media.canPlayType('video/ogg; codecs=\"theora\"').replace(/no/, '');\n\n default:\n return false;\n }\n } else if (this.isAudio) {\n switch (type) {\n case 'audio/mpeg':\n return media.canPlayType('audio/mpeg;').replace(/no/, '');\n\n case 'audio/ogg':\n return media.canPlayType('audio/ogg; codecs=\"vorbis\"').replace(/no/, '');\n\n case 'audio/wav':\n return media.canPlayType('audio/wav; codecs=\"1\"').replace(/no/, '');\n\n default:\n return false;\n }\n }\n } catch (e) {\n return false;\n }\n\n // If we got this far, we're stuffed\n return false;\n },\n\n // Check for textTracks support\n textTracks: 'textTracks' in document.createElement('video'),\n\n // Check for passive event listener support\n // https://github.com/WICG/EventListenerOptions/blob/gh-pages/explainer.md\n // https://www.youtube.com/watch?v=NPM6172J22g\n passiveListeners: (() => {\n // Test via a getter in the options object to see if the passive property is accessed\n let supported = false;\n try {\n const options = Object.defineProperty({}, 'passive', {\n get() {\n supported = true;\n return null;\n },\n });\n window.addEventListener('test', null, options);\n } catch (e) {\n // Do nothing\n }\n\n return supported;\n })(),\n\n // <input type=\"range\"> Sliders\n rangeInput: (() => {\n const range = document.createElement('input');\n range.type = 'range';\n return range.type === 'range';\n })(),\n\n // Touch\n // Remember a device can be moust + touch enabled\n touch: 'ontouchstart' in document.documentElement,\n\n // Detect transitions support\n transitions: utils.transitionEnd !== false,\n\n // Reduced motion iOS & MacOS setting\n // https://webkit.org/blog/7551/responsive-design-for-motion/\n reducedMotion: 'matchMedia' in window && window.matchMedia('(prefers-reduced-motion)').matches,\n};\n\nexport default support;\n","// ==========================================================================\n// Plyr fullscreen API\n// ==========================================================================\n\nimport utils from './utils';\n\n// Determine the prefix\nconst prefix = (() => {\n let value = false;\n\n if (utils.is.function(document.cancelFullScreen)) {\n value = '';\n } else {\n // Check for fullscreen support by vendor prefix\n [\n 'webkit',\n 'o',\n 'moz',\n 'ms',\n 'khtml',\n ].some(pre => {\n if (utils.is.function(document[`${pre}CancelFullScreen`])) {\n value = pre;\n return true;\n } else if (utils.is.function(document.msExitFullscreen) && document.msFullscreenEnabled) {\n // Special case for MS (when isn't it?)\n value = 'ms';\n return true;\n }\n\n return false;\n });\n }\n\n return value;\n})();\n\n// Fullscreen API\nconst fullscreen = {\n // Get the prefix\n prefix,\n\n // Check if we can use it\n enabled: document.fullscreenEnabled || document.webkitFullscreenEnabled || document.mozFullScreenEnabled || document.msFullscreenEnabled,\n\n // Yet again Microsoft awesomeness,\n // Sometimes the prefix is 'ms', sometimes 'MS' to keep you on your toes\n eventType: prefix === 'ms' ? 'MSFullscreenChange' : `${prefix}fullscreenchange`,\n\n // Is an element fullscreen\n isFullScreen(element) {\n if (!fullscreen.enabled) {\n return false;\n }\n\n const target = utils.is.nullOrUndefined(element) ? document.body : element;\n\n switch (prefix) {\n case '':\n return document.fullscreenElement === target;\n\n case 'moz':\n return document.mozFullScreenElement === target;\n\n default:\n return document[`${prefix}FullscreenElement`] === target;\n }\n },\n\n // Make an element fullscreen\n requestFullScreen(element) {\n if (!fullscreen.enabled) {\n return false;\n }\n\n const target = utils.is.nullOrUndefined(element) ? document.body : element;\n\n return !prefix.length ? target.requestFullScreen() : target[prefix + (prefix === 'ms' ? 'RequestFullscreen' : 'RequestFullScreen')]();\n },\n\n // Bail from fullscreen\n cancelFullScreen() {\n if (!fullscreen.enabled) {\n return false;\n }\n\n return !prefix.length ? document.cancelFullScreen() : document[prefix + (prefix === 'ms' ? 'ExitFullscreen' : 'CancelFullScreen')]();\n },\n\n // Get the current element\n element() {\n if (!fullscreen.enabled) {\n return null;\n }\n\n return !prefix.length ? document.fullscreenElement : document[`${prefix}FullscreenElement`];\n },\n\n // Setup fullscreen\n setup() {\n if (!this.supported.ui || this.isAudio || !this.config.fullscreen.enabled) {\n return;\n }\n\n // Check for native support\n const nativeSupport = fullscreen.enabled;\n\n if (nativeSupport || (this.config.fullscreen.fallback && !utils.inFrame())) {\n this.debug.log(`${nativeSupport ? 'Native' : 'Fallback'} fullscreen enabled`);\n\n // Add styling hook to show button\n utils.toggleClass(this.elements.container, this.config.classNames.fullscreen.enabled, true);\n } else {\n this.debug.log('Fullscreen not supported and fallback disabled');\n }\n\n // Toggle state\n if (this.elements.buttons && this.elements.buttons.fullscreen) {\n utils.toggleState(this.elements.buttons.fullscreen, false);\n }\n\n // Trap focus in container\n utils.trapFocus.call(this);\n },\n};\n\nexport default fullscreen;\n","// Default config\nconst defaults = {\n // Disable\n enabled: true,\n\n // Custom media title\n title: '',\n\n // Logging to console\n debug: false,\n\n // Auto play (if supported)\n autoplay: false,\n\n // Only allow one media playing at once (vimeo only)\n autopause: true,\n\n // Default time to skip when rewind/fast forward\n seekTime: 10,\n\n // Default volume\n volume: 1,\n muted: false,\n\n // Pass a custom duration\n duration: null,\n\n // Display the media duration on load in the current time position\n // If you have opted to display both duration and currentTime, this is ignored\n displayDuration: true,\n\n // Invert the current time to be a countdown\n invertTime: true,\n\n // Clicking the currentTime inverts it's value to show time left rather than elapsed\n toggleInvert: true,\n\n // Aspect ratio (for embeds)\n ratio: '16:9',\n\n // Click video container to play/pause\n clickToPlay: true,\n\n // Auto hide the controls\n hideControls: true,\n\n // Revert to poster on finish (HTML5 - will cause reload)\n showPosterOnEnd: false,\n\n // Disable the standard context menu\n disableContextMenu: true,\n\n // Sprite (for icons)\n loadSprite: true,\n iconPrefix: 'plyr',\n iconUrl: 'https://cdn.plyr.io/2.0.10/plyr.svg',\n\n // Blank video (used to prevent errors on source change)\n blankVideo: 'https://cdn.plyr.io/static/blank.mp4',\n\n // Quality default\n quality: {\n default: 'default',\n options: [\n 'hd2160',\n 'hd1440',\n 'hd1080',\n 'hd720',\n 'large',\n 'medium',\n 'small',\n 'tiny',\n 'default',\n ],\n },\n\n // Set loops\n loop: {\n active: false,\n // start: null,\n // end: null,\n },\n\n // Speed default and options to display\n speed: {\n selected: 1,\n options: [\n 0.5,\n 0.75,\n 1,\n 1.25,\n 1.5,\n 1.75,\n 2,\n ],\n },\n\n // Keyboard shortcut settings\n keyboard: {\n focused: true,\n global: false,\n },\n\n // Display tooltips\n tooltips: {\n controls: false,\n seek: true,\n },\n\n // Captions settings\n captions: {\n active: false,\n language: window.navigator.language.split('-')[0],\n },\n\n // Fullscreen settings\n fullscreen: {\n enabled: true, // Allow fullscreen?\n fallback: true, // Fallback for vintage browsers\n },\n\n // Local storage\n storage: {\n enabled: true,\n key: 'plyr',\n },\n\n // Default controls\n controls: [\n 'play-large',\n 'play',\n 'progress',\n 'current-time',\n 'mute',\n 'volume',\n 'captions',\n 'settings',\n 'pip',\n 'airplay',\n 'fullscreen',\n ],\n settings: [\n 'captions',\n 'quality',\n 'speed',\n ],\n\n // Localisation\n i18n: {\n restart: 'Restart',\n rewind: 'Rewind {seektime} secs',\n play: 'Play',\n pause: 'Pause',\n forward: 'Forward {seektime} secs',\n seek: 'Seek',\n played: 'Played',\n buffered: 'Buffered',\n currentTime: 'Current time',\n duration: 'Duration',\n volume: 'Volume',\n mute: 'Mute',\n unmute: 'Unmute',\n enableCaptions: 'Enable captions',\n disableCaptions: 'Disable captions',\n enterFullscreen: 'Enter fullscreen',\n exitFullscreen: 'Exit fullscreen',\n frameTitle: 'Player for {title}',\n captions: 'Captions',\n settings: 'Settings',\n speed: 'Speed',\n quality: 'Quality',\n loop: 'Loop',\n start: 'Start',\n end: 'End',\n all: 'All',\n reset: 'Reset',\n none: 'None',\n disabled: 'Disabled',\n },\n\n // URLs\n urls: {\n vimeo: {\n api: 'https://player.vimeo.com/api/player.js',\n },\n youtube: {\n api: 'https://www.youtube.com/iframe_api',\n },\n },\n\n // Custom control listeners\n listeners: {\n seek: null,\n play: null,\n pause: null,\n restart: null,\n rewind: null,\n forward: null,\n mute: null,\n volume: null,\n captions: null,\n fullscreen: null,\n pip: null,\n airplay: null,\n speed: null,\n quality: null,\n loop: null,\n language: null,\n },\n\n // Events to watch and bubble\n events: [\n // Events to watch on HTML5 media elements and bubble\n // https://developer.mozilla.org/en/docs/Web/Guide/Events/Media_events\n 'ended',\n 'progress',\n 'stalled',\n 'playing',\n 'waiting',\n 'canplay',\n 'canplaythrough',\n 'loadstart',\n 'loadeddata',\n 'loadedmetadata',\n 'timeupdate',\n 'volumechange',\n 'play',\n 'pause',\n 'error',\n 'seeking',\n 'seeked',\n 'emptied',\n 'ratechange',\n 'cuechange',\n\n // Custom events\n 'enterfullscreen',\n 'exitfullscreen',\n 'captionsenabled',\n 'captionsdisabled',\n 'languagechange',\n 'controlshidden',\n 'controlsshown',\n 'ready',\n\n // YouTube\n 'statechange',\n 'qualitychange',\n 'qualityrequested',\n ],\n\n // Selectors\n // Change these to match your template if using custom HTML\n selectors: {\n editable: 'input, textarea, select, [contenteditable]',\n container: '.plyr',\n controls: {\n container: null,\n wrapper: '.plyr__controls',\n },\n labels: '[data-plyr]',\n buttons: {\n play: '[data-plyr=\"play\"]',\n pause: '[data-plyr=\"pause\"]',\n restart: '[data-plyr=\"restart\"]',\n rewind: '[data-plyr=\"rewind\"]',\n forward: '[data-plyr=\"fast-forward\"]',\n mute: '[data-plyr=\"mute\"]',\n captions: '[data-plyr=\"captions\"]',\n fullscreen: '[data-plyr=\"fullscreen\"]',\n pip: '[data-plyr=\"pip\"]',\n airplay: '[data-plyr=\"airplay\"]',\n settings: '[data-plyr=\"settings\"]',\n loop: '[data-plyr=\"loop\"]',\n },\n inputs: {\n seek: '[data-plyr=\"seek\"]',\n volume: '[data-plyr=\"volume\"]',\n speed: '[data-plyr=\"speed\"]',\n language: '[data-plyr=\"language\"]',\n quality: '[data-plyr=\"quality\"]',\n },\n display: {\n currentTime: '.plyr__time--current',\n duration: '.plyr__time--duration',\n buffer: '.plyr__progress--buffer',\n played: '.plyr__progress--played',\n loop: '.plyr__progress--loop',\n volume: '.plyr__volume--display',\n },\n progress: '.plyr__progress',\n captions: '.plyr__captions',\n menu: {\n quality: '.js-plyr__menu__list--quality',\n },\n },\n\n // Class hooks added to the player in different states\n classNames: {\n video: 'plyr__video-wrapper',\n embed: 'plyr__video-embed',\n control: 'plyr__control',\n type: 'plyr--{0}',\n provider: 'plyr--{0}',\n stopped: 'plyr--stopped',\n playing: 'plyr--playing',\n loading: 'plyr--loading',\n error: 'plyr--has-error',\n hover: 'plyr--hover',\n tooltip: 'plyr__tooltip',\n hidden: 'plyr__sr-only',\n hideControls: 'plyr--hide-controls',\n isIos: 'plyr--is-ios',\n isTouch: 'plyr--is-touch',\n uiSupported: 'plyr--full-ui',\n noTransition: 'plyr--no-transition',\n menu: {\n value: 'plyr__menu__value',\n badge: 'plyr__badge',\n open: 'plyr--menu-open',\n },\n captions: {\n enabled: 'plyr--captions-enabled',\n active: 'plyr--captions-active',\n },\n fullscreen: {\n enabled: 'plyr--fullscreen-enabled',\n fallback: 'plyr--fullscreen-fallback',\n },\n pip: {\n supported: 'plyr--pip-supported',\n active: 'plyr--pip-active',\n },\n airplay: {\n supported: 'plyr--airplay-supported',\n active: 'plyr--airplay-active',\n },\n tabFocus: 'plyr__tab-focus',\n },\n\n // API keys\n keys: {\n google: null,\n },\n};\n\nexport default defaults;\n","// ==========================================================================\n// Console wrapper\n// ==========================================================================\n\nconst noop = () => {};\n\nexport default class Console {\n constructor(player) {\n this.enabled = window.console && player.config.debug;\n\n if (this.enabled) {\n this.log('Debugging enabled');\n }\n }\n\n get log() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.log, console) : noop;\n }\n get warn() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.warn, console) : noop;\n }\n get error() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.error, console) : noop;\n }\n}\n","// ==========================================================================\n// Plyr storage\n// ==========================================================================\n\nimport utils from './utils';\n\nclass Storage {\n constructor(player) {\n this.enabled = player.config.storage.enabled;\n this.key = player.config.storage.key;\n }\n\n // Check for actual support (see if we can use it)\n static get supported() {\n if (!('localStorage' in window)) {\n return false;\n }\n\n const test = '___test';\n\n // Try to use it (it might be disabled, e.g. user is in private mode)\n // see: https://github.com/sampotts/plyr/issues/131\n try {\n window.localStorage.setItem(test, test);\n window.localStorage.removeItem(test);\n return true;\n } catch (e) {\n return false;\n }\n }\n\n get(key) {\n const store = window.localStorage.getItem(this.key);\n\n if (!Storage.supported || utils.is.empty(store)) {\n return null;\n }\n\n const json = JSON.parse(store);\n\n return utils.is.string(key) && key.length ? json[key] : json;\n }\n\n set(object) {\n // Bail if we don't have localStorage support or it's disabled\n if (!Storage.supported || !this.enabled) {\n return;\n }\n\n // Can only store objectst\n if (!utils.is.object(object)) {\n return;\n }\n\n // Get current storage\n let storage = this.get();\n\n // Default to empty object\n if (utils.is.empty(storage)) {\n storage = {};\n }\n\n // Update the working copy of the values\n utils.extend(storage, object);\n\n // Update storage\n window.localStorage.setItem(this.key, JSON.stringify(storage));\n }\n}\n\nexport default Storage;\n","// ==========================================================================\n// Plyr Event Listeners\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport ui from './ui';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst listeners = {\n // Global listeners\n global() {\n let last = null;\n\n // Get the key code for an event\n const getKeyCode = event => (event.keyCode ? event.keyCode : event.which);\n\n // Handle key press\n const handleKey = event => {\n const code = getKeyCode(event);\n const pressed = event.type === 'keydown';\n const repeat = pressed && code === last;\n\n // Bail if a modifier key is set\n if (event.altKey || event.ctrlKey || event.metaKey || event.shiftKey) {\n return;\n }\n\n // If the event is bubbled from the media element\n // Firefox doesn't get the keycode for whatever reason\n if (!utils.is.number(code)) {\n return;\n }\n\n // Seek by the number keys\n const seekByKey = () => {\n // Divide the max duration into 10th's and times by the number value\n this.currentTime = this.duration / 10 * (code - 48);\n };\n\n // Handle the key on keydown\n // Reset on keyup\n if (pressed) {\n // Which keycodes should we prevent default\n const preventDefault = [\n 48,\n 49,\n 50,\n 51,\n 52,\n 53,\n 54,\n 56,\n 57,\n 32,\n 75,\n 38,\n 40,\n 77,\n 39,\n 37,\n 70,\n 67,\n 73,\n 76,\n 79,\n ];\n\n // Check focused element\n // and if the focused element is not editable (e.g. text input)\n // and any that accept key input http://webaim.org/techniques/keyboard/\n const focused = utils.getFocusElement();\n if (utils.is.element(focused) && utils.matches(focused, this.config.selectors.editable)) {\n return;\n }\n\n // If the code is found prevent default (e.g. prevent scrolling for arrows)\n if (preventDefault.includes(code)) {\n event.preventDefault();\n event.stopPropagation();\n }\n\n switch (code) {\n case 48:\n case 49:\n case 50:\n case 51:\n case 52:\n case 53:\n case 54:\n case 55:\n case 56:\n case 57:\n // 0-9\n if (!repeat) {\n seekByKey();\n }\n break;\n\n case 32:\n case 75:\n // Space and K key\n if (!repeat) {\n this.togglePlay();\n }\n break;\n\n case 38:\n // Arrow up\n this.increaseVolume(0.1);\n break;\n\n case 40:\n // Arrow down\n this.decreaseVolume(0.1);\n break;\n\n case 77:\n // M key\n if (!repeat) {\n this.muted = !this.muted;\n }\n break;\n\n case 39:\n // Arrow forward\n this.forward();\n break;\n\n case 37:\n // Arrow back\n this.rewind();\n break;\n\n case 70:\n // F key\n this.toggleFullscreen();\n break;\n\n case 67:\n // C key\n if (!repeat) {\n this.toggleCaptions();\n }\n break;\n\n case 76:\n // L key\n this.loop = !this.loop;\n break;\n\n /* case 73:\n this.setLoop('start');\n break;\n\n case 76:\n this.setLoop();\n break;\n\n case 79:\n this.setLoop('end');\n break; */\n\n default:\n break;\n }\n\n // Escape is handle natively when in full screen\n // So we only need to worry about non native\n if (!fullscreen.enabled && this.fullscreen.active && code === 27) {\n this.toggleFullscreen();\n }\n\n // Store last code for next cycle\n last = code;\n } else {\n last = null;\n }\n };\n\n // Keyboard shortcuts\n if (this.config.keyboard.global) {\n utils.on(window, 'keydown keyup', handleKey, false);\n } else if (this.config.keyboard.focused) {\n utils.on(this.elements.container, 'keydown keyup', handleKey, false);\n }\n\n // Detect tab focus\n // Remove class on blur/focusout\n utils.on(this.elements.container, 'focusout', event => {\n utils.toggleClass(event.target, this.config.classNames.tabFocus, false);\n });\n\n // Add classname to tabbed elements\n utils.on(this.elements.container, 'keydown', event => {\n if (event.keyCode !== 9) {\n return;\n }\n\n // Delay the adding of classname until the focus has changed\n // This event fires before the focusin event\n window.setTimeout(() => {\n utils.toggleClass(utils.getFocusElement(), this.config.classNames.tabFocus, true);\n }, 0);\n });\n\n // Toggle controls visibility based on mouse movement\n if (this.config.hideControls) {\n // Toggle controls on mouse events and entering fullscreen\n utils.on(this.elements.container, 'mouseenter mouseleave mousemove touchstart touchend touchmove enterfullscreen exitfullscreen', event => {\n this.toggleControls(event);\n });\n }\n\n // Handle user exiting fullscreen by escaping etc\n if (fullscreen.enabled) {\n utils.on(document, fullscreen.eventType, event => {\n this.toggleFullscreen(event);\n });\n\n // Fullscreen toggle on double click\n utils.on(this.elements.container, 'dblclick', event => {\n this.toggleFullscreen(event);\n });\n }\n },\n\n // Listen for media events\n media() {\n // Time change on media\n utils.on(this.media, 'timeupdate seeking', event => ui.timeUpdate.call(this, event));\n\n // Display duration\n utils.on(this.media, 'durationchange loadedmetadata', event => ui.durationUpdate.call(this, event));\n\n // Check for audio tracks on load\n // We can't use `loadedmetadata` as it doesn't seem to have audio tracks at that point\n utils.on(this.media, 'loadeddata', () => {\n utils.toggleHidden(this.elements.volume, !this.hasAudio);\n utils.toggleHidden(this.elements.buttons.mute, !this.hasAudio);\n });\n\n // Handle the media finishing\n utils.on(this.media, 'ended', () => {\n // Show poster on end\n if (this.isHTML5 && this.isVideo && this.config.showPosterOnEnd) {\n // Restart\n this.restart();\n\n // Re-load media\n this.media.load();\n }\n });\n\n // Check for buffer progress\n utils.on(this.media, 'progress playing', event => ui.updateProgress.call(this, event));\n\n // Handle native mute\n utils.on(this.media, 'volumechange', event => ui.updateVolume.call(this, event));\n\n // Handle native play/pause\n utils.on(this.media, 'playing play pause ended', event => ui.checkPlaying.call(this, event));\n\n // Loading\n utils.on(this.media, 'stalled waiting canplay seeked playing', event => ui.checkLoading.call(this, event));\n\n // Check if media failed to load\n // utils.on(this.media, 'play', event => ui.checkFailed.call(this, event));\n\n // Click video\n if (this.supported.ui && this.config.clickToPlay && !this.isAudio) {\n // Re-fetch the wrapper\n const wrapper = utils.getElement.call(this, `.${this.config.classNames.video}`);\n\n // Bail if there's no wrapper (this should never happen)\n if (!utils.is.element(wrapper)) {\n return;\n }\n\n // On click play, pause ore restart\n utils.on(wrapper, 'click', () => {\n // Touch devices will just show controls (if we're hiding controls)\n if (this.config.hideControls && support.touch && !this.paused) {\n return;\n }\n\n if (this.paused) {\n this.play();\n } else if (this.ended) {\n this.restart();\n this.play();\n } else {\n this.pause();\n }\n });\n }\n\n // Disable right click\n if (this.supported.ui && this.config.disableContextMenu) {\n utils.on(\n this.media,\n 'contextmenu',\n event => {\n event.preventDefault();\n },\n false\n );\n }\n\n // Volume change\n utils.on(this.media, 'volumechange', () => {\n // Save to storage\n this.storage.set({ volume: this.volume, muted: this.muted });\n });\n\n // Speed change\n utils.on(this.media, 'ratechange', () => {\n // Update UI\n controls.updateSetting.call(this, 'speed');\n\n // Save to storage\n this.storage.set({ speed: this.speed });\n });\n\n // Quality change\n utils.on(this.media, 'qualitychange', () => {\n // Update UI\n controls.updateSetting.call(this, 'quality');\n\n // Save to storage\n this.storage.set({ quality: this.quality });\n });\n\n // Caption language change\n utils.on(this.media, 'languagechange', () => {\n // Update UI\n controls.updateSetting.call(this, 'captions');\n\n // Save to storage\n this.storage.set({ language: this.language });\n });\n\n // Captions toggle\n utils.on(this.media, 'captionsenabled captionsdisabled', () => {\n // Update UI\n controls.updateSetting.call(this, 'captions');\n\n // Save to storage\n this.storage.set({ captions: this.captions.active });\n });\n\n // Proxy events to container\n // Bubble up key events for Edge\n utils.on(this.media, this.config.events.concat([\n 'keyup',\n 'keydown',\n ]).join(' '), event => {\n let detail = {};\n\n // Get error details from media\n if (event.type === 'error') {\n detail = this.media.error;\n }\n\n utils.dispatchEvent.call(this, this.elements.container, event.type, true, detail);\n });\n },\n\n // Listen for control events\n controls() {\n // IE doesn't support input event, so we fallback to change\n const inputEvent = browser.isIE ? 'change' : 'input';\n\n // Trigger custom and default handlers\n const proxy = (event, handlerKey, defaultHandler) => {\n const customHandler = this.config.listeners[handlerKey];\n\n // Execute custom handler\n if (utils.is.function(customHandler)) {\n customHandler.call(this, event);\n }\n\n // Only call default handler if not prevented in custom handler\n if (!event.defaultPrevented && utils.is.function(defaultHandler)) {\n defaultHandler.call(this, event);\n }\n };\n\n // Play/pause toggle\n utils.on(this.elements.buttons.play, 'click', event =>\n proxy(event, 'play', () => {\n this.togglePlay();\n })\n );\n\n // Pause\n utils.on(this.elements.buttons.restart, 'click', event =>\n proxy(event, 'restart', () => {\n this.restart();\n })\n );\n\n // Rewind\n utils.on(this.elements.buttons.rewind, 'click', event =>\n proxy(event, 'rewind', () => {\n this.rewind();\n })\n );\n\n // Rewind\n utils.on(this.elements.buttons.forward, 'click', event =>\n proxy(event, 'forward', () => {\n this.forward();\n })\n );\n\n // Mute toggle\n utils.on(this.elements.buttons.mute, 'click', event =>\n proxy(event, 'mute', () => {\n this.muted = !this.muted;\n })\n );\n\n // Captions toggle\n utils.on(this.elements.buttons.captions, 'click', event =>\n proxy(event, 'captions', () => {\n this.toggleCaptions();\n })\n );\n\n // Fullscreen toggle\n utils.on(this.elements.buttons.fullscreen, 'click', event =>\n proxy(event, 'fullscreen', () => {\n this.toggleFullscreen();\n })\n );\n\n // Picture-in-Picture\n utils.on(this.elements.buttons.pip, 'click', event =>\n proxy(event, 'pip', () => {\n this.pip = 'toggle';\n })\n );\n\n // Airplay\n utils.on(this.elements.buttons.airplay, 'click', event =>\n proxy(event, 'airplay', () => {\n this.airplay();\n })\n );\n\n // Settings menu\n utils.on(this.elements.buttons.settings, 'click', event => {\n controls.toggleMenu.call(this, event);\n });\n\n // Click anywhere closes menu\n utils.on(document.documentElement, 'click', event => {\n controls.toggleMenu.call(this, event);\n });\n\n // Settings menu\n utils.on(this.elements.settings.form, 'click', event => {\n event.stopPropagation();\n\n // Settings menu items - use event delegation as items are added/removed\n if (utils.matches(event.target, this.config.selectors.inputs.language)) {\n proxy(event, 'language', () => {\n this.language = event.target.value;\n });\n } else if (utils.matches(event.target, this.config.selectors.inputs.quality)) {\n proxy(event, 'quality', () => {\n this.quality = event.target.value;\n });\n } else if (utils.matches(event.target, this.config.selectors.inputs.speed)) {\n proxy(event, 'speed', () => {\n this.speed = parseFloat(event.target.value);\n });\n } else {\n controls.showTab.call(this, event);\n }\n });\n\n // Seek\n utils.on(this.elements.inputs.seek, inputEvent, event =>\n proxy(event, 'seek', () => {\n this.currentTime = event.target.value / event.target.max * this.duration;\n })\n );\n\n // Current time invert\n // Only if one time element is used for both currentTime and duration\n if (this.config.toggleInvert && !utils.is.element(this.elements.display.duration)) {\n utils.on(this.elements.display.currentTime, 'click', () => {\n // Do nothing if we're at the start\n if (this.currentTime === 0) {\n return;\n }\n\n this.config.invertTime = !this.config.invertTime;\n ui.timeUpdate.call(this);\n });\n }\n\n // Volume\n utils.on(this.elements.inputs.volume, inputEvent, event =>\n proxy(event, 'volume', () => {\n this.volume = event.target.value;\n })\n );\n\n // Polyfill for lower fill in <input type=\"range\"> for webkit\n if (browser.isWebkit) {\n utils.on(utils.getElements.call(this, 'input[type=\"range\"]'), 'input', event => {\n controls.updateRangeFill.call(this, event.target);\n });\n }\n\n // Seek tooltip\n utils.on(this.elements.progress, 'mouseenter mouseleave mousemove', event => controls.updateSeekTooltip.call(this, event));\n\n // Toggle controls visibility based on mouse movement\n if (this.config.hideControls) {\n // Watch for cursor over controls so they don't hide when trying to interact\n utils.on(this.elements.controls, 'mouseenter mouseleave', event => {\n this.elements.controls.hover = event.type === 'mouseenter';\n });\n\n // Watch for cursor over controls so they don't hide when trying to interact\n utils.on(this.elements.controls, 'mousedown mouseup touchstart touchend touchcancel', event => {\n this.elements.controls.pressed = [\n 'mousedown',\n 'touchstart',\n ].includes(event.type);\n });\n\n // Focus in/out on controls\n utils.on(this.elements.controls, 'focusin focusout', event => {\n this.toggleControls(event);\n });\n }\n\n // Mouse wheel for volume\n utils.on(\n this.elements.inputs.volume,\n 'wheel',\n event =>\n proxy(event, 'volume', () => {\n // Detect \"natural\" scroll - suppored on OS X Safari only\n // Other browsers on OS X will be inverted until support improves\n const inverted = event.webkitDirectionInvertedFromDevice;\n const step = 1 / 50;\n let direction = 0;\n\n // Scroll down (or up on natural) to decrease\n if (event.deltaY < 0 || event.deltaX > 0) {\n if (inverted) {\n this.decreaseVolume(step);\n direction = -1;\n } else {\n this.increaseVolume(step);\n direction = 1;\n }\n }\n\n // Scroll up (or down on natural) to increase\n if (event.deltaY > 0 || event.deltaX < 0) {\n if (inverted) {\n this.increaseVolume(step);\n direction = 1;\n } else {\n this.decreaseVolume(step);\n direction = -1;\n }\n }\n\n // Don't break page scrolling at max and min\n if ((direction === 1 && this.media.volume < 1) || (direction === -1 && this.media.volume > 0)) {\n event.preventDefault();\n }\n }),\n false\n );\n },\n};\n\nexport default listeners;\n","// ==========================================================================\n// Plyr UI\n// ==========================================================================\n\nimport utils from './utils';\nimport captions from './captions';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport listeners from './listeners';\n\nconst ui = {\n addStyleHook() {\n utils.toggleClass(this.elements.container, this.config.selectors.container.replace('.', ''), true);\n utils.toggleClass(this.elements.container, this.config.classNames.uiSupported, this.supported.ui);\n },\n\n // Toggle native HTML5 media controls\n toggleNativeControls(toggle = false) {\n if (toggle && this.isHTML5) {\n this.media.setAttribute('controls', '');\n } else {\n this.media.removeAttribute('controls');\n }\n },\n\n // Setup the UI\n build() {\n // Re-attach media element listeners\n // TODO: Use event bubbling\n listeners.media.call(this);\n\n // Don't setup interface if no support\n if (!this.supported.ui) {\n this.debug.warn(`Basic support only for ${this.provider} ${this.type}`);\n\n // Remove controls\n utils.removeElement.call(this, 'controls');\n\n // Remove large play\n utils.removeElement.call(this, 'buttons.play');\n\n // Restore native controls\n ui.toggleNativeControls.call(this, true);\n\n // Bail\n return;\n }\n\n // Inject custom controls if not present\n if (!utils.is.element(this.elements.controls)) {\n // Inject custom controls\n controls.inject.call(this);\n\n // Re-attach control listeners\n listeners.controls.call(this);\n }\n\n // If there's no controls, bail\n if (!utils.is.element(this.elements.controls)) {\n return;\n }\n\n // Remove native controls\n ui.toggleNativeControls.call(this);\n\n // Setup fullscreen\n fullscreen.setup.call(this);\n\n // Captions\n captions.setup.call(this);\n\n // Reset volume\n this.volume = null;\n\n // Reset mute state\n this.muted = null;\n\n // Reset speed\n this.speed = null;\n\n // Reset loop state\n this.loop = null;\n\n // Reset quality options\n this.options.quality = [];\n\n // Reset time display\n ui.timeUpdate.call(this);\n\n // Update the UI\n ui.checkPlaying.call(this);\n\n // Ready for API calls\n this.ready = true;\n\n // Ready event at end of execution stack\n utils.dispatchEvent.call(this, this.media, 'ready');\n\n // Set the title\n ui.setTitle.call(this);\n },\n\n // Setup aria attribute for play and iframe title\n setTitle() {\n // Find the current text\n let label = this.config.i18n.play;\n\n // If there's a media title set, use that for the label\n if (utils.is.string(this.config.title) && !utils.is.empty(this.config.title)) {\n label += `, ${this.config.title}`;\n\n // Set container label\n this.elements.container.setAttribute('aria-label', this.config.title);\n }\n\n // If there's a play button, set label\n if (utils.is.nodeList(this.elements.buttons.play)) {\n Array.from(this.elements.buttons.play).forEach(button => {\n button.setAttribute('aria-label', label);\n });\n }\n\n // Set iframe title\n // https://github.com/sampotts/plyr/issues/124\n if (this.isEmbed) {\n const iframe = utils.getElement.call(this, 'iframe');\n\n if (!utils.is.element(iframe)) {\n return;\n }\n\n // Default to media type\n const title = !utils.is.empty(this.config.title) ? this.config.title : 'video';\n\n iframe.setAttribute('title', this.config.i18n.frameTitle.replace('{title}', title));\n }\n },\n\n // Check playing state\n checkPlaying() {\n // Class hooks\n utils.toggleClass(this.elements.container, this.config.classNames.playing, this.playing);\n utils.toggleClass(this.elements.container, this.config.classNames.stopped, this.paused);\n\n // Set aria state\n if (utils.is.nodeList(this.elements.buttons.play)) {\n Array.from(this.elements.buttons.play).forEach(button => utils.toggleState(button, this.playing));\n }\n\n // Toggle controls\n this.toggleControls(!this.playing);\n },\n\n // Check if media is loading\n checkLoading(event) {\n this.loading = [\n 'stalled',\n 'waiting',\n ].includes(event.type);\n\n // Clear timer\n clearTimeout(this.timers.loading);\n\n // Timer to prevent flicker when seeking\n this.timers.loading = setTimeout(() => {\n // Toggle container class hook\n utils.toggleClass(this.elements.container, this.config.classNames.loading, this.loading);\n\n // Show controls if loading, hide if done\n this.toggleControls(this.loading);\n }, this.loading ? 250 : 0);\n },\n\n // Check if media failed to load\n checkFailed() {\n // https://developer.mozilla.org/en-US/docs/Web/API/HTMLMediaElement/networkState\n this.failed = this.media.networkState === 3;\n\n if (this.failed) {\n utils.toggleClass(this.elements.container, this.config.classNames.loading, false);\n utils.toggleClass(this.elements.container, this.config.classNames.error, true);\n }\n\n // Clear timer\n clearTimeout(this.timers.failed);\n\n // Timer to prevent flicker when seeking\n this.timers.loading = setTimeout(() => {\n // Toggle container class hook\n utils.toggleClass(this.elements.container, this.config.classNames.loading, this.loading);\n\n // Show controls if loading, hide if done\n this.toggleControls(this.loading);\n }, this.loading ? 250 : 0);\n },\n\n // Update volume UI and storage\n updateVolume() {\n if (!this.supported.ui) {\n return;\n }\n\n // Update range\n if (utils.is.element(this.elements.inputs.volume)) {\n ui.setRange.call(this, this.elements.inputs.volume, this.muted ? 0 : this.volume);\n }\n\n // Update mute state\n if (utils.is.element(this.elements.buttons.mute)) {\n utils.toggleState(this.elements.buttons.mute, this.muted || this.volume === 0);\n }\n },\n\n // Update seek value and lower fill\n setRange(target, value = 0) {\n if (!utils.is.element(target)) {\n return;\n }\n\n // eslint-disable-next-line\n target.value = value;\n\n // Webkit range fill\n controls.updateRangeFill.call(this, target);\n },\n\n // Set <progress> value\n setProgress(target, input) {\n const value = utils.is.number(input) ? input : 0;\n const progress = utils.is.element(target) ? target : this.elements.display.buffer;\n\n // Update value and label\n if (utils.is.element(progress)) {\n progress.value = value;\n\n // Update text label inside\n const label = progress.getElementsByTagName('span')[0];\n if (utils.is.element(label)) {\n label.childNodes[0].nodeValue = value;\n }\n }\n },\n\n // Update <progress> elements\n updateProgress(event) {\n if (!this.supported.ui || !utils.is.event(event)) {\n return;\n }\n\n let value = 0;\n\n if (event) {\n switch (event.type) {\n // Video playing\n case 'timeupdate':\n case 'seeking':\n value = utils.getPercentage(this.currentTime, this.duration);\n\n // Set seek range value only if it's a 'natural' time event\n if (event.type === 'timeupdate') {\n ui.setRange.call(this, this.elements.inputs.seek, value);\n }\n\n break;\n\n // Check buffer status\n case 'playing':\n case 'progress':\n value = (() => {\n const { buffered } = this.media;\n\n if (buffered && buffered.length) {\n // HTML5\n return utils.getPercentage(buffered.end(0), this.duration);\n } else if (utils.is.number(buffered)) {\n // YouTube returns between 0 and 1\n return buffered * 100;\n }\n\n return 0;\n })();\n\n ui.setProgress.call(this, this.elements.display.buffer, value);\n\n break;\n\n default:\n break;\n }\n }\n },\n\n // Update the displayed time\n updateTimeDisplay(target = null, time = 0, inverted = false) {\n // Bail if there's no element to display or the value isn't a number\n if (!utils.is.element(target) || !utils.is.number(time)) {\n return;\n }\n\n // Format time component to add leading zero\n const format = value => `0${value}`.slice(-2);\n\n // Helpers\n const getHours = value => parseInt((value / 60 / 60) % 60, 10);\n const getMinutes = value => parseInt((value / 60) % 60, 10);\n const getSeconds = value => parseInt(value % 60, 10);\n\n // Breakdown to hours, mins, secs\n let hours = getHours(time);\n const mins = getMinutes(time);\n const secs = getSeconds(time);\n\n // Do we need to display hours?\n if (getHours(this.duration) > 0) {\n hours = `${hours}:`;\n } else {\n hours = '';\n }\n\n // Render\n // eslint-disable-next-line no-param-reassign\n target.textContent = `${inverted ? '-' : ''}${hours}${format(mins)}:${format(secs)}`;\n },\n\n // Handle time change event\n timeUpdate(event) {\n // Only invert if only one time element is displayed and used for both duration and currentTime\n const invert = !utils.is.element(this.elements.display.duration) && this.config.invertTime;\n\n // Duration\n ui.updateTimeDisplay.call(this, this.elements.display.currentTime, invert ? this.duration - this.currentTime : this.currentTime, invert);\n\n // Ignore updates while seeking\n if (event && event.type === 'timeupdate' && this.media.seeking) {\n return;\n }\n\n // Playing progress\n ui.updateProgress.call(this, event);\n },\n\n // Show the duration on metadataloaded\n durationUpdate() {\n if (!this.supported.ui) {\n return;\n }\n\n // If there's only one time display, display duration there\n if (!utils.is.element(this.elements.display.duration) && this.config.displayDuration && this.paused) {\n ui.updateTimeDisplay.call(this, this.elements.display.currentTime, this.duration);\n }\n\n // If there's a duration element, update content\n if (utils.is.element(this.elements.display.duration)) {\n ui.updateTimeDisplay.call(this, this.elements.display.duration, this.duration);\n }\n\n // Update the tooltip (if visible)\n controls.updateSeekTooltip.call(this);\n },\n};\n\nexport default ui;\n","// ==========================================================================\n// Plyr controls\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport ui from './ui';\nimport captions from './captions';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst controls = {\n // Webkit polyfill for lower fill range\n updateRangeFill(target) {\n // WebKit only\n if (!browser.isWebkit) {\n return;\n }\n\n // Get range from event if event passed\n const range = utils.is.event(target) ? target.target : target;\n\n // Needs to be a valid <input type='range'>\n if (!utils.is.element(range) || range.getAttribute('type') !== 'range') {\n return;\n }\n\n // Set CSS custom property\n range.style.setProperty('--value', `${range.value / range.max * 100}%`);\n },\n\n // Get icon URL\n getIconUrl() {\n return {\n url: this.config.iconUrl,\n absolute: this.config.iconUrl.indexOf('http') === 0 || (browser.isIE && !window.svg4everybody),\n };\n },\n\n // Create <svg> icon\n createIcon(type, attributes) {\n const namespace = 'http://www.w3.org/2000/svg';\n const iconUrl = controls.getIconUrl.call(this);\n const iconPath = `${!iconUrl.absolute ? iconUrl.url : ''}#${this.config.iconPrefix}`;\n\n // Create <svg>\n const icon = document.createElementNS(namespace, 'svg');\n utils.setAttributes(\n icon,\n utils.extend(attributes, {\n role: 'presentation',\n })\n );\n\n // Create the <use> to reference sprite\n const use = document.createElementNS(namespace, 'use');\n const path = `${iconPath}-${type}`;\n\n // Set `href` attributes\n // https://github.com/sampotts/plyr/issues/460\n // https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/xlink:href\n if ('href' in use) {\n use.setAttributeNS('http://www.w3.org/1999/xlink', 'href', path);\n } else {\n use.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href', path);\n }\n\n // Add <use> to <svg>\n icon.appendChild(use);\n\n return icon;\n },\n\n // Create hidden text label\n createLabel(type, attr) {\n let text = this.config.i18n[type];\n const attributes = Object.assign({}, attr);\n\n switch (type) {\n case 'pip':\n text = 'PIP';\n break;\n\n case 'airplay':\n text = 'AirPlay';\n break;\n\n default:\n break;\n }\n\n if ('class' in attributes) {\n attributes.class += ` ${this.config.classNames.hidden}`;\n } else {\n attributes.class = this.config.classNames.hidden;\n }\n\n return utils.createElement('span', attributes, text);\n },\n\n // Create a badge\n createBadge(text) {\n if (utils.is.empty(text)) {\n return null;\n }\n\n const badge = utils.createElement('span', {\n class: this.config.classNames.menu.value,\n });\n\n badge.appendChild(\n utils.createElement(\n 'span',\n {\n class: this.config.classNames.menu.badge,\n },\n text\n )\n );\n\n return badge;\n },\n\n // Create a <button>\n createButton(buttonType, attr) {\n const button = utils.createElement('button');\n const attributes = Object.assign({}, attr);\n let type = buttonType;\n\n let toggle = false;\n let label;\n let icon;\n let labelPressed;\n let iconPressed;\n\n if (!('type' in attributes)) {\n attributes.type = 'button';\n }\n\n if ('class' in attributes) {\n if (attributes.class.includes(this.config.classNames.control)) {\n attributes.class += ` ${this.config.classNames.control}`;\n }\n } else {\n attributes.class = this.config.classNames.control;\n }\n\n // Large play button\n switch (type) {\n case 'play':\n toggle = true;\n label = 'play';\n labelPressed = 'pause';\n icon = 'play';\n iconPressed = 'pause';\n break;\n\n case 'mute':\n toggle = true;\n label = 'mute';\n labelPressed = 'unmute';\n icon = 'volume';\n iconPressed = 'muted';\n break;\n\n case 'captions':\n toggle = true;\n label = 'enableCaptions';\n labelPressed = 'disableCaptions';\n icon = 'captions-off';\n iconPressed = 'captions-on';\n break;\n\n case 'fullscreen':\n toggle = true;\n label = 'enterFullscreen';\n labelPressed = 'exitFullscreen';\n icon = 'enter-fullscreen';\n iconPressed = 'exit-fullscreen';\n break;\n\n case 'play-large':\n attributes.class += ` ${this.config.classNames.control}--overlaid`;\n type = 'play';\n label = 'play';\n icon = 'play';\n break;\n\n default:\n label = type;\n icon = type;\n }\n\n // Setup toggle icon and labels\n if (toggle) {\n // Icon\n button.appendChild(controls.createIcon.call(this, iconPressed, { class: 'icon--pressed' }));\n button.appendChild(controls.createIcon.call(this, icon, { class: 'icon--not-pressed' }));\n\n // Label/Tooltip\n button.appendChild(controls.createLabel.call(this, labelPressed, { class: 'label--pressed' }));\n button.appendChild(controls.createLabel.call(this, label, { class: 'label--not-pressed' }));\n\n // Add aria attributes\n attributes['aria-pressed'] = false;\n attributes['aria-label'] = this.config.i18n[label];\n } else {\n button.appendChild(controls.createIcon.call(this, icon));\n button.appendChild(controls.createLabel.call(this, label));\n }\n\n // Merge attributes\n utils.extend(attributes, utils.getAttributesFromSelector(this.config.selectors.buttons[type], attributes));\n\n utils.setAttributes(button, attributes);\n\n this.elements.buttons[type] = button;\n\n return button;\n },\n\n // Create an <input type='range'>\n createRange(type, attributes) {\n // Seek label\n const label = utils.createElement(\n 'label',\n {\n for: attributes.id,\n class: this.config.classNames.hidden,\n },\n this.config.i18n[type]\n );\n\n // Seek input\n const input = utils.createElement(\n 'input',\n utils.extend(\n utils.getAttributesFromSelector(this.config.selectors.inputs[type]),\n {\n type: 'range',\n min: 0,\n max: 100,\n step: 0.01,\n value: 0,\n autocomplete: 'off',\n },\n attributes\n )\n );\n\n this.elements.inputs[type] = input;\n\n // Set the fill for webkit now\n controls.updateRangeFill.call(this, input);\n\n return {\n label,\n input,\n };\n },\n\n // Create a <progress>\n createProgress(type, attributes) {\n const progress = utils.createElement(\n 'progress',\n utils.extend(\n utils.getAttributesFromSelector(this.config.selectors.display[type]),\n {\n min: 0,\n max: 100,\n value: 0,\n },\n attributes\n )\n );\n\n // Create the label inside\n if (type !== 'volume') {\n progress.appendChild(utils.createElement('span', null, '0'));\n\n let suffix = '';\n switch (type) {\n case 'played':\n suffix = this.config.i18n.played;\n break;\n\n case 'buffer':\n suffix = this.config.i18n.buffered;\n break;\n\n default:\n break;\n }\n\n progress.textContent = `% ${suffix.toLowerCase()}`;\n }\n\n this.elements.display[type] = progress;\n\n return progress;\n },\n\n // Create time display\n createTime(type) {\n const container = utils.createElement('div', {\n class: 'plyr__time',\n });\n\n container.appendChild(\n utils.createElement(\n 'span',\n {\n class: this.config.classNames.hidden,\n },\n this.config.i18n[type]\n )\n );\n\n container.appendChild(utils.createElement('span', utils.getAttributesFromSelector(this.config.selectors.display[type]), '00:00'));\n\n this.elements.display[type] = container;\n\n return container;\n },\n\n // Create a settings menu item\n createMenuItem(value, list, type, title, badge = null, checked = false) {\n const item = utils.createElement('li');\n\n const label = utils.createElement('label', {\n class: this.config.classNames.control,\n });\n\n const radio = utils.createElement(\n 'input',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.inputs[type]), {\n type: 'radio',\n name: `plyr-${type}`,\n value,\n checked,\n class: 'plyr__sr-only',\n })\n );\n\n const faux = utils.createElement('span', { 'aria-hidden': true });\n\n label.appendChild(radio);\n label.appendChild(faux);\n label.insertAdjacentHTML('beforeend', title);\n\n if (utils.is.element(badge)) {\n label.appendChild(badge);\n }\n\n item.appendChild(label);\n list.appendChild(item);\n },\n\n // Update hover tooltip for seeking\n updateSeekTooltip(event) {\n // Bail if setting not true\n if (\n !this.config.tooltips.seek ||\n !utils.is.element(this.elements.inputs.seek) ||\n !utils.is.element(this.elements.display.seekTooltip) ||\n this.duration === 0\n ) {\n return;\n }\n\n // Calculate percentage\n let percent = 0;\n const clientRect = this.elements.inputs.seek.getBoundingClientRect();\n const visible = `${this.config.classNames.tooltip}--visible`;\n\n // Determine percentage, if already visible\n if (utils.is.event(event)) {\n percent = 100 / clientRect.width * (event.pageX - clientRect.left);\n } else if (utils.hasClass(this.elements.display.seekTooltip, visible)) {\n percent = parseFloat(this.elements.display.seekTooltip.style.left, 10);\n } else {\n return;\n }\n\n // Set bounds\n if (percent < 0) {\n percent = 0;\n } else if (percent > 100) {\n percent = 100;\n }\n\n // Display the time a click would seek to\n ui.updateTimeDisplay.call(this, this.elements.display.seekTooltip, this.duration / 100 * percent);\n\n // Set position\n this.elements.display.seekTooltip.style.left = `${percent}%`;\n\n // Show/hide the tooltip\n // If the event is a moues in/out and percentage is inside bounds\n if (utils.is.event(event) && [\n 'mouseenter',\n 'mouseleave',\n ].includes(event.type)) {\n utils.toggleClass(this.elements.display.seekTooltip, visible, event.type === 'mouseenter');\n }\n },\n\n // Hide/show a tab\n toggleTab(setting, toggle) {\n const tab = this.elements.settings.tabs[setting];\n const pane = this.elements.settings.panes[setting];\n\n utils.toggleHidden(tab, !toggle);\n utils.toggleHidden(pane, !toggle);\n },\n\n // Set the YouTube quality menu\n // TODO: Support for HTML5\n setQualityMenu(options) {\n const type = 'quality';\n const list = this.elements.settings.panes.quality.querySelector('ul');\n\n // Set options if passed and filter based on config\n if (utils.is.array(options)) {\n this.options.quality = options.filter(quality => this.config.quality.options.includes(quality));\n } else {\n this.options.quality = this.config.quality.options;\n }\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.options.quality) && this.isYouTube;\n controls.toggleTab.call(this, type, toggle);\n\n // If we're hiding, nothing more to do\n if (!toggle) {\n return;\n }\n\n // Empty the menu\n utils.emptyElement(list);\n\n // Get the badge HTML for HD, 4K etc\n const getBadge = quality => {\n let label = '';\n\n switch (quality) {\n case 'hd2160':\n label = '4K';\n break;\n\n case 'hd1440':\n label = 'WQHD';\n break;\n\n case 'hd1080':\n label = 'HD';\n break;\n\n case 'hd720':\n label = 'HD';\n break;\n\n default:\n break;\n }\n\n if (!label.length) {\n return null;\n }\n\n return controls.createBadge.call(this, label);\n };\n\n this.options.quality.forEach(quality =>\n controls.createMenuItem.call(this, quality, list, type, controls.getLabel.call(this, 'quality', quality), getBadge(quality))\n );\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Translate a value into a nice label\n // TODO: Localisation\n getLabel(setting, value) {\n switch (setting) {\n case 'speed':\n return value === 1 ? 'Normal' : `${value}×`;\n\n case 'quality':\n switch (value) {\n case 'hd2160':\n return '2160P';\n case 'hd1440':\n return '1440P';\n case 'hd1080':\n return '1080P';\n case 'hd720':\n return '720P';\n case 'large':\n return '480P';\n case 'medium':\n return '360P';\n case 'small':\n return '240P';\n case 'tiny':\n return 'Tiny';\n case 'default':\n return 'Auto';\n default:\n return value;\n }\n\n case 'captions':\n return controls.getLanguage.call(this);\n\n default:\n return null;\n }\n },\n\n // Update the selected setting\n updateSetting(setting, container) {\n const pane = this.elements.settings.panes[setting];\n let value = null;\n let list = container;\n\n switch (setting) {\n case 'captions':\n value = this.captions.active ? this.captions.language : '';\n break;\n\n default:\n value = this[setting];\n\n // Get default\n if (utils.is.empty(value)) {\n value = this.config[setting].default;\n }\n\n // Unsupported value\n if (!this.options[setting].includes(value)) {\n this.debug.warn(`Unsupported value of '${value}' for ${setting}`);\n return;\n }\n\n // Disabled value\n if (!this.config[setting].options.includes(value)) {\n this.debug.warn(`Disabled value of '${value}' for ${setting}`);\n return;\n }\n\n break;\n }\n\n // Get the list if we need to\n if (!utils.is.element(list)) {\n list = pane && pane.querySelector('ul');\n }\n\n // Update the label\n if (!utils.is.empty(value)) {\n const label = this.elements.settings.tabs[setting].querySelector(`.${this.config.classNames.menu.value}`);\n label.innerHTML = controls.getLabel.call(this, setting, value);\n }\n\n // Find the radio option\n const target = list && list.querySelector(`input[value=\"${value}\"]`);\n\n if (utils.is.element(target)) {\n // Check it\n target.checked = true;\n }\n },\n\n // Set the looping options\n /* setLoopMenu() {\n const options = ['start', 'end', 'all', 'reset'];\n const list = this.elements.settings.panes.loop.querySelector('ul');\n\n // Show the pane and tab\n utils.toggleHidden(this.elements.settings.tabs.loop, false);\n utils.toggleHidden(this.elements.settings.panes.loop, false);\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.loop.options);\n controls.toggleTab.call(this, 'loop', toggle);\n\n // Empty the menu\n utils.emptyElement(list);\n\n options.forEach(option => {\n const item = utils.createElement('li');\n\n const button = utils.createElement(\n 'button',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.buttons.loop), {\n type: 'button',\n class: this.config.classNames.control,\n 'data-plyr-loop-action': option,\n }),\n this.config.i18n[option]\n );\n\n if (['start', 'end'].includes(option)) {\n const badge = controls.createBadge.call(this, '00:00');\n button.appendChild(badge);\n }\n\n item.appendChild(button);\n list.appendChild(item);\n });\n }, */\n\n // Get current selected caption language\n // TODO: rework this to user the getter in the API?\n getLanguage() {\n if (!this.supported.ui) {\n return null;\n }\n\n if (!support.textTracks || !captions.getTracks.call(this).length) {\n return this.config.i18n.none;\n }\n\n if (this.captions.active) {\n const currentTrack = captions.getCurrentTrack.call(this);\n\n if (utils.is.track(currentTrack)) {\n return currentTrack.label;\n }\n }\n\n return this.config.i18n.disabled;\n },\n\n // Set a list of available captions languages\n setCaptionsMenu() {\n // TODO: Captions or language? Currently it's mixed\n const type = 'captions';\n const list = this.elements.settings.panes.captions.querySelector('ul');\n\n // Toggle the pane and tab\n const hasTracks = captions.getTracks.call(this).length;\n controls.toggleTab.call(this, type, hasTracks);\n\n // Empty the menu\n utils.emptyElement(list);\n\n // If there's no captions, bail\n if (!hasTracks) {\n return;\n }\n\n // Re-map the tracks into just the data we need\n const tracks = captions.getTracks.call(this).map(track => ({\n language: track.language,\n label: !utils.is.empty(track.label) ? track.label : track.language.toUpperCase(),\n }));\n\n // Add the \"None\" option to turn off captions\n tracks.unshift({\n language: '',\n label: this.config.i18n.none,\n });\n\n // Generate options\n tracks.forEach(track => {\n controls.createMenuItem.call(\n this,\n track.language,\n list,\n 'language',\n track.label || track.language,\n controls.createBadge.call(this, track.language.toUpperCase()),\n track.language.toLowerCase() === this.captions.language.toLowerCase()\n );\n });\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Set a list of available captions languages\n setSpeedMenu() {\n const type = 'speed';\n\n // Set the default speeds\n if (!utils.is.object(this.options.speed) || !Object.keys(this.options.speed).length) {\n this.options.speed = [\n 0.5,\n 0.75,\n 1,\n 1.25,\n 1.5,\n 1.75,\n 2,\n ];\n }\n\n // Set options if passed and filter based on config\n this.options.speed = this.options.speed.filter(speed => this.config.speed.options.includes(speed));\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.options.speed);\n controls.toggleTab.call(this, type, toggle);\n\n // If we're hiding, nothing more to do\n if (!toggle) {\n return;\n }\n\n // Get the list to populate\n const list = this.elements.settings.panes.speed.querySelector('ul');\n\n // Show the pane and tab\n utils.toggleHidden(this.elements.settings.tabs.speed, false);\n utils.toggleHidden(this.elements.settings.panes.speed, false);\n\n // Empty the menu\n utils.emptyElement(list);\n\n // Create items\n this.options.speed.forEach(speed => controls.createMenuItem.call(this, speed, list, type, controls.getLabel.call(this, 'speed', speed)));\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Show/hide menu\n toggleMenu(event) {\n const { form } = this.elements.settings;\n const button = this.elements.buttons.settings;\n const show = utils.is.boolean(event) ? event : utils.is.element(form) && form.getAttribute('aria-hidden') === 'true';\n\n if (utils.is.event(event)) {\n const isMenuItem = utils.is.element(form) && form.contains(event.target);\n const isButton = event.target === this.elements.buttons.settings;\n\n // If the click was inside the form or if the click\n // wasn't the button or menu item and we're trying to\n // show the menu (a doc click shouldn't show the menu)\n if (isMenuItem || (!isMenuItem && !isButton && show)) {\n return;\n }\n\n // Prevent the toggle being caught by the doc listener\n if (isButton) {\n event.stopPropagation();\n }\n }\n\n // Set form and button attributes\n if (utils.is.element(button)) {\n button.setAttribute('aria-expanded', show);\n }\n\n if (utils.is.element(form)) {\n form.setAttribute('aria-hidden', !show);\n utils.toggleClass(this.elements.container, this.config.classNames.menu.open, show);\n\n if (show) {\n form.removeAttribute('tabindex');\n } else {\n form.setAttribute('tabindex', -1);\n }\n }\n },\n\n // Get the natural size of a tab\n getTabSize(tab) {\n const clone = tab.cloneNode(true);\n clone.style.position = 'absolute';\n clone.style.opacity = 0;\n clone.setAttribute('aria-hidden', false);\n\n // Prevent input's being unchecked due to the name being identical\n Array.from(clone.querySelectorAll('input[name]')).forEach(input => {\n const name = input.getAttribute('name');\n input.setAttribute('name', `${name}-clone`);\n });\n\n // Append to parent so we get the \"real\" size\n tab.parentNode.appendChild(clone);\n\n // Get the sizes before we remove\n const width = clone.scrollWidth;\n const height = clone.scrollHeight;\n\n // Remove from the DOM\n utils.removeElement(clone);\n\n return {\n width,\n height,\n };\n },\n\n // Toggle Menu\n showTab(event) {\n const { menu } = this.elements.settings;\n const tab = event.target;\n const show = tab.getAttribute('aria-expanded') === 'false';\n const pane = document.getElementById(tab.getAttribute('aria-controls'));\n\n // Nothing to show, bail\n if (!utils.is.element(pane)) {\n return;\n }\n\n // Are we targetting a tab? If not, bail\n const isTab = pane.getAttribute('role') === 'tabpanel';\n if (!isTab) {\n return;\n }\n\n // Hide all other tabs\n // Get other tabs\n const current = menu.querySelector('[role=\"tabpanel\"][aria-hidden=\"false\"]');\n const container = current.parentNode;\n\n // Set other toggles to be expanded false\n Array.from(menu.querySelectorAll(`[aria-controls=\"${current.getAttribute('id')}\"]`)).forEach(toggle => {\n toggle.setAttribute('aria-expanded', false);\n });\n\n // If we can do fancy animations, we'll animate the height/width\n if (support.transitions && !support.reducedMotion) {\n // Set the current width as a base\n container.style.width = `${current.scrollWidth}px`;\n container.style.height = `${current.scrollHeight}px`;\n\n // Get potential sizes\n const size = controls.getTabSize.call(this, pane);\n\n // Restore auto height/width\n const restore = e => {\n // We're only bothered about height and width on the container\n if (e.target !== container || ![\n 'width',\n 'height',\n ].includes(e.propertyName)) {\n return;\n }\n\n // Revert back to auto\n container.style.width = '';\n container.style.height = '';\n\n // Only listen once\n utils.off(container, utils.transitionEnd, restore);\n };\n\n // Listen for the transition finishing and restore auto height/width\n utils.on(container, utils.transitionEnd, restore);\n\n // Set dimensions to target\n container.style.width = `${size.width}px`;\n container.style.height = `${size.height}px`;\n }\n\n // Set attributes on current tab\n current.setAttribute('aria-hidden', true);\n current.setAttribute('tabindex', -1);\n\n // Set attributes on target\n pane.setAttribute('aria-hidden', !show);\n tab.setAttribute('aria-expanded', show);\n pane.removeAttribute('tabindex');\n\n // Focus the first item\n pane.querySelectorAll('button:not(:disabled), input:not(:disabled), [tabindex]')[0].focus();\n },\n\n // Build the default HTML\n // TODO: Set order based on order in the config.controls array?\n create(data) {\n // Do nothing if we want no controls\n if (utils.is.empty(this.config.controls)) {\n return null;\n }\n\n // Create the container\n const container = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.controls.wrapper));\n\n // Restart button\n if (this.config.controls.includes('restart')) {\n container.appendChild(controls.createButton.call(this, 'restart'));\n }\n\n // Rewind button\n if (this.config.controls.includes('rewind')) {\n container.appendChild(controls.createButton.call(this, 'rewind'));\n }\n\n // Play/Pause button\n if (this.config.controls.includes('play')) {\n container.appendChild(controls.createButton.call(this, 'play'));\n // container.appendChild(controls.createButton.call(this, 'pause'));\n }\n\n // Fast forward button\n if (this.config.controls.includes('fast-forward')) {\n container.appendChild(controls.createButton.call(this, 'fast-forward'));\n }\n\n // Progress\n if (this.config.controls.includes('progress')) {\n const progress = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.progress));\n\n // Seek range slider\n const seek = controls.createRange.call(this, 'seek', {\n id: `plyr-seek-${data.id}`,\n });\n progress.appendChild(seek.label);\n progress.appendChild(seek.input);\n\n // Buffer progress\n progress.appendChild(controls.createProgress.call(this, 'buffer'));\n\n // TODO: Add loop display indicator\n\n // Seek tooltip\n if (this.config.tooltips.seek) {\n const tooltip = utils.createElement(\n 'span',\n {\n role: 'tooltip',\n class: this.config.classNames.tooltip,\n },\n '00:00'\n );\n\n progress.appendChild(tooltip);\n this.elements.display.seekTooltip = tooltip;\n }\n\n this.elements.progress = progress;\n container.appendChild(this.elements.progress);\n }\n\n // Media current time display\n if (this.config.controls.includes('current-time')) {\n container.appendChild(controls.createTime.call(this, 'currentTime'));\n }\n\n // Media duration display\n if (this.config.controls.includes('duration')) {\n container.appendChild(controls.createTime.call(this, 'duration'));\n }\n\n // Toggle mute button\n if (this.config.controls.includes('mute')) {\n container.appendChild(controls.createButton.call(this, 'mute'));\n }\n\n // Volume range control\n if (this.config.controls.includes('volume')) {\n const volume = utils.createElement('div', {\n class: 'plyr__volume',\n });\n\n // Set the attributes\n const attributes = {\n max: 1,\n step: 0.05,\n value: this.config.volume,\n };\n\n // Create the volume range slider\n const range = controls.createRange.call(\n this,\n 'volume',\n utils.extend(attributes, {\n id: `plyr-volume-${data.id}`,\n })\n );\n volume.appendChild(range.label);\n volume.appendChild(range.input);\n\n this.elements.volume = volume;\n\n container.appendChild(volume);\n }\n\n // Toggle captions button\n if (this.config.controls.includes('captions')) {\n container.appendChild(controls.createButton.call(this, 'captions'));\n }\n\n // Settings button / menu\n if (this.config.controls.includes('settings') && !utils.is.empty(this.config.settings)) {\n const menu = utils.createElement('div', {\n class: 'plyr__menu',\n });\n\n menu.appendChild(\n controls.createButton.call(this, 'settings', {\n id: `plyr-settings-toggle-${data.id}`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}`,\n 'aria-expanded': false,\n })\n );\n\n const form = utils.createElement('form', {\n class: 'plyr__menu__container',\n id: `plyr-settings-${data.id}`,\n 'aria-hidden': true,\n 'aria-labelled-by': `plyr-settings-toggle-${data.id}`,\n role: 'tablist',\n tabindex: -1,\n });\n\n const inner = utils.createElement('div');\n\n const home = utils.createElement('div', {\n id: `plyr-settings-${data.id}-home`,\n 'aria-hidden': false,\n 'aria-labelled-by': `plyr-settings-toggle-${data.id}`,\n role: 'tabpanel',\n });\n\n // Create the tab list\n const tabs = utils.createElement('ul', {\n role: 'tablist',\n });\n\n // Build the tabs\n this.config.settings.forEach(type => {\n const tab = utils.createElement('li', {\n role: 'tab',\n hidden: '',\n });\n\n const button = utils.createElement(\n 'button',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.buttons.settings), {\n type: 'button',\n class: `${this.config.classNames.control} ${this.config.classNames.control}--forward`,\n id: `plyr-settings-${data.id}-${type}-tab`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}-${type}`,\n 'aria-expanded': false,\n }),\n this.config.i18n[type]\n );\n\n const value = utils.createElement('span', {\n class: this.config.classNames.menu.value,\n });\n\n // Speed contains HTML entities\n value.innerHTML = data[type];\n\n button.appendChild(value);\n tab.appendChild(button);\n tabs.appendChild(tab);\n\n this.elements.settings.tabs[type] = tab;\n });\n\n home.appendChild(tabs);\n inner.appendChild(home);\n\n // Build the panes\n this.config.settings.forEach(type => {\n const pane = utils.createElement('div', {\n id: `plyr-settings-${data.id}-${type}`,\n 'aria-hidden': true,\n 'aria-labelled-by': `plyr-settings-${data.id}-${type}-tab`,\n role: 'tabpanel',\n tabindex: -1,\n hidden: '',\n });\n\n const back = utils.createElement(\n 'button',\n {\n type: 'button',\n class: `${this.config.classNames.control} ${this.config.classNames.control}--back`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}-home`,\n 'aria-expanded': false,\n },\n this.config.i18n[type]\n );\n\n pane.appendChild(back);\n\n const options = utils.createElement('ul');\n\n pane.appendChild(options);\n inner.appendChild(pane);\n\n this.elements.settings.panes[type] = pane;\n });\n\n form.appendChild(inner);\n menu.appendChild(form);\n container.appendChild(menu);\n\n this.elements.settings.form = form;\n this.elements.settings.menu = menu;\n }\n\n // Picture in picture button\n if (this.config.controls.includes('pip') && support.pip) {\n container.appendChild(controls.createButton.call(this, 'pip'));\n }\n\n // Airplay button\n if (this.config.controls.includes('airplay') && support.airplay) {\n container.appendChild(controls.createButton.call(this, 'airplay'));\n }\n\n // Toggle fullscreen button\n if (this.config.controls.includes('fullscreen')) {\n container.appendChild(controls.createButton.call(this, 'fullscreen'));\n }\n\n // Larger overlaid play button\n if (this.config.controls.includes('play-large')) {\n this.elements.container.appendChild(controls.createButton.call(this, 'play-large'));\n }\n\n this.elements.controls = container;\n\n if (this.config.controls.includes('settings') && this.config.settings.includes('speed')) {\n controls.setSpeedMenu.call(this);\n }\n\n return container;\n },\n\n // Insert controls\n inject() {\n // Sprite\n if (this.config.loadSprite) {\n const icon = controls.getIconUrl.call(this);\n\n // Only load external sprite using AJAX\n if (icon.absolute) {\n utils.loadSprite(icon.url, 'sprite-plyr');\n }\n }\n\n // Create a unique ID\n this.id = Math.floor(Math.random() * 10000);\n\n // Null by default\n let container = null;\n\n // HTML passed as the option\n if (utils.is.string(this.config.controls)) {\n container = this.config.controls;\n } else if (utils.is.function(this.config.controls)) {\n // A custom function to build controls\n // The function can return a HTMLElement or String\n container = this.config.controls({\n id: this.id,\n seektime: this.config.seekTime,\n title: this.config.title,\n });\n } else {\n // Create controls\n container = controls.create.call(this, {\n id: this.id,\n seektime: this.config.seekTime,\n speed: this.speed,\n quality: this.quality,\n captions: controls.getLanguage.call(this),\n // TODO: Looping\n // loop: 'None',\n });\n }\n\n // Controls container\n let target;\n\n // Inject to custom location\n if (utils.is.string(this.config.selectors.controls.container)) {\n target = document.querySelector(this.config.selectors.controls.container);\n }\n\n // Inject into the container by default\n if (!utils.is.element(target)) {\n target = this.elements.container;\n }\n\n // Inject controls HTML\n if (utils.is.element(container)) {\n target.appendChild(container);\n } else {\n target.insertAdjacentHTML('beforeend', container);\n }\n\n // Find the elements if need be\n if (utils.is.element(this.elements.controls)) {\n utils.findElements.call(this);\n }\n\n // Edge sometimes doesn't finish the paint so force a redraw\n if (window.navigator.userAgent.includes('Edge')) {\n utils.repaint(target);\n }\n\n // Setup tooltips\n if (this.config.tooltips.controls) {\n const labels = utils.getElements.call(\n this,\n [\n this.config.selectors.controls.wrapper,\n ' ',\n this.config.selectors.labels,\n ' .',\n this.config.classNames.hidden,\n ].join('')\n );\n\n Array.from(labels).forEach(label => {\n utils.toggleClass(label, this.config.classNames.hidden, false);\n utils.toggleClass(label, this.config.classNames.tooltip, true);\n label.setAttribute('role', 'tooltip');\n });\n }\n },\n};\n\nexport default controls;\n","// ==========================================================================\n// Plyr Captions\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport controls from './controls';\n\nconst captions = {\n // Setup captions\n setup() {\n // Requires UI support\n if (!this.supported.ui) {\n return;\n }\n\n // Set default language if not set\n const stored = this.storage.get('language');\n\n if (!utils.is.empty(stored)) {\n this.captions.language = stored;\n }\n\n if (utils.is.empty(this.captions.language)) {\n this.captions.language = this.config.captions.language.toLowerCase();\n }\n\n // Set captions enabled state if not set\n if (!utils.is.boolean(this.captions.active)) {\n const active = this.storage.get('captions');\n\n if (utils.is.boolean(active)) {\n this.captions.active = active;\n } else {\n this.captions.active = this.config.captions.active;\n }\n }\n\n // Only Vimeo and HTML5 video supported at this point\n if (!this.isVideo || this.isYouTube || (this.isHTML5 && !support.textTracks)) {\n // Clear menu and hide\n if (this.config.controls.includes('settings') && this.config.settings.includes('captions')) {\n controls.setCaptionsMenu.call(this);\n }\n\n return;\n }\n\n // Inject the container\n if (!utils.is.element(this.elements.captions)) {\n this.elements.captions = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.captions));\n\n utils.insertAfter(this.elements.captions, this.elements.wrapper);\n }\n\n // Set the class hook\n utils.toggleClass(this.elements.container, this.config.classNames.captions.enabled, !utils.is.empty(captions.getTracks.call(this)));\n\n // If no caption file exists, hide container for caption text\n if (utils.is.empty(captions.getTracks.call(this))) {\n return;\n }\n\n // Set language\n captions.setLanguage.call(this);\n\n // Enable UI\n captions.show.call(this);\n\n // Set available languages in list\n if (this.config.controls.includes('settings') && this.config.settings.includes('captions')) {\n controls.setCaptionsMenu.call(this);\n }\n },\n\n // Set the captions language\n setLanguage() {\n // Setup HTML5 track rendering\n if (this.isHTML5 && this.isVideo) {\n captions.getTracks.call(this).forEach(track => {\n // Remove previous bindings\n utils.on(track, 'cuechange', event => captions.setCue.call(this, event));\n\n // Turn off native caption rendering to avoid double captions\n // eslint-disable-next-line\n track.mode = 'hidden';\n });\n\n // Get current track\n const currentTrack = captions.getCurrentTrack.call(this);\n\n // Check if suported kind\n if (utils.is.track(currentTrack)) {\n // If we change the active track while a cue is already displayed we need to update it\n if (Array.from(currentTrack.activeCues || []).length) {\n captions.setCue.call(this, currentTrack);\n }\n }\n } else if (this.isVimeo && this.captions.active) {\n this.embed.enableTextTrack(this.language);\n }\n },\n\n // Get the tracks\n getTracks() {\n // Return empty array at least\n if (utils.is.nullOrUndefined(this.media)) {\n return [];\n }\n\n // Only get accepted kinds\n return Array.from(this.media.textTracks || []).filter(track => [\n 'captions',\n 'subtitles',\n ].includes(track.kind));\n },\n\n // Get the current track for the current language\n getCurrentTrack() {\n return captions.getTracks.call(this).find(track => track.language.toLowerCase() === this.language);\n },\n\n // Display active caption if it contains text\n setCue(input) {\n // Get the track from the event if needed\n const track = utils.is.event(input) ? input.target : input;\n const active = track.activeCues[0];\n const currentTrack = captions.getCurrentTrack.call(this);\n\n // Only display current track\n if (track !== currentTrack) {\n return;\n }\n\n // Display a cue, if there is one\n if (utils.is.cue(active)) {\n captions.setText.call(this, active.getCueAsHTML());\n } else {\n captions.setText.call(this, null);\n }\n\n utils.dispatchEvent.call(this, this.media, 'cuechange');\n },\n\n // Set the current caption\n setText(input) {\n // Requires UI\n if (!this.supported.ui) {\n return;\n }\n\n if (utils.is.element(this.elements.captions)) {\n const content = utils.createElement('span');\n\n // Empty the container\n utils.emptyElement(this.elements.captions);\n\n // Default to empty\n const caption = !utils.is.nullOrUndefined(input) ? input : '';\n\n // Set the span content\n if (utils.is.string(caption)) {\n content.textContent = caption.trim();\n } else {\n content.appendChild(caption);\n }\n\n // Set new caption text\n this.elements.captions.appendChild(content);\n } else {\n this.debug.warn('No captions element to render to');\n }\n },\n\n // Display captions container and button (for initialization)\n show() {\n // If there's no caption toggle, bail\n if (!utils.is.element(this.elements.buttons.captions)) {\n return;\n }\n\n // Try to load the value from storage\n let active = this.storage.get('captions');\n\n // Otherwise fall back to the default config\n if (!utils.is.boolean(active)) {\n ({ active } = this.config.captions);\n } else {\n this.captions.active = active;\n }\n\n if (active) {\n utils.toggleClass(this.elements.container, this.config.classNames.captions.active, true);\n utils.toggleState(this.elements.buttons.captions, true);\n }\n },\n};\n\nexport default captions;\n","// ==========================================================================\n// YouTube plugin\n// ==========================================================================\n\nimport utils from './../utils';\nimport controls from './../controls';\nimport ui from './../ui';\n\nconst youtube = {\n setup() {\n const videoId = utils.parseYouTubeId(this.embedId);\n\n // Remove old containers\n const containers = utils.getElements.call(this, `[id^=\"${this.provider}-\"]`);\n Array.from(containers).forEach(utils.removeElement);\n\n // Add embed class for responsive\n utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true);\n\n // Set aspect ratio\n youtube.setAspectRatio.call(this);\n\n // Set ID\n this.media.setAttribute('id', utils.generateId(this.provider));\n\n // Setup API\n if (utils.is.object(window.YT)) {\n youtube.ready.call(this, videoId);\n } else {\n // Load the API\n utils.loadScript(this.config.urls.youtube.api);\n\n // Setup callback for the API\n // YouTube has it's own system of course...\n window.onYouTubeReadyCallbacks = window.onYouTubeReadyCallbacks || [];\n\n // Add to queue\n window.onYouTubeReadyCallbacks.push(() => {\n youtube.ready.call(this, videoId);\n });\n\n // Set callback to process queue\n window.onYouTubeIframeAPIReady = () => {\n window.onYouTubeReadyCallbacks.forEach(callback => {\n callback();\n });\n };\n }\n },\n\n // Get the media title\n getTitle() {\n // Try via undocumented API method first\n // This method disappears now and then though...\n // https://github.com/sampotts/plyr/issues/709\n if (utils.is.function(this.embed.getVideoData)) {\n const { title } = this.embed.getVideoData();\n\n if (utils.is.empty(title)) {\n this.config.title = title;\n ui.setTitle.call(this);\n return;\n }\n }\n\n // Or via Google API\n const key = this.config.keys.google;\n const videoId = utils.parseYouTubeId(this.embedId);\n if (utils.is.string(key) && !utils.is.empty(key)) {\n const url = `https://www.googleapis.com/youtube/v3/videos?id=${videoId}&key=${key}&fields=items(snippet(title))&part=snippet`;\n\n fetch(url)\n .then(response => (response.ok ? response.json() : null))\n .then(result => {\n if (result !== null && utils.is.object(result)) {\n this.config.title = result.items[0].snippet.title;\n ui.setTitle.call(this);\n }\n })\n .catch(() => {});\n }\n },\n\n // Set aspect ratio\n setAspectRatio() {\n const ratio = this.config.ratio.split(':');\n this.elements.wrapper.style.paddingBottom = `${100 / ratio[0] * ratio[1]}%`;\n },\n\n // API ready\n ready(videoId) {\n const player = this;\n\n // Setup instance\n // https://developers.google.com/youtube/iframe_api_reference\n player.embed = new window.YT.Player(player.media.id, {\n videoId,\n playerVars: {\n autoplay: player.config.autoplay ? 1 : 0, // Autoplay\n controls: player.supported.ui ? 0 : 1, // Only show controls if not fully supported\n rel: 0, // No related vids\n showinfo: 0, // Hide info\n iv_load_policy: 3, // Hide annotations\n modestbranding: 1, // Hide logos as much as possible (they still show one in the corner when paused)\n disablekb: 1, // Disable keyboard as we handle it\n playsinline: 1, // Allow iOS inline playback\n\n // Tracking for stats\n origin: window && window.location.hostname,\n widget_referrer: window && window.location.href,\n\n // Captions are flaky on YouTube\n cc_load_policy: this.captions.active ? 1 : 0,\n cc_lang_pref: this.config.captions.language,\n },\n events: {\n onError(event) {\n // If we've already fired an error, don't do it again\n // YouTube fires onError twice\n if (utils.is.object(player.media.error)) {\n return;\n }\n\n const detail = {\n code: event.data,\n };\n\n // Messages copied from https://developers.google.com/youtube/iframe_api_reference#onError\n switch (event.data) {\n case 2:\n detail.message =\n 'The request contains an invalid parameter value. For example, this error occurs if you specify a video ID that does not have 11 characters, or if the video ID contains invalid characters, such as exclamation points or asterisks.';\n break;\n\n case 5:\n detail.message =\n 'The requested content cannot be played in an HTML5 player or another error related to the HTML5 player has occurred.';\n break;\n\n case 100:\n detail.message =\n 'The video requested was not found. This error occurs when a video has been removed (for any reason) or has been marked as private.';\n break;\n\n case 101:\n case 150:\n detail.message = 'The owner of the requested video does not allow it to be played in embedded players.';\n break;\n\n default:\n detail.message = 'An unknown error occured';\n break;\n }\n\n player.media.error = detail;\n\n utils.dispatchEvent.call(player, player.media, 'error');\n },\n onPlaybackQualityChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Get current quality\n player.media.quality = instance.getPlaybackQuality();\n\n utils.dispatchEvent.call(player, player.media, 'qualitychange');\n },\n onPlaybackRateChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Get current speed\n player.media.playbackRate = instance.getPlaybackRate();\n\n utils.dispatchEvent.call(player, player.media, 'ratechange');\n },\n onReady(event) {\n // Get the instance\n const instance = event.target;\n\n // Get the title\n youtube.getTitle.call(player);\n\n // Create a faux HTML5 API using the YouTube API\n player.media.play = () => {\n instance.playVideo();\n player.media.paused = false;\n };\n\n player.media.pause = () => {\n instance.pauseVideo();\n player.media.paused = true;\n };\n\n player.media.stop = () => {\n instance.stopVideo();\n player.media.paused = true;\n };\n\n player.media.duration = instance.getDuration();\n player.media.paused = true;\n\n // Seeking\n player.media.currentTime = 0;\n Object.defineProperty(player.media, 'currentTime', {\n get() {\n return Number(instance.getCurrentTime());\n },\n set(time) {\n // Set seeking flag\n player.media.seeking = true;\n\n // Trigger seeking\n utils.dispatchEvent.call(player, player.media, 'seeking');\n\n // Seek after events sent\n instance.seekTo(time);\n },\n });\n\n // Playback speed\n Object.defineProperty(player.media, 'playbackRate', {\n get() {\n return instance.getPlaybackRate();\n },\n set(input) {\n instance.setPlaybackRate(input);\n },\n });\n\n // Quality\n Object.defineProperty(player.media, 'quality', {\n get() {\n return instance.getPlaybackQuality();\n },\n set(input) {\n // Trigger request event\n utils.dispatchEvent.call(player, player.media, 'qualityrequested', false, {\n quality: input,\n });\n\n instance.setPlaybackQuality(input);\n },\n });\n\n // Volume\n let { volume } = player.config;\n Object.defineProperty(player.media, 'volume', {\n get() {\n return volume;\n },\n set(input) {\n volume = input;\n instance.setVolume(volume * 100);\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n },\n });\n\n // Muted\n let { muted } = player.config;\n Object.defineProperty(player.media, 'muted', {\n get() {\n return muted;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : muted;\n muted = toggle;\n instance[toggle ? 'mute' : 'unMute']();\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n },\n });\n\n // Source\n Object.defineProperty(player.media, 'currentSrc', {\n get() {\n return instance.getVideoUrl();\n },\n });\n\n // Ended\n Object.defineProperty(player.media, 'ended', {\n get() {\n return player.currentTime === player.duration;\n },\n });\n\n // Get available speeds\n player.options.speed = instance.getAvailablePlaybackRates();\n\n // Set the tabindex to avoid focus entering iframe\n if (player.supported.ui) {\n player.media.setAttribute('tabindex', -1);\n }\n\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n\n // Reset timer\n window.clearInterval(player.timers.buffering);\n\n // Setup buffering\n player.timers.buffering = window.setInterval(() => {\n // Get loaded % from YouTube\n player.media.buffered = instance.getVideoLoadedFraction();\n\n // Trigger progress only when we actually buffer something\n if (player.media.lastBuffered === null || player.media.lastBuffered < player.media.buffered) {\n utils.dispatchEvent.call(player, player.media, 'progress');\n }\n\n // Set last buffer point\n player.media.lastBuffered = player.media.buffered;\n\n // Bail if we're at 100%\n if (player.media.buffered === 1) {\n window.clearInterval(player.timers.buffering);\n\n // Trigger event\n utils.dispatchEvent.call(player, player.media, 'canplaythrough');\n }\n }, 200);\n\n // Rebuild UI\n window.setTimeout(() => ui.build.call(player), 50);\n },\n onStateChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Reset timer\n window.clearInterval(player.timers.playing);\n\n // Handle events\n // -1 Unstarted\n // 0 Ended\n // 1 Playing\n // 2 Paused\n // 3 Buffering\n // 5 Video cued\n switch (event.data) {\n case 0:\n player.media.paused = true;\n\n // YouTube doesn't support loop for a single video, so mimick it.\n if (player.media.loop) {\n // YouTube needs a call to `stopVideo` before playing again\n instance.stopVideo();\n instance.playVideo();\n } else {\n utils.dispatchEvent.call(player, player.media, 'ended');\n }\n\n break;\n\n case 1:\n // If we were seeking, fire seeked event\n if (player.media.seeking) {\n utils.dispatchEvent.call(player, player.media, 'seeked');\n }\n player.media.seeking = false;\n\n // Only fire play if paused before\n if (player.media.paused) {\n utils.dispatchEvent.call(player, player.media, 'play');\n }\n player.media.paused = false;\n\n utils.dispatchEvent.call(player, player.media, 'playing');\n\n // Poll to get playback progress\n player.timers.playing = window.setInterval(() => {\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n }, 50);\n\n // Check duration again due to YouTube bug\n // https://github.com/sampotts/plyr/issues/374\n // https://code.google.com/p/gdata-issues/issues/detail?id=8690\n if (player.media.duration !== instance.getDuration()) {\n player.media.duration = instance.getDuration();\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n }\n\n // Get quality\n controls.setQualityMenu.call(player, instance.getAvailableQualityLevels());\n\n break;\n\n case 2:\n player.media.paused = true;\n\n utils.dispatchEvent.call(player, player.media, 'pause');\n\n break;\n\n default:\n break;\n }\n\n utils.dispatchEvent.call(player, player.elements.container, 'statechange', false, {\n code: event.data,\n });\n },\n },\n });\n },\n};\n\nexport default youtube;\n","// ==========================================================================\n// Vimeo plugin\n// ==========================================================================\n\nimport utils from './../utils';\nimport captions from './../captions';\nimport ui from './../ui';\n\nconst vimeo = {\n setup() {\n // Remove old containers\n const containers = utils.getElements.call(this, `[id^=\"${this.provider}-\"]`);\n Array.from(containers).forEach(utils.removeElement);\n\n // Add embed class for responsive\n utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true);\n\n // Set intial ratio\n vimeo.setAspectRatio.call(this);\n\n // Set ID\n this.media.setAttribute('id', utils.generateId(this.provider));\n\n // Load the API if not already\n if (!utils.is.object(window.Vimeo)) {\n utils.loadScript(this.config.urls.vimeo.api, () => {\n vimeo.ready.call(this);\n });\n } else {\n vimeo.ready.call(this);\n }\n },\n\n // Set aspect ratio\n // For Vimeo we have an extra 300% height <div> to hide the standard controls and UI\n setAspectRatio(input) {\n const ratio = utils.is.string(input) ? input.split(':') : this.config.ratio.split(':');\n const padding = 100 / ratio[0] * ratio[1];\n const height = 200;\n const offset = (height - padding) / (height / 50);\n this.elements.wrapper.style.paddingBottom = `${padding}%`;\n this.media.style.transform = `translateY(-${offset}%)`;\n },\n\n // API Ready\n ready() {\n const player = this;\n\n // Get Vimeo params for the iframe\n const options = {\n loop: player.config.loop.active,\n autoplay: player.autoplay,\n byline: false,\n portrait: false,\n title: false,\n speed: true,\n transparent: 0,\n gesture: 'media',\n };\n const params = utils.buildUrlParameters(options);\n const id = utils.parseVimeoId(player.embedId);\n\n // Build an iframe\n const iframe = utils.createElement('iframe');\n const src = `https://player.vimeo.com/video/${id}?${params}`;\n iframe.setAttribute('src', src);\n iframe.setAttribute('allowfullscreen', '');\n player.media.appendChild(iframe);\n\n // Setup instance\n // https://github.com/vimeo/player.js\n player.embed = new window.Vimeo.Player(iframe);\n\n player.media.paused = true;\n player.media.currentTime = 0;\n\n // Create a faux HTML5 API using the Vimeo API\n player.media.play = () => {\n player.embed.play().then(() => {\n player.media.paused = false;\n });\n };\n\n player.media.pause = () => {\n player.embed.pause().then(() => {\n player.media.paused = true;\n });\n };\n\n player.media.stop = () => {\n player.embed.stop().then(() => {\n player.media.paused = true;\n player.currentTime = 0;\n });\n };\n\n // Seeking\n let { currentTime } = player.media;\n Object.defineProperty(player.media, 'currentTime', {\n get() {\n return currentTime;\n },\n set(time) {\n // Get current paused state\n // Vimeo will automatically play on seek\n const { paused } = player.media;\n\n // Set seeking flag\n player.media.seeking = true;\n\n // Trigger seeking\n utils.dispatchEvent.call(player, player.media, 'seeking');\n\n // Seek after events\n player.embed.setCurrentTime(time);\n\n // Restore pause state\n if (paused) {\n player.pause();\n }\n },\n });\n\n // Playback speed\n let speed = player.config.speed.selected;\n Object.defineProperty(player.media, 'playbackRate', {\n get() {\n return speed;\n },\n set(input) {\n player.embed.setPlaybackRate(input).then(() => {\n speed = input;\n utils.dispatchEvent.call(player, player.media, 'ratechange');\n });\n },\n });\n\n // Volume\n let { volume } = player.config;\n Object.defineProperty(player.media, 'volume', {\n get() {\n return volume;\n },\n set(input) {\n player.embed.setVolume(input).then(() => {\n volume = input;\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n });\n },\n });\n\n // Muted\n let { muted } = player.config;\n Object.defineProperty(player.media, 'muted', {\n get() {\n return muted;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : false;\n\n player.embed.setVolume(toggle ? 0 : player.config.volume).then(() => {\n muted = toggle;\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n });\n },\n });\n\n // Loop\n let { loop } = player.config;\n Object.defineProperty(player.media, 'loop', {\n get() {\n return loop;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : player.config.loop.active;\n\n player.embed.setLoop(toggle).then(() => {\n loop = toggle;\n });\n },\n });\n\n // Source\n let currentSrc;\n player.embed.getVideoUrl().then(value => {\n currentSrc = value;\n });\n Object.defineProperty(player.media, 'currentSrc', {\n get() {\n return currentSrc;\n },\n });\n\n // Ended\n Object.defineProperty(player.media, 'ended', {\n get() {\n return player.currentTime === player.duration;\n },\n });\n\n // Set aspect ratio based on video size\n Promise.all([\n player.embed.getVideoWidth(),\n player.embed.getVideoHeight(),\n ]).then(dimensions => {\n const ratio = utils.getAspectRatio(dimensions[0], dimensions[1]);\n vimeo.setAspectRatio.call(this, ratio);\n });\n\n // Set autopause\n player.embed.setAutopause(player.config.autopause).then(state => {\n player.config.autopause = state;\n });\n\n // Get title\n player.embed.getVideoTitle().then(title => {\n player.config.title = title;\n ui.setTitle.call(this);\n });\n\n // Get current time\n player.embed.getCurrentTime().then(value => {\n currentTime = value;\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n });\n\n // Get duration\n player.embed.getDuration().then(value => {\n player.media.duration = value;\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n });\n\n // Get captions\n player.embed.getTextTracks().then(tracks => {\n player.media.textTracks = tracks;\n captions.setup.call(player);\n });\n\n player.embed.on('cuechange', data => {\n let cue = null;\n\n if (data.cues.length) {\n cue = utils.stripHTML(data.cues[0].text);\n }\n\n captions.setText.call(player, cue);\n });\n\n player.embed.on('loaded', () => {\n if (utils.is.element(player.embed.element) && player.supported.ui) {\n const frame = player.embed.element;\n\n // Fix keyboard focus issues\n // https://github.com/sampotts/plyr/issues/317\n frame.setAttribute('tabindex', -1);\n }\n });\n\n player.embed.on('play', () => {\n // Only fire play if paused before\n if (player.media.paused) {\n utils.dispatchEvent.call(player, player.media, 'play');\n }\n player.media.paused = false;\n utils.dispatchEvent.call(player, player.media, 'playing');\n });\n\n player.embed.on('pause', () => {\n player.media.paused = true;\n utils.dispatchEvent.call(player, player.media, 'pause');\n });\n\n player.embed.on('timeupdate', data => {\n player.media.seeking = false;\n currentTime = data.seconds;\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n });\n\n player.embed.on('progress', data => {\n player.media.buffered = data.percent;\n utils.dispatchEvent.call(player, player.media, 'progress');\n\n // Check all loaded\n if (parseInt(data.percent, 10) === 1) {\n utils.dispatchEvent.call(player, player.media, 'canplaythrough');\n }\n });\n\n player.embed.on('seeked', () => {\n player.media.seeking = false;\n utils.dispatchEvent.call(player, player.media, 'seeked');\n utils.dispatchEvent.call(player, player.media, 'play');\n });\n\n player.embed.on('ended', () => {\n player.media.paused = true;\n utils.dispatchEvent.call(player, player.media, 'ended');\n });\n\n player.embed.on('error', detail => {\n player.media.error = detail;\n utils.dispatchEvent.call(player, player.media, 'error');\n });\n\n // Rebuild UI\n window.setTimeout(() => ui.build.call(player), 0);\n },\n};\n\nexport default vimeo;\n","// ==========================================================================\n// Plyr Media\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport youtube from './plugins/youtube';\nimport vimeo from './plugins/vimeo';\nimport ui from './ui';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst media = {\n // Setup media\n setup() {\n // If there's no media, bail\n if (!this.media) {\n this.debug.warn('No media element found!');\n return;\n }\n\n // Add type class\n utils.toggleClass(this.elements.container, this.config.classNames.type.replace('{0}', this.type), true);\n\n // Add provider class\n utils.toggleClass(this.elements.container, this.config.classNames.provider.replace('{0}', this.provider), true);\n\n // Add video class for embeds\n // This will require changes if audio embeds are added\n if (this.isEmbed) {\n utils.toggleClass(this.elements.container, this.config.classNames.type.replace('{0}', 'video'), true);\n }\n\n if (this.supported.ui) {\n // Check for picture-in-picture support\n utils.toggleClass(this.elements.container, this.config.classNames.pip.supported, support.pip && this.isHTML5 && this.isVideo);\n\n // Check for airplay support\n utils.toggleClass(this.elements.container, this.config.classNames.airplay.supported, support.airplay && this.isHTML5);\n\n // If there's no autoplay attribute, assume the video is stopped and add state class\n utils.toggleClass(this.elements.container, this.config.classNames.stopped, this.config.autoplay);\n\n // Add iOS class\n utils.toggleClass(this.elements.container, this.config.classNames.isIos, browser.isIos);\n\n // Add touch class\n utils.toggleClass(this.elements.container, this.config.classNames.isTouch, support.touch);\n }\n\n // Inject the player wrapper\n if (this.isVideo || this.isYouTube || this.isVimeo) {\n // Create the wrapper div\n this.elements.wrapper = utils.createElement('div', {\n class: this.config.classNames.video,\n });\n\n // Wrap the video in a container\n utils.wrap(this.media, this.elements.wrapper);\n }\n\n if (this.isEmbed) {\n switch (this.provider) {\n case 'youtube':\n youtube.setup.call(this);\n break;\n\n case 'vimeo':\n vimeo.setup.call(this);\n break;\n\n default:\n break;\n }\n } else if (this.isHTML5) {\n ui.setTitle.call(this);\n }\n },\n\n // Cancel current network requests\n // See https://github.com/sampotts/plyr/issues/174\n cancelRequests() {\n if (!this.isHTML5) {\n return;\n }\n\n // Remove child sources\n Array.from(this.media.querySelectorAll('source')).forEach(utils.removeElement);\n\n // Set blank video src attribute\n // This is to prevent a MEDIA_ERR_SRC_NOT_SUPPORTED error\n // Info: http://stackoverflow.com/questions/32231579/how-to-properly-dispose-of-an-html5-video-and-close-socket-or-connection\n this.media.setAttribute('src', this.config.blankVideo);\n\n // Load the new empty source\n // This will cancel existing requests\n // See https://github.com/sampotts/plyr/issues/174\n this.media.load();\n\n // Debugging\n this.debug.log('Cancelled network requests');\n },\n};\n\nexport default media;\n","// ==========================================================================\n// Plyr source update\n// ==========================================================================\n\nimport { providers } from './types';\nimport utils from './utils';\nimport media from './media';\nimport ui from './ui';\nimport support from './support';\n\nconst source = {\n // Add elements to HTML5 media (source, tracks, etc)\n insertElements(type, attributes) {\n if (utils.is.string(attributes)) {\n utils.insertElement(type, this.media, {\n src: attributes,\n });\n } else if (utils.is.array(attributes)) {\n attributes.forEach(attribute => {\n utils.insertElement(type, this.media, attribute);\n });\n }\n },\n\n // Update source\n // Sources are not checked for support so be careful\n change(input) {\n if (!utils.is.object(input) || !('sources' in input) || !input.sources.length) {\n this.debug.warn('Invalid source format');\n return;\n }\n\n // Cancel current network requests\n media.cancelRequests.call(this);\n\n // Destroy instance and re-setup\n this.destroy.call(\n this,\n () => {\n // TODO: Reset menus here\n\n // Remove elements\n utils.removeElement(this.media);\n this.media = null;\n\n // Reset class name\n if (utils.is.element(this.elements.container)) {\n this.elements.container.removeAttribute('class');\n }\n\n // Set the type and provider\n this.type = input.type;\n this.provider = !utils.is.empty(input.sources[0].provider) ? input.sources[0].provider : providers.html5;\n\n // Check for support\n this.supported = support.check(this.type, this.provider, this.config.inline);\n\n // Create new markup\n switch (`${this.provider}:${this.type}`) {\n case 'html5:video':\n this.media = utils.createElement('video');\n break;\n\n case 'html5:audio':\n this.media = utils.createElement('audio');\n break;\n\n case 'youtube:video':\n case 'vimeo:video':\n this.media = utils.createElement('div');\n this.embedId = input.sources[0].src;\n break;\n\n default:\n break;\n }\n\n // Inject the new element\n this.elements.container.appendChild(this.media);\n\n // Autoplay the new source?\n if (utils.is.boolean(input.autoplay)) {\n this.config.autoplay = input.autoplay;\n }\n\n // Set attributes for audio and video\n if (this.isHTML5) {\n if (this.config.crossorigin) {\n this.media.setAttribute('crossorigin', '');\n }\n if (this.config.autoplay) {\n this.media.setAttribute('autoplay', '');\n }\n if ('poster' in input) {\n this.media.setAttribute('poster', input.poster);\n }\n if (this.config.loop.active) {\n this.media.setAttribute('loop', '');\n }\n if (this.config.muted) {\n this.media.setAttribute('muted', '');\n }\n if (this.config.inline) {\n this.media.setAttribute('playsinline', '');\n }\n }\n\n // Restore class hook\n ui.addStyleHook.call(this);\n\n // Set new sources for html5\n if (this.isHTML5) {\n source.insertElements.call(this, 'source', input.sources);\n }\n\n // Set video title\n this.config.title = input.title;\n\n // Set up from scratch\n media.setup.call(this);\n\n // HTML5 stuff\n if (this.isHTML5) {\n // Setup captions\n if ('tracks' in input) {\n source.insertElements.call(this, 'track', input.tracks);\n }\n\n // Load HTML5 sources\n this.media.load();\n }\n\n // If HTML5 or embed but not fully supported, setupInterface and call ready now\n if (this.isHTML5 || (this.isEmbed && !this.supported.ui)) {\n // Setup interface\n ui.build.call(this);\n }\n },\n true\n );\n },\n};\n\nexport default source;\n","// ==========================================================================\n// Plyr\n// plyr.js v3.0.0-beta.1\n// https://github.com/sampotts/plyr\n// License: The MIT License (MIT)\n// ==========================================================================\n\nimport { providers, types } from './types';\nimport defaults from './defaults';\nimport support from './support';\nimport utils from './utils';\n\nimport Console from './console';\nimport Storage from './storage';\n\nimport captions from './captions';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport listeners from './listeners';\nimport media from './media';\nimport source from './source';\nimport ui from './ui';\n\n// Private properties\n// TODO: Use a WeakMap for private globals\n// const globals = new WeakMap();\n\n// Globals\nlet scrollPosition = {\n x: 0,\n y: 0,\n};\n\n// Plyr instance\nclass Plyr {\n constructor(target, options) {\n this.timers = {};\n\n // State\n this.ready = false;\n this.loading = false;\n this.failed = false;\n\n // Set the media element\n this.media = target;\n\n // String selector passed\n if (utils.is.string(this.media)) {\n this.media = document.querySelectorAll(this.media);\n }\n\n // jQuery, NodeList or Array passed, use first element\n if ((window.jQuery && this.media instanceof jQuery) || utils.is.nodeList(this.media) || utils.is.array(this.media)) {\n // eslint-disable-next-line\n this.media = this.media[0];\n }\n\n // Set config\n this.config = utils.extend(\n {},\n defaults,\n options,\n (() => {\n try {\n return JSON.parse(this.media.getAttribute('data-plyr-config'));\n } catch (e) {\n return {};\n }\n })()\n );\n\n // Elements cache\n this.elements = {\n container: null,\n buttons: {},\n display: {},\n progress: {},\n inputs: {},\n settings: {\n menu: null,\n panes: {},\n tabs: {},\n },\n captions: null,\n };\n\n // Captions\n this.captions = {\n active: null,\n currentTrack: null,\n };\n\n // Fullscreen\n this.fullscreen = {\n active: false,\n };\n\n // Options\n this.options = {\n speed: [],\n quality: [],\n };\n\n // Debugging\n // TODO: move to globals\n this.debug = new Console(this);\n\n // Log config options and support\n this.debug.log('Config', this.config);\n this.debug.log('Support', support);\n\n // We need an element to setup\n if (utils.is.nullOrUndefined(this.media) || !utils.is.element(this.media)) {\n this.debug.error('Setup failed: no suitable element passed');\n return;\n }\n\n // Bail if the element is initialized\n if (this.media.plyr) {\n this.debug.warn('Target already setup');\n return;\n }\n\n // Bail if not enabled\n if (!this.config.enabled) {\n this.debug.error('Setup failed: disabled by config');\n return;\n }\n\n // Bail if disabled or no basic support\n // You may want to disable certain UAs etc\n if (!support.check().api) {\n this.debug.error('Setup failed: no support');\n return;\n }\n\n // Cache original element state for .destroy()\n this.elements.original = this.media.cloneNode(true);\n\n // Set media type based on tag or data attribute\n // Supported: video, audio, vimeo, youtube\n const type = this.media.tagName.toLowerCase();\n\n // Embed attributes\n const attributes = {\n provider: 'data-plyr-provider',\n id: 'data-plyr-embed-id',\n };\n\n // Different setup based on type\n switch (type) {\n // TODO: Handle passing an iframe for true progressive enhancement\n // case 'iframe':\n case 'div':\n this.type = types.video; // Audio will come later for external providers\n this.provider = this.media.getAttribute(attributes.provider);\n this.embedId = this.media.getAttribute(attributes.id);\n\n if (utils.is.empty(this.provider) || !Object.keys(providers).includes(this.provider)) {\n this.debug.error('Setup failed: Invalid provider');\n return;\n }\n\n // Try and get the embed id\n if (utils.is.empty(this.embedId)) {\n this.debug.error('Setup failed: Embed ID or URL missing');\n return;\n }\n\n // Clean up\n this.media.removeAttribute(attributes.provider);\n this.media.removeAttribute(attributes.id);\n\n break;\n\n case 'video':\n case 'audio':\n this.type = type;\n this.provider = providers.html5;\n\n if (this.media.hasAttribute('crossorigin')) {\n this.config.crossorigin = true;\n }\n\n if (this.media.hasAttribute('autoplay')) {\n this.config.autoplay = true;\n }\n\n if (this.media.hasAttribute('playsinline')) {\n this.config.inline = true;\n }\n\n if (this.media.hasAttribute('muted')) {\n this.config.muted = true;\n }\n\n if (this.media.hasAttribute('loop')) {\n this.config.loop.active = true;\n }\n\n break;\n\n default:\n this.debug.error('Setup failed: unsupported type');\n return;\n }\n\n // Setup local storage for user settings\n this.storage = new Storage(this);\n\n // Check for support again but with type\n this.supported = support.check(this.type, this.provider, this.config.inline);\n\n // If no support for even API, bail\n if (!this.supported.api) {\n this.debug.error('Setup failed: no support');\n return;\n }\n\n // Store reference\n this.media.plyr = this;\n\n // Wrap media\n this.elements.container = utils.createElement('div');\n utils.wrap(this.media, this.elements.container);\n\n // Allow focus to be captured\n this.elements.container.setAttribute('tabindex', 0);\n\n // Global listeners\n listeners.global.call(this);\n\n // Add style hook\n ui.addStyleHook.call(this);\n\n // Setup media\n media.setup.call(this);\n\n // Listen for events if debugging\n if (this.config.debug) {\n utils.on(this.elements.container, this.config.events.join(' '), event => {\n this.debug.log(`event: ${event.type}`);\n });\n }\n\n // Setup interface\n // If embed but not fully supported, build interface now to avoid flash of controls\n if (this.isHTML5 || (this.isEmbed && !this.supported.ui)) {\n ui.build.call(this);\n }\n }\n\n // ---------------------------------------\n // API\n // ---------------------------------------\n\n /**\n * Types and provider helpers\n */\n get isHTML5() {\n return this.provider === providers.html5;\n }\n get isEmbed() {\n return this.isYouTube || this.isVimeo;\n }\n get isYouTube() {\n return this.provider === providers.youtube;\n }\n get isVimeo() {\n return this.provider === providers.vimeo;\n }\n get isVideo() {\n return this.type === types.video;\n }\n get isAudio() {\n return this.type === types.audio;\n }\n\n /**\n * Play the media\n */\n play() {\n return this.media.play();\n }\n\n /**\n * Pause the media\n */\n pause() {\n if (!this.playing) {\n return;\n }\n\n this.media.pause();\n }\n\n /**\n * Get paused state\n */\n get paused() {\n return this.media.paused;\n }\n\n /**\n * Get playing state\n */\n get playing() {\n return !this.paused && !this.ended && (this.isHTML5 ? this.media.readyState > 2 : true);\n }\n\n /**\n * Get ended state\n */\n get ended() {\n return this.media.ended;\n }\n\n /**\n * Toggle playback based on current status\n * @param {boolean} input\n */\n togglePlay(input) {\n // Toggle based on current state if nothing passed\n const toggle = utils.is.boolean(input) ? input : !this.playing;\n\n if (toggle) {\n this.play();\n } else {\n this.pause();\n }\n }\n\n /**\n * Stop playback\n */\n stop() {\n this.restart();\n this.pause();\n }\n\n /**\n * Restart playback\n */\n restart() {\n this.currentTime = 0;\n }\n\n /**\n * Rewind\n * @param {number} seekTime - how far to rewind in seconds. Defaults to the config.seekTime\n */\n rewind(seekTime) {\n this.currentTime = this.currentTime - (utils.is.number(seekTime) ? seekTime : this.config.seekTime);\n }\n\n /**\n * Fast forward\n * @param {number} seekTime - how far to fast forward in seconds. Defaults to the config.seekTime\n */\n forward(seekTime) {\n this.currentTime = this.currentTime + (utils.is.number(seekTime) ? seekTime : this.config.seekTime);\n }\n\n /**\n * Seek to a time\n * @param {number} input - where to seek to in seconds. Defaults to 0 (the start)\n */\n set currentTime(input) {\n let targetTime = 0;\n\n if (utils.is.number(input)) {\n targetTime = input;\n }\n\n // Normalise targetTime\n if (targetTime < 0) {\n targetTime = 0;\n } else if (targetTime > this.duration) {\n targetTime = this.duration;\n }\n\n // Set\n this.media.currentTime = targetTime.toFixed(4);\n\n // Logging\n this.debug.log(`Seeking to ${this.currentTime} seconds`);\n }\n\n /**\n * Get current time\n */\n get currentTime() {\n return Number(this.media.currentTime);\n }\n\n /**\n * Get seeking status\n */\n get seeking() {\n return this.media.seeking;\n }\n\n /**\n * Get the duration of the current media\n */\n get duration() {\n // Faux duration set via config\n const fauxDuration = parseInt(this.config.duration, 10);\n\n // True duration\n const realDuration = Number(this.media.duration);\n\n // If custom duration is funky, use regular duration\n return !Number.isNaN(fauxDuration) ? fauxDuration : realDuration;\n }\n\n /**\n * Set the player volume\n * @param {number} value - must be between 0 and 1. Defaults to the value from local storage and config.volume if not set in storage\n */\n set volume(value) {\n let volume = value;\n const max = 1;\n const min = 0;\n\n if (utils.is.string(volume)) {\n volume = Number(volume);\n }\n\n // Load volume from storage if no value specified\n if (!utils.is.number(volume)) {\n volume = this.storage.get('volume');\n }\n\n // Use config if all else fails\n if (!utils.is.number(volume)) {\n ({ volume } = this.config);\n }\n\n // Maximum is volumeMax\n if (volume > max) {\n volume = max;\n }\n // Minimum is volumeMin\n if (volume < min) {\n volume = min;\n }\n\n // Update config\n this.config.volume = volume;\n\n // Set the player volume\n this.media.volume = volume;\n\n // If muted, and we're increasing volume, reset muted state\n if (this.muted && volume > 0) {\n this.muted = false;\n }\n }\n\n /**\n * Get the current player volume\n */\n get volume() {\n return this.media.volume;\n }\n\n /**\n * Increase volume\n * @param {boolean} step - How much to decrease by (between 0 and 1)\n */\n increaseVolume(step) {\n const volume = this.media.muted ? 0 : this.volume;\n this.volume = volume + (utils.is.number(step) ? step : 1);\n }\n\n /**\n * Decrease volume\n * @param {boolean} step - How much to decrease by (between 0 and 1)\n */\n decreaseVolume(step) {\n const volume = this.media.muted ? 0 : this.volume;\n this.volume = volume - (utils.is.number(step) ? step : 1);\n }\n\n /**\n * Set muted state\n * @param {boolean} mute\n */\n set muted(mute) {\n let toggle = mute;\n\n // Load muted state from storage\n if (!utils.is.boolean(toggle)) {\n toggle = this.storage.get('muted');\n }\n\n // Use config if all else fails\n if (!utils.is.boolean(toggle)) {\n toggle = this.config.muted;\n }\n\n // Update config\n this.config.muted = toggle;\n\n // Set mute on the player\n this.media.muted = toggle;\n }\n\n /**\n * Get current muted state\n */\n get muted() {\n return this.media.muted;\n }\n\n /**\n * Check if the media has audio\n */\n get hasAudio() {\n // Assume yes for all non HTML5 (as we can't tell...)\n if (!this.isHTML5) {\n return true;\n }\n\n // Get audio tracks\n return this.media.mozHasAudio || Boolean(this.media.webkitAudioDecodedByteCount) || Boolean(this.media.audioTracks && this.media.audioTracks.length);\n }\n\n /**\n * Set playback speed\n * @param {decimal} speed - the speed of playback (0.5-2.0)\n */\n set speed(input) {\n let speed = null;\n\n if (utils.is.number(input)) {\n speed = input;\n }\n\n if (!utils.is.number(speed)) {\n speed = this.storage.get('speed');\n }\n\n if (!utils.is.number(speed)) {\n speed = this.config.speed.selected;\n }\n\n // Set min/max\n if (speed < 0.1) {\n speed = 0.1;\n }\n if (speed > 2.0) {\n speed = 2.0;\n }\n\n if (!this.config.speed.options.includes(speed)) {\n this.debug.warn(`Unsupported speed (${speed})`);\n return;\n }\n\n // Update config\n this.config.speed.selected = speed;\n\n // Set media speed\n this.media.playbackRate = speed;\n }\n\n /**\n * Get current playback speed\n */\n get speed() {\n return this.media.playbackRate;\n }\n\n /**\n * Set playback quality\n * Currently YouTube only\n * @param {string} input - Quality level\n */\n set quality(input) {\n let quality = null;\n\n if (utils.is.string(input)) {\n quality = input;\n }\n\n if (!utils.is.string(quality)) {\n quality = this.storage.get('quality');\n }\n\n if (!utils.is.string(quality)) {\n quality = this.config.quality.selected;\n }\n\n if (!this.options.quality.includes(quality)) {\n this.debug.warn(`Unsupported quality option (${quality})`);\n return;\n }\n\n // Update config\n this.config.quality.selected = quality;\n\n // Set quality\n this.media.quality = quality;\n }\n\n /**\n * Get current quality level\n */\n get quality() {\n return this.media.quality;\n }\n\n /**\n * Toggle loop\n * TODO: Finish fancy new logic. Set the indicator on load as user may pass loop as config\n * @param {boolean} input - Whether to loop or not\n */\n set loop(input) {\n const toggle = utils.is.boolean(input) ? input : this.config.loop.active;\n this.config.loop.active = toggle;\n this.media.loop = toggle;\n\n // Set default to be a true toggle\n /* const type = ['start', 'end', 'all', 'none', 'toggle'].includes(input) ? input : 'toggle';\n\n switch (type) {\n case 'start':\n if (this.config.loop.end && this.config.loop.end <= this.currentTime) {\n this.config.loop.end = null;\n }\n this.config.loop.start = this.currentTime;\n // this.config.loop.indicator.start = this.elements.display.played.value;\n break;\n\n case 'end':\n if (this.config.loop.start >= this.currentTime) {\n return this;\n }\n this.config.loop.end = this.currentTime;\n // this.config.loop.indicator.end = this.elements.display.played.value;\n break;\n\n case 'all':\n this.config.loop.start = 0;\n this.config.loop.end = this.duration - 2;\n this.config.loop.indicator.start = 0;\n this.config.loop.indicator.end = 100;\n break;\n\n case 'toggle':\n if (this.config.loop.active) {\n this.config.loop.start = 0;\n this.config.loop.end = null;\n } else {\n this.config.loop.start = 0;\n this.config.loop.end = this.duration - 2;\n }\n break;\n\n default:\n this.config.loop.start = 0;\n this.config.loop.end = null;\n break;\n } */\n }\n\n /**\n * Get current loop state\n */\n get loop() {\n return this.media.loop;\n }\n\n /**\n * Set new media source\n * @param {object} input - The new source object (see docs)\n */\n set source(input) {\n source.change.call(this, input);\n }\n\n /**\n * Get current source\n */\n get source() {\n return this.media.currentSrc;\n }\n\n /**\n * Set the poster image for a HTML5 video\n * @param {input} - the URL for the new poster image\n */\n set poster(input) {\n if (!this.isHTML5 || !this.isVideo) {\n this.debug.warn('Poster can only be set on HTML5 video');\n return;\n }\n\n if (utils.is.string(input)) {\n this.media.setAttribute('poster', input);\n }\n }\n\n /**\n * Get the current poster image\n */\n get poster() {\n if (!this.isHTML5 || !this.isVideo) {\n return null;\n }\n\n return this.media.getAttribute('poster');\n }\n\n /**\n * Set the autoplay state\n * @param {boolean} input - Whether to autoplay or not\n */\n set autoplay(input) {\n const toggle = utils.is.boolean(input) ? input : this.config.autoplay;\n this.config.autoplay = toggle;\n }\n\n /**\n * Get the current autoplay state\n */\n get autoplay() {\n return this.config.autoplay;\n }\n\n /**\n * Toggle captions\n * @param {boolean} input - Whether to enable captions\n */\n toggleCaptions(input) {\n // If there's no full support, or there's no caption toggle\n if (!this.supported.ui || !utils.is.element(this.elements.buttons.captions)) {\n return;\n }\n\n // If the method is called without parameter, toggle based on current value\n const show = utils.is.boolean(input) ? input : this.elements.container.className.indexOf(this.config.classNames.captions.active) === -1;\n\n // Nothing to change...\n if (this.captions.active === show) {\n return;\n }\n\n // Set global\n this.captions.active = show;\n\n // Toggle state\n utils.toggleState(this.elements.buttons.captions, this.captions.active);\n\n // Add class hook\n utils.toggleClass(this.elements.container, this.config.classNames.captions.active, this.captions.active);\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, this.captions.active ? 'captionsenabled' : 'captionsdisabled');\n }\n\n /**\n * Set the captions language\n * @param {string} - Two character ISO language code (e.g. EN, FR, PT, etc)\n */\n set language(input) {\n // Nothing specified\n if (!utils.is.string(input)) {\n return;\n }\n\n // Toggle captions based on input\n this.toggleCaptions(!utils.is.empty(input));\n\n // If empty string is passed, assume disable captions\n if (utils.is.empty(input)) {\n return;\n }\n\n // Normalize\n const language = input.toLowerCase();\n\n // If nothing to change, bail\n if (this.language === language) {\n return;\n }\n\n // Update config\n this.captions.language = language;\n\n // Clear caption\n captions.setText.call(this, null);\n\n // Update captions\n captions.setLanguage.call(this);\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, 'languagechange');\n }\n\n /**\n * Get the current captions language\n */\n get language() {\n return this.captions.language;\n }\n\n /**\n * Toggle fullscreen playback\n * Requires user input event\n * @param {event} event\n */\n toggleFullscreen(event) {\n // Video only\n if (this.isAudio) {\n return;\n }\n\n // Check for native support\n if (fullscreen.enabled) {\n if (utils.is.event(event) && event.type === fullscreen.eventType) {\n // If it's a fullscreen change event, update the state\n this.fullscreen.active = fullscreen.isFullScreen(this.elements.container);\n } else {\n // Else it's a user request to enter or exit\n if (!this.fullscreen.active) {\n fullscreen.requestFullScreen(this.elements.container);\n } else {\n fullscreen.cancelFullScreen();\n }\n\n return;\n }\n } else {\n // Otherwise, it's a simple toggle\n this.fullscreen.active = !this.fullscreen.active;\n\n // Add class hook\n utils.toggleClass(this.elements.container, this.config.classNames.fullscreen.fallback, this.fullscreen.active);\n\n // Make sure we don't lose scroll position\n if (this.fullscreen.active) {\n scrollPosition = {\n x: window.pageXOffset || 0,\n y: window.pageYOffset || 0,\n };\n } else {\n window.scrollTo(scrollPosition.x, scrollPosition.y);\n }\n\n // Bind/unbind escape key\n document.body.style.overflow = this.fullscreen.active ? 'hidden' : '';\n }\n\n // Set button state\n if (utils.is.element(this.elements.buttons.fullscreen)) {\n utils.toggleState(this.elements.buttons.fullscreen, this.fullscreen.active);\n }\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, this.fullscreen.active ? 'enterfullscreen' : 'exitfullscreen');\n }\n\n /**\n * Toggle picture-in-picture playback on WebKit/MacOS\n * TODO: update player with state, support, enabled\n * TODO: detect outside changes\n */\n set pip(input) {\n const states = {\n pip: 'picture-in-picture',\n inline: 'inline',\n };\n\n // Bail if no support\n if (!support.pip) {\n return;\n }\n\n // Toggle based on current state if not passed\n const toggle = utils.is.boolean(input) ? input : this.pip === states.inline;\n\n // Toggle based on current state\n this.media.webkitSetPresentationMode(toggle ? states.pip : states.inline);\n }\n\n /**\n * Get the current picture-in-picture state\n */\n get pip() {\n if (!support.pip) {\n return null;\n }\n\n return this.media.webkitPresentationMode;\n }\n\n /**\n * Trigger the airplay dialog\n * TODO: update player with state, support, enabled\n */\n airplay() {\n // Show dialog if supported\n if (support.airplay) {\n this.media.webkitShowPlaybackTargetPicker();\n }\n }\n\n /**\n * Toggle the player controls\n * @param {boolean} toggle - Whether to show the controls\n */\n toggleControls(toggle) {\n // We need controls of course...\n if (!utils.is.element(this.elements.controls)) {\n return;\n }\n\n // Don't hide if no UI support or it's audio\n if (!this.supported.ui || this.isAudio) {\n return;\n }\n\n let delay = 0;\n let show = toggle;\n let isEnterFullscreen = false;\n\n // Get toggle state if not set\n if (!utils.is.boolean(toggle)) {\n if (utils.is.event(toggle)) {\n // Is the enter fullscreen event\n isEnterFullscreen = toggle.type === 'enterfullscreen';\n\n // Whether to show controls\n show = [\n 'mouseenter',\n 'mousemove',\n 'touchstart',\n 'touchmove',\n 'focusin',\n ].includes(toggle.type);\n\n // Delay hiding on move events\n if ([\n 'mousemove',\n 'touchmove',\n 'touchend',\n ].includes(toggle.type)) {\n delay = 2000;\n }\n\n // Delay a little more for keyboard users\n if (toggle.type === 'focusin') {\n delay = 3000;\n utils.toggleClass(this.elements.controls, this.config.classNames.noTransition, true);\n }\n } else {\n show = utils.hasClass(this.elements.container, this.config.classNames.hideControls);\n }\n }\n\n // Clear timer on every call\n window.clearTimeout(this.timers.controls);\n\n // If the mouse is not over the controls, set a timeout to hide them\n if (show || this.paused || this.loading) {\n // Check if controls toggled\n const toggled = utils.toggleClass(this.elements.container, this.config.classNames.hideControls, false);\n\n // Trigger event\n if (toggled) {\n utils.dispatchEvent.call(this, this.media, 'controlsshown');\n }\n\n // Always show controls when paused or if touch\n if (this.paused || this.loading) {\n return;\n }\n\n // Delay for hiding on touch\n if (support.touch) {\n delay = 3000;\n }\n }\n\n // If toggle is false or if we're playing (regardless of toggle),\n // then set the timer to hide the controls\n if (!show || this.playing) {\n this.timers.controls = window.setTimeout(() => {\n /* this.debug.warn({\n pressed: this.elements.controls.pressed,\n hover: this.elements.controls.pressed,\n playing: this.playing,\n paused: this.paused,\n loading: this.loading,\n }); */\n\n // If the mouse is over the controls (and not entering fullscreen), bail\n if ((this.elements.controls.pressed || this.elements.controls.hover) && !isEnterFullscreen) {\n return;\n }\n\n // Restore transition behaviour\n if (!utils.hasClass(this.elements.container, this.config.classNames.hideControls)) {\n utils.toggleClass(this.elements.controls, this.config.classNames.noTransition, false);\n }\n\n // Check if controls toggled\n const toggled = utils.toggleClass(this.elements.container, this.config.classNames.hideControls, true);\n\n // Trigger event and close menu\n if (toggled) {\n utils.dispatchEvent.call(this, this.media, 'controlshidden');\n\n if (this.config.controls.includes('settings') && !utils.is.empty(this.config.settings)) {\n controls.toggleMenu.call(this, false);\n }\n }\n }, delay);\n }\n }\n\n /**\n * Add event listeners\n * @param {string} event - Event type\n * @param {function} callback - Callback for when event occurs\n */\n on(event, callback) {\n utils.on(this.elements.container, event, callback);\n }\n\n /**\n * Remove event listeners\n * @param {string} event - Event type\n * @param {function} callback - Callback for when event occurs\n */\n off(event, callback) {\n utils.off(this.elements.container, event, callback);\n }\n\n /**\n * Destroy an instance\n * Event listeners are removed when elements are removed\n * http://stackoverflow.com/questions/12528049/if-a-dom-element-is-removed-are-its-listeners-also-removed-from-memory\n * @param {function} callback - Callback for when destroy is complete\n * @param {boolean} soft - Whether it's a soft destroy (for source changes etc)\n */\n destroy(callback, soft = false) {\n const done = () => {\n // Reset overflow (incase destroyed while in fullscreen)\n document.body.style.overflow = '';\n\n // GC for embed\n this.embed = null;\n this.embedId = null;\n\n // If it's a soft destroy, make minimal changes\n if (soft) {\n if (Object.keys(this.elements).length) {\n // Remove buttons\n if (this.elements.buttons && this.elements.buttons.play) {\n Array.from(this.elements.buttons.play).forEach(button => utils.removeElement(button));\n }\n\n // Remove others\n utils.removeElement(this.elements.captions);\n utils.removeElement(this.elements.controls);\n utils.removeElement(this.elements.wrapper);\n\n // Clear for GC\n this.elements.buttons.play = null;\n this.elements.captions = null;\n this.elements.controls = null;\n this.elements.wrapper = null;\n }\n\n // Callback\n if (utils.is.function(callback)) {\n callback();\n }\n } else {\n // Replace the container with the original element provided\n const parent = this.elements.container.parentNode;\n\n if (utils.is.element(parent)) {\n parent.replaceChild(this.elements.original, this.elements.container);\n }\n\n // Event\n utils.dispatchEvent.call(this, this.elements.original, 'destroyed', true);\n\n // Callback\n if (utils.is.function(callback)) {\n callback.call(this.elements.original);\n }\n\n // Clear for GC\n this.elements = null;\n }\n };\n\n // Type specific stuff\n switch (`${this.provider}:${this.type}`) {\n case 'html5:video':\n case 'html5:audio':\n // Restore native video controls\n ui.toggleNativeControls.call(this, true);\n\n // Clean up\n done();\n\n break;\n\n case 'youtube:video':\n // Clear timers\n window.clearInterval(this.timers.buffering);\n window.clearInterval(this.timers.playing);\n\n // Destroy YouTube API\n this.embed.destroy();\n\n // Clean up\n done();\n\n break;\n\n case 'vimeo:video':\n // Destroy Vimeo API\n // then clean up (wait, to prevent postmessage errors)\n this.embed.unload().then(done);\n\n // Vimeo does not always return\n window.setTimeout(done, 200);\n\n break;\n\n default:\n break;\n }\n }\n\n /**\n * Check for support for a mime type (HTML5 only)\n * @param {string} type - Mime type\n */\n supports(type) {\n return support.mime.call(this, type);\n }\n\n /**\n * Check for support\n * @param {string} type - Player type (audio/video)\n * @param {string} provider - Provider (html5/youtube/vimeo)\n * @param {bool} inline - Where player has `playsinline` sttribute\n */\n static supported(type, provider, inline) {\n return support.check(type, provider, inline);\n }\n\n /**\n * Load an SVG sprite into the page\n * @param {string} url - URL for the SVG sprite\n * @param {string} [id] - Unique ID\n */\n static loadSprite(url, id) {\n return utils.loadSprite(url, id);\n }\n}\n\nexport default Plyr;\n"],"names":["element","type","range","value","providers","types","defaults","window","navigator","language","split","utils","input","this","instanceof","Plyr","getConstructor","Object","Number","isNaN","String","Boolean","Function","nullOrUndefined","Array","isArray","WeakMap","NodeList","Element","Text","Event","TextTrackCue","VTTCue","TextTrack","string","kind","array","nodeList","length","object","keys","constructor","document","documentMode","documentElement","style","test","userAgent","platform","url","callback","current","querySelector","callbacks","push","createElement","is","function","addEventListener","forEach","cb","call","event","src","first","getElementsByTagName","parentNode","insertBefore","id","hasId","querySelectorAll","container","toggleHidden","setAttribute","support","storage","cached","localStorage","getItem","data","JSON","parse","content","then","response","ok","text","setItem","stringify","catch","updateSprite","innerHTML","body","childNodes","prefix","Math","floor","random","self","top","e","elements","wrapper","targets","from","reverse","index","child","cloneNode","parent","sibling","nextSibling","appendChild","attributes","setAttributes","textContent","target","removeChild","lastChild","empty","key","sel","existingAttributes","existing","selector","s","trim","className","replace","parts","charAt","class","toggle","contains","classList","removeAttribute","prototype","matches","webkitMatchesSelector","mozMatchesSelector","msMatchesSelector","includes","controls","getElement","config","selectors","buttons","getElements","play","pause","restart","rewind","forward","mute","pip","airplay","settings","captions","fullscreen","progress","inputs","seek","volume","display","buffer","duration","currentTime","seekTooltip","classNames","tooltip","error","debug","warn","toggleNativeControls","focused","activeElement","focusable","last","on","keyCode","_this","active","getFocusElement","shiftKey","focus","preventDefault","passive","capture","Node","toggleListener","events","options","boolean","passiveListeners","bubbles","detail","CustomEvent","assign","dispatchEvent","pressed","getAttribute","state","max","toFixed","sources","source","shift","extend","match","RegExp","$2","number","map","encodeURIComponent","join","fragment","createDocumentFragment","firstChild","innerText","width","height","ratio","getRatio","w","h","find","undefined","setTimeout","offsetHeight","provider","inline","api","ui","browser","getBrowser","playsInline","isIPhone","video","rangeInput","audio","webkitSetPresentationMode","WebKitPlaybackTargetAvailabilityEvent","media","isHTML5","canPlayType","isVideo","isAudio","supported","defineProperty","transitionEnd","matchMedia","noop","Console","player","enabled","console","log","bind","Storage","store","json","get","removeItem","cancelFullScreen","some","pre","msExitFullscreen","msFullscreenEnabled","fullscreenEnabled","webkitFullscreenEnabled","mozFullScreenEnabled","fullscreenElement","mozFullScreenElement","requestFullScreen","nativeSupport","fallback","inFrame","toggleClass","toggleState","trapFocus","listeners","handleKey","code","which","repeat","altKey","ctrlKey","metaKey","editable","stopPropagation","togglePlay","increaseVolume","decreaseVolume","muted","toggleFullscreen","toggleCaptions","loop","keyboard","global","tabFocus","hideControls","toggleControls","eventType","timeUpdate","durationUpdate","_this2","hasAudio","showPosterOnEnd","load","updateProgress","updateVolume","checkPlaying","checkLoading","clickToPlay","touch","paused","ended","disableContextMenu","set","updateSetting","speed","quality","concat","inputEvent","isIE","proxy","handlerKey","defaultHandler","customHandler","_this3","defaultPrevented","toggleMenu","form","parseFloat","showTab","toggleInvert","invertTime","isWebkit","updateRangeFill","updateSeekTooltip","hover","inverted","webkitDirectionInvertedFromDevice","direction","deltaY","deltaX","uiSupported","removeElement","inject","setup","ready","setTitle","label","i18n","title","isEmbed","iframe","frameTitle","playing","stopped","button","loading","timers","failed","networkState","setRange","nodeValue","buffered","getPercentage","_this4","end","setProgress","time","format","slice","getHours","parseInt","hours","mins","secs","invert","updateTimeDisplay","seeking","displayDuration","setProperty","iconUrl","indexOf","svg4everybody","namespace","getIconUrl","iconPath","absolute","iconPrefix","icon","createElementNS","use","path","setAttributeNS","attr","hidden","badge","menu","buttonType","labelPressed","iconPressed","control","createIcon","createLabel","getAttributesFromSelector","suffix","played","toLowerCase","list","checked","item","radio","faux","aria-hidden","insertAdjacentHTML","tooltips","percent","clientRect","getBoundingClientRect","visible","pageX","left","hasClass","setting","tab","tabs","pane","panes","filter","isYouTube","toggleTab","emptyElement","createMenuItem","getLabel","createBadge","getBadge","getLanguage","default","textTracks","getTracks","none","currentTrack","getCurrentTrack","track","disabled","hasTracks","tracks","toUpperCase","unshift","show","isMenuItem","isButton","open","clone","position","opacity","name","scrollWidth","scrollHeight","getElementById","transitions","reducedMotion","size","getTabSize","restore","propertyName","off","createButton","createRange","createProgress","createTime","inner","home","back","setSpeedMenu","loadSprite","seekTime","create","findElements","repaint","labels","_this5","stored","setCaptionsMenu","insertAfter","setLanguage","setCue","mode","activeCues","isVimeo","embed","enableTextTrack","cue","setText","getCueAsHTML","caption","youtube","videoId","parseYouTubeId","embedId","containers","setAspectRatio","generateId","YT","loadScript","urls","onYouTubeReadyCallbacks","onYouTubeIframeAPIReady","getVideoData","google","result","items","snippet","paddingBottom","Player","autoplay","location","hostname","href","message","instance","getPlaybackQuality","playbackRate","getPlaybackRate","getTitle","playVideo","pauseVideo","stop","stopVideo","getDuration","getCurrentTime","seekTo","setPlaybackRate","setPlaybackQuality","setVolume","getVideoUrl","getAvailablePlaybackRates","clearInterval","buffering","setInterval","getVideoLoadedFraction","lastBuffered","build","setQualityMenu","getAvailableQualityLevels","vimeo","Vimeo","padding","offset","transform","params","buildUrlParameters","parseVimeoId","setCurrentTime","selected","setLoop","currentSrc","all","getVideoWidth","getVideoHeight","getAspectRatio","dimensions","setAutopause","autopause","getVideoTitle","getTextTracks","cues","stripHTML","seconds","isIos","isTouch","wrap","blankVideo","insertElement","attribute","cancelRequests","destroy","html5","check","crossorigin","poster","addStyleHook","insertElements","scrollPosition","jQuery","plyr","original","tagName","hasAttribute","step","isFullScreen","pageXOffset","pageYOffset","scrollTo","x","y","overflow","webkitShowPlaybackTargetPicker","delay","isEnterFullscreen","noTransition","clearTimeout","soft","done","replaceChild","unload","mime","readyState","targetTime","fauxDuration","realDuration","mozHasAudio","webkitAudioDecodedByteCount","audioTracks","change","states","webkitPresentationMode"],"mappings":"uLAIA,ICopBcA,EASAC,EClhBAC,ECvINC,EHJKC,SACF,gBACE,gBACF,SAGEC,SACF,cACA,SIXLC,YAEO,QAGF,UAGA,YAGG,aAGC,WAGD,UAGF,SACD,WAGG,sBAIO,cAGL,gBAGE,QAGP,oBAGM,gBAGC,mBAGG,sBAGG,cAGR,aACA,eACH,iDAGG,wDAIC,mBAEL,SACA,SACA,SACA,QACA,QACA,SACA,QACA,OACA,0BAMI,mBAOE,WAEN,GACA,IACA,EACA,KACA,IACA,KACA,uBAMK,UACD,uBAKE,QACJ,qBAKE,WACEC,OAAOC,UAAUC,SAASC,MAAM,KAAK,yBAKtC,YACC,qBAKD,MACJ,kBAKL,aACA,OACA,WACA,eACA,OACA,SACA,WACA,WACA,MACA,UACA,wBAGA,WACA,UACA,uBAKS,iBACD,8BACF,aACC,gBACE,+BACH,cACE,kBACE,uBACG,wBACH,kBACF,cACF,cACE,wBACQ,kCACC,mCACA,kCACD,6BACJ,8BACF,oBACA,iBACH,gBACE,eACH,aACC,YACF,UACA,YACE,aACD,gBACI,6BAMD,uDAGA,uDAMH,UACA,WACC,aACE,YACD,aACC,UACH,YACE,cACE,gBACE,SACP,aACI,WACF,aACE,UACH,cACI,sBAQV,WACA,UACA,UACA,UACA,UACA,iBACA,YACA,aACA,iBACA,aACA,eACA,OACA,QACA,QACA,UACA,SACA,UACA,aACA,8BAIA,iBACA,kBACA,mBACA,iBACA,iBACA,gBACA,sBAIA,gBACA,wCAMU,uDACC,4BAEI,aACF,0BAEL,4BAEE,2BACC,8BACE,+BACD,+BACC,kCACH,8BACI,oCACE,+BACP,4BACI,iCACC,8BACJ,mCAGA,4BACE,6BACD,+BACG,iCACD,8CAGI,gCACH,+BACF,iCACA,+BACF,+BACE,mCAEF,2BACA,gCAEG,oDAMN,4BACA,4BACE,qBACH,qBACI,oBACD,wBACA,wBACA,sBACF,wBACA,sBACE,uBACD,6BACM,4BACP,uBACE,6BACI,6BACC,kCAEH,0BACA,mBACD,qCAGG,gCACD,6CAGC,oCACC,4CAGC,6BACH,u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"}
\ No newline at end of file +{"version":3,"file":"plyr.js","sources":["src/js/types.js","src/js/utils.js","src/js/support.js","src/js/fullscreen.js","src/js/defaults.js","src/js/console.js","src/js/storage.js","src/js/listeners.js","src/js/ui.js","src/js/controls.js","src/js/captions.js","src/js/plugins/youtube.js","src/js/plugins/vimeo.js","src/js/media.js","src/js/source.js","src/js/plyr.js"],"sourcesContent":["// ==========================================================================\n// Plyr supported types and providers\n// ==========================================================================\n\nexport const providers = {\n html5: 'html5',\n youtube: 'youtube',\n vimeo: 'vimeo',\n};\n\nexport const types = {\n audio: 'audio',\n video: 'video',\n};\n\nexport default { providers, types };\n","// ==========================================================================\n// Plyr utils\n// ==========================================================================\n\nimport support from './support';\nimport { providers } from './types';\n\nconst utils = {\n // Check variable types\n is: {\n plyr(input) {\n return this.instanceof(input, Plyr);\n },\n object(input) {\n return this.getConstructor(input) === Object;\n },\n number(input) {\n return this.getConstructor(input) === Number && !Number.isNaN(input);\n },\n string(input) {\n return this.getConstructor(input) === String;\n },\n boolean(input) {\n return this.getConstructor(input) === Boolean;\n },\n function(input) {\n return this.getConstructor(input) === Function;\n },\n array(input) {\n return !this.nullOrUndefined(input) && Array.isArray(input);\n },\n weakMap(input) {\n return this.instanceof(input, window.WeakMap);\n },\n nodeList(input) {\n return this.instanceof(input, window.NodeList);\n },\n element(input) {\n return this.instanceof(input, window.Element);\n },\n textNode(input) {\n return this.getConstructor(input) === Text;\n },\n event(input) {\n return this.instanceof(input, window.Event);\n },\n cue(input) {\n return this.instanceof(input, window.TextTrackCue) || this.instanceof(input, window.VTTCue);\n },\n track(input) {\n return this.instanceof(input, TextTrack) || (!this.nullOrUndefined(input) && this.string(input.kind));\n },\n nullOrUndefined(input) {\n return input === null || typeof input === 'undefined';\n },\n empty(input) {\n return (\n this.nullOrUndefined(input) ||\n ((this.string(input) || this.array(input) || this.nodeList(input)) && !input.length) ||\n (this.object(input) && !Object.keys(input).length)\n );\n },\n instanceof(input, constructor) {\n return Boolean(input && constructor && input instanceof constructor);\n },\n getConstructor(input) {\n return !this.nullOrUndefined(input) ? input.constructor : null;\n },\n },\n\n // Unfortunately, due to mixed support, UA sniffing is required\n getBrowser() {\n return {\n isIE: /* @cc_on!@ */ false || !!document.documentMode,\n isWebkit: 'WebkitAppearance' in document.documentElement.style && !/Edge/.test(navigator.userAgent),\n isIPhone: /(iPhone|iPod)/gi.test(navigator.platform),\n isIos: /(iPad|iPhone|iPod)/gi.test(navigator.platform),\n };\n },\n\n // Load an external script\n loadScript(url, callback) {\n const current = document.querySelector(`script[src=\"${url}\"]`);\n\n // Check script is not already referenced, if so wait for load\n if (current !== null) {\n current.callbacks = current.callbacks || [];\n current.callbacks.push(callback);\n return;\n }\n\n // Build the element\n const element = document.createElement('script');\n\n // Callback queue\n element.callbacks = element.callbacks || [];\n element.callbacks.push(callback);\n\n // Bind callback\n if (utils.is.function(callback)) {\n element.addEventListener(\n 'load',\n event => {\n element.callbacks.forEach(cb => cb.call(null, event));\n element.callbacks = null;\n },\n false,\n );\n }\n\n // Set the URL after binding callback\n element.src = url;\n\n // Inject\n const first = document.getElementsByTagName('script')[0];\n first.parentNode.insertBefore(element, first);\n },\n\n // Load an external SVG sprite\n loadSprite(url, id) {\n if (!utils.is.string(url)) {\n return;\n }\n\n const prefix = 'cache-';\n const hasId = utils.is.string(id);\n let isCached = false;\n\n function updateSprite(data) {\n // Inject content\n this.innerHTML = data;\n\n // Inject the SVG to the body\n document.body.insertBefore(this, document.body.childNodes[0]);\n }\n\n // Only load once\n if (!hasId || !document.querySelectorAll(`#${id}`).length) {\n // Create container\n const container = document.createElement('div');\n utils.toggleHidden(container, true);\n\n if (hasId) {\n container.setAttribute('id', id);\n }\n\n // Check in cache\n if (support.storage) {\n const cached = window.localStorage.getItem(prefix + id);\n isCached = cached !== null;\n\n if (isCached) {\n const data = JSON.parse(cached);\n updateSprite.call(container, data.content);\n return;\n }\n }\n\n // Get the sprite\n fetch(url)\n .then(response => (response.ok ? response.text() : null))\n .then(text => {\n if (text === null) {\n return;\n }\n\n if (support.storage) {\n window.localStorage.setItem(\n prefix + id,\n JSON.stringify({\n content: text,\n }),\n );\n }\n\n updateSprite.call(container, text);\n })\n .catch(() => {});\n }\n },\n\n // Generate a random ID\n generateId(prefix) {\n return `${prefix}-${Math.floor(Math.random() * 10000)}`;\n },\n\n // Determine if we're in an iframe\n inFrame() {\n try {\n return window.self !== window.top;\n } catch (e) {\n return true;\n }\n },\n\n // Wrap an element\n wrap(elements, wrapper) {\n // Convert `elements` to an array, if necessary.\n const targets = elements.length ? elements : [elements];\n\n // Loops backwards to prevent having to clone the wrapper on the\n // first element (see `child` below).\n Array.from(targets)\n .reverse()\n .forEach((element, index) => {\n const child = index > 0 ? wrapper.cloneNode(true) : wrapper;\n\n // Cache the current parent and sibling.\n const parent = element.parentNode;\n const sibling = element.nextSibling;\n\n // Wrap the element (is automatically removed from its current\n // parent).\n child.appendChild(element);\n\n // If the element had a sibling, insert the wrapper before\n // the sibling to maintain the HTML structure; otherwise, just\n // append it to the parent.\n if (sibling) {\n parent.insertBefore(child, sibling);\n } else {\n parent.appendChild(child);\n }\n });\n },\n\n // Create a DocumentFragment\n createElement(type, attributes, text) {\n // Create a new <element>\n const element = document.createElement(type);\n\n // Set all passed attributes\n if (utils.is.object(attributes)) {\n utils.setAttributes(element, attributes);\n }\n\n // Add text node\n if (utils.is.string(text)) {\n element.textContent = text;\n }\n\n // Return built element\n return element;\n },\n\n // Inaert an element after another\n insertAfter(element, target) {\n target.parentNode.insertBefore(element, target.nextSibling);\n },\n\n // Insert a DocumentFragment\n insertElement(type, parent, attributes, text) {\n // Inject the new <element>\n parent.appendChild(utils.createElement(type, attributes, text));\n },\n\n // Remove an element\n removeElement(element) {\n if (!utils.is.element(element) || !utils.is.element(element.parentNode)) {\n return null;\n }\n\n element.parentNode.removeChild(element);\n\n return element;\n },\n\n // Remove all child elements\n emptyElement(element) {\n let { length } = element.childNodes;\n\n while (length > 0) {\n element.removeChild(element.lastChild);\n length -= 1;\n }\n },\n\n // Replace element\n replaceElement(newChild, oldChild) {\n if (!utils.is.element(oldChild) || !utils.is.element(oldChild.parentNode) || !utils.is.element(newChild)) {\n return null;\n }\n\n oldChild.parentNode.replaceChild(newChild, oldChild);\n\n return newChild;\n },\n\n // Set attributes\n setAttributes(element, attributes) {\n if (!utils.is.element(element) || utils.is.empty(attributes)) {\n return;\n }\n\n Object.keys(attributes).forEach(key => {\n element.setAttribute(key, attributes[key]);\n });\n },\n\n // Get an attribute object from a string selector\n getAttributesFromSelector(sel, existingAttributes) {\n // For example:\n // '.test' to { class: 'test' }\n // '#test' to { id: 'test' }\n // '[data-test=\"test\"]' to { 'data-test': 'test' }\n\n if (!utils.is.string(sel) || utils.is.empty(sel)) {\n return {};\n }\n\n const attributes = {};\n const existing = existingAttributes;\n\n sel.split(',').forEach(s => {\n // Remove whitespace\n const selector = s.trim();\n const className = selector.replace('.', '');\n const stripped = selector.replace(/[[\\]]/g, '');\n\n // Get the parts and value\n const parts = stripped.split('=');\n const key = parts[0];\n const value = parts.length > 1 ? parts[1].replace(/[\"']/g, '') : '';\n\n // Get the first character\n const start = selector.charAt(0);\n\n switch (start) {\n case '.':\n // Add to existing classname\n if (utils.is.object(existing) && utils.is.string(existing.class)) {\n existing.class += ` ${className}`;\n }\n\n attributes.class = className;\n break;\n\n case '#':\n // ID selector\n attributes.id = selector.replace('#', '');\n break;\n\n case '[':\n // Attribute selector\n attributes[key] = value;\n\n break;\n\n default:\n break;\n }\n });\n\n return attributes;\n },\n\n // Toggle class on an element\n toggleClass(element, className, toggle) {\n if (utils.is.element(element)) {\n const contains = element.classList.contains(className);\n\n element.classList[toggle ? 'add' : 'remove'](className);\n\n return (toggle && !contains) || (!toggle && contains);\n }\n\n return null;\n },\n\n // Has class name\n hasClass(element, className) {\n return utils.is.element(element) && element.classList.contains(className);\n },\n\n // Toggle hidden attribute on an element\n toggleHidden(element, toggle) {\n if (!utils.is.element(element)) {\n return;\n }\n\n if (toggle) {\n element.setAttribute('hidden', '');\n } else {\n element.removeAttribute('hidden');\n }\n },\n\n // Element matches selector\n matches(element, selector) {\n const prototype = { Element };\n\n function match() {\n return Array.from(document.querySelectorAll(selector)).includes(this);\n }\n\n const matches = prototype.matches || prototype.webkitMatchesSelector || prototype.mozMatchesSelector || prototype.msMatchesSelector || match;\n\n return matches.call(element, selector);\n },\n\n // Find all elements\n getElements(selector) {\n return this.elements.container.querySelectorAll(selector);\n },\n\n // Find a single element\n getElement(selector) {\n return this.elements.container.querySelector(selector);\n },\n\n // Find the UI controls and store references in custom controls\n // TODO: Allow settings menus with custom controls\n findElements() {\n try {\n this.elements.controls = utils.getElement.call(this, this.config.selectors.controls.wrapper);\n\n // Buttons\n this.elements.buttons = {\n play: utils.getElements.call(this, this.config.selectors.buttons.play),\n pause: utils.getElement.call(this, this.config.selectors.buttons.pause),\n restart: utils.getElement.call(this, this.config.selectors.buttons.restart),\n rewind: utils.getElement.call(this, this.config.selectors.buttons.rewind),\n forward: utils.getElement.call(this, this.config.selectors.buttons.forward),\n mute: utils.getElement.call(this, this.config.selectors.buttons.mute),\n pip: utils.getElement.call(this, this.config.selectors.buttons.pip),\n airplay: utils.getElement.call(this, this.config.selectors.buttons.airplay),\n settings: utils.getElement.call(this, this.config.selectors.buttons.settings),\n captions: utils.getElement.call(this, this.config.selectors.buttons.captions),\n fullscreen: utils.getElement.call(this, this.config.selectors.buttons.fullscreen),\n };\n\n // Progress\n this.elements.progress = utils.getElement.call(this, this.config.selectors.progress);\n\n // Inputs\n this.elements.inputs = {\n seek: utils.getElement.call(this, this.config.selectors.inputs.seek),\n volume: utils.getElement.call(this, this.config.selectors.inputs.volume),\n };\n\n // Display\n this.elements.display = {\n buffer: utils.getElement.call(this, this.config.selectors.display.buffer),\n duration: utils.getElement.call(this, this.config.selectors.display.duration),\n currentTime: utils.getElement.call(this, this.config.selectors.display.currentTime),\n };\n\n // Seek tooltip\n if (utils.is.element(this.elements.progress)) {\n this.elements.display.seekTooltip = this.elements.progress.querySelector(`.${this.config.classNames.tooltip}`);\n }\n\n return true;\n } catch (error) {\n // Log it\n this.debug.warn('It looks like there is a problem with your custom controls HTML', error);\n\n // Restore native video controls\n this.toggleNativeControls(true);\n\n return false;\n }\n },\n\n // Get the focused element\n getFocusElement() {\n let focused = document.activeElement;\n\n if (!focused || focused === document.body) {\n focused = null;\n } else {\n focused = document.querySelector(':focus');\n }\n\n return focused;\n },\n\n // Trap focus inside container\n trapFocus() {\n const focusable = utils.getElements.call(this, 'button:not(:disabled), input:not(:disabled), [tabindex]');\n const first = focusable[0];\n const last = focusable[focusable.length - 1];\n\n utils.on(\n this.elements.container,\n 'keydown',\n event => {\n // Bail if not tab key or not fullscreen\n if (event.key !== 'Tab' || event.keyCode !== 9 || !this.fullscreen.active) {\n return;\n }\n\n // Get the current focused element\n const focused = utils.getFocusElement();\n\n if (focused === last && !event.shiftKey) {\n // Move focus to first element that can be tabbed if Shift isn't used\n first.focus();\n event.preventDefault();\n } else if (focused === first && event.shiftKey) {\n // Move focus to last element that can be tabbed if Shift is used\n last.focus();\n event.preventDefault();\n }\n },\n false,\n );\n },\n\n // Toggle event listener\n toggleListener(elements, event, callback, toggle, passive, capture) {\n // Bail if no elements\n if (utils.is.nullOrUndefined(elements)) {\n return;\n }\n\n // If a nodelist is passed, call itself on each node\n if (utils.is.nodeList(elements)) {\n // Create listener for each node\n Array.from(elements).forEach(element => {\n if (element instanceof Node) {\n utils.toggleListener.call(null, element, event, callback, toggle, passive, capture);\n }\n });\n\n return;\n }\n\n // Allow multiple events\n const events = event.split(' ');\n\n // Build options\n // Default to just capture boolean\n let options = utils.is.boolean(capture) ? capture : false;\n\n // If passive events listeners are supported\n if (support.passiveListeners) {\n options = {\n // Whether the listener can be passive (i.e. default never prevented)\n passive: utils.is.boolean(passive) ? passive : true,\n // Whether the listener is a capturing listener or not\n capture: utils.is.boolean(capture) ? capture : false,\n };\n }\n\n // If a single node is passed, bind the event listener\n events.forEach(type => {\n elements[toggle ? 'addEventListener' : 'removeEventListener'](type, callback, options);\n });\n },\n\n // Bind event handler\n on(element, events, callback, passive, capture) {\n utils.toggleListener(element, events, callback, true, passive, capture);\n },\n\n // Unbind event handler\n off(element, events, callback, passive, capture) {\n utils.toggleListener(element, events, callback, false, passive, capture);\n },\n\n // Trigger event\n dispatchEvent(element, type, bubbles, detail) {\n // Bail if no element\n if (!element || !type) {\n return;\n }\n\n // Create and dispatch the event\n const event = new CustomEvent(type, {\n bubbles: utils.is.boolean(bubbles) ? bubbles : false,\n detail: Object.assign({}, detail, {\n plyr: this instanceof Plyr ? this : null,\n }),\n });\n\n // Dispatch the event\n element.dispatchEvent(event);\n },\n\n // Toggle aria-pressed state on a toggle button\n // http://www.ssbbartgroup.com/blog/how-not-to-misuse-aria-states-properties-and-roles\n toggleState(element, input) {\n // Bail if no target\n if (!utils.is.element(element)) {\n return;\n }\n\n // Get state\n const pressed = element.getAttribute('aria-pressed') === 'true';\n const state = utils.is.boolean(input) ? input : !pressed;\n\n // Set the attribute on target\n element.setAttribute('aria-pressed', state);\n },\n\n // Get percentage\n getPercentage(current, max) {\n if (current === 0 || max === 0 || Number.isNaN(current) || Number.isNaN(max)) {\n return 0;\n }\n return (current / max * 100).toFixed(2);\n },\n\n // Deep extend destination object with N more objects\n extend(target = {}, ...sources) {\n if (!sources.length) {\n return target;\n }\n\n const source = sources.shift();\n\n if (!utils.is.object(source)) {\n return target;\n }\n\n Object.keys(source).forEach(key => {\n if (utils.is.object(source[key])) {\n if (!Object.keys(target).includes(key)) {\n Object.assign(target, { [key]: {} });\n }\n\n utils.extend(target[key], source[key]);\n } else {\n Object.assign(target, { [key]: source[key] });\n }\n });\n\n return utils.extend(target, ...sources);\n },\n\n // Get the provider for a given URL\n getProviderByUrl(url) {\n // YouTube\n if (/^(https?:\\/\\/)?(www\\.)?(youtube\\.com|youtu\\.?be)\\/.+$/.test(url)) {\n return providers.youtube;\n }\n\n // Vimeo\n if (/^https?:\\/\\/player.vimeo.com\\/video\\/\\d{8,}(?=\\b|\\/)/.test(url)) {\n return providers.vimeo;\n }\n\n return null;\n },\n\n // Parse YouTube ID from URL\n parseYouTubeId(url) {\n if (utils.is.empty(url)) {\n return null;\n }\n\n const regex = /^.*(youtu.be\\/|v\\/|u\\/\\w\\/|embed\\/|watch\\?v=|&v=)([^#&?]*).*/;\n return url.match(regex) ? RegExp.$2 : url;\n },\n\n // Parse Vimeo ID from URL\n parseVimeoId(url) {\n if (utils.is.empty(url)) {\n return null;\n }\n\n if (utils.is.number(Number(url))) {\n return url;\n }\n\n const regex = /^.*(vimeo.com\\/|video\\/)(\\d+).*/;\n return url.match(regex) ? RegExp.$2 : url;\n },\n\n // Convert a URL to a location object\n parseUrl(url) {\n const parser = document.createElement('a');\n parser.href = url;\n return parser;\n },\n\n // Get URL query parameters\n getUrlParams(input) {\n let search = input;\n\n // Parse URL if needed\n if (input.startsWith('http://') || input.startsWith('https://')) {\n ({ search } = this.parseUrl(input));\n }\n\n if (this.is.empty(search)) {\n return null;\n }\n\n const hashes = search.slice(search.indexOf('?') + 1).split('&');\n\n return hashes.reduce((params, hash) => {\n const [\n key,\n val,\n ] = hash.split('=');\n\n return Object.assign(params, { [key]: decodeURIComponent(val) });\n }, {});\n },\n\n // Convert object to URL parameters\n buildUrlParams(input) {\n if (!utils.is.object(input)) {\n return '';\n }\n\n return Object.keys(input)\n .map(key => `${encodeURIComponent(key)}=${encodeURIComponent(input[key])}`)\n .join('&');\n },\n\n // Remove HTML from a string\n stripHTML(source) {\n const fragment = document.createDocumentFragment();\n const element = document.createElement('div');\n fragment.appendChild(element);\n element.innerHTML = source;\n return fragment.firstChild.innerText;\n },\n\n // Get aspect ratio for dimensions\n getAspectRatio(width, height) {\n const getRatio = (w, h) => (h === 0 ? w : getRatio(h, w % h));\n const ratio = getRatio(width, height);\n return `${width / ratio}:${height / ratio}`;\n },\n\n // Get the transition end event\n transitionEnd: (() => {\n const element = document.createElement('span');\n\n const events = {\n WebkitTransition: 'webkitTransitionEnd',\n MozTransition: 'transitionend',\n OTransition: 'oTransitionEnd otransitionend',\n transition: 'transitionend',\n };\n\n const type = Object.keys(events).find(event => element.style[event] !== undefined);\n\n return typeof type === 'string' ? type : false;\n })(),\n\n // Force repaint of element\n repaint(element) {\n window.setTimeout(() => {\n element.setAttribute('hidden', '');\n element.offsetHeight; // eslint-disable-line\n element.removeAttribute('hidden');\n }, 0);\n },\n};\n\nexport default utils;\n","// ==========================================================================\n// Plyr support checks\n// ==========================================================================\n\nimport utils from './utils';\n\n// Check for feature support\nconst support = {\n // Basic support\n audio: 'canPlayType' in document.createElement('audio'),\n video: 'canPlayType' in document.createElement('video'),\n\n // Check for support\n // Basic functionality vs full UI\n check(type, provider, inline) {\n let api = false;\n let ui = false;\n const browser = utils.getBrowser();\n const playsInline = browser.isIPhone && inline && support.inline;\n\n switch (`${provider}:${type}`) {\n case 'html5:video':\n api = support.video;\n ui = api && support.rangeInput && (!browser.isIPhone || playsInline);\n break;\n\n case 'html5:audio':\n api = support.audio;\n ui = api && support.rangeInput;\n break;\n\n case 'youtube:video':\n api = true;\n ui = support.rangeInput && (!browser.isIPhone || playsInline);\n break;\n\n case 'vimeo:video':\n api = true;\n ui = support.rangeInput && !browser.isIPhone;\n break;\n\n default:\n api = support.audio && support.video;\n ui = api && support.rangeInput;\n }\n\n return {\n api,\n ui,\n };\n },\n\n // Picture-in-picture support\n // Safari only currently\n pip: (() => {\n const browser = utils.getBrowser();\n return !browser.isIPhone && utils.is.function(utils.createElement('video').webkitSetPresentationMode);\n })(),\n\n // Airplay support\n // Safari only currently\n airplay: utils.is.function(window.WebKitPlaybackTargetAvailabilityEvent),\n\n // Inline playback support\n // https://webkit.org/blog/6784/new-video-policies-for-ios/\n inline: 'playsInline' in document.createElement('video'),\n\n // Check for mime type support against a player instance\n // Credits: http://diveintohtml5.info/everything.html\n // Related: http://www.leanbackplayer.com/test/h5mt.html\n mime(type) {\n const { media } = this;\n\n try {\n // Bail if no checking function\n if (!this.isHTML5 || !utils.is.function(media.canPlayType)) {\n return false;\n }\n\n // Type specific checks\n if (this.isVideo) {\n switch (type) {\n case 'video/webm':\n return media.canPlayType('video/webm; codecs=\"vp8, vorbis\"').replace(/no/, '');\n\n case 'video/mp4':\n return media.canPlayType('video/mp4; codecs=\"avc1.42E01E, mp4a.40.2\"').replace(/no/, '');\n\n case 'video/ogg':\n return media.canPlayType('video/ogg; codecs=\"theora\"').replace(/no/, '');\n\n default:\n return false;\n }\n } else if (this.isAudio) {\n switch (type) {\n case 'audio/mpeg':\n return media.canPlayType('audio/mpeg;').replace(/no/, '');\n\n case 'audio/ogg':\n return media.canPlayType('audio/ogg; codecs=\"vorbis\"').replace(/no/, '');\n\n case 'audio/wav':\n return media.canPlayType('audio/wav; codecs=\"1\"').replace(/no/, '');\n\n default:\n return false;\n }\n }\n } catch (e) {\n return false;\n }\n\n // If we got this far, we're stuffed\n return false;\n },\n\n // Check for textTracks support\n textTracks: 'textTracks' in document.createElement('video'),\n\n // Check for passive event listener support\n // https://github.com/WICG/EventListenerOptions/blob/gh-pages/explainer.md\n // https://www.youtube.com/watch?v=NPM6172J22g\n passiveListeners: (() => {\n // Test via a getter in the options object to see if the passive property is accessed\n let supported = false;\n try {\n const options = Object.defineProperty({}, 'passive', {\n get() {\n supported = true;\n return null;\n },\n });\n window.addEventListener('test', null, options);\n } catch (e) {\n // Do nothing\n }\n\n return supported;\n })(),\n\n // <input type=\"range\"> Sliders\n rangeInput: (() => {\n const range = document.createElement('input');\n range.type = 'range';\n return range.type === 'range';\n })(),\n\n // Touch\n // Remember a device can be moust + touch enabled\n touch: 'ontouchstart' in document.documentElement,\n\n // Detect transitions support\n transitions: utils.transitionEnd !== false,\n\n // Reduced motion iOS & MacOS setting\n // https://webkit.org/blog/7551/responsive-design-for-motion/\n reducedMotion: 'matchMedia' in window && window.matchMedia('(prefers-reduced-motion)').matches,\n};\n\nexport default support;\n","// ==========================================================================\n// Plyr fullscreen API\n// ==========================================================================\n\nimport utils from './utils';\n\n// Determine the prefix\nconst prefix = (() => {\n let value = false;\n\n if (utils.is.function(document.cancelFullScreen)) {\n value = '';\n } else {\n // Check for fullscreen support by vendor prefix\n [\n 'webkit',\n 'o',\n 'moz',\n 'ms',\n 'khtml',\n ].some(pre => {\n if (utils.is.function(document[`${pre}CancelFullScreen`])) {\n value = pre;\n return true;\n } else if (utils.is.function(document.msExitFullscreen) && document.msFullscreenEnabled) {\n // Special case for MS (when isn't it?)\n value = 'ms';\n return true;\n }\n\n return false;\n });\n }\n\n return value;\n})();\n\n// Fullscreen API\nconst fullscreen = {\n // Get the prefix\n prefix,\n\n // Check if we can use it\n enabled: document.fullscreenEnabled || document.webkitFullscreenEnabled || document.mozFullScreenEnabled || document.msFullscreenEnabled,\n\n // Yet again Microsoft awesomeness,\n // Sometimes the prefix is 'ms', sometimes 'MS' to keep you on your toes\n eventType: prefix === 'ms' ? 'MSFullscreenChange' : `${prefix}fullscreenchange`,\n\n // Is an element fullscreen\n isFullScreen(element) {\n if (!fullscreen.enabled) {\n return false;\n }\n\n const target = utils.is.nullOrUndefined(element) ? document.body : element;\n\n switch (prefix) {\n case '':\n return document.fullscreenElement === target;\n\n case 'moz':\n return document.mozFullScreenElement === target;\n\n default:\n return document[`${prefix}FullscreenElement`] === target;\n }\n },\n\n // Make an element fullscreen\n requestFullScreen(element) {\n if (!fullscreen.enabled) {\n return false;\n }\n\n const target = utils.is.nullOrUndefined(element) ? document.body : element;\n\n return !prefix.length ? target.requestFullScreen() : target[prefix + (prefix === 'ms' ? 'RequestFullscreen' : 'RequestFullScreen')]();\n },\n\n // Bail from fullscreen\n cancelFullScreen() {\n if (!fullscreen.enabled) {\n return false;\n }\n\n return !prefix.length ? document.cancelFullScreen() : document[prefix + (prefix === 'ms' ? 'ExitFullscreen' : 'CancelFullScreen')]();\n },\n\n // Get the current element\n element() {\n if (!fullscreen.enabled) {\n return null;\n }\n\n return !prefix.length ? document.fullscreenElement : document[`${prefix}FullscreenElement`];\n },\n\n // Setup fullscreen\n setup() {\n if (!this.supported.ui || this.isAudio || !this.config.fullscreen.enabled) {\n return;\n }\n\n // Check for native support\n const nativeSupport = fullscreen.enabled;\n\n if (nativeSupport || (this.config.fullscreen.fallback && !utils.inFrame())) {\n this.debug.log(`${nativeSupport ? 'Native' : 'Fallback'} fullscreen enabled`);\n\n // Add styling hook to show button\n utils.toggleClass(this.elements.container, this.config.classNames.fullscreen.enabled, true);\n } else {\n this.debug.log('Fullscreen not supported and fallback disabled');\n }\n\n // Toggle state\n if (this.elements.buttons && this.elements.buttons.fullscreen) {\n utils.toggleState(this.elements.buttons.fullscreen, false);\n }\n\n // Trap focus in container\n utils.trapFocus.call(this);\n },\n};\n\nexport default fullscreen;\n","// Default config\nconst defaults = {\n // Disable\n enabled: true,\n\n // Custom media title\n title: '',\n\n // Logging to console\n debug: false,\n\n // Auto play (if supported)\n autoplay: false,\n\n // Only allow one media playing at once (vimeo only)\n autopause: true,\n\n // Default time to skip when rewind/fast forward\n seekTime: 10,\n\n // Default volume\n volume: 1,\n muted: false,\n\n // Pass a custom duration\n duration: null,\n\n // Display the media duration on load in the current time position\n // If you have opted to display both duration and currentTime, this is ignored\n displayDuration: true,\n\n // Invert the current time to be a countdown\n invertTime: true,\n\n // Clicking the currentTime inverts it's value to show time left rather than elapsed\n toggleInvert: true,\n\n // Aspect ratio (for embeds)\n ratio: '16:9',\n\n // Click video container to play/pause\n clickToPlay: true,\n\n // Auto hide the controls\n hideControls: true,\n\n // Revert to poster on finish (HTML5 - will cause reload)\n showPosterOnEnd: false,\n\n // Disable the standard context menu\n disableContextMenu: true,\n\n // Sprite (for icons)\n loadSprite: true,\n iconPrefix: 'plyr',\n iconUrl: 'https://cdn.plyr.io/2.0.10/plyr.svg',\n\n // Blank video (used to prevent errors on source change)\n blankVideo: 'https://cdn.plyr.io/static/blank.mp4',\n\n // Quality default\n quality: {\n default: 'default',\n options: [\n 'hd2160',\n 'hd1440',\n 'hd1080',\n 'hd720',\n 'large',\n 'medium',\n 'small',\n 'tiny',\n 'default',\n ],\n },\n\n // Set loops\n loop: {\n active: false,\n // start: null,\n // end: null,\n },\n\n // Speed default and options to display\n speed: {\n selected: 1,\n options: [\n 0.5,\n 0.75,\n 1,\n 1.25,\n 1.5,\n 1.75,\n 2,\n ],\n },\n\n // Keyboard shortcut settings\n keyboard: {\n focused: true,\n global: false,\n },\n\n // Display tooltips\n tooltips: {\n controls: false,\n seek: true,\n },\n\n // Captions settings\n captions: {\n active: false,\n language: window.navigator.language.split('-')[0],\n },\n\n // Fullscreen settings\n fullscreen: {\n enabled: true, // Allow fullscreen?\n fallback: true, // Fallback for vintage browsers\n },\n\n // Local storage\n storage: {\n enabled: true,\n key: 'plyr',\n },\n\n // Default controls\n controls: [\n 'play-large',\n 'play',\n 'progress',\n 'current-time',\n 'mute',\n 'volume',\n 'captions',\n 'settings',\n 'pip',\n 'airplay',\n 'fullscreen',\n ],\n settings: [\n 'captions',\n 'quality',\n 'speed',\n ],\n\n // Localisation\n i18n: {\n restart: 'Restart',\n rewind: 'Rewind {seektime} secs',\n play: 'Play',\n pause: 'Pause',\n forward: 'Forward {seektime} secs',\n seek: 'Seek',\n played: 'Played',\n buffered: 'Buffered',\n currentTime: 'Current time',\n duration: 'Duration',\n volume: 'Volume',\n mute: 'Mute',\n unmute: 'Unmute',\n enableCaptions: 'Enable captions',\n disableCaptions: 'Disable captions',\n enterFullscreen: 'Enter fullscreen',\n exitFullscreen: 'Exit fullscreen',\n frameTitle: 'Player for {title}',\n captions: 'Captions',\n settings: 'Settings',\n speed: 'Speed',\n quality: 'Quality',\n loop: 'Loop',\n start: 'Start',\n end: 'End',\n all: 'All',\n reset: 'Reset',\n none: 'None',\n disabled: 'Disabled',\n },\n\n // URLs\n urls: {\n vimeo: {\n api: 'https://player.vimeo.com/api/player.js',\n },\n youtube: {\n api: 'https://www.youtube.com/iframe_api',\n },\n },\n\n // Custom control listeners\n listeners: {\n seek: null,\n play: null,\n pause: null,\n restart: null,\n rewind: null,\n forward: null,\n mute: null,\n volume: null,\n captions: null,\n fullscreen: null,\n pip: null,\n airplay: null,\n speed: null,\n quality: null,\n loop: null,\n language: null,\n },\n\n // Events to watch and bubble\n events: [\n // Events to watch on HTML5 media elements and bubble\n // https://developer.mozilla.org/en/docs/Web/Guide/Events/Media_events\n 'ended',\n 'progress',\n 'stalled',\n 'playing',\n 'waiting',\n 'canplay',\n 'canplaythrough',\n 'loadstart',\n 'loadeddata',\n 'loadedmetadata',\n 'timeupdate',\n 'volumechange',\n 'play',\n 'pause',\n 'error',\n 'seeking',\n 'seeked',\n 'emptied',\n 'ratechange',\n 'cuechange',\n\n // Custom events\n 'enterfullscreen',\n 'exitfullscreen',\n 'captionsenabled',\n 'captionsdisabled',\n 'languagechange',\n 'controlshidden',\n 'controlsshown',\n 'ready',\n\n // YouTube\n 'statechange',\n 'qualitychange',\n 'qualityrequested',\n ],\n\n // Selectors\n // Change these to match your template if using custom HTML\n selectors: {\n editable: 'input, textarea, select, [contenteditable]',\n container: '.plyr',\n controls: {\n container: null,\n wrapper: '.plyr__controls',\n },\n labels: '[data-plyr]',\n buttons: {\n play: '[data-plyr=\"play\"]',\n pause: '[data-plyr=\"pause\"]',\n restart: '[data-plyr=\"restart\"]',\n rewind: '[data-plyr=\"rewind\"]',\n forward: '[data-plyr=\"fast-forward\"]',\n mute: '[data-plyr=\"mute\"]',\n captions: '[data-plyr=\"captions\"]',\n fullscreen: '[data-plyr=\"fullscreen\"]',\n pip: '[data-plyr=\"pip\"]',\n airplay: '[data-plyr=\"airplay\"]',\n settings: '[data-plyr=\"settings\"]',\n loop: '[data-plyr=\"loop\"]',\n },\n inputs: {\n seek: '[data-plyr=\"seek\"]',\n volume: '[data-plyr=\"volume\"]',\n speed: '[data-plyr=\"speed\"]',\n language: '[data-plyr=\"language\"]',\n quality: '[data-plyr=\"quality\"]',\n },\n display: {\n currentTime: '.plyr__time--current',\n duration: '.plyr__time--duration',\n buffer: '.plyr__progress--buffer',\n played: '.plyr__progress--played',\n loop: '.plyr__progress--loop',\n volume: '.plyr__volume--display',\n },\n progress: '.plyr__progress',\n captions: '.plyr__captions',\n menu: {\n quality: '.js-plyr__menu__list--quality',\n },\n },\n\n // Class hooks added to the player in different states\n classNames: {\n video: 'plyr__video-wrapper',\n embed: 'plyr__video-embed',\n control: 'plyr__control',\n type: 'plyr--{0}',\n provider: 'plyr--{0}',\n stopped: 'plyr--stopped',\n playing: 'plyr--playing',\n loading: 'plyr--loading',\n error: 'plyr--has-error',\n hover: 'plyr--hover',\n tooltip: 'plyr__tooltip',\n hidden: 'plyr__sr-only',\n hideControls: 'plyr--hide-controls',\n isIos: 'plyr--is-ios',\n isTouch: 'plyr--is-touch',\n uiSupported: 'plyr--full-ui',\n noTransition: 'plyr--no-transition',\n menu: {\n value: 'plyr__menu__value',\n badge: 'plyr__badge',\n open: 'plyr--menu-open',\n },\n captions: {\n enabled: 'plyr--captions-enabled',\n active: 'plyr--captions-active',\n },\n fullscreen: {\n enabled: 'plyr--fullscreen-enabled',\n fallback: 'plyr--fullscreen-fallback',\n },\n pip: {\n supported: 'plyr--pip-supported',\n active: 'plyr--pip-active',\n },\n airplay: {\n supported: 'plyr--airplay-supported',\n active: 'plyr--airplay-active',\n },\n tabFocus: 'plyr__tab-focus',\n },\n\n // API keys\n keys: {\n google: null,\n },\n};\n\nexport default defaults;\n","// ==========================================================================\n// Console wrapper\n// ==========================================================================\n\nconst noop = () => {};\n\nexport default class Console {\n constructor(enabled = false) {\n this.enabled = window.console && enabled;\n\n if (this.enabled) {\n this.log('Debugging enabled');\n }\n }\n\n get log() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.log, console) : noop;\n }\n get warn() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.warn, console) : noop;\n }\n get error() {\n // eslint-disable-next-line no-console\n return this.enabled ? Function.prototype.bind.call(console.error, console) : noop;\n }\n}\n","// ==========================================================================\n// Plyr storage\n// ==========================================================================\n\nimport utils from './utils';\n\nclass Storage {\n constructor(player) {\n this.enabled = player.config.storage.enabled;\n this.key = player.config.storage.key;\n }\n\n // Check for actual support (see if we can use it)\n static get supported() {\n if (!('localStorage' in window)) {\n return false;\n }\n\n const test = '___test';\n\n // Try to use it (it might be disabled, e.g. user is in private mode)\n // see: https://github.com/sampotts/plyr/issues/131\n try {\n window.localStorage.setItem(test, test);\n window.localStorage.removeItem(test);\n return true;\n } catch (e) {\n return false;\n }\n }\n\n get(key) {\n const store = window.localStorage.getItem(this.key);\n\n if (!Storage.supported || utils.is.empty(store)) {\n return null;\n }\n\n const json = JSON.parse(store);\n\n return utils.is.string(key) && key.length ? json[key] : json;\n }\n\n set(object) {\n // Bail if we don't have localStorage support or it's disabled\n if (!Storage.supported || !this.enabled) {\n return;\n }\n\n // Can only store objectst\n if (!utils.is.object(object)) {\n return;\n }\n\n // Get current storage\n let storage = this.get();\n\n // Default to empty object\n if (utils.is.empty(storage)) {\n storage = {};\n }\n\n // Update the working copy of the values\n utils.extend(storage, object);\n\n // Update storage\n window.localStorage.setItem(this.key, JSON.stringify(storage));\n }\n}\n\nexport default Storage;\n","// ==========================================================================\n// Plyr Event Listeners\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport ui from './ui';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst listeners = {\n // Global listeners\n global() {\n let last = null;\n\n // Get the key code for an event\n const getKeyCode = event => (event.keyCode ? event.keyCode : event.which);\n\n // Handle key press\n const handleKey = event => {\n const code = getKeyCode(event);\n const pressed = event.type === 'keydown';\n const repeat = pressed && code === last;\n\n // Bail if a modifier key is set\n if (event.altKey || event.ctrlKey || event.metaKey || event.shiftKey) {\n return;\n }\n\n // If the event is bubbled from the media element\n // Firefox doesn't get the keycode for whatever reason\n if (!utils.is.number(code)) {\n return;\n }\n\n // Seek by the number keys\n const seekByKey = () => {\n // Divide the max duration into 10th's and times by the number value\n this.currentTime = this.duration / 10 * (code - 48);\n };\n\n // Handle the key on keydown\n // Reset on keyup\n if (pressed) {\n // Which keycodes should we prevent default\n const preventDefault = [\n 48,\n 49,\n 50,\n 51,\n 52,\n 53,\n 54,\n 56,\n 57,\n 32,\n 75,\n 38,\n 40,\n 77,\n 39,\n 37,\n 70,\n 67,\n 73,\n 76,\n 79,\n ];\n\n // Check focused element\n // and if the focused element is not editable (e.g. text input)\n // and any that accept key input http://webaim.org/techniques/keyboard/\n const focused = utils.getFocusElement();\n if (utils.is.element(focused) && utils.matches(focused, this.config.selectors.editable)) {\n return;\n }\n\n // If the code is found prevent default (e.g. prevent scrolling for arrows)\n if (preventDefault.includes(code)) {\n event.preventDefault();\n event.stopPropagation();\n }\n\n switch (code) {\n case 48:\n case 49:\n case 50:\n case 51:\n case 52:\n case 53:\n case 54:\n case 55:\n case 56:\n case 57:\n // 0-9\n if (!repeat) {\n seekByKey();\n }\n break;\n\n case 32:\n case 75:\n // Space and K key\n if (!repeat) {\n this.togglePlay();\n }\n break;\n\n case 38:\n // Arrow up\n this.increaseVolume(0.1);\n break;\n\n case 40:\n // Arrow down\n this.decreaseVolume(0.1);\n break;\n\n case 77:\n // M key\n if (!repeat) {\n this.muted = !this.muted;\n }\n break;\n\n case 39:\n // Arrow forward\n this.forward();\n break;\n\n case 37:\n // Arrow back\n this.rewind();\n break;\n\n case 70:\n // F key\n this.toggleFullscreen();\n break;\n\n case 67:\n // C key\n if (!repeat) {\n this.toggleCaptions();\n }\n break;\n\n case 76:\n // L key\n this.loop = !this.loop;\n break;\n\n /* case 73:\n this.setLoop('start');\n break;\n\n case 76:\n this.setLoop();\n break;\n\n case 79:\n this.setLoop('end');\n break; */\n\n default:\n break;\n }\n\n // Escape is handle natively when in full screen\n // So we only need to worry about non native\n if (!fullscreen.enabled && this.fullscreen.active && code === 27) {\n this.toggleFullscreen();\n }\n\n // Store last code for next cycle\n last = code;\n } else {\n last = null;\n }\n };\n\n // Keyboard shortcuts\n if (this.config.keyboard.global) {\n utils.on(window, 'keydown keyup', handleKey, false);\n } else if (this.config.keyboard.focused) {\n utils.on(this.elements.container, 'keydown keyup', handleKey, false);\n }\n\n // Detect tab focus\n // Remove class on blur/focusout\n utils.on(this.elements.container, 'focusout', event => {\n utils.toggleClass(event.target, this.config.classNames.tabFocus, false);\n });\n\n // Add classname to tabbed elements\n utils.on(this.elements.container, 'keydown', event => {\n if (event.keyCode !== 9) {\n return;\n }\n\n // Delay the adding of classname until the focus has changed\n // This event fires before the focusin event\n window.setTimeout(() => {\n utils.toggleClass(utils.getFocusElement(), this.config.classNames.tabFocus, true);\n }, 0);\n });\n\n // Toggle controls visibility based on mouse movement\n if (this.config.hideControls) {\n // Toggle controls on mouse events and entering fullscreen\n utils.on(this.elements.container, 'mouseenter mouseleave mousemove touchstart touchend touchmove enterfullscreen exitfullscreen', event => {\n this.toggleControls(event);\n });\n }\n\n // Handle user exiting fullscreen by escaping etc\n if (fullscreen.enabled) {\n utils.on(document, fullscreen.eventType, event => {\n this.toggleFullscreen(event);\n });\n\n // Fullscreen toggle on double click\n utils.on(this.elements.container, 'dblclick', event => {\n this.toggleFullscreen(event);\n });\n }\n },\n\n // Listen for media events\n media() {\n // Time change on media\n utils.on(this.media, 'timeupdate seeking', event => ui.timeUpdate.call(this, event));\n\n // Display duration\n utils.on(this.media, 'durationchange loadedmetadata', event => ui.durationUpdate.call(this, event));\n\n // Check for audio tracks on load\n // We can't use `loadedmetadata` as it doesn't seem to have audio tracks at that point\n utils.on(this.media, 'loadeddata', () => {\n utils.toggleHidden(this.elements.volume, !this.hasAudio);\n utils.toggleHidden(this.elements.buttons.mute, !this.hasAudio);\n });\n\n // Handle the media finishing\n utils.on(this.media, 'ended', () => {\n // Show poster on end\n if (this.isHTML5 && this.isVideo && this.config.showPosterOnEnd) {\n // Restart\n this.restart();\n\n // Re-load media\n this.media.load();\n }\n });\n\n // Check for buffer progress\n utils.on(this.media, 'progress playing', event => ui.updateProgress.call(this, event));\n\n // Handle native mute\n utils.on(this.media, 'volumechange', event => ui.updateVolume.call(this, event));\n\n // Handle native play/pause\n utils.on(this.media, 'playing play pause ended', event => ui.checkPlaying.call(this, event));\n\n // Loading\n utils.on(this.media, 'stalled waiting canplay seeked playing', event => ui.checkLoading.call(this, event));\n\n // Check if media failed to load\n // utils.on(this.media, 'play', event => ui.checkFailed.call(this, event));\n\n // Click video\n if (this.supported.ui && this.config.clickToPlay && !this.isAudio) {\n // Re-fetch the wrapper\n const wrapper = utils.getElement.call(this, `.${this.config.classNames.video}`);\n\n // Bail if there's no wrapper (this should never happen)\n if (!utils.is.element(wrapper)) {\n return;\n }\n\n // On click play, pause ore restart\n utils.on(wrapper, 'click', () => {\n // Touch devices will just show controls (if we're hiding controls)\n if (this.config.hideControls && support.touch && !this.paused) {\n return;\n }\n\n if (this.paused) {\n this.play();\n } else if (this.ended) {\n this.restart();\n this.play();\n } else {\n this.pause();\n }\n });\n }\n\n // Disable right click\n if (this.supported.ui && this.config.disableContextMenu) {\n utils.on(\n this.media,\n 'contextmenu',\n event => {\n event.preventDefault();\n },\n false\n );\n }\n\n // Volume change\n utils.on(this.media, 'volumechange', () => {\n // Save to storage\n this.storage.set({ volume: this.volume, muted: this.muted });\n });\n\n // Speed change\n utils.on(this.media, 'ratechange', () => {\n // Update UI\n controls.updateSetting.call(this, 'speed');\n\n // Save to storage\n this.storage.set({ speed: this.speed });\n });\n\n // Quality change\n utils.on(this.media, 'qualitychange', () => {\n // Update UI\n controls.updateSetting.call(this, 'quality');\n\n // Save to storage\n this.storage.set({ quality: this.quality });\n });\n\n // Caption language change\n utils.on(this.media, 'languagechange', () => {\n // Update UI\n controls.updateSetting.call(this, 'captions');\n\n // Save to storage\n this.storage.set({ language: this.language });\n });\n\n // Captions toggle\n utils.on(this.media, 'captionsenabled captionsdisabled', () => {\n // Update UI\n controls.updateSetting.call(this, 'captions');\n\n // Save to storage\n this.storage.set({ captions: this.captions.active });\n });\n\n // Proxy events to container\n // Bubble up key events for Edge\n utils.on(this.media, this.config.events.concat([\n 'keyup',\n 'keydown',\n ]).join(' '), event => {\n let detail = {};\n\n // Get error details from media\n if (event.type === 'error') {\n detail = this.media.error;\n }\n\n utils.dispatchEvent.call(this, this.elements.container, event.type, true, detail);\n });\n },\n\n // Listen for control events\n controls() {\n // IE doesn't support input event, so we fallback to change\n const inputEvent = browser.isIE ? 'change' : 'input';\n\n // Trigger custom and default handlers\n const proxy = (event, handlerKey, defaultHandler) => {\n const customHandler = this.config.listeners[handlerKey];\n\n // Execute custom handler\n if (utils.is.function(customHandler)) {\n customHandler.call(this, event);\n }\n\n // Only call default handler if not prevented in custom handler\n if (!event.defaultPrevented && utils.is.function(defaultHandler)) {\n defaultHandler.call(this, event);\n }\n };\n\n // Play/pause toggle\n utils.on(this.elements.buttons.play, 'click', event =>\n proxy(event, 'play', () => {\n this.togglePlay();\n })\n );\n\n // Pause\n utils.on(this.elements.buttons.restart, 'click', event =>\n proxy(event, 'restart', () => {\n this.restart();\n })\n );\n\n // Rewind\n utils.on(this.elements.buttons.rewind, 'click', event =>\n proxy(event, 'rewind', () => {\n this.rewind();\n })\n );\n\n // Rewind\n utils.on(this.elements.buttons.forward, 'click', event =>\n proxy(event, 'forward', () => {\n this.forward();\n })\n );\n\n // Mute toggle\n utils.on(this.elements.buttons.mute, 'click', event =>\n proxy(event, 'mute', () => {\n this.muted = !this.muted;\n })\n );\n\n // Captions toggle\n utils.on(this.elements.buttons.captions, 'click', event =>\n proxy(event, 'captions', () => {\n this.toggleCaptions();\n })\n );\n\n // Fullscreen toggle\n utils.on(this.elements.buttons.fullscreen, 'click', event =>\n proxy(event, 'fullscreen', () => {\n this.toggleFullscreen();\n })\n );\n\n // Picture-in-Picture\n utils.on(this.elements.buttons.pip, 'click', event =>\n proxy(event, 'pip', () => {\n this.pip = 'toggle';\n })\n );\n\n // Airplay\n utils.on(this.elements.buttons.airplay, 'click', event =>\n proxy(event, 'airplay', () => {\n this.airplay();\n })\n );\n\n // Settings menu\n utils.on(this.elements.buttons.settings, 'click', event => {\n controls.toggleMenu.call(this, event);\n });\n\n // Click anywhere closes menu\n utils.on(document.documentElement, 'click', event => {\n controls.toggleMenu.call(this, event);\n });\n\n // Settings menu\n utils.on(this.elements.settings.form, 'click', event => {\n event.stopPropagation();\n\n // Settings menu items - use event delegation as items are added/removed\n if (utils.matches(event.target, this.config.selectors.inputs.language)) {\n proxy(event, 'language', () => {\n this.language = event.target.value;\n });\n } else if (utils.matches(event.target, this.config.selectors.inputs.quality)) {\n proxy(event, 'quality', () => {\n this.quality = event.target.value;\n });\n } else if (utils.matches(event.target, this.config.selectors.inputs.speed)) {\n proxy(event, 'speed', () => {\n this.speed = parseFloat(event.target.value);\n });\n } else {\n controls.showTab.call(this, event);\n }\n });\n\n // Seek\n utils.on(this.elements.inputs.seek, inputEvent, event =>\n proxy(event, 'seek', () => {\n this.currentTime = event.target.value / event.target.max * this.duration;\n })\n );\n\n // Current time invert\n // Only if one time element is used for both currentTime and duration\n if (this.config.toggleInvert && !utils.is.element(this.elements.display.duration)) {\n utils.on(this.elements.display.currentTime, 'click', () => {\n // Do nothing if we're at the start\n if (this.currentTime === 0) {\n return;\n }\n\n this.config.invertTime = !this.config.invertTime;\n ui.timeUpdate.call(this);\n });\n }\n\n // Volume\n utils.on(this.elements.inputs.volume, inputEvent, event =>\n proxy(event, 'volume', () => {\n this.volume = event.target.value;\n })\n );\n\n // Polyfill for lower fill in <input type=\"range\"> for webkit\n if (browser.isWebkit) {\n utils.on(utils.getElements.call(this, 'input[type=\"range\"]'), 'input', event => {\n controls.updateRangeFill.call(this, event.target);\n });\n }\n\n // Seek tooltip\n utils.on(this.elements.progress, 'mouseenter mouseleave mousemove', event => controls.updateSeekTooltip.call(this, event));\n\n // Toggle controls visibility based on mouse movement\n if (this.config.hideControls) {\n // Watch for cursor over controls so they don't hide when trying to interact\n utils.on(this.elements.controls, 'mouseenter mouseleave', event => {\n this.elements.controls.hover = event.type === 'mouseenter';\n });\n\n // Watch for cursor over controls so they don't hide when trying to interact\n utils.on(this.elements.controls, 'mousedown mouseup touchstart touchend touchcancel', event => {\n this.elements.controls.pressed = [\n 'mousedown',\n 'touchstart',\n ].includes(event.type);\n });\n\n // Focus in/out on controls\n utils.on(this.elements.controls, 'focusin focusout', event => {\n this.toggleControls(event);\n });\n }\n\n // Mouse wheel for volume\n utils.on(\n this.elements.inputs.volume,\n 'wheel',\n event =>\n proxy(event, 'volume', () => {\n // Detect \"natural\" scroll - suppored on OS X Safari only\n // Other browsers on OS X will be inverted until support improves\n const inverted = event.webkitDirectionInvertedFromDevice;\n const step = 1 / 50;\n let direction = 0;\n\n // Scroll down (or up on natural) to decrease\n if (event.deltaY < 0 || event.deltaX > 0) {\n if (inverted) {\n this.decreaseVolume(step);\n direction = -1;\n } else {\n this.increaseVolume(step);\n direction = 1;\n }\n }\n\n // Scroll up (or down on natural) to increase\n if (event.deltaY > 0 || event.deltaX < 0) {\n if (inverted) {\n this.increaseVolume(step);\n direction = 1;\n } else {\n this.decreaseVolume(step);\n direction = -1;\n }\n }\n\n // Don't break page scrolling at max and min\n if ((direction === 1 && this.media.volume < 1) || (direction === -1 && this.media.volume > 0)) {\n event.preventDefault();\n }\n }),\n false\n );\n },\n};\n\nexport default listeners;\n","// ==========================================================================\n// Plyr UI\n// ==========================================================================\n\nimport utils from './utils';\nimport captions from './captions';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport listeners from './listeners';\n\nconst ui = {\n addStyleHook() {\n utils.toggleClass(this.elements.container, this.config.selectors.container.replace('.', ''), true);\n utils.toggleClass(this.elements.container, this.config.classNames.uiSupported, this.supported.ui);\n },\n\n // Toggle native HTML5 media controls\n toggleNativeControls(toggle = false) {\n if (toggle && this.isHTML5) {\n this.media.setAttribute('controls', '');\n } else {\n this.media.removeAttribute('controls');\n }\n },\n\n // Setup the UI\n build() {\n // Re-attach media element listeners\n // TODO: Use event bubbling\n listeners.media.call(this);\n\n // Don't setup interface if no support\n if (!this.supported.ui) {\n this.debug.warn(`Basic support only for ${this.provider} ${this.type}`);\n\n // Remove controls\n utils.removeElement.call(this, 'controls');\n\n // Remove large play\n utils.removeElement.call(this, 'buttons.play');\n\n // Restore native controls\n ui.toggleNativeControls.call(this, true);\n\n // Bail\n return;\n }\n\n // Inject custom controls if not present\n if (!utils.is.element(this.elements.controls)) {\n // Inject custom controls\n controls.inject.call(this);\n\n // Re-attach control listeners\n listeners.controls.call(this);\n }\n\n // If there's no controls, bail\n if (!utils.is.element(this.elements.controls)) {\n return;\n }\n\n // Remove native controls\n ui.toggleNativeControls.call(this);\n\n // Setup fullscreen\n fullscreen.setup.call(this);\n\n // Captions\n captions.setup.call(this);\n\n // Reset volume\n this.volume = null;\n\n // Reset mute state\n this.muted = null;\n\n // Reset speed\n this.speed = null;\n\n // Reset loop state\n this.loop = null;\n\n // Reset quality options\n this.options.quality = [];\n\n // Reset time display\n ui.timeUpdate.call(this);\n\n // Update the UI\n ui.checkPlaying.call(this);\n\n // Ready for API calls\n this.ready = true;\n\n // Ready event at end of execution stack\n utils.dispatchEvent.call(this, this.media, 'ready');\n\n // Set the title\n ui.setTitle.call(this);\n },\n\n // Setup aria attribute for play and iframe title\n setTitle() {\n // Find the current text\n let label = this.config.i18n.play;\n\n // If there's a media title set, use that for the label\n if (utils.is.string(this.config.title) && !utils.is.empty(this.config.title)) {\n label += `, ${this.config.title}`;\n\n // Set container label\n this.elements.container.setAttribute('aria-label', this.config.title);\n }\n\n // If there's a play button, set label\n if (utils.is.nodeList(this.elements.buttons.play)) {\n Array.from(this.elements.buttons.play).forEach(button => {\n button.setAttribute('aria-label', label);\n });\n }\n\n // Set iframe title\n // https://github.com/sampotts/plyr/issues/124\n if (this.isEmbed) {\n const iframe = utils.getElement.call(this, 'iframe');\n\n if (!utils.is.element(iframe)) {\n return;\n }\n\n // Default to media type\n const title = !utils.is.empty(this.config.title) ? this.config.title : 'video';\n\n iframe.setAttribute('title', this.config.i18n.frameTitle.replace('{title}', title));\n }\n },\n\n // Check playing state\n checkPlaying() {\n // Class hooks\n utils.toggleClass(this.elements.container, this.config.classNames.playing, this.playing);\n utils.toggleClass(this.elements.container, this.config.classNames.stopped, this.paused);\n\n // Set aria state\n if (utils.is.nodeList(this.elements.buttons.play)) {\n Array.from(this.elements.buttons.play).forEach(button => utils.toggleState(button, this.playing));\n }\n\n // Toggle controls\n this.toggleControls(!this.playing);\n },\n\n // Check if media is loading\n checkLoading(event) {\n this.loading = [\n 'stalled',\n 'waiting',\n ].includes(event.type);\n\n // Clear timer\n clearTimeout(this.timers.loading);\n\n // Timer to prevent flicker when seeking\n this.timers.loading = setTimeout(() => {\n // Toggle container class hook\n utils.toggleClass(this.elements.container, this.config.classNames.loading, this.loading);\n\n // Show controls if loading, hide if done\n this.toggleControls(this.loading);\n }, this.loading ? 250 : 0);\n },\n\n // Check if media failed to load\n checkFailed() {\n // https://developer.mozilla.org/en-US/docs/Web/API/HTMLMediaElement/networkState\n this.failed = this.media.networkState === 3;\n\n if (this.failed) {\n utils.toggleClass(this.elements.container, this.config.classNames.loading, false);\n utils.toggleClass(this.elements.container, this.config.classNames.error, true);\n }\n\n // Clear timer\n clearTimeout(this.timers.failed);\n\n // Timer to prevent flicker when seeking\n this.timers.loading = setTimeout(() => {\n // Toggle container class hook\n utils.toggleClass(this.elements.container, this.config.classNames.loading, this.loading);\n\n // Show controls if loading, hide if done\n this.toggleControls(this.loading);\n }, this.loading ? 250 : 0);\n },\n\n // Update volume UI and storage\n updateVolume() {\n if (!this.supported.ui) {\n return;\n }\n\n // Update range\n if (utils.is.element(this.elements.inputs.volume)) {\n ui.setRange.call(this, this.elements.inputs.volume, this.muted ? 0 : this.volume);\n }\n\n // Update mute state\n if (utils.is.element(this.elements.buttons.mute)) {\n utils.toggleState(this.elements.buttons.mute, this.muted || this.volume === 0);\n }\n },\n\n // Update seek value and lower fill\n setRange(target, value = 0) {\n if (!utils.is.element(target)) {\n return;\n }\n\n // eslint-disable-next-line\n target.value = value;\n\n // Webkit range fill\n controls.updateRangeFill.call(this, target);\n },\n\n // Set <progress> value\n setProgress(target, input) {\n const value = utils.is.number(input) ? input : 0;\n const progress = utils.is.element(target) ? target : this.elements.display.buffer;\n\n // Update value and label\n if (utils.is.element(progress)) {\n progress.value = value;\n\n // Update text label inside\n const label = progress.getElementsByTagName('span')[0];\n if (utils.is.element(label)) {\n label.childNodes[0].nodeValue = value;\n }\n }\n },\n\n // Update <progress> elements\n updateProgress(event) {\n if (!this.supported.ui || !utils.is.event(event)) {\n return;\n }\n\n let value = 0;\n\n if (event) {\n switch (event.type) {\n // Video playing\n case 'timeupdate':\n case 'seeking':\n value = utils.getPercentage(this.currentTime, this.duration);\n\n // Set seek range value only if it's a 'natural' time event\n if (event.type === 'timeupdate') {\n ui.setRange.call(this, this.elements.inputs.seek, value);\n }\n\n break;\n\n // Check buffer status\n case 'playing':\n case 'progress':\n value = (() => {\n const { buffered } = this.media;\n\n if (buffered && buffered.length) {\n // HTML5\n return utils.getPercentage(buffered.end(0), this.duration);\n } else if (utils.is.number(buffered)) {\n // YouTube returns between 0 and 1\n return buffered * 100;\n }\n\n return 0;\n })();\n\n ui.setProgress.call(this, this.elements.display.buffer, value);\n\n break;\n\n default:\n break;\n }\n }\n },\n\n // Update the displayed time\n updateTimeDisplay(target = null, time = 0, inverted = false) {\n // Bail if there's no element to display or the value isn't a number\n if (!utils.is.element(target) || !utils.is.number(time)) {\n return;\n }\n\n // Format time component to add leading zero\n const format = value => `0${value}`.slice(-2);\n\n // Helpers\n const getHours = value => parseInt((value / 60 / 60) % 60, 10);\n const getMinutes = value => parseInt((value / 60) % 60, 10);\n const getSeconds = value => parseInt(value % 60, 10);\n\n // Breakdown to hours, mins, secs\n let hours = getHours(time);\n const mins = getMinutes(time);\n const secs = getSeconds(time);\n\n // Do we need to display hours?\n if (getHours(this.duration) > 0) {\n hours = `${hours}:`;\n } else {\n hours = '';\n }\n\n // Render\n // eslint-disable-next-line no-param-reassign\n target.textContent = `${inverted ? '-' : ''}${hours}${format(mins)}:${format(secs)}`;\n },\n\n // Handle time change event\n timeUpdate(event) {\n // Only invert if only one time element is displayed and used for both duration and currentTime\n const invert = !utils.is.element(this.elements.display.duration) && this.config.invertTime;\n\n // Duration\n ui.updateTimeDisplay.call(this, this.elements.display.currentTime, invert ? this.duration - this.currentTime : this.currentTime, invert);\n\n // Ignore updates while seeking\n if (event && event.type === 'timeupdate' && this.media.seeking) {\n return;\n }\n\n // Playing progress\n ui.updateProgress.call(this, event);\n },\n\n // Show the duration on metadataloaded\n durationUpdate() {\n if (!this.supported.ui) {\n return;\n }\n\n // If there's only one time display, display duration there\n if (!utils.is.element(this.elements.display.duration) && this.config.displayDuration && this.paused) {\n ui.updateTimeDisplay.call(this, this.elements.display.currentTime, this.duration);\n }\n\n // If there's a duration element, update content\n if (utils.is.element(this.elements.display.duration)) {\n ui.updateTimeDisplay.call(this, this.elements.display.duration, this.duration);\n }\n\n // Update the tooltip (if visible)\n controls.updateSeekTooltip.call(this);\n },\n};\n\nexport default ui;\n","// ==========================================================================\n// Plyr controls\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport ui from './ui';\nimport captions from './captions';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst controls = {\n // Webkit polyfill for lower fill range\n updateRangeFill(target) {\n // WebKit only\n if (!browser.isWebkit) {\n return;\n }\n\n // Get range from event if event passed\n const range = utils.is.event(target) ? target.target : target;\n\n // Needs to be a valid <input type='range'>\n if (!utils.is.element(range) || range.getAttribute('type') !== 'range') {\n return;\n }\n\n // Set CSS custom property\n range.style.setProperty('--value', `${range.value / range.max * 100}%`);\n },\n\n // Get icon URL\n getIconUrl() {\n return {\n url: this.config.iconUrl,\n absolute: this.config.iconUrl.indexOf('http') === 0 || (browser.isIE && !window.svg4everybody),\n };\n },\n\n // Create <svg> icon\n createIcon(type, attributes) {\n const namespace = 'http://www.w3.org/2000/svg';\n const iconUrl = controls.getIconUrl.call(this);\n const iconPath = `${!iconUrl.absolute ? iconUrl.url : ''}#${this.config.iconPrefix}`;\n\n // Create <svg>\n const icon = document.createElementNS(namespace, 'svg');\n utils.setAttributes(\n icon,\n utils.extend(attributes, {\n role: 'presentation',\n })\n );\n\n // Create the <use> to reference sprite\n const use = document.createElementNS(namespace, 'use');\n const path = `${iconPath}-${type}`;\n\n // Set `href` attributes\n // https://github.com/sampotts/plyr/issues/460\n // https://developer.mozilla.org/en-US/docs/Web/SVG/Attribute/xlink:href\n if ('href' in use) {\n use.setAttributeNS('http://www.w3.org/1999/xlink', 'href', path);\n } else {\n use.setAttributeNS('http://www.w3.org/1999/xlink', 'xlink:href', path);\n }\n\n // Add <use> to <svg>\n icon.appendChild(use);\n\n return icon;\n },\n\n // Create hidden text label\n createLabel(type, attr) {\n let text = this.config.i18n[type];\n const attributes = Object.assign({}, attr);\n\n switch (type) {\n case 'pip':\n text = 'PIP';\n break;\n\n case 'airplay':\n text = 'AirPlay';\n break;\n\n default:\n break;\n }\n\n if ('class' in attributes) {\n attributes.class += ` ${this.config.classNames.hidden}`;\n } else {\n attributes.class = this.config.classNames.hidden;\n }\n\n return utils.createElement('span', attributes, text);\n },\n\n // Create a badge\n createBadge(text) {\n if (utils.is.empty(text)) {\n return null;\n }\n\n const badge = utils.createElement('span', {\n class: this.config.classNames.menu.value,\n });\n\n badge.appendChild(\n utils.createElement(\n 'span',\n {\n class: this.config.classNames.menu.badge,\n },\n text\n )\n );\n\n return badge;\n },\n\n // Create a <button>\n createButton(buttonType, attr) {\n const button = utils.createElement('button');\n const attributes = Object.assign({}, attr);\n let type = buttonType;\n\n let toggle = false;\n let label;\n let icon;\n let labelPressed;\n let iconPressed;\n\n if (!('type' in attributes)) {\n attributes.type = 'button';\n }\n\n if ('class' in attributes) {\n if (attributes.class.includes(this.config.classNames.control)) {\n attributes.class += ` ${this.config.classNames.control}`;\n }\n } else {\n attributes.class = this.config.classNames.control;\n }\n\n // Large play button\n switch (type) {\n case 'play':\n toggle = true;\n label = 'play';\n labelPressed = 'pause';\n icon = 'play';\n iconPressed = 'pause';\n break;\n\n case 'mute':\n toggle = true;\n label = 'mute';\n labelPressed = 'unmute';\n icon = 'volume';\n iconPressed = 'muted';\n break;\n\n case 'captions':\n toggle = true;\n label = 'enableCaptions';\n labelPressed = 'disableCaptions';\n icon = 'captions-off';\n iconPressed = 'captions-on';\n break;\n\n case 'fullscreen':\n toggle = true;\n label = 'enterFullscreen';\n labelPressed = 'exitFullscreen';\n icon = 'enter-fullscreen';\n iconPressed = 'exit-fullscreen';\n break;\n\n case 'play-large':\n attributes.class += ` ${this.config.classNames.control}--overlaid`;\n type = 'play';\n label = 'play';\n icon = 'play';\n break;\n\n default:\n label = type;\n icon = type;\n }\n\n // Setup toggle icon and labels\n if (toggle) {\n // Icon\n button.appendChild(controls.createIcon.call(this, iconPressed, { class: 'icon--pressed' }));\n button.appendChild(controls.createIcon.call(this, icon, { class: 'icon--not-pressed' }));\n\n // Label/Tooltip\n button.appendChild(controls.createLabel.call(this, labelPressed, { class: 'label--pressed' }));\n button.appendChild(controls.createLabel.call(this, label, { class: 'label--not-pressed' }));\n\n // Add aria attributes\n attributes['aria-pressed'] = false;\n attributes['aria-label'] = this.config.i18n[label];\n } else {\n button.appendChild(controls.createIcon.call(this, icon));\n button.appendChild(controls.createLabel.call(this, label));\n }\n\n // Merge attributes\n utils.extend(attributes, utils.getAttributesFromSelector(this.config.selectors.buttons[type], attributes));\n\n utils.setAttributes(button, attributes);\n\n this.elements.buttons[type] = button;\n\n return button;\n },\n\n // Create an <input type='range'>\n createRange(type, attributes) {\n // Seek label\n const label = utils.createElement(\n 'label',\n {\n for: attributes.id,\n class: this.config.classNames.hidden,\n },\n this.config.i18n[type]\n );\n\n // Seek input\n const input = utils.createElement(\n 'input',\n utils.extend(\n utils.getAttributesFromSelector(this.config.selectors.inputs[type]),\n {\n type: 'range',\n min: 0,\n max: 100,\n step: 0.01,\n value: 0,\n autocomplete: 'off',\n },\n attributes\n )\n );\n\n this.elements.inputs[type] = input;\n\n // Set the fill for webkit now\n controls.updateRangeFill.call(this, input);\n\n return {\n label,\n input,\n };\n },\n\n // Create a <progress>\n createProgress(type, attributes) {\n const progress = utils.createElement(\n 'progress',\n utils.extend(\n utils.getAttributesFromSelector(this.config.selectors.display[type]),\n {\n min: 0,\n max: 100,\n value: 0,\n },\n attributes\n )\n );\n\n // Create the label inside\n if (type !== 'volume') {\n progress.appendChild(utils.createElement('span', null, '0'));\n\n let suffix = '';\n switch (type) {\n case 'played':\n suffix = this.config.i18n.played;\n break;\n\n case 'buffer':\n suffix = this.config.i18n.buffered;\n break;\n\n default:\n break;\n }\n\n progress.textContent = `% ${suffix.toLowerCase()}`;\n }\n\n this.elements.display[type] = progress;\n\n return progress;\n },\n\n // Create time display\n createTime(type) {\n const container = utils.createElement('div', {\n class: 'plyr__time',\n });\n\n container.appendChild(\n utils.createElement(\n 'span',\n {\n class: this.config.classNames.hidden,\n },\n this.config.i18n[type]\n )\n );\n\n container.appendChild(utils.createElement('span', utils.getAttributesFromSelector(this.config.selectors.display[type]), '00:00'));\n\n this.elements.display[type] = container;\n\n return container;\n },\n\n // Create a settings menu item\n createMenuItem(value, list, type, title, badge = null, checked = false) {\n const item = utils.createElement('li');\n\n const label = utils.createElement('label', {\n class: this.config.classNames.control,\n });\n\n const radio = utils.createElement(\n 'input',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.inputs[type]), {\n type: 'radio',\n name: `plyr-${type}`,\n value,\n checked,\n class: 'plyr__sr-only',\n })\n );\n\n const faux = utils.createElement('span', { 'aria-hidden': true });\n\n label.appendChild(radio);\n label.appendChild(faux);\n label.insertAdjacentHTML('beforeend', title);\n\n if (utils.is.element(badge)) {\n label.appendChild(badge);\n }\n\n item.appendChild(label);\n list.appendChild(item);\n },\n\n // Update hover tooltip for seeking\n updateSeekTooltip(event) {\n // Bail if setting not true\n if (\n !this.config.tooltips.seek ||\n !utils.is.element(this.elements.inputs.seek) ||\n !utils.is.element(this.elements.display.seekTooltip) ||\n this.duration === 0\n ) {\n return;\n }\n\n // Calculate percentage\n let percent = 0;\n const clientRect = this.elements.inputs.seek.getBoundingClientRect();\n const visible = `${this.config.classNames.tooltip}--visible`;\n\n // Determine percentage, if already visible\n if (utils.is.event(event)) {\n percent = 100 / clientRect.width * (event.pageX - clientRect.left);\n } else if (utils.hasClass(this.elements.display.seekTooltip, visible)) {\n percent = parseFloat(this.elements.display.seekTooltip.style.left, 10);\n } else {\n return;\n }\n\n // Set bounds\n if (percent < 0) {\n percent = 0;\n } else if (percent > 100) {\n percent = 100;\n }\n\n // Display the time a click would seek to\n ui.updateTimeDisplay.call(this, this.elements.display.seekTooltip, this.duration / 100 * percent);\n\n // Set position\n this.elements.display.seekTooltip.style.left = `${percent}%`;\n\n // Show/hide the tooltip\n // If the event is a moues in/out and percentage is inside bounds\n if (utils.is.event(event) && [\n 'mouseenter',\n 'mouseleave',\n ].includes(event.type)) {\n utils.toggleClass(this.elements.display.seekTooltip, visible, event.type === 'mouseenter');\n }\n },\n\n // Hide/show a tab\n toggleTab(setting, toggle) {\n const tab = this.elements.settings.tabs[setting];\n const pane = this.elements.settings.panes[setting];\n\n utils.toggleHidden(tab, !toggle);\n utils.toggleHidden(pane, !toggle);\n },\n\n // Set the YouTube quality menu\n // TODO: Support for HTML5\n setQualityMenu(options) {\n const type = 'quality';\n const list = this.elements.settings.panes.quality.querySelector('ul');\n\n // Set options if passed and filter based on config\n if (utils.is.array(options)) {\n this.options.quality = options.filter(quality => this.config.quality.options.includes(quality));\n } else {\n this.options.quality = this.config.quality.options;\n }\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.options.quality) && this.isYouTube;\n controls.toggleTab.call(this, type, toggle);\n\n // If we're hiding, nothing more to do\n if (!toggle) {\n return;\n }\n\n // Empty the menu\n utils.emptyElement(list);\n\n // Get the badge HTML for HD, 4K etc\n const getBadge = quality => {\n let label = '';\n\n switch (quality) {\n case 'hd2160':\n label = '4K';\n break;\n\n case 'hd1440':\n label = 'WQHD';\n break;\n\n case 'hd1080':\n label = 'HD';\n break;\n\n case 'hd720':\n label = 'HD';\n break;\n\n default:\n break;\n }\n\n if (!label.length) {\n return null;\n }\n\n return controls.createBadge.call(this, label);\n };\n\n this.options.quality.forEach(quality =>\n controls.createMenuItem.call(this, quality, list, type, controls.getLabel.call(this, 'quality', quality), getBadge(quality))\n );\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Translate a value into a nice label\n // TODO: Localisation\n getLabel(setting, value) {\n switch (setting) {\n case 'speed':\n return value === 1 ? 'Normal' : `${value}×`;\n\n case 'quality':\n switch (value) {\n case 'hd2160':\n return '2160P';\n case 'hd1440':\n return '1440P';\n case 'hd1080':\n return '1080P';\n case 'hd720':\n return '720P';\n case 'large':\n return '480P';\n case 'medium':\n return '360P';\n case 'small':\n return '240P';\n case 'tiny':\n return 'Tiny';\n case 'default':\n return 'Auto';\n default:\n return value;\n }\n\n case 'captions':\n return controls.getLanguage.call(this);\n\n default:\n return null;\n }\n },\n\n // Update the selected setting\n updateSetting(setting, container) {\n const pane = this.elements.settings.panes[setting];\n let value = null;\n let list = container;\n\n switch (setting) {\n case 'captions':\n value = this.captions.active ? this.captions.language : '';\n break;\n\n default:\n value = this[setting];\n\n // Get default\n if (utils.is.empty(value)) {\n value = this.config[setting].default;\n }\n\n // Unsupported value\n if (!this.options[setting].includes(value)) {\n this.debug.warn(`Unsupported value of '${value}' for ${setting}`);\n return;\n }\n\n // Disabled value\n if (!this.config[setting].options.includes(value)) {\n this.debug.warn(`Disabled value of '${value}' for ${setting}`);\n return;\n }\n\n break;\n }\n\n // Get the list if we need to\n if (!utils.is.element(list)) {\n list = pane && pane.querySelector('ul');\n }\n\n // Update the label\n if (!utils.is.empty(value)) {\n const label = this.elements.settings.tabs[setting].querySelector(`.${this.config.classNames.menu.value}`);\n label.innerHTML = controls.getLabel.call(this, setting, value);\n }\n\n // Find the radio option\n const target = list && list.querySelector(`input[value=\"${value}\"]`);\n\n if (utils.is.element(target)) {\n // Check it\n target.checked = true;\n }\n },\n\n // Set the looping options\n /* setLoopMenu() {\n const options = ['start', 'end', 'all', 'reset'];\n const list = this.elements.settings.panes.loop.querySelector('ul');\n\n // Show the pane and tab\n utils.toggleHidden(this.elements.settings.tabs.loop, false);\n utils.toggleHidden(this.elements.settings.panes.loop, false);\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.loop.options);\n controls.toggleTab.call(this, 'loop', toggle);\n\n // Empty the menu\n utils.emptyElement(list);\n\n options.forEach(option => {\n const item = utils.createElement('li');\n\n const button = utils.createElement(\n 'button',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.buttons.loop), {\n type: 'button',\n class: this.config.classNames.control,\n 'data-plyr-loop-action': option,\n }),\n this.config.i18n[option]\n );\n\n if (['start', 'end'].includes(option)) {\n const badge = controls.createBadge.call(this, '00:00');\n button.appendChild(badge);\n }\n\n item.appendChild(button);\n list.appendChild(item);\n });\n }, */\n\n // Get current selected caption language\n // TODO: rework this to user the getter in the API?\n getLanguage() {\n if (!this.supported.ui) {\n return null;\n }\n\n if (!support.textTracks || !captions.getTracks.call(this).length) {\n return this.config.i18n.none;\n }\n\n if (this.captions.active) {\n const currentTrack = captions.getCurrentTrack.call(this);\n\n if (utils.is.track(currentTrack)) {\n return currentTrack.label;\n }\n }\n\n return this.config.i18n.disabled;\n },\n\n // Set a list of available captions languages\n setCaptionsMenu() {\n // TODO: Captions or language? Currently it's mixed\n const type = 'captions';\n const list = this.elements.settings.panes.captions.querySelector('ul');\n\n // Toggle the pane and tab\n const hasTracks = captions.getTracks.call(this).length;\n controls.toggleTab.call(this, type, hasTracks);\n\n // Empty the menu\n utils.emptyElement(list);\n\n // If there's no captions, bail\n if (!hasTracks) {\n return;\n }\n\n // Re-map the tracks into just the data we need\n const tracks = captions.getTracks.call(this).map(track => ({\n language: track.language,\n label: !utils.is.empty(track.label) ? track.label : track.language.toUpperCase(),\n }));\n\n // Add the \"None\" option to turn off captions\n tracks.unshift({\n language: '',\n label: this.config.i18n.none,\n });\n\n // Generate options\n tracks.forEach(track => {\n controls.createMenuItem.call(\n this,\n track.language,\n list,\n 'language',\n track.label || track.language,\n controls.createBadge.call(this, track.language.toUpperCase()),\n track.language.toLowerCase() === this.captions.language.toLowerCase()\n );\n });\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Set a list of available captions languages\n setSpeedMenu() {\n const type = 'speed';\n\n // Set the default speeds\n if (!utils.is.object(this.options.speed) || !Object.keys(this.options.speed).length) {\n this.options.speed = [\n 0.5,\n 0.75,\n 1,\n 1.25,\n 1.5,\n 1.75,\n 2,\n ];\n }\n\n // Set options if passed and filter based on config\n this.options.speed = this.options.speed.filter(speed => this.config.speed.options.includes(speed));\n\n // Toggle the pane and tab\n const toggle = !utils.is.empty(this.options.speed);\n controls.toggleTab.call(this, type, toggle);\n\n // If we're hiding, nothing more to do\n if (!toggle) {\n return;\n }\n\n // Get the list to populate\n const list = this.elements.settings.panes.speed.querySelector('ul');\n\n // Show the pane and tab\n utils.toggleHidden(this.elements.settings.tabs.speed, false);\n utils.toggleHidden(this.elements.settings.panes.speed, false);\n\n // Empty the menu\n utils.emptyElement(list);\n\n // Create items\n this.options.speed.forEach(speed => controls.createMenuItem.call(this, speed, list, type, controls.getLabel.call(this, 'speed', speed)));\n\n controls.updateSetting.call(this, type, list);\n },\n\n // Show/hide menu\n toggleMenu(event) {\n const { form } = this.elements.settings;\n const button = this.elements.buttons.settings;\n const show = utils.is.boolean(event) ? event : utils.is.element(form) && form.getAttribute('aria-hidden') === 'true';\n\n if (utils.is.event(event)) {\n const isMenuItem = utils.is.element(form) && form.contains(event.target);\n const isButton = event.target === this.elements.buttons.settings;\n\n // If the click was inside the form or if the click\n // wasn't the button or menu item and we're trying to\n // show the menu (a doc click shouldn't show the menu)\n if (isMenuItem || (!isMenuItem && !isButton && show)) {\n return;\n }\n\n // Prevent the toggle being caught by the doc listener\n if (isButton) {\n event.stopPropagation();\n }\n }\n\n // Set form and button attributes\n if (utils.is.element(button)) {\n button.setAttribute('aria-expanded', show);\n }\n\n if (utils.is.element(form)) {\n form.setAttribute('aria-hidden', !show);\n utils.toggleClass(this.elements.container, this.config.classNames.menu.open, show);\n\n if (show) {\n form.removeAttribute('tabindex');\n } else {\n form.setAttribute('tabindex', -1);\n }\n }\n },\n\n // Get the natural size of a tab\n getTabSize(tab) {\n const clone = tab.cloneNode(true);\n clone.style.position = 'absolute';\n clone.style.opacity = 0;\n clone.setAttribute('aria-hidden', false);\n\n // Prevent input's being unchecked due to the name being identical\n Array.from(clone.querySelectorAll('input[name]')).forEach(input => {\n const name = input.getAttribute('name');\n input.setAttribute('name', `${name}-clone`);\n });\n\n // Append to parent so we get the \"real\" size\n tab.parentNode.appendChild(clone);\n\n // Get the sizes before we remove\n const width = clone.scrollWidth;\n const height = clone.scrollHeight;\n\n // Remove from the DOM\n utils.removeElement(clone);\n\n return {\n width,\n height,\n };\n },\n\n // Toggle Menu\n showTab(event) {\n const { menu } = this.elements.settings;\n const tab = event.target;\n const show = tab.getAttribute('aria-expanded') === 'false';\n const pane = document.getElementById(tab.getAttribute('aria-controls'));\n\n // Nothing to show, bail\n if (!utils.is.element(pane)) {\n return;\n }\n\n // Are we targetting a tab? If not, bail\n const isTab = pane.getAttribute('role') === 'tabpanel';\n if (!isTab) {\n return;\n }\n\n // Hide all other tabs\n // Get other tabs\n const current = menu.querySelector('[role=\"tabpanel\"][aria-hidden=\"false\"]');\n const container = current.parentNode;\n\n // Set other toggles to be expanded false\n Array.from(menu.querySelectorAll(`[aria-controls=\"${current.getAttribute('id')}\"]`)).forEach(toggle => {\n toggle.setAttribute('aria-expanded', false);\n });\n\n // If we can do fancy animations, we'll animate the height/width\n if (support.transitions && !support.reducedMotion) {\n // Set the current width as a base\n container.style.width = `${current.scrollWidth}px`;\n container.style.height = `${current.scrollHeight}px`;\n\n // Get potential sizes\n const size = controls.getTabSize.call(this, pane);\n\n // Restore auto height/width\n const restore = e => {\n // We're only bothered about height and width on the container\n if (e.target !== container || ![\n 'width',\n 'height',\n ].includes(e.propertyName)) {\n return;\n }\n\n // Revert back to auto\n container.style.width = '';\n container.style.height = '';\n\n // Only listen once\n utils.off(container, utils.transitionEnd, restore);\n };\n\n // Listen for the transition finishing and restore auto height/width\n utils.on(container, utils.transitionEnd, restore);\n\n // Set dimensions to target\n container.style.width = `${size.width}px`;\n container.style.height = `${size.height}px`;\n }\n\n // Set attributes on current tab\n current.setAttribute('aria-hidden', true);\n current.setAttribute('tabindex', -1);\n\n // Set attributes on target\n pane.setAttribute('aria-hidden', !show);\n tab.setAttribute('aria-expanded', show);\n pane.removeAttribute('tabindex');\n\n // Focus the first item\n pane.querySelectorAll('button:not(:disabled), input:not(:disabled), [tabindex]')[0].focus();\n },\n\n // Build the default HTML\n // TODO: Set order based on order in the config.controls array?\n create(data) {\n // Do nothing if we want no controls\n if (utils.is.empty(this.config.controls)) {\n return null;\n }\n\n // Create the container\n const container = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.controls.wrapper));\n\n // Restart button\n if (this.config.controls.includes('restart')) {\n container.appendChild(controls.createButton.call(this, 'restart'));\n }\n\n // Rewind button\n if (this.config.controls.includes('rewind')) {\n container.appendChild(controls.createButton.call(this, 'rewind'));\n }\n\n // Play/Pause button\n if (this.config.controls.includes('play')) {\n container.appendChild(controls.createButton.call(this, 'play'));\n // container.appendChild(controls.createButton.call(this, 'pause'));\n }\n\n // Fast forward button\n if (this.config.controls.includes('fast-forward')) {\n container.appendChild(controls.createButton.call(this, 'fast-forward'));\n }\n\n // Progress\n if (this.config.controls.includes('progress')) {\n const progress = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.progress));\n\n // Seek range slider\n const seek = controls.createRange.call(this, 'seek', {\n id: `plyr-seek-${data.id}`,\n });\n progress.appendChild(seek.label);\n progress.appendChild(seek.input);\n\n // Buffer progress\n progress.appendChild(controls.createProgress.call(this, 'buffer'));\n\n // TODO: Add loop display indicator\n\n // Seek tooltip\n if (this.config.tooltips.seek) {\n const tooltip = utils.createElement(\n 'span',\n {\n role: 'tooltip',\n class: this.config.classNames.tooltip,\n },\n '00:00'\n );\n\n progress.appendChild(tooltip);\n this.elements.display.seekTooltip = tooltip;\n }\n\n this.elements.progress = progress;\n container.appendChild(this.elements.progress);\n }\n\n // Media current time display\n if (this.config.controls.includes('current-time')) {\n container.appendChild(controls.createTime.call(this, 'currentTime'));\n }\n\n // Media duration display\n if (this.config.controls.includes('duration')) {\n container.appendChild(controls.createTime.call(this, 'duration'));\n }\n\n // Toggle mute button\n if (this.config.controls.includes('mute')) {\n container.appendChild(controls.createButton.call(this, 'mute'));\n }\n\n // Volume range control\n if (this.config.controls.includes('volume')) {\n const volume = utils.createElement('div', {\n class: 'plyr__volume',\n });\n\n // Set the attributes\n const attributes = {\n max: 1,\n step: 0.05,\n value: this.config.volume,\n };\n\n // Create the volume range slider\n const range = controls.createRange.call(\n this,\n 'volume',\n utils.extend(attributes, {\n id: `plyr-volume-${data.id}`,\n })\n );\n volume.appendChild(range.label);\n volume.appendChild(range.input);\n\n this.elements.volume = volume;\n\n container.appendChild(volume);\n }\n\n // Toggle captions button\n if (this.config.controls.includes('captions')) {\n container.appendChild(controls.createButton.call(this, 'captions'));\n }\n\n // Settings button / menu\n if (this.config.controls.includes('settings') && !utils.is.empty(this.config.settings)) {\n const menu = utils.createElement('div', {\n class: 'plyr__menu',\n });\n\n menu.appendChild(\n controls.createButton.call(this, 'settings', {\n id: `plyr-settings-toggle-${data.id}`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}`,\n 'aria-expanded': false,\n })\n );\n\n const form = utils.createElement('form', {\n class: 'plyr__menu__container',\n id: `plyr-settings-${data.id}`,\n 'aria-hidden': true,\n 'aria-labelled-by': `plyr-settings-toggle-${data.id}`,\n role: 'tablist',\n tabindex: -1,\n });\n\n const inner = utils.createElement('div');\n\n const home = utils.createElement('div', {\n id: `plyr-settings-${data.id}-home`,\n 'aria-hidden': false,\n 'aria-labelled-by': `plyr-settings-toggle-${data.id}`,\n role: 'tabpanel',\n });\n\n // Create the tab list\n const tabs = utils.createElement('ul', {\n role: 'tablist',\n });\n\n // Build the tabs\n this.config.settings.forEach(type => {\n const tab = utils.createElement('li', {\n role: 'tab',\n hidden: '',\n });\n\n const button = utils.createElement(\n 'button',\n utils.extend(utils.getAttributesFromSelector(this.config.selectors.buttons.settings), {\n type: 'button',\n class: `${this.config.classNames.control} ${this.config.classNames.control}--forward`,\n id: `plyr-settings-${data.id}-${type}-tab`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}-${type}`,\n 'aria-expanded': false,\n }),\n this.config.i18n[type]\n );\n\n const value = utils.createElement('span', {\n class: this.config.classNames.menu.value,\n });\n\n // Speed contains HTML entities\n value.innerHTML = data[type];\n\n button.appendChild(value);\n tab.appendChild(button);\n tabs.appendChild(tab);\n\n this.elements.settings.tabs[type] = tab;\n });\n\n home.appendChild(tabs);\n inner.appendChild(home);\n\n // Build the panes\n this.config.settings.forEach(type => {\n const pane = utils.createElement('div', {\n id: `plyr-settings-${data.id}-${type}`,\n 'aria-hidden': true,\n 'aria-labelled-by': `plyr-settings-${data.id}-${type}-tab`,\n role: 'tabpanel',\n tabindex: -1,\n hidden: '',\n });\n\n const back = utils.createElement(\n 'button',\n {\n type: 'button',\n class: `${this.config.classNames.control} ${this.config.classNames.control}--back`,\n 'aria-haspopup': true,\n 'aria-controls': `plyr-settings-${data.id}-home`,\n 'aria-expanded': false,\n },\n this.config.i18n[type]\n );\n\n pane.appendChild(back);\n\n const options = utils.createElement('ul');\n\n pane.appendChild(options);\n inner.appendChild(pane);\n\n this.elements.settings.panes[type] = pane;\n });\n\n form.appendChild(inner);\n menu.appendChild(form);\n container.appendChild(menu);\n\n this.elements.settings.form = form;\n this.elements.settings.menu = menu;\n }\n\n // Picture in picture button\n if (this.config.controls.includes('pip') && support.pip) {\n container.appendChild(controls.createButton.call(this, 'pip'));\n }\n\n // Airplay button\n if (this.config.controls.includes('airplay') && support.airplay) {\n container.appendChild(controls.createButton.call(this, 'airplay'));\n }\n\n // Toggle fullscreen button\n if (this.config.controls.includes('fullscreen')) {\n container.appendChild(controls.createButton.call(this, 'fullscreen'));\n }\n\n // Larger overlaid play button\n if (this.config.controls.includes('play-large')) {\n this.elements.container.appendChild(controls.createButton.call(this, 'play-large'));\n }\n\n this.elements.controls = container;\n\n if (this.config.controls.includes('settings') && this.config.settings.includes('speed')) {\n controls.setSpeedMenu.call(this);\n }\n\n return container;\n },\n\n // Insert controls\n inject() {\n // Sprite\n if (this.config.loadSprite) {\n const icon = controls.getIconUrl.call(this);\n\n // Only load external sprite using AJAX\n if (icon.absolute) {\n utils.loadSprite(icon.url, 'sprite-plyr');\n }\n }\n\n // Create a unique ID\n this.id = Math.floor(Math.random() * 10000);\n\n // Null by default\n let container = null;\n\n // HTML passed as the option\n if (utils.is.string(this.config.controls)) {\n container = this.config.controls;\n } else if (utils.is.function(this.config.controls)) {\n // A custom function to build controls\n // The function can return a HTMLElement or String\n container = this.config.controls({\n id: this.id,\n seektime: this.config.seekTime,\n title: this.config.title,\n });\n } else {\n // Create controls\n container = controls.create.call(this, {\n id: this.id,\n seektime: this.config.seekTime,\n speed: this.speed,\n quality: this.quality,\n captions: controls.getLanguage.call(this),\n // TODO: Looping\n // loop: 'None',\n });\n }\n\n // Controls container\n let target;\n\n // Inject to custom location\n if (utils.is.string(this.config.selectors.controls.container)) {\n target = document.querySelector(this.config.selectors.controls.container);\n }\n\n // Inject into the container by default\n if (!utils.is.element(target)) {\n target = this.elements.container;\n }\n\n // Inject controls HTML\n if (utils.is.element(container)) {\n target.appendChild(container);\n } else {\n target.insertAdjacentHTML('beforeend', container);\n }\n\n // Find the elements if need be\n if (utils.is.element(this.elements.controls)) {\n utils.findElements.call(this);\n }\n\n // Edge sometimes doesn't finish the paint so force a redraw\n if (window.navigator.userAgent.includes('Edge')) {\n utils.repaint(target);\n }\n\n // Setup tooltips\n if (this.config.tooltips.controls) {\n const labels = utils.getElements.call(\n this,\n [\n this.config.selectors.controls.wrapper,\n ' ',\n this.config.selectors.labels,\n ' .',\n this.config.classNames.hidden,\n ].join('')\n );\n\n Array.from(labels).forEach(label => {\n utils.toggleClass(label, this.config.classNames.hidden, false);\n utils.toggleClass(label, this.config.classNames.tooltip, true);\n label.setAttribute('role', 'tooltip');\n });\n }\n },\n};\n\nexport default controls;\n","// ==========================================================================\n// Plyr Captions\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport controls from './controls';\n\nconst captions = {\n // Setup captions\n setup() {\n // Requires UI support\n if (!this.supported.ui) {\n return;\n }\n\n // Set default language if not set\n const stored = this.storage.get('language');\n\n if (!utils.is.empty(stored)) {\n this.captions.language = stored;\n }\n\n if (utils.is.empty(this.captions.language)) {\n this.captions.language = this.config.captions.language.toLowerCase();\n }\n\n // Set captions enabled state if not set\n if (!utils.is.boolean(this.captions.active)) {\n const active = this.storage.get('captions');\n\n if (utils.is.boolean(active)) {\n this.captions.active = active;\n } else {\n this.captions.active = this.config.captions.active;\n }\n }\n\n // Only Vimeo and HTML5 video supported at this point\n if (!this.isVideo || this.isYouTube || (this.isHTML5 && !support.textTracks)) {\n // Clear menu and hide\n if (this.config.controls.includes('settings') && this.config.settings.includes('captions')) {\n controls.setCaptionsMenu.call(this);\n }\n\n return;\n }\n\n // Inject the container\n if (!utils.is.element(this.elements.captions)) {\n this.elements.captions = utils.createElement('div', utils.getAttributesFromSelector(this.config.selectors.captions));\n\n utils.insertAfter(this.elements.captions, this.elements.wrapper);\n }\n\n // Set the class hook\n utils.toggleClass(this.elements.container, this.config.classNames.captions.enabled, !utils.is.empty(captions.getTracks.call(this)));\n\n // If no caption file exists, hide container for caption text\n if (utils.is.empty(captions.getTracks.call(this))) {\n return;\n }\n\n // Set language\n captions.setLanguage.call(this);\n\n // Enable UI\n captions.show.call(this);\n\n // Set available languages in list\n if (this.config.controls.includes('settings') && this.config.settings.includes('captions')) {\n controls.setCaptionsMenu.call(this);\n }\n },\n\n // Set the captions language\n setLanguage() {\n // Setup HTML5 track rendering\n if (this.isHTML5 && this.isVideo) {\n captions.getTracks.call(this).forEach(track => {\n // Remove previous bindings\n utils.on(track, 'cuechange', event => captions.setCue.call(this, event));\n\n // Turn off native caption rendering to avoid double captions\n // eslint-disable-next-line\n track.mode = 'hidden';\n });\n\n // Get current track\n const currentTrack = captions.getCurrentTrack.call(this);\n\n // Check if suported kind\n if (utils.is.track(currentTrack)) {\n // If we change the active track while a cue is already displayed we need to update it\n if (Array.from(currentTrack.activeCues || []).length) {\n captions.setCue.call(this, currentTrack);\n }\n }\n } else if (this.isVimeo && this.captions.active) {\n this.embed.enableTextTrack(this.language);\n }\n },\n\n // Get the tracks\n getTracks() {\n // Return empty array at least\n if (utils.is.nullOrUndefined(this.media)) {\n return [];\n }\n\n // Only get accepted kinds\n return Array.from(this.media.textTracks || []).filter(track => [\n 'captions',\n 'subtitles',\n ].includes(track.kind));\n },\n\n // Get the current track for the current language\n getCurrentTrack() {\n return captions.getTracks.call(this).find(track => track.language.toLowerCase() === this.language);\n },\n\n // Display active caption if it contains text\n setCue(input) {\n // Get the track from the event if needed\n const track = utils.is.event(input) ? input.target : input;\n const active = track.activeCues[0];\n const currentTrack = captions.getCurrentTrack.call(this);\n\n // Only display current track\n if (track !== currentTrack) {\n return;\n }\n\n // Display a cue, if there is one\n if (utils.is.cue(active)) {\n captions.setText.call(this, active.getCueAsHTML());\n } else {\n captions.setText.call(this, null);\n }\n\n utils.dispatchEvent.call(this, this.media, 'cuechange');\n },\n\n // Set the current caption\n setText(input) {\n // Requires UI\n if (!this.supported.ui) {\n return;\n }\n\n if (utils.is.element(this.elements.captions)) {\n const content = utils.createElement('span');\n\n // Empty the container\n utils.emptyElement(this.elements.captions);\n\n // Default to empty\n const caption = !utils.is.nullOrUndefined(input) ? input : '';\n\n // Set the span content\n if (utils.is.string(caption)) {\n content.textContent = caption.trim();\n } else {\n content.appendChild(caption);\n }\n\n // Set new caption text\n this.elements.captions.appendChild(content);\n } else {\n this.debug.warn('No captions element to render to');\n }\n },\n\n // Display captions container and button (for initialization)\n show() {\n // If there's no caption toggle, bail\n if (!utils.is.element(this.elements.buttons.captions)) {\n return;\n }\n\n // Try to load the value from storage\n let active = this.storage.get('captions');\n\n // Otherwise fall back to the default config\n if (!utils.is.boolean(active)) {\n ({ active } = this.config.captions);\n } else {\n this.captions.active = active;\n }\n\n if (active) {\n utils.toggleClass(this.elements.container, this.config.classNames.captions.active, true);\n utils.toggleState(this.elements.buttons.captions, true);\n }\n },\n};\n\nexport default captions;\n","// ==========================================================================\n// YouTube plugin\n// ==========================================================================\n\nimport utils from './../utils';\nimport controls from './../controls';\nimport ui from './../ui';\n\nconst youtube = {\n setup() {\n // Add embed class for responsive\n utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true);\n\n // Set aspect ratio\n youtube.setAspectRatio.call(this);\n\n // Setup API\n if (utils.is.object(window.YT) && utils.is.function(window.YT.Player)) {\n youtube.ready.call(this);\n } else {\n // Load the API\n utils.loadScript(this.config.urls.youtube.api);\n\n // Setup callback for the API\n // YouTube has it's own system of course...\n window.onYouTubeReadyCallbacks = window.onYouTubeReadyCallbacks || [];\n\n // Add to queue\n window.onYouTubeReadyCallbacks.push(() => {\n youtube.ready.call(this);\n });\n\n // Set callback to process queue\n window.onYouTubeIframeAPIReady = () => {\n window.onYouTubeReadyCallbacks.forEach(callback => {\n callback();\n });\n };\n }\n },\n\n // Get the media title\n getTitle(videoId) {\n // Try via undocumented API method first\n // This method disappears now and then though...\n // https://github.com/sampotts/plyr/issues/709\n if (utils.is.function(this.embed.getVideoData)) {\n const { title } = this.embed.getVideoData();\n\n if (utils.is.empty(title)) {\n this.config.title = title;\n ui.setTitle.call(this);\n return;\n }\n }\n\n // Or via Google API\n const key = this.config.keys.google;\n if (utils.is.string(key) && !utils.is.empty(key)) {\n const url = `https://www.googleapis.com/youtube/v3/videos?id=${videoId}&key=${key}&fields=items(snippet(title))&part=snippet`;\n\n fetch(url)\n .then(response => (response.ok ? response.json() : null))\n .then(result => {\n if (result !== null && utils.is.object(result)) {\n this.config.title = result.items[0].snippet.title;\n ui.setTitle.call(this);\n }\n })\n .catch(() => {});\n }\n },\n\n // Set aspect ratio\n setAspectRatio() {\n const ratio = this.config.ratio.split(':');\n this.elements.wrapper.style.paddingBottom = `${100 / ratio[0] * ratio[1]}%`;\n },\n\n // API ready\n ready() {\n const player = this;\n\n // Ignore already setup (race condition)\n const currentId = player.media.getAttribute('id');\n if (!utils.is.empty(currentId) && currentId.startsWith('youtube-')) {\n return;\n }\n\n // Replace the <iframe> with a <div> due to YouTube API issues\n const videoId = utils.parseYouTubeId(player.media.getAttribute('src'));\n const id = utils.generateId(player.provider);\n const container = utils.createElement('div', { id });\n player.media = utils.replaceElement(container, player.media);\n\n // Setup instance\n // https://developers.google.com/youtube/iframe_api_reference\n player.embed = new window.YT.Player(id, {\n videoId,\n playerVars: {\n autoplay: player.config.autoplay ? 1 : 0, // Autoplay\n controls: player.supported.ui ? 0 : 1, // Only show controls if not fully supported\n rel: 0, // No related vids\n showinfo: 0, // Hide info\n iv_load_policy: 3, // Hide annotations\n modestbranding: 1, // Hide logos as much as possible (they still show one in the corner when paused)\n disablekb: 1, // Disable keyboard as we handle it\n playsinline: 1, // Allow iOS inline playback\n\n // Tracking for stats\n origin: window && window.location.hostname,\n widget_referrer: window && window.location.href,\n\n // Captions are flaky on YouTube\n cc_load_policy: player.captions.active ? 1 : 0,\n cc_lang_pref: player.config.captions.language,\n },\n events: {\n onError(event) {\n // If we've already fired an error, don't do it again\n // YouTube fires onError twice\n if (utils.is.object(player.media.error)) {\n return;\n }\n\n const detail = {\n code: event.data,\n };\n\n // Messages copied from https://developers.google.com/youtube/iframe_api_reference#onError\n switch (event.data) {\n case 2:\n detail.message =\n 'The request contains an invalid parameter value. For example, this error occurs if you specify a video ID that does not have 11 characters, or if the video ID contains invalid characters, such as exclamation points or asterisks.';\n break;\n\n case 5:\n detail.message =\n 'The requested content cannot be played in an HTML5 player or another error related to the HTML5 player has occurred.';\n break;\n\n case 100:\n detail.message =\n 'The video requested was not found. This error occurs when a video has been removed (for any reason) or has been marked as private.';\n break;\n\n case 101:\n case 150:\n detail.message = 'The owner of the requested video does not allow it to be played in embedded players.';\n break;\n\n default:\n detail.message = 'An unknown error occured';\n break;\n }\n\n player.media.error = detail;\n\n utils.dispatchEvent.call(player, player.media, 'error');\n },\n onPlaybackQualityChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Get current quality\n player.media.quality = instance.getPlaybackQuality();\n\n utils.dispatchEvent.call(player, player.media, 'qualitychange');\n },\n onPlaybackRateChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Get current speed\n player.media.playbackRate = instance.getPlaybackRate();\n\n utils.dispatchEvent.call(player, player.media, 'ratechange');\n },\n onReady(event) {\n // Get the instance\n const instance = event.target;\n\n // Get the title\n youtube.getTitle.call(player, videoId);\n\n // Create a faux HTML5 API using the YouTube API\n player.media.play = () => {\n instance.playVideo();\n player.media.paused = false;\n };\n\n player.media.pause = () => {\n instance.pauseVideo();\n player.media.paused = true;\n };\n\n player.media.stop = () => {\n instance.stopVideo();\n player.media.paused = true;\n };\n\n player.media.duration = instance.getDuration();\n player.media.paused = true;\n\n // Seeking\n player.media.currentTime = 0;\n Object.defineProperty(player.media, 'currentTime', {\n get() {\n return Number(instance.getCurrentTime());\n },\n set(time) {\n // Set seeking flag\n player.media.seeking = true;\n\n // Trigger seeking\n utils.dispatchEvent.call(player, player.media, 'seeking');\n\n // Seek after events sent\n instance.seekTo(time);\n },\n });\n\n // Playback speed\n Object.defineProperty(player.media, 'playbackRate', {\n get() {\n return instance.getPlaybackRate();\n },\n set(input) {\n instance.setPlaybackRate(input);\n },\n });\n\n // Quality\n Object.defineProperty(player.media, 'quality', {\n get() {\n return instance.getPlaybackQuality();\n },\n set(input) {\n // Trigger request event\n utils.dispatchEvent.call(player, player.media, 'qualityrequested', false, {\n quality: input,\n });\n\n instance.setPlaybackQuality(input);\n },\n });\n\n // Volume\n let { volume } = player.config;\n Object.defineProperty(player.media, 'volume', {\n get() {\n return volume;\n },\n set(input) {\n volume = input;\n instance.setVolume(volume * 100);\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n },\n });\n\n // Muted\n let { muted } = player.config;\n Object.defineProperty(player.media, 'muted', {\n get() {\n return muted;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : muted;\n muted = toggle;\n instance[toggle ? 'mute' : 'unMute']();\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n },\n });\n\n // Source\n Object.defineProperty(player.media, 'currentSrc', {\n get() {\n return instance.getVideoUrl();\n },\n });\n\n // Ended\n Object.defineProperty(player.media, 'ended', {\n get() {\n return player.currentTime === player.duration;\n },\n });\n\n // Get available speeds\n player.options.speed = instance.getAvailablePlaybackRates();\n\n // Set the tabindex to avoid focus entering iframe\n if (player.supported.ui) {\n player.media.setAttribute('tabindex', -1);\n }\n\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n\n // Reset timer\n window.clearInterval(player.timers.buffering);\n\n // Setup buffering\n player.timers.buffering = window.setInterval(() => {\n // Get loaded % from YouTube\n player.media.buffered = instance.getVideoLoadedFraction();\n\n // Trigger progress only when we actually buffer something\n if (player.media.lastBuffered === null || player.media.lastBuffered < player.media.buffered) {\n utils.dispatchEvent.call(player, player.media, 'progress');\n }\n\n // Set last buffer point\n player.media.lastBuffered = player.media.buffered;\n\n // Bail if we're at 100%\n if (player.media.buffered === 1) {\n window.clearInterval(player.timers.buffering);\n\n // Trigger event\n utils.dispatchEvent.call(player, player.media, 'canplaythrough');\n }\n }, 200);\n\n // Rebuild UI\n window.setTimeout(() => ui.build.call(player), 50);\n },\n onStateChange(event) {\n // Get the instance\n const instance = event.target;\n\n // Reset timer\n window.clearInterval(player.timers.playing);\n\n // Handle events\n // -1 Unstarted\n // 0 Ended\n // 1 Playing\n // 2 Paused\n // 3 Buffering\n // 5 Video cued\n switch (event.data) {\n case 0:\n player.media.paused = true;\n\n // YouTube doesn't support loop for a single video, so mimick it.\n if (player.media.loop) {\n // YouTube needs a call to `stopVideo` before playing again\n instance.stopVideo();\n instance.playVideo();\n } else {\n utils.dispatchEvent.call(player, player.media, 'ended');\n }\n\n break;\n\n case 1:\n // If we were seeking, fire seeked event\n if (player.media.seeking) {\n utils.dispatchEvent.call(player, player.media, 'seeked');\n }\n player.media.seeking = false;\n\n // Only fire play if paused before\n if (player.media.paused) {\n utils.dispatchEvent.call(player, player.media, 'play');\n }\n player.media.paused = false;\n\n utils.dispatchEvent.call(player, player.media, 'playing');\n\n // Poll to get playback progress\n player.timers.playing = window.setInterval(() => {\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n }, 50);\n\n // Check duration again due to YouTube bug\n // https://github.com/sampotts/plyr/issues/374\n // https://code.google.com/p/gdata-issues/issues/detail?id=8690\n if (player.media.duration !== instance.getDuration()) {\n player.media.duration = instance.getDuration();\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n }\n\n // Get quality\n controls.setQualityMenu.call(player, instance.getAvailableQualityLevels());\n\n break;\n\n case 2:\n player.media.paused = true;\n\n utils.dispatchEvent.call(player, player.media, 'pause');\n\n break;\n\n default:\n break;\n }\n\n utils.dispatchEvent.call(player, player.elements.container, 'statechange', false, {\n code: event.data,\n });\n },\n },\n });\n },\n};\n\nexport default youtube;\n","// ==========================================================================\n// Vimeo plugin\n// ==========================================================================\n\nimport utils from './../utils';\nimport captions from './../captions';\nimport ui from './../ui';\n\nconst vimeo = {\n setup() {\n // Add embed class for responsive\n utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true);\n\n // Set intial ratio\n vimeo.setAspectRatio.call(this);\n\n // Load the API if not already\n if (!utils.is.object(window.Vimeo)) {\n utils.loadScript(this.config.urls.vimeo.api, () => {\n vimeo.ready.call(this);\n });\n } else {\n vimeo.ready.call(this);\n }\n },\n\n // Set aspect ratio\n // For Vimeo we have an extra 300% height <div> to hide the standard controls and UI\n setAspectRatio(input) {\n const ratio = utils.is.string(input) ? input.split(':') : this.config.ratio.split(':');\n const padding = 100 / ratio[0] * ratio[1];\n const height = 200;\n const offset = (height - padding) / (height / 50);\n this.elements.wrapper.style.paddingBottom = `${padding}%`;\n this.media.style.transform = `translateY(-${offset}%)`;\n },\n\n // API Ready\n ready() {\n const player = this;\n\n // Get Vimeo params for the iframe\n const options = {\n loop: player.config.loop.active,\n autoplay: player.autoplay,\n byline: false,\n portrait: false,\n title: false,\n speed: true,\n transparent: 0,\n gesture: 'media',\n };\n const params = utils.buildUrlParams(options);\n const id = utils.parseVimeoId(player.media.getAttribute('src'));\n\n // Build an iframe\n const iframe = utils.createElement('iframe');\n const src = `https://player.vimeo.com/video/${id}?${params}`;\n iframe.setAttribute('src', src);\n iframe.setAttribute('allowfullscreen', '');\n iframe.setAttribute('allowtransparency', '');\n iframe.setAttribute('allow', 'autoplay');\n\n // Inject the package\n const wrapper = utils.createElement('div');\n wrapper.appendChild(iframe);\n player.media = utils.replaceElement(wrapper, player.media);\n\n // Setup instance\n // https://github.com/vimeo/player.js\n player.embed = new window.Vimeo.Player(iframe);\n\n player.media.paused = true;\n player.media.currentTime = 0;\n\n // Create a faux HTML5 API using the Vimeo API\n player.media.play = () => {\n player.embed.play().then(() => {\n player.media.paused = false;\n });\n };\n\n player.media.pause = () => {\n player.embed.pause().then(() => {\n player.media.paused = true;\n });\n };\n\n player.media.stop = () => {\n player.embed.stop().then(() => {\n player.media.paused = true;\n player.currentTime = 0;\n });\n };\n\n // Seeking\n let { currentTime } = player.media;\n Object.defineProperty(player.media, 'currentTime', {\n get() {\n return currentTime;\n },\n set(time) {\n // Get current paused state\n // Vimeo will automatically play on seek\n const { paused } = player.media;\n\n // Set seeking flag\n player.media.seeking = true;\n\n // Trigger seeking\n utils.dispatchEvent.call(player, player.media, 'seeking');\n\n // Seek after events\n player.embed.setCurrentTime(time);\n\n // Restore pause state\n if (paused) {\n player.pause();\n }\n },\n });\n\n // Playback speed\n let speed = player.config.speed.selected;\n Object.defineProperty(player.media, 'playbackRate', {\n get() {\n return speed;\n },\n set(input) {\n player.embed.setPlaybackRate(input).then(() => {\n speed = input;\n utils.dispatchEvent.call(player, player.media, 'ratechange');\n });\n },\n });\n\n // Volume\n let { volume } = player.config;\n Object.defineProperty(player.media, 'volume', {\n get() {\n return volume;\n },\n set(input) {\n player.embed.setVolume(input).then(() => {\n volume = input;\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n });\n },\n });\n\n // Muted\n let { muted } = player.config;\n Object.defineProperty(player.media, 'muted', {\n get() {\n return muted;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : false;\n\n player.embed.setVolume(toggle ? 0 : player.config.volume).then(() => {\n muted = toggle;\n utils.dispatchEvent.call(player, player.media, 'volumechange');\n });\n },\n });\n\n // Loop\n let { loop } = player.config;\n Object.defineProperty(player.media, 'loop', {\n get() {\n return loop;\n },\n set(input) {\n const toggle = utils.is.boolean(input) ? input : player.config.loop.active;\n\n player.embed.setLoop(toggle).then(() => {\n loop = toggle;\n });\n },\n });\n\n // Source\n let currentSrc;\n player.embed.getVideoUrl().then(value => {\n currentSrc = value;\n });\n Object.defineProperty(player.media, 'currentSrc', {\n get() {\n return currentSrc;\n },\n });\n\n // Ended\n Object.defineProperty(player.media, 'ended', {\n get() {\n return player.currentTime === player.duration;\n },\n });\n\n // Set aspect ratio based on video size\n Promise.all([\n player.embed.getVideoWidth(),\n player.embed.getVideoHeight(),\n ]).then(dimensions => {\n const ratio = utils.getAspectRatio(dimensions[0], dimensions[1]);\n vimeo.setAspectRatio.call(this, ratio);\n });\n\n // Set autopause\n player.embed.setAutopause(player.config.autopause).then(state => {\n player.config.autopause = state;\n });\n\n // Get title\n player.embed.getVideoTitle().then(title => {\n player.config.title = title;\n ui.setTitle.call(this);\n });\n\n // Get current time\n player.embed.getCurrentTime().then(value => {\n currentTime = value;\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n });\n\n // Get duration\n player.embed.getDuration().then(value => {\n player.media.duration = value;\n utils.dispatchEvent.call(player, player.media, 'durationchange');\n });\n\n // Get captions\n player.embed.getTextTracks().then(tracks => {\n player.media.textTracks = tracks;\n captions.setup.call(player);\n });\n\n player.embed.on('cuechange', data => {\n let cue = null;\n\n if (data.cues.length) {\n cue = utils.stripHTML(data.cues[0].text);\n }\n\n captions.setText.call(player, cue);\n });\n\n player.embed.on('loaded', () => {\n if (utils.is.element(player.embed.element) && player.supported.ui) {\n const frame = player.embed.element;\n\n // Fix keyboard focus issues\n // https://github.com/sampotts/plyr/issues/317\n frame.setAttribute('tabindex', -1);\n }\n });\n\n player.embed.on('play', () => {\n // Only fire play if paused before\n if (player.media.paused) {\n utils.dispatchEvent.call(player, player.media, 'play');\n }\n player.media.paused = false;\n utils.dispatchEvent.call(player, player.media, 'playing');\n });\n\n player.embed.on('pause', () => {\n player.media.paused = true;\n utils.dispatchEvent.call(player, player.media, 'pause');\n });\n\n player.embed.on('timeupdate', data => {\n player.media.seeking = false;\n currentTime = data.seconds;\n utils.dispatchEvent.call(player, player.media, 'timeupdate');\n });\n\n player.embed.on('progress', data => {\n player.media.buffered = data.percent;\n utils.dispatchEvent.call(player, player.media, 'progress');\n\n // Check all loaded\n if (parseInt(data.percent, 10) === 1) {\n utils.dispatchEvent.call(player, player.media, 'canplaythrough');\n }\n });\n\n player.embed.on('seeked', () => {\n player.media.seeking = false;\n utils.dispatchEvent.call(player, player.media, 'seeked');\n utils.dispatchEvent.call(player, player.media, 'play');\n });\n\n player.embed.on('ended', () => {\n player.media.paused = true;\n utils.dispatchEvent.call(player, player.media, 'ended');\n });\n\n player.embed.on('error', detail => {\n player.media.error = detail;\n utils.dispatchEvent.call(player, player.media, 'error');\n });\n\n // Rebuild UI\n window.setTimeout(() => ui.build.call(player), 0);\n },\n};\n\nexport default vimeo;\n","// ==========================================================================\n// Plyr Media\n// ==========================================================================\n\nimport support from './support';\nimport utils from './utils';\nimport youtube from './plugins/youtube';\nimport vimeo from './plugins/vimeo';\nimport ui from './ui';\n\n// Sniff out the browser\nconst browser = utils.getBrowser();\n\nconst media = {\n // Setup media\n setup() {\n // If there's no media, bail\n if (!this.media) {\n this.debug.warn('No media element found!');\n return;\n }\n\n // Add type class\n utils.toggleClass(this.elements.container, this.config.classNames.type.replace('{0}', this.type), true);\n\n // Add provider class\n utils.toggleClass(this.elements.container, this.config.classNames.provider.replace('{0}', this.provider), true);\n\n // Add video class for embeds\n // This will require changes if audio embeds are added\n if (this.isEmbed) {\n utils.toggleClass(this.elements.container, this.config.classNames.type.replace('{0}', 'video'), true);\n }\n\n if (this.supported.ui) {\n // Check for picture-in-picture support\n utils.toggleClass(this.elements.container, this.config.classNames.pip.supported, support.pip && this.isHTML5 && this.isVideo);\n\n // Check for airplay support\n utils.toggleClass(this.elements.container, this.config.classNames.airplay.supported, support.airplay && this.isHTML5);\n\n // If there's no autoplay attribute, assume the video is stopped and add state class\n utils.toggleClass(this.elements.container, this.config.classNames.stopped, this.config.autoplay);\n\n // Add iOS class\n utils.toggleClass(this.elements.container, this.config.classNames.isIos, browser.isIos);\n\n // Add touch class\n utils.toggleClass(this.elements.container, this.config.classNames.isTouch, support.touch);\n }\n\n // Inject the player wrapper\n if (this.isVideo) {\n // Create the wrapper div\n this.elements.wrapper = utils.createElement('div', {\n class: this.config.classNames.video,\n });\n\n // Wrap the video in a container\n utils.wrap(this.media, this.elements.wrapper);\n }\n\n if (this.isEmbed) {\n switch (this.provider) {\n case 'youtube':\n youtube.setup.call(this);\n break;\n\n case 'vimeo':\n vimeo.setup.call(this);\n break;\n\n default:\n break;\n }\n } else if (this.isHTML5) {\n ui.setTitle.call(this);\n }\n },\n\n // Cancel current network requests\n // See https://github.com/sampotts/plyr/issues/174\n cancelRequests() {\n if (!this.isHTML5) {\n return;\n }\n\n // Remove child sources\n Array.from(this.media.querySelectorAll('source')).forEach(utils.removeElement);\n\n // Set blank video src attribute\n // This is to prevent a MEDIA_ERR_SRC_NOT_SUPPORTED error\n // Info: http://stackoverflow.com/questions/32231579/how-to-properly-dispose-of-an-html5-video-and-close-socket-or-connection\n this.media.setAttribute('src', this.config.blankVideo);\n\n // Load the new empty source\n // This will cancel existing requests\n // See https://github.com/sampotts/plyr/issues/174\n this.media.load();\n\n // Debugging\n this.debug.log('Cancelled network requests');\n },\n};\n\nexport default media;\n","// ==========================================================================\n// Plyr source update\n// ==========================================================================\n\nimport { providers } from './types';\nimport utils from './utils';\nimport media from './media';\nimport ui from './ui';\nimport support from './support';\n\nconst source = {\n // Add elements to HTML5 media (source, tracks, etc)\n insertElements(type, attributes) {\n if (utils.is.string(attributes)) {\n utils.insertElement(type, this.media, {\n src: attributes,\n });\n } else if (utils.is.array(attributes)) {\n attributes.forEach(attribute => {\n utils.insertElement(type, this.media, attribute);\n });\n }\n },\n\n // Update source\n // Sources are not checked for support so be careful\n change(input) {\n if (!utils.is.object(input) || !('sources' in input) || !input.sources.length) {\n this.debug.warn('Invalid source format');\n return;\n }\n\n // Cancel current network requests\n media.cancelRequests.call(this);\n\n // Destroy instance and re-setup\n this.destroy.call(\n this,\n () => {\n // TODO: Reset menus here\n\n // Remove elements\n utils.removeElement(this.media);\n this.media = null;\n\n // Reset class name\n if (utils.is.element(this.elements.container)) {\n this.elements.container.removeAttribute('class');\n }\n\n // Set the type and provider\n this.type = input.type;\n this.provider = !utils.is.empty(input.sources[0].provider) ? input.sources[0].provider : providers.html5;\n\n // Check for support\n this.supported = support.check(this.type, this.provider, this.config.inline);\n\n // Create new markup\n switch (`${this.provider}:${this.type}`) {\n case 'html5:video':\n this.media = utils.createElement('video');\n break;\n\n case 'html5:audio':\n this.media = utils.createElement('audio');\n break;\n\n case 'youtube:video':\n case 'vimeo:video':\n this.media = utils.createElement('div', {\n src: input.sources[0].src,\n });\n break;\n\n default:\n break;\n }\n\n // Inject the new element\n this.elements.container.appendChild(this.media);\n\n // Autoplay the new source?\n if (utils.is.boolean(input.autoplay)) {\n this.config.autoplay = input.autoplay;\n }\n\n // Set attributes for audio and video\n if (this.isHTML5) {\n if (this.config.crossorigin) {\n this.media.setAttribute('crossorigin', '');\n }\n if (this.config.autoplay) {\n this.media.setAttribute('autoplay', '');\n }\n if ('poster' in input) {\n this.media.setAttribute('poster', input.poster);\n }\n if (this.config.loop.active) {\n this.media.setAttribute('loop', '');\n }\n if (this.config.muted) {\n this.media.setAttribute('muted', '');\n }\n if (this.config.inline) {\n this.media.setAttribute('playsinline', '');\n }\n }\n\n // Restore class hook\n ui.addStyleHook.call(this);\n\n // Set new sources for html5\n if (this.isHTML5) {\n source.insertElements.call(this, 'source', input.sources);\n }\n\n // Set video title\n this.config.title = input.title;\n\n // Set up from scratch\n media.setup.call(this);\n\n // HTML5 stuff\n if (this.isHTML5) {\n // Setup captions\n if ('tracks' in input) {\n source.insertElements.call(this, 'track', input.tracks);\n }\n\n // Load HTML5 sources\n this.media.load();\n }\n\n // If HTML5 or embed but not fully supported, setupInterface and call ready now\n if (this.isHTML5 || (this.isEmbed && !this.supported.ui)) {\n // Setup interface\n ui.build.call(this);\n }\n },\n true,\n );\n },\n};\n\nexport default source;\n","// ==========================================================================\n// Plyr\n// plyr.js v3.0.0-beta.1\n// https://github.com/sampotts/plyr\n// License: The MIT License (MIT)\n// ==========================================================================\n\nimport { providers, types } from './types';\nimport defaults from './defaults';\nimport support from './support';\nimport utils from './utils';\n\nimport Console from './console';\nimport Storage from './storage';\n\nimport captions from './captions';\nimport controls from './controls';\nimport fullscreen from './fullscreen';\nimport listeners from './listeners';\nimport media from './media';\nimport source from './source';\nimport ui from './ui';\n\n// Private properties\n// TODO: Use a WeakMap for private globals\n// const globals = new WeakMap();\n\n// Globals\nlet scrollPosition = {\n x: 0,\n y: 0,\n};\n\n// Plyr instance\nclass Plyr {\n constructor(target, options) {\n this.timers = {};\n\n // State\n this.ready = false;\n this.loading = false;\n this.failed = false;\n\n // Set the media element\n this.media = target;\n\n // String selector passed\n if (utils.is.string(this.media)) {\n this.media = document.querySelectorAll(this.media);\n }\n\n // jQuery, NodeList or Array passed, use first element\n if ((window.jQuery && this.media instanceof jQuery) || utils.is.nodeList(this.media) || utils.is.array(this.media)) {\n // eslint-disable-next-line\n this.media = this.media[0];\n }\n\n // Set config\n this.config = utils.extend(\n {},\n defaults,\n options,\n (() => {\n try {\n return JSON.parse(this.media.getAttribute('data-plyr-config'));\n } catch (e) {\n return {};\n }\n })(),\n );\n\n // Elements cache\n this.elements = {\n container: null,\n buttons: {},\n display: {},\n progress: {},\n inputs: {},\n settings: {\n menu: null,\n panes: {},\n tabs: {},\n },\n captions: null,\n };\n\n // Captions\n this.captions = {\n active: null,\n currentTrack: null,\n };\n\n // Fullscreen\n this.fullscreen = {\n active: false,\n };\n\n // Options\n this.options = {\n speed: [],\n quality: [],\n };\n\n // Debugging\n // TODO: move to globals\n this.debug = new Console(this.config.debug);\n\n // Log config options and support\n this.debug.log('Config', this.config);\n this.debug.log('Support', support);\n\n // We need an element to setup\n if (utils.is.nullOrUndefined(this.media) || !utils.is.element(this.media)) {\n this.debug.error('Setup failed: no suitable element passed');\n return;\n }\n\n // Bail if the element is initialized\n if (this.media.plyr) {\n this.debug.warn('Target already setup');\n return;\n }\n\n // Bail if not enabled\n if (!this.config.enabled) {\n this.debug.error('Setup failed: disabled by config');\n return;\n }\n\n // Bail if disabled or no basic support\n // You may want to disable certain UAs etc\n if (!support.check().api) {\n this.debug.error('Setup failed: no support');\n return;\n }\n\n // Cache original element state for .destroy()\n this.elements.original = this.media.cloneNode(true);\n\n // Set media type based on tag or data attribute\n // Supported: video, audio, vimeo, youtube\n const type = this.media.tagName.toLowerCase();\n\n // Embed properties\n let iframe = null;\n let url = null;\n let params = null;\n\n // Different setup based on type\n switch (type) {\n case 'div':\n // Find the frame\n iframe = this.media.querySelector('iframe');\n\n // <iframe> required\n if (!utils.is.element(iframe)) {\n this.debug.error('Setup failed: <iframe> is missing');\n return;\n }\n\n // Audio will come later for external providers\n this.type = types.video;\n\n // Detect provider\n url = iframe.getAttribute('src');\n this.provider = utils.getProviderByUrl(url);\n\n // Get attributes from URL and set config\n params = utils.getUrlParams(url);\n if (!utils.is.empty(params)) {\n const truthy = [\n '1',\n 'true',\n ];\n\n if (truthy.includes(params.autoplay)) {\n this.config.autoplay = true;\n }\n if (truthy.includes(params.playsinline)) {\n this.config.inline = true;\n }\n if (truthy.includes(params.loop)) {\n this.config.loop.active = true;\n }\n }\n\n // Unsupported provider\n if (utils.is.empty(this.provider) || !Object.keys(providers).includes(this.provider)) {\n this.debug.error('Setup failed: Invalid provider');\n return;\n }\n\n // Rework elements\n this.elements.container = this.media;\n this.media = iframe;\n\n // Reset classname\n this.elements.container.className = '';\n\n break;\n\n case 'video':\n case 'audio':\n this.type = type;\n this.provider = providers.html5;\n\n // Get config from attributes\n if (this.media.hasAttribute('crossorigin')) {\n this.config.crossorigin = true;\n }\n if (this.media.hasAttribute('autoplay')) {\n this.config.autoplay = true;\n }\n if (this.media.hasAttribute('playsinline')) {\n this.config.inline = true;\n }\n if (this.media.hasAttribute('muted')) {\n this.config.muted = true;\n }\n if (this.media.hasAttribute('loop')) {\n this.config.loop.active = true;\n }\n\n break;\n\n default:\n this.debug.error('Setup failed: unsupported type');\n return;\n }\n\n // Setup local storage for user settings\n this.storage = new Storage(this);\n\n // Check for support again but with type\n this.supported = support.check(this.type, this.provider, this.config.inline);\n\n // If no support for even API, bail\n if (!this.supported.api) {\n this.debug.error('Setup failed: no support');\n return;\n }\n\n // Store reference\n this.media.plyr = this;\n\n // Wrap media\n if (!utils.is.element(this.elements.container)) {\n this.elements.container = utils.createElement('div');\n utils.wrap(this.media, this.elements.container);\n }\n\n // Allow focus to be captured\n this.elements.container.setAttribute('tabindex', 0);\n\n // Global listeners\n listeners.global.call(this);\n\n // Add style hook\n ui.addStyleHook.call(this);\n\n // Setup media\n media.setup.call(this);\n\n // Listen for events if debugging\n if (this.config.debug) {\n utils.on(this.elements.container, this.config.events.join(' '), event => {\n this.debug.log(`event: ${event.type}`);\n });\n }\n\n // Setup interface\n // If embed but not fully supported, build interface now to avoid flash of controls\n if (this.isHTML5 || (this.isEmbed && !this.supported.ui)) {\n ui.build.call(this);\n }\n }\n\n // ---------------------------------------\n // API\n // ---------------------------------------\n\n /**\n * Types and provider helpers\n */\n get isHTML5() {\n return this.provider === providers.html5;\n }\n get isEmbed() {\n return this.isYouTube || this.isVimeo;\n }\n get isYouTube() {\n return this.provider === providers.youtube;\n }\n get isVimeo() {\n return this.provider === providers.vimeo;\n }\n get isVideo() {\n return this.type === types.video;\n }\n get isAudio() {\n return this.type === types.audio;\n }\n\n /**\n * Play the media\n */\n play() {\n return this.media.play();\n }\n\n /**\n * Pause the media\n */\n pause() {\n if (!this.playing) {\n return;\n }\n\n this.media.pause();\n }\n\n /**\n * Get paused state\n */\n get paused() {\n return this.media.paused;\n }\n\n /**\n * Get playing state\n */\n get playing() {\n return !this.paused && !this.ended && (this.isHTML5 ? this.media.readyState > 2 : true);\n }\n\n /**\n * Get ended state\n */\n get ended() {\n return this.media.ended;\n }\n\n /**\n * Toggle playback based on current status\n * @param {boolean} input\n */\n togglePlay(input) {\n // Toggle based on current state if nothing passed\n const toggle = utils.is.boolean(input) ? input : !this.playing;\n\n if (toggle) {\n this.play();\n } else {\n this.pause();\n }\n }\n\n /**\n * Stop playback\n */\n stop() {\n this.restart();\n this.pause();\n }\n\n /**\n * Restart playback\n */\n restart() {\n this.currentTime = 0;\n }\n\n /**\n * Rewind\n * @param {number} seekTime - how far to rewind in seconds. Defaults to the config.seekTime\n */\n rewind(seekTime) {\n this.currentTime = this.currentTime - (utils.is.number(seekTime) ? seekTime : this.config.seekTime);\n }\n\n /**\n * Fast forward\n * @param {number} seekTime - how far to fast forward in seconds. Defaults to the config.seekTime\n */\n forward(seekTime) {\n this.currentTime = this.currentTime + (utils.is.number(seekTime) ? seekTime : this.config.seekTime);\n }\n\n /**\n * Seek to a time\n * @param {number} input - where to seek to in seconds. Defaults to 0 (the start)\n */\n set currentTime(input) {\n let targetTime = 0;\n\n if (utils.is.number(input)) {\n targetTime = input;\n }\n\n // Normalise targetTime\n if (targetTime < 0) {\n targetTime = 0;\n } else if (targetTime > this.duration) {\n targetTime = this.duration;\n }\n\n // Set\n this.media.currentTime = targetTime.toFixed(4);\n\n // Logging\n this.debug.log(`Seeking to ${this.currentTime} seconds`);\n }\n\n /**\n * Get current time\n */\n get currentTime() {\n return Number(this.media.currentTime);\n }\n\n /**\n * Get seeking status\n */\n get seeking() {\n return this.media.seeking;\n }\n\n /**\n * Get the duration of the current media\n */\n get duration() {\n // Faux duration set via config\n const fauxDuration = parseInt(this.config.duration, 10);\n\n // True duration\n const realDuration = Number(this.media.duration);\n\n // If custom duration is funky, use regular duration\n return !Number.isNaN(fauxDuration) ? fauxDuration : realDuration;\n }\n\n /**\n * Set the player volume\n * @param {number} value - must be between 0 and 1. Defaults to the value from local storage and config.volume if not set in storage\n */\n set volume(value) {\n let volume = value;\n const max = 1;\n const min = 0;\n\n if (utils.is.string(volume)) {\n volume = Number(volume);\n }\n\n // Load volume from storage if no value specified\n if (!utils.is.number(volume)) {\n volume = this.storage.get('volume');\n }\n\n // Use config if all else fails\n if (!utils.is.number(volume)) {\n ({ volume } = this.config);\n }\n\n // Maximum is volumeMax\n if (volume > max) {\n volume = max;\n }\n // Minimum is volumeMin\n if (volume < min) {\n volume = min;\n }\n\n // Update config\n this.config.volume = volume;\n\n // Set the player volume\n this.media.volume = volume;\n\n // If muted, and we're increasing volume, reset muted state\n if (this.muted && volume > 0) {\n this.muted = false;\n }\n }\n\n /**\n * Get the current player volume\n */\n get volume() {\n return this.media.volume;\n }\n\n /**\n * Increase volume\n * @param {boolean} step - How much to decrease by (between 0 and 1)\n */\n increaseVolume(step) {\n const volume = this.media.muted ? 0 : this.volume;\n this.volume = volume + (utils.is.number(step) ? step : 1);\n }\n\n /**\n * Decrease volume\n * @param {boolean} step - How much to decrease by (between 0 and 1)\n */\n decreaseVolume(step) {\n const volume = this.media.muted ? 0 : this.volume;\n this.volume = volume - (utils.is.number(step) ? step : 1);\n }\n\n /**\n * Set muted state\n * @param {boolean} mute\n */\n set muted(mute) {\n let toggle = mute;\n\n // Load muted state from storage\n if (!utils.is.boolean(toggle)) {\n toggle = this.storage.get('muted');\n }\n\n // Use config if all else fails\n if (!utils.is.boolean(toggle)) {\n toggle = this.config.muted;\n }\n\n // Update config\n this.config.muted = toggle;\n\n // Set mute on the player\n this.media.muted = toggle;\n }\n\n /**\n * Get current muted state\n */\n get muted() {\n return this.media.muted;\n }\n\n /**\n * Check if the media has audio\n */\n get hasAudio() {\n // Assume yes for all non HTML5 (as we can't tell...)\n if (!this.isHTML5) {\n return true;\n }\n\n // Get audio tracks\n return this.media.mozHasAudio || Boolean(this.media.webkitAudioDecodedByteCount) || Boolean(this.media.audioTracks && this.media.audioTracks.length);\n }\n\n /**\n * Set playback speed\n * @param {decimal} speed - the speed of playback (0.5-2.0)\n */\n set speed(input) {\n let speed = null;\n\n if (utils.is.number(input)) {\n speed = input;\n }\n\n if (!utils.is.number(speed)) {\n speed = this.storage.get('speed');\n }\n\n if (!utils.is.number(speed)) {\n speed = this.config.speed.selected;\n }\n\n // Set min/max\n if (speed < 0.1) {\n speed = 0.1;\n }\n if (speed > 2.0) {\n speed = 2.0;\n }\n\n if (!this.config.speed.options.includes(speed)) {\n this.debug.warn(`Unsupported speed (${speed})`);\n return;\n }\n\n // Update config\n this.config.speed.selected = speed;\n\n // Set media speed\n this.media.playbackRate = speed;\n }\n\n /**\n * Get current playback speed\n */\n get speed() {\n return this.media.playbackRate;\n }\n\n /**\n * Set playback quality\n * Currently YouTube only\n * @param {string} input - Quality level\n */\n set quality(input) {\n let quality = null;\n\n if (utils.is.string(input)) {\n quality = input;\n }\n\n if (!utils.is.string(quality)) {\n quality = this.storage.get('quality');\n }\n\n if (!utils.is.string(quality)) {\n quality = this.config.quality.selected;\n }\n\n if (!this.options.quality.includes(quality)) {\n this.debug.warn(`Unsupported quality option (${quality})`);\n return;\n }\n\n // Update config\n this.config.quality.selected = quality;\n\n // Set quality\n this.media.quality = quality;\n }\n\n /**\n * Get current quality level\n */\n get quality() {\n return this.media.quality;\n }\n\n /**\n * Toggle loop\n * TODO: Finish fancy new logic. Set the indicator on load as user may pass loop as config\n * @param {boolean} input - Whether to loop or not\n */\n set loop(input) {\n const toggle = utils.is.boolean(input) ? input : this.config.loop.active;\n this.config.loop.active = toggle;\n this.media.loop = toggle;\n\n // Set default to be a true toggle\n /* const type = ['start', 'end', 'all', 'none', 'toggle'].includes(input) ? input : 'toggle';\n\n switch (type) {\n case 'start':\n if (this.config.loop.end && this.config.loop.end <= this.currentTime) {\n this.config.loop.end = null;\n }\n this.config.loop.start = this.currentTime;\n // this.config.loop.indicator.start = this.elements.display.played.value;\n break;\n\n case 'end':\n if (this.config.loop.start >= this.currentTime) {\n return this;\n }\n this.config.loop.end = this.currentTime;\n // this.config.loop.indicator.end = this.elements.display.played.value;\n break;\n\n case 'all':\n this.config.loop.start = 0;\n this.config.loop.end = this.duration - 2;\n this.config.loop.indicator.start = 0;\n this.config.loop.indicator.end = 100;\n break;\n\n case 'toggle':\n if (this.config.loop.active) {\n this.config.loop.start = 0;\n this.config.loop.end = null;\n } else {\n this.config.loop.start = 0;\n this.config.loop.end = this.duration - 2;\n }\n break;\n\n default:\n this.config.loop.start = 0;\n this.config.loop.end = null;\n break;\n } */\n }\n\n /**\n * Get current loop state\n */\n get loop() {\n return this.media.loop;\n }\n\n /**\n * Set new media source\n * @param {object} input - The new source object (see docs)\n */\n set source(input) {\n source.change.call(this, input);\n }\n\n /**\n * Get current source\n */\n get source() {\n return this.media.currentSrc;\n }\n\n /**\n * Set the poster image for a HTML5 video\n * @param {input} - the URL for the new poster image\n */\n set poster(input) {\n if (!this.isHTML5 || !this.isVideo) {\n this.debug.warn('Poster can only be set on HTML5 video');\n return;\n }\n\n if (utils.is.string(input)) {\n this.media.setAttribute('poster', input);\n }\n }\n\n /**\n * Get the current poster image\n */\n get poster() {\n if (!this.isHTML5 || !this.isVideo) {\n return null;\n }\n\n return this.media.getAttribute('poster');\n }\n\n /**\n * Set the autoplay state\n * @param {boolean} input - Whether to autoplay or not\n */\n set autoplay(input) {\n const toggle = utils.is.boolean(input) ? input : this.config.autoplay;\n this.config.autoplay = toggle;\n }\n\n /**\n * Get the current autoplay state\n */\n get autoplay() {\n return this.config.autoplay;\n }\n\n /**\n * Toggle captions\n * @param {boolean} input - Whether to enable captions\n */\n toggleCaptions(input) {\n // If there's no full support, or there's no caption toggle\n if (!this.supported.ui || !utils.is.element(this.elements.buttons.captions)) {\n return;\n }\n\n // If the method is called without parameter, toggle based on current value\n const show = utils.is.boolean(input) ? input : this.elements.container.className.indexOf(this.config.classNames.captions.active) === -1;\n\n // Nothing to change...\n if (this.captions.active === show) {\n return;\n }\n\n // Set global\n this.captions.active = show;\n\n // Toggle state\n utils.toggleState(this.elements.buttons.captions, this.captions.active);\n\n // Add class hook\n utils.toggleClass(this.elements.container, this.config.classNames.captions.active, this.captions.active);\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, this.captions.active ? 'captionsenabled' : 'captionsdisabled');\n }\n\n /**\n * Set the captions language\n * @param {string} - Two character ISO language code (e.g. EN, FR, PT, etc)\n */\n set language(input) {\n // Nothing specified\n if (!utils.is.string(input)) {\n return;\n }\n\n // Toggle captions based on input\n this.toggleCaptions(!utils.is.empty(input));\n\n // If empty string is passed, assume disable captions\n if (utils.is.empty(input)) {\n return;\n }\n\n // Normalize\n const language = input.toLowerCase();\n\n // If nothing to change, bail\n if (this.language === language) {\n return;\n }\n\n // Update config\n this.captions.language = language;\n\n // Clear caption\n captions.setText.call(this, null);\n\n // Update captions\n captions.setLanguage.call(this);\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, 'languagechange');\n }\n\n /**\n * Get the current captions language\n */\n get language() {\n return this.captions.language;\n }\n\n /**\n * Toggle fullscreen playback\n * Requires user input event\n * @param {event} event\n */\n toggleFullscreen(event) {\n // Video only\n if (this.isAudio) {\n return;\n }\n\n // Check for native support\n if (fullscreen.enabled) {\n if (utils.is.event(event) && event.type === fullscreen.eventType) {\n // If it's a fullscreen change event, update the state\n this.fullscreen.active = fullscreen.isFullScreen(this.elements.container);\n } else {\n // Else it's a user request to enter or exit\n if (!this.fullscreen.active) {\n fullscreen.requestFullScreen(this.elements.container);\n } else {\n fullscreen.cancelFullScreen();\n }\n\n return;\n }\n } else {\n // Otherwise, it's a simple toggle\n this.fullscreen.active = !this.fullscreen.active;\n\n // Add class hook\n utils.toggleClass(this.elements.container, this.config.classNames.fullscreen.fallback, this.fullscreen.active);\n\n // Make sure we don't lose scroll position\n if (this.fullscreen.active) {\n scrollPosition = {\n x: window.pageXOffset || 0,\n y: window.pageYOffset || 0,\n };\n } else {\n window.scrollTo(scrollPosition.x, scrollPosition.y);\n }\n\n // Bind/unbind escape key\n document.body.style.overflow = this.fullscreen.active ? 'hidden' : '';\n }\n\n // Set button state\n if (utils.is.element(this.elements.buttons.fullscreen)) {\n utils.toggleState(this.elements.buttons.fullscreen, this.fullscreen.active);\n }\n\n // Trigger an event\n utils.dispatchEvent.call(this, this.media, this.fullscreen.active ? 'enterfullscreen' : 'exitfullscreen');\n }\n\n /**\n * Toggle picture-in-picture playback on WebKit/MacOS\n * TODO: update player with state, support, enabled\n * TODO: detect outside changes\n */\n set pip(input) {\n const states = {\n pip: 'picture-in-picture',\n inline: 'inline',\n };\n\n // Bail if no support\n if (!support.pip) {\n return;\n }\n\n // Toggle based on current state if not passed\n const toggle = utils.is.boolean(input) ? input : this.pip === states.inline;\n\n // Toggle based on current state\n this.media.webkitSetPresentationMode(toggle ? states.pip : states.inline);\n }\n\n /**\n * Get the current picture-in-picture state\n */\n get pip() {\n if (!support.pip) {\n return null;\n }\n\n return this.media.webkitPresentationMode;\n }\n\n /**\n * Trigger the airplay dialog\n * TODO: update player with state, support, enabled\n */\n airplay() {\n // Show dialog if supported\n if (support.airplay) {\n this.media.webkitShowPlaybackTargetPicker();\n }\n }\n\n /**\n * Toggle the player controls\n * @param {boolean} toggle - Whether to show the controls\n */\n toggleControls(toggle) {\n // We need controls of course...\n if (!utils.is.element(this.elements.controls)) {\n return;\n }\n\n // Don't hide if no UI support or it's audio\n if (!this.supported.ui || this.isAudio) {\n return;\n }\n\n let delay = 0;\n let show = toggle;\n let isEnterFullscreen = false;\n\n // Get toggle state if not set\n if (!utils.is.boolean(toggle)) {\n if (utils.is.event(toggle)) {\n // Is the enter fullscreen event\n isEnterFullscreen = toggle.type === 'enterfullscreen';\n\n // Whether to show controls\n show = [\n 'mouseenter',\n 'mousemove',\n 'touchstart',\n 'touchmove',\n 'focusin',\n ].includes(toggle.type);\n\n // Delay hiding on move events\n if ([\n 'mousemove',\n 'touchmove',\n 'touchend',\n ].includes(toggle.type)) {\n delay = 2000;\n }\n\n // Delay a little more for keyboard users\n if (toggle.type === 'focusin') {\n delay = 3000;\n utils.toggleClass(this.elements.controls, this.config.classNames.noTransition, true);\n }\n } else {\n show = utils.hasClass(this.elements.container, this.config.classNames.hideControls);\n }\n }\n\n // Clear timer on every call\n window.clearTimeout(this.timers.controls);\n\n // If the mouse is not over the controls, set a timeout to hide them\n if (show || this.paused || this.loading) {\n // Check if controls toggled\n const toggled = utils.toggleClass(this.elements.container, this.config.classNames.hideControls, false);\n\n // Trigger event\n if (toggled) {\n utils.dispatchEvent.call(this, this.media, 'controlsshown');\n }\n\n // Always show controls when paused or if touch\n if (this.paused || this.loading) {\n return;\n }\n\n // Delay for hiding on touch\n if (support.touch) {\n delay = 3000;\n }\n }\n\n // If toggle is false or if we're playing (regardless of toggle),\n // then set the timer to hide the controls\n if (!show || this.playing) {\n this.timers.controls = window.setTimeout(() => {\n /* this.debug.warn({\n pressed: this.elements.controls.pressed,\n hover: this.elements.controls.pressed,\n playing: this.playing,\n paused: this.paused,\n loading: this.loading,\n }); */\n\n // If the mouse is over the controls (and not entering fullscreen), bail\n if ((this.elements.controls.pressed || this.elements.controls.hover) && !isEnterFullscreen) {\n return;\n }\n\n // Restore transition behaviour\n if (!utils.hasClass(this.elements.container, this.config.classNames.hideControls)) {\n utils.toggleClass(this.elements.controls, this.config.classNames.noTransition, false);\n }\n\n // Check if controls toggled\n const toggled = utils.toggleClass(this.elements.container, this.config.classNames.hideControls, true);\n\n // Trigger event and close menu\n if (toggled) {\n utils.dispatchEvent.call(this, this.media, 'controlshidden');\n\n if (this.config.controls.includes('settings') && !utils.is.empty(this.config.settings)) {\n controls.toggleMenu.call(this, false);\n }\n }\n }, delay);\n }\n }\n\n /**\n * Add event listeners\n * @param {string} event - Event type\n * @param {function} callback - Callback for when event occurs\n */\n on(event, callback) {\n utils.on(this.elements.container, event, callback);\n }\n\n /**\n * Remove event listeners\n * @param {string} event - Event type\n * @param {function} callback - Callback for when event occurs\n */\n off(event, callback) {\n utils.off(this.elements.container, event, callback);\n }\n\n /**\n * Destroy an instance\n * Event listeners are removed when elements are removed\n * http://stackoverflow.com/questions/12528049/if-a-dom-element-is-removed-are-its-listeners-also-removed-from-memory\n * @param {function} callback - Callback for when destroy is complete\n * @param {boolean} soft - Whether it's a soft destroy (for source changes etc)\n */\n destroy(callback, soft = false) {\n const done = () => {\n // Reset overflow (incase destroyed while in fullscreen)\n document.body.style.overflow = '';\n\n // GC for embed\n this.embed = null;\n\n // If it's a soft destroy, make minimal changes\n if (soft) {\n if (Object.keys(this.elements).length) {\n // Remove buttons\n if (this.elements.buttons && this.elements.buttons.play) {\n Array.from(this.elements.buttons.play).forEach(button => utils.removeElement(button));\n }\n\n // Remove others\n utils.removeElement(this.elements.captions);\n utils.removeElement(this.elements.controls);\n utils.removeElement(this.elements.wrapper);\n\n // Clear for GC\n this.elements.buttons.play = null;\n this.elements.captions = null;\n this.elements.controls = null;\n this.elements.wrapper = null;\n }\n\n // Callback\n if (utils.is.function(callback)) {\n callback();\n }\n } else {\n // Replace the container with the original element provided\n utils.replaceElement(this.elements.original, this.elements.container);\n\n // Event\n utils.dispatchEvent.call(this, this.elements.original, 'destroyed', true);\n\n // Callback\n if (utils.is.function(callback)) {\n callback.call(this.elements.original);\n }\n\n // Clear for GC\n this.elements = null;\n }\n };\n\n // Type specific stuff\n switch (`${this.provider}:${this.type}`) {\n case 'html5:video':\n case 'html5:audio':\n // Restore native video controls\n ui.toggleNativeControls.call(this, true);\n\n // Clean up\n done();\n\n break;\n\n case 'youtube:video':\n // Clear timers\n window.clearInterval(this.timers.buffering);\n window.clearInterval(this.timers.playing);\n\n // Destroy YouTube API\n if (this.embed !== null) {\n this.embed.destroy();\n }\n\n // Clean up\n done();\n\n break;\n\n case 'vimeo:video':\n // Destroy Vimeo API\n // then clean up (wait, to prevent postmessage errors)\n if (this.embed !== null) {\n this.embed.unload().then(done);\n }\n\n // Vimeo does not always return\n window.setTimeout(done, 200);\n\n break;\n\n default:\n break;\n }\n }\n\n /**\n * Check for support for a mime type (HTML5 only)\n * @param {string} type - Mime type\n */\n supports(type) {\n return support.mime.call(this, type);\n }\n\n /**\n * Check for support\n * @param {string} type - Player type (audio/video)\n * @param {string} provider - Provider (html5/youtube/vimeo)\n * @param {bool} inline - Where player has `playsinline` sttribute\n */\n static supported(type, provider, inline) {\n return support.check(type, provider, inline);\n }\n\n /**\n * Load an SVG sprite into the page\n * @param {string} url - URL for the SVG sprite\n * @param {string} [id] - Unique ID\n */\n static loadSprite(url, id) {\n return utils.loadSprite(url, id);\n }\n}\n\nexport default Plyr;\n"],"names":["element","type","range","value","providers","types","defaults","window","navigator","language","split","utils","input","this","instanceof","Plyr","getConstructor","Object","Number","isNaN","String","Boolean","Function","nullOrUndefined","Array","isArray","WeakMap","NodeList","Element","Text","Event","TextTrackCue","VTTCue","TextTrack","string","kind","array","nodeList","length","object","keys","constructor","document","documentMode","documentElement","style","test","userAgent","platform","url","callback","current","querySelector","callbacks","push","createElement","is","function","addEventListener","forEach","cb","call","event","src","first","getElementsByTagName","parentNode","insertBefore","id","hasId","querySelectorAll","container","toggleHidden","setAttribute","support","storage","cached","localStorage","getItem","data","JSON","parse","content","then","response","ok","text","setItem","stringify","catch","updateSprite","innerHTML","body","childNodes","prefix","Math","floor","random","self","top","e","elements","wrapper","targets","from","reverse","index","child","cloneNode","parent","sibling","nextSibling","appendChild","attributes","setAttributes","textContent","target","removeChild","lastChild","newChild","oldChild","replaceChild","empty","key","sel","existingAttributes","existing","selector","s","trim","className","replace","parts","charAt","class","toggle","contains","classList","removeAttribute","prototype","matches","webkitMatchesSelector","mozMatchesSelector","msMatchesSelector","includes","controls","getElement","config","selectors","buttons","getElements","play","pause","restart","rewind","forward","mute","pip","airplay","settings","captions","fullscreen","progress","inputs","seek","volume","display","buffer","duration","currentTime","seekTooltip","classNames","tooltip","error","debug","warn","toggleNativeControls","focused","activeElement","focusable","last","on","keyCode","_this","active","getFocusElement","shiftKey","focus","preventDefault","passive","capture","Node","toggleListener","events","options","boolean","passiveListeners","bubbles","detail","CustomEvent","assign","dispatchEvent","pressed","getAttribute","state","max","toFixed","sources","source","shift","extend","youtube","vimeo","match","RegExp","$2","number","parser","href","search","startsWith","parseUrl","slice","indexOf","reduce","params","hash","val","decodeURIComponent","map","encodeURIComponent","join","fragment","createDocumentFragment","firstChild","innerText","width","height","ratio","getRatio","w","h","find","undefined","setTimeout","offsetHeight","provider","inline","api","ui","browser","getBrowser","playsInline","isIPhone","video","rangeInput","audio","webkitSetPresentationMode","WebKitPlaybackTargetAvailabilityEvent","media","isHTML5","canPlayType","isVideo","isAudio","supported","defineProperty","transitionEnd","matchMedia","noop","Console","enabled","console","log","bind","Storage","player","store","json","get","removeItem","cancelFullScreen","some","pre","msExitFullscreen","msFullscreenEnabled","fullscreenEnabled","webkitFullscreenEnabled","mozFullScreenEnabled","fullscreenElement","mozFullScreenElement","requestFullScreen","nativeSupport","fallback","inFrame","toggleClass","toggleState","trapFocus","listeners","handleKey","code","which","repeat","altKey","ctrlKey","metaKey","editable","stopPropagation","togglePlay","increaseVolume","decreaseVolume","muted","toggleFullscreen","toggleCaptions","loop","keyboard","global","tabFocus","hideControls","toggleControls","eventType","timeUpdate","durationUpdate","_this2","hasAudio","showPosterOnEnd","load","updateProgress","updateVolume","checkPlaying","checkLoading","clickToPlay","touch","paused","ended","disableContextMenu","set","updateSetting","speed","quality","concat","inputEvent","isIE","proxy","handlerKey","defaultHandler","customHandler","_this3","defaultPrevented","toggleMenu","form","parseFloat","showTab","toggleInvert","invertTime","isWebkit","updateRangeFill","updateSeekTooltip","hover","inverted","webkitDirectionInvertedFromDevice","direction","deltaY","deltaX","uiSupported","removeElement","inject","setup","ready","setTitle","label","i18n","title","isEmbed","iframe","frameTitle","playing","stopped","button","loading","timers","failed","networkState","setRange","nodeValue","buffered","getPercentage","_this4","end","setProgress","time","format","getHours","parseInt","hours","mins","secs","invert","updateTimeDisplay","seeking","displayDuration","setProperty","iconUrl","svg4everybody","namespace","getIconUrl","iconPath","absolute","iconPrefix","icon","createElementNS","use","path","setAttributeNS","attr","hidden","badge","menu","buttonType","labelPressed","iconPressed","control","createIcon","createLabel","getAttributesFromSelector","suffix","played","toLowerCase","list","checked","item","radio","faux","aria-hidden","insertAdjacentHTML","tooltips","percent","clientRect","getBoundingClientRect","visible","pageX","left","hasClass","setting","tab","tabs","pane","panes","filter","isYouTube","toggleTab","emptyElement","createMenuItem","getLabel","createBadge","getBadge","getLanguage","default","textTracks","getTracks","none","currentTrack","getCurrentTrack","track","disabled","hasTracks","tracks","toUpperCase","unshift","show","isMenuItem","isButton","open","clone","position","opacity","name","scrollWidth","scrollHeight","getElementById","transitions","reducedMotion","size","getTabSize","restore","propertyName","off","createButton","createRange","createProgress","createTime","inner","home","back","setSpeedMenu","loadSprite","seekTime","create","findElements","repaint","labels","_this5","stored","setCaptionsMenu","insertAfter","setLanguage","setCue","mode","activeCues","isVimeo","embed","enableTextTrack","cue","setText","getCueAsHTML","caption","setAspectRatio","YT","Player","loadScript","urls","onYouTubeReadyCallbacks","onYouTubeIframeAPIReady","videoId","getVideoData","google","result","items","snippet","paddingBottom","currentId","parseYouTubeId","generateId","replaceElement","autoplay","location","hostname","message","instance","getPlaybackQuality","playbackRate","getPlaybackRate","getTitle","playVideo","pauseVideo","stop","stopVideo","getDuration","getCurrentTime","seekTo","setPlaybackRate","setPlaybackQuality","setVolume","getVideoUrl","getAvailablePlaybackRates","clearInterval","buffering","setInterval","getVideoLoadedFraction","lastBuffered","build","setQualityMenu","getAvailableQualityLevels","Vimeo","padding","offset","transform","buildUrlParams","parseVimeoId","setCurrentTime","selected","setLoop","currentSrc","all","getVideoWidth","getVideoHeight","getAspectRatio","dimensions","setAutopause","autopause","getVideoTitle","getTextTracks","cues","stripHTML","seconds","isIos","isTouch","wrap","blankVideo","insertElement","attribute","cancelRequests","destroy","html5","check","crossorigin","poster","addStyleHook","insertElements","scrollPosition","jQuery","plyr","original","tagName","getProviderByUrl","getUrlParams","truthy","playsinline","hasAttribute","step","isFullScreen","pageXOffset","pageYOffset","scrollTo","x","y","overflow","webkitShowPlaybackTargetPicker","delay","isEnterFullscreen","noTransition","clearTimeout","soft","done","unload","mime","readyState","targetTime","fauxDuration","realDuration","mozHasAudio","webkitAudioDecodedByteCount","audioTracks","change","states","webkitPresentationMode"],"mappings":"uLAIA,ICutBcA,EASAC,ECrlBAC,ECvINC,EHJKC,SACF,gBACE,gBACF,SAGEC,SACF,cACA,SIXLC,YAEO,QAGF,UAGA,YAGG,aAGC,WAGD,UAGF,SACD,WAGG,sBAIO,cAGL,gBAGE,QAGP,oBAGM,gBAGC,mBAGG,sBAGG,cAGR,aACA,eACH,iDAGG,wDAIC,mBAEL,SACA,SACA,SACA,QACA,QACA,SACA,QACA,OACA,0BAMI,mBAOE,WAEN,GACA,IACA,EACA,KACA,IACA,KACA,uBAMK,UACD,uBAKE,QACJ,qBAKE,WACEC,OAAOC,UAAUC,SAASC,MAAM,KAAK,yBAKtC,YACC,qBAKD,MACJ,kBAKL,aACA,OACA,WACA,eACA,OACA,SACA,WACA,WACA,MACA,UACA,wBAGA,WACA,UACA,uBAKS,iBACD,8BACF,aACC,gBACE,+BACH,cACE,kBACE,uBACG,wBACH,kBACF,cACF,cACE,wBACQ,kCACC,mCACA,kCACD,6BACJ,8BACF,oBACA,iBACH,gBACE,eACH,aACC,YACF,UACA,YACE,aACD,gBACI,6BAMD,uDAGA,uDAMH,UACA,WACC,aACE,YACD,aACC,UACH,YACE,cACE,gBACE,SACP,aACI,WACF,aACE,UACH,cACI,sBAQV,WACA,UACA,UACA,UACA,UACA,iBACA,YACA,aACA,iBACA,aACA,eACA,OACA,QACA,QACA,UACA,SACA,UACA,aACA,8BAIA,iBACA,kBACA,mBACA,iBACA,iBACA,gBACA,sBAIA,gBACA,wCAMU,uDACC,4BAEI,aACF,0BAEL,4BAEE,2BACC,8BACE,+BACD,+BACC,kCACH,8BACI,oCACE,+BACP,4BACI,iCACC,8BACJ,mCAGA,4BACE,6BACD,+BACG,iCACD,8CAGI,gCACH,+BACF,iCACA,+BACF,+BACE,mCAEF,2BACA,gCAEG,oDAMN,4BACA,4BACE,qBACH,qBACI,oBACD,wBACA,wBACA,sBACF,wBACA,sBACE,uBACD,6BACM,4BACP,uBACE,6BACI,6BACC,kCAEH,0BACA,mBACD,qCAGG,gCACD,6CAGC,oCACC,4CAGC,6BACH,u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"}
\ No newline at end of file diff --git a/package.json b/package.json index b65161a8..6dffeeef 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "plyr", - "version": "3.0.0-beta.1", + "version": "3.0.0-beta.2", "description": "A simple, accessible and customizable HTML5, YouTube and Vimeo media player", "homepage": "https://plyr.io", "main": "./dist", @@ -34,7 +34,7 @@ "gulp-util": "^3.0.8", "rollup-plugin-babel": "^3.0.3", "rollup-plugin-commonjs": "^8.2.6", - "rollup-plugin-node-resolve": "^3.0.0", + "rollup-plugin-node-resolve": "^3.0.2", "rollup-plugin-uglify": "^2.0.1", "run-sequence": "^2.2.1", "stylelint": "^8.4.0", @@ -46,15 +46,7 @@ "stylelint-selector-bem-pattern": "^2.0.0", "uglify-es": "^3.3.5" }, - "keywords": [ - "HTML5 Video", - "HTML5 Audio", - "Media Player", - "DASH", - "Shaka", - "WordPress", - "HLS" - ], + "keywords": ["HTML5 Video", "HTML5 Audio", "Media Player", "DASH", "Shaka", "WordPress", "HLS"], "repository": { "type": "git", "url": "git://github.com/sampotts/plyr.git" @@ -91,22 +91,24 @@ Plyr extends upon the standard HTML5 markup so that's all you need for those typ </audio> ``` -For YouTube and Vimeo, Plyr uses the standard YouTube API markup (an empty `<div>`): +For YouTube and Vimeo players, Plyr uses progressive enhancement to enhance the default `<iframe>` embeds. Below are some examples. The `plyr__video-embed` classname will make the embed responsive. You can add the `autoplay`, `loop` and `playsinline` (YouTube only) query parameters to the URL and they will be set as config options automatically. For YouTube, the `origin` should be updated to reflect the domain you're hosting the embed on, or you can opt to omit it. #### YouTube embed ```html -<div id="player" data-plyr-provider="youtube" data-plyr-embed-id="bTqVqk7FSmY"></div> +<div class="plyr__video-embed" id="player"> + <iframe src="https://www.youtube.com/embed/bTqVqk7FSmY?origin=https://plyr.io&iv_load_policy=3&modestbranding=1&playsinline=1&showinfo=0&rel=0&enablejsapi=1" allowfullscreen allowtransparency allow="autoplay"></iframe> +</div> ``` #### Vimeo embed ```html -<div id="player" data-plyr-provider="vimeo" data-plyr-embed-id="143418951"></div> +<div class="plyr__video-embed" id="player"> + <iframe src="https://player.vimeo.com/video/76979871?loop=false&byline=false&portrait=false&title=false&speed=true&transparent=0&gesture=media" allowfullscreen allowtransparency allow="autoplay"></iframe> +</div> ``` -Note: In both cases, `data-plyr-embed-id` value can be the ID or URL for the media. - ### JavaScript Include the `plyr.js` script before the closing `</body>` tag and then call `plyr.setup()`. More info on `setup()` can be found under @@ -120,7 +122,7 @@ Include the `plyr.js` script before the closing `</body>` tag and then call `ply If you want to use our CDN (provided by [Fastly](https://www.fastly.com/)) for the JavaScript, you can use the following: ```html -<script src="https://cdn.plyr.io/3.0.0-beta.1/plyr.js"></script> +<script src="https://cdn.plyr.io/3.0.0-beta.2/plyr.js"></script> ``` ### CSS @@ -134,13 +136,13 @@ Include the `plyr.css` stylsheet into your `<head>` If you want to use our CDN (provided by [Fastly](https://www.fastly.com/)) for the default CSS, you can use the following: ```html -<link rel="stylesheet" href="https://cdn.plyr.io/3.0.0-beta.1/plyr.css"> +<link rel="stylesheet" href="https://cdn.plyr.io/3.0.0-beta.2/plyr.css"> ``` ### SVG Sprite The SVG sprite is loaded automatically from our CDN (provided by [Fastly](https://www.fastly.com/)). To change this, see the [options](#options) below. For -reference, the CDN hosted SVG sprite can be found at `https://cdn.plyr.io/3.0.0-beta.1/plyr.svg`. +reference, the CDN hosted SVG sprite can be found at `https://cdn.plyr.io/3.0.0-beta.2/plyr.svg`. ## Advanced @@ -211,8 +213,7 @@ Passing a [NodeList](https://developer.mozilla.org/en-US/docs/Web/API/NodeList): const player = new Plyr(document.querySelectorAll('.js-player')); ``` -The NodeList, HTMLElement or string selector can be the target `<video>`, `<audio>` or `[data-plyr-provider]` (for embeds) element itself or a container -element. +The NodeList, HTMLElement or string selector can be the target `<video>`, `<audio>`, or `<div>` wrapper for embeds The second argument for the constructor is the [#options](options) object: diff --git a/src/js/console.js b/src/js/console.js index c5389970..7c5ec1b4 100644 --- a/src/js/console.js +++ b/src/js/console.js @@ -5,8 +5,8 @@ const noop = () => {}; export default class Console { - constructor(player) { - this.enabled = window.console && player.config.debug; + constructor(enabled = false) { + this.enabled = window.console && enabled; if (this.enabled) { this.log('Debugging enabled'); diff --git a/src/js/media.js b/src/js/media.js index 4019c1a7..3fbd9774 100644 --- a/src/js/media.js +++ b/src/js/media.js @@ -50,7 +50,7 @@ const media = { } // Inject the player wrapper - if (this.isVideo || this.isYouTube || this.isVimeo) { + if (this.isVideo) { // Create the wrapper div this.elements.wrapper = utils.createElement('div', { class: this.config.classNames.video, diff --git a/src/js/plugins/vimeo.js b/src/js/plugins/vimeo.js index c77ecd20..48d46037 100644 --- a/src/js/plugins/vimeo.js +++ b/src/js/plugins/vimeo.js @@ -8,19 +8,12 @@ import ui from './../ui'; const vimeo = { setup() { - // Remove old containers - const containers = utils.getElements.call(this, `[id^="${this.provider}-"]`); - Array.from(containers).forEach(utils.removeElement); - // Add embed class for responsive utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true); // Set intial ratio vimeo.setAspectRatio.call(this); - // Set ID - this.media.setAttribute('id', utils.generateId(this.provider)); - // Load the API if not already if (!utils.is.object(window.Vimeo)) { utils.loadScript(this.config.urls.vimeo.api, () => { @@ -57,15 +50,21 @@ const vimeo = { transparent: 0, gesture: 'media', }; - const params = utils.buildUrlParameters(options); - const id = utils.parseVimeoId(player.embedId); + const params = utils.buildUrlParams(options); + const id = utils.parseVimeoId(player.media.getAttribute('src')); // Build an iframe const iframe = utils.createElement('iframe'); const src = `https://player.vimeo.com/video/${id}?${params}`; iframe.setAttribute('src', src); iframe.setAttribute('allowfullscreen', ''); - player.media.appendChild(iframe); + iframe.setAttribute('allowtransparency', ''); + iframe.setAttribute('allow', 'autoplay'); + + // Inject the package + const wrapper = utils.createElement('div'); + wrapper.appendChild(iframe); + player.media = utils.replaceElement(wrapper, player.media); // Setup instance // https://github.com/vimeo/player.js diff --git a/src/js/plugins/youtube.js b/src/js/plugins/youtube.js index 67f1ca95..bec342a7 100644 --- a/src/js/plugins/youtube.js +++ b/src/js/plugins/youtube.js @@ -8,24 +8,15 @@ import ui from './../ui'; const youtube = { setup() { - const videoId = utils.parseYouTubeId(this.embedId); - - // Remove old containers - const containers = utils.getElements.call(this, `[id^="${this.provider}-"]`); - Array.from(containers).forEach(utils.removeElement); - // Add embed class for responsive utils.toggleClass(this.elements.wrapper, this.config.classNames.embed, true); // Set aspect ratio youtube.setAspectRatio.call(this); - // Set ID - this.media.setAttribute('id', utils.generateId(this.provider)); - // Setup API - if (utils.is.object(window.YT)) { - youtube.ready.call(this, videoId); + if (utils.is.object(window.YT) && utils.is.function(window.YT.Player)) { + youtube.ready.call(this); } else { // Load the API utils.loadScript(this.config.urls.youtube.api); @@ -36,7 +27,7 @@ const youtube = { // Add to queue window.onYouTubeReadyCallbacks.push(() => { - youtube.ready.call(this, videoId); + youtube.ready.call(this); }); // Set callback to process queue @@ -49,7 +40,7 @@ const youtube = { }, // Get the media title - getTitle() { + getTitle(videoId) { // Try via undocumented API method first // This method disappears now and then though... // https://github.com/sampotts/plyr/issues/709 @@ -65,7 +56,6 @@ const youtube = { // Or via Google API const key = this.config.keys.google; - const videoId = utils.parseYouTubeId(this.embedId); if (utils.is.string(key) && !utils.is.empty(key)) { const url = `https://www.googleapis.com/youtube/v3/videos?id=${videoId}&key=${key}&fields=items(snippet(title))&part=snippet`; @@ -88,12 +78,24 @@ const youtube = { }, // API ready - ready(videoId) { + ready() { const player = this; + // Ignore already setup (race condition) + const currentId = player.media.getAttribute('id'); + if (!utils.is.empty(currentId) && currentId.startsWith('youtube-')) { + return; + } + + // Replace the <iframe> with a <div> due to YouTube API issues + const videoId = utils.parseYouTubeId(player.media.getAttribute('src')); + const id = utils.generateId(player.provider); + const container = utils.createElement('div', { id }); + player.media = utils.replaceElement(container, player.media); + // Setup instance // https://developers.google.com/youtube/iframe_api_reference - player.embed = new window.YT.Player(player.media.id, { + player.embed = new window.YT.Player(id, { videoId, playerVars: { autoplay: player.config.autoplay ? 1 : 0, // Autoplay @@ -110,8 +112,8 @@ const youtube = { widget_referrer: window && window.location.href, // Captions are flaky on YouTube - cc_load_policy: this.captions.active ? 1 : 0, - cc_lang_pref: this.config.captions.language, + cc_load_policy: player.captions.active ? 1 : 0, + cc_lang_pref: player.config.captions.language, }, events: { onError(event) { @@ -179,7 +181,7 @@ const youtube = { const instance = event.target; // Get the title - youtube.getTitle.call(player); + youtube.getTitle.call(player, videoId); // Create a faux HTML5 API using the YouTube API player.media.play = () => { diff --git a/src/js/plyr.js b/src/js/plyr.js index 0bb0a89c..dfb07302 100644 --- a/src/js/plyr.js +++ b/src/js/plyr.js @@ -1,6 +1,6 @@ // ========================================================================== // Plyr -// plyr.js v3.0.0-beta.1 +// plyr.js v3.0.0-beta.2 // https://github.com/sampotts/plyr // License: The MIT License (MIT) // ========================================================================== @@ -66,7 +66,7 @@ class Plyr { } catch (e) { return {}; } - })() + })(), ); // Elements cache @@ -103,7 +103,7 @@ class Plyr { // Debugging // TODO: move to globals - this.debug = new Console(this); + this.debug = new Console(this.config.debug); // Log config options and support this.debug.log('Config', this.config); @@ -141,35 +141,61 @@ class Plyr { // Supported: video, audio, vimeo, youtube const type = this.media.tagName.toLowerCase(); - // Embed attributes - const attributes = { - provider: 'data-plyr-provider', - id: 'data-plyr-embed-id', - }; + // Embed properties + let iframe = null; + let url = null; + let params = null; // Different setup based on type switch (type) { - // TODO: Handle passing an iframe for true progressive enhancement - // case 'iframe': case 'div': - this.type = types.video; // Audio will come later for external providers - this.provider = this.media.getAttribute(attributes.provider); - this.embedId = this.media.getAttribute(attributes.id); + // Find the frame + iframe = this.media.querySelector('iframe'); - if (utils.is.empty(this.provider) || !Object.keys(providers).includes(this.provider)) { - this.debug.error('Setup failed: Invalid provider'); + // <iframe> required + if (!utils.is.element(iframe)) { + this.debug.error('Setup failed: <iframe> is missing'); return; } - // Try and get the embed id - if (utils.is.empty(this.embedId)) { - this.debug.error('Setup failed: Embed ID or URL missing'); + // Audio will come later for external providers + this.type = types.video; + + // Detect provider + url = iframe.getAttribute('src'); + this.provider = utils.getProviderByUrl(url); + + // Get attributes from URL and set config + params = utils.getUrlParams(url); + if (!utils.is.empty(params)) { + const truthy = [ + '1', + 'true', + ]; + + if (truthy.includes(params.autoplay)) { + this.config.autoplay = true; + } + if (truthy.includes(params.playsinline)) { + this.config.inline = true; + } + if (truthy.includes(params.loop)) { + this.config.loop.active = true; + } + } + + // Unsupported provider + if (utils.is.empty(this.provider) || !Object.keys(providers).includes(this.provider)) { + this.debug.error('Setup failed: Invalid provider'); return; } - // Clean up - this.media.removeAttribute(attributes.provider); - this.media.removeAttribute(attributes.id); + // Rework elements + this.elements.container = this.media; + this.media = iframe; + + // Reset classname + this.elements.container.className = ''; break; @@ -178,22 +204,19 @@ class Plyr { this.type = type; this.provider = providers.html5; + // Get config from attributes if (this.media.hasAttribute('crossorigin')) { this.config.crossorigin = true; } - if (this.media.hasAttribute('autoplay')) { this.config.autoplay = true; } - if (this.media.hasAttribute('playsinline')) { this.config.inline = true; } - if (this.media.hasAttribute('muted')) { this.config.muted = true; } - if (this.media.hasAttribute('loop')) { this.config.loop.active = true; } @@ -221,8 +244,10 @@ class Plyr { this.media.plyr = this; // Wrap media - this.elements.container = utils.createElement('div'); - utils.wrap(this.media, this.elements.container); + if (!utils.is.element(this.elements.container)) { + this.elements.container = utils.createElement('div'); + utils.wrap(this.media, this.elements.container); + } // Allow focus to be captured this.elements.container.setAttribute('tabindex', 0); @@ -1054,7 +1079,6 @@ class Plyr { // GC for embed this.embed = null; - this.embedId = null; // If it's a soft destroy, make minimal changes if (soft) { @@ -1082,11 +1106,7 @@ class Plyr { } } else { // Replace the container with the original element provided - const parent = this.elements.container.parentNode; - - if (utils.is.element(parent)) { - parent.replaceChild(this.elements.original, this.elements.container); - } + utils.replaceElement(this.elements.original, this.elements.container); // Event utils.dispatchEvent.call(this, this.elements.original, 'destroyed', true); @@ -1119,7 +1139,9 @@ class Plyr { window.clearInterval(this.timers.playing); // Destroy YouTube API - this.embed.destroy(); + if (this.embed !== null) { + this.embed.destroy(); + } // Clean up done(); @@ -1129,7 +1151,9 @@ class Plyr { case 'vimeo:video': // Destroy Vimeo API // then clean up (wait, to prevent postmessage errors) - this.embed.unload().then(done); + if (this.embed !== null) { + this.embed.unload().then(done); + } // Vimeo does not always return window.setTimeout(done, 200); diff --git a/src/js/source.js b/src/js/source.js index 80620bdf..9a6b219c 100644 --- a/src/js/source.js +++ b/src/js/source.js @@ -67,8 +67,9 @@ const source = { case 'youtube:video': case 'vimeo:video': - this.media = utils.createElement('div'); - this.embedId = input.sources[0].src; + this.media = utils.createElement('div', { + src: input.sources[0].src, + }); break; default: @@ -136,7 +137,7 @@ const source = { ui.build.call(this); } }, - true + true, ); }, }; diff --git a/src/js/utils.js b/src/js/utils.js index d9dd3df1..930d3e9f 100644 --- a/src/js/utils.js +++ b/src/js/utils.js @@ -3,6 +3,7 @@ // ========================================================================== import support from './support'; +import { providers } from './types'; const utils = { // Check variable types @@ -103,7 +104,7 @@ const utils = { element.callbacks.forEach(cb => cb.call(null, event)); element.callbacks = null; }, - false + false, ); } @@ -168,7 +169,7 @@ const utils = { prefix + id, JSON.stringify({ content: text, - }) + }), ); } @@ -274,6 +275,17 @@ const utils = { } }, + // Replace element + replaceElement(newChild, oldChild) { + if (!utils.is.element(oldChild) || !utils.is.element(oldChild.parentNode) || !utils.is.element(newChild)) { + return null; + } + + oldChild.parentNode.replaceChild(newChild, oldChild); + + return newChild; + }, + // Set attributes setAttributes(element, attributes) { if (!utils.is.element(element) || utils.is.empty(attributes)) { @@ -491,7 +503,7 @@ const utils = { event.preventDefault(); } }, - false + false, ); }, @@ -617,14 +629,37 @@ const utils = { return utils.extend(target, ...sources); }, + // Get the provider for a given URL + getProviderByUrl(url) { + // YouTube + if (/^(https?:\/\/)?(www\.)?(youtube\.com|youtu\.?be)\/.+$/.test(url)) { + return providers.youtube; + } + + // Vimeo + if (/^https?:\/\/player.vimeo.com\/video\/\d{8,}(?=\b|\/)/.test(url)) { + return providers.vimeo; + } + + return null; + }, + // Parse YouTube ID from URL parseYouTubeId(url) { + if (utils.is.empty(url)) { + return null; + } + const regex = /^.*(youtu.be\/|v\/|u\/\w\/|embed\/|watch\?v=|&v=)([^#&?]*).*/; return url.match(regex) ? RegExp.$2 : url; }, // Parse Vimeo ID from URL parseVimeoId(url) { + if (utils.is.empty(url)) { + return null; + } + if (utils.is.number(Number(url))) { return url; } @@ -633,8 +668,40 @@ const utils = { return url.match(regex) ? RegExp.$2 : url; }, + // Convert a URL to a location object + parseUrl(url) { + const parser = document.createElement('a'); + parser.href = url; + return parser; + }, + + // Get URL query parameters + getUrlParams(input) { + let search = input; + + // Parse URL if needed + if (input.startsWith('http://') || input.startsWith('https://')) { + ({ search } = this.parseUrl(input)); + } + + if (this.is.empty(search)) { + return null; + } + + const hashes = search.slice(search.indexOf('?') + 1).split('&'); + + return hashes.reduce((params, hash) => { + const [ + key, + val, + ] = hash.split('='); + + return Object.assign(params, { [key]: decodeURIComponent(val) }); + }, {}); + }, + // Convert object to URL parameters - buildUrlParameters(input) { + buildUrlParams(input) { if (!utils.is.object(input)) { return ''; } diff --git a/src/sass/components/embed.scss b/src/sass/components/embed.scss index d9349052..56916f17 100644 --- a/src/sass/components/embed.scss +++ b/src/sass/components/embed.scss @@ -11,6 +11,7 @@ height: 0; padding-bottom: to-percentage($padding); + position: relative; iframe { border: 0; |