s.js 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156
  1. const methodMap = [
  2. [
  3. 'requestFullscreen',
  4. 'exitFullscreen',
  5. 'fullscreenElement',
  6. 'fullscreenEnabled',
  7. 'fullscreenchange',
  8. 'fullscreenerror',
  9. ],
  10. // New WebKit
  11. [
  12. 'webkitRequestFullscreen',
  13. 'webkitExitFullscreen',
  14. 'webkitFullscreenElement',
  15. 'webkitFullscreenEnabled',
  16. 'webkitfullscreenchange',
  17. 'webkitfullscreenerror',
  18. ],
  19. // Old WebKit
  20. [
  21. 'webkitRequestFullScreen',
  22. 'webkitCancelFullScreen',
  23. 'webkitCurrentFullScreenElement',
  24. 'webkitCancelFullScreen',
  25. 'webkitfullscreenchange',
  26. 'webkitfullscreenerror',
  27. ],
  28. [
  29. 'mozRequestFullScreen',
  30. 'mozCancelFullScreen',
  31. 'mozFullScreenElement',
  32. 'mozFullScreenEnabled',
  33. 'mozfullscreenchange',
  34. 'mozfullscreenerror',
  35. ],
  36. [
  37. 'msRequestFullscreen',
  38. 'msExitFullscreen',
  39. 'msFullscreenElement',
  40. 'msFullscreenEnabled',
  41. 'MSFullscreenChange',
  42. 'MSFullscreenError',
  43. ],
  44. ];
  45. const nativeAPI = (() => {
  46. if (typeof document === 'undefined') {
  47. return false;
  48. }
  49. const unprefixedMethods = methodMap[0];
  50. const returnValue = {};
  51. for (const methodList of methodMap) {
  52. const exitFullscreenMethod = methodList[1];
  53. if (exitFullscreenMethod in document) {
  54. for (const [index, method] of methodList.entries()) {
  55. returnValue[unprefixedMethods[index]] = method;
  56. }
  57. return returnValue;
  58. }
  59. }
  60. return false;
  61. })();
  62. const eventNameMap = {
  63. change: nativeAPI.fullscreenchange,
  64. error: nativeAPI.fullscreenerror,
  65. };
  66. let screenfull = {
  67. request(element = document.documentElement, options) {
  68. return new Promise((resolve, reject) => {
  69. const onFullScreenEntered = () => {
  70. screenfull.off('change', onFullScreenEntered);
  71. resolve();
  72. };
  73. screenfull.on('change', onFullScreenEntered);
  74. const returnPromise = element[nativeAPI.requestFullscreen](options);
  75. if (returnPromise instanceof Promise) {
  76. returnPromise.then(onFullScreenEntered).catch(reject);
  77. }
  78. });
  79. },
  80. exit() {
  81. return new Promise((resolve, reject) => {
  82. if (!screenfull.isFullscreen) {
  83. resolve();
  84. return;
  85. }
  86. const onFullScreenExit = () => {
  87. screenfull.off('change', onFullScreenExit);
  88. resolve();
  89. };
  90. screenfull.on('change', onFullScreenExit);
  91. const returnPromise = document[nativeAPI.exitFullscreen]();
  92. if (returnPromise instanceof Promise) {
  93. returnPromise.then(onFullScreenExit).catch(reject);
  94. }
  95. });
  96. },
  97. toggle(element, options) {
  98. return screenfull.isFullscreen
  99. ? screenfull.exit()
  100. : screenfull.request(element, options);
  101. },
  102. onchange(callback) {
  103. screenfull.on('change', callback);
  104. },
  105. onerror(callback) {
  106. screenfull.on('error', callback);
  107. },
  108. on(event, callback) {
  109. const eventName = eventNameMap[event];
  110. if (eventName) {
  111. document.addEventListener(eventName, callback, false);
  112. }
  113. },
  114. off(event, callback) {
  115. const eventName = eventNameMap[event];
  116. if (eventName) {
  117. document.removeEventListener(eventName, callback, false);
  118. }
  119. },
  120. raw: nativeAPI,
  121. };
  122. Object.defineProperties(screenfull, {
  123. isFullscreen: {
  124. get: () => Boolean(document[nativeAPI.fullscreenElement]),
  125. },
  126. element: {
  127. enumerable: true,
  128. get: () => document[nativeAPI.fullscreenElement] ?? undefined,
  129. },
  130. isEnabled: {
  131. enumerable: true,
  132. // Coerce to boolean in case of old WebKit.
  133. get: () => Boolean(document[nativeAPI.fullscreenEnabled]),
  134. },
  135. });
  136. if (!nativeAPI) {
  137. screenfull = { isEnabled: false };
  138. }
  139. export default screenfull;