您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
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.
当前为
// ==UserScript== // @name programajaponesonline.com.br - Use spacebar keyboard shortcut to pause/play audios, and more... // @namespace secretx_scripts // @match *://portal.programajaponesonline.com.br/playlist/ // @match *://portal.programajaponesonline.com.br/*/*/* // @version 2023.01.03 // @author SecretX // @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. // @grant GM_addStyle // @grant GM_getValue // @grant GM_setValue // @run-at document-start // @require https://craig.global.ssl.fastly.net/js/mousetrap/mousetrap.min.js?a4098 // @icon https://i.imgur.com/vn8ClVJ.png // @license GNU LGPLv3 // ==/UserScript== const keybindHandlers = { playPauseAudioKey: playPauseAudioPlaying, stopAudioKey: stopAudioPlaying, increaseVolumeKey: () => forEachAudio(stepUpVolume), decreaseVolumeKey: () => forEachAudio(stepDownVolume), } Object.defineProperty(HTMLElement.prototype, "isVisible", { value: function() { return (this.offsetParent !== null); }, writable: true, configurable: true }); Object.defineProperty(Array.prototype, "insert", { value: function(index, ...items) { return [...this.slice(0, index), ...items, ...this.slice(index)] }, writable: true, configurable: true }); function loadSetting(name) { return GM_SuperValue.get(name); } function saveSetting(name, value) { GM_SuperValue.set(name, value); } function getPlaylistOptions() { const cached = loadSetting("playlist_options"); if (cached != null) return cached; const defaults = { enableShortcutKeys: true, rememberAudioRepetitionAmount: true, linkAudioVolumes: true, playPauseAudioKey: ["space", "playpausemedia"], stopAudioKey: ["stopmedia"], // Stop media increaseVolumeKey: ["+"], // Plus Numpad decreaseVolumeKey: ["-"], // Minus Numpad volumeStep: 0.05, // 5% volumeMin: 0.0, // 0% volumeMax: 1.0, // 100% }; saveSetting("playlist_options", defaults); return defaults; } function savePlaylistOptions(playlistOptions) { saveSetting("playlist_options", playlistOptions); } function getLessionsOptions() { const cached = loadSetting("lessions_options"); if (cached != null) return cached; const defaults = { enableHideTranslationButton: true, showFuriganaOnMouseOver: false, audioVolume: 1.0, audioSpeed: 1.0, }; saveSetting("lessions_options", defaults); return defaults; } function saveLessionsOptions(playlistOptions) { saveSetting("lessions_options", playlistOptions); } function showElem(htmlElem) { if (htmlElem == null) return; htmlElem.classList.remove("hidden"); } function hideElem(htmlElem) { if (htmlElem == null) return; htmlElem.classList.add("hidden"); } const getAudios = () => Array.from(document.querySelectorAll("audio")); const forEachAudio = action => getAudios().forEach(audio => action(audio)); const getPlayingAudio = () => { const audios = getAudios().filter(audio => audio.isVisible()); return audios.length === 1 ? audios[0] : null; } const setVolumeOfAllAudios = volume => forEachAudio(audio => audio.volume = Math.max(0.0, Math.min(1.0, volume))); const setMuteOfAllAudios = isMuted => forEachAudio(audio => audio.muted = isMuted); const toggleAudio = (audio) => audio.paused ? audio.play() : audio.pause(); const stepDownVolume = (audio) => audio.volume = Math.max(audio.volume - getPlaylistOptions()["volumeStep"], getPlaylistOptions()["volumeMin"]); const stepUpVolume = (audio) => audio.volume = Math.min(audio.volume + getPlaylistOptions()["volumeStep"], getPlaylistOptions()["volumeMax"]); function loadPreviousSettings() { const previousAudioVolume = loadSetting("audio_volume"); if (previousAudioVolume != null) setVolumeOfAllAudios(previousAudioVolume); setMuteOfAllAudios(loadSetting("audio_muted") === true); } function linkAllAudioVolumeSliders(audios) { audios.forEach(audio => audio.addEventListener("volumechange", () => { setVolumeOfAllAudios(audio.volume) setMuteOfAllAudios(audio.muted); saveSetting("audio_volume", audio.volume); saveSetting("audio_muted", audio.muted); })); } function playPauseAudioPlaying() { const playingAudio = getPlayingAudio(); if (playingAudio == null) { console.info("No audio is playing"); return; } toggleAudio(playingAudio); } function stopAudioPlaying() { const playingAudio = getPlayingAudio(); if (playingAudio == null) { console.info("No audio is playing"); return; } if (!playingAudio.paused) playingAudio.pause(); } // UI const mainDivId = "jpo_toolkit"; const parentDivId = `${mainDivId}_parent`; const formId = `${mainDivId}_form`; const openButtonId = "jpo_toolkit_open_button"; const closeButtonId = "jpo_toolkit_close_button"; function injectToolkitOverlayCss() { const css = ` @import url("https://fonts.googleapis.com/css?family=Bebas+Neue:400|Inter:400"); :root { --black: rgba(0, 0, 0, 1); --baby-powder: rgba(252, 252, 252, 1); --font-size-s: 0.7rem; --font-size-m: 0.8rem; --font-size-l: 1.7rem; --font-family-bebas_neue: "Bebas Neue"; --font-family-inter: "Inter"; } #${parentDivId} { width: 24rem; min-height: 9rem; margin-top: 0; margin-right: 0; margin-bottom: 0; padding: 0.5rem 1.5rem 1.75rem 1.5rem; border: 0 none; background: var(--baby-powder); display: block; position: fixed; z-index: 16000001; right: 1rem; top: 9rem; border-radius: 5px; box-shadow:0px 4px 4px rgba(0, 0, 0, 0.25); } .jpo_toolkit_title_bar { display: flex; justify-content: space-between; } .jpo_toolkit_title_bar > h2 { color: var(--black); font-family: var(--font-family-bebas_neue); font-size: var(--font-size-l); font-weight: 400; letter-spacing: 0; line-height: normal; margin-bottom: 0; } #${closeButtonId} { display: flex; justify-content: center; align-items: center; border-radius: 2.5rem; margin-right: -0.8rem; } #${closeButtonId} > img { height: var(--font-size-l); width: var(--font-size-l); filter: contrast(30%) opacity(70%); } div.toolkit_form_section { display: grid; font-size: var(--font-size-m); } div.toolkit_form_section_second_onwards { margin-top: 0.5em } div.toolkit_form_field_label { display: flex; align-items: center; } div.toolkit_form_field_label > label { color: var(--black); font-family: var(--font-family-inter); font-size: var(--font-size-m); font-weight: 400; letter-spacing: 0; line-height: normal; } .hidden { display: none !important; } .rectangle_box { margin-top: 1.7rem; position: relative; padding: 1rem 0.5rem 0.5rem 0.5rem; border: 1px solid; border-color: #d7d7d7; border-radius: 5px; width: 100%; } .overlap_group { align-items: flex-end; background-color: var(--baby-powder); display: flex; justify-content: flex-end; position: absolute; top: 0; margin: -0.6rem 0 0 0.5rem; padding: 0 0.2rem; } .section_title { color: #959595; font-family: var(--font-family-bebas_neue); font-size: 1rem; font-weight: 400; letter-spacing: 0; line-height: normal; } .jpo_toolkit_input { width: 3.7rem; padding: 4px 4px 4px 8px; border-radius: 7px; border: 1px solid #ccc; font-size: var(--font-size-m); } .go_to_right { margin-left: auto; } `; // Use Violent Monkey global function to inject our CSS onto page GM_addStyle(css); } function createToolkitOverlay() { const div = document.createElement("div"); div.id = parentDivId; hideElem(div); const playlistOptions = getPlaylistOptions(); const html = ` <div id="${mainDivId}"> <div class="jpo_toolkit_title_bar"> <h2>PJO Toolkit️</h2> <div id="${closeButtonId}"> <img src="${closeIcon()}" alt="Close this window"/> </div> </div> <p style="font-size: 0.7rem; font-style: italic; margin-bottom: 0.25rem">Work in progress!</p> <form id="${formId}" name="${formId}" action=""> <div class="toolkit_form_section"> <div class="rectangle_box"> <div class="overlap_group"> <div class="section_title">Playlist</div> </div> <div class="toolkit_form_field_label"> <label for="enable_shortcut_keys">Ativar botões de atalho</label> <input type="checkbox" id="enable_shortcut_keys" class="go_to_right"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="remember_audio_repetition_amount">Lembrar do número de repetição dos áudios</label> <input type="checkbox" id="remember_audio_repetition_amount" class="go_to_right"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="link_audio_volumes">Unificar volume de todos os áudios</label> <input type="checkbox" id="link_audio_volumes" class="go_to_right"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="volume_min">Volume mínimo (0-100)</label> <input type="number" id="volume_min" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeMin * 100.0)}" min="0" max="100"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="volume_max">Volume máximo (0-100)</label> <input type="number" id="volume_max" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeMax * 100.0)}" min="0" max="100"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="volume_step">Volume step (1-100)</label> <input type="number" id="volume_step" class="jpo_toolkit_input go_to_right" value="${Math.floor(playlistOptions.volumeStep * 100.0)}" min="1" max="100"/> </div> </div> <div class="rectangle_box"> <div class="overlap_group"> <div class="section_title">Lessons</div> </div> <div class="toolkit_form_field_label"> <label for="hide_translation_button">Exibir botão de esconder tradução de textos</label> <input type="checkbox" id="hide_translation_button" class="go_to_right"/> </div> <div class="toolkit_form_field_label toolkit_form_section_second_onwards"> <label for="show_furigana_on_mouse_over">Furigana somente ao passar o mouse</label> <input type="checkbox" id="show_furigana_on_mouse_over" class="go_to_right"/> </div> </div> </form> </div> `.trim(); div.innerHTML = html; setPlaylistInteractListeners(div); return div; } function setPlaylistInteractListeners(div) { const setPlaylistCheckboxOption = (checkbox, property) => { const playlistOptions = getPlaylistOptions(); const isChecked = !playlistOptions[property]; checkbox.attributes.checked = isChecked; playlistOptions[property] = isChecked; savePlaylistOptions(playlistOptions); console.info(`Setting '${property}' to '${isChecked}'`); }; const preparePlaylistCheckbox = (inputSelector, property) => { const checkbox = div.querySelector(inputSelector); checkbox.addEventListener("input", () => setPlaylistCheckboxOption(checkbox, property), false); if (getPlaylistOptions()[property]) checkbox.checked = true; } preparePlaylistCheckbox("input#enable_shortcut_keys", "enableShortcutKeys"); preparePlaylistCheckbox("input#remember_audio_repetition_amount", "rememberAudioRepetitionAmount"); preparePlaylistCheckbox("input#link_audio_volumes", "linkAudioVolumes"); const setPlaylistVolumeOption = (input, property) => { const playlistOptions = getPlaylistOptions(); const newValue = Math.min(input.max, Math.max(input.min, input.value)) / 100.0; playlistOptions[property] = newValue; savePlaylistOptions(playlistOptions); console.info(`Setting '${property}' to '${newValue}'`); }; const playlistVolumeMin = div.querySelector("input#volume_min"); playlistVolumeMin.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeMin, "volumeMin"), false); const playlistVolumeMax = div.querySelector("input#volume_max"); playlistVolumeMax.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeMax, "volumeMax"), false); const playlistVolumeStep = div.querySelector("input#volume_step"); playlistVolumeStep.addEventListener("input", () => setPlaylistVolumeOption(playlistVolumeStep, "volumeStep"), false); const setLessionsCheckboxOption = (checkbox, property) => { const lessionsOptions = getLessionsOptions(); const isChecked = !lessionsOptions[property]; checkbox.attributes.checked = isChecked; lessionsOptions[property] = isChecked; saveLessionsOptions(lessionsOptions); console.info(`Setting '${property}' to '${isChecked}'`); }; const hideTranslationButton = div.querySelector("input#hide_translation_button"); hideTranslationButton.addEventListener("input", () => setLessionsCheckboxOption(hideTranslationButton, "enableHideTranslationButton"), false); if (getLessionsOptions().enableHideTranslationButton) hideTranslationButton.checked = true; const showFuriganaOnMouseOver = div.querySelector("input#show_furigana_on_mouse_over"); showFuriganaOnMouseOver.addEventListener("input", () => setLessionsCheckboxOption(hideTranslationButton, "showFuriganaOnMouseOver"), false); if (getLessionsOptions().showFuriganaOnMouseOver) showFuriganaOnMouseOver.checked = true; div.querySelector(`#${closeButtonId}`).addEventListener("click", () => hideElem(div), false); } function appendOpenToolkitButton() { const userInfoDiv = document.querySelector("div.user-info"); const userInfoItems = userInfoDiv.innerHTML.split("|"); const newItems = userInfoItems.insert(1, `<a id='${openButtonId}'>PJO Toolkit</a>`); userInfoDiv.innerHTML = newItems.join(" | "); document.querySelector(`#${openButtonId}`).addEventListener("click", () => showElem(document.querySelector(`#${parentDivId}`))); } function injectToolkitOverlay() { injectToolkitOverlayCss(); const mainDiv = document.querySelector("div#primary"); if (mainDiv == null) return; mainDiv.appendChild(createToolkitOverlay()); appendOpenToolkitButton(); } // Main functions function rememberAudioRepeatNumber() { const audioRepeat = loadSetting("audio_repeat_amount") ?? {}; // div.playlist-title > div > input.playlist-item-loop const audioDivs = Array.from(document.querySelectorAll("li.playlist-item-active")); for (const audioDiv of audioDivs) { const audioInput = audioDiv.querySelector("div.playlist-title > div > input.playlist-item-loop"); // Will become something like 'Nihongo Rise - Tópico 08アンケート調査 (Questionário de pesquisa.)' const audioName = Array.from(audioDiv.querySelectorAll("div.playlist-title > div > span.jp-title")) .map(span => span.innerText) .reduce((a, b) => a + b, "") .trim(); if (audioInput == null || audioName == null || audioName.length === 0) continue; const previousRepeatAmount = audioRepeat[audioName]; if (previousRepeatAmount != null && audioInput.value === "1") { // Restore previous repeat value audioInput.value = previousRepeatAmount; } audioInput.addEventListener("input", () => { console.info(`Changed '${audioName}' repeat amount to '${audioInput.value}'`); audioRepeat[audioName] = audioInput.value; saveSetting("audio_repeat_amount", audioRepeat); }, false); } } function bindKeyboardShortcuts() { for (const [actionName, keyShortcuts] of Object.entries(getPlaylistOptions())) { const executorMethod = keybindHandlers[actionName]; if (executorMethod == null) continue; keyShortcuts.forEach(keyShortcut => Mousetrap.bind(keyShortcut, () => { executorMethod(); return false; // prevents default browser behavior })); } console.log("Bound keyboard shortcuts!"); } function configurePlaylist() { if (window.location.pathname !== "/playlist/") return; const audios = getAudios(); if (audios.length === 0) { console.info("No audios found on your playlist, skipping script..."); return; } const playlistOptions = getPlaylistOptions(); if (playlistOptions.linkAudioVolumes) { loadPreviousSettings(); linkAllAudioVolumeSliders(audios); } if (playlistOptions.rememberAudioRepetitionAmount) rememberAudioRepeatNumber(); if (playlistOptions.enableShortcutKeys) bindKeyboardShortcuts(); } function configureLessonAudios() { const lessionsOptions = getLessionsOptions(); getAudios().forEach(audio => { console.info(`Setting audio of lesson to ${lessionsOptions["audioVolume"]} at speed ${lessionsOptions["audioSpeed"]}`); audio.volume = lessionsOptions["audioVolume"]; audio.playbackRate = lessionsOptions["audioSpeed"]; }); } function addTextLessionHideTranslationButton() { const textTranslationDivs = Array.from(document.querySelectorAll("div.portuguese_block")); if (textTranslationDivs.length === 0) { console.info("This lession have no text"); return; } const firstTranslationDiv = textTranslationDivs[0]; const translationDivStyle = window.getComputedStyle(firstTranslationDiv); const audioDiv = document.querySelector("div#audio-text"); if (audioDiv == null) { console.error("Could not find audio div to add the hide/show button!"); return; } const firstTextLineDiv = audioDiv.nextSibling; if (!isMobile) { firstTextLineDiv.style.marginTop = "1.3em"; } else { firstTextLineDiv.style.marginTop = "2.0em"; } const toggleDiv = document.createElement("div"); toggleDiv.style.display = "flex"; toggleDiv.style.margin = translationDivStyle.margin; if (!isMobile) { toggleDiv.style.width = "80%"; toggleDiv.style.marginTop = "1.5em"; } else { toggleDiv.style.width = translationDivStyle.width; toggleDiv.style.marginTop = "2.2em"; } const toggleButton = document.createElement("button"); toggleButton.style.height = "2.4rem"; toggleButton.style.width = "8.35rem"; if (isMobile) { toggleButton.style.borderRadius = "0.5rem"; } toggleDiv.appendChild(toggleButton); let isHidden = loadSetting("text_translation_hidden") === true; updateTranslationButtonText(isHidden, toggleButton); const defaultDisplayMode = firstTranslationDiv.style.display ?? "block"; toggleButton.addEventListener("click", function() { isHidden = toggleTranslationDisplay(isHidden, toggleButton, textTranslationDivs, defaultDisplayMode); saveSetting("text_translation_hidden", isHidden); }, false); if (isHidden) { for (let translationDiv of textTranslationDivs) { translationDiv.style.display = "none"; } } audioDiv.parentNode.insertBefore(toggleDiv, firstTextLineDiv); } function toggleTranslationDisplay(isHidden, toggleButton, textTranslationDivs, defaultDisplayMode) { for (let translationDiv of textTranslationDivs) { if (isHidden) { translationDiv.style.display = defaultDisplayMode; } else { translationDiv.style.display = "none"; } } updateTranslationButtonText(!isHidden, toggleButton); return !isHidden; } function updateTranslationButtonText(isHidden, toggleButton) { if (isHidden) { toggleButton.innerText = "Mostrar tradução"; } else { toggleButton.innerText = "Esconder tradução"; } } function makeFuriganaAppearOnHover() { const furiganaTexts = Array.from(document.querySelectorAll("div.furigana_text")); for (const sentenceDiv of furiganaTexts) { const rts = Array.from(sentenceDiv.querySelectorAll("rt")); if (rts.length === 0) return; rts.forEach(rt => rt.style.visibility = "hidden"); sentenceDiv.addEventListener("mouseenter", () => rts.forEach(rt => rt.style.visibility = "visible"), false); sentenceDiv.addEventListener("mouseleave", () => rts.forEach(rt => rt.style.visibility = "hidden"), false); } } function configureLesson() { if (!(/(\/.+){3,}/i.test(window.location.pathname))) return; const lessionsOptions = getLessionsOptions(); configureLessonAudios(); if (lessionsOptions.enableHideTranslationButton) addTextLessionHideTranslationButton(); if (lessionsOptions.showFuriganaOnMouseOver) makeFuriganaAppearOnHover(); } function addAdditionalKeybinds() { Mousetrap.addKeycodes({ 178: "stopmedia", 179: "playpausemedia", }); } window.addEventListener("DOMContentLoaded", function () { 'use strict'; addAdditionalKeybinds(); configurePlaylist(); configureLesson(); injectToolkitOverlay(); }); const isMobile = (function(){ let check = false; (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); return check; })(); /** * Icons. */ function closeIcon() { return ``; }