You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 

509 lines
15 KiB

  1. /*!
  2. ckin v0.0.1: Custom HTML5 Media Player Skins.
  3. (c) 2017
  4. MIT License
  5. git+https://github.com/hunzaboy/ckin.git
  6. */
  7. // Source: https://gist.github.com/k-gun/c2ea7c49edf7b757fe9561ba37cb19ca;
  8. (function () {
  9. // helpers
  10. var regExp = function regExp(name) {
  11. return new RegExp('(^| )' + name + '( |$)');
  12. };
  13. var forEach = function forEach(list, fn, scope) {
  14. for (var i = 0; i < list.length; i++) {
  15. fn.call(scope, list[i]);
  16. }
  17. };
  18. // class list object with basic methods
  19. function ClassList(element) {
  20. this.element = element;
  21. }
  22. ClassList.prototype = {
  23. add: function add() {
  24. forEach(arguments, function (name) {
  25. if (!this.contains(name)) {
  26. this.element.className += ' ' + name;
  27. }
  28. }, this);
  29. },
  30. remove: function remove() {
  31. forEach(arguments, function (name) {
  32. this.element.className = this.element.className.replace(regExp(name), '');
  33. }, this);
  34. },
  35. toggle: function toggle(name) {
  36. return this.contains(name) ? (this.remove(name), false) : (this.add(name), true);
  37. },
  38. contains: function contains(name) {
  39. return regExp(name).test(this.element.className);
  40. },
  41. // bonus..
  42. replace: function replace(oldName, newName) {
  43. this.remove(oldName), this.add(newName);
  44. }
  45. };
  46. // IE8/9, Safari
  47. if (!('classList' in Element.prototype)) {
  48. Object.defineProperty(Element.prototype, 'classList', {
  49. get: function get() {
  50. return new ClassList(this);
  51. }
  52. });
  53. }
  54. // replace() support for others
  55. if (window.DOMTokenList && DOMTokenList.prototype.replace == null) {
  56. DOMTokenList.prototype.replace = ClassList.prototype.replace;
  57. }
  58. })();
  59. (function () {
  60. if (typeof NodeList.prototype.forEach === "function") return false;
  61. NodeList.prototype.forEach = Array.prototype.forEach;
  62. })();
  63. // Unfortunately, due to scattered support, browser sniffing is required
  64. function browserSniff() {
  65. var nVer = navigator.appVersion,
  66. nAgt = navigator.userAgent,
  67. browserName = navigator.appName,
  68. fullVersion = '' + parseFloat(navigator.appVersion),
  69. majorVersion = parseInt(navigator.appVersion, 10),
  70. nameOffset,
  71. verOffset,
  72. ix;
  73. // MSIE 11
  74. if (navigator.appVersion.indexOf("Windows NT") !== -1 && navigator.appVersion.indexOf("rv:11") !== -1) {
  75. browserName = "IE";
  76. fullVersion = "11;";
  77. }
  78. // MSIE
  79. else if ((verOffset = nAgt.indexOf("MSIE")) !== -1) {
  80. browserName = "IE";
  81. fullVersion = nAgt.substring(verOffset + 5);
  82. }
  83. // Chrome
  84. else if ((verOffset = nAgt.indexOf("Chrome")) !== -1) {
  85. browserName = "Chrome";
  86. fullVersion = nAgt.substring(verOffset + 7);
  87. }
  88. // Safari
  89. else if ((verOffset = nAgt.indexOf("Safari")) !== -1) {
  90. browserName = "Safari";
  91. fullVersion = nAgt.substring(verOffset + 7);
  92. if ((verOffset = nAgt.indexOf("Version")) !== -1) {
  93. fullVersion = nAgt.substring(verOffset + 8);
  94. }
  95. }
  96. // Firefox
  97. else if ((verOffset = nAgt.indexOf("Firefox")) !== -1) {
  98. browserName = "Firefox";
  99. fullVersion = nAgt.substring(verOffset + 8);
  100. }
  101. // In most other browsers, "name/version" is at the end of userAgent
  102. else if ((nameOffset = nAgt.lastIndexOf(' ') + 1) < (verOffset = nAgt.lastIndexOf('/'))) {
  103. browserName = nAgt.substring(nameOffset, verOffset);
  104. fullVersion = nAgt.substring(verOffset + 1);
  105. if (browserName.toLowerCase() == browserName.toUpperCase()) {
  106. browserName = navigator.appName;
  107. }
  108. }
  109. // Trim the fullVersion string at semicolon/space if present
  110. if ((ix = fullVersion.indexOf(";")) !== -1) {
  111. fullVersion = fullVersion.substring(0, ix);
  112. }
  113. if ((ix = fullVersion.indexOf(" ")) !== -1) {
  114. fullVersion = fullVersion.substring(0, ix);
  115. }
  116. // Get major version
  117. majorVersion = parseInt('' + fullVersion, 10);
  118. if (isNaN(majorVersion)) {
  119. fullVersion = '' + parseFloat(navigator.appVersion);
  120. majorVersion = parseInt(navigator.appVersion, 10);
  121. }
  122. // Return data
  123. return [browserName, majorVersion];
  124. }
  125. var obj = {};
  126. obj.browserInfo = browserSniff();
  127. obj.browserName = obj.browserInfo[0];
  128. obj.browserVersion = obj.browserInfo[1];
  129. wrapPlayers();
  130. /* Get Our Elements */
  131. var players = document.querySelectorAll('.ckin__player');
  132. var iconPlay = '<i class="ckin-play"></i>';
  133. var iconPause = '<i class="ckin-pause"></i>';
  134. var iconVolumeMute = '<i class="ckin-volume-mute"></i>';
  135. var iconVolumeHigh = '<i class="ckin-volume-high"></i>';
  136. var iconVolumeMedium = '<i class="ckin-volume-medium"></i>';
  137. var iconVolumeLow = '<i class="ckin-volume-low"></i>';
  138. var iconExpand = '<i class="ckin-expand"></i>';
  139. var iconCompress = '<i class="ckin-compress"></i>';
  140. players.forEach(function (player) {
  141. var video = player.querySelector('video');
  142. if (null === video) {
  143. return;
  144. }
  145. var skin = attachSkin(video.dataset.ckin);
  146. player.classList.add(skin);
  147. var overlay = video.dataset.overlay;
  148. addOverlay(player, overlay);
  149. var title = showTitle(skin, video.dataset.title);
  150. if (title) {
  151. player.insertAdjacentHTML('beforeend', title);
  152. }
  153. var html = buildControls(skin);
  154. player.insertAdjacentHTML('beforeend', html);
  155. var color = video.dataset.color;
  156. addColor(player, color);
  157. var playerControls = player.querySelector('.' + skin + '__controls');
  158. var progress = player.querySelector('.progress');;
  159. var progressBar = player.querySelector('.progress__filled');
  160. var progressTime = player.querySelector('.progress__time');
  161. var toggle = player.querySelectorAll('.toggle');
  162. var skipButtons = player.querySelectorAll('[data-skip]');
  163. var ranges = player.querySelectorAll('.' + skin + '__slider');
  164. var volumeButton = player.querySelector('.volume');
  165. var fullScreenButton = player.querySelector('.fullscreen');
  166. if (obj.browserName === "IE" && (obj.browserVersion === 8 || obj.browserVersion === 9)) {
  167. showControls(video);
  168. playerControls.style.display = "none";
  169. }
  170. video.addEventListener('click', function () {
  171. togglePlay(this, player, true);
  172. });
  173. video.addEventListener('play', function () {
  174. updateButton(this, toggle);
  175. });
  176. video.addEventListener('pause', function () {
  177. updateButton(this, toggle);
  178. });
  179. video.addEventListener('timeupdate', function () {
  180. handleProgress(this, progressBar, progressTime);
  181. });
  182. toggle.forEach(function (button) {
  183. return button.addEventListener('click', function () {
  184. togglePlay(video, player, true);
  185. });
  186. });
  187. volumeButton.addEventListener('click', function () {
  188. toggleVolume(video, volumeButton);
  189. });
  190. var mousedown = false;
  191. progress.addEventListener('click', function (e) {
  192. scrub(e, video, progress, progressTime);
  193. });
  194. progress.addEventListener('mousemove', function (e) {
  195. return mousedown && scrub(e, video, progress, progressTime);
  196. });
  197. progress.addEventListener('mousedown', function () {
  198. return mousedown = true;
  199. });
  200. progress.addEventListener('mouseup', function () {
  201. return mousedown = false;
  202. });
  203. fullScreenButton.addEventListener('click', function (e) {
  204. return toggleFullScreen(player, fullScreenButton);
  205. });
  206. addListenerMulti(player, 'webkitfullscreenchange mozfullscreenchange fullscreenchange MSFullscreenChange', function (e) {
  207. return onFullScreen(e, player);
  208. });
  209. });
  210. players.forEach(function (player) {
  211. var audio = player.querySelector('audio');
  212. if (null === audio) {
  213. return;
  214. }
  215. var skin = attachSkin(audio.dataset.ckin);
  216. player.classList.add(skin);
  217. var overlay = audio.dataset.overlay;
  218. addOverlay(player, overlay);
  219. var title = showTitle(skin, audio.dataset.title);
  220. if (title) {
  221. player.insertAdjacentHTML('beforeend', title);
  222. }
  223. var html = buildControls(skin);
  224. player.insertAdjacentHTML('beforeend', html);
  225. var color = audio.dataset.color;
  226. addColor(player, color);
  227. var playerControls = player.querySelector('.' + skin + '__controls');
  228. var progress = player.querySelector('.progress');;
  229. var progressBar = player.querySelector('.progress__filled');
  230. var progressTime = player.querySelector('.progress__time');
  231. var toggle = player.querySelectorAll('.toggle');
  232. var skipButtons = player.querySelectorAll('[data-skip]');
  233. var ranges = player.querySelectorAll('.' + skin + '__slider');
  234. var volumeButton = player.querySelector('.volume');
  235. var fullScreenButton = player.querySelector('.fullscreen');
  236. if (obj.browserName === "IE" && (obj.browserVersion === 8 || obj.browserVersion === 9)) {
  237. showControls(audio);
  238. playerControls.style.display = "none";
  239. }
  240. audio.addEventListener('click', function () {
  241. togglePlay(this, player, false);
  242. });
  243. audio.addEventListener('play', function () {
  244. updateButton(this, toggle);
  245. });
  246. audio.addEventListener('pause', function () {
  247. updateButton(this, toggle);
  248. });
  249. audio.addEventListener('timeupdate', function () {
  250. handleProgress(this, progressBar, progressTime);
  251. });
  252. toggle.forEach(function (button) {
  253. return button.addEventListener('click', function () {
  254. togglePlay(audio, player, false);
  255. });
  256. });
  257. volumeButton.addEventListener('click', function () {
  258. toggleVolume(audio, volumeButton);
  259. });
  260. var mousedown = false;
  261. progress.addEventListener('click', function (e) {
  262. scrub(e, audio, progress, progressTime);
  263. });
  264. progress.addEventListener('mousemove', function (e) {
  265. return mousedown && scrub(e, audio, progress, progressTime);
  266. });
  267. progress.addEventListener('mousedown', function () {
  268. return mousedown = true;
  269. });
  270. progress.addEventListener('mouseup', function () {
  271. return mousedown = false;
  272. });
  273. fullScreenButton.addEventListener('click', function (e) {
  274. return toggleFullScreen(player, fullScreenButton);
  275. });
  276. addListenerMulti(player, 'webkitfullscreenchange mozfullscreenchange fullscreenchange MSFullscreenChange', function (e) {
  277. return onFullScreen(e, player);
  278. });
  279. });
  280. function showControls(media) {
  281. media.setAttribute("controls", "controls");
  282. }
  283. function togglePlay(media, player, autoHide) {
  284. var method = media.paused ? 'play' : 'pause';
  285. media[method]();
  286. if (autoHide) {
  287. media.paused ? player.classList.remove('is-playing') : player.classList.add('is-playing');
  288. }
  289. }
  290. function updateButton(media, toggle) {
  291. var icon = media.paused ? iconPlay : iconPause;
  292. toggle.forEach(function (button) {
  293. return button.innerHTML = icon;
  294. });
  295. }
  296. function skip() {
  297. media.currentTime += parseFloat(this.dataset.skip);
  298. }
  299. function toggleVolume(media, volumeButton) {
  300. var level = media.volume;
  301. var icon = iconVolumeMedium;
  302. if (level == 1) {
  303. level = 0;
  304. icon = iconVolumeMute;
  305. } else if (level == 0.66) {
  306. level = 1;
  307. icon = iconVolumeHigh;
  308. } else if (level == 0.33) {
  309. level = 0.66;
  310. icon = iconVolumeMedium;
  311. } else {
  312. level = 0.33;
  313. icon = iconVolumeLow;
  314. }
  315. media['volume'] = level;
  316. volumeButton.innerHTML = icon;
  317. }
  318. function handleRangeUpdate() {
  319. media[this.name] = this.value;
  320. }
  321. function handleProgress(media, progressBar, progressTime) {
  322. var percent = media.currentTime / media.duration * 100;
  323. progressBar.style.flexBasis = percent + '%';
  324. progressTime.innerText = formatTime(media.currentTime) + "/" + formatTime(media.duration);
  325. }
  326. function scrub(e, media, progress, progressTime) {
  327. var scrubTime = e.offsetX / progress.offsetWidth * media.duration;
  328. media.currentTime = scrubTime;
  329. progressTime.innerText = formatTime(media.currentTime) + "/" + formatTime(media.duration);
  330. }
  331. function formatTime(time) {
  332. var date = new Date(null);
  333. date.setSeconds(time); // specify value for SECONDS here
  334. return date.toISOString().substr(11, 8);
  335. }
  336. function wrapPlayers() {
  337. var videos = document.querySelectorAll('video');
  338. videos.forEach(function (video) {
  339. var wrapper = document.createElement('div');
  340. wrapper.classList.add('ckin__player');
  341. video.parentNode.insertBefore(wrapper, video);
  342. wrapper.appendChild(video);
  343. });
  344. var audios = document.querySelectorAll('audio');
  345. audios.forEach(function (audio) {
  346. var wrapper = document.createElement('div');
  347. wrapper.classList.add('ckin__player');
  348. wrapper.classList.add('audio');
  349. audio.parentNode.insertBefore(wrapper, audio);
  350. wrapper.appendChild(audio);
  351. });
  352. }
  353. function buildControls(skin) {
  354. var html = [];
  355. html.push('<button class="' + skin + '__button--big toggle" title="Toggle Play">' + iconPlay + '</button>');
  356. html.push('<div class="' + skin + '__controls ckin__controls">');
  357. html.push('<button class="' + skin + '__button toggle" title="Toggle Media">' + iconPlay + '</button>', '<div class="progress">', '<div class="progress__filled"></div>', '<p class="progress__breaker"></p>', '<div class="progress__time"></div>', '</div>', '<button class="' + skin + '__button volume" title="Volume">' + iconVolumeMedium + '</button>', '<button class="' + skin + '__button fullscreen" title="Full Screen">' + iconExpand + '</button>');
  358. html.push('</div>');
  359. return html.join('');
  360. }
  361. function attachSkin(skin) {
  362. if (typeof skin != 'undefined' && skin != '') {
  363. return skin;
  364. } else {
  365. return 'default';
  366. }
  367. }
  368. function showTitle(skin, title) {
  369. if (typeof title != 'undefined' && title != '') {
  370. return '<div class="' + skin + '__title">' + title + '</div>';
  371. } else {
  372. return false;
  373. }
  374. }
  375. function addOverlay(player, overlay) {
  376. if (overlay == 1) {
  377. player.classList.add('ckin__overlay');
  378. } else if (overlay == 2) {
  379. player.classList.add('ckin__overlay--2');
  380. } else {
  381. return;
  382. }
  383. }
  384. function addColor(player, color) {
  385. if (typeof color != 'undefined' && color != '') {
  386. var buttons = player.querySelectorAll('button');
  387. var progress = player.querySelector('.progress__filled');
  388. progress.style.background = color;
  389. buttons.forEach(function (button) {
  390. return button.style.color = color;
  391. });
  392. }
  393. }
  394. function toggleFullScreen(player, fullScreenButton) {
  395. // let isFullscreen = false;
  396. if (!document.fullscreenElement && // alternative standard method
  397. !document.mozFullScreenElement && !document.webkitFullscreenElement && !document.msFullscreenElement) {
  398. player.classList.add('ckin__fullscreen');
  399. if (player.requestFullscreen) {
  400. player.requestFullscreen();
  401. } else if (player.mozRequestFullScreen) {
  402. player.mozRequestFullScreen(); // Firefox
  403. } else if (player.webkitRequestFullscreen) {
  404. player.webkitRequestFullscreen(); // Chrome and Safari
  405. } else if (player.msRequestFullscreen) {
  406. player.msRequestFullscreen();
  407. }
  408. isFullscreen = true;
  409. fullScreenButton.innerHTML = iconCompress;
  410. } else {
  411. player.classList.remove('ckin__fullscreen');
  412. if (document.cancelFullScreen) {
  413. document.cancelFullScreen();
  414. } else if (document.mozCancelFullScreen) {
  415. document.mozCancelFullScreen();
  416. } else if (document.webkitCancelFullScreen) {
  417. document.webkitCancelFullScreen();
  418. } else if (document.msExitFullscreen) {
  419. document.msExitFullscreen();
  420. }
  421. isFullscreen = false;
  422. fullScreenButton.innerHTML = iconExpand;
  423. }
  424. }
  425. function onFullScreen(e, player) {
  426. var isFullscreenNow = document.webkitFullscreenElement !== null;
  427. if (!isFullscreenNow) {
  428. player.classList.remove('ckin__fullscreen');
  429. player.querySelector('.fullscreen').innerHTML = iconExpand;
  430. } else {
  431. // player.querySelector('.fullscreen').innerHTML = iconExpand;
  432. }
  433. }
  434. function addListenerMulti(element, eventNames, listener) {
  435. var events = eventNames.split(' ');
  436. for (var i = 0, iLen = events.length; i < iLen; i++) {
  437. element.addEventListener(events[i], listener, false);
  438. }
  439. }