Grok Quota Panel

Displays remaining queries / rate limits on grok.com (updated for remainingQueries)

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey, Greasemonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да инсталирате разширение, като например Tampermonkey .

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Violentmonkey.

За да инсталирате този скрипт, трябва да имате инсталирано разширение като Tampermonkey или Userscripts.

За да инсталирате скрипта, трябва да инсталирате разширение като Tampermonkey.

За да инсталирате този скрипт, трябва да имате инсталиран скриптов мениджър.

(Вече имам скриптов мениджър, искам да го инсталирам!)

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да инсталирате разширение като Stylus.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

За да инсталирате този стил, трябва да имате инсталиран мениджър на потребителски стилове.

(Вече имам инсталиран мениджър на стиловете, искам да го инсталирам!)

// ==UserScript==
// @name         Grok Quota Panel
// @namespace    aspen138
// @version      2026.04.18.2112
// @description  Displays remaining queries / rate limits on grok.com (updated for remainingQueries)
// @author       aspen138
// @match        https://grok.com/*
// @icon         https://grok.com/images/favicon-light.png
// @run-at       document-end
// @grant        none
// @license      MIT
// @icon         data:image/svg;base64,<svg width="512" height="512" viewBox="0 0 512 512" fill="none" xmlns="http://www.w3.org/2000/svg">
<g clip-path="url(#clip0_304_111)">
<foreignObject x="-24" y="-22" width="560" height="560"><div xmlns="http://www.w3.org/1999/xhtml" style="backdrop-filter:blur(16px);clip-path:url(#bgblur_1_304_111_clip_path);height:100%;width:100%"></div></foreignObject><g filter="url(#filter0_d_304_111)" data-figma-bg-blur-radius="32">
<path d="M0 256C0 166.392 0 121.587 17.439 87.3615C32.7787 57.2556 57.2556 32.7787 87.3615 17.439C121.587 0 166.392 0 256 0C345.608 0 390.413 0 424.638 17.439C454.744 32.7787 479.221 57.2556 494.561 87.3615C512 121.587 512 166.392 512 256C512 345.608 512 390.413 494.561 424.638C479.221 454.744 454.744 479.221 424.638 494.561C390.413 512 345.608 512 256 512C166.392 512 121.587 512 87.3615 494.561C57.2556 479.221 32.7787 454.744 17.439 424.638C0 390.413 0 345.608 0 256Z" fill="#050505"/>
<path d="M256 2.5C300.845 2.5 334.329 2.50228 360.884 4.67188C387.41 6.83911 406.802 11.156 423.504 19.666C453.139 34.7661 477.234 58.8607 492.334 88.4961C500.844 105.198 505.161 124.59 507.328 151.116C509.498 177.671 509.5 211.155 509.5 256C509.5 300.845 509.498 334.329 507.328 360.884C505.161 387.41 500.844 406.802 492.334 423.504C477.234 453.139 453.139 477.234 423.504 492.334C406.802 500.844 387.41 505.161 360.884 507.328C334.329 509.498 300.845 509.5 256 509.5C211.155 509.5 177.671 509.498 151.116 507.328C124.59 505.161 105.198 500.844 88.4961 492.334C58.8607 477.234 34.7661 453.139 19.666 423.504C11.156 406.802 6.83911 387.41 4.67188 360.884C2.50228 334.329 2.5 300.845 2.5 256C2.5 211.155 2.50228 177.671 4.67188 151.116C6.83911 124.59 11.156 105.198 19.666 88.4961C34.7661 58.8607 58.8607 34.7661 88.4961 19.666C105.198 11.156 124.59 6.83911 151.116 4.67188C177.671 2.50228 211.155 2.5 256 2.5Z" stroke="black" stroke-opacity="0.05" stroke-width="5"/>
</g>
</g>
<mask id="path-3-inside-1_304_111" fill="white">
<path d="M210.484 312.759L343.465 210.383C349.984 205.364 359.302 207.322 362.408 215.117C378.758 256.231 371.454 305.64 338.925 339.563C306.397 373.487 261.137 380.927 219.768 363.983L174.577 385.803C239.394 432.008 318.104 420.581 367.289 369.251C406.303 328.564 418.386 273.104 407.088 223.091L407.19 223.198C390.807 149.726 411.218 120.359 453.03 60.3072C454.02 58.8833 455.01 57.4595 456 56L400.978 113.382V113.204L210.45 312.794"/>
<path d="M183.042 337.641C136.519 291.294 144.54 219.567 184.236 178.203C213.59 147.59 261.683 135.096 303.666 153.464L348.755 131.75C340.632 125.627 330.221 119.042 318.275 114.414C264.277 91.2407 199.63 102.774 155.735 148.516C113.513 192.549 100.236 260.254 123.036 318.027C140.069 361.206 112.148 391.748 84.0229 422.575C74.0561 433.503 64.0553 444.431 56 456L183.007 337.677"/>
</mask>
<path d="M210.484 312.759L343.465 210.383C349.984 205.364 359.302 207.322 362.408 215.117C378.758 256.231 371.454 305.64 338.925 339.563C306.397 373.487 261.137 380.927 219.768 363.983L174.577 385.803C239.394 432.008 318.104 420.581 367.289 369.251C406.303 328.564 418.386 273.104 407.088 223.091L407.19 223.198C390.807 149.726 411.218 120.359 453.03 60.3072C454.02 58.8833 455.01 57.4595 456 56L400.978 113.382V113.204L210.45 312.794" fill="#FCFCFC"/>
<path d="M183.042 337.641C136.519 291.294 144.54 219.567 184.236 178.203C213.59 147.59 261.683 135.096 303.666 153.464L348.755 131.75C340.632 125.627 330.221 119.042 318.275 114.414C264.277 91.2407 199.63 102.774 155.735 148.516C113.513 192.549 100.236 260.254 123.036 318.027C140.069 361.206 112.148 391.748 84.0229 422.575C74.0561 433.503 64.0553 444.431 56 456L183.007 337.677" fill="#FCFCFC"/>
<path d="M343.465 210.383L344.38 211.571L344.38 211.571L343.465 210.383ZM362.408 215.117L363.802 214.563L363.802 214.562L362.408 215.117ZM219.768 363.983L220.337 362.594L219.718 362.341L219.116 362.632L219.768 363.983ZM174.577 385.803L173.924 384.453L171.643 385.554L173.706 387.025L174.577 385.803ZM367.289 369.251L366.207 368.213L366.206 368.213L367.289 369.251ZM407.088 223.091L408.171 222.053L404.436 218.157L405.625 223.421L407.088 223.091ZM407.19 223.198L406.108 224.236L409.823 228.111L408.654 222.871L407.19 223.198ZM453.03 60.3072L454.262 61.1643L454.262 61.1634L453.03 60.3072ZM456 56L457.241 56.842L454.917 54.9618L456 56ZM400.978 113.382H399.478V117.113L402.061 114.42L400.978 113.382ZM400.978 113.204H402.478V109.46L399.893 112.168L400.978 113.204ZM184.236 178.203L185.318 179.242L185.319 179.242L184.236 178.203ZM303.666 153.464L303.065 154.838L303.696 155.114L304.317 154.815L303.666 153.464ZM348.755 131.75L349.406 133.101L351.624 132.033L349.658 130.552L348.755 131.75ZM318.275 114.414L317.683 115.793L317.708 115.803L317.733 115.813L318.275 114.414ZM155.735 148.516L154.653 147.477L154.653 147.478L155.735 148.516ZM123.036 318.027L124.432 317.477L124.432 317.477L123.036 318.027ZM84.0229 422.575L82.9148 421.564L82.9146 421.564L84.0229 422.575ZM56 456L54.769 455.143L57.0225 457.098L56 456ZM210.484 312.759L211.399 313.948L344.38 211.571L343.465 210.383L342.55 209.194L209.569 311.57L210.484 312.759ZM343.465 210.383L344.38 211.571C350.204 207.087 358.32 208.909 361.015 215.672L362.408 215.117L363.802 214.562C360.285 205.734 349.764 203.64 342.55 209.194L343.465 210.383ZM362.408 215.117L361.015 215.671C377.163 256.279 369.936 305.055 337.842 338.525L338.925 339.563L340.008 340.601C372.972 306.224 380.353 256.183 363.802 214.563L362.408 215.117ZM338.925 339.563L337.842 338.525C305.769 371.974 261.175 379.321 220.337 362.594L219.768 363.983L219.2 365.371C261.099 382.532 307.024 375 340.008 340.601L338.925 339.563ZM219.768 363.983L219.116 362.632L173.924 384.453L174.577 385.803L175.229 387.154L220.42 365.333L219.768 363.983ZM174.577 385.803L173.706 387.025C239.15 433.675 318.689 422.14 368.372 370.289L367.289 369.251L366.206 368.213C317.52 419.023 239.639 430.34 175.447 384.582L174.577 385.803ZM367.289 369.251L368.372 370.289C407.764 329.208 419.95 273.221 408.551 222.76L407.088 223.091L405.625 223.421C416.822 272.987 404.842 327.92 366.207 368.213L367.289 369.251ZM407.088 223.091L406.005 224.129L406.108 224.236L407.19 223.198L408.273 222.16L408.171 222.053L407.088 223.091ZM407.19 223.198L408.654 222.871C400.505 186.325 401.538 160.91 409.705 137.209C417.905 113.414 433.319 91.2421 454.262 61.1643L453.03 60.3072L451.799 59.4501C430.929 89.4239 415.232 111.961 406.869 136.231C398.473 160.594 397.492 186.599 405.726 223.524L407.19 223.198ZM453.03 60.3072L454.262 61.1634C455.251 59.7406 456.246 58.3097 457.241 56.842L456 56L454.759 55.158C453.774 56.6092 452.789 58.0261 451.799 59.451L453.03 60.3072ZM456 56L454.917 54.9618L399.896 112.344L400.978 113.382L402.061 114.42L457.083 57.0382L456 56ZM400.978 113.382H402.478V113.204H400.978H399.478V113.382H400.978ZM183.042 337.641L184.1 336.578C138.255 290.907 146.087 220.121 185.318 179.242L184.236 178.203L183.154 177.165C142.993 219.012 134.782 291.681 181.983 338.704L183.042 337.641ZM184.236 178.203L185.319 179.242C214.275 149.044 261.706 136.743 303.065 154.838L303.666 153.464L304.267 152.09C261.66 133.449 212.906 146.137 183.154 177.165L184.236 178.203ZM303.666 153.464L304.317 154.815L349.406 133.101L348.755 131.75L348.105 130.398L303.015 152.112L303.666 153.464ZM348.755 131.75L349.658 130.552C341.462 124.374 330.93 117.708 318.817 113.015L318.275 114.414L317.733 115.813C329.513 120.376 339.802 126.88 347.853 132.948L348.755 131.75ZM318.275 114.414L318.867 113.036C264.299 89.6178 198.984 101.281 154.653 147.477L155.735 148.516L156.818 149.554C200.276 104.267 264.255 92.8636 317.683 115.793L318.275 114.414ZM155.735 148.516L154.653 147.478C112.02 191.939 98.6279 260.265 121.641 318.578L123.036 318.027L124.432 317.477C101.844 260.242 115.006 193.159 156.818 149.554L155.735 148.516ZM123.036 318.027L121.641 318.578C129.98 339.717 127.326 357.704 119.12 374.267C110.865 390.928 97.0134 406.111 82.9148 421.564L84.0229 422.575L85.131 423.586C99.1576 408.212 113.329 392.711 121.808 375.599C130.335 358.389 133.125 339.516 124.432 317.477L123.036 318.027ZM84.0229 422.575L82.9146 421.564C72.9633 432.475 62.8911 443.478 54.769 455.143L56 456L57.231 456.857C65.2195 445.384 75.149 434.531 85.1312 423.586L84.0229 422.575ZM56 456L57.0225 457.098L184.03 338.774L183.007 337.677L181.985 336.579L54.9775 454.902L56 456ZM400.978 113.204L399.893 112.168L209.365 311.759L210.45 312.794L211.535 313.83L402.063 114.24L400.978 113.204Z" fill="#FCFCFC" mask="url(#path-3-inside-1_304_111)"/>
<defs>
<filter id="filter0_d_304_111" x="-24" y="-22" width="560" height="560" filterUnits="userSpaceOnUse" color-interpolation-filters="sRGB">
<feFlood flood-opacity="0" result="BackgroundImageFix"/>
<feColorMatrix in="SourceAlpha" type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 127 0" result="hardAlpha"/>
<feOffset dy="2"/>
<feGaussianBlur stdDeviation="12"/>
<feComposite in2="hardAlpha" operator="out"/>
<feColorMatrix type="matrix" values="0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0 0.16 0"/>
<feBlend mode="normal" in2="BackgroundImageFix" result="effect1_dropShadow_304_111"/>
<feBlend mode="normal" in="SourceGraphic" in2="effect1_dropShadow_304_111" result="shape"/>
</filter>
<clipPath id="bgblur_1_304_111_clip_path" transform="translate(24 22)"><path d="M0 256C0 166.392 0 121.587 17.439 87.3615C32.7787 57.2556 57.2556 32.7787 87.3615 17.439C121.587 0 166.392 0 256 0C345.608 0 390.413 0 424.638 17.439C454.744 32.7787 479.221 57.2556 494.561 87.3615C512 121.587 512 166.392 512 256C512 345.608 512 390.413 494.561 424.638C479.221 454.744 454.744 479.221 424.638 494.561C390.413 512 345.608 512 256 512C166.392 512 121.587 512 87.3615 494.561C57.2556 479.221 32.7787 454.744 17.439 424.638C0 390.413 0 345.608 0 256Z"/>
</clipPath><clipPath id="clip0_304_111">
<rect width="512" height="512" fill="white"/>
</clipPath>
</defs>
</svg>

