Mistral AI 채팅 내보내기

Mistral AI 채팅 대화를 마크다운 형식으로 내보내기

이 스크립트를 설치하려면 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         Mistral AI Chat Exporter
// @name:en      Mistral AI Chat Exporter
// @name:zh      Mistral AI 聊天导出器
// @name:zh-CN   Mistral AI 聊天导出器
// @name:zh-TW   Mistral AI 聊天匯出器
// @name:ja      Mistral AI チャットエクスポーター
// @name:es      Exportador de Chat de Mistral AI
// @name:fr      Exportateur de Chat Mistral AI
// @name:de      Mistral AI Chat-Exporteur
// @name:it      Esportatore Chat Mistral AI
// @name:ru      Экспортёр чата Mistral AI
// @name:pt      Exportador de Chat Mistral AI
// @name:ko      Mistral AI 채팅 내보내기
// @name:ar      مصدر دردشة Mistral AI
// @name:hi      Mistral AI चैट निर्यातक
// @name:el      Εξαγωγέας Συνομιλιών Mistral AI
// @description  Export Mistral AI chat conversations to markdown
// @description:en Export Mistral AI chat conversations to markdown format
// @description:zh 导出Mistral AI聊天对话为Markdown格式
// @description:zh-CN 导出Mistral AI聊天对话为Markdown格式
// @description:zh-TW 匯出 Mistral AI 聊天對話為 Markdown 格式
// @description:ja Mistral AIのチャット会話をMarkdown形式でエクスポート
// @description:es Exportar conversaciones de chat de Mistral AI a formato markdown
// @description:fr Exporter les conversations de chat Mistral AI au format markdown
// @description:de Mistral AI Chat-Unterhaltungen in Markdown-Format exportieren
// @description:it Esporta conversazioni chat di Mistral AI in formato markdown
// @description:ru Экспорт разговоров чата Mistral AI в формат markdown
// @description:pt Exportar conversas de chat do Mistral AI para formato markdown
// @description:ko Mistral AI 채팅 대화를 마크다운 형식으로 내보내기
// @description:ar تصدير محادثات دردشة Mistral AI إلى تنسيق markdown
// @description:hi Mistral AI चैट वार्तालापों को markdown प्रारूप में निर्यात करें
// @description:el Εξαγωγή συνομιλιών του Mistral AI σε μορφή markdown
// @namespace    http://tampermonkey.net/
// @version      1.0.2
// @author       aspen138
// @match        https://chat.mistral.ai/chat/*
// @grant        none
// @license      MIT
// @icon         
// ==/UserScript==

