Play Video Locally

Gives you a simple button on certain media servers to send the video to a local player.

// ==UserScript==
// @name         Play Video Locally
// @namespace    https://naeembolchhi.github.io/
// @version      0.6
// @description  Gives you a simple button on certain media servers to send the video to a local player.
// @author       NaeemBolchhi
// @match        http*://circleftp.net/*
// @match        http*://10.16.100.244/*
// @match        http*://freedrivemovie.com/*
// @match        http*://movies.discoveryftp.net/*
// @license      GPL-3.0-or-later
// @icon         
// @run-at       document-idle
// @grant        none
// ==/UserScript==

// Supported video formats
const vform = [".mp4",".mkv"];

// Hide video players (change value to "invisible" to hide a player, and to "visible" to show it)
const vvid = {
    "potplayer": "visible",
    "vlcplayer": "visible",
    "xplayer": "visible",
    "mxplayer": "visible",
    "mxplayerpro": "visible"
};

// SVG Icons
const icons = {
    "toggle":`<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 24 24"><path d="M0 0h24v24H0z" fill="none"/><path d="M7 11H1v2h6v-2zm2.17-3.24L7.05 5.64 5.64 7.05l2.12 2.12 1.41-1.41zM13 1h-2v6h2V1zm5.36 6.05l-1.41-1.41-2.12 2.12 1.41 1.41 2.12-2.12zM17 11v2h6v-2h-6zm-5-2c-1.66 0-3 1.34-3 3s1.34 3 3 3 3-1.34 3-3-1.34-3-3-3zm2.83 7.24l2.12 2.12 1.41-1.41-2.12-2.12-1.41 1.41zm-9.19.71l1.41 1.41 2.12-2.12-1.41-1.41-2.12 2.12zM11 23h2v-6h-2v6z"/></svg>`,
    "next":`<svg xmlns="http://www.w3.org/2000/svg" xml:space="preserve" x="0" y="0" version="1.1" viewBox="0 0 300 300"><path d="M150.4 180.1c-5.1-5.6-9.7-10.6-14.2-15.7-29.7-32.7-59.3-65.4-89-98.1-4.9-5.4-10.8-8.8-18.2-9.6C15.6 55.2 2.5 65.3.4 78.5c-1.4 9.4 1.7 17.2 8 24.1 24.9 27.3 49.7 54.7 74.5 82 14.8 16.3 30.1 32.1 44.1 49 10.9 13.1 34.8 12.8 46.1 0 38.9-43.9 78.6-87 118-130.4 6.1-6.7 9.9-14 8.7-23.3-1.4-10.7-7.3-18.3-17.5-21.8-10.9-3.8-20.7-1.1-28.5 7.3-18.5 20.1-36.7 40.3-55 60.5l-47.1 51.9c-.4.7-.7 1.4-1.3 2.3z"/></svg>`,
    "prev":`<svg xmlns="http://www.w3.org/2000/svg" xml:space="preserve" x="0" y="0" version="1.1" viewBox="0 0 300 300"><path d="M150.4 119.7c-5.1 5.6-9.7 10.6-14.2 15.7-29.7 32.7-59.3 65.4-89 98.1-4.9 5.4-10.8 8.8-18.2 9.6-13.4 1.5-26.5-8.6-28.6-21.8-1.4-9.4 1.7-17.2 8-24.1 24.9-27.3 49.7-54.7 74.5-82 14.8-16.3 30.1-32.1 44.1-49 10.9-13.1 34.8-12.8 46.1 0 38.9 43.9 78.6 87 118 130.4 6.1 6.7 9.9 14 8.7 23.3-1.4 10.7-7.3 18.3-17.5 21.8-10.9 3.8-20.7 1.1-28.5-7.3-18.5-20.1-36.7-40.3-55-60.5L151.7 122c-.4-.7-.7-1.4-1.3-2.3z"/></svg>`,
    "player":``,
    "mobile":``,
    "pc":``
};

// Get URL variables
function getVar(vlink) {
    let vars = {};
    let parts = vlink.replace(/[?&]+([^=&]+)=([^&]*)/gi, function(m,key,value) {
        vars[key] = value;
    });
    return vars;
}

// Generate protocol URL
function getURL(player, url, text) {
    let matchList = [
        {"key": "none", "val": `${url}`},
        {"key": "potplayer", "val": `potplayer://${url}`},
        {"key": "vlcplayer", "val": `vlc://${url}`},
        {"key": "xplayer", "val": `intent:${url}#Intent;package=video.player.videoplayer;S.title=${text};end`},
        {"key": "mxplayer", "val": `intent:${url}#Intent;package=com.mxtech.videoplayer.ad;S.title=${text};end`},
        {"key": "mxplayerpro", "val": `intent:${url}#Intent;package=com.mxtech.videoplayer.pro;S.title=${text};end`}
    ]

    for (let x = 0; x < matchList.length; x++) {
        if (player === matchList[x].key) {
            return matchList[x].val;
        }
    }
}

