Esporta conversazioni chat di Mistral AI in formato markdown
// ==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 });
})();