programajaponesonline.com.br - Use spacebar keyboard shortcut to pause/play audios, and more...

This script does a lot of simple things, like remembering what volume you have previously set to your audios, sync the audio volumes as if they are one (including mute), add space bar and media button keyboard shortcut support to play/pause the audio, add volume change keyboard shortcut through Add (+) and Subtract (-) keys of your numpad. It also sets the audio and speed of all lesson audios to the configured value, so you don't have to blow your ears when hearing them.

当前为 2023-01-03 提交的版本,查看 最新版本

  1. // ==UserScript==
  2. // @name programajaponesonline.com.br - Use spacebar keyboard shortcut to pause/play audios, and more...
  3. // @namespace secretx_scripts
  4. // @match *://portal.programajaponesonline.com.br/playlist/
  5. // @match *://portal.programajaponesonline.com.br/*/*/*
  6. // @version 2023.01.03
  7. // @author SecretX
  8. // @description This script does a lot of simple things, like remembering what volume you have previously set to your audios, sync the audio volumes as if they are one (including mute), add space bar and media button keyboard shortcut support to play/pause the audio, add volume change keyboard shortcut through Add (+) and Subtract (-) keys of your numpad. It also sets the audio and speed of all lesson audios to the configured value, so you don't have to blow your ears when hearing them.
  9. // @grant GM_addStyle
  10. // @grant GM_getValue
  11. // @grant GM_setValue
  12. // @run-at document-start
  13. // @require https://craig.global.ssl.fastly.net/js/mousetrap/mousetrap.min.js?a4098
  14. // @icon https://i.imgur.com/vn8ClVJ.png
  15. // @license GNU LGPLv3
  16. // ==/UserScript==
  17.  
  18. const keybindHandlers = {
  19. playPauseAudioKey: playPauseAudioPlaying,
  20. stopAudioKey: stopAudioPlaying,
  21. increaseVolumeKey: () => forEachAudio(stepUpVolume),
  22. decreaseVolumeKey: () => forEachAudio(stepDownVolume),
  23. }
  24.  
  25. Object.defineProperty(HTMLElement.prototype, "isVisible", {
  26. value: function() {
  27. return (this.offsetParent !== null);
  28. },
  29. writable: true,
  30. configurable: true
  31. });
  32.  
  33. Object.defineProperty(Array.prototype, "insert", {
  34. value: function(index, ...items) {
  35. return [...this.slice(0, index), ...items, ...this.slice(index)]
  36. },
  37. writable: true,
  38. configurable: true
  39. });
  40.  
  41. function loadSetting(name) {
  42. return GM_SuperValue.get(name);
  43. }
  44.  
  45. function saveSetting(name, value) {
  46. GM_SuperValue.set(name, value);
  47. }
  48.  
  49. function getPlaylistOptions() {
  50. const cached = loadSetting("playlist_options");
  51. if (cached != null) return cached;
  52.  
  53. const defaults = {
  54. enableShortcutKeys: true,
  55. rememberAudioRepetitionAmount: true,
  56. linkAudioVolumes: true,
  57. playPauseAudioKey: ["space", "playpausemedia"],
  58. stopAudioKey: ["stopmedia"], // Stop media
  59. increaseVolumeKey: ["+"], // Plus Numpad
  60. decreaseVolumeKey: ["-"], // Minus Numpad
  61. volumeStep: 0.05, // 5%
  62. volumeMin: 0.0, // 0%
  63. volumeMax: 1.0, // 100%
  64. };
  65. saveSetting("playlist_options", defaults);
  66. return defaults;
  67. }
  68.  
  69. function savePlaylistOptions(playlistOptions) {
  70. saveSetting("playlist_options", playlistOptions);
  71. }
  72.  
  73. function getLessionsOptions() {
  74. const cached = loadSetting("lessions_options");
  75. if (cached != null) return cached;
  76.  
  77. const defaults = {
  78. enableHideTranslationButton: true,
  79. showFuriganaOnMouseOver: false,
  80. audioVolume: 1.0,
  81. audioSpeed: 1.0,
  82. };
  83. saveSetting("lessions_options", defaults);
  84. return defaults;
  85. }
  86.  
  87. function saveLessionsOptions(playlistOptions) {
  88. saveSetting("lessions_options", playlistOptions);
  89. }
  90.  
  91. function showElem(htmlElem) {
  92. if (htmlElem == null) return;
  93. htmlElem.classList.remove("hidden");
  94. }
  95.  
  96. function hideElem(htmlElem) {
  97. if (htmlElem == null) return;
  98. htmlElem.classList.add("hidden");
  99. }
  100.  
  101. const getAudios = () => Array.from(document.querySelectorAll("audio"));
  102. const forEachAudio = action => getAudios().forEach(audio => action(audio));
  103. const getPlayingAudio = () => {
  104. const audios = getAudios().filter(audio => audio.isVisible());
  105. return audios.length === 1 ? audios[0] : null;
  106. }
  107. const setVolumeOfAllAudios = volume => forEachAudio(audio => audio.volume = Math.max(0.0, Math.min(1.0, volume)));
  108. const setMuteOfAllAudios = isMuted => forEachAudio(audio => audio.muted = isMuted);
  109. const toggleAudio = (audio) => audio.paused ? audio.play() : audio.pause();
  110. const stepDownVolume = (audio) => audio.volume = Math.max(audio.volume - getPlaylistOptions()["volumeStep"], getPlaylistOptions()["volumeMin"]);
  111. const stepUpVolume = (audio) => audio.volume = Math.min(audio.volume + getPlaylistOptions()["volumeStep"], getPlaylistOptions()["volumeMax"]);
  112.  
  113. function loadPreviousSettings() {
  114. const previousAudioVolume = loadSetting("audio_volume");
  115. if (previousAudioVolume != null) setVolumeOfAllAudios(previousAudioVolume);
  116. setMuteOfAllAudios(loadSetting("audio_muted") === true);
  117. }
  118.  
  119. function linkAllAudioVolumeSliders(audios) {
  120. audios.forEach(audio => audio.addEventListener("volumechange", () => {
  121. setVolumeOfAllAudios(audio.volume)
  122. setMuteOfAllAudios(audio.muted);
  123. saveSetting("audio_volume", audio.volume);
  124. saveSetting("audio_muted", audio.muted);
  125. }));
  126. }
  127.  
  128. function playPauseAudioPlaying() {
  129. const playingAudio = getPlayingAudio();
  130. if (playingAudio == null) {
  131. console.info("No audio is playing");
  132. return;
  133. }
  134. toggleAudio(playingAudio);
  135. }
  136.  
  137. function stopAudioPlaying() {
  138. const playingAudio = getPlayingAudio();
  139. if (playingAudio == null) {
  140. console.info("No audio is playing");
  141. return;
  142. }
  143. if (!playingAudio.paused) playingAudio.pause();
  144. }
  145.  
  146. // UI
  147.  
  148. const mainDivId = "jpo_toolkit";
  149. const parentDivId = `${mainDivId}_parent`;
  150. const formId = `${mainDivId}_form`;
  151. const openButtonId = "jpo_toolkit_open_button";
  152. const closeButtonId = "jpo_toolkit_close_button";
  153.  
  154. function injectToolkitOverlayCss() {
  155. const css = `
  156. @import url("https://fonts.googleapis.com/css?family=Bebas+Neue:400|Inter:400");
  157. :root {
  158. --black: rgba(0, 0, 0, 1);
  159. --baby-powder: rgba(252, 252, 252, 1);
  160. --font-size-s: 0.7rem;
  161. --font-size-m: 0.8rem;
  162. --font-size-l: 1.7rem;
  163. --font-family-bebas_neue: "Bebas Neue";
  164. --font-family-inter: "Inter";
  165. }
  166.  
  167. #${parentDivId} {
  168. width: 24rem;
  169. min-height: 9rem;
  170. margin-top: 0;
  171. margin-right: 0;
  172. margin-bottom: 0;
  173. padding: 0.5rem 1.5rem 1.75rem 1.5rem;
  174. border: 0 none;
  175. background: var(--baby-powder);
  176. display: block;
  177. position: fixed;
  178. z-index: 16000001;
  179. right: 1rem;
  180. top: 9rem;
  181. border-radius: 5px;
  182. box-shadow:0px 4px 4px rgba(0, 0, 0, 0.25);
  183. }
  184. .jpo_toolkit_title_bar {
  185. display: flex;
  186. justify-content: space-between;
  187. }
  188. .jpo_toolkit_title_bar > h2 {
  189. color: var(--black);
  190. font-family: var(--font-family-bebas_neue);
  191. font-size: var(--font-size-l);
  192. font-weight: 400;
  193. letter-spacing: 0;
  194. line-height: normal;
  195. margin-bottom: 0;
  196. }
  197. #${closeButtonId} {
  198. display: flex;
  199. justify-content: center;
  200. align-items: center;
  201. border-radius: 2.5rem;
  202. margin-right: -0.8rem;
  203. }
  204. #${closeButtonId} > img {
  205. height: var(--font-size-l);
  206. width: var(--font-size-l);
  207. filter: contrast(30%) opacity(70%);
  208. }
  209. div.toolkit_form_section {
  210. display: grid;
  211. font-size: var(--font-size-m);
  212. }
  213. div.toolkit_form_section_second_onwards {
  214. margin-top: 0.5em
  215. }
  216. div.toolkit_form_field_label {
  217. display: flex;
  218. align-items: center;
  219. }
  220. div.toolkit_form_field_label > label {
  221. color: var(--black);
  222. font-family: var(--font-family-inter);
  223. font-size: var(--font-size-m);
  224. font-weight: 400;
  225. letter-spacing: 0;
  226. line-height: normal;
  227. }
  228. .hidden {
  229. display: none !important;
  230. }
  231. .rectangle_box {
  232. margin-top: 1.7rem;
  233. position: relative;
  234. padding: 1rem 0.5rem 0.5rem 0.5rem;
  235. border: 1px solid;
  236. border-color: #d7d7d7;
  237. border-radius: 5px;
  238. width: 100%;
  239. }
  240. .overlap_group {
  241. align-items: flex-end;
  242. background-color: var(--baby-powder);
  243. display: flex;
  244. justify-content: flex-end;
  245. position: absolute;
  246. top: 0;
  247. margin: -0.6rem 0 0 0.5rem;
  248. padding: 0 0.2rem;
  249. }
  250. .section_title {
  251. color: #959595;
  252. font-family: var(--font-family-bebas_neue);
  253. font-size: 1rem;
  254. font-weight: 400;
  255. letter-spacing: 0;
  256. line-height: normal;
  257. }
  258. .jpo_toolkit_input {
  259. width: 3.7rem;
  260. padding: 4px 4px 4px 8px;
  261. border-radius: 7px;
  262. border: 1px solid #ccc;
  263. font-size: var(--font-size-m);
  264. }
  265. .go_to_right {
  266. margin-left: auto;
  267. }
  268. `;
  269.  
  270. // Use Violent Monkey global function to inject our CSS onto page
  271. GM_addStyle(css);
  272. }
  273.  
  274. function createToolkitOverlay() {
  275. const div = document.createElement("div");
  276. div.id = parentDivId;
  277. hideElem(div);
  278.  
  279. const playlistOptions = getPlaylistOptions();
  280.  
  281. const html = `
  282. <div id="${mainDivId}">
  283. <div class="jpo_toolkit_title_bar">
  284. <h2>PJO Toolkit️</h2>
  285. <div id="${closeButtonId}">
  286. <img src="${closeIcon()}" alt="Close this window"/>
  287. </div>
  288. </div>
  289. <p style="font-size: 0.7rem; font-style: italic; margin-bottom: 0.25rem">Work in progress!</p>
  290. <form id="${formId}" name="${formId}" action="">
  291. <div class="toolkit_form_section">
  292. <div class="rectangle_box">
  293. <div class="overlap_group">
  294. <div class="section_title">Playlist</div>
  295. </div>
  296. <div class="toolkit_form_field_label">
  297. <label for="enable_shortcut_keys">Ativar botões de atalho</label>
  298. <input type="checkbox" id="enable_shortcut_keys" class="go_to_right"/>
  299. </div>
  300. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  301. <label for="remember_audio_repetition_amount">Lembrar do número de repetição dos áudios</label>
  302. <input type="checkbox" id="remember_audio_repetition_amount" class="go_to_right"/>
  303. </div>
  304. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  305. <label for="link_audio_volumes">Unificar volume de todos os áudios</label>
  306. <input type="checkbox" id="link_audio_volumes" class="go_to_right"/>
  307. </div>
  308. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  309. <label for="volume_min">Volume mínimo (0-100)</label>
  310. <input type="number" id="volume_min" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeMin * 100.0)}" min="0" max="100"/>
  311. </div>
  312. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  313. <label for="volume_max">Volume máximo (0-100)</label>
  314. <input type="number" id="volume_max" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeMax * 100.0)}" min="0" max="100"/>
  315. </div>
  316. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  317. <label for="volume_step">Volume step (1-100)</label>
  318. <input type="number" id="volume_step" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeStep * 100.0)}" min="1" max="100"/>
  319. </div>
  320. </div>
  321. <div class="rectangle_box">
  322. <div class="overlap_group">
  323. <div class="section_title">Lessons</div>
  324. </div>
  325. <div class="toolkit_form_field_label">
  326. <label for="hide_translation_button">Exibir botão de esconder tradução de textos</label>
  327. <input type="checkbox" id="hide_translation_button" class="go_to_right"/>
  328. </div>
  329. <div class="toolkit_form_field_label toolkit_form_section_second_onwards">
  330. <label for="show_furigana_on_mouse_over">Furigana somente ao passar o mouse</label>
  331. <input type="checkbox" id="show_furigana_on_mouse_over" class="go_to_right"/>
  332. </div>
  333. </div>
  334. </form>
  335. </div>
  336. `.trim();
  337. div.innerHTML = html;
  338.  
  339. setPlaylistInteractListeners(div);
  340.  
  341. return div;
  342. }
  343.  
  344. function setPlaylistInteractListeners(div) {
  345. const setPlaylistCheckboxOption = (checkbox, property) => {
  346. const playlistOptions = getPlaylistOptions();
  347. const isChecked = !playlistOptions[property];
  348. checkbox.attributes.checked = isChecked;
  349. playlistOptions[property] = isChecked;
  350. savePlaylistOptions(playlistOptions);
  351. console.info(`Setting '${property}' to '${isChecked}'`);
  352. };
  353. const preparePlaylistCheckbox = (inputSelector, property) => {
  354. const checkbox = div.querySelector(inputSelector);
  355. checkbox.addEventListener("input", () => setPlaylistCheckboxOption(checkbox, property), false);
  356. if (getPlaylistOptions()[property]) checkbox.checked = true;
  357. }
  358. preparePlaylistCheckbox("input#enable_shortcut_keys", "enableShortcutKeys");
  359. preparePlaylistCheckbox("input#remember_audio_repetition_amount", "rememberAudioRepetitionAmount");
  360. preparePlaylistCheckbox("input#link_audio_volumes", "linkAudioVolumes");
  361.  
  362. const setPlaylistVolumeOption = (input, property) => {
  363. const playlistOptions = getPlaylistOptions();
  364. const newValue = Math.min(input.max, Math.max(input.min, input.value)) / 100.0;
  365. playlistOptions[property] = newValue;
  366. savePlaylistOptions(playlistOptions);
  367. console.info(`Setting '${property}' to '${newValue}'`);
  368. };
  369. const playlistVolumeMin = div.querySelector("input#volume_min");
  370. playlistVolumeMin.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeMin, "volumeMin"), false);
  371. const playlistVolumeMax = div.querySelector("input#volume_max");
  372. playlistVolumeMax.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeMax, "volumeMax"), false);
  373. const playlistVolumeStep = div.querySelector("input#volume_step");
  374. playlistVolumeStep.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeStep, "volumeStep"), false);
  375.  
  376. const setLessionsCheckboxOption = (checkbox, property) => {
  377. const lessionsOptions = getLessionsOptions();
  378. const isChecked = !lessionsOptions[property];
  379. checkbox.attributes.checked = isChecked;
  380. lessionsOptions[property] = isChecked;
  381. saveLessionsOptions(lessionsOptions);
  382. console.info(`Setting '${property}' to '${isChecked}'`);
  383. };
  384. const hideTranslationButton = div.querySelector("input#hide_translation_button");
  385. hideTranslationButton.addEventListener("input", () => setLessionsCheckboxOption(hideTranslationButton, "enableHideTranslationButton"), false);
  386. if (getLessionsOptions().enableHideTranslationButton) hideTranslationButton.checked = true;
  387. const showFuriganaOnMouseOver = div.querySelector("input#show_furigana_on_mouse_over");
  388. showFuriganaOnMouseOver.addEventListener("input", () => setLessionsCheckboxOption(hideTranslationButton, "showFuriganaOnMouseOver"), false);
  389. if (getLessionsOptions().showFuriganaOnMouseOver) showFuriganaOnMouseOver.checked = true;
  390. div.querySelector(`#${closeButtonId}`).addEventListener("click", () => hideElem(div), false);
  391. }
  392.  
  393. function appendOpenToolkitButton() {
  394. const userInfoDiv = document.querySelector("div.user-info");
  395. const userInfoItems = userInfoDiv.innerHTML.split("|");
  396. const newItems = userInfoItems.insert(1, `<a id='${openButtonId}'>PJO Toolkit</a>`);
  397. userInfoDiv.innerHTML = newItems.join(" | ");
  398. document.querySelector(`#${openButtonId}`).addEventListener("click", () => showElem(document.querySelector(`#${parentDivId}`)));
  399. }
  400.  
  401. function injectToolkitOverlay() {
  402. injectToolkitOverlayCss();
  403. const mainDiv = document.querySelector("div#primary");
  404. if (mainDiv == null) return;
  405. mainDiv.appendChild(createToolkitOverlay());
  406. appendOpenToolkitButton();
  407. }
  408.  
  409. // Main functions
  410.  
  411. function rememberAudioRepeatNumber() {
  412. const audioRepeat = loadSetting("audio_repeat_amount") ?? {};
  413.  
  414. // div.playlist-title > div > input.playlist-item-loop
  415. const audioDivs = Array.from(document.querySelectorAll("li.playlist-item-active"));
  416. for (const audioDiv of audioDivs) {
  417. const audioInput = audioDiv.querySelector("div.playlist-title > div > input.playlist-item-loop");
  418. // Will become something like 'Nihongo Rise - Tópico 08アンケート調査 (Questionário de pesquisa.)'
  419. const audioName = Array.from(audioDiv.querySelectorAll("div.playlist-title > div > span.jp-title"))
  420. .map(span => span.innerText)
  421. .reduce((a, b) => a + b, "")
  422. .trim();
  423.  
  424. if (audioInput == null || audioName == null || audioName.length === 0) continue;
  425.  
  426. const previousRepeatAmount = audioRepeat[audioName];
  427. if (previousRepeatAmount != null && audioInput.value === "1") {
  428. // Restore previous repeat value
  429. audioInput.value = previousRepeatAmount;
  430. }
  431.  
  432. audioInput.addEventListener("input", () => {
  433. console.info(`Changed '${audioName}' repeat amount to '${audioInput.value}'`);
  434. audioRepeat[audioName] = audioInput.value;
  435. saveSetting("audio_repeat_amount", audioRepeat);
  436. }, false);
  437. }
  438. }
  439.  
  440. function bindKeyboardShortcuts() {
  441. for (const [actionName, keyShortcuts] of Object.entries(getPlaylistOptions())) {
  442. const executorMethod = keybindHandlers[actionName];
  443. if (executorMethod == null) continue;
  444.  
  445. keyShortcuts.forEach(keyShortcut => Mousetrap.bind(keyShortcut, () => {
  446. executorMethod();
  447. return false; // prevents default browser behavior
  448. }));
  449. }
  450. console.log("Bound keyboard shortcuts!");
  451. }
  452.  
  453. function configurePlaylist() {
  454. if (window.location.pathname !== "/playlist/") return;
  455. const audios = getAudios();
  456. if (audios.length === 0) {
  457. console.info("No audios found on your playlist, skipping script...");
  458. return;
  459. }
  460. const playlistOptions = getPlaylistOptions();
  461. if (playlistOptions.linkAudioVolumes) {
  462. loadPreviousSettings();
  463. linkAllAudioVolumeSliders(audios);
  464. }
  465. if (playlistOptions.rememberAudioRepetitionAmount) rememberAudioRepeatNumber();
  466. if (playlistOptions.enableShortcutKeys) bindKeyboardShortcuts();
  467. }
  468.  
  469. function configureLessonAudios() {
  470. const lessionsOptions = getLessionsOptions();
  471. getAudios().forEach(audio => {
  472. console.info(`Setting audio of lesson to ${lessionsOptions["audioVolume"]} at speed ${lessionsOptions["audioSpeed"]}`);
  473. audio.volume = lessionsOptions["audioVolume"];
  474. audio.playbackRate = lessionsOptions["audioSpeed"];
  475. });
  476. }
  477.  
  478. function addTextLessionHideTranslationButton() {
  479. const textTranslationDivs = Array.from(document.querySelectorAll("div.portuguese_block"));
  480. if (textTranslationDivs.length === 0) {
  481. console.info("This lession have no text");
  482. return;
  483. }
  484. const firstTranslationDiv = textTranslationDivs[0];
  485. const translationDivStyle = window.getComputedStyle(firstTranslationDiv);
  486.  
  487. const audioDiv = document.querySelector("div#audio-text");
  488. if (audioDiv == null) {
  489. console.error("Could not find audio div to add the hide/show button!");
  490. return;
  491. }
  492. const firstTextLineDiv = audioDiv.nextSibling;
  493. if (!isMobile) {
  494. firstTextLineDiv.style.marginTop = "1.3em";
  495. } else {
  496. firstTextLineDiv.style.marginTop = "2.0em";
  497. }
  498.  
  499. const toggleDiv = document.createElement("div");
  500. toggleDiv.style.display = "flex";
  501. toggleDiv.style.margin = translationDivStyle.margin;
  502. if (!isMobile) {
  503. toggleDiv.style.width = "80%";
  504. toggleDiv.style.marginTop = "1.5em";
  505. } else {
  506. toggleDiv.style.width = translationDivStyle.width;
  507. toggleDiv.style.marginTop = "2.2em";
  508. }
  509.  
  510. const toggleButton = document.createElement("button");
  511. toggleButton.style.height = "2.4rem";
  512. toggleButton.style.width = "8.35rem";
  513. if (isMobile) {
  514. toggleButton.style.borderRadius = "0.5rem";
  515. }
  516. toggleDiv.appendChild(toggleButton);
  517.  
  518. let isHidden = loadSetting("text_translation_hidden") === true;
  519. updateTranslationButtonText(isHidden, toggleButton);
  520. const defaultDisplayMode = firstTranslationDiv.style.display ?? "block";
  521.  
  522. toggleButton.addEventListener("click", function() {
  523. isHidden = toggleTranslationDisplay(isHidden, toggleButton, textTranslationDivs, defaultDisplayMode);
  524. saveSetting("text_translation_hidden", isHidden);
  525. }, false);
  526.  
  527. if (isHidden) {
  528. for (let translationDiv of textTranslationDivs) {
  529. translationDiv.style.display = "none";
  530. }
  531. }
  532. audioDiv.parentNode.insertBefore(toggleDiv, firstTextLineDiv);
  533. }
  534.  
  535. function toggleTranslationDisplay(isHidden, toggleButton, textTranslationDivs, defaultDisplayMode) {
  536. for (let translationDiv of textTranslationDivs) {
  537. if (isHidden) {
  538. translationDiv.style.display = defaultDisplayMode;
  539. } else {
  540. translationDiv.style.display = "none";
  541. }
  542. }
  543. updateTranslationButtonText(!isHidden, toggleButton);
  544. return !isHidden;
  545. }
  546.  
  547. function updateTranslationButtonText(isHidden, toggleButton) {
  548. if (isHidden) {
  549. toggleButton.innerText = "Mostrar tradução";
  550. } else {
  551. toggleButton.innerText = "Esconder tradução";
  552. }
  553. }
  554.  
  555. function makeFuriganaAppearOnHover() {
  556. const furiganaTexts = Array.from(document.querySelectorAll("div.furigana_text"));
  557. for (const sentenceDiv of furiganaTexts) {
  558. const rts = Array.from(sentenceDiv.querySelectorAll("rt"));
  559. if (rts.length === 0) return;
  560. rts.forEach(rt => rt.style.visibility = "hidden");
  561.  
  562. sentenceDiv.addEventListener("mouseenter", () => rts.forEach(rt => rt.style.visibility = "visible"), false);
  563. sentenceDiv.addEventListener("mouseleave", () => rts.forEach(rt => rt.style.visibility = "hidden"), false);
  564. }
  565. }
  566.  
  567. function configureLesson() {
  568. if (!(/(\/.+){3,}/i.test(window.location.pathname))) return;
  569. const lessionsOptions = getLessionsOptions();
  570. configureLessonAudios();
  571. if (lessionsOptions.enableHideTranslationButton) addTextLessionHideTranslationButton();
  572. if (lessionsOptions.showFuriganaOnMouseOver) makeFuriganaAppearOnHover();
  573. }
  574.  
  575. function addAdditionalKeybinds() {
  576. Mousetrap.addKeycodes({
  577. 178: "stopmedia",
  578. 179: "playpausemedia",
  579. });
  580. }
  581.  
  582. window.addEventListener("DOMContentLoaded", function () {
  583. 'use strict';
  584. addAdditionalKeybinds();
  585. configurePlaylist();
  586. configureLesson();
  587. injectToolkitOverlay();
  588. });
  589.  
  590. const isMobile = (function(){
  591. let check = false;
  592. (function(a){if(/(android|bb\d+|meego).+mobile|avantgo|bada\/|blackberry|blazer|compal|elaine|fennec|hiptop|iemobile|ip(hone|od)|iris|kindle|lge |maemo|midp|mmp|mobile.+firefox|netfront|opera m(ob|in)i|palm( os)?|phone|p(ixi|re)\/|plucker|pocket|psp|series(4|6)0|symbian|treo|up\.(browser|link)|vodafone|wap|windows ce|xda|xiino/i.test(a)||/1207|6310|6590|3gso|4thp|50[1-6]i|770s|802s|a wa|abac|ac(er|oo|s\-)|ai(ko|rn)|al(av|ca|co)|amoi|an(ex|ny|yw)|aptu|ar(ch|go)|as(te|us)|attw|au(di|\-m|r |s )|avan|be(ck|ll|nq)|bi(lb|rd)|bl(ac|az)|br(e|v)w|bumb|bw\-(n|u)|c55\/|capi|ccwa|cdm\-|cell|chtm|cldc|cmd\-|co(mp|nd)|craw|da(it|ll|ng)|dbte|dc\-s|devi|dica|dmob|do(c|p)o|ds(12|\-d)|el(49|ai)|em(l2|ul)|er(ic|k0)|esl8|ez([4-7]0|os|wa|ze)|fetc|fly(\-|_)|g1 u|g560|gene|gf\-5|g\-mo|go(\.w|od)|gr(ad|un)|haie|hcit|hd\-(m|p|t)|hei\-|hi(pt|ta)|hp( i|ip)|hs\-c|ht(c(\-| |_|a|g|p|s|t)|tp)|hu(aw|tc)|i\-(20|go|ma)|i230|iac( |\-|\/)|ibro|idea|ig01|ikom|im1k|inno|ipaq|iris|ja(t|v)a|jbro|jemu|jigs|kddi|keji|kgt( |\/)|klon|kpt |kwc\-|kyo(c|k)|le(no|xi)|lg( g|\/(k|l|u)|50|54|\-[a-w])|libw|lynx|m1\-w|m3ga|m50\/|ma(te|ui|xo)|mc(01|21|ca)|m\-cr|me(rc|ri)|mi(o8|oa|ts)|mmef|mo(01|02|bi|de|do|t(\-| |o|v)|zz)|mt(50|p1|v )|mwbp|mywa|n10[0-2]|n20[2-3]|n30(0|2)|n50(0|2|5)|n7(0(0|1)|10)|ne((c|m)\-|on|tf|wf|wg|wt)|nok(6|i)|nzph|o2im|op(ti|wv)|oran|owg1|p800|pan(a|d|t)|pdxg|pg(13|\-([1-8]|c))|phil|pire|pl(ay|uc)|pn\-2|po(ck|rt|se)|prox|psio|pt\-g|qa\-a|qc(07|12|21|32|60|\-[2-7]|i\-)|qtek|r380|r600|raks|rim9|ro(ve|zo)|s55\/|sa(ge|ma|mm|ms|ny|va)|sc(01|h\-|oo|p\-)|sdk\/|se(c(\-|0|1)|47|mc|nd|ri)|sgh\-|shar|sie(\-|m)|sk\-0|sl(45|id)|sm(al|ar|b3|it|t5)|so(ft|ny)|sp(01|h\-|v\-|v )|sy(01|mb)|t2(18|50)|t6(00|10|18)|ta(gt|lk)|tcl\-|tdg\-|tel(i|m)|tim\-|t\-mo|to(pl|sh)|ts(70|m\-|m3|m5)|tx\-9|up(\.b|g1|si)|utst|v400|v750|veri|vi(rg|te)|vk(40|5[0-3]|\-v)|vm40|voda|vulc|vx(52|53|60|61|70|80|81|83|85|98)|w3c(\-| )|webc|whit|wi(g |nc|nw)|wmlb|wonu|x700|yas\-|your|zeto|zte\-/i.test(a.substr(0,4))) check = true;})(navigator.userAgent||navigator.vendor||window.opera);
  593. return check;
  594. })();
  595.  
  596. /**
  597. * Icons.
  598. */
  599. function closeIcon() {
  600. return ``;
  601. }