Source: ui/remote_button.js

  1. /*! @license
  2. * Shaka Player
  3. * Copyright 2016 Google LLC
  4. * SPDX-License-Identifier: Apache-2.0
  5. */
  6. goog.provide('shaka.ui.RemoteButton');
  7. goog.require('shaka.Player');
  8. goog.require('shaka.ui.Controls');
  9. goog.require('shaka.ui.Element');
  10. goog.require('shaka.ui.Enums');
  11. goog.require('shaka.ui.Locales');
  12. goog.require('shaka.ui.Localization');
  13. goog.require('shaka.ui.OverflowMenu');
  14. goog.require('shaka.ui.Utils');
  15. goog.require('shaka.util.Dom');
  16. goog.require('shaka.util.Platform');
  17. goog.requireType('shaka.ui.Controls');
  18. /**
  19. * @extends {shaka.ui.Element}
  20. * @final
  21. * @export
  22. */
  23. shaka.ui.RemoteButton = class extends shaka.ui.Element {
  24. /**
  25. * @param {!HTMLElement} parent
  26. * @param {!shaka.ui.Controls} controls
  27. */
  28. constructor(parent, controls) {
  29. super(parent, controls);
  30. /** @private {!HTMLButtonElement} */
  31. this.remoteButton_ = shaka.util.Dom.createButton();
  32. this.remoteButton_.classList.add('shaka-remote-button');
  33. this.remoteButton_.classList.add('shaka-tooltip');
  34. this.remoteButton_.ariaPressed = 'false';
  35. /** @private {!HTMLElement} */
  36. this.remoteIcon_ = shaka.util.Dom.createHTMLElement('i');
  37. this.remoteIcon_.classList.add('material-icons-round');
  38. let icon = shaka.ui.Enums.MaterialDesignIcons.CAST;
  39. const safariVersion = shaka.util.Platform.safariVersion();
  40. if (safariVersion && safariVersion >= 13) {
  41. icon = shaka.ui.Enums.MaterialDesignIcons.AIRPLAY;
  42. }
  43. this.remoteIcon_.textContent = icon;
  44. this.remoteButton_.appendChild(this.remoteIcon_);
  45. const label = shaka.util.Dom.createHTMLElement('label');
  46. label.classList.add('shaka-overflow-button-label');
  47. label.classList.add('shaka-overflow-menu-only');
  48. this.remoteNameSpan_ = shaka.util.Dom.createHTMLElement('span');
  49. label.appendChild(this.remoteNameSpan_);
  50. this.remoteCurrentSelectionSpan_ =
  51. shaka.util.Dom.createHTMLElement('span');
  52. this.remoteCurrentSelectionSpan_.classList.add(
  53. 'shaka-current-selection-span');
  54. label.appendChild(this.remoteCurrentSelectionSpan_);
  55. this.remoteButton_.appendChild(label);
  56. this.parent.appendChild(this.remoteButton_);
  57. /** @private {number} */
  58. this.callbackId_ = -1;
  59. // Setup strings in the correct language
  60. this.updateLocalizedStrings_();
  61. shaka.ui.Utils.setDisplay(this.remoteButton_, false);
  62. if (!this.video.remote || this.video.disableRemotePlayback) {
  63. this.remoteButton_.classList.add('shaka-hidden');
  64. } else {
  65. this.eventManager.listen(
  66. this.localization, shaka.ui.Localization.LOCALE_UPDATED, () => {
  67. this.updateLocalizedStrings_();
  68. });
  69. this.eventManager.listen(
  70. this.localization, shaka.ui.Localization.LOCALE_CHANGED, () => {
  71. this.updateLocalizedStrings_();
  72. });
  73. this.eventManager.listen(this.controls, 'caststatuschanged', () => {
  74. this.updateRemoteState_();
  75. });
  76. this.eventManager.listen(this.remoteButton_, 'click', () => {
  77. this.video.remote.prompt();
  78. });
  79. this.eventManager.listen(this.video.remote, 'connect', () => {
  80. this.updateRemoteState_();
  81. });
  82. this.eventManager.listen(this.video.remote, 'connecting', () => {
  83. this.updateRemoteState_();
  84. });
  85. this.eventManager.listen(this.video.remote, 'disconnect', () => {
  86. this.updateRemoteState_();
  87. });
  88. this.eventManager.listen(this.player, 'loaded', () => {
  89. this.updateRemoteState_();
  90. });
  91. this.updateRemoteState_();
  92. }
  93. }
  94. /** @override */
  95. release() {
  96. if (this.video.remote && this.callbackId_ != -1) {
  97. this.video.remote.cancelWatchAvailability(this.callbackId_).catch(() => {
  98. // Ignore this error.
  99. });
  100. }
  101. super.release();
  102. }
  103. /**
  104. * @private
  105. */
  106. async updateRemoteState_() {
  107. if (this.controls.getCastProxy().canCast() &&
  108. this.controls.isCastAllowed()) {
  109. shaka.ui.Utils.setDisplay(this.remoteButton_, false);
  110. if (this.callbackId_ != -1) {
  111. this.video.remote.cancelWatchAvailability(this.callbackId_);
  112. this.callbackId_ = -1;
  113. }
  114. } else if (this.video.remote.state == 'disconnected') {
  115. const handleAvailabilityChange = (availability) => {
  116. if (this.player) {
  117. const loadMode = this.player.getLoadMode();
  118. const srcMode = loadMode == shaka.Player.LoadMode.SRC_EQUALS;
  119. shaka.ui.Utils.setDisplay(
  120. this.remoteButton_, srcMode && availability);
  121. } else {
  122. shaka.ui.Utils.setDisplay(this.remoteButton_, false);
  123. }
  124. };
  125. try {
  126. if (this.callbackId_ != -1) {
  127. await this.video.remote.cancelWatchAvailability(this.callbackId_);
  128. this.callbackId_ = -1;
  129. }
  130. } catch (e) {
  131. // Ignore this error.
  132. }
  133. try {
  134. const id = await this.video.remote.watchAvailability(
  135. handleAvailabilityChange);
  136. this.callbackId_ = id;
  137. } catch (e) {
  138. handleAvailabilityChange(/* availability= */ true);
  139. }
  140. } else if (this.callbackId_ != -1) {
  141. // If remote device is connecting or connected, we should stop
  142. // watching remote device availability to save power.
  143. await this.video.remote.cancelWatchAvailability(this.callbackId_);
  144. this.callbackId_ = -1;
  145. }
  146. }
  147. /**
  148. * @private
  149. */
  150. updateLocalizedStrings_() {
  151. const LocIds = shaka.ui.Locales.Ids;
  152. let text = this.localization.resolve(LocIds.CAST);
  153. const safariVersion = shaka.util.Platform.safariVersion();
  154. if (safariVersion && safariVersion >= 13) {
  155. text = this.localization.resolve(LocIds.AIRPLAY);
  156. }
  157. this.remoteButton_.ariaLabel = text;
  158. this.remoteNameSpan_.textContent = text;
  159. }
  160. };
  161. /**
  162. * @implements {shaka.extern.IUIElement.Factory}
  163. * @final
  164. */
  165. shaka.ui.RemoteButton.Factory = class {
  166. /** @override */
  167. create(rootElement, controls) {
  168. return new shaka.ui.RemoteButton(rootElement, controls);
  169. }
  170. };
  171. shaka.ui.OverflowMenu.registerElement(
  172. 'remote', new shaka.ui.RemoteButton.Factory());
  173. shaka.ui.Controls.registerElement(
  174. 'remote', new shaka.ui.RemoteButton.Factory());