(function () {
    'use strict';

    // Create export button
    function createExportButton() {
        if (document.getElementById('mistral-export-button')) return;

        const button = document.createElement('button');
        button.id = 'mistral-export-button';
        button.innerHTML = `
            <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round" style="margin-right: 6px;">
                <path d="M21 15v4a2 2 0 0 1-2 2H5a2 2 0 0 1-2-2v-4"></path>
                <polyline points="7 10 12 15 17 10"></polyline>
                <line x1="12" y1="15" x2="12" y2="3"></line>
            </svg>
            Export
        `;

        // Default fallback styles
        let styles = {
            position: 'fixed',
            top: '0.5rem',
            right: '5.5rem', // Positioned to the left of the user profile/menu usually in top right
            zIndex: '9999',
            display: 'flex',
            alignItems: 'center',
            justifyContent: 'center',
            padding: '8px 12px',
            backgroundColor: '#fff', // Fallback
            color: '#000',      // Fallback
            border: '1px solid #e5e5e5', // Fallback
            borderRadius: '0.5rem',
            fontSize: '0.875rem',
            fontWeight: '500',
            cursor: 'pointer',
            transition: 'all 0.2s cubic-bezier(0.4, 0, 0.2, 1)',
            boxShadow: '0 1px 2px 0 rgba(0, 0, 0, 0.05)'
        };

        // element to mimic (look for a button in the header or shared styles)
        // Trying to find a "Share" button or similar secondary button in the header
        const selectorsToTry = [
            'header button[class*="secondary"]',
            'header button',
            'button[aria-label="New chat"]',
            '.flex.gap-2 button'
        ];

        let referenceBtn = null;
        for (const sel of selectorsToTry) {
            const found = document.querySelector(sel);
            if (found && found.offsetParent !== null) { // Check if visible
                referenceBtn = found;
                break;
            }
        }

        // If we found a reference button, try to copy its computed styles
        if (referenceBtn) {
            try {
                const computed = window.getComputedStyle(referenceBtn);
                styles.backgroundColor = computed.backgroundColor;
                styles.color = computed.color;
                styles.borderRadius = computed.borderRadius;
                styles.fontFamily = computed.fontFamily;
                styles.fontSize = computed.fontSize;
                styles.border = computed.border;
                // If the reference button has no border, keep our default or check box-shadow
                if (computed.borderWidth === '0px' && !computed.boxShadow) {
                    styles.border = '1px solid transparent'; // Mimic clean look but ensure visibility if transparent bg
                }

                // If background is transparent, it might be an icon-only button or rely on parents.
                // In that case, we might want to default to a "surface" look.
                if (computed.backgroundColor === 'rgba(0, 0, 0, 0)' || computed.backgroundColor === 'transparent') {
                    styles.backgroundColor = '#ffffff';
                    styles.border = '1px solid #e5e7eb';
                }
            } catch (e) {
                console.warn('Mistral Exporter: Could not copy styles', e);
            }
        }

        // Apply styles
        Object.assign(button.style, styles);

        // Add hover effect logic manually since we can't easily copy :hover state
        button.onmouseover = () => {
            button.style.opacity = '0.9';
            button.style.transform = 'translateY(-1px)';
        };
        button.onmouseout = () => {
            button.style.opacity = '1';
            button.style.transform = 'translateY(0)';
        };

        button.onclick = exportChat;
        document.body.appendChild(button);
    }

    // Function to extract text content from code blocks
    function extractCodeContent(codeElement) {
        const codeText = codeElement.querySelector('code');
        if (codeText) {
            return codeText.textContent || codeText.innerText;
        }
        return codeElement.textContent || codeElement.innerText;
    }

    // Function to process message content and convert to markdown
    function processMessageContent(contentDiv) {
        let markdown = '';

        // Handle different types of content
        const children = contentDiv.children;

        for (let i = 0; i < children.length; i++) {
            const child = children[i];

            // Handle paragraphs
            if (child.tagName === 'P') {
                const textContent = child.textContent || child.innerText;
                if (textContent.trim()) {
                    markdown += textContent.trim() + '\n\n';
                }
            }

            // Handle code blocks
            else if (child.tagName === 'PRE') {
                const codeContent = extractCodeContent(child);
                const languageElement = child.querySelector('[class*="language-"]');
                let language = '';

                if (languageElement) {
                    const classList = languageElement.className;
                    const match = classList.match(/language-(\w+)/);
                    if (match) {
                        language = match[1];
                    }
                }

                // Also check for language indicators in header
                const headerSpan = child.querySelector('span.text-xs.capitalize');
                if (headerSpan && !language) {
                    language = headerSpan.textContent || '';
                }

                markdown += '```' + language + '\n' + codeContent.trim() + '\n```\n\n';
            }

            // Handle headings
            else if (child.tagName && child.tagName.match(/^H[1-6]$/)) {
                const level = parseInt(child.tagName.charAt(1));
                const headingText = child.textContent || child.innerText;
                markdown += '#'.repeat(level) + ' ' + headingText.trim() + '\n\n';
            }

            // Handle lists
            else if (child.tagName === 'UL') {
                const listItems = child.querySelectorAll('li');
                listItems.forEach(li => {
                    const itemText = li.textContent || li.innerText;
                    markdown += '- ' + itemText.trim() + '\n';
                });
                markdown += '\n';
            }

            else if (child.tagName === 'OL') {
                const listItems = child.querySelectorAll('li');
                listItems.forEach((li, index) => {
                    const itemText = li.textContent || li.innerText;
                    markdown += `${index + 1}. ` + itemText.trim() + '\n';
                });
                markdown += '\n';
            }

            // Handle other elements by extracting text
            else {
                const textContent = child.textContent || child.innerText;
                if (textContent.trim()) {
                    markdown += textContent.trim() + '\n\n';
                }
            }
        }

        return markdown.trim();
    }

    // Main export function
    function exportChat() {
        try {
            // Find all message containers
            const messageContainers = document.querySelectorAll('[data-message-author-role]');

            if (messageContainers.length === 0) {
                alert('No messages found to export. Make sure you are on a chat page with messages.');
                return;
            }

            let markdown = '# Mistral AI Chat Export\n\n';
            markdown += `**Exported on:** ${new Date().toLocaleString()}\n\n`;
            markdown += '---\n\n';

            messageContainers.forEach((container, index) => {
                const role = container.getAttribute('data-message-author-role');
                const messageId = container.getAttribute('data-message-id');

                // Find the timestamp
                let timestamp = '';
                const timestampElement = container.querySelector('.text-sm.text-hint');
                if (timestampElement) {
                    timestamp = timestampElement.textContent || timestampElement.innerText;
                }

                // Find the message content
                let content = '';

                if (role === 'user') {
                    // User messages - look for select-text content
                    const userContent = container.querySelector('.select-text');
                    if (userContent) {
                        // Handle user messages with potential code blocks
                        const spans = userContent.querySelectorAll('span.whitespace-pre-wrap');
                        let userText = '';
                        let inCodeBlock = false;
                        let codeLanguage = '';

                        spans.forEach(span => {
                            const text = span.textContent || span.innerText;
                            if (text === '```') {
                                if (!inCodeBlock) {
                                    inCodeBlock = true;
                                    userText += '```';
                                } else {
                                    inCodeBlock = false;
                                    userText += '\n```\n';
                                }
                            } else {
                                if (inCodeBlock) {
                                    userText += text + '\n';
                                } else {
                                    userText += text;
                                }
                            }
                        });

                        content = userText.trim();
                    }
                } else if (role === 'assistant') {
                    // Assistant messages - look for markdown container
                    const markdownContainer = container.querySelector('.markdown-container-style');
                    if (markdownContainer) {
                        content = processMessageContent(markdownContainer);
                    }
                }

                // Add message to markdown
                if (content) {
                    const roleTitle = role === 'user' ? '👤 User' : '🤖 Assistant';
                    markdown += `## ${roleTitle}`;
                    if (timestamp) {
                        markdown += ` *(${timestamp})*`;
                    }
                    markdown += '\n\n';
                    markdown += content + '\n\n';
                    markdown += '---\n\n';
                }
            });

            // Create and download file
            const blob = new Blob([markdown], { type: 'text/markdown;charset=utf-8' });
            const url = URL.createObjectURL(blob);
            const a = document.createElement('a');
            a.href = url;

            // Generate filename with current date
            const now = new Date();
            const dateStr = now.toISOString().split('T')[0];
            const timeStr = now.toTimeString().split(' ')[0].replace(/:/g, '-');
            a.download = `mistral-chat-${dateStr}_${timeStr}.md`;

            document.body.appendChild(a);
            a.click();
            document.body.removeChild(a);
            URL.revokeObjectURL(url);

            // Show success message
            const successMsg = document.createElement('div');
            successMsg.innerText = '✅ Chat exported successfully!';
            successMsg.style.cssText = `
                position: fixed;
                top: 80px;
                right: 20px;
                background: #10b981;
                color: white;
                padding: 10px 15px;
                border-radius: 8px;
                z-index: 10000;
                font-weight: 600;
                animation: fadeInOut 3s ease-in-out;
            `;

            // Add CSS animation
            const style = document.createElement('style');
            style.textContent = `
                @keyframes fadeInOut {
                    0% { opacity: 0; transform: translateY(-10px); }
                    20%, 80% { opacity: 1; transform: translateY(0); }
                    100% { opacity: 0; transform: translateY(-10px); }
                }
            `;
            document.head.appendChild(style);

            document.body.appendChild(successMsg);
            setTimeout(() => {
                if (document.body.contains(successMsg)) {
                    document.body.removeChild(successMsg);
                }
            }, 3000);

        } catch (error) {
            console.error('Export error:', error);
            alert('An error occurred while exporting the chat. Please check the console for details.');
        }
    }

    // Initialize the script when page loads
    function init() {
        // Wait for the page to load completely
        if (document.readyState === 'loading') {
            document.addEventListener('DOMContentLoaded', init);
            return;
        }

        // Add a small delay to ensure all elements are rendered
        setTimeout(() => {
            createExportButton();
        }, 2000);
    }

    // Start initialization
    init();

    // Also handle navigation changes (for SPAs)
    let currentUrl = location.href;
    new MutationObserver(() => {
        if (location.href !== currentUrl) {
            currentUrl = location.href;
            // Re-initialize on page change
            setTimeout(() => {
                if (!document.querySelector('button:contains("📝 Export to MD")')) {
                    createExportButton();
                }
            }, 2000);
        }
    }).observe(document, { subtree: true, childList: true });

})();