ChatGPT 채팅 코드 복사 및 코드 내보내기 버튼

존재하다 chatgpt.com 상단 코드 블록의 오른쪽 하단에 애니메이션을 추가합니다.“코드 복사”버튼과 ChatGPT 응답으로 코드 블록에 내보내기 버튼이 추가됩니다.,코드 블록의 클래스 이름으로 감지된 프로그래밍 언어를 기반으로 파일에 코드를 저장하라는 메시지를 사용자에게 표시합니다.。

// ==UserScript==
// @name              ChatGPT Chat copy code and export code buttons
// @description       exist chatgpt.com Add an animation in the lower right corner of the upper code block“Copy code”button and a for ChatGPT Code block in response adds export button,Prompts the user to save code to a file based on the programming language detected by the code block’s class name。
// @name:zh           ChatGPT 聊天复制代码和导出代码按钮
// @description:zh    在 chatgpt.com 上代码块的右下角添加一个带有动画的“复制代码”按钮和一个为 ChatGPT 响应中的代码块添加导出按钮,提示用户根据代码块的类名检测到的编程语言将代码保存为文件。
// @name:zh-CN        ChatGPT 聊天复制代码和导出代码按钮
// @description:zh-CN 在 chatgpt.com 上代码块的右下角添加一个带有动画的“复制代码”按钮和一个为 ChatGPT 响应中的代码块添加导出按钮,提示用户根据代码块的类名检测到的编程语言将代码保存为文件。
// @name:ar           ChatGPT نسخ رمز الدردشة وأزرار رمز التصدير
// @description:ar    يخرج chatgpt.com أضف رسمًا متحركًا في الركن الأيمن السفلي من كتلة التعليمات البرمجية العلوية“نسخ الرمز”زر و ل ChatGPT تضيف كتلة التعليمات البرمجية في الاستجابة زر التصدير,يطالب المستخدم بحفظ التعليمات البرمجية في ملف بناءً على لغة البرمجة التي اكتشفها اسم فئة كتلة التعليمات البرمجية。
// @name:bg           ChatGPT Бутони за копиране на код за чат и експортиране на код
// @description:bg    съществуват chatgpt.com Добавете анимация в долния десен ъгъл на горния кодов блок“Копирайте кода”бутон и за ChatGPT Кодовият блок в отговор добавя бутон за експортиране,Подканва потребителя да запише код във файл въз основа на езика за програмиране, открит от името на класа на кодовия блок。
// @name:cs           ChatGPT Chat kopírovat kód a exportovat tlačítka kódu
// @description:cs    existovat chatgpt.com Přidejte animaci do pravého dolního rohu horního bloku kódu“Kopírovat kód”tlačítko a pro ChatGPT Blok kódu jako odpověď přidá tlačítko exportu,Vyzve uživatele k uložení kódu do souboru na základě programovacího jazyka zjištěného názvem třídy bloku kódu。
// @name:da           ChatGPT Chat kopi kode og eksport kode knapper
// @description:da    eksistere chatgpt.com Tilføj en animation i nederste højre hjørne af den øverste kodeblok“Kopiér kode”knap og en for ChatGPT Kodeblok som svar tilføjer eksportknap,Beder brugeren om at gemme kode i en fil baseret på det programmeringssprog, der registreres af kodeblokkens klassenavn。
// @name:de           ChatGPT Chat-Schaltflächen zum Kopieren und Exportieren von Code
// @description:de    existieren chatgpt.com Fügen Sie in der unteren rechten Ecke des oberen Codeblocks eine Animation hinzu“Code kopieren”Knopf und ein für ChatGPT Als Antwort fügt der Codeblock eine Schaltfläche zum Exportieren hinzu,Fordert den Benutzer auf, Code in einer Datei zu speichern, basierend auf der Programmiersprache, die durch den Klassennamen des Codeblocks erkannt wird。
// @name:el           ChatGPT Κουμπιά κωδικού αντιγραφής συνομιλίας και κωδικού εξαγωγής
// @description:el    υπάρχω chatgpt.com Προσθέστε μια κινούμενη εικόνα στην κάτω δεξιά γωνία του επάνω μπλοκ κώδικα“Αντιγραφή κωδικού”κουμπί και ένα για ChatGPT Το μπλοκ κώδικα σε απόκριση προσθέτει το κουμπί εξαγωγής,Προτρέπει τον χρήστη να αποθηκεύσει κώδικα σε ένα αρχείο με βάση τη γλώσσα προγραμματισμού που ανιχνεύεται από το όνομα κλάσης του μπλοκ κώδικα。
// @name:en           ChatGPT Chat copy code and export code buttons
// @description:en    exist chatgpt.com Add an animation in the lower right corner of the upper code block“Copy code”button and a for ChatGPT Code block in response adds export button,Prompts the user to save code to a file based on the programming language detected by the code block’s class name。
// @name:eo           ChatGPT Babilejo kopikodo kaj eksportkodo butonoj
// @description:eo    ekzisti chatgpt.com Aldonu animacion en la malsupra dekstra angulo de la supra kodbloko“Kopiu kodon”butono kaj a por ChatGPT Kodbloko en respondo aldonas eksportbutonon,Instigas la uzanton konservi kodon al dosiero bazita sur la programlingvo detektita de la klasnomo de la kodbloko。
// @name:es           ChatGPT Botones de copiar código y exportar código de chat
// @description:es    existir chatgpt.com Agregue una animación en la esquina inferior derecha del bloque de código superior“Copiar código”botón y un para ChatGPT El bloque de código en respuesta agrega el botón de exportación,Solicita al usuario que guarde el código en un archivo según el lenguaje de programación detectado por el nombre de clase del bloque de código.。
// @name:fi           ChatGPT Chat kopioi koodi ja vientikoodin painikkeet
// @description:fi    olemassa chatgpt.com Lisää animaatio ylemmän koodilohkon oikeaan alakulmaan“Kopioi koodi”-painiketta ja for ChatGPT Koodilohko vastauksena lisää vientipainikkeen,Kehottaa käyttäjää tallentamaan koodin tiedostoon koodilohkon luokan nimen havaitseman ohjelmointikielen perusteella。
// @name:fr           ChatGPT Boutons de copie du code de chat et d’exportation du code
// @description:fr    exister chatgpt.com Ajoutez une animation dans le coin inférieur droit du bloc de code supérieur“Copier le code”bouton et un pour ChatGPT Le bloc de code en réponse ajoute un bouton d’exportation,Invite l’utilisateur à enregistrer le code dans un fichier basé sur le langage de programmation détecté par le nom de classe du bloc de code。
// @name:he           ChatGPT לחצני צ’אט העתקת קוד וייצוא קוד
// @description:he    לְהִתְקַיֵם chatgpt.com הוסף אנימציה בפינה הימנית התחתונה של גוש הקוד העליון“העתק קוד”כפתור ו- for ChatGPT בלוק קוד בתגובה מוסיף לחצן ייצוא,מבקש מהמשתמש לשמור קוד בקובץ המבוסס על שפת התכנות שזוהתה על ידי שם המחלקה של בלוק הקוד。
// @name:hr           ChatGPT Gumbi za kopiranje koda i izvoz koda
// @description:hr    postojati chatgpt.com Dodajte animaciju u donji desni kut gornjeg bloka koda“Kopiraj kod”gumb i for ChatGPT Blok koda kao odgovor dodaje gumb za izvoz,Traži od korisnika da spremi kod u datoteku na temelju programskog jezika otkrivenog nazivom klase kodnog bloka。
// @name:hu           ChatGPT Chat kódmásolás és kódexportálás gombok
// @description:hu    létezik chatgpt.com Adjon hozzá animációt a felső kódblokk jobb alsó sarkába“Kód másolása”gombot és egy for ChatGPT A kódblokk válaszként hozzáadja az export gombot,Arra kéri a felhasználót, hogy a kódblokk osztályneve által észlelt programozási nyelv alapján mentse el a kódot egy fájlba。
// @name:id           ChatGPT Tombol salin kode obrolan dan ekspor kode
// @description:id    ada chatgpt.com Tambahkan animasi di sudut kanan bawah blok kode atas“Salin kode”tombol dan untuk ChatGPT Blok kode sebagai respons menambahkan tombol ekspor,Meminta pengguna untuk menyimpan kode ke file berdasarkan bahasa pemrograman yang terdeteksi oleh nama kelas blok kode。
// @name:it           ChatGPT Chatta copia il codice ed esporta i pulsanti del codice
// @description:it    esistere chatgpt.com Aggiungi un’animazione nell’angolo in basso a destra del blocco di codice superiore“Copia il codice”pulsante e un for ChatGPT Il blocco di codice in risposta aggiunge il pulsante di esportazione,Richiede all’utente di salvare il codice in un file in base al linguaggio di programmazione rilevato dal nome della classe del blocco di codice。
// @name:ja           ChatGPT チャットのコードのコピー ボタンとコードのエクスポート ボタン
// @description:ja    存在する chatgpt.com 上部のコード ブロックの右下隅にアニメーションを追加します。“コードをコピーする”ボタンと for ChatGPT 応答のコード ブロックはエクスポート ボタンを追加します,コード ブロックのクラス名によって検出されたプログラミング言語に基づいて、コードをファイルに保存するようにユーザーにプロンプトを表示します。。
// @name:ka           ChatGPT ჩეთის კოპირების კოდის და ექსპორტის კოდის ღილაკები
// @description:ka    არსებობს chatgpt.com დაამატეთ ანიმაცია ზედა კოდის ბლოკის ქვედა მარჯვენა კუთხეში“დააკოპირეთ კოდი”ღილაკი და ამისთვის ChatGPT კოდის ბლოკი პასუხად ამატებს ექსპორტის ღილაკს,სთხოვს მომხმარებელს შეინახოს კოდი ფაილში კოდის ბლოკის კლასის სახელით აღმოჩენილი პროგრამირების ენის საფუძველზე。
// @name:ko           ChatGPT 채팅 코드 복사 및 코드 내보내기 버튼
// @description:ko    존재하다 chatgpt.com 상단 코드 블록의 오른쪽 하단에 애니메이션을 추가합니다.“코드 복사”버튼과 ChatGPT 응답으로 코드 블록에 내보내기 버튼이 추가됩니다.,코드 블록의 클래스 이름으로 감지된 프로그래밍 언어를 기반으로 파일에 코드를 저장하라는 메시지를 사용자에게 표시합니다.。
// @name:nl           ChatGPT Chat kopieer code en exportcodeknoppen
// @description:nl    bestaan chatgpt.com Voeg een animatie toe in de rechter benedenhoek van het bovenste codeblok“Kopieer code”knop en een voor ChatGPT Codeblok als reactie voegt een exportknop toe,Vraagt de gebruiker om code op te slaan in een bestand op basis van de programmeertaal die wordt gedetecteerd door de klassenaam van het codeblok。
// @name:nb           ChatGPT Chat kopier kode og eksport kode knapper
// @description:nb    eksistere chatgpt.com Legg til en animasjon i nedre høyre hjørne av den øvre kodeblokken“Kopier koden”knapp og en for ChatGPT Kodeblokk som svar legger til eksportknapp,Ber brukeren om å lagre kode i en fil basert på programmeringsspråket oppdaget av kodeblokkens klassenavn。
// @name:pl           ChatGPT Przyciski kopiowania kodu i eksportowania kodu na czacie
// @description:pl    istnieć chatgpt.com Dodaj animację w prawym dolnym rogu górnego bloku kodu“Skopiuj kod”przycisk i for ChatGPT Blok kodu w odpowiedzi dodaje przycisk eksportu,Monituje użytkownika o zapisanie kodu w pliku w oparciu o język programowania wykryty przez nazwę klasy bloku kodu。
// @name:pt-BR        ChatGPT Botões de copiar código de bate-papo e exportar código
// @description:pt-BR existir chatgpt.com Adicione uma animação no canto inferior direito do bloco de código superior“Copiar código”botão e um para ChatGPT Bloco de código em resposta adiciona botão de exportação,Solicita ao usuário que salve o código em um arquivo com base na linguagem de programação detectada pelo nome da classe do bloco de código。
// @name:ro           ChatGPT Butoanele de copiere a codului de chat și a codului de export
// @description:ro    exista chatgpt.com Adăugați o animație în colțul din dreapta jos al blocului de cod de sus“Copiați codul”buton și un pentru ChatGPT Blocarea codului ca răspuns adaugă butonul de export,Solicită utilizatorului să salveze codul într-un fișier pe baza limbajului de programare detectat de numele clasei blocului de cod。
// @name:ru           ChatGPT Кнопки копирования кода чата и экспорта кода
// @description:ru    существовать chatgpt.com Добавьте анимацию в правом нижнем углу верхнего блока кода.“Скопировать код”кнопка и для ChatGPT Блок кода в ответ добавляет кнопку экспорта,Предлагает пользователю сохранить код в файл на основе языка программирования, определенного по имени класса блока кода.。
// @name:sk           ChatGPT Kopírovanie kódu chatu a tlačidlá exportu kódu
// @description:sk    existujú chatgpt.com Pridajte animáciu do pravého dolného rohu horného bloku kódu“Kopírovať kód”tlačidlo a pre ChatGPT Blok kódu v odpovedi pridá tlačidlo exportu,Vyzve používateľa, aby uložil kód do súboru na základe programovacieho jazyka, ktorý sa zistil podľa názvu triedy bloku kódu。
// @name:sr           ChatGPT Дугмад за копирање кода за ћаскање и дугмад за извоз кода
// @description:sr    постоје chatgpt.com Додајте анимацију у доњи десни угао горњег кодног блока“Копирај код”дугме и за ChatGPT Блок кода као одговор додаје дугме за извоз,Тражи од корисника да сачува код у датотеци на основу програмског језика који је откривен именом класе блока кода。
// @name:sv           ChatGPT Chat kopiera kod och export kod knappar
// @description:sv    existera chatgpt.com Lägg till en animation i det nedre högra hörnet av det övre kodblocket“Kopiera kod”knapp och en för ChatGPT Kodblock som svar lägger till exportknapp,Ber användaren att spara kod till en fil baserat på programmeringsspråket som identifieras av kodblockets klassnamn。
// @name:th           ChatGPT ปุ่มคัดลอกรหัสแชทและปุ่มส่งออกรหัส
// @description:th    มีอยู่ chatgpt.com เพิ่มภาพเคลื่อนไหวที่มุมขวาล่างของบล็อกโค้ดด้านบน“คัดลอกรหัส”ปุ่ม และสำหรับ ChatGPT บล็อกโค้ดในการตอบสนองเพิ่มปุ่มส่งออก,พร้อมต์ให้ผู้ใช้บันทึกโค้ดลงในไฟล์ตามภาษาการเขียนโปรแกรมที่ตรวจพบโดยชื่อคลาสของบล็อคโค้ด。
// @name:tr           ChatGPT Sohbet kodunu kopyala ve kodu dışa aktar düğmeleri
// @description:tr    var olmak chatgpt.com Üst kod bloğunun sağ alt köşesine bir animasyon ekleyin“Kodu kopyala”düğme ve bir için ChatGPT Yanıt olarak kod bloğu dışa aktarma düğmesi ekler,Kullanıcıdan, kod bloğunun sınıf adı tarafından algılanan programlama diline dayalı olarak kodu bir dosyaya kaydetmesini ister。
// @name:ug           ChatGPT پاراڭلىشىش كودى ۋە ئېكسپورت كود كۇنۇپكىسى
// @description:ug    مەۋجۇت chatgpt.com ئۈستۈنكى كود توپىنىڭ ئوڭ ئوڭ بۇلۇڭىغا كارتون قوشۇڭ“كودنى كۆچۈرۈڭ”كۇنۇپكا ۋە ئۈچۈن ChatGPT بۇنىڭغا جاۋابەن كود توسۇش ئېكسپورت كۇنۇپكىسىنى قوشىدۇ,ئىشلەتكۈچىنىڭ كود بۆلەكنىڭ سىنىپ ئىسمى تەرىپىدىن بايقالغان پروگرامما تىلىغا ئاساسەن كودنى ھۆججەتكە ساقلىشىنى تەلەپ قىلىدۇ。
// @name:uk           ChatGPT Кнопки копіювання та експорту коду чату
// @description:uk    існують chatgpt.com Додайте анімацію в нижній правий кут верхнього блоку коду“Скопіюйте код”кнопка і for ChatGPT Блок коду у відповідь додає кнопку експорту,Пропонує користувачеві зберегти код у файл на основі мови програмування, визначеної назвою класу блоку коду。
// @name:vi           ChatGPT Nút trò chuyện sao chép mã và xuất mã
// @description:vi    hiện hữu chatgpt.com Thêm hình động ở góc dưới bên phải của khối mã phía trên“Sao chép mã”nút và một cho ChatGPT Khối mã phản hồi thêm nút xuất,Nhắc người dùng lưu mã vào tệp dựa trên ngôn ngữ lập trình được phát hiện bởi tên lớp của khối mã。
// @name:zh-TW        ChatGPT 聊天複製程式碼和匯出程式碼按鈕
// @description:zh-TW 在 chatgpt.com 上代碼區塊的右下角添加一個帶有動畫的“複製程式碼”按鈕和一個為 ChatGPT 響應中的程式碼區塊新增匯出按鈕,提示使用者根據程式碼區塊的類別名稱偵測到的程式語言將程式碼儲存為文件。
// @name:zh-HK        ChatGPT 聊天複製程式碼和匯出程式碼按鈕
// @description:zh-HK 在 chatgpt.com 上代碼區塊的右下角添加一個帶有動畫的“複製程式碼”按鈕和一個為 ChatGPT 響應中的程式碼區塊新增匯出按鈕,提示使用者根據程式碼區塊的類別名稱偵測到的程式語言將程式碼儲存為文件。
// @name:fr-CA        ChatGPT Boutons de copie du code de chat et d’exportation du code
// @description:fr-CA exister chatgpt.com Ajoutez une animation dans le coin inférieur droit du bloc de code supérieur“Copier le code”bouton et un pour ChatGPT Le bloc de code en réponse ajoute un bouton d’exportation,Invite l’utilisateur à enregistrer le code dans un fichier basé sur le langage de programmation détecté par le nom de classe du bloc de code。
// @match             https://chatgpt.com/*
// @match             https://share.nezhagpt.cloud/*
// @author            YodaBets,人民的勤务员 <china.qinwuyuan@gmail.com>
// @namespace         https://github.com/ChinaGodMan/UserScripts
// @supportURL        https://github.com/ChinaGodMan/UserScripts/issues
// @homepageURL       https://github.com/ChinaGodMan/UserScripts
// @license           MIT
// @icon              
// @compatible        chrome
// @compatible        firefox
// @compatible        edge
// @compatible        opera
// @compatible        safari
// @version           1.2.0.0
// ==/UserScript==
(function () {
    'use strict'
    var EXPORT = true
    const copyToClipboard = (text, button) => {
        navigator.clipboard.writeText(text).then(() => {
            console.log('Copied code to clipboard')
            button.innerHTML = 'Copied!'
            setTimeout(() => {
                button.innerHTML = '' // Clear the text
                const svgIcon = getSVGIcon() // Re-add the SVG icon
                button.appendChild(svgIcon)
            }, 2000)
        }, (err) => {
            console.error('Failed to copy code: ', err)
        })
    }
    function getSVGIcon(isEx = false) {
        const svgIcon = document.createElementNS('http://www.w3.org/2000/svg', 'svg')
        svgIcon.setAttribute('width', '24')
        svgIcon.setAttribute('height', '24')
        svgIcon.setAttribute('fill', 'none')
        svgIcon.setAttribute('viewBox', '0 0 24 24')
        svgIcon.classList.add('icon-sm')
        const path = document.createElementNS('http://www.w3.org/2000/svg', 'path')
        path.setAttribute('fill', 'currentColor')
        path.setAttribute('fill-rule', 'evenodd')
        if (isEx) {
            path.setAttribute('transform', 'rotate(180 12 12)')
            path.setAttribute('d', 'M5 20a1 1 0 0 1 1-1h12a1 1 0 1 1 0 2H6a1 1 0 0 1-1-1zm7-2a1 1 0 0 1-1-1V8.414L9.707 10.707a1 1 0 1 1-1.414-1.414l4-4a1 1 0 0 1 1.414 0l4 4a1 1 0 0 1-1.414 1.414L13 8.414V17a1 1 0 0 1-1 1z')
        } else {
            path.setAttribute('d', 'M7 5a3 3 0 0 1 3-3h9a3 3 0 0 1 3 3v9a3 3 0 0 1-3 3h-2v2a3 3 0 0 1-3 3H5a3 3 0 0 1-3-3v-9a3 3 0 0 1 3-3h2zm2 2h5a3 3 0 0 1 3 3v5h2a1 1 0 0 0 1-1V5a1 1 0 0 0-1-1h-9a1 1 0 0 0-1 1zM5 9a1 1 0 0 0-1 1v9a1 1 0 0 0 1 1h9a1 1 0 0 0 1-1v-9a1 1 0 0 0-1-1z')
        }
        path.setAttribute('clip-rule', 'evenodd')
        svgIcon.appendChild(path)
        return svgIcon
    }
    const addButton = (elem) => {
        const button = document.createElement('button')
        const svgIcon = getSVGIcon()
        button.appendChild(svgIcon)
        button.style.position = 'absolute'
        button.style.bottom = '8px'
        button.style.right = '8px'
        button.style.fontSize = '12px'
        button.style.padding = '4px 8px'
        button.style.border = '1px solid #ccc'
        button.style.borderRadius = '3px'
        button.style.background = 'rgba(0,0,0,0.1)'
        button.style.color = 'white'
        button.style.cursor = 'pointer'
        button.style.zIndex = '10'
        button.style.transition = 'background-color 0.3s ease'
        button.addEventListener('click', (e) => {
            e.stopPropagation()
            copyToClipboard(elem.querySelector('code').textContent, button)
        })
        button.addEventListener('mouseover', () => {
            button.style.backgroundColor = 'rgba(0,0,0,0.2)'
        })
        button.addEventListener('mouseout', () => {
            button.style.backgroundColor = 'rgba(0,0,0,0.1)'
        })
        elem.style.position = 'relative'
        elem.appendChild(button)
    }
    const addexButton = (elem) => {
        const button = document.createElement('button')
        const svgIcon = getSVGIcon(true)
        button.appendChild(svgIcon)
        button.style.position = 'absolute'
        button.style.bottom = '8px'
        button.style.right = '48px'
        button.style.fontSize = '12px'
        button.style.padding = '4px 8px'
        button.style.border = '1px solid #ccc'
        button.style.borderRadius = '3px'
        button.style.background = 'rgba(0,0,0,0.1)'
        button.style.color = 'white'
        button.style.cursor = 'pointer'
        button.style.zIndex = '10'
        button.style.transition = 'background-color 0.3s ease'
        button.addEventListener('click', (e) => {
            var languageDiv = elem.querySelector('div.flex.items-center.text-token-text-secondary')
            e.stopPropagation()
            exportCode(elem, languageDiv.textContent)
        })
        elem.style.position = 'relative'
        elem.appendChild(button)
    }
    const observeCodeBlocks = () => {
        const codeBlocks = document.querySelectorAll('pre:not(.copy-code-processed)')
        if (codeBlocks.length) {
            codeBlocks.forEach(block => {
                if (block.querySelector('div.flex.items-center.text-token-text-secondary')) {
                    // console.log(block)
                    addButton(block)
                    if (EXPORT) {
                        addexButton(block)
                    }
                    block.classList.add('copy-code-processed')
                }

            })
        }
    }
    async function exportCode(codeBlock, language) {
        let fileName
        let fileExtension
        let mimeType
        // Determine filename, extension, and MIME type based on language
        switch (language) {
            case 'javascript':
            case 'js':
                fileName = 'script'
                fileExtension = '.js'
                mimeType = 'application/javascript'
                break
            case 'html':
                fileName = 'index'
                fileExtension = '.html'
                mimeType = 'text/html'
                break
            case 'css':
                fileName = 'styles'
                fileExtension = '.css'
                mimeType = 'text/css'
                break
            case 'python':
            case 'py':
                fileName = 'main'
                fileExtension = '.py'
                mimeType = 'text/x-python'
                break
            default:
                // If language cannot be determined from <span>, fallback to provided language
                switch (language.toLowerCase()) {
                    case 'javascript':
                    case 'js':
                        fileName = 'script'
                        fileExtension = '.js'
                        mimeType = 'application/javascript'
                        break
                    case 'html':
                        fileName = 'index'
                        fileExtension = '.html'
                        mimeType = 'text/html'
                        break
                    case 'css':
                        fileName = 'styles'
                        fileExtension = '.css'
                        mimeType = 'text/css'
                        break
                    case 'python':
                    case 'py':
                        fileName = 'main'
                        fileExtension = '.py'
                        mimeType = 'text/x-python'
                        break
                    case 'java':
                        fileName = 'Main'
                        fileExtension = '.java'
                        mimeType = 'text/x-java-source'
                        break
                    case 'kotlin':
                        fileName = 'Main'
                        fileExtension = '.kt'
                        mimeType = 'text/x-kotlin'
                        break
                    case 'c++':
                    case 'cpp':
                        fileName = 'main'
                        fileExtension = '.cpp'
                        mimeType = 'text/x-c++src'
                        break
                    case 'c#':
                    case 'csharp':
                        fileName = 'Program'
                        fileExtension = '.cs'
                        mimeType = 'text/x-csharp'
                        break
                    case 'c':
                        fileName = 'main'
                        fileExtension = '.c'
                        mimeType = 'text/x-csrc'
                        break
                    case 'ruby':
                        fileName = 'script'
                        fileExtension = '.rb'
                        mimeType = 'text/x-ruby'
                        break
                    case 'rust':
                        fileName = 'main'
                        fileExtension = '.rs'
                        mimeType = 'text/x-rustsrc'
                        break
                    case 'php':
                        fileName = 'script'
                        fileExtension = '.php'
                        mimeType = 'text/x-php'
                        break
                    case 'swift':
                        fileName = 'main'
                        fileExtension = '.swift'
                        mimeType = 'text/x-swift'
                        break
                    case 'typescript':
                    case 'ts':
                        fileName = 'script'
                        fileExtension = '.ts'
                        mimeType = 'application/typescript'
                        break
                    case 'go':
                        fileName = 'main'
                        fileExtension = '.go'
                        mimeType = 'text/x-go'
                        break
                    case 'perl':
                        fileName = 'script'
                        fileExtension = '.pl'
                        mimeType = 'text/x-perl'
                        break
                    case 'lua':
                        fileName = 'script'
                        fileExtension = '.lua'
                        mimeType = 'text/x-lua'
                        break
                    case 'powershell':
                    case 'ps':
                        fileName = 'script'
                        fileExtension = '.ps1'
                        mimeType = 'application/x-powershell'
                        break
                    case 'json':
                        fileName = 'data'
                        fileExtension = '.json'
                        mimeType = 'application/json'
                        break
                    case 'bat':
                        fileName = 'script'
                        fileExtension = '.bat'
                        mimeType = 'application/bat'
                        break
                    case 'md':
                    case 'markdown':
                        fileName = 'document'
                        fileExtension = '.md'
                        mimeType = 'text/markdown'
                        break
                    default:
                        fileName = 'code'
                        fileExtension = '.txt'
                        mimeType = 'text/plain'
                        break
                }
                break
        }
        // Create a Blob object with the code content
        const blob = new Blob([codeBlock.querySelector('code').textContent], { type: mimeType })
        try {
            if (window.showSaveFilePicker) {
                // Use File System Access API if available
                const fileHandle = await window.showSaveFilePicker({
                    suggestedName: fileName + fileExtension,
                    types: [
                        {
                            description: language,
                            accept: {
                                [mimeType]: [fileExtension]
                            }
                        }
                    ]
                })
                const writable = await fileHandle.createWritable()
                await writable.write(blob)
                await writable.close()
            } else {
                // Fallback for browsers that do not support showSaveFilePicker
                const downloadLink = document.createElement('a')
                downloadLink.href = URL.createObjectURL(blob)
                downloadLink.download = fileName + fileExtension
                downloadLink.style.display = 'none'
                document.body.appendChild(downloadLink)
                downloadLink.click()
                URL.revokeObjectURL(downloadLink.href)
                document.body.removeChild(downloadLink)
            }
        } catch (error) {
            console.error('Save file dialog was canceled or failed', error)
        }
    }
    setTimeout(() => {
        const observer = new MutationObserver(observeCodeBlocks)
        observer.observe(document.body, { childList: true, subtree: true })
        observeCodeBlocks()
    }, 5000)
})()