// ==UserScript==
// @name Claude helper (对话导出\字数统计\时间显示)
// @name:zh-CN Claude 助手 (对话导出\字数统计\时间显示)
// @version 0.6.5
// @description ✴️1、可以导出 claude ai对话的内容。✴️2、统计当前字数 (包括粘贴、上传、article的内容,含换行符/markdown语法符号等)。✴️3、显示对话的时间、模型信息、Token用量。ℹ️显示的信息均来自网页内本身存在但未显示的属性值。
// @author Yearly
// @match https://claude.ai/*
// @include https://*claude*.com/*
// @match https://chat.kelaode.ai/*
// @include https://*claude*.cc/*
// @icon 
// @license AGPL-v3.0
// @namespace https://greasyfork.org/zh-CN/scripts/502829-claude-helper
// @supportURL https://greasyfork.org/zh-CN/scripts/502829-claude-helper
// @homepageURL https://greasyfork.org/zh-CN/scripts/502829-claude-helper
// @grant GM_addStyle
// @grant GM_xmlhttpRequest
// @grant GM_registerMenuCommand
// @grant GM_download
// @noframes
// ==/UserScript==
(function() {
GM_addStyle(`
.xl\:max-w-\[48rem\] {
width:95% !important;
max-width:96% !important;
}
div.mx-auto.md:max-w-3xl {
max-width: calc(100% - 10px);
}
div.mx-auto.flex {
max-width: calc(100% - 10px);
}
body > div.flex.min-h-screen.w-full div.flex.flex-col div.flex.gap-2 div.mt-1.max-h-96.w-full.overflow-y-auto.break-words > div.ProseMirror.break-words{
max-width:90%;
}
body > div.flex.min-h-screen.w-full > div > main > div.top-5.z-10.mx-auto.w-full.max-w-2xl.md{
max-width:100%;
}
body > div.flex.min-h-screen.w-full > div > main > div.mx-auto.w-full.max-w-2xl.px-1.md {
max-width:100%;
}
body > div.flex.min-h-screen.w-full > div > main.max-w-7xl {
max-width: 90rem;
}
main > div.composer-parent article > div.text-base > div.mx-auto {
max-width: 95%;
}
main article > div.text-base > div.mx-auto {
max-width: 95%;
}
div.relative.flex.w-full.overflow-x-hidden.overflow-y-scroll > div.relative.mx-auto.flex.h-full.w-full > div.flex.mx-auto.w-full > div[data-test-render-count] > div.mb-1.mt-1 > div.group.relative.inline-flex {
border: 1px solid #dfded7;
}
`);
// model info
function conversation_model() {
let conversation = document.querySelector("body > div.flex.min-h-screen.w-full > div > div.flex.h-screen") ;
if(!conversation) return null;
let reactProps = Object.keys(conversation).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return null;
let conversProps = conversation[reactProps];
if (!conversProps) return null;
let model = conversProps.children[1]?.props?.children[0]?.props?.conversation?.model; //claude-3-5-sonnet-20240620
return model;
}
// tokensSoFar
function conversation_tokensSoFar() {
let conversation = document.querySelector("body > div.flex.min-h-screen.w-full > div > div.flex.h-screen") ;
if(!conversation) return null;
let reactProps = Object.keys(conversation).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return null;
let conversProps = conversation[reactProps];
if (!conversProps) return null;
let tokensSoFar = conversProps.children[1]?.props?.children[0]?.props?.conversation?.tokensSoFar;
return tokensSoFar;
}
// msg count
var last_uuid = '', last_length = 0;
function get_msg_count() {
let mainScreen = document.querySelector("body > div.flex.min-h-screen.w-full > div > div > div.relative.flex.w-full > div.relative.mx-auto.flex");
if(!mainScreen) return;
let tx_cnts = 0, tx_sz = 0;
let rx_cnts = 0, rx_sz = 0;
let fp_cnts = 0, fp_sz = 0, img_cnts = 0;
let i = 0;
let reactProps = Object.keys(mainScreen).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return null;
let msgProps = mainScreen[reactProps];
let Msgs = (msgProps.children[1].props.children[0]);
/*
if (Msgs && Msgs.length > 0) {
let newest_msg = Msgs[Msgs.length-1].newest_msgs.props.children[0].props.msg;
if(newest_msg.content[0].text) {
let uuid = newest_msg.uuid;
let length = newest_msg.content[0].text.length;
if (uuid == last_uuid && length == last_length) {
return null;
}
last_uuid = uuid;
last_length = length;
}
}else {
return null;
}
*/
Msgs.forEach(function(msg_item){
let msg = msg_item.props.children[0].props.msg;
if(msg.sender == "human" && msg.content[0].text) {
tx_cnts +=1;
tx_sz += msg.content[0].text.length;
for(i = 0; i < msg.attachments.length; i++) {
tx_sz += msg.attachments[i].file_size;
fp_cnts += 1;
fp_sz += msg.attachments[i].file_size;;
}
img_cnts += msg.files.length;
} else if(msg.sender == "assistant" && msg.content[0].text) {
rx_cnts +=1;
rx_sz += msg.content[0].text.length;
}
});
return {
tx_cnts: tx_cnts, tx_sz: tx_sz,
rx_cnts: rx_cnts, rx_sz: rx_sz,
fp_cnts: fp_cnts, fp_sz: fp_sz,
img_cnts: img_cnts,
};
}
function msg_counter_main() {
let fieldset = document.querySelector("body > div.flex.min-h-screen.w-full fieldset") || document.querySelector("body > div.flex.min-h-screen.w-full > div > div > div.relative.flex.w-full> div.relative.mx-auto.flex.h-full.w-full > div.sticky.bottom-0.mx-auto.w-full");
if (fieldset) {
let ret = get_msg_count();
if(!ret) return;
let count_result = document.querySelector("#claude-msg-counter")
if(!count_result) {
count_result = document.createElement("pre");
count_result.id = "claude-msg-counter";
count_result.className="border-0.5 relative z-[5] text-text-200 border-accent-pro-100/20 bg-accent-pro-900 rounded-t-xl border-b-0"
count_result.style = "font-size:12px; padding: 5px 7px 14px; margin: -3px 0px -12px; text-wrap: pretty; z-index: 6;";
let targetParent = fieldset.querySelector("div.flex.md\\:px-2.flex-col-reverse");
if(targetParent) {
targetParent.insertBefore(count_result, targetParent.firstChild);
} else if(fieldset.querySelector("div.flex.w-full.flex-col.items-center")) {
fieldset.querySelector("div.flex.w-full.flex-col.items-center").before(count_result);
}
}
let all_length = ret.tx_sz + ret.rx_sz ;
let file_info = ""
let img_file_info = ""
if (ret.fp_cnts) file_info = ` (包含${ret.fp_cnts}个上传或粘贴文本,${ret.fp_sz}字)`
if (ret.img_cnts) img_file_info = ` (另有${ret.img_cnts}个非文本内容的上传或粘贴,不能计量字数)`
const model = conversation_model();
const token = conversation_tokensSoFar();
let model_info = '';
if (model) {
model_info = `【模型】${model}。`;
}
let token_info = '';
if (token) {
token_info = `【tokensSoFar】${token}。`;
}
count_result.innerText = `【统计】已发出:${ret.tx_cnts}条,${ret.tx_sz}字${file_info}; 已回复:${ret.rx_cnts}条,${ret.rx_sz}字; 总计:${all_length}字${img_file_info}。${model_info}${token_info}`;
}
}
setInterval(() => {
msg_counter_main();
}, 1600);
// show message date/time
function show_msg_time() {
let mainScreen = document.querySelector("body > div.flex.min-h-screen.w-full > div > div.flex.h-screen") ;
if(!mainScreen) return;
const msg_divs = mainScreen.querySelectorAll("div[data-test-render-count] > div.mb-1.mt-1, div[data-test-render-count] > div > div[data-is-streaming].group");
msg_divs.forEach(function(msg_div){
if (msg_div.nextSibling) return;
let reactProps = Object.keys(msg_div).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return;
let divProps = msg_div[reactProps];
let updated_at = divProps.children?.[1]?.props?.message?.updated_at ?? divProps.children?.[1]?.props?.children?.[2]?.props?.message?.updated_at;
if (!updated_at) return;
const date = new Date(updated_at);
if (!date) return;
const localDateStr = date.toLocaleString();
let timeNode = document.createElement("div");
timeNode.innerText = localDateStr;
timeNode.className = 'msg-uptime';
msg_div.after(timeNode);
});
}
GM_addStyle(`
div[data-test-render-count] > div > .msg-uptime {
margin: 1px 5px 5px; font-size: 13px; font-weight: 300;
}
div[data-test-render-count] > .msg-uptime {
margin: -2px 5px 5px; font-size: 13px; font-weight: 300;
}
`);
setInterval(() => {
show_msg_time();
}, 2100);
// Add Download Button
function createPersistentElement(selector, createElementCallback) {
function ensureElement() {
const targetElement = document.querySelector(selector);
if (targetElement) {
if (!targetElement.querySelector('.-added-element')) {
const newElement = createElementCallback();
newElement.classList.add('-added-element');
targetElement.appendChild(newElement);
}
}
}
ensureElement();
const observer = new MutationObserver(() => {
ensureElement();
});
observer.observe(document.body, {
childList: true,
subtree: true
});
}
function get_account_email() {
let user_menu = document.querySelector('button[data-testid="user-menu-button"][id^=radix-] > div.relative.flex.w-full.items-center');
if(!user_menu) return '';
let reactProps = Object.keys(user_menu).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return '';
let __reactProps = user_menu[reactProps];
if (!__reactProps) return '';
let account = __reactProps.children[0]?.props?.account?.email_address || '';
console.log(account);
return account;
}
function get_msg_context() {
let context = "";
// let mainScreen = document.querySelector("body > div.flex.min-h-screen.w-full > div > div.flex.h-screen") ;
let mainScreen = document.querySelector("body > div.flex.min-h-screen.w-full > div > div > div.relative.flex.w-full > div.relative.mx-auto.flex");
if(!mainScreen) {
console.warn("not found div")
return null;
}
let tx_cnts = 0, tx_sz = 0;
let rx_cnts = 0, rx_sz = 0;
let fp_cnts = 0, fp_sz = 0;
let i = 0;
let reactProps = Object.keys(mainScreen).find(key => key.startsWith('__reactProps$'));
if (!reactProps) {
console.warn("not found reactProps")
return null;
}
let msgProps = mainScreen[reactProps];
let convID = (msgProps.children[0]?.props?.conversationUuid);
let name = (msgProps.children[1]?.props?.name) || document.title;
let Msgs = (msgProps.children[1].props.children[0]);
if ( !convID || !name || !Msgs && !Msgs.length <= 0) {
console.warn("not found Msg", convID , name, Msgs);
return null;
}
const model = conversation_model();
const token = conversation_tokensSoFar();
let model_info = '';
if (model) {
model_info = `- model : ${model}\n`;
}
let token_info = '';
if (token) {
token_info = `- tokensSoFar : ${token}\n`;
}
let time_info='';
let time_start_str = '';
let time_start = Msgs[0].props.children[0].props.msg.updated_at || Msgs[0].props.children[0].props.msg.created_at;
if (time_start) {
const date = new Date(time_start);
if (date) { time_start_str = date.toLocaleString(); }
}
let time_last_str = '';
let time_last = Msgs[Msgs.length-1].updated_at || Msgs[Msgs.length-1].created_at;
if (time_last) {
const date = new Date(time_last);
if (date) { time_last_str = date.toLocaleString(); }
}
if(time_start_str && time_last_str) {
time_info = `- time : ${time_start_str} ~ ${time_last_str}\n`
};
let account_info = ''
let account_email = get_account_email();
if (account_email) {
account_info = `- account : ${account_email}\n`;
}
context += `# ${name}\n\n${account_info}${model_info}${token_info}${time_info}- conversationUUID : ${convID}\n`;
// Msgs.forEach(function(msg){
Msgs.forEach(function(msg_item){
let msg = msg_item.props.children[0].props.msg;
context += `\n## ${msg.sender}:\n\n`
context += msg.text + msg.content[0].text + '\n'
for(i = 0; i < msg.attachments.length; i++) {
context += `\nfile: ${msg.attachments[i].file_name}\n`
if(msg.attachments[i].extracted_content) {
context += `\n\`\`\`file_context\n ${msg.attachments[i].extracted_content}\n\`\`\`\n`;
}
}
for(i = 0; i < msg.files.length; i++) {
context += `file: ${msg.files[i].file_name}\n`
if(msg.files[i].preview_url) {
context += `preview_url: ${window.location.origin + msg.files[i].preview_url}\n`;
}
}
context += `\n------------------------------------------------------\n`
});
let blob = new Blob([context], {type: 'text/plain;charset=utf-8'});
let fileUrl = URL.createObjectURL(blob);
let tempLink = document.createElement('a');
tempLink.href = fileUrl;
let fileTitle = name.replaceAll(' ','_') + ".ClaudeAI.export.md";
tempLink.setAttribute('download', fileTitle);
tempLink.style.display = 'none';
document.body.appendChild(tempLink);
tempLink.click();
document.body.removeChild(tempLink);
URL.revokeObjectURL(fileUrl);
return;
}
function createDownloadButton() {
const button = document.createElement("button");
button.className = "inline-flex items-center justify-center relative shrink-0 ring-offset-2 ring-offset-bg-300 ring-accent-main-100 focus-visible:outline-none focus-visible:ring-1 disabled:pointer-events-none disabled:opacity-50 disabled:shadow-none disabled:drop-shadow-none text-text-200 transition-all font-styrene active:bg-bg-400 hover:bg-bg-500/40 hover:text-text-100 h-9 w-9 rounded-md active:scale-95 shrink-0";
button.innerHTML = `<svg width="20" height="20" viewBox="0 0 32 32" xmlns="http://www.w3.org/2000/svg" fill="none"><path stroke="#535358" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M27 7H5a2 2 0 0 0-2 2v14a2 2 0 0 0 2 2h22a2 2 0 0 0 2-2V9a2 2 0 0 0-2-2"/><path stroke="#535358" stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M15 20v-8l-4 4-4-4v8m12-3.5 3.5 3.5 3.5-3.5M22.5 20v-9"/></svg>`;
button.title="Download Conversation"
button.addEventListener("click", () => {
get_msg_context();
});
return button;
}
// 添加按钮
createPersistentElement("body > div.flex.min-h-screen.w-full div.sticky.items-center div.right-3 div.hidden.flex-row-reverse", createDownloadButton);
})();
(function() {
function AddDownloadAllChats(){
let user_menu = document.querySelector('button[data-testid="user-menu-button"][id^=radix-] > div.relative.flex.w-full.items-center');
if(!user_menu) {
setTimeout(() => AddDownloadAllChats(), 1000);
return null;
}
let reactProps = Object.keys(user_menu).find(key => key.startsWith('__reactProps$'));
if (!reactProps) return null;
let __reactProps = user_menu[reactProps];
if (!__reactProps) return null;
let master_id = __reactProps.children[0]?.props?.account?.memberships[0]?.organization?.uuid;
let account = __reactProps.children[0]?.props?.account?.email_address || '';
console.log(master_id);
let progressWindow, progressBar, statusText, downloadButton, debugInfo;
function createProgressWindow() {
progressWindow = document.createElement('div');
progressWindow.style.cssText = `
position: fixed;
top: 20px;
right: 20px;
width: 300px;
padding: 10px;
background: white;
border: 1px solid #ccc;
box-shadow: 0 0 10px rgba(0,0,0,0.1);
z-index: 99999;
`;
statusText = document.createElement('div');
progressWindow.appendChild(statusText);
progressBar = document.createElement('progress');
progressBar.style.width = '100%';
progressBar.max = 100;
progressWindow.appendChild(progressBar);
downloadButton = document.createElement('button');
downloadButton.textContent = 'Download TAR';
downloadButton.style.cssText = `
display: none;
margin-top: 10px;
padding: 5px 10px;
background-color: #4CAF50;
color: white;
border: none;
cursor: pointer;
`;
progressWindow.appendChild(downloadButton);
debugInfo = document.createElement('div');
debugInfo.style.marginTop = '10px';
progressWindow.appendChild(debugInfo);
document.body.appendChild(progressWindow);
}
function updateProgress(percent, status) {
progressBar.value = percent;
statusText.textContent = status;
}
function updateDebugInfo(info) {
debugInfo.textContent = info;
console.log('Debug:', info);
}
function makeRequest(url) {
return fetch(url)
.then(response => {
if (!response.ok) {
throw new Error(`Request failed: ${response.status}`);
}
return response.json();
})
.catch(error => {
if (error instanceof SyntaxError) {
throw new Error(`Failed to parse JSON: ${error}`);
}
throw error;
});
}
function createTarHeader(filename, size) {
const header = new Uint8Array(512);
const encoder = new TextEncoder();
encoder.encodeInto(filename, header.subarray(0, 100)); // File name
encoder.encodeInto('000644 \0', header.subarray(100, 108)); // File mode (default to 644 octal)
encoder.encodeInto('0000000 \0', header.subarray(108, 116)); // Owner's numeric user ID (default to 0)
encoder.encodeInto('0000000 \0', header.subarray(116, 124)); // Group's numeric user ID (default to 0)
encoder.encodeInto(size.toString(8).padStart(11, '0') + ' ', header.subarray(124, 136)); // File size in bytes (octal)
const mtime = Math.floor(Date.now() / 1000).toString(8).padStart(11, '0') + ' '; // Last modification time in numeric Unix time format (octal)
encoder.encodeInto(mtime, header.subarray(136, 148));
encoder.encodeInto(' ', header.subarray(148, 156)); // Checksum for header block (calculate later)
header[156] = '0'.charCodeAt(0); // Type flag (default to '0' for normal file)
let checksum = 0;
for (let i = 0; i < 512; i++) {
checksum += header[i];
}
encoder.encodeInto(checksum.toString(8).padStart(6, '0') + '\0 ', header.subarray(148, 156)); // Calculate and set the checksum
return header;
}
function createTarFile(files) {
const chunks = [];
for (const [filename, content] of Object.entries(files)) {
const encoder = new TextEncoder();
const fileContent = encoder.encode(content);
const header = createTarHeader(filename, fileContent.length);
chunks.push(header);
chunks.push(fileContent);
// Pad the file content to a multiple of 512 bytes
const padding = 512 - (fileContent.length % 512);
if (padding < 512) {
chunks.push(new Uint8Array(padding));
}
}
// Add two 512-byte blocks filled with zeros to mark the end of the archive
chunks.push(new Uint8Array(1024));
return new Blob(chunks, { type: 'application/x-tar' });
}
var jsonfiles = {};
async function backupAllChats() {
createProgressWindow();
updateProgress(0, 'Fetching conversation list...');
try {
const conversations = await makeRequest(`/api/organizations/${master_id}/chat_conversations`);
updateDebugInfo(`Total conversations: ${conversations.length}`);
const totalConversations = conversations.length;
for (let i = 0; i < conversations.length; i++) {
const conversation = conversations[i];
updateProgress((i / totalConversations) * 100, `Fetching conversation ${i + 1} of ${totalConversations}`);
try {
const detailedConversation = await makeRequest(`/api/organizations/${master_id}/chat_conversations/${conversation.uuid}?tree=True&rendering_mode=raw`);
if (detailedConversation.chat_messages?.length == 0) {
continue;
}
const fileContent = JSON.stringify(detailedConversation, null, 2);
let fileName = `${conversation.name.replaceAll(' ','_').substr(0,50)}_${conversation.updated_at.substr(0,10)}`;
fileName+='.json';
jsonfiles[fileName] = fileContent;
updateDebugInfo(`Added file: ${fileName}, Size: ${fileContent.length} bytes`);
} catch (error) {
console.error(`Error fetching conversation ${conversation.uuid}:`, error);
updateDebugInfo(`Error fetching conversation ${conversation.uuid}: ${error}`);
}
}
updateProgress(95, 'Generating Tar file...');
updateDebugInfo('Starting Tar generation...');
const tarBlob = createTarFile(jsonfiles);
const sizeInKB = (tarBlob.size / (1024)).toFixed(2);
updateDebugInfo(`Tar file size: ${sizeInKB} KB`);
downloadButton.style.display = 'block';
updateProgress(100, 'Click the button below to download');
downloadButton.onclick = function() {
const url = window.URL.createObjectURL(tarBlob);
const a = document.createElement('a');
a.href = url;
const timeStr = (new Date()).toLocaleString().replace(/[\/:]/g,'-').replaceAll(' ','_');
a.download = `ClaudeBackup_${account}_${timeStr}.tar`;
document.body.appendChild(a);
a.click();
window.URL.revokeObjectURL(url);
document.body.removeChild(a);
updateProgress(100, 'Download started!');
setTimeout(() => progressWindow.remove(), 3000);
};
} catch (error) {
console.error('Error:', error);
updateProgress(100, 'Error occurred. Check console for details.');
updateDebugInfo(`Error: ${error}`);
}
}
GM_registerMenuCommand("备份全部对话记录(Json格式)", backupAllChats);
};
setTimeout(() => AddDownloadAllChats(), 1000);
// 将页面上的重置时间用更完整的本地时间表示
let lastProcessedTime = '';
const updateInterval = 2000; // 每2秒更新一次
function convertTime(shortTime) {
const now = new Date();
const [time, period] = shortTime.split(' ');
let [hours, minutes] = time.split(':');
hours = parseInt(hours);
if (period.toLowerCase() === 'pm' && hours !== 12) {
hours += 12;
} else if (period.toLowerCase() === 'am' && hours === 12) {
hours = 0;
}
now.setHours(hours);
now.setMinutes(minutes || 0);
now.setSeconds(0);
now.setMilliseconds(0);
return now.toLocaleString();
}
function updateTime() {
const element = document.querySelector("div.sticky div.w-full > div.items-center > div.text-danger-000 > div > div.text-sm > span");
if (!element) return;
const originalText = element.getAttribute('data-original-time') || element.textContent.trim();
if (!element.hasAttribute('data-original-time')) {
element.setAttribute('data-original-time', originalText);
}
if (originalText !== lastProcessedTime) {
const fullTime = convertTime(originalText);
element.textContent = `${originalText} (${fullTime})`;
lastProcessedTime = originalText;
}
}
function debounce(func, wait) {
let timeout;
return function executedFunction(...args) {
const later = () => {
clearTimeout(timeout);
func(...args);
};
clearTimeout(timeout);
timeout = setTimeout(later, wait);
};
}
const debouncedUpdateTime = debounce(updateTime, 350);
function checkForChanges() {
debouncedUpdateTime();
requestAnimationFrame(checkForChanges);
}
requestAnimationFrame(checkForChanges);
setInterval(updateTime, updateInterval);
})();