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