programajaponesonline.com.br - Toolkit

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. There's also a function to hide furigana (so it only appears when you hover the mouse over the paragraphs).

// ==UserScript==
// @name        programajaponesonline.com.br - Toolkit
// @name:pt-BR  programajaponesonline.com.br - Toolkit
// @namespace   secretx_scripts
// @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. There's also a function to hide furigana (so it only appears when you hover the mouse over the paragraphs).
// @description:pt-br Esse script faz várias coisinhas básicas, como arrumar volume dos áudios que você está escutando, unificar o volume dos áudios (incluindo o mute), adiciona alguns atalhos para tocar os áudios da playlist (como por exemplo colocar a barra de espaço para sempre pausar o áudio atual), adiciona a possibilidade de aumentar o volume usando a tecla de Mais (+) e Menos (-) do seu numpad. Tem também uma função de ocultar furigana (deixar pra aparecer somente quando passar o mouse por cima).
// @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();
    document.body.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 ``;
}

var GM_SuperValue = new function () {

    var JSON_MarkerStr  = 'json_val: ';
    var FunctionMarker  = 'function_code: ';

    function ReportError (msg) {
        if (console && console.error)
            console.log (msg);
        else
            throw new Error (msg);
    }

    //--- Check that the environment is proper.
    if (typeof GM_setValue != "function")
        ReportError ('This library requires Greasemonkey! GM_setValue is missing.');
    if (typeof GM_getValue != "function")
        ReportError ('This library requires Greasemonkey! GM_getValue is missing.');


    /*--- set ()
        GM_setValue (http://wiki.greasespot.net/GM_setValue) only stores:
        strings, booleans, and integers (a limitation of using Firefox
        preferences for storage).

        This function extends that to allow storing any data type.

        Parameters:
            varName
                String: The unique (within this script) name for this value.
                Should be restricted to valid Javascript identifier characters.
            varValue
                Any valid javascript value.  Just note that it is not advisable to
                store too much data in the Firefox preferences.

        Returns:
            undefined
    */
    this.set = function (varName, varValue) {

        if ( ! varName) {
            ReportError ('Illegal varName sent to GM_SuperValue.set().');
            return;
        }
        if (/[^\w _-]/.test (varName) ) {
            ReportError ('Suspect, probably illegal, varName sent to GM_SuperValue.set().');
        }

        switch (typeof varValue) {
            case 'undefined':
                ReportError ('Illegal varValue sent to GM_SuperValue.set().');
                break;
            case 'boolean':
            case 'string':
                //--- These 2 types are safe to store, as is.
                GM_setValue (varName, varValue);
                break;
            case 'number':
                /*--- Numbers are ONLY safe if they are integers.
                    Note that hex numbers, EG 0xA9, get converted
                    and stored as decimals, EG 169, automatically.
                    That's a feature of JavaScript.

                    Also, only a 32-bit, signed integer is allowed.
                    So we only process +/-2147483647 here.
                */
                if (varValue === parseInt (varValue)  &&  Math.abs (varValue) < 2147483647)
                {
                    GM_setValue (varName, varValue);
                    break;
                }
            case 'object':
                /*--- For all other cases (but functions), and for
                    unsafe numbers, store the value as a JSON string.
                */
                var safeStr = JSON_MarkerStr + JSON.stringify (varValue);
                GM_setValue (varName, safeStr);
                break;
            case 'function':
                /*--- Functions need special handling.
                */
                var safeStr = FunctionMarker + varValue.toString ();
                GM_setValue (varName, safeStr);
                break;

            default:
                ReportError ('Unknown type in GM_SuperValue.set()!');
                break;
        }
    }//-- End of set()


    /*--- get ()
        GM_getValue (http://wiki.greasespot.net/GM_getValue) only retieves:
        strings, booleans, and integers (a limitation of using Firefox
        preferences for storage).

        This function extends that to allow retrieving any data type -- as
        long as it was stored with GM_SuperValue.set().

        Parameters:
            varName
                String: The property name to get. See GM_SuperValue.set for details.
            defaultValue
                Optional. Any value to be returned, when no value has previously
                been set.

        Returns:
            When this name has been set...
                The variable or function value as previously set.

            When this name has not been set, and a default is provided...
                The value passed in as a default

            When this name has not been set, and default is not provided...
                undefined
    */
    this.get = function (varName, defaultValue) {

        if ( ! varName) {
            ReportError ('Illegal varName sent to GM_SuperValue.get().');
            return;
        }
        if (/[^\w _-]/.test (varName) ) {
            ReportError ('Suspect, probably illegal, varName sent to GM_SuperValue.get().');
        }

        //--- Attempt to get the value from storage.
        var varValue    = GM_getValue (varName);
        if (!varValue)
            return defaultValue;

        //--- We got a value from storage. Now unencode it, if necessary.
        if (typeof varValue == "string") {
            //--- Is it a JSON value?
            var regxp       = new RegExp ('^' + JSON_MarkerStr + '(.+)$');
            var m           = varValue.match (regxp);
            if (m  &&  m.length > 1) {
                varValue    = JSON.parse ( m[1] );
                return varValue;
            }

            //--- Is it a function?
            var regxp       = new RegExp ('^' + FunctionMarker + '((?:.|\n|\r)+)$');
            var m           = varValue.match (regxp);
            if (m  &&  m.length > 1) {
                varValue    = eval ('(' + m[1] + ')');
                return varValue;
            }
        }

        return varValue;
    }//-- End of get()


    /*--- runTestCases ()
        Tests storage and retrieval every every knid of value.
        Note: makes extensive use of the console.

        Parameters:
            bUseConsole
                Boolean: If this is true, uses the console environment to store
                the data.  Useful for dev test.
        Returns:
            true, if pass.  false, otherwise.
    */
    this.runTestCases = function (bUseConsole) {

        if (bUseConsole) {
            //--- Set up the environment to use local JS, and not the GM environment.
            this.testStorage    = {};
            var context         = this;
            this.oldSetFunc     = (typeof GM_setValue == "function") ? GM_setValue : null;
            this.oldGetFunc     = (typeof GM_getValue == "function") ? GM_getValue : null;

            GM_setValue    = function (varName, varValue) {
                console.log ('Storing: ', varName, ' as: ', varValue);
                context.testStorage[varName] = varValue;
            }

            GM_getValue    = function (varName, defaultValue) {
                var varValue    = context.testStorage[varName];
                if (!varValue)
                    varValue    = defaultValue;

                console.log ('Retrieving: ', varName, '. Got: ', varValue);

                return varValue;
            }
        }

        var dataBefore  =   [null, true, 1, 1.1, -1.0, 2.0E9,  2.77E9,  2.0E-9, 0xA9, 'string',
            [1,2,3], {a:1, B:2}, function () {a=7; console.log ("Neat! Ain't it?"); }
        ];

        for (var J = 0;  J <= dataBefore.length;  J++)
        {
            var X       = dataBefore[J];
            console.log (J, ': ', typeof X, X);

            this.set ('Test item ' + J, X);
            console.log ('\n');
        }

        console.log ('\n***********************\n***********************\n\n');

        var dataAfter   = [];

        for (var J = 0;  J < dataBefore.length;  J++)
        {
            var X       = this.get ('Test item ' + J);
            dataAfter.push (X);
            console.log ('\n');
        }

        console.log (dataBefore);
        console.log (dataAfter);

        dataAfter[12]();

        //--- Now test for pass/fail.  The objects won't be identical but contenets are.
        var bPass;
        if (dataBefore.toString()  ==  dataAfter.toString() ) {
            var pfStr   = 'PASS!';
            bPass       = true;
        } else {
            var pfStr   = 'FAIL!';
            bPass       = false;
        }
        console.log ( "\n***************************        \
                       \n***************************        \
                       \n***************************        \
                       \n*****     " + pfStr + "       *****        \
                       \n***************************        \
                       \n***************************        \
                       \n***************************\n");

        if (bUseConsole) {
            //--- Restore the GM functions.
            GM_setValue    = this.oldSetFunc;
            GM_getValue    = this.oldGetFunc;
        }
        else {
            //--- Clean up the FF storage!

            for (var J = 0;  J < dataBefore.length;  J++)
            {
                GM_deleteValue ('Test item ' + J);
            }
        }

        return bPass;

    }//-- End of runTestCases()
};


//GM_SuperValue.runTestCases  (true);

//--- EOF