// Make any new element
function elemake(tag, innr, attr) {
    let element = document.createElement(tag);
    if (innr) {element.innerHTML = innr;}
    if (!attr) {return element;}

    for (let x = 0; x < attr.key.length; x++) {
        element.setAttribute(attr.key[x], attr.val[x]);
    }
    return element;
}

// let sample = elemake("div","ja monchay",{"key":["id","class"],"val":["oneDiv","nav pointer"]});
// document.body.appendChild(sample);

// Find supported videos in "src" and "href" of all tags
function getVids() {
    let allLinks = document.querySelectorAll('*[src], *[href]'),
        print = [];

    for (let x = 0; x < allLinks.length; x++) {
        var tmpl = "";
        if (allLinks[x].href !== undefined) {tmpl = allLinks[x].href;}
        if (allLinks[x].src !== undefined) {tmpl = allLinks[x].src;}

        if (tmpl.match(/source/)) {
            tmpl = decodeURIComponent(getVar(tmpl).source);
        }

        var ext = tmpl.replace(/.*(....)$/,'$1'),
            okay = 'no';

        for (let y = 0; y < vform.length; y++) {
            if (ext === vform[y]) {okay = 'yes';}
        }

        if (okay === 'yes' && print.indexOf(tmpl) === -1) {
            print.push(tmpl);
        }
    }

    window.vidlinks = JSON.stringify(print);
}

// Determine visibility
function visble(what, txt) {
    if (vvid[what] === "visible") {
        return `<a class="avid ${what}">Open with ${txt}</a>`;
    } else {
        return `<a class="avid ${what} hidden">Open with ${txt}</a>`;
    }
}

// Visualize the interface
function inTerface() {
    const inStyle = `<style type="text/css">
    #playVideoLocally, #playVideoLocally * {
        outline: none;
    }
    #playVideoLocally {
        position: fixed;
        display: block;
        z-index: 2147483647;
    }
    #playVideoLocally section {
        position: fixed;
        bottom: 0;
        left: 0;
        background: transparent;
        height: 100%;
        width: 100%;
        display: block;
        z-index: 2147483641;
        transition: opacity .1s linear, filter .1s linear, transform .1s linear;
    }
    #playVideoLocally section:not(.clicked) {
        opacity: 0;
        filter: blur(6px);
        transform: translateY(100%);
    }
    #playVideoLocally section > backdrop {
        position: fixed;
        background: transparent;
        height: 100svh;
        width: 100svw;
        display: block;
        z-index: 2147483640;
    }
    #playVideoLocally > .btn-main {
        position: fixed;
        bottom: 0;
        left: 0;
        z-index: 2147483642;
        display: flex;
        align-items: center;
        justify-content: center;
        height: 40px;
        width: 40px;
        padding: 10px;
        margin: 0 0 20px 20px;
        border-radius: 100%;
        fill: #cdd6f4;
        background: #11111b;
        cursor: pointer;
        box-shadow: 0 0 4px 1px rgba(0, 0, 0, 0.25);
        transition: opacity .1s linear;
    }
    #playVideoLocally > .btn-main > svg {
        transition: transform .1s linear;
    }
    #playVideoLocally > .btn-main.clicked > svg {
        transform: rotate(-90deg);
    }
    #playVideoLocally > .btn-main:not(.clicked) {
        opacity: .5;
    }
    #playVideoLocally > .btn-main:not(.clicked):hover {
        fill: #ffffff;
        opacity: 1;
    }
    #playVideoLocally > .btn-main.clicked:hover {
        opacity: 1;
        color: #ffffff;
        fill: #ffffff;
        background: #444654;
    }
    #playVideoLocally section > .section-sub {
        position: fixed;
        bottom: 0;
        left: 0;
        z-index: 2147483641;
        margin: 0 0 70px 20px;
        display: flex;
        flex-direction: column;
        gap: 10px;
    }
    #playVideoLocally section > .section-sub > span {
        height: 40px;
        width: 40px;
        padding: 10px;
        border-radius: 100%;
        display: flex;
        align-items: center;
        justify-content: center;
        font-size: 14px;
        font-weight: 700;
        color: #cdd6f4;
        fill: #cdd6f4;
        background: #11111b;
        box-shadow: 0 0 4px 1px rgba(0, 0, 0, 0.25);
        cursor: pointer;
    }
    #playVideoLocally section > .section-sub > .current {
        cursor: default;
    }
    #playVideoLocally section > .section-sub > span:hover {
        color: #ffffff;
        fill: #ffffff;
        background: #444654;
    }
    #playVideoLocally section > .section-sub > span.disabled {
        fill: #999999 !important;
        cursor: not-allowed;
    }
    #playVideoLocally section > .section-main {
        position: fixed;
        bottom: 0;
        left: 0;
        z-index: 2147483641;
        margin: 0 0 20px 70px;
        display: flex;
        flex-direction: column;
        gap: 10px;
    }
    #playVideoLocally section > .section-main > a {
        height: 40px;
        min-width: 40px;
        width: fit-content;
        display: flex;
        align-items: center;
        padding: 15px;
        font-size: 14px;
        border-radius: 500px;
        color: #cdd6f4;
        fill: #cdd6f4;
        background: #11111b;
        text-decoration: none;
        box-shadow: 0 0 4px 1px rgba(0, 0, 0, 0.25);
    }
    #playVideoLocally section > .section-main > a:hover {
        color: #ffffff;
        fill: #ffffff;
        background: #444654;
    }
    #playVideoLocally .next > svg {
        transform: translateY(1px);
    }
    #playVideoLocally .prev > svg {
        transform: translateY(-1px);
    }
    #playVideoLocally .next:not(.disabled):active, #playVideoLocally .prev:not(.disabled):active {
        transform: scale(.75);
    }
    #playVideoLocally .hidden {
        display: none !important;
    }
    </style>`;

    const inSide = `
    <div class="btn-main">${icons.toggle}</div>
    <section>
        <div class="section-sub">
            <span class="prev">${icons.prev}</span>
            <span class="current"></span>
            <span class="next">${icons.next}</span>
        </div>
        <div class="section-main">
            ${visble('potplayer', 'PotPlayer')}
            ${visble('vlcplayer', 'VLC Player')}
            ${visble('xplayer', 'XPlayer')}
            ${visble('mxplayer', 'MX Player')}
            ${visble('mxplayerpro', 'MX Player Pro')}
            <a class="avid raw filename"></a>
        </div>
        <backdrop></backdrop>
    </section>
    `;

    let mainDiv = elemake("div", inStyle + inSide, {"key":["id"],"val":["playVideoLocally"]});
    document.body.appendChild(mainDiv);
}

