Source: ui/vr_manager.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ui.VRManager');
  7. goog.require('shaka.log');
  8. goog.require('shaka.ui.VRWebgl');
  9. goog.require('shaka.util.EventManager');
  10. goog.require('shaka.util.FakeEvent');
  11. goog.require('shaka.util.FakeEventTarget');
  12. goog.require('shaka.util.IReleasable');
  13. goog.require('shaka.util.Platform');
  14. goog.requireType('shaka.Player');
  15. /**
  16. * @implements {shaka.util.IReleasable}
  17. */
  18. shaka.ui.VRManager = class extends shaka.util.FakeEventTarget {
  19. /**
  20. * @param {!HTMLElement} container
  21. * @param {?HTMLCanvasElement} canvas
  22. * @param {!HTMLMediaElement} video
  23. * @param {!shaka.Player} player
  24. * @param {shaka.extern.UIConfiguration} config
  25. */
  26. constructor(container, canvas, video, player, config) {
  27. super();
  28. /** @private {!HTMLElement} */
  29. this.container_ = container;
  30. /** @private {?HTMLCanvasElement} */
  31. this.canvas_ = canvas;
  32. /** @private {!HTMLMediaElement} */
  33. this.video_ = video;
  34. /** @private {!shaka.Player} */
  35. this.player_ = player;
  36. /** @private {shaka.extern.UIConfiguration} */
  37. this.config_ = config;
  38. /** @private {shaka.util.EventManager} */
  39. this.loadEventManager_ = new shaka.util.EventManager();
  40. /** @private {shaka.util.EventManager} */
  41. this.eventManager_ = new shaka.util.EventManager();
  42. /** @private {?WebGLRenderingContext} */
  43. this.gl_ = this.getGL_();
  44. /** @private {?shaka.ui.VRWebgl} */
  45. this.vrWebgl_ = null;
  46. /** @private {boolean} */
  47. this.onGesture_ = false;
  48. /** @private {number} */
  49. this.prevX_ = 0;
  50. /** @private {number} */
  51. this.prevY_ = 0;
  52. /** @private {number} */
  53. this.prevAlpha_ = 0;
  54. /** @private {number} */
  55. this.prevBeta_ = 0;
  56. /** @private {number} */
  57. this.prevGamma_ = 0;
  58. /** @private {?string} */
  59. this.vrAsset_ = null;
  60. this.loadEventManager_.listen(player, 'loading', () => {
  61. if (this.vrWebgl_) {
  62. this.vrWebgl_.reset();
  63. }
  64. this.checkVrStatus_();
  65. });
  66. this.loadEventManager_.listen(player, 'spatialvideoinfo', (event) => {
  67. /** @type {shaka.extern.SpatialVideoInfo} */
  68. const spatialInfo = event['detail'];
  69. let unsupported = false;
  70. switch (spatialInfo.projection) {
  71. case 'hequ':
  72. unsupported = spatialInfo.hfov != 360;
  73. this.vrAsset_ = 'equirectangular';
  74. break;
  75. case 'fish':
  76. this.vrAsset_ = 'equirectangular';
  77. unsupported = true;
  78. break;
  79. default:
  80. this.vrAsset_ = null;
  81. break;
  82. }
  83. if (unsupported) {
  84. shaka.log.warning('Unsupported VR projection or hfov', spatialInfo);
  85. }
  86. this.checkVrStatus_();
  87. });
  88. this.loadEventManager_.listen(player, 'nospatialvideoinfo', () => {
  89. this.vrAsset_ = null;
  90. this.checkVrStatus_();
  91. });
  92. this.loadEventManager_.listen(player, 'unloading', () => {
  93. this.vrAsset_ = null;
  94. this.checkVrStatus_();
  95. });
  96. this.checkVrStatus_();
  97. }
  98. /**
  99. * @override
  100. */
  101. release() {
  102. if (this.loadEventManager_) {
  103. this.loadEventManager_.release();
  104. this.loadEventManager_ = null;
  105. }
  106. if (this.eventManager_) {
  107. this.eventManager_.release();
  108. this.eventManager_ = null;
  109. }
  110. if (this.vrWebgl_) {
  111. this.vrWebgl_.release();
  112. this.vrWebgl_ = null;
  113. }
  114. // FakeEventTarget implements IReleasable
  115. super.release();
  116. }
  117. /**
  118. * @param {!shaka.extern.UIConfiguration} config
  119. */
  120. configure(config) {
  121. this.config_ = config;
  122. this.checkVrStatus_();
  123. }
  124. /**
  125. * Returns if a VR is capable.
  126. *
  127. * @return {boolean}
  128. */
  129. canPlayVR() {
  130. return !!this.gl_;
  131. }
  132. /**
  133. * Returns if a VR is supported.
  134. *
  135. * @return {boolean}
  136. */
  137. isPlayingVR() {
  138. return !!this.vrWebgl_;
  139. }
  140. /**
  141. * Reset VR view.
  142. */
  143. reset() {
  144. if (!this.vrWebgl_) {
  145. shaka.log.alwaysWarn('Not playing VR content');
  146. return;
  147. }
  148. this.vrWebgl_.reset();
  149. }
  150. /**
  151. * Get the angle of the north.
  152. *
  153. * @return {?number}
  154. */
  155. getNorth() {
  156. if (!this.vrWebgl_) {
  157. shaka.log.alwaysWarn('Not playing VR content');
  158. return null;
  159. }
  160. return this.vrWebgl_.getNorth();
  161. }
  162. /**
  163. * Returns the field of view.
  164. *
  165. * @return {?number}
  166. */
  167. getFieldOfView() {
  168. if (!this.vrWebgl_) {
  169. shaka.log.alwaysWarn('Not playing VR content');
  170. return null;
  171. }
  172. return this.vrWebgl_.getFieldOfView();
  173. }
  174. /**
  175. * Set the field of view.
  176. *
  177. * @param {number} fieldOfView
  178. */
  179. setFieldOfView(fieldOfView) {
  180. if (!this.vrWebgl_) {
  181. shaka.log.alwaysWarn('Not playing VR content');
  182. return;
  183. }
  184. if (fieldOfView < 0) {
  185. shaka.log.alwaysWarn('Field of view should be greater than 0');
  186. fieldOfView = 0;
  187. } else if (fieldOfView > 100) {
  188. shaka.log.alwaysWarn('Field of view should be less than 100');
  189. fieldOfView = 100;
  190. }
  191. this.vrWebgl_.setFieldOfView(fieldOfView);
  192. }
  193. /**
  194. * Toggle stereoscopic mode.
  195. */
  196. toggleStereoscopicMode() {
  197. if (!this.vrWebgl_) {
  198. shaka.log.alwaysWarn('Not playing VR content');
  199. return;
  200. }
  201. this.vrWebgl_.toggleStereoscopicMode();
  202. }
  203. /**
  204. * Returns true if stereoscopic mode is enabled.
  205. *
  206. * @return {boolean}
  207. */
  208. isStereoscopicModeEnabled() {
  209. if (!this.vrWebgl_) {
  210. shaka.log.alwaysWarn('Not playing VR content');
  211. return false;
  212. }
  213. return this.vrWebgl_.isStereoscopicModeEnabled();
  214. }
  215. /**
  216. * Increment the yaw in X angle in degrees.
  217. *
  218. * @param {number} angle
  219. */
  220. incrementYaw(angle) {
  221. if (!this.vrWebgl_) {
  222. shaka.log.alwaysWarn('Not playing VR content');
  223. return;
  224. }
  225. this.vrWebgl_.rotateViewGlobal(
  226. angle * shaka.ui.VRManager.TO_RADIANS_, 0, 0);
  227. }
  228. /**
  229. * Increment the pitch in X angle in degrees.
  230. *
  231. * @param {number} angle
  232. */
  233. incrementPitch(angle) {
  234. if (!this.vrWebgl_) {
  235. shaka.log.alwaysWarn('Not playing VR content');
  236. return;
  237. }
  238. this.vrWebgl_.rotateViewGlobal(
  239. 0, angle * shaka.ui.VRManager.TO_RADIANS_, 0);
  240. }
  241. /**
  242. * Increment the roll in X angle in degrees.
  243. *
  244. * @param {number} angle
  245. */
  246. incrementRoll(angle) {
  247. if (!this.vrWebgl_) {
  248. shaka.log.alwaysWarn('Not playing VR content');
  249. return;
  250. }
  251. this.vrWebgl_.rotateViewGlobal(
  252. 0, 0, angle * shaka.ui.VRManager.TO_RADIANS_);
  253. }
  254. /**
  255. * @private
  256. */
  257. checkVrStatus_() {
  258. if (!this.canvas_) {
  259. return;
  260. }
  261. if ((this.config_.displayInVrMode || this.vrAsset_)) {
  262. const newProjectionMode =
  263. this.vrAsset_ || this.config_.defaultVrProjectionMode;
  264. if (!this.vrWebgl_) {
  265. this.canvas_.style.display = '';
  266. this.init_(newProjectionMode);
  267. this.dispatchEvent(new shaka.util.FakeEvent(
  268. 'vrstatuschanged',
  269. (new Map()).set('newStatus', this.isPlayingVR())));
  270. } else {
  271. const currentProjectionMode = this.vrWebgl_.getProjectionMode();
  272. if (currentProjectionMode != newProjectionMode) {
  273. this.eventManager_.removeAll();
  274. this.vrWebgl_.release();
  275. this.init_(newProjectionMode);
  276. // Re-initialization the status does not change.
  277. }
  278. }
  279. } else if (!this.config_.displayInVrMode && !this.vrAsset_ &&
  280. this.vrWebgl_) {
  281. this.canvas_.style.display = 'none';
  282. this.eventManager_.removeAll();
  283. this.vrWebgl_.release();
  284. this.vrWebgl_ = null;
  285. this.dispatchEvent(new shaka.util.FakeEvent(
  286. 'vrstatuschanged',
  287. (new Map()).set('newStatus', this.isPlayingVR())));
  288. }
  289. }
  290. /**
  291. * @param {string} projectionMode
  292. * @private
  293. */
  294. init_(projectionMode) {
  295. if (this.gl_ && this.canvas_) {
  296. this.vrWebgl_ = new shaka.ui.VRWebgl(
  297. this.video_, this.player_, this.canvas_, this.gl_, projectionMode);
  298. this.setupVRListerners_();
  299. }
  300. }
  301. /**
  302. * @return {?WebGLRenderingContext}
  303. * @private
  304. */
  305. getGL_() {
  306. if (!this.canvas_) {
  307. return null;
  308. }
  309. // The user interface is not intended for devices that are controlled with
  310. // a remote control, and WebGL may run slowly on these devices.
  311. if (shaka.util.Platform.isSmartTV()) {
  312. return null;
  313. }
  314. const webglContexts = [
  315. 'webgl2',
  316. 'webgl',
  317. ];
  318. for (const webgl of webglContexts) {
  319. const gl = this.canvas_.getContext(webgl);
  320. if (gl) {
  321. return /** @type {!WebGLRenderingContext} */(gl);
  322. }
  323. }
  324. return null;
  325. }
  326. /**
  327. * @private
  328. */
  329. setupVRListerners_() {
  330. // Start
  331. this.eventManager_.listen(this.container_, 'mousedown', (event) => {
  332. if (!this.onGesture_) {
  333. this.gestureStart_(event.clientX, event.clientY);
  334. }
  335. });
  336. if (navigator.maxTouchPoints > 0) {
  337. this.eventManager_.listen(this.container_, 'touchstart', (e) => {
  338. if (!this.onGesture_) {
  339. const event = /** @type {!TouchEvent} */(e);
  340. this.gestureStart_(
  341. event.touches[0].clientX, event.touches[0].clientY);
  342. }
  343. });
  344. }
  345. // Zoom
  346. this.eventManager_.listen(this.container_, 'wheel', (e) => {
  347. if (!this.onGesture_) {
  348. const event = /** @type {!WheelEvent} */(e);
  349. this.vrWebgl_.zoom(event.deltaY);
  350. event.preventDefault();
  351. event.stopPropagation();
  352. }
  353. });
  354. // Move
  355. this.eventManager_.listen(this.container_, 'mousemove', (event) => {
  356. if (this.onGesture_) {
  357. this.gestureMove_(event.clientX, event.clientY);
  358. }
  359. });
  360. if (navigator.maxTouchPoints > 0) {
  361. this.eventManager_.listen(this.container_, 'touchmove', (e) => {
  362. if (this.onGesture_) {
  363. const event = /** @type {!TouchEvent} */(e);
  364. this.gestureMove_(
  365. event.touches[0].clientX, event.touches[0].clientY);
  366. }
  367. e.preventDefault();
  368. });
  369. }
  370. // End
  371. this.eventManager_.listen(this.container_, 'mouseleave', () => {
  372. this.onGesture_ = false;
  373. });
  374. this.eventManager_.listen(this.container_, 'mouseup', () => {
  375. this.onGesture_ = false;
  376. });
  377. if (navigator.maxTouchPoints > 0) {
  378. this.eventManager_.listen(this.container_, 'touchend', () => {
  379. this.onGesture_ = false;
  380. });
  381. }
  382. // Detect device movement
  383. let deviceOrientationListener = false;
  384. if (window.DeviceOrientationEvent) {
  385. // See: https://dev.to/li/how-to-requestpermission-for-devicemotion-and-deviceorientation-events-in-ios-13-46g2
  386. if (typeof DeviceMotionEvent.requestPermission == 'function') {
  387. const userGestureListener = () => {
  388. DeviceMotionEvent.requestPermission().then((newPermissionState) => {
  389. if (newPermissionState !== 'granted' ||
  390. deviceOrientationListener) {
  391. return;
  392. }
  393. deviceOrientationListener = true;
  394. this.setupDeviceOrientationListener_();
  395. });
  396. };
  397. DeviceMotionEvent.requestPermission().then((permissionState) => {
  398. this.eventManager_.unlisten(
  399. this.container_, 'click', userGestureListener);
  400. this.eventManager_.unlisten(
  401. this.container_, 'mouseup', userGestureListener);
  402. if (navigator.maxTouchPoints > 0) {
  403. this.eventManager_.unlisten(
  404. this.container_, 'touchend', userGestureListener);
  405. }
  406. if (permissionState !== 'granted') {
  407. this.eventManager_.listenOnce(
  408. this.container_, 'click', userGestureListener);
  409. this.eventManager_.listenOnce(
  410. this.container_, 'mouseup', userGestureListener);
  411. if (navigator.maxTouchPoints > 0) {
  412. this.eventManager_.listenOnce(
  413. this.container_, 'touchend', userGestureListener);
  414. }
  415. return;
  416. }
  417. deviceOrientationListener = true;
  418. this.setupDeviceOrientationListener_();
  419. }).catch(() => {
  420. this.eventManager_.unlisten(
  421. this.container_, 'click', userGestureListener);
  422. this.eventManager_.unlisten(
  423. this.container_, 'mouseup', userGestureListener);
  424. if (navigator.maxTouchPoints > 0) {
  425. this.eventManager_.unlisten(
  426. this.container_, 'touchend', userGestureListener);
  427. }
  428. this.eventManager_.listenOnce(
  429. this.container_, 'click', userGestureListener);
  430. this.eventManager_.listenOnce(
  431. this.container_, 'mouseup', userGestureListener);
  432. if (navigator.maxTouchPoints > 0) {
  433. this.eventManager_.listenOnce(
  434. this.container_, 'touchend', userGestureListener);
  435. }
  436. });
  437. } else {
  438. deviceOrientationListener = true;
  439. this.setupDeviceOrientationListener_();
  440. }
  441. }
  442. }
  443. /**
  444. * @private
  445. */
  446. setupDeviceOrientationListener_() {
  447. this.eventManager_.listen(window, 'deviceorientation', (e) => {
  448. const event = /** @type {!DeviceOrientationEvent} */(e);
  449. let alphaDif = (event.alpha || 0) - this.prevAlpha_;
  450. let betaDif = (event.beta || 0) - this.prevBeta_;
  451. let gammaDif = (event.gamma || 0) - this.prevGamma_;
  452. if (Math.abs(alphaDif) > 10 || Math.abs(betaDif) > 10 ||
  453. Math.abs(gammaDif) > 5) {
  454. alphaDif = 0;
  455. gammaDif = 0;
  456. betaDif = 0;
  457. }
  458. this.prevAlpha_ = event.alpha || 0;
  459. this.prevBeta_ = event.beta || 0;
  460. this.prevGamma_ = event.gamma || 0;
  461. const toRadians = shaka.ui.VRManager.TO_RADIANS_;
  462. const orientation = screen.orientation.angle;
  463. if (orientation == 90 || orientation == -90) {
  464. this.vrWebgl_.rotateViewGlobal(
  465. alphaDif * toRadians * -1, gammaDif * toRadians * -1, 0);
  466. } else {
  467. this.vrWebgl_.rotateViewGlobal(
  468. alphaDif * toRadians * -1, betaDif * toRadians, 0);
  469. }
  470. });
  471. }
  472. /**
  473. * @param {number} x
  474. * @param {number} y
  475. * @private
  476. */
  477. gestureStart_(x, y) {
  478. this.onGesture_ = true;
  479. this.prevX_ = x;
  480. this.prevY_ = y;
  481. }
  482. /**
  483. * @param {number} x
  484. * @param {number} y
  485. * @private
  486. */
  487. gestureMove_(x, y) {
  488. const touchScaleFactor = -0.60 * Math.PI / 180;
  489. this.vrWebgl_.rotateViewGlobal((x - this.prevX_) * touchScaleFactor,
  490. (y - this.prevY_) * -1 * touchScaleFactor, 0);
  491. this.prevX_ = x;
  492. this.prevY_ = y;
  493. }
  494. };
  495. /**
  496. * @constant {number}
  497. * @private
  498. */
  499. shaka.ui.VRManager.TO_RADIANS_ = Math.PI / 180;