// ==/UserScript==



// backup
// @icon         https://grok.com/images/favicon.svg
// @icon         https://grok.com/images/android-chrome-512x512.png

(function () {
    'use strict';

    const MODELS = [
        { modelName: 'grok-4-heavy',   requestKind: 'DEFAULT' },
        { modelName: 'grok-4-thinking', requestKind: 'DEFAULT' },
        { modelName: 'grok-4',         requestKind: 'DEFAULT' },
        { modelName: 'grok-3',         requestKind: 'DEFAULT' },
        { modelName: 'grok-2',         requestKind: 'DEFAULT' }
    ];

    const CONFIG = {
        API: {
            endpoint: 'https://grok.com/rest/rate-limits',
            headers: {
                'Content-Type': 'application/json',
                'Accept': '*/*',
                'Origin': 'https://grok.com',
                'Referer': 'https://grok.com/',
                'Accept-Language': 'en-US,en;q=0.9'
            }
        },
        UI: {
            styles: `
                .grok-rate-limit-wrapper {
                    position: fixed;
                    bottom: 1rem;
                    right: 1rem;
                    z-index: 10000;
                    background: rgba(0, 0, 0, 0.78);
                    border-radius: 8px;
                    color: #ddd;
                    font-family: system-ui, -apple-system, sans-serif;
                    font-size: 0.77rem;
                    line-height: 1.4;
                    box-shadow: 0 5px 15px rgba(0, 0, 0, 0.35);
                    max-width: 340px;
                    padding: 0.6rem 0.7rem;
                    text-align: left;
                }
                .grok-rate-limit-wrapper .header {
                    display: flex;
                    justify-content: space-between;
                    align-items: center;
                    margin-bottom: 0.3rem;
                    font-weight: 600;
                    color: #fff;
                }
                .grok-rate-limit-wrapper button {
                    background: #374151;
                    color: #ddd;
                    border: none;
                    padding: 0.2rem 0.5rem;
                    border-radius: 4px;
                    cursor: pointer;
                    font-size: 0.75rem;
                }
                .grok-rate-limit-wrapper button:hover { background: #4b5563; }
                .grok-rate-limit {
                    white-space: nowrap;
                    overflow: hidden;
                    text-overflow: ellipsis;
                }
                .grok-rate-limit b { color: #22c55e; }
            `,
            refreshInterval: 8000
        },
        TIME: {
            SECONDS_PER_HOUR: 3600,
            SECONDS_PER_DAY: 86400
        }
    };

    let currentIndex = 0;
    let cachedResults = new Map(); // modelName -> data

    const utils = {
        generateId: () => Math.random().toString(16).slice(2),

        formatTimeWindow: (seconds) => {
            if (!seconds) return { value: '?', unit: 'h' };
            const hours = seconds / CONFIG.TIME.SECONDS_PER_HOUR;
            if (hours >= 24) return { value: (seconds / CONFIG.TIME.SECONDS_PER_DAY).toFixed(1), unit: 'd' };
            return { value: hours.toFixed(1), unit: 'h' };
        }
    };

    const ui = {
        createStyles: () => {
            const style = document.createElement('style');
            style.textContent = CONFIG.UI.styles;
            document.head.appendChild(style);
        },

        createMenu: () => {
            const wrapper = document.createElement('div');
            wrapper.className = 'grok-rate-limit-wrapper';

            wrapper.innerHTML = `
                <div class="header">
                    <span id="current_model">Loading...</span>
                    <div>
                        <button id="btn_prev">← Prev</button>
                        <button id="btn_next">Next →</button>
                    </div>
                </div>
                <div id="rate_queries" class="grok-rate-limit">Queries: N/A</div>
                <div id="rate_params" class="grok-rate-limit">Model: ... | Kind: ...</div>
                <div id="rate_low" class="grok-rate-limit">Low Effort: null</div>
                <div id="rate_high" class="grok-rate-limit">High Effort: null</div>
            `;

            document.body.appendChild(wrapper);

            // Button listeners
            document.getElementById('btn_prev').addEventListener('click', () => {
                currentIndex = (currentIndex - 1 + MODELS.length) % MODELS.length;
                renderCurrent();
            });

            document.getElementById('btn_next').addEventListener('click', () => {
                currentIndex = (currentIndex + 1) % MODELS.length;
                renderCurrent();
            });
        },

        updateDisplay: (model, data) => {
            document.getElementById('current_model').textContent = model.modelName;

            const queriesElem = document.getElementById('rate_queries');
            const paramsElem = document.getElementById('rate_params');
            const lowElem = document.getElementById('rate_low');
            const highElem = document.getElementById('rate_high');

            // Queries
            if (data && typeof data.remainingQueries === 'number') {
                const { value, unit } = utils.formatTimeWindow(data.windowSizeSeconds);
                const color = data.remainingQueries > 3 ? '#22c55e' : data.remainingQueries > 0 ? '#eab308' : '#ef4444';
                queriesElem.innerHTML = `Queries: <b style="color:${color}">${data.remainingQueries}</b>/${data.totalQueries || '?'} (${value}${unit})`;
            } else {
                queriesElem.textContent = 'Queries: N/A (no queries field)';
            }

            // Parameters
            paramsElem.textContent = `Model: ${model.modelName} | Kind: ${model.requestKind}`;

            // Low / High Effort
            lowElem.textContent = `Low Effort: ${data?.lowEffortRateLimits ? JSON.stringify(data.lowEffortRateLimits) : 'null'}`;
            highElem.textContent = `High Effort: ${data?.highEffortRateLimits ? JSON.stringify(data.highEffortRateLimits) : 'null'}`;
        }
    };

    const api = {
        async fetchForModel(model) {
            const cacheKey = model.modelName;
            if (cachedResults.has(cacheKey)) {
                return cachedResults.get(cacheKey);
            }

            try {
                const headers = {
                    ...CONFIG.API.headers,
                    'User-Agent': navigator.userAgent,
                    'X-Xai-Request-Id': utils.generateId()
                };

                const response = await fetch(CONFIG.API.endpoint, {
                    method: 'POST',
                    headers,
                    body: JSON.stringify({
                        modelName: model.modelName,
                        requestKind: model.requestKind
                    })
                });

                if (!response.ok) throw new Error(`HTTP ${response.status}`);

                const json = await response.json();
                cachedResults.set(cacheKey, json);
                return json;
            } catch (error) {
                console.warn(`Failed to fetch for ${model.modelName}`, error);
                const errorData = { error: true };
                cachedResults.set(cacheKey, errorData);
                return errorData;
            }
        }
    };

    const renderCurrent = async () => {
        const model = MODELS[currentIndex];
        const data = await api.fetchForModel(model);
        ui.updateDisplay(model, data);
    };

    const init = () => {
        ui.createStyles();
        ui.createMenu();
        renderCurrent();                    // initial load
        setInterval(renderCurrent, CONFIG.UI.refreshInterval);
    };

    if (document.readyState === 'complete' || document.readyState === 'interactive') {
        init();
    } else {
        document.addEventListener('DOMContentLoaded', init);
    }
})();