T3.chat - Markdown code un-indenter

Automatically un-indents code blocks in the message input.

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Greasemonkey 油猴子Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Violentmonkey 暴力猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴Userscripts ,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey 篡改猴,才能安装此脚本。

您需要先安装一款用户脚本管理器扩展后才能安装此脚本。

(我已经安装了用户脚本管理器,让我安装!)

Advertisement:

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展,比如 Stylus,才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

您需要先安装一款用户样式管理器扩展后才能安装此样式。

(我已经安装了用户样式管理器,让我安装!)

Advertisement:

// ==UserScript==
// @name        T3.chat - Markdown code un-indenter
// @namespace   Violentmonkey Scripts
// @match       https://beta.t3.chat/*
// @match       https://t3.chat/*
// @grant       none
// @version     1.0
// @author      koza.dev
// @description Automatically un-indents code blocks in the message input.
// @license MIT
// ==/UserScript==

(function() {
    let debounceTimer

    const unindent = (text) => {
        return text.replace(/```([\s\S]*?)```/g, (match, code) => {
            const lines = code.split('\n')

            // Filter out empty lines to find the actual minimum indentation
            const contentLines = lines.filter((line) => line.trim().length > 0)
            if (contentLines.length === 0) return match

            const minIndent = Math.min(
                ...contentLines.map((line) => line.match(/^\s*/)[0].length)
            )

            const processedCode = lines
                .map((line) => (line.length >= minIndent ? line.slice(minIndent) : line.trimStart()))
                .join('\n')

            return '```' + processedCode + '```'
        })
    }

    const processInput = (target) => {
        const originalValue = target.value
        const newValue = unindent(originalValue)

        if (originalValue !== newValue) {
            const start = target.selectionStart
            const end = target.selectionEnd

            target.value = newValue

            // Restore cursor position
            target.selectionStart = start
            target.selectionEnd = end

            // Trigger Svelte 5 reactivity
            target.dispatchEvent(
                new InputEvent('input', {
                    bubbles: true,
                    cancelable: true,
                    inputType: 'insertText'
                })
            )
            target.dispatchEvent(new Event('change', { bubbles: true }))
        }
    }

    window.addEventListener('input', (e) => {
        if (e.target.id !== 'chat-input') return

        clearTimeout(debounceTimer)
        debounceTimer = setTimeout(() => {
            processInput(e.target)
        }, 500)
    })
})()