Greasy Fork is available in English.

ChatGPT fix for missing caret

Fixes ChatGPT bug where caret disappears after pasting CRLF-based newlines (Windows standard); converts CRLF to LF and pastes properly in ChatGPT ProseMirror editor

이 스크립트를 설치하려면 Tampermonkey, Greasemonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

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

이 스크립트를 설치하려면 Tampermonkey 또는 Violentmonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey 또는 Userscripts와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 Tampermonkey와 같은 확장 프로그램이 필요합니다.

이 스크립트를 설치하려면 유저 스크립트 관리자 확장 프로그램이 필요합니다.

(이미 유저 스크립트 관리자가 설치되어 있습니다. 설치를 진행합니다!)

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 Stylus와 같은 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

이 스타일을 설치하려면 유저 스타일 관리자 확장 프로그램이 필요합니다.

(이미 유저 스타일 관리자가 설치되어 있습니다. 설치를 진행합니다!)

// ==UserScript==
// @name         ChatGPT fix for missing caret
// @namespace    http://tampermonkey.net/
// @version      1.1
// @description  Fixes ChatGPT bug where caret disappears after pasting CRLF-based newlines (Windows standard); converts CRLF to LF and pastes properly in ChatGPT ProseMirror editor
// @license      MIT
// @match        https://chat.openai.com/*
// @match        https://chatgpt.com/*
// @grant        none
// @homepage     https://greasyfork.org/en/scripts/559039/
// ==/UserScript==

(function ()
{
    'use strict';

    document.addEventListener('paste', function (event)
    {
        const target = event.target;
        if (!(target instanceof HTMLElement)) return;

        const prosemirror = target.closest('.ProseMirror');
        if (!prosemirror) return;

        const clipdata = event.clipboardData;
        if (!clipdata) return;

        const text = clipdata.getData('text/plain');
        if (!text || !text.includes('\r\n')) return;

        event.preventDefault();

        const normalized = text.replace(/\r\n?/g, '\n');
        const datatransfer = new DataTransfer();
        datatransfer.setData('text/plain', normalized);

        const syntheticpaste = new ClipboardEvent('paste',
        {
            clipboardData: datatransfer,
            bubbles: true,
            cancelable: true
        });

        target.dispatchEvent(syntheticpaste);
    }, true);
})();