Chat Squircle

Adds corner-shape: squircle to chat input boxes on ChatGPT, Gemini, Grok, and AI Studio

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, Greasemonkey alebo Violentmonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey, % alebo Violentmonkey.

Na nainštalovanie skriptu si budete musieť nainštalovať rozšírenie, ako napríklad Tampermonkey alebo Userscripts.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie, ako napríklad Tampermonkey.

Na inštaláciu tohto skriptu je potrebné nainštalovať rozšírenie správcu používateľských skriptov.

(Už mám správcu používateľských skriptov, nechajte ma ho nainštalovať!)

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie, ako napríklad Stylus.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

Na inštaláciu tohto štýlu je potrebné nainštalovať rozšírenie správcu používateľských štýlov.

(Už mám správcu používateľských štýlov, nechajte ma ho nainštalovať!)

// ==UserScript==
// @name         Chat Squircle
// @namespace    https://loongphy.com
// @version      1.1
// @description  Adds corner-shape: squircle to chat input boxes on ChatGPT, Gemini, Grok, and AI Studio
// @author       loongphy
// @match        https://chatgpt.com/*
// @match        https://gemini.google.com/*
// @match        https://grok.com/*
// @match        https://aistudio.google.com/*
// @grant        none
// @run-at       document-idle
// ==/UserScript==

(function() {
    'use strict';

    const SQUIRCLE_CSS = `
        corner-shape: squircle;
    `;

    const CONFIG = [
        {
            domain: 'chatgpt.com',
            selector: '#prompt-textarea',
            targetParent: true,
            parentDepth: 3
        },
        {
            domain: 'chatgpt.com',
            selector: '.user-message-bubble-color',
            targetParent: false,
            parentDepth: 0
        },
        {
            domain: 'chatgpt.com',
            selector: '.divide-token-border-default',
            targetParent: false,
            parentDepth: 0
        },
        {
            domain: 'gemini.google.com',
            selector: '.ql-editor',
            targetParent: true,
            parentDepth: 7
        },
        {
            domain: 'grok.com',
            selector: '[contenteditable="true"]',
            targetParent: true,
            parentDepth: 4
        },
        {
            domain: 'aistudio.google.com',
            selector: '.prompt-input-wrapper',
            targetParent: false, 
            parentDepth: 0
        }
    ];

    function getDomain() {
        return window.location.hostname;
    }

    function applySquircle() {
        const domain = getDomain();
        // Find all configs that match the current domain
        const activeConfigs = CONFIG.filter(config => domain.includes(config.domain));
        
        if (activeConfigs.length === 0) return;
        
        activeConfigs.forEach(config => {
            const elements = document.querySelectorAll(config.selector);

            elements.forEach(el => {
                let target = el;
                
                if (config.targetParent) {
                    // Traverse up to find the container that likely has the border
                    // This is heuristic: look for a div with a border or background
                    let parent = el.parentElement;
                    for(let i=0; i<config.parentDepth && parent; i++) {
                         target = parent;
                         parent = parent.parentElement;
                    }
                }

                // Apply the style directly to the target element
                if (!target.dataset.squircleApplied) {
                    target.style.cssText += SQUIRCLE_CSS;
                    target.dataset.squircleApplied = "true";
                }
            });
        });
    }

    // Initial run
    applySquircle();

    // Observe changes for dynamic SPAs
    const observer = new MutationObserver((mutations) => {
        applySquircle();
    });

    observer.observe(document.body, {
        childList: true,
        subtree: true
    });

})();