您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
将 ChatGPT 和 Grok 网站的聊天记录导出为普通 Markdown 格式,可在 Typora 中准确打开。
// ==UserScript== // @name Export ChatGPT/Gemini/Grok conversations as Markdown // @name:zh-CN Export ChatGPT/Gemini/Grok conversations as Markdown // @name:zh-TW Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ar Export ChatGPT/Gemini/Grok conversations as Markdown // @name:bg Export ChatGPT/Gemini/Grok conversations as Markdown // @name:cs Export ChatGPT/Gemini/Grok conversations as Markdown // @name:da Export ChatGPT/Gemini/Grok conversations as Markdown // @name:de Export ChatGPT/Gemini/Grok conversations as Markdown // @name:el Export ChatGPT/Gemini/Grok conversations as Markdown // @name:en Export ChatGPT/Gemini/Grok conversations as Markdown // @name:eo Export ChatGPT/Gemini/Grok conversations as Markdown // @name:es Export ChatGPT/Gemini/Grok conversations as Markdown // @name:es-419 Export ChatGPT/Gemini/Grok conversations as Markdown // @name:fi Export ChatGPT/Gemini/Grok conversations as Markdown // @name:fr Export ChatGPT/Gemini/Grok conversations as Markdown // @name:fr-CA Export ChatGPT/Gemini/Grok conversations as Markdown // @name:he Export ChatGPT/Gemini/Grok conversations as Markdown // @name:hr Export ChatGPT/Gemini/Grok conversations as Markdown // @name:hu Export ChatGPT/Gemini/Grok conversations as Markdown // @name:id Export ChatGPT/Gemini/Grok conversations as Markdown // @name:it Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ja Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ka Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ko Export ChatGPT/Gemini/Grok conversations as Markdown // @name:nb Export ChatGPT/Gemini/Grok conversations as Markdown // @name:nl Export ChatGPT/Gemini/Grok conversations as Markdown // @name:pl Export ChatGPT/Gemini/Grok conversations as Markdown // @name:pt-BR Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ro Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ru Export ChatGPT/Gemini/Grok conversations as Markdown // @name:sv Export ChatGPT/Gemini/Grok conversations as Markdown // @name:th Export ChatGPT/Gemini/Grok conversations as Markdown // @name:tr Export ChatGPT/Gemini/Grok conversations as Markdown // @name:uk Export ChatGPT/Gemini/Grok conversations as Markdown // @name:ug Export ChatGPT/Gemini/Grok conversations as Markdown // @name:vi Export ChatGPT/Gemini/Grok conversations as Markdown // @description Export chat history from ChatGPT and Grok websites to normal format as Markdown, which can be opened via typora exactly. // @description:zh-CN 将 ChatGPT 和 Grok 网站的聊天记录导出为普通 Markdown 格式,可在 Typora 中准确打开。 // @description:zh-TW 將 ChatGPT 和 Grok 網站的聊天記錄導出為普通 Markdown 格式,可於 Typora 中精確打開。 // @description:ar تصدير محفوظات الدردشة من موقعي ChatGPT و Grok إلى تنسيق Markdown عادي يمكن فتحه عبر Typora بدقة. // @description:bg Експортиране на историята на чатовете от уебсайтовете ChatGPT и Grok в обикновен Markdown формат, който може да се отвори точно чрез Typora. // @description:cs Exportovat historii chatu z webů ChatGPT a Grok do obyčejného formátu Markdown, který lze přesně otevřít v Typora. // @description:da Eksportér chat‑historik fra ChatGPT- og Grok‑websteder til normalt Markdown‑format, som kan åbnes præcist i Typora. // @description:de Exportiere den Chatverlauf von den ChatGPT‑ und Grok‑Websites in ein normales Markdown‑Format, das exakt in Typora geöffnet werden kann. // @description:el Εξαγωγή του ιστορικού συνομιλίας από τις ιστοσελίδες ChatGPT και Grok σε απλό μορφότυπο Markdown, που μπορεί να ανοιχτεί ακριβώς με το Typora. // @description:en Export chat history from ChatGPT and Grok websites to normal format as Markdown, which can be opened via typora exactly. // @description:eo Eksporti babilhistorion de ChatGPT- kaj Grok‑retejoj al norma Markdown‑formato, kiu povas esti malfermita ĝuste en Typora. // @description:es Exportar el historial de chat de los sitios web ChatGPT y Grok a un formato normal Markdown, que se pueda abrir exactamente con Typora. // @description:es-419 Exportar el historial de chat de los sitios web ChatGPT y Grok a un formato normal Markdown, que se pueda abrir exactamente con Typora. // @description:fi Vie ChatGPT:n ja Grok-verkkosivustojen chätähistoria tavalliseen Markdown‑muotoon, jonka Typora avaa täsmälleen oikein. // @description:fr Exporter l’historique de discussion des sites ChatGPT et Grok vers un format Markdown normal, pouvant être ouvert exactement avec Typora. // @description:fr-CA Exporter l’historique des conversations des sites ChatGPT et Grok dans un format Markdown standard, pouvant être ouvert précisément avec Typora. // @description:he ייצוא היסטוריית הצ'אט מאתרי ChatGPT ו‑Grok לפורמט Markdown רגיל, שניתן לפתוח ב‑Typora בדיוק. // @description:hr Izvezi povijest razgovora s web‑mjesta ChatGPT i Grok u obični Markdown format, koji se točno može otvoriti u Typora. // @description:hu A ChatGPT és Grok webhelyek csevegési előzményeinek exportálása normál Markdown formátumba, amely pontosan megnyitható a Typora segítségével. // @description:id Ekspor riwayat obrolan dari situs web ChatGPT dan Grok ke format Markdown biasa, yang bisa dibuka persis dengan Typora. // @description:it Esporta la cronologia della chat dai siti ChatGPT e Grok in formato Markdown normale, che può essere aperto esattamente con Typora. // @description:ja ChatGPT と Grok のウェブサイトからチャット履歴を通常の Markdown 形式にエクスポートし、Typora で正確に開けるようにします。 // @description:ka ChatGPT და Grok ვებსაიტების ჩატის ისტორიის ექსპორტი ნორმალურ Markdown ფორმატში, რომელიც Typora-ში ზუსტად იხსნება. // @description:ko ChatGPT 및 Grok 웹사이트의 채팅 기록을 일반 Markdown 형식으로 내보내 Typora에서 정확하게 열 수 있습니다. // @description:nb Eksporter chattehistorikk fra ChatGPT- og Grok-nettsteder til vanlig Markdown-format, som kan åpnes nøyaktig i Typora. // @description:nl Exporteer de chatgeschiedenis van de ChatGPT- en Grok-websites naar normaal Markdown‑formaat, dat precies met Typora geopend kan worden. // @description:pl Eksportuj historię czatów ze stron ChatGPT i Grok do zwykłego formatu Markdown, który można dokładnie otworzyć w Typora. // @description:pt-BR Exporte o histórico de conversa dos sites ChatGPT e Grok para um formato Markdown normal, que possa ser aberto exatamente via Typora. // @description:ro Exportă istoricul conversației de pe site‑urile ChatGPT și Grok într‑un format Markdown normal, care poate fi deschis exact în Typora. // @description:ru Экспорт истории чата с сайтов ChatGPT и Grok в обычный формат Markdown, который можно точно открыть в Typora. // @description:sv Exportera chatthistorik från ChatGPT- och Grok-webbplatser till vanligt Markdown-format, som kan öppnas exakt i Typora. // @description:th ส่งออกประวัติแชทจากเว็บไซต์ ChatGPT และ Grok ไปยังรูปแบบ Markdown ปกติ ซึ่งสามารถเปิดได้อย่างถูกต้องผ่าน Typora // @description:tr ChatGPT ve Grok web sitelerinden sohbet geçmişini normal Markdown formatına aktarın; Typora ile tam olarak açılabilir. // @description:uk Експортувати історію чату з веб‑сайтів ChatGPT і Grok у звичайний формат Markdown, який точно відкривається в Typora. // @description:ug ChatGPT ۋە Grok تور بەتلىرىدىكى چەت تارىخنى ئادەتتىكى Markdown فورماتىغا چىقارسىڭىز بولىدۇ، بۇنى Typora قوشۇلمىسىدە تولۇق ئاچىشقا بولىدۇ. // @description:vi Xuất lịch sử trò chuyện từ các trang ChatGPT và Grok sang định dạng Markdown bình thường, có thể mở chính xác bằng Typora. // @namespace Elior_Chatgpt_XX // @version 1.1.1 // @author Elior // @icon  // @include *://chatgpt.com/* // @include *://grok.com/* // @include *://gemini.google.com/* // @noframes // @license MIT // @run-at document-idle // @grant GM_registerMenuCommand // @grant GM_openInTab // @grant GM.openInTab // @grant GM_addStyle // @grant GM_setValue // @grant GM_getValue // @grant GM_xmlhttpRequest // ==/UserScript== (function () { 'use strict'; /*! * Copyright (c) 2024 - 2025, Elior. All rights reserved. * * Permission is hereby granted, free of charge, to any person obtaining a copy * of this software and associated documentation files (the "Software"), to deal * in the Software without restriction, including without limitation the rights * to use, copy, modify, merge, publish, distribute, sublicense, and/or sell * copies of the Software, and to permit persons to whom the Software is * furnished to do so, subject to the following conditions: * * The above copyright notice and this permission notice shall be included in * all copies or substantial portions of the Software. * * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, * * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE * AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, * OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE * SOFTWARE. */ const CommonUtil = { onPageLoad: function(callback) { if (document.readyState === "complete") { callback(); } else { window.addEventListener("DOMContentLoaded", callback, { once: true }); window.addEventListener("load", callback, { once: true }); } }, addStyle: function(style) { GM_addStyle(style); }, createElement: function(tag, options = {}) { const element = document.createElement(tag); if (options.text) { element.textContent = options.text; } if (options.html) { element.innerHTML = options.html; } if (options.style) { Object.assign(element.style, options.style); } if (options.className) { element.className = options.className; } if (options.attributes) { for (let [key, value] of Object.entries(options.attributes)) { element.setAttribute(key, value); } } if (options.childrens) { options.childrens.forEach((child) => { element.appendChild(child); }); } return element; }, openInTab: function(url, options = { "active": true, "insert": true, "setParent": true }) { if (typeof GM_openInTab === "function") { GM_openInTab(url, options); } else { GM.openInTab(url, options); } }, waitForElementByInterval: function(selector, target = document.body, allowEmpty = true, delay = 10, maxDelay = 10 * 1e3) { return new Promise((resolve, reject) => { let totalDelay = 0; let element = target.querySelector(selector); let result = allowEmpty ? !!element : !!element && !!element.innerHTML; if (result) { resolve(element); } const elementInterval = setInterval(() => { if (totalDelay >= maxDelay) { clearInterval(elementInterval); resolve(null); } element = target.querySelector(selector); result = allowEmpty ? !!element : !!element && !!element.innerHTML; if (result) { clearInterval(elementInterval); resolve(element); } else { totalDelay += delay; } }, delay); }); } }; const HtmlToMarkdown = { to: function(html, platform) { const parser = new DOMParser(); const doc = parser.parseFromString(html, "text/html"); const isChatGPT = platform === "chatGPT", isGemini = platform === "gemini", isGrok = platform === "grok"; if (!isGemini) { doc.querySelectorAll("span.katex-html").forEach((element) => element.remove()); } doc.querySelectorAll("mrow").forEach((mrow) => mrow.remove()); doc.querySelectorAll('annotation[encoding="application/x-tex"]').forEach((element) => { if (element.closest(".katex-display")) { const latex = element.textContent; const trimmedLatex = latex.trim(); element.replaceWith(` $$ ${trimmedLatex} $$ `); } else { const latex = element.textContent; const trimmedLatex = latex.trim(); element.replaceWith(`$${trimmedLatex}$`); } }); doc.querySelectorAll("strong, b").forEach((bold) => { const markdownBold = `**${bold.textContent}**`; bold.parentNode.replaceChild(document.createTextNode(markdownBold), bold); }); doc.querySelectorAll("em, i").forEach((italic) => { const markdownItalic = `*${italic.textContent}*`; italic.parentNode.replaceChild(document.createTextNode(markdownItalic), italic); }); doc.querySelectorAll("p code").forEach((code) => { const markdownCode = `\`${code.textContent}\``; code.parentNode.replaceChild(document.createTextNode(markdownCode), code); }); doc.querySelectorAll("a").forEach((link) => { const markdownLink = `[${link.textContent}](${link.href})`; link.parentNode.replaceChild(document.createTextNode(markdownLink), link); }); doc.querySelectorAll("img").forEach((img) => { const markdownImage = ``; img.parentNode.replaceChild(document.createTextNode(markdownImage), img); }); if (isChatGPT) { doc.querySelectorAll("pre").forEach((pre) => { const codeType = pre.querySelector("div > div:first-child")?.textContent || ""; const markdownCode = pre.querySelector("div > div:nth-child(3) > code")?.textContent || pre.textContent; pre.innerHTML = ` \`\`\`${codeType} ${markdownCode} \`\`\``; }); } else if (isGrok) { doc.querySelectorAll("div.not-prose").forEach((div) => { const codeType = div.querySelector("div > div > span")?.textContent || ""; const markdownCode = div.querySelector("div > div:nth-child(3) > code")?.textContent || div.textContent; div.innerHTML = ` \`\`\`${codeType} ${markdownCode} \`\`\``; }); } else if (isGemini) { doc.querySelectorAll("code-block").forEach((div) => { const codeType = div.querySelector("div > div > span")?.textContent || ""; const markdownCode = div.querySelector("div > div:nth-child(2) > div > pre")?.textContent || div.textContent; div.innerHTML = ` \`\`\`${codeType} ${markdownCode} \`\`\``; }); } doc.querySelectorAll("ul").forEach((ul) => { let markdown2 = ""; ul.querySelectorAll(":scope > li").forEach((li) => { markdown2 += `- ${li.textContent.trim()} `; }); ul.parentNode.replaceChild(document.createTextNode("\n" + markdown2.trim()), ul); }); doc.querySelectorAll("ol").forEach((ol) => { let markdown2 = ""; ol.querySelectorAll(":scope > li").forEach((li, index) => { markdown2 += `${index + 1}. ${li.textContent.trim()} `; }); ol.parentNode.replaceChild(document.createTextNode("\n" + markdown2.trim()), ol); }); for (let i = 1; i <= 6; i++) { doc.querySelectorAll(`h${i}`).forEach((header) => { const markdownHeader = ` ${"#".repeat(i)} ${header.textContent} `; header.parentNode.replaceChild(document.createTextNode(markdownHeader), header); }); } doc.querySelectorAll("p").forEach((p) => { const markdownParagraph = "\n" + p.textContent + "\n"; p.parentNode.replaceChild(document.createTextNode(markdownParagraph), p); }); doc.querySelectorAll("table").forEach((table) => { let markdown2 = ""; table.querySelectorAll("thead tr").forEach((tr) => { tr.querySelectorAll("th").forEach((th) => { markdown2 += `| ${th.textContent} `; }); markdown2 += "|\n"; tr.querySelectorAll("th").forEach(() => { markdown2 += "| ---- "; }); markdown2 += "|\n"; }); table.querySelectorAll("tbody tr").forEach((tr) => { tr.querySelectorAll("td").forEach((td) => { markdown2 += `| ${td.textContent} `; }); markdown2 += "|\n"; }); table.parentNode.replaceChild(document.createTextNode("\n" + markdown2.trim() + "\n"), table); }); let markdown = doc.body.innerHTML.replace(/<[^>]*>/g, ""); markdown = markdown.replaceAll(/- >/g, "- $\\gt$"); markdown = markdown.replaceAll(/>/g, ">"); markdown = markdown.replaceAll(/</g, "<"); markdown = markdown.replaceAll(/≥/g, ">="); markdown = markdown.replaceAll(/≤/g, "<="); markdown = markdown.replaceAll(/≠/g, "\\neq"); return markdown.trim(); } }; const Download = { start: function(data, filename, type) { var file = new Blob([data], { type }); if (window.navigator.msSaveOrOpenBlob) { window.navigator.msSaveOrOpenBlob(file, filename); } else { var a = document.createElement("a"), url = URL.createObjectURL(file); a.href = url; a.download = filename; document.body.appendChild(a); a.click(); setTimeout(function() { document.body.removeChild(a); window.URL.revokeObjectURL(url); }, 0); } } }; const Chat = { sanitizeFilename: function(input, replacement = "_") { const illegalRe = /[\/\\\?\%\*\:\|"<>\.]/g; const controlRe = /[\x00-\x1f\x80-\x9f]/g; const reservedRe = /^\.+$/; const windowsReservedRe = /^(con|prn|aux|nul|com[1-9]|lpt[1-9])$/i; let name = input.replace(illegalRe, replacement).replace(controlRe, replacement).replace(/\s+/g, " ").trim(); if (reservedRe.test(name)) name = "file"; if (windowsReservedRe.test(name)) name = `file_${name}`; return name || "untitled"; }, getConversationElements: function() { const currentUrl = window.location.href; const result = []; let platform = ""; let title = ""; if (currentUrl.includes("openai.com") || currentUrl.includes("chatgpt.com")) { platform = "chatGPT"; title = document.querySelector("#history a[data-active]")?.textContent; result.push(...document.querySelectorAll("div[data-message-id]")); } else if (currentUrl.includes("grok.com")) { platform = "grok"; result.push(...document.querySelectorAll("div.message-bubble")); } else if (currentUrl.includes("gemini.google.com")) { platform = "gemini"; title = document.querySelector("conversations-list div.selected")?.textContent; const userQueries = document.querySelectorAll("user-query-content"); const modelResponses = document.querySelectorAll("model-response"); for (let i = 0; i < userQueries.length; i++) { if (i < modelResponses.length) { result.push(userQueries[i]); result.push(modelResponses[i]); } else { result.push(userQueries[i]); } } } return { "result": result, "platform": platform, "title": title }; }, exportChatAsMarkdown: function() { let markdownContent = ""; const { result, platform, title } = this.getConversationElements(); const filename = (this.sanitizeFilename(title) || "chat_export") + ".md"; for (let i = 0; i < result.length; i += 2) { if (!result[i + 1]) break; let userText = result[i].textContent.trim(); let answerHtml = result[i + 1].innerHTML.trim(); userText = HtmlToMarkdown.to(userText, platform); answerHtml = HtmlToMarkdown.to(answerHtml, platform); markdownContent += ` # Q: ${userText} # A: ${answerHtml}`; } markdownContent = markdownContent.replace(/&/g, "&"); if (markdownContent) { Download.start(markdownContent, filename, "text/markdown"); } } }; var css_248z = ".chat-gpt-document-block{align-items:center;border:1px solid #e5e5e5;border-radius:35px;cursor:pointer;display:flex;font-size:15px;justify-content:center;left:50%;padding:5px 15px;position:fixed;top:9px;transform:translateX(-50%);z-index:99999999999!important}.chat-gpt-document-icon-sm{margin-right:5px}.chat-gpt-document-btn-content{align-items:center;display:flex}"; const Export = { addStyle: function() { CommonUtil.addStyle(css_248z); }, createSvgIcon: function() { const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg"); svg.setAttribute("class", "chat-gpt-document-icon-sm chat-gpt-document-btn-content"); svg.setAttribute("xmlns", "http://www.w3.org/2000/svg"); svg.setAttribute("fill", "none"); svg.setAttribute("viewBox", "0 0 24 24"); svg.setAttribute("width", "16"); svg.setAttribute("height", "16"); svg.setAttribute("stroke-width", "1.5"); svg.setAttribute("stroke", "currentColor"); const path = document.createElementNS("http://www.w3.org/2000/svg", "path"); path.setAttribute("stroke-linecap", "round"); path.setAttribute("stroke-linejoin", "round"); path.setAttribute("d", "M19.5 14.25v-2.625a3.375 3.375 0 0 0-3.375-3.375h-1.5A1.125 1.125 0 0 1 13.5 7.125v-1.5a3.375 3.375 0 0 0-3.375-3.375H8.25m.75 12 3 3m0 0 3-3m-3 3v-6m-1.5-9H5.625c-.621 0-1.125.504-1.125 1.125v17.25c0 .621.504 1.125 1.125 1.125h12.75c.621 0 1.125-.504 1.125-1.125V11.25a9 9 0 0 0-9-9Z"); svg.appendChild(path); document.body.appendChild(svg); return svg; }, generateHtml: function() { const outerDiv = CommonUtil.createElement("div", { className: "chat-gpt-document-block", childrens: [ this.createSvgIcon(), CommonUtil.createElement("div", { className: "chat-gpt-document-btn-content", text: "Save As PDF" }) ] }); (document.body || document.documentElement).appendChild(outerDiv); outerDiv.addEventListener("click", function() { Chat.exportChatAsMarkdown(); }); }, start: function() { this.addStyle(); this.generateHtml(); } }; (() => { if (typeof trustedTypes !== "undefined" && trustedTypes.defaultPolicy === null) { let s = (s2) => s2; trustedTypes.createPolicy("default", { createHTML: s, createScriptURL: s, createScript: s }); } })(); Export.start(); }());
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址