// ==UserScript==
// @name 文本网页自由复制-Markdown (可拖动按钮)
// @namespace http://tampermonkey.net/
// @version 3.2.0
// @description 修复了Turndown库加载的竞态条件问题,并优化了复制逻辑,确保稳定可靠地将选定内容复制为Markdown。
// @author shenfangda
// @match *://*/*
// @exclude https://accounts.google.com/*
// @exclude https://*.google.com/sorry/*
// @exclude https://mail.google.com/*
// @exclude /^https?://localhost[:/]/
// @exclude /^file:///*/
// @grant GM_setClipboard
// @require https://unpkg.com/turndown/dist/turndown.js
// @license MIT
// @icon  LTIuMDEgNC41LTQuNSA0LjUtNC41LTIuMDEtNC41LTQuNSAyLjAxLTQuNSA0LjUtNC41eiIvPjwvc3ZnPg==
// ==/UserScript==
(function () {
'use strict';
// --- Configuration ---
const BUTTON_TEXT_DEFAULT = '复制为 Markdown';
const BUTTON_TEXT_SELECTING_FREE = '选择区域中... (ESC 取消)';
const BUTTON_TEXT_SELECTING_DIV = '点击元素复制 (ESC 取消)';
const BUTTON_TEXT_COPIED = '已复制!';
const BUTTON_TEXT_FAILED = '复制失败!';
const TEMP_MESSAGE_DURATION = 2000; // ms
const DEBUG = true; // 设置为 true 以获取更详细的日志记录
// --- Logging ---
const log = (msg) => console.log(`[Markdown - 复制] ${msg}`);
const debugLog = (msg) => DEBUG && console.log(`[Markdown - 复制调试] ${msg}`);
// --- State ---
let isSelecting = false;
let isDivMode = false;
let startX, startY;
let selectionBox = null;
let highlightedDiv = null;
let copyBtn = null;
let originalButtonText = BUTTON_TEXT_DEFAULT;
let messageTimeout = null;
let turndownService = null;
// --- DOM Ready Check ---
function onDOMReady(callback) {
if (document.readyState === 'loading') {
document.addEventListener('DOMContentLoaded', callback);
} else {
callback();
}
}
// --- Main Initialization ---
function initScript() {
log(`Attempting init on ${window.location.href}`);
if (window.self !== window.top) {
log('Script is running in an iframe, aborting.');
return;
}
if (!document.body || !document.head) {
log('Error: document.body or document.head not found. Retrying...');
setTimeout(initScript, 500);
return;
}
log('DOM ready, initializing script.');
turndownService = new TurndownService({
headingStyle: 'atx',
hr: '---',
bulletListMarker: '-',
codeBlockStyle: 'fenced',
emDelimiter: '*',
});
injectStyles();
if (!createButton()) return;
setupEventListeners();
log('Initialization complete.');
}
// --- CSS Injection ---
function injectStyles() {
const STYLES = `
.markdown-copy-btn {
position: fixed;
top: 15px;
right: 15px;
z-index: 2147483646;
padding: 8px 14px;
background-color: #4CAF50;
color: white;
border: none;
border-radius: 5px;
cursor: pointer;
font-size: 13px;
font-family: sans-serif;
box-shadow: 0 2px 5px rgba(0,0,0,0.2);
transition: background-color 0.2s ease-in-out, transform 0.2s ease-in-out, box-shadow 0.2s ease-in-out;
line-height: 1.4;
text-align: center;
user-select: none;
}
.markdown-copy-btn:hover {
background-color: #45a049;
transform: translateY(-1px);
box-shadow: 0 4px 8px rgba(0,0,0,0.25);
}
.markdown-copy-btn.mc-copied { background-color: #3a8f40; }
.markdown-copy-btn.mc-failed { background-color: #c0392b; }
.markdown-copy-btn.dragging {
cursor: move;
transform: scale(1.05);
box-shadow: 0 6px 12px rgba(0,0,0,0.3);
}
.markdown-copy-selection-box {
position: absolute;
border: 2px dashed #4CAF50;
background-color: rgba(76, 175, 80, 0.1);
z-index: 2147483645;
pointer-events: none;
box-sizing: border-box;
}
.markdown-copy-div-highlight {
outline: 2px solid #4CAF50!important;
background-color: rgba(76, 175, 80, 0.1)!important;
box-shadow: inset 0 0 0 2px rgba(76, 175, 80, 0.5)!important;
transition: all 0.1s ease-in-out;
cursor: pointer;
}
`;
try {
const styleSheet = document.createElement('style');
styleSheet.id = 'markdown-copy-styles';
styleSheet.textContent = STYLES.trim().replace(/\s{2,}/g, ' ');
document.head.appendChild(styleSheet);
debugLog('Styles injected.');
} catch (error) {
log(`Error injecting styles: ${error.message}`);
}
}
// --- Button Creation ---
function createButton() {
if (document.getElementById('markdown-copy-btn-main')) {
log('Button already exists.');
copyBtn = document.getElementById('markdown-copy-btn-main');
return true;
}
try {
copyBtn = document.createElement('button');
copyBtn.id = 'markdown-copy-btn-main';
copyBtn.className = 'markdown-copy-btn';
copyBtn.textContent = BUTTON_TEXT_DEFAULT;
originalButtonText = BUTTON_TEXT_DEFAULT;
document.body.appendChild(copyBtn);
const savedPos = localStorage.getItem('markdown-copy-btn-pos');
if (savedPos) {
try {
const pos = JSON.parse(savedPos);
if (pos && typeof pos.left !== 'undefined' && typeof pos.top !== 'undefined') {
copyBtn.style.left = pos.left;
copyBtn.style.top = pos.top;
copyBtn.style.right = 'auto';
debugLog(`Restored button position to ${pos.left}, ${pos.top}`);
}
} catch (e) {
log('Error parsing saved button position.');
localStorage.removeItem('markdown-copy-btn-pos');
}
}
debugLog('Button created and added.');
return true;
} catch (error) {
log(`Error creating button: ${error.message}`);
return false;
}
}
// --- Dragging Logic for Button ---
let isDragging = false;
let dragStartX, dragStartY;
let btnStartX, btnStartY;
let wasDragged = false;
function makeDraggable(btn) {
btn.addEventListener('mousedown', (e) => {
if (e.button !== 0 || isSelecting) {
return;
}
e.stopPropagation();
isDragging = true;
wasDragged = false;
dragStartX = e.clientX;
dragStartY = e.clientY;
const rect = btn.getBoundingClientRect();
btnStartX = rect.left;
btnStartY = rect.top;
document.addEventListener('mousemove', handleDragMove, { capture: true });
document.addEventListener('mouseup', handleDragEnd, { capture: true });
});
}
function handleDragMove(e) {
if (!isDragging) return;
if (!wasDragged) {
const dx = Math.abs(e.clientX - dragStartX);
const dy = Math.abs(e.clientY - dragStartY);
if (dx > 5 || dy > 5) {
wasDragged = true;
copyBtn.classList.add('dragging');
document.body.style.cursor = 'move';
}
}
if (wasDragged) {
e.stopPropagation();
e.preventDefault();
const deltaX = e.clientX - dragStartX;
const deltaY = e.clientY - dragStartY;
let newLeft = btnStartX + deltaX;
let newTop = btnStartY + deltaY;
const btnRect = copyBtn.getBoundingClientRect();
const viewportWidth = window.innerWidth;
const viewportHeight = window.innerHeight;
if (newLeft < 0) newLeft = 0;
else if (newLeft + btnRect.width > viewportWidth) newLeft = viewportWidth - btnRect.width;
if (newTop < 0) newTop = 0;
else if (newTop + btnRect.height > viewportHeight) newTop = viewportHeight - btnRect.height;
copyBtn.style.left = `${newLeft}px`;
copyBtn.style.top = `${newTop}px`;
copyBtn.style.right = 'auto';
}
}
function handleDragEnd(e) {
if (!isDragging) return;
if (wasDragged) {
e.stopPropagation();
e.preventDefault();
const pos = { left: copyBtn.style.left, top: copyBtn.style.top };
try {
localStorage.setItem('markdown-copy-btn-pos', JSON.stringify(pos));
debugLog(`Saved button position: ${JSON.stringify(pos)}`);
} catch (err) {
log(`Error saving button position: ${err.message}`);
}
}
isDragging = false;
copyBtn.classList.remove('dragging');
if (isSelecting) {
document.body.style.cursor = isDivMode ? 'pointer' : 'crosshair';
} else {
document.body.style.cursor = 'default';
}
document.removeEventListener('mousemove', handleDragMove, { capture: true });
document.removeEventListener('mouseup', handleDragEnd, { capture: true });
}
// --- Event Listeners Setup ---
function setupEventListeners() {
if (!copyBtn) {
log("Error: Button not found for adding listeners.");
return;
}
copyBtn.addEventListener('click', handleButtonClick);
document.addEventListener('mousedown', handleMouseDown, true);
document.addEventListener('mousemove', handleMouseMove, true);
document.addEventListener('mouseup', handleMouseUp, true);
document.addEventListener('mouseover', handleMouseOverDiv);
document.addEventListener('click', handleClickDiv, true);
document.addEventListener('keydown', handleKeyDown);
makeDraggable(copyBtn); // Make the button draggable
debugLog('Event listeners added.');
}
// --- Button Click Logic ---
function handleButtonClick(e) {
if (wasDragged) {
return;
}
e.stopPropagation();
if (!isSelecting) {
isSelecting = true;
isDivMode = true;
setButtonState(BUTTON_TEXT_SELECTING_DIV);
document.body.style.cursor = 'pointer';
log('Entered Div Selection Mode.');
} else if (isDivMode) {
isDivMode = false;
setButtonState(BUTTON_TEXT_SELECTING_FREE);
document.body.style.cursor = 'crosshair';
log('Switched to Free Selection Mode.');
removeDivHighlight();
} else {
resetSelectionState();
log('Selection cancelled by button click.');
}
}
// --- Mouse Event Handlers for Selection ---
function handleMouseDown(e) {
if (isSelecting && !isDivMode && e.button === 0) {
e.preventDefault();
e.stopPropagation();
startX = e.pageX;
startY = e.pageY;
selectionBox = document.createElement('div');
selectionBox.className = 'markdown-copy-selection-box';
selectionBox.style.left = `${startX}px`;
selectionBox.style.top = `${startY}px`;
document.body.appendChild(selectionBox);
}
}
function handleMouseMove(e) {
if (selectionBox) {
const currentX = e.pageX;
const currentY = e.pageY;
const width = Math.abs(currentX - startX);
const height = Math.abs(currentY - startY);
const left = Math.min(currentX, startX);
const top = Math.min(currentY, startY);
selectionBox.style.width = `${width}px`;
selectionBox.style.height = `${height}px`;
selectionBox.style.left = `${left}px`;
selectionBox.style.top = `${top}px`;
} else if (isDivMode && isSelecting) {
handleMouseOverDiv(e);
}
}
function handleMouseUp(e) {
if (selectionBox) {
const rect = selectionBox.getBoundingClientRect();
document.body.removeChild(selectionBox);
selectionBox = null;
if (rect.width > 5 && rect.height > 5) {
copyContentInRect(rect);
}
resetSelectionState();
}
}
function handleMouseOverDiv(e) {
if (!isDivMode || !isSelecting) return;
const target = e.target;
if (target === copyBtn || target.closest('.markdown-copy-btn')) return;
if (highlightedDiv && highlightedDiv !== target) {
removeDivHighlight();
}
if (target && target.nodeType === 1 && !target.classList.contains('markdown-copy-div-highlight')) {
highlightedDiv = target;
highlightedDiv.classList.add('markdown-copy-div-highlight');
}
}
function handleClickDiv(e) {
if (isDivMode && isSelecting && highlightedDiv) {
e.preventDefault();
e.stopPropagation();
const divToCopy = highlightedDiv;
removeDivHighlight();
copyElementAsMarkdown(divToCopy);
resetSelectionState();
}
}
// --- Keyboard Event Handler ---
function handleKeyDown(e) {
if (e.key === 'Escape' && isSelecting) {
resetSelectionState();
log('Selection cancelled by ESC key.');
}
}
// --- State Management ---
function setButtonState(text, temporary = false, success = null) {
if (messageTimeout) {
clearTimeout(messageTimeout);
messageTimeout = null;
}
copyBtn.textContent = text;
copyBtn.classList.remove('mc-copied', 'mc-failed');
if (success === true) copyBtn.classList.add('mc-copied');
if (success === false) copyBtn.classList.add('mc-failed');
if (temporary) {
messageTimeout = setTimeout(() => {
copyBtn.textContent = originalButtonText;
copyBtn.classList.remove('mc-copied', 'mc-failed');
}, TEMP_MESSAGE_DURATION);
} else {
originalButtonText = text;
}
}
function resetSelectionState() {
isSelecting = false;
isDivMode = false;
document.body.style.cursor = 'default';
setButtonState(BUTTON_TEXT_DEFAULT);
if (selectionBox) {
document.body.removeChild(selectionBox);
selectionBox = null;
}
removeDivHighlight();
}
function removeDivHighlight() {
if (highlightedDiv) {
highlightedDiv.classList.remove('markdown-copy-div-highlight');
highlightedDiv = null;
}
}
// --- Core Copying Logic ---
async function copyContentInRect(rect) {
try {
const container = document.createElement('div');
const elementsInRect = [];
const walker = document.createTreeWalker(document.body, NodeFilter.SHOW_ELEMENT, null, false);
while (walker.nextNode()) {
const node = walker.currentNode;
if (node.closest && node.closest('.markdown-copy-btn')) continue;
if (isNodeVisible(node) && isNodeInRect(node, rect)) {
elementsInRect.push(node);
}
}
if (elementsInRect.length === 0) {
log('没有在选定区域找到可复制的内容。');
return;
}
const topLevelElements = elementsInRect.filter(el => {
let parent = el.parentElement;
while (parent) {
if (elementsInRect.includes(parent)) {
return false;
}
parent = parent.parentElement;
}
return true;
});
topLevelElements.forEach(el => container.appendChild(el.cloneNode(true)));
const htmlContent = container.innerHTML;
debugLog(`复制的 HTML 内容: ${htmlContent}`);
if (!htmlContent.trim()) {
log('没有在选定区域找到可复制的内容。');
return;
}
const markdown = turndownService.turndown(htmlContent);
await GM_setClipboard(markdown, 'text');
setButtonState(BUTTON_TEXT_COPIED, true, true);
log('内容已复制为 Markdown。');
} catch (error) {
log(`复制失败: ${error.message}`);
debugLog(`复制失败的错误: ${error.stack}`);
setButtonState(BUTTON_TEXT_FAILED, true, false);
}
}
async function copyElementAsMarkdown(element) {
try {
const markdown = turndownService.turndown(element);
await GM_setClipboard(markdown, 'text');
setButtonState(BUTTON_TEXT_COPIED, true, true);
log('元素已复制为 Markdown。');
} catch (error) {
log(`复制失败: ${error.message}`);
debugLog(`复制失败的元素: ${element.outerHTML}`);
debugLog(`复制失败的错误: ${error.stack}`);
setButtonState(BUTTON_TEXT_FAILED, true, false);
}
}
// --- Utility Functions ---
function isNodeInRect(node, rect) {
const nodeRect = node.getBoundingClientRect();
return (
nodeRect.top < rect.bottom &&
nodeRect.bottom > rect.top &&
nodeRect.left < rect.right &&
nodeRect.right > rect.left
);
}
function isNodeVisible(node) {
return !!(node.offsetWidth || node.offsetHeight || node.getClientRects().length);
}
// --- Script Entry Point ---
function checkLibsReady(callback) {
debugLog('正在检查 Turndown 库...');
if (typeof TurndownService !== 'undefined' && typeof TurndownService === 'function') {
debugLog('Turndown 库已就绪。');
callback();
} else {
let attempts = 0;
const interval = setInterval(() => {
attempts++;
if (typeof TurndownService !== 'undefined' && typeof TurndownService === 'function') {
clearInterval(interval);
debugLog(`Turndown 库在 ${attempts} 次尝试后加载。`);
callback();
} else if (attempts > 20) { // Timeout after 2 seconds
clearInterval(interval);
log('错误:Turndown 库加载超时。请检查网络连接或脚本的 @require URL。');
if(copyBtn) { // Check if button exists before trying to update it
copyBtn.textContent = '库加载失败';
copyBtn.classList.add('mc-failed');
copyBtn.disabled = true;
} else { // If button doesn't exist yet, create it to show the error
createButton();
if(copyBtn){
copyBtn.textContent = '库加载失败';
copyBtn.classList.add('mc-failed');
copyBtn.disabled = true;
}
}
}
}, 100);
}
}
onDOMReady(() => {
checkLibsReady(initScript);
});
})();