ckin.js 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397
  1. /*!
  2. ckin v0.0.1: Custom HTML5 Video 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 iconVolumeMedium = '<i class="ckin-volume-medium"></i>';
  136. var iconVolumeLow = '<i class="ckin-volume-low"></i>';
  137. var iconExpand = '<i class="ckin-expand"></i>';
  138. var iconCompress = '<i class="ckin-compress"></i>';
  139. players.forEach(function (player) {
  140. var video = player.querySelector('video');
  141. var skin = attachSkin(video.dataset.ckin);
  142. player.classList.add(skin);
  143. var overlay = video.dataset.overlay;
  144. addOverlay(player, overlay);
  145. var title = showTitle(skin, video.dataset.title);
  146. if (title) {
  147. player.insertAdjacentHTML('beforeend', title);
  148. }
  149. var html = buildControls(skin);
  150. player.insertAdjacentHTML('beforeend', html);
  151. var color = video.dataset.color;
  152. addColor(player, color);
  153. var playerControls = player.querySelector('.' + skin + '__controls');
  154. var progress = player.querySelector('.progress');;
  155. var progressBar = player.querySelector('.progress__filled');
  156. var toggle = player.querySelectorAll('.toggle');
  157. var skipButtons = player.querySelectorAll('[data-skip]');
  158. var ranges = player.querySelectorAll('.' + skin + '__slider');
  159. var volumeButton = player.querySelector('.volume');
  160. var fullScreenButton = player.querySelector('.fullscreen');
  161. if (obj.browserName === "IE" && (obj.browserVersion === 8 || obj.browserVersion === 9)) {
  162. showControls(video);
  163. playerControls.style.display = "none";
  164. }
  165. video.addEventListener('click', function () {
  166. togglePlay(this, player);
  167. });
  168. video.addEventListener('play', function () {
  169. updateButton(this, toggle);
  170. });
  171. video.addEventListener('pause', function () {
  172. updateButton(this, toggle);
  173. });
  174. video.addEventListener('timeupdate', function () {
  175. handleProgress(this, progressBar);
  176. });
  177. toggle.forEach(function (button) {
  178. return button.addEventListener('click', function () {
  179. togglePlay(video, player);
  180. });
  181. });
  182. volumeButton.addEventListener('click', function () {
  183. toggleVolume(video, volumeButton);
  184. });
  185. var mousedown = false;
  186. progress.addEventListener('click', function (e) {
  187. scrub(e, video, progress);
  188. });
  189. progress.addEventListener('mousemove', function (e) {
  190. return mousedown && scrub(e, video, progress);
  191. });
  192. progress.addEventListener('mousedown', function () {
  193. return mousedown = true;
  194. });
  195. progress.addEventListener('mouseup', function () {
  196. return mousedown = false;
  197. });
  198. fullScreenButton.addEventListener('click', function (e) {
  199. return toggleFullScreen(player, fullScreenButton);
  200. });
  201. addListenerMulti(player, 'webkitfullscreenchange mozfullscreenchange fullscreenchange MSFullscreenChange', function (e) {
  202. return onFullScreen(e, player);
  203. });
  204. });
  205. function showControls(video) {
  206. video.setAttribute("controls", "controls");
  207. }
  208. function togglePlay(video, player) {
  209. var method = video.paused ? 'play' : 'pause';
  210. video[method]();
  211. video.paused ? player.classList.remove('is-playing') : player.classList.add('is-playing');
  212. }
  213. function updateButton(video, toggle) {
  214. var icon = video.paused ? iconPlay : iconPause;
  215. toggle.forEach(function (button) {
  216. return button.innerHTML = icon;
  217. });
  218. }
  219. function skip() {
  220. video.currentTime += parseFloat(this.dataset.skip);
  221. }
  222. function toggleVolume(video, volumeButton) {
  223. var level = video.volume;
  224. var icon = iconVolumeMedium;
  225. if (level == 1) {
  226. level = 0;
  227. icon = iconVolumeMute;
  228. } else if (level == 0.5) {
  229. level = 1;
  230. icon = iconVolumeMedium;
  231. } else {
  232. level = 0.5;
  233. icon = iconVolumeLow;
  234. }
  235. video['volume'] = level;
  236. volumeButton.innerHTML = icon;
  237. }
  238. function handleRangeUpdate() {
  239. video[this.name] = this.value;
  240. }
  241. function handleProgress(video, progressBar) {
  242. var percent = video.currentTime / video.duration * 100;
  243. progressBar.style.flexBasis = percent + '%';
  244. }
  245. function scrub(e, video, progress) {
  246. var scrubTime = e.offsetX / progress.offsetWidth * video.duration;
  247. video.currentTime = scrubTime;
  248. }
  249. function wrapPlayers() {
  250. var videos = document.querySelectorAll('video');
  251. videos.forEach(function (video) {
  252. var wrapper = document.createElement('div');
  253. wrapper.classList.add('ckin__player');
  254. video.parentNode.insertBefore(wrapper, video);
  255. wrapper.appendChild(video);
  256. });
  257. }
  258. function buildControls(skin) {
  259. var html = [];
  260. html.push('<button class="' + skin + '__button--big toggle" title="Toggle Play">' + iconPlay + '</button>');
  261. html.push('<div class="' + skin + '__controls ckin__controls">');
  262. html.push('<button class="' + skin + '__button toggle" title="Toggle Video">' + iconPlay + '</button>', '<div class="progress">', '<div class="progress__filled"></div>', '</div>', '<button class="' + skin + '__button volume" title="Volume">' + iconVolumeMedium + '</button>', '<button class="' + skin + '__button fullscreen" title="Full Screen">' + iconExpand + '</button>');
  263. html.push('</div>');
  264. return html.join('');
  265. }
  266. function attachSkin(skin) {
  267. if (typeof skin != 'undefined' && skin != '') {
  268. return skin;
  269. } else {
  270. return 'default';
  271. }
  272. }
  273. function showTitle(skin, title) {
  274. if (typeof title != 'undefined' && title != '') {
  275. return '<div class="' + skin + '__title">' + title + '</div>';
  276. } else {
  277. return false;
  278. }
  279. }
  280. function addOverlay(player, overlay) {
  281. if (overlay == 1) {
  282. player.classList.add('ckin__overlay');
  283. } else if (overlay == 2) {
  284. player.classList.add('ckin__overlay--2');
  285. } else {
  286. return;
  287. }
  288. }
  289. function addColor(player, color) {
  290. if (typeof color != 'undefined' && color != '') {
  291. var buttons = player.querySelectorAll('button');
  292. var progress = player.querySelector('.progress__filled');
  293. progress.style.background = color;
  294. buttons.forEach(function (button) {
  295. return button.style.color = color;
  296. });
  297. }
  298. }
  299. function toggleFullScreen(player, fullScreenButton) {
  300. // let isFullscreen = false;
  301. if (!document.fullscreenElement && // alternative standard method
  302. !document.mozFullScreenElement && !document.webkitFullscreenElement && !document.msFullscreenElement) {
  303. player.classList.add('ckin__fullscreen');
  304. if (player.requestFullscreen) {
  305. player.requestFullscreen();
  306. } else if (player.mozRequestFullScreen) {
  307. player.mozRequestFullScreen(); // Firefox
  308. } else if (player.webkitRequestFullscreen) {
  309. player.webkitRequestFullscreen(); // Chrome and Safari
  310. } else if (player.msRequestFullscreen) {
  311. player.msRequestFullscreen();
  312. }
  313. isFullscreen = true;
  314. fullScreenButton.innerHTML = iconCompress;
  315. } else {
  316. player.classList.remove('ckin__fullscreen');
  317. if (document.cancelFullScreen) {
  318. document.cancelFullScreen();
  319. } else if (document.mozCancelFullScreen) {
  320. document.mozCancelFullScreen();
  321. } else if (document.webkitCancelFullScreen) {
  322. document.webkitCancelFullScreen();
  323. } else if (document.msExitFullscreen) {
  324. document.msExitFullscreen();
  325. }
  326. isFullscreen = false;
  327. fullScreenButton.innerHTML = iconExpand;
  328. }
  329. }
  330. function onFullScreen(e, player) {
  331. var isFullscreenNow = document.webkitFullscreenElement !== null;
  332. if (!isFullscreenNow) {
  333. player.classList.remove('ckin__fullscreen');
  334. player.querySelector('.fullscreen').innerHTML = iconExpand;
  335. } else {
  336. // player.querySelector('.fullscreen').innerHTML = iconExpand;
  337. }
  338. }
  339. function addListenerMulti(element, eventNames, listener) {
  340. var events = eventNames.split(' ');
  341. for (var i = 0, iLen = events.length; i < iLen; i++) {
  342. element.addEventListener(events[i], listener, false);
  343. }
  344. }