bootstrap.js 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186
  1. document.addEventListener("DOMContentLoaded", () => {
  2. handleSubmitButtons();
  3. if (!document.querySelector("body[data-disable-keyboard-shortcuts=true]")) {
  4. const keyboardHandler = new KeyboardHandler();
  5. keyboardHandler.on("g u", () => goToPage("unread"));
  6. keyboardHandler.on("g b", () => goToPage("starred"));
  7. keyboardHandler.on("g h", () => goToPage("history"));
  8. keyboardHandler.on("g f", goToFeedOrFeeds);
  9. keyboardHandler.on("g c", () => goToPage("categories"));
  10. keyboardHandler.on("g s", () => goToPage("settings"));
  11. keyboardHandler.on("g g", () => goToPrevious(TOP));
  12. keyboardHandler.on("G", () => goToNext(BOTTOM));
  13. keyboardHandler.on("ArrowLeft", goToPrevious);
  14. keyboardHandler.on("ArrowRight", goToNext);
  15. keyboardHandler.on("k", goToPrevious);
  16. keyboardHandler.on("p", goToPrevious);
  17. keyboardHandler.on("j", goToNext);
  18. keyboardHandler.on("n", goToNext);
  19. keyboardHandler.on("h", () => goToPage("previous"));
  20. keyboardHandler.on("l", () => goToPage("next"));
  21. keyboardHandler.on("z t", scrollToCurrentItem);
  22. keyboardHandler.on("o", openSelectedItem);
  23. keyboardHandler.on("Enter", () => openSelectedItem());
  24. keyboardHandler.on("v", () => openOriginalLink(false));
  25. keyboardHandler.on("V", () => openOriginalLink(true));
  26. keyboardHandler.on("c", () => openCommentLink(false));
  27. keyboardHandler.on("C", () => openCommentLink(true));
  28. keyboardHandler.on("m", () => handleEntryStatus("next"));
  29. keyboardHandler.on("M", () => handleEntryStatus("previous"));
  30. keyboardHandler.on("A", markPageAsRead);
  31. keyboardHandler.on("s", () => handleSaveEntry());
  32. keyboardHandler.on("d", handleFetchOriginalContent);
  33. keyboardHandler.on("f", () => handleBookmark());
  34. keyboardHandler.on("F", goToFeed);
  35. keyboardHandler.on("R", handleRefreshAllFeeds);
  36. keyboardHandler.on("?", showKeyboardShortcuts);
  37. keyboardHandler.on("+", goToAddSubscription);
  38. keyboardHandler.on("#", unsubscribeFromFeed);
  39. keyboardHandler.on("/", () => goToPage("search"));
  40. keyboardHandler.on("a", () => {
  41. const enclosureElement = document.querySelector('.entry-enclosures');
  42. if (enclosureElement) {
  43. enclosureElement.toggleAttribute('open');
  44. }
  45. });
  46. keyboardHandler.on("Escape", () => ModalHandler.close());
  47. keyboardHandler.listen();
  48. }
  49. const touchHandler = new TouchHandler();
  50. touchHandler.listen();
  51. if (WebAuthnHandler.isWebAuthnSupported()) {
  52. const webauthnHandler = new WebAuthnHandler();
  53. onClick("#webauthn-delete", () => { webauthnHandler.removeAllCredentials(); });
  54. const registerButton = document.getElementById("webauthn-register");
  55. if (registerButton !== null) {
  56. registerButton.disabled = false;
  57. onClick("#webauthn-register", () => {
  58. webauthnHandler.register().catch((err) => WebAuthnHandler.showErrorMessage(err));
  59. });
  60. }
  61. const loginButton = document.getElementById("webauthn-login");
  62. if (loginButton !== null) {
  63. const abortController = new AbortController();
  64. loginButton.disabled = false;
  65. onClick("#webauthn-login", () => {
  66. const usernameField = document.getElementById("form-username");
  67. if (usernameField !== null) {
  68. abortController.abort();
  69. webauthnHandler.login(usernameField.value).catch(err => WebAuthnHandler.showErrorMessage(err));
  70. }
  71. });
  72. webauthnHandler.conditionalLogin(abortController).catch(err => WebAuthnHandler.showErrorMessage(err));
  73. }
  74. }
  75. onClick(":is(a, button)[data-save-entry]", (event) => handleSaveEntry(event.target));
  76. onClick(":is(a, button)[data-toggle-bookmark]", (event) => handleBookmark(event.target));
  77. onClick(":is(a, button)[data-fetch-content-entry]", handleFetchOriginalContent);
  78. onClick(":is(a, button)[data-share-status]", handleShare);
  79. onClick(":is(a, button)[data-action=markPageAsRead]", (event) => handleConfirmationMessage(event.target, markPageAsRead));
  80. onClick(":is(a, button)[data-toggle-status]", (event) => handleEntryStatus("next", event.target));
  81. onClick(":is(a, button)[data-confirm]", (event) => handleConfirmationMessage(event.target, (url, redirectURL) => {
  82. const request = new RequestBuilder(url);
  83. request.withCallback((response) => {
  84. if (redirectURL) {
  85. window.location.href = redirectURL;
  86. } else if (response && response.redirected && response.url) {
  87. window.location.href = response.url;
  88. } else {
  89. window.location.reload();
  90. }
  91. });
  92. request.execute();
  93. }));
  94. onClick("a[data-original-link='true']", (event) => {
  95. handleEntryStatus("next", event.target, true);
  96. }, true);
  97. onAuxClick("a[data-original-link='true']", (event) => {
  98. if (event.button === 1) {
  99. handleEntryStatus("next", event.target, true);
  100. }
  101. }, true);
  102. checkMenuToggleModeByLayout();
  103. window.addEventListener("resize", checkMenuToggleModeByLayout, { passive: true });
  104. fixVoiceOverDetailsSummaryBug();
  105. const logoElement = document.querySelector(".logo");
  106. if (logoElement) {
  107. logoElement.addEventListener("click", toggleMainMenu);
  108. logoElement.addEventListener("keydown", toggleMainMenu);
  109. }
  110. onClick(".header nav li", (event) => onClickMainMenuListItem(event));
  111. if ("serviceWorker" in navigator) {
  112. const scriptElement = document.getElementById("service-worker-script");
  113. if (scriptElement) {
  114. navigator.serviceWorker.register(ttpolicy.createScriptURL(scriptElement.src));
  115. }
  116. }
  117. window.addEventListener('beforeinstallprompt', (e) => {
  118. let deferredPrompt = e;
  119. const promptHomeScreen = document.getElementById('prompt-home-screen');
  120. if (promptHomeScreen) {
  121. promptHomeScreen.style.display = "block";
  122. const btnAddToHomeScreen = document.getElementById('btn-add-to-home-screen');
  123. if (btnAddToHomeScreen) {
  124. btnAddToHomeScreen.addEventListener('click', (e) => {
  125. e.preventDefault();
  126. deferredPrompt.prompt();
  127. deferredPrompt.userChoice.then(() => {
  128. deferredPrompt = null;
  129. promptHomeScreen.style.display = "none";
  130. });
  131. });
  132. }
  133. }
  134. });
  135. // Save and resume media position
  136. const lastPositionElements = document.querySelectorAll("audio[data-last-position],video[data-last-position]");
  137. lastPositionElements.forEach((element) => {
  138. if (element.dataset.lastPosition) {
  139. element.currentTime = element.dataset.lastPosition;
  140. }
  141. element.ontimeupdate = () => handlePlayerProgressionSaveAndMarkAsReadOnCompletion(element);
  142. });
  143. // Set media playback rate
  144. const playbackRateElements = document.querySelectorAll("audio[data-playback-rate],video[data-playback-rate]");
  145. playbackRateElements.forEach((element) => {
  146. if (element.dataset.playbackRate) {
  147. element.playbackRate = element.dataset.playbackRate;
  148. if (element.dataset.enclosureId){
  149. // In order to display properly the speed we need to do it on bootstrap.
  150. // Could not do it backend side because I didn't know how to do it because of the template inclusion and
  151. // the way the initial playback speed is handled. See enclosure_media_controls.html if you want to try to fix this
  152. document.querySelectorAll(`span.speed-indicator[data-enclosure-id="${element.dataset.enclosureId}"]`).forEach((speedI)=>{
  153. speedI.innerText = `${parseFloat(element.dataset.playbackRate).toFixed(2)}x`;
  154. });
  155. }
  156. }
  157. });
  158. // Set enclosure media controls handlers
  159. const mediaControlsElements = document.querySelectorAll("button[data-enclosure-action]");
  160. mediaControlsElements.forEach((element) => {
  161. element.addEventListener("click", () => handleMediaControl(element));
  162. });
  163. });