您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
从 ChatGPT 导出你的 GPTs 数据
- // ==UserScript==
- // @name ChatGPT GPTs Exporter
- // @name:zh-CN ChatGPT GPTs 导出工具
- // @name:zh-TW ChatGPT GPTs 導出工具
- // @namespace https://github.com/lroolle/chatgpt-degraded
- // @version 0.1.1
- // @description Export your GPTs data from ChatGPT
- // @description:zh-CN 从 ChatGPT 导出你的 GPTs 数据
- // @description:zh-TW 從 ChatGPT 導出你的 GPTs 數據
- // @author lroolle
- // @license AGPL-3.0
- // @match *://chat.openai.com/gpts/*
- // @match *://chatgpt.com/gpts/*
- // @grant GM_xmlhttpRequest
- // @grant GM_registerMenuCommand
- // @grant GM_setClipboard
- // @grant unsafeWindow
- // @run-at document-start
- // @icon 
- // @homepageURL https://github.com/lroolle/chatgpt-degraded
- // @supportURL https://github.com/lroolle/chatgpt-degraded/issues
- // ==/UserScript==
- (function() {
- 'use strict';
- // Store GPTs data
- let gptsData = [];
- let isExporting = false;
- // Storage key for persisted data
- const STORAGE_KEY = 'chatgpt_gpts_data';
- // Load persisted data
- function loadPersistedData() {
- try {
- const stored = localStorage.getItem(STORAGE_KEY);
- return stored ? JSON.parse(stored) : [];
- } catch (error) {
- console.error('Error loading persisted GPTs data:', error);
- return [];
- }
- }
- // Save data to persistence
- function persistData(data) {
- try {
- // Create a map of existing data by ID
- const existingData = loadPersistedData();
- const dataMap = new Map(existingData.map(item => [item.id, item]));
- // Update or add new data
- data.forEach(item => {
- dataMap.set(item.id, item);
- });
- // Convert map back to array and save
- const mergedData = Array.from(dataMap.values());
- localStorage.setItem(STORAGE_KEY, JSON.stringify(mergedData));
- return mergedData;
- } catch (error) {
- console.error('Error persisting GPTs data:', error);
- return data;
- }
- }
- // Clear persisted data
- function clearPersistedData() {
- try {
- localStorage.removeItem(STORAGE_KEY);
- } catch (error) {
- console.error('Error clearing persisted GPTs data:', error);
- }
- }
- // i18n support
- const i18n = {
- 'en': {
- exportBtn: 'Export GPTs',
- exportJSON: 'Export as JSON',
- exportCSV: 'Export as CSV',
- clearData: 'Clear Data',
- exporting: 'Exporting...',
- copied: 'Copied to clipboard!',
- noData: 'No GPTs data found',
- error: 'Error exporting GPTs',
- dataCleared: 'Data cleared successfully',
- totalGPTs: 'Total GPTs: '
- },
- 'zh-CN': {
- exportBtn: '导出 GPTs',
- exportJSON: '导出为 JSON',
- exportCSV: '导出为 CSV',
- clearData: '清除数据',
- exporting: '导出中...',
- copied: '已复制到剪贴板!',
- noData: '未找到 GPTs 数据',
- error: '导出 GPTs 时出错',
- dataCleared: '数据已清除',
- totalGPTs: '总计 GPTs: '
- },
- 'zh-TW': {
- exportBtn: '導出 GPTs',
- exportJSON: '導出為 JSON',
- exportCSV: '導出為 CSV',
- clearData: '清除數據',
- exporting: '導出中...',
- copied: '已複製到剪貼板!',
- noData: '未找到 GPTs 數據',
- error: '導出 GPTs 時出錯',
- dataCleared: '數據已清除',
- totalGPTs: '總計 GPTs: '
- }
- };
- // Get user language
- const userLang = (navigator.language || 'en').toLowerCase();
- const lang = i18n[userLang] ? userLang :
- userLang.startsWith('zh-tw') ? 'zh-TW' :
- userLang.startsWith('zh') ? 'zh-CN' : 'en';
- const t = key => i18n[lang][key] || i18n.en[key];
- // Intercept fetch requests
- const originalFetch = unsafeWindow.fetch;
- unsafeWindow.fetch = async function(resource, options) {
- const response = await originalFetch(resource, options);
- const url = typeof resource === 'string' ? resource : resource?.url;
- // Check if this is the GPTs list request
- if (url && url.includes('/public-api/gizmos/discovery/mine')) {
- try {
- const clonedResponse = response.clone();
- const data = await clonedResponse.json();
- if (data?.list?.items) {
- const newGPTsData = data.list.items.map(item => {
- const gpt = item.resource.gizmo;
- return {
- id: gpt.id,
- name: gpt.display.name || '',
- description: gpt.display.description || '',
- instructions: gpt.instructions || '',
- created_at: gpt.created_at,
- updated_at: gpt.updated_at,
- version: gpt.version,
- tools: item.resource.tools.map(tool => tool.type),
- prompt_starters: gpt.display.prompt_starters || [],
- share_recipient: gpt.share_recipient,
- num_interactions: gpt.num_interactions
- };
- });
- // Persist and update the data
- gptsData = persistData(newGPTsData);
- // Update the counter in UI
- updateGPTsCounter(gptsData.length);
- }
- } catch (error) {
- console.error('Error intercepting GPTs data:', error);
- }
- }
- return response;
- };
- // Update GPTs counter in UI
- function updateGPTsCounter(count) {
- const counter = document.getElementById('gpts-counter');
- if (counter) {
- counter.textContent = `${t('totalGPTs')}${count}`;
- }
- }
- // Convert GPTs data to CSV
- function convertToCSV(data) {
- const headers = [
- 'ID',
- 'Name',
- 'Description',
- 'Instructions',
- 'Created At',
- 'Updated At',
- 'Version',
- 'Tools',
- 'Prompt Starters',
- 'Share Recipient',
- 'Interactions'
- ];
- const rows = data.map(gpt => [
- gpt.id,
- `"${(gpt.name || '').replace(/"/g, '""')}"`,
- `"${(gpt.description || '').replace(/"/g, '""')}"`,
- `"${(gpt.instructions || '').replace(/"/g, '""')}"`,
- gpt.created_at,
- gpt.updated_at,
- gpt.version,
- `"${(gpt.tools || []).join(', ')}"`,
- `"${(gpt.prompt_starters || []).join(', ').replace(/"/g, '""')}"`,
- gpt.share_recipient,
- gpt.num_interactions
- ]);
- return [headers.join(','), ...rows.map(row => row.join(','))].join('\n');
- }
- // Export GPTs data
- function exportGPTs(format = 'json') {
- if (isExporting) return;
- isExporting = true;
- try {
- // Load all persisted data
- const allData = loadPersistedData();
- if (!allData.length) {
- alert(t('noData'));
- isExporting = false;
- return;
- }
- let exportContent, filename, mimeType;
- if (format === 'csv') {
- exportContent = convertToCSV(allData);
- filename = `chatgpt-gpts-export-${new Date().toISOString().split('T')[0]}.csv`;
- mimeType = 'text/csv';
- } else {
- const exportData = {
- exported_at: new Date().toISOString(),
- total_gpts: allData.length,
- gpts: allData
- };
- exportContent = JSON.stringify(exportData, null, 2);
- filename = `chatgpt-gpts-export-${new Date().toISOString().split('T')[0]}.json`;
- mimeType = 'application/json';
- }
- GM_setClipboard(exportContent);
- // Create and download file
- const blob = new Blob([exportContent], { type: mimeType });
- const url = URL.createObjectURL(blob);
- const a = document.createElement('a');
- a.href = url;
- a.download = filename;
- document.body.appendChild(a);
- a.click();
- document.body.removeChild(a);
- URL.revokeObjectURL(url);
- alert(t('copied'));
- } catch (error) {
- console.error('Error exporting GPTs:', error);
- alert(t('error'));
- } finally {
- isExporting = false;
- }
- }
- // Register menu commands
- GM_registerMenuCommand(t('exportJSON'), () => exportGPTs('json'));
- GM_registerMenuCommand(t('exportCSV'), () => exportGPTs('csv'));
- // Add export button to UI
- function addExportButton() {
- const styles = document.createElement('style');
- styles.textContent = `
- .gpts-exporter {
- position: fixed;
- bottom: 20px;
- right: 20px;
- z-index: 10000;
- display: flex;
- flex-direction: column;
- gap: 8px;
- background: var(--surface-primary, rgba(255, 255, 255, 0.9));
- padding: 12px;
- border-radius: 8px;
- box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
- border: 1px solid var(--border-light, rgba(0, 0, 0, 0.1));
- transition: transform 0.2s ease, opacity 0.2s ease;
- }
- .gpts-exporter:hover {
- transform: translateY(-2px);
- }
- .gpts-exporter.collapsed {
- transform: translateX(calc(100% + 20px));
- }
- .gpts-exporter button {
- padding: 8px 16px;
- background-color: var(--success-color, #10a37f);
- color: #fff;
- border: none;
- border-radius: 4px;
- cursor: pointer;
- font-size: 14px;
- font-weight: 500;
- transition: all 0.2s ease;
- display: flex;
- align-items: center;
- justify-content: center;
- gap: 6px;
- min-width: 140px;
- }
- .gpts-exporter button:hover {
- opacity: 0.9;
- transform: translateY(-1px);
- }
- .gpts-exporter button:active {
- transform: translateY(0);
- }
- .gpts-exporter .toggle-btn {
- position: absolute;
- left: -32px;
- top: 50%;
- transform: translateY(-50%);
- width: 24px;
- height: 24px;
- background: var(--success-color, #10a37f);
- border: none;
- border-radius: 4px 0 0 4px;
- cursor: pointer;
- display: flex;
- align-items: center;
- justify-content: center;
- padding: 0;
- min-width: unset;
- }
- .gpts-exporter .toggle-btn svg {
- width: 16px;
- height: 16px;
- fill: #fff;
- transition: transform 0.2s ease;
- }
- .gpts-exporter.collapsed .toggle-btn svg {
- transform: rotate(180deg);
- }
- .gpts-exporter button.json-btn {
- background-color: var(--success-color, #10a37f);
- }
- .gpts-exporter button.csv-btn {
- background-color: var(--primary-color, #0ea5e9);
- }
- .gpts-exporter .counter {
- font-size: 12px;
- color: var(--text-secondary, #666);
- text-align: center;
- margin-bottom: 4px;
- }
- .gpts-exporter button.clear-btn {
- background-color: var(--error-color, #dc2626);
- font-size: 12px;
- padding: 4px 8px;
- }
- `;
- document.head.appendChild(styles);
- const container = document.createElement('div');
- container.className = 'gpts-exporter';
- // Add GPTs counter
- const counter = document.createElement('div');
- counter.id = 'gpts-counter';
- counter.className = 'counter';
- counter.textContent = `${t('totalGPTs')}${loadPersistedData().length}`;
- const toggleBtn = document.createElement('button');
- toggleBtn.className = 'toggle-btn';
- toggleBtn.innerHTML = '<svg viewBox="0 0 24 24"><path d="M15.41 7.41L14 6l-6 6 6 6 1.41-1.41L10.83 12z"/></svg>';
- toggleBtn.addEventListener('click', () => {
- container.classList.toggle('collapsed');
- });
- const jsonBtn = document.createElement('button');
- jsonBtn.className = 'json-btn';
- jsonBtn.innerHTML = `
- <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
- <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"/>
- <polyline points="7 10 12 15 17 10"/>
- <line x1="12" y1="15" x2="12" y2="3"/>
- </svg>
- ${t('exportJSON')}
- `;
- jsonBtn.addEventListener('click', () => exportGPTs('json'));
- const csvBtn = document.createElement('button');
- csvBtn.className = 'csv-btn';
- csvBtn.innerHTML = `
- <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2">
- <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z"/>
- <polyline points="14 2 14 8 20 8"/>
- <line x1="16" y1="13" x2="8" y2="13"/>
- <line x1="16" y1="17" x2="8" y2="17"/>
- <polyline points="10 9 9 9 8 9"/>
- </svg>
- ${t('exportCSV')}
- `;
- csvBtn.addEventListener('click', () => exportGPTs('csv'));
- // Add clear data button
- const clearBtn = document.createElement('button');
- clearBtn.className = 'clear-btn';
- clearBtn.textContent = t('clearData');
- clearBtn.addEventListener('click', () => {
- if (confirm(t('clearData') + '?')) {
- clearPersistedData();
- gptsData = [];
- updateGPTsCounter(0);
- alert(t('dataCleared'));
- }
- });
- container.appendChild(counter);
- container.appendChild(toggleBtn);
- container.appendChild(jsonBtn);
- container.appendChild(csvBtn);
- container.appendChild(clearBtn);
- document.body.appendChild(container);
- }
- // Initialize
- if (document.readyState === 'loading') {
- document.addEventListener('DOMContentLoaded', addExportButton);
- } else {
- addExportButton();
- }
- })();