thisvid infinite scroll and filter

[s]try to take over the world![/s] infinite scroll, filter: public, private, duration

As of 2024-01-26. See the latest version.

You will need to install an extension such as Tampermonkey, Greasemonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Violentmonkey to install this script.

You will need to install an extension such as Tampermonkey or Userscripts to install this script.

You will need to install an extension such as Tampermonkey to install this script.

You will need to install a user script manager extension to install this script.

(I already have a user script manager, let me install it!)

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install an extension such as Stylus to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

You will need to install a user style manager extension to install this style.

(I already have a user style manager, let me install it!)

// ==UserScript==
// @name         thisvid infinite scroll and filter
// @license      MIT
// @namespace    http://tampermonkey.net/
// @version      2024-01-18
// @description  [s]try to take over the world![/s] infinite scroll, filter: public, private, duration
// @author       homo
// @match        https://thisvid.com/
// @match        https://thisvid.com/latest-updates/*
// @match        https://thisvid.com/tags/*
// @match        https://thisvid.com/categories/*
// @match        https://thisvid.com/*/?q=*
// @match        https://thisvid.com/members/*
// @exclude      https://thisvid.com/videos/*
// @icon         
// @grant        GM_addStyle
// ==/UserScript==
(function () {
    'use strict';

    console.clear();
    console.log('\n\ntapermonkey-thisvidscript\n\n');

    class Utils {
        static $ = (x, e = document.body) => e.querySelector(x);
        static $$ = (x, e = document.body) => e.querySelectorAll(x);

        static findElementsBetweenIds = (startId, endId) => {
            const startElement = document.getElementById(startId);
            const endElement = document.getElementById(endId);

            if (!startElement || !endElement) {
                return [];
            }

            const elementsBetween = [];
            let currentElement = startElement.nextElementSibling;

            while (currentElement && currentElement !== endElement) {
                elementsBetween.push(currentElement);
                currentElement = currentElement.nextElementSibling;
            }

            return elementsBetween;
        };

        static parseHTML = (str) => {
            var temp = document.createElement('html');
            temp.innerHTML = str;
            return temp;
        };

        static fetchHtml = async (url) =>
        fetch(url)
        .then((r) => r.text())
        .then((d) => parseHTML(d));

        static getRandomRgb = () => {
            const n = Math.round(0xffffff * Math.random());
            return `rgb(${n >> 16},${(n >> 8) & 255},${n & 255})`;
        };

        static parseCSSUrl = (s) => s.match(/(?<=\(").*(?="\))/)[0];

        static timeToSeconds = (t) =>
        t
        .match(/\d+/gm)
        .reverse()
        .map((s, i) => parseInt(s) * Math.pow(60, i))
        .reduce((a, b) => a + b);

        static circularShift = (n, c = 6, s = 1) => (n + s) % c || c;

        static isElementInViewport = (el) => {
            var rect = el.getBoundingClientRect();
            return (
                rect.top >= 0 &&
                rect.left >= 0 &&
                rect.bottom <=
                (window.innerHeight ||
                 document.documentElement
                 .clientHeight) /* or $(window).height() */ &&
                rect.right <=
                (window.innerWidth ||
                 document.documentElement.clientWidth) /* or $(window).width() */
            );
        };

        static toggleClass = (element, className, condition) => {
            if (condition) {
                element.classList.add(className);
            } else {
                element.classList.remove(className);
            }
        };

        static parseDom = (html) => {
            return new DOMParser().parseFromString(html, 'text/html').body.firstChild;
        }
    }

    const {
        $,
        $$,
        findElementsBetweenIds,
        fetchHtml,
        parseHTML,
        parseCSSUrl,
        circularShift,
        isElementInViewport,
        timeToSeconds,
        getRandomRgb,
        toggleClass,
        parseDom
    } = Utils;

    class Tick {
        constructor(interval) {
            this.interval = interval;
        }

        start(callback) {
            if (this.ticker) {
                this.stop();
            }
            callback();
            this.ticker = setInterval(callback, this.interval);
        }

        stop() {
            clearInterval(this.ticker);
            this.ticker = false;
        }
    }

    class QueueDuplicatesFilter {
        constructor(queueLimit = 120) {
            this.queueLimit = queueLimit;
            this.queue = [];
        }

        put(item) {
            const isUnique = !this.queue.includes(item);
            if (isUnique) {
                if (this.queue.length > this.queueLimit) {
                    this.queue.shift();
                }
                this.queue.push(item);
            }
            return isUnique;
        }
    }

    class ReactiveLocalStorage {
        constructor(data) {
            if (data) {
                Object.assign(this, data);
                this.observeProps(this);
            }
        }

        getLS(prop) {
            const val = localStorage.getItem(prop);
            if (val !== null) {
                return JSON.parse(val);
            } else {
                return null;
            }
        }

        setLS(prop, value) {
            localStorage.setItem(prop, JSON.stringify(value));
        }

        toObservable(obj, prop) {
            var value = this.getLS(prop) || obj[prop];

            Object.defineProperty(obj, prop, {
                get() {
                    return value;
                },
                set(newValue) {
                    this.setLS(prop, newValue);
                    value = newValue;
                },
            });
        }

        observeProps(obj) {
            for (const [key, _] of Object.entries(obj)) {
                this.toObservable(obj, key);
            }
        }
    }

    const SCROLL_RESET_DELAY = 500;
    const ANIMATION_DELAY = 750;

    class DomManager {
        constructor() {
            GM_addStyle(`
            .filtered-private, .filtered-duration, .filtered-public
            { display: none !important; } `);
        }

        static thumbIsPrivate(t) {
            return t.firstElementChild.classList.contains('private');
        }

        static filterPrivate(container = document.body) {
            const tumbs = $$('.tumbpu', container);
            for (const t of tumbs) {
                toggleClass(t, 'filtered-private',
                            state.filterPrivate && DomManager.thumbIsPrivate(t));
            }
        }

        static filterPublic(container = document.body) {
            const tumbs = $$('.tumbpu', container);
            for (const t of tumbs) {
                toggleClass(t, 'filtered-public',
                            state.filterPublic && !DomManager.thumbIsPrivate(t));
            }
        }

        static filterByDuration(container = document.body) {
            const elements = $$('.duration', container);
            const {
                filterDurationFrom: from,
                filterDurationTo: to,
                filterDuration,
            } = state;
            for (const t of elements) {
                const te = t.parentElement.parentElement;
                let flag = false;
                if (filterDuration) {
                    const ts = timeToSeconds(t.innerText);
                    flag = ts < from || ts > to;
                }
                toggleClass(te, 'filtered-duration', filterDuration && flag);
            }
        }

        static createThumbsContainer() {
            const c = document.createElement('div');
            c.className = 'thumbs-items';
            return c;
        }

        static handleLoadedHTML = (htmlPage, mount, useStateContainer = true) => {
            const thumbs = $$('.tumbpu', htmlPage);

            let container = !useStateContainer
            ? DomManager.createThumbsContainer()
            : (state.container =
               state.container || DomManager.createThumbsContainer());

            for (const thumbElement of thumbs) {
                if (!uniquesQueue.put(thumbElement.getAttribute('href'))) {
                    thumbElement.remove();
                } else {
                    const privateEl = $('.private', thumbElement);
                    const img = $('img', thumbElement);
                    if (privateEl) {
                        img.src = parseCSSUrl(privateEl.style.background);
                        privateEl.style.background = '#000';
                    } else {
                        img.src = img.getAttribute('data-original');
                    }
                    img.classList.add('tracking');
                    container.appendChild(thumbElement);
                }
            }

            if (state.filterPrivate) filterPrivate(container);
            if (state.filterPublic) filterPublic(container);
            if (state.filterDuration) filterByDuration(container);
            mount.before(container);
        };
    }

    class PreviewManager {
        constructor() {
            this.tick = new Tick(ANIMATION_DELAY);
        }

        iteratePreviewImages(src) {
            return src.replace(/(\d)\.jpg$/, (_, num) => circularShift(parseInt(num)) + '.jpg');
        }

        animatePreview = (e) => {
            const { target: el, type } = e;
            if (el.tagName === 'IMG' && el.classList.contains('tracking')) {
                if (type === 'mouseout') {
                    this.tick.stop();
                    if (el.getAttribute('orig')) el.src = el.getAttribute('orig');
                }
                if (type === 'mouseover') {
                    this.tick.start(() => {
                        if (!el.getAttribute('orig')) el.setAttribute('orig', el.src);
                        el.src = this.iteratePreviewImages(el.src);
                    });
                }
            }
        };

        listen(e) {
            e.addEventListener('mouseout', this.animatePreview);
            e.addEventListener('mouseover', this.animatePreview);
        }
    }

    class PaginationPageManager {
        constructor() {
            this.pagination = $('.pagination');
            handleLoadedHTML(document.body, this.pagination);
            previewManager.listen(this.pagination.parentElement);
            this.paginationGenerator = this.createNextPageGenerator();
            this.generatorDone = false;
            this.infiniteScrollTriggered = false;
            this.tick = new Tick(SCROLL_RESET_DELAY);
            this.fixScrollViewPort();
        }

        fixScrollViewPort(){
            this.tick.start(() => {
                if (this.generatorDone) this.tick.stop();
                if (isElementInViewport(this.pagination)) this.generatorConsumer();
            });
        }

        async generatorConsumer() {
            const { value, done } = this.paginationGenerator.next();
            this.generatorDone = done;
            if (!done) {
                const nextPageHTML = await fetchHtml(value);
                const prevScrollPos = document.documentElement.scrollTop;
                handleLoadedHTML(nextPageHTML, this.pagination);
                window.scrollTo(0, prevScrollPos);
            }
        }

        createNextPageGenerator() {
            const pl = $('.pagination-next').previousElementSibling.firstElementChild.textContent;
            const offsetLast = parseInt(pl);

            let {origin, pathname, search } = window.location;
            let offset;
            [pathname, offset='1'] = pathname.split(/(\d+\/)$/);
            offset = parseInt(offset);
            pathname = pathname === '/' ? '/latest-updates/' : pathname;

            function* nextPageGenerator() {
                for (let c = offset + 1; c <= offsetLast; c++) {
                    const url = `${origin}${pathname}${c}/${search}`;
                    console.log(url);
                    yield url;
                }
            }

            return nextPageGenerator();
        }

        resetScroller = () => {
            this.infiniteScrollTriggered = false;
            window.dispatchEvent(new CustomEvent('scroll'));
        };

        infiniteScroll = () => {
            const inViewport = isElementInViewport(this.pagination);
            if (inViewport === this.infiniteScrollTriggered) return;
            this.infiniteScrollTriggered = inViewport;
            if (inViewport) this.generatorConsumer();
        };
    }

    class Router {
        constructor() {
            this.route();
        }

        route() {
            const { pathname } = window.location;
            if ($('.pagination-next') instanceof HTMLElement) {
                this.handlePaginationPage();
            } else if (/\/members\/\d+\/$/.test(pathname)) {
                this.handleMemberPage();
            } else if (/\/tag\//.test(pathname) || /\/?q=.*/.test(pathname)) {
                this.handlePageWithVideosButNoPagination();
            }
        }

        handlePageWithVideosButNoPagination() {
            const vid = $('.tumbpu');
            if (!vid) return;
            handleLoadedHTML(document.body, vid.parentElement);
            previewManager.listen(vid.parentElement);
        }

        handlePaginationPage() {
            this.paginationManager = new PaginationPageManager();
        }

        handleMemberPage() {
            const privates = $('#list_videos_private_videos_items');
            if (privates) {
                const mistakes = findElementsBetweenIds(
                    'list_videos_private_videos_items',
                    'list_videos_favourite_videos'
                );
                mistakes.forEach((m) => privates.appendChild(m));
                handleLoadedHTML(privates, privates, false);
            }

            const favorites = $('#list_videos_favourite_videos');
            if (favorites) {
                const mountTo = favorites.firstElementChild.nextElementSibling;
                handleLoadedHTML(favorites, mountTo, false);
            }

            if (privates || favorites) {
                previewManager.listen((privates || favorites).parentElement);
            }
        }
    }

    class UI {
        templateHTML = `
<div id="tapermonkey-app">
  <div class="subbox">
    <input type="checkbox" id="filterPrivate" name="filterPrivate" ${
      state.filterPrivate ? 'checked' : ''
    }/>
    <label for="filterPrivate">filter private</label>
    <input type="checkbox" id="filterPublic" name="filterPublic" ${
      state.filterPublic ? 'checked' : ''
    }/>
    <label for="filterPublic">filter public</label>
  </div>
  <div class="subbox">
    <input type="checkbox" id="filterl" name="filterl" ${
      state.filterDuration ? 'checked' : ''
    } />
    <label for="filterl">filter duration seconds</label>

    <input type="number" placeholder="min sec" step="10"
      min="0" max="100000" id="minL" value=${state.filterDurationFrom} />
    <input type="number" placeholder="max sec" step="10"
      min="0" max="100000" id="maxL" value=${state.filterDurationTo} />
  </div>
</div>`;

        templateCSS = `
    #tapermonkey-app {
      background: #151515;
      color: #fff;
      padding: 10px;
      position: fixed;
      z-index: 9999;
      bottom: 10px;
      right: 10px;
      border-radius: 15px;
      width: max-content;
      box-shadow: 20px 20px 60px #000000, -20px -20px 60px #000000;
    }
    #tapermonkey-app .subbox {
      background: #2c2c2c;
      border-radius: 5px;
      padding: 4px;
      margin: 6px;
    }
    #tapermonkey-app .subbox input[type=number] {
      padding-left: 10px;
      width: 5rem;
      background: #26282b;
    }
    #tapermonkey-app .subbox input[type=checkbox] {
      margin-left: 5px;
    }
    #tapermonkey-app .subbox label {
      user-select: none;
    }
    #tapermonkey-app .subbox input,
    #tapermonkey-app .subbox label {
      padding-left: 8px;
      float: none;
      width: auto;
      font-family: monospace;
      font-size: 0.8rem;
    }`;

        constructor() {
            document.body.appendChild(parseDom(this.templateHTML));
            this.tapermonkeyAppTemplate = document.querySelector('#tapermonkey-app');
            GM_addStyle(this.templateCSS);
            this.control();
        }

        control() {
            this.tapermonkeyAppTemplate.addEventListener('click', (e) => {
                if (e.target.id === 'filterPublic') {
                    state.filterPublic = e.target.checked;
                    filterPublic();
                }
                if (e.target.id === 'filterPrivate') {
                    state.filterPrivate = e.target.checked;
                    filterPrivate();
                }
                if (e.target.id === 'filterl') {
                    state.filterDuration = e.target.checked;
                    filterByDuration();
                }
                if (e.target.id === 'minL') {
                    state.filterDurationFrom =
                        parseInt(e.target.value) || state.filterDurationFrom;
                    filterByDuration();
                }
                if (e.target.id === 'maxL') {
                    state.filterDurationTo =
                        parseInt(e.target.value) || state.filterDurationTo;
                    filterByDuration();
                }
            });
        }
    }

    const state = new ReactiveLocalStorage({
        filterDurationFrom: 600,
        filterDurationTo: 100000,
        filterDuration: false,
        filterPrivate: false,
        filterPublic: false,
        infiniteScrollTriggered: false,
    });
    const uniquesQueue = new QueueDuplicatesFilter();
    const domManager = new DomManager();
    const { filterPrivate, filterPublic, filterByDuration, handleLoadedHTML } = DomManager;
    const previewManager = new PreviewManager();
    const router = new Router();
    const ui = new UI();
})();