// Make it interactive
function inTeract(num) {
    if (num > JSON.parse(window.vidlinks).length - 1) {
        var numX = JSON.parse(window.vidlinks).length - 1;
    } else if (num < 0) {
        var numX = 0;
    } else {
        var numX = num;
    }

    let disLink = JSON.parse(window.vidlinks)[numX],
        disTitle = decodeURIComponent(disLink.replace(/.*\/(.*)$/,'$1'));

    document.querySelector('#playVideoLocally .filename').innerText = disTitle;

    document.querySelector('#playVideoLocally .avid.raw').href = getURL('none', disLink, disTitle);
    document.querySelector('#playVideoLocally .avid.potplayer').href = getURL('potplayer', disLink, disTitle);
    document.querySelector('#playVideoLocally .avid.vlcplayer').href = getURL('vlcplayer', disLink, disTitle);
    document.querySelector('#playVideoLocally .avid.xplayer').href = getURL('xplayer', disLink, disTitle);
    document.querySelector('#playVideoLocally .avid.mxplayer').href = getURL('mxplayer', disLink, disTitle);
    document.querySelector('#playVideoLocally .avid.mxplayerpro').href = getURL('mxplayerpro', disLink, disTitle);

    document.querySelector('#playVideoLocally .prev').classList.remove('disabled','hidden');
    document.querySelector('#playVideoLocally .next').classList.remove('disabled','hidden');

    document.querySelector('#playVideoLocally .current').innerText = numX + 1;
    document.querySelector('#playVideoLocally .current').setAttribute('data-num', numX);

    if (numX === 0) {
        document.querySelector('#playVideoLocally .prev').classList.add('disabled');
    } else if (numX === JSON.parse(window.vidlinks).length - 1) {
        document.querySelector('#playVideoLocally .next').classList.add('disabled');
    }

    if (JSON.parse(window.vidlinks).length === 1) {
        document.querySelector('#playVideoLocally .prev').classList.add('hidden');
        document.querySelector('#playVideoLocally .next').classList.add('hidden');
    }
}

// Event listeners
function inVent() {
    let prev = document.querySelector('#playVideoLocally .prev'),
        next = document.querySelector('#playVideoLocally .next'),
        current = document.querySelector('#playVideoLocally .current'),
        main = document.querySelector('#playVideoLocally .btn-main'),
        section = document.querySelector('#playVideoLocally section'),
        backdrop = document.querySelector('#playVideoLocally backdrop');

    document.addEventListener('click', function(e) {
        if (e.target == prev || prev.contains(e.target)) {
            inTeract(parseInt(current.getAttribute('data-num')) - 1);
        }
        if (e.target == next || next.contains(e.target)) {
            inTeract(parseInt(current.getAttribute('data-num')) + 1);
        }
        if (e.target == main || main.contains(e.target)) {
            try {getVids();} catch {}
            try {inTeract(0);} catch {}
            if (main.classList.contains('clicked')) {
                main.classList.remove('clicked');
                section.classList.remove('clicked');
            } else {
                main.classList.add('clicked');
                section.classList.add('clicked');
            }
        }
        if (e.target == backdrop || backdrop.contains(e.target)) {
            main.click();
        }
    });
}

if (!document.body.classList.contains('jwplayer')) {
    try {getVids();} catch {}
    try {inTerface();} catch {}
    try {inTeract(0);} catch {}
    try {inVent();} catch {}
}