WatchNow IMDb TMDB

Quickly redirect IMDb and TMDB titles to vidbinge.dev with custom settings.

// ==UserScript==
// @name         WatchNow IMDb TMDB
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  Quickly redirect IMDb and TMDB titles to vidbinge.dev with custom settings.
// @author       Bitgineer https://github.com/bitgineer
// @match        https://www.imdb.com/title/tt*/*
// @match        https://www.themoviedb.org/*/*/*/*
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_addStyle
// @run-at       document-end
// @license MIT
// ==/UserScript==

(function() {
    'use strict';

    /*************** Styles Injection ***************/
    const styles = `
    /* WatchNow Styles */
    #WatchNow-container {
        position: fixed;
        bottom: 20px;
        right: 20px;
        width: 350px;
        background-color: #ffffff;
        border: 1px solid #cccccc;
        border-radius: 8px;
        box-shadow: 0 4px 8px rgba(0,0,0,0.2);
        z-index: 10000;
        font-family: 'Roboto', sans-serif;
        padding: 20px;
        box-sizing: border-box;
    }

    #WatchNow-container h2 {
        font-size: 1.4rem;
        margin-bottom: 20px;
        text-align: center;
        color: #6200ea;
    }

    .wn-button {
        display: flex;
        align-items: center;
        justify-content: center;
        padding: 12px 16px;
        font-size: 1rem;
        background-color: #6200ea;
        color: #ffffff;
        border: none;
        border-radius: 4px;
        cursor: pointer;
        transition: background-color 0.3s, box-shadow 0.3s;
        width: 100%;
        margin-bottom: 15px;
        box-shadow: 0 2px 4px rgba(0, 0, 0, 0.2);
    }

    .wn-button:hover {
        background-color: #3700b3;
        box-shadow: 0 4px 8px rgba(0, 0, 0, 0.3);
    }

    .wn-toggle-container {
        display: flex;
        align-items: center;
        justify-content: space-between;
        margin-bottom: 15px;
    }

    .wn-toggle-label {
        font-size: 1rem;
        flex: 1;
        margin-right: 10px;
    }

    .wn-toggle-switch {
        position: relative;
        width: 50px;
        height: 24px;
        flex-shrink: 0;
    }

    .wn-toggle-switch input {
        opacity: 0;
        width: 0;
        height: 0;
    }

    .wn-slider {
        position: absolute;
        cursor: pointer;
        top: 0;
        left: 0;
        right: 0;
        bottom: 0;
        background-color: #ccc;
        transition: 0.4s;
        border-radius: 24px;
    }

    .wn-slider:before {
        position: absolute;
        content: "";
        height: 18px;
        width: 18px;
        left: 3px;
        bottom: 3px;
        background-color: white;
        transition: 0.4s;
        border-radius: 50%;
    }

    .wn-toggle-switch input:checked + .wn-slider {
        background-color: #6200ea;
    }

    .wn-toggle-switch input:checked + .wn-slider:before {
        transform: translateX(26px);
    }

    .wn-section {
        margin-bottom: 15px;
        display: none;
    }

    .wn-section.active {
        display: block;
    }

    .wn-section label {
        display: block;
        margin-bottom: 8px;
        font-size: 0.95rem;
    }

    .wn-radio-group {
        display: flex;
        justify-content: space-around;
        margin-bottom: 15px;
        flex-wrap: wrap;
    }

    .wn-radio-group label {
        display: flex;
        align-items: center;
        font-size: 0.95rem;
        cursor: pointer;
        margin-bottom: 8px;
        width: 45%;
    }

    .wn-radio-group input {
        margin-right: 6px;
        flex-shrink: 0;
    }

    .wn-input-group {
        display: flex;
        flex-direction: column;
        gap: 10px;
    }

    .wn-input-group label {
        font-size: 0.95rem;
    }

    .wn-input-group input {
        padding: 8px 12px;
        font-size: 1rem;
        border: 1px solid #cccccc;
        border-radius: 4px;
        outline: none;
        transition: border-color 0.3s;
        width: 100%;
    }

    .wn-input-group input:focus {
        border-color: #6200ea;
    }

    .wn-footer {
        font-size: 0.85rem;
        color: #777777;
        text-align: center;
    }

    /* Snackbar Styles */
    #wn-snackbar {
        position: fixed;
        bottom: 20px;
        left: 50%;
        transform: translateX(-50%);
        background-color: #333333;
        color: #ffffff;
        padding: 10px 20px;
        border-radius: 4px;
        opacity: 0;
        transition: opacity 0.5s ease, bottom 0.5s ease;
        z-index: 10001;
        font-size: 0.9rem;
        max-width: 80%;
        text-align: center;
        pointer-events: none;
    }

    #wn-snackbar.show {
        opacity: 1;
        bottom: 30px;
    }

    /* Responsive Design */
    @media (max-width: 400px) {
        #WatchNow-container {
            width: 90%;
            right: 5%;
            bottom: 10px;
            padding: 15px;
        }

        #WatchNow-container h2 {
            font-size: 1.2rem;
        }

        .wn-button {
            font-size: 0.95rem;
            padding: 10px 12px;
        }

        .wn-toggle-label, .wn-section label, .wn-radio-group label, .wn-input-group label {
            font-size: 0.85rem;
        }
    }
    `;

    GM_addStyle(styles);

    /*************** UI Injection ***************/
    const container = document.createElement('div');
    container.id = 'WatchNow-container';
    container.innerHTML = `
        <h2>WatchNow</h2>

        <button id="wn-redirectButton" class="wn-button">
            ▶️ Watch Now!
        </button>

        <div class="wn-toggle-container">
            <span class="wn-toggle-label">Enable Redirect</span>
            <label class="wn-toggle-switch">
                <input type="checkbox" id="wn-redirectCheckbox">
                <span class="wn-slider"></span>
            </label>
        </div>

        <!-- Redirect Options Section -->
        <div id="wn-redirectOptions" class="wn-section">
            <label>Open Redirect:</label>
            <div class="wn-radio-group">
                <label>
                    <input type="radio" name="wn-redirectTarget" value="same" id="wn-redirectSame">
                    Same Tab
                </label>
                <label>
                    <input type="radio" name="wn-redirectTarget" value="new" id="wn-redirectNew">
                    New Tab
                </label>
            </div>
        </div>

        <!-- Season and Episode Inputs -->
        <div id="wn-seasonEpisodeOptions" class="wn-section">
            <div class="wn-input-group">
                <label for="wn-seasonInput">Season:</label>
                <input type="number" id="wn-seasonInput" min="1" value="1">
            </div>
            <div class="wn-input-group">
                <label for="wn-episodeInput">Episode:</label>
                <input type="number" id="wn-episodeInput" min="1" value="1">
            </div>
        </div>

        <div class="wn-footer">
            <p>Stream effortlessly with WatchNow.</p>
        </div>
    `;
    document.body.appendChild(container);

    // Snackbar Element
    const snackbar = document.createElement('div');
    snackbar.id = 'wn-snackbar';
    document.body.appendChild(snackbar);

    /*************** Utility Functions ***************/
    // Function to show snackbar messages
    function showSnackbar(message) {
        snackbar.textContent = message;
        snackbar.classList.add('show');
        setTimeout(() => {
            snackbar.classList.remove('show');
        }, 3000);
    }

    // Function to toggle visibility of sections
    function toggleSections(show) {
        const redirectOptions = document.getElementById('wn-redirectOptions');
        const seasonEpisodeOptions = document.getElementById('wn-seasonEpisodeOptions');
        if (show) {
            redirectOptions.classList.add('active');
            seasonEpisodeOptions.classList.add('active');
        } else {
            redirectOptions.classList.remove('active');
            seasonEpisodeOptions.classList.remove('active');
        }
    }

    // Function to extract IMDb ID using a robust regex
    function getImdbId(url) {
        const regex = /\/title\/(tt\d{7,8})/; // IMDb IDs typically have 7 or 8 digits
        const match = url.match(regex);
        return match ? match[1] : null;
    }

    // Function to determine the title type on IMDb by inspecting specific list items
    function determineImdbTitleType() {
        const listItems = document.querySelectorAll('ul.ipc-inline-list.ipc-inline-list--show-dividers li.ipc-inline-list__item');
        for (const item of listItems) {
            const text = item.textContent.trim().toLowerCase();
            if (text.includes('tv series') || text.includes('tv mini-series')) {
                return 'tv';
            }
            // You can add more conditions here for other types if needed
        }
        return 'movie'; // Default to 'movie' if no TV indicators are found
    }

    // Function to extract TMDB ID and type from the URL
    function getTmdbInfo(url) {
        const regex = /https:\/\/www\.themoviedb\.org\/(movie|tv)\/(\d+)/;
        const match = url.match(regex);
        if (match) {
            return { type: match[1], id: match[2] };
        }
        return null;
    }

    // Function to build the redirect URL with optional season and episode for TV
    function buildRedirectUrl(titleType, id, season, episode) {
        let url = `https://vidbinge.dev/embed/${titleType}/${id}`;
        if (titleType === 'tv' && season && episode) {
            url += `/${season}/${episode}`;
        }
        return url;
    }

    /*************** Settings Management ***************/
    // Load existing settings
    let redirectEnabled = GM_getValue('redirectEnabled', false);
    let redirectTarget = GM_getValue('redirectTarget', 'same');
    let season = GM_getValue('season', 1);
    let episode = GM_getValue('episode', 1);

    // Initialize UI with settings
    const redirectCheckbox = document.getElementById('wn-redirectCheckbox');
    const redirectSame = document.getElementById('wn-redirectSame');
    const redirectNew = document.getElementById('wn-redirectNew');
    const seasonInput = document.getElementById('wn-seasonInput');
    const episodeInput = document.getElementById('wn-episodeInput');

    redirectCheckbox.checked = redirectEnabled;
    toggleSections(redirectEnabled);

    if (redirectTarget === 'same') {
        redirectSame.checked = true;
    } else {
        redirectNew.checked = true;
    }

    seasonInput.value = season;
    episodeInput.value = episode;

    /*************** Event Listeners ***************/
    // Event listener for the redirect checkbox
    redirectCheckbox.addEventListener('change', function() {
        const isEnabled = redirectCheckbox.checked;
        GM_setValue('redirectEnabled', isEnabled);
        toggleSections(isEnabled);
    });

    // Event listeners for radio buttons
    redirectSame.addEventListener('change', function() {
        if (this.checked) {
            GM_setValue('redirectTarget', 'same');
        }
    });

    redirectNew.addEventListener('change', function() {
        if (this.checked) {
            GM_setValue('redirectTarget', 'new');
        }
    });

    // Event listeners for Season and Episode inputs
    seasonInput.addEventListener('input', function() {
        let val = parseInt(seasonInput.value, 10);
        if (isNaN(val) || val < 1) {
            val = 1;
            seasonInput.value = val;
        }
        GM_setValue('season', val);
    });

    episodeInput.addEventListener('input', function() {
        let val = parseInt(episodeInput.value, 10);
        if (isNaN(val) || val < 1) {
            val = 1;
            episodeInput.value = val;
        }
        GM_setValue('episode', val);
    });

    // Event listener for the redirect button
    document.getElementById('wn-redirectButton').addEventListener('click', function() {
        const url = window.location.href;

        // Check if URL is IMDb
        const imdbId = getImdbId(url);
        if (imdbId) {
            const titleType = determineImdbTitleType();

            if (titleType === 'tv') {
                const currentSeason = GM_getValue('season', 1);
                const currentEpisode = GM_getValue('episode', 1);
                const redirectUrl = buildRedirectUrl(titleType, imdbId, currentSeason, currentEpisode);
                performRedirect(redirectUrl);
            } else {
                // For movies, no season and episode
                const redirectUrl = buildRedirectUrl(titleType, imdbId);
                performRedirect(redirectUrl);
            }
            return;
        }

        // Check if URL is TMDB
        const tmdbInfo = getTmdbInfo(url);
        if (tmdbInfo) {
            const { type, id } = tmdbInfo;
            if (type === 'tv') {
                const currentSeason = GM_getValue('season', 1);
                const currentEpisode = GM_getValue('episode', 1);
                const redirectUrl = buildRedirectUrl(type, id, currentSeason, currentEpisode);
                performRedirect(redirectUrl);
            } else {
                // For movies, no season and episode
                const redirectUrl = buildRedirectUrl(type, id);
                performRedirect(redirectUrl);
            }
            return;
        }

        showSnackbar('This is not a supported title page.');
    });

    /*************** Redirection Function ***************/
    function performRedirect(url) {
        if (!GM_getValue('redirectEnabled', false)) {
            showSnackbar('Redirect is disabled in settings.');
            return;
        }

        const target = GM_getValue('redirectTarget', 'same');
        if (target === 'same') {
            window.location.href = url;
            showSnackbar('Redirecting...');
        } else if (target === 'new') {
            window.open(url, '_blank');
            showSnackbar('Redirecting to new tab...');
        } else {
            showSnackbar('Invalid redirect target.');
        }
    }

    /*************** Optional: Auto Redirect ***************/
    // Uncomment the following block if you want to auto redirect when enabled
    /*
    if (redirectEnabled) {
        const url = window.location.href;

        // Check if URL is IMDb
        const imdbId = getImdbId(url);
        if (imdbId) {
            const titleType = determineImdbTitleType();

            if (titleType === 'tv') {
                const currentSeason = GM_getValue('season', 1);
                const currentEpisode = GM_getValue('episode', 1);
                const redirectUrl = buildRedirectUrl(titleType, imdbId, currentSeason, currentEpisode);
                performRedirect(redirectUrl);
            } else {
                // For movies, no season and episode
                const redirectUrl = buildRedirectUrl(titleType, imdbId);
                performRedirect(redirectUrl);
            }
            return;
        }

        // Check if URL is TMDB
        const tmdbInfo = getTmdbInfo(url);
        if (tmdbInfo) {
            const { type, id } = tmdbInfo;
            if (type === 'tv') {
                const currentSeason = GM_getValue('season', 1);
                const currentEpisode = GM_getValue('episode', 1);
                const redirectUrl = buildRedirectUrl(type, id, currentSeason, currentEpisode);
                performRedirect(redirectUrl);
            } else {
                // For movies, no season and episode
                const redirectUrl = buildRedirectUrl(type, id);
                performRedirect(redirectUrl);
            }
            return;
        }

        console.error('This is not a valid IMDb or TMDB title page.');
    }
    */

})();