您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
将网页内容转换为 Markdown 格式的工具,支持复制、下载、发送到 GitHub 和 Obsidian 等功能。
当前为
// ==UserScript== // @name Markdown Web Clipper // @name:zh-CN Markdown Web Clipper // @description A tool to convert web content into Markdown format, supporting features like copy, download, and sending to GitHub and Obsidian. // @description:zh-CN 将网页内容转换为 Markdown 格式的工具,支持复制、下载、发送到 GitHub 和 Obsidian 等功能。 // @author shiquda,人民的勤务员 <[email protected]> // @namespace https://github.com/ChinaGodMan/UserScripts // @supportURL https://github.com/ChinaGodMan/UserScripts/issues // @homepageURL https://github.com/ChinaGodMan/UserScripts // @license MIT // @grant GM_addStyle // @grant GM_registerMenuCommand // @grant GM_setClipboard // @grant GM_setValue // @grant GM_getValue // @require https://code.jquery.com/jquery-3.6.0.min.js // @require https://cdnjs.cloudflare.com/ajax/libs/jqueryui/1.12.1/jquery-ui.min.js // @require https://unpkg.com/turndown/dist/turndown.js // @require https://unpkg.com/@guyplusplus/turndown-plugin-gfm/dist/turndown-plugin-gfm.js // @require https://cdnjs.cloudflare.com/ajax/libs/marked/12.0.0/marked.min.js // @match *://*/* // @icon  // @compatible chrome // @compatible firefox // @compatible edge // @compatible opera // @compatible safari // @compatible kiwi // @compatible qq // @compatible via // @compatible brave // @version 2025.03.18.0713 // @created 2025-03-18 07:13:13 // @modified 2025-03-18 07:13:13 // ==/UserScript== /** * File: web-clipper.user.js * Project: UserScripts * File Created: 2025/03/18,Tuesday 07:13:13 * Author: 人民的勤务员@ChinaGodMan ([email protected]) * ----- * Last Modified: 2025/03/18,Tuesday 08:48:06 * Modified By: 人民的勤务员@ChinaGodMan ([email protected]) * ----- * License: MIT License * Copyright © 2024 - 2025 ChinaGodMan,Inc */ //! https://greasyfork.org/zh-CN/scripts/486888 (function () { 'use strict' // GitHub 信息设置, GitHub settings // 替换为你的GitHub令牌,Replace with your GitHub Token const github_token = '' // 替换为你的GitHub仓库所有者,Replace with your GitHub repository owner const github_owner = '' // 比如 ChinaGodMan,Example: ChinaGodMan // 替换为你的GitHub仓库名称,Replace with your GitHub repository name const github_repo = '' // 比如 UserScripts,Example: UserScripts // User Config // Short cut const shortCutUserConfig = { /* Example: "Shift": false, "Ctrl": true, "Alt": false, "Key": "m" */ } // Obsidian const obsidianUserConfig = { /* Example: "my note": [ "Inbox/Web/", "Collection/Web/Reading/" ] */ } const userLang = (navigator.languages && navigator.languages[0]) || navigator.language || 'en' const translations = { en: { copy: 'Copy to clipboard', copied: 'Copied successfully!', download_md: 'Download as MD', send_to_github: 'Send to Github', send_to_obsidian: 'Send to Obsidian', github_failed: 'Creation failed:', github_success: 'Creation succeeded:', configure: 'Please configure your GitHub information first', menu: 'Convert to Markdown', guide: ` - Use **arrow keys** to select elements: - Up: Select parent element - Down: Select the first child element - Left: Select the previous sibling element - Right: Select the next sibling element - Use **scroll wheel** to zoom in and out: * Up: Select parent element - Down: Select the first child element - Click to select an element - Press \`Esc\` key to cancel selection ` }, 'zh-CN,zh,zh-SG': { copy: '复制到剪辑版', copied: '复制成功!', download_md: '下载为MD', send_to_github: '保存到GitHub', send_to_obsidian: '发送到Obsidian', github_failed: '创建失败:', github_success: '创建成功:', configure: '请先配置你的GitHub 信息', menu: '转换为Markdown', guide: ` - 使用**方向键**选择元素 - 上:选择父元素 - 下:选择第一个子元素 - 左:选择上一个兄弟元素 - 右:选择下一个兄弟元素 - 使用**滚轮**放大缩小 - 上:选择父元素 - 下:选择第一个子元素 - 点击元素选择 - 按下 \`Esc\` 键取消选择 ` }, 'zh-TW,zh-HK,zh-MO': { copy: '複製到剪貼簿', copied: '複製成功!', download_md: '下載為MD', send_to_github: '保存到GitHub', send_to_obsidian: '發送到Obsidian', github_failed: '創建失敗:', github_success: '創建成功:', configure: '請先配置你的GitHub 信息', menu: '轉換為Markdown', guide: ` - 使用**方向鍵**選擇元素 - 上:選擇父元素 - 下:選擇第一個子元素 - 左:選擇上一個兄弟元素 - 右:選擇下一個兄弟元素 - 使用**滾輪**放大縮小 - 上:選擇父元素 - 下:選擇第一個子元素 - 點擊元素選擇 - 按下 \`Esc\` 鍵取消選擇 ` }, vi: { copy: 'Sao chép vào clipboard', copied: 'Sao chép thành công!', download_md: 'Tải xuống dưới dạng MD', send_to_github: 'Gửi đến Github', send_to_obsidian: 'Gửi đến Obsidian', github_failed: 'Tạo thất bại:', github_success: 'Tạo thành công:', configure: 'Vui lòng cấu hình thông tin GitHub của bạn trước', menu: 'Chuyển đổi sang Markdown', guide: ` - Sử dụng **phím mũi tên** để chọn các phần tử: - Lên: Chọn phần tử cha - Xuống: Chọn phần tử con đầu tiên - Trái: Chọn phần tử anh em trước - Phải: Chọn phần tử anh em sau - Sử dụng **bánh xe cuộn** để phóng to và thu nhỏ: - Lên: Chọn phần tử cha - Xuống: Chọn phần tử con đầu tiên - Nhấp để chọn một phần tử - Nhấn phím \`Esc\` để hủy chọn ` }, ja: { copy: 'クリップボードにコピー', copied: 'コピー成功!', download_md: 'MDとしてダウンロード', send_to_github: 'Githubに送信', send_to_obsidian: 'Obsidianに送信', github_failed: '作成失敗:', github_success: '作成成功:', configure: 'まずGitHub情報を設定してください', menu: 'Markdownに変換', guide: ` - **矢印キー**を使用して要素を選択します: - 上: 親要素を選択 - 下: 最初の子要素を選択 - 左: 前の兄弟要素を選択 - 右: 次の兄弟要素を選択 - **スクロールホイール**を使用してズームインおよびズームアウトします: - 上: 親要素を選択 - 下: 最初の子要素を選択 - 要素をクリックして選択 - \`Esc\`キーを押して選択をキャンセル ` }, ko: { copy: '클립보드에 복사', copied: '복사 성공!', download_md: 'MD로 다운로드', send_to_github: 'Github에 보내기', send_to_obsidian: 'Obsidian에 보내기', github_failed: '생성 실패:', github_success: '생성 성공:', configure: '먼저 GitHub 정보를 구성하십시오', menu: 'Markdown으로 변환', guide: ` - **화살표 키**를 사용하여 요소를 선택합니다: - 위: 부모 요소 선택 - 아래: 첫 번째 자식 요소 선택 - 왼쪽: 이전 형제 요소 선택 - 오른쪽: 다음 형제 요소 선택 - **스크롤 휠**을 사용하여 확대 및 축소합니다: - 위: 부모 요소 선택 - 아래: 첫 번째 자식 요소 선택 - 요소를 선택하려면 클릭 - 선택을 취소하려면 \`Esc\` 키를 누르십시오 ` }, fr: { copy: 'Copier dans le presse-papiers', copied: 'Copié avec succès!', download_md: 'Télécharger en tant que MD', send_to_github: 'Envoyer à Github', send_to_obsidian: 'Envoyer à Obsidian', github_failed: 'Échec de la création:', github_success: 'Création réussie:', configure: 'Veuillez d\'abord configurer vos informations GitHub', menu: 'Convertir en Markdown', guide: ` - Utilisez les **touches fléchées** pour sélectionner les éléments: - Haut: Sélectionner l'élément parent - Bas: Sélectionner le premier élément enfant - Gauche: Sélectionner l'élément frère précédent - Droite: Sélectionner l'élément frère suivant - Utilisez la **molette de défilement** pour zoomer et dézoomer: - Haut: Sélectionner l'élément parent - Bas: Sélectionner le premier élément enfant - Cliquez pour sélectionner un élément - Appuyez sur la touche \`Esc\` pour annuler la sélection ` }, it: { copy: 'Copia negli appunti', copied: 'Copiato con successo!', download_md: 'Scarica come MD', send_to_github: 'Invia a Github', send_to_obsidian: 'Invia a Obsidian', github_failed: 'Creazione fallita:', github_success: 'Creazione riuscita:', configure: 'Si prega di configurare prima le informazioni di GitHub', menu: 'Converti in Markdown', guide: ` - Usa i **tasti freccia** per selezionare gli elementi: - Su: Seleziona l'elemento padre - Giù: Seleziona il primo elemento figlio - Sinistra: Seleziona l'elemento fratello precedente - Destra: Seleziona l'elemento fratello successivo - Usa la **rotella di scorrimento** per ingrandire e ridurre: - Su: Seleziona l'elemento padre - Giù: Seleziona il primo elemento figlio - Clicca per selezionare un elemento - Premi il tasto \`Esc\` per annullare la selezione ` }, de: { copy: 'In die Zwischenablage kopieren', copied: 'Erfolgreich kopiert!', download_md: 'Als MD herunterladen', send_to_github: 'An Github senden', send_to_obsidian: 'An Obsidian senden', github_failed: 'Erstellung fehlgeschlagen:', github_success: 'Erstellung erfolgreich:', configure: 'Bitte konfigurieren Sie zuerst Ihre GitHub-Informationen', menu: 'In Markdown konvertieren', guide: ` - Verwenden Sie die **Pfeiltasten**, um Elemente auszuwählen: - Oben: Elternelement auswählen - Unten: Erstes Kindelement auswählen - Links: Vorheriges Geschwisterelement auswählen - Rechts: Nächstes Geschwisterelement auswählen - Verwenden Sie das **Scrollrad**, um hinein- und herauszuzoomen: - Oben: Elternelement auswählen - Unten: Erstes Kindelement auswählen - Klicken Sie, um ein Element auszuwählen - Drücken Sie die \`Esc\`-Taste, um die Auswahl abzubrechen ` } } const getTranslations = (lang) => { for (const key in translations) { if (key === lang || key.split(',').includes(lang)) { return translations[key] } } return translations['en'] } const translate = new Proxy( function (key) { const lang = userLang const strings = getTranslations(lang) return strings[key] || translations['en'][key] }, { get(target, prop) { const lang = userLang const strings = getTranslations(lang) return strings[prop] || translations['en'][prop] } } ) // 全局变量 var isSelecting = false var selectedElement = null let shortCutConfig, obsidianConfig // 读取配置 // 初始化快捷键配置 let storedShortCutConfig = GM_getValue('shortCutConfig') if (Object.keys(shortCutUserConfig).length !== 0) { GM_setValue('shortCutConfig', JSON.stringify(shortCutUserConfig)) shortCutConfig = shortCutUserConfig } else if (storedShortCutConfig) { shortCutConfig = JSON.parse(storedShortCutConfig) } // 初始化Obsidian配置 let storedObsidianConfig = GM_getValue('obsidianConfig') if (Object.keys(obsidianUserConfig).length !== 0) { GM_setValue('obsidianConfig', JSON.stringify(obsidianUserConfig)) obsidianConfig = obsidianUserConfig } else if (storedObsidianConfig) { obsidianConfig = JSON.parse(storedObsidianConfig) } // HTML2Markdown function convertToMarkdown(element) { var html = element.outerHTML let turndownMd = turndownService.turndown(html) turndownMd = turndownMd.replaceAll('[\n\n]', '[]') // 防止 <a> 元素嵌套的暂时方法,并不完善 return turndownMd } // 预览 function showMarkdownModal(markdown) { var $modal = $(` <div class="h2m-modal-overlay"> <div class="h2m-modal"> <textarea>${markdown}</textarea> <div class="h2m-preview">${marked.parse(markdown)}</div> <div class="h2m-buttons"> <button class="h2m-copy">${translate.copy}</button> <button class="h2m-download">${translate.download_md}</button> <button class="h2m-github">${translate.send_to_github}</button> <select class="h2m-obsidian-select">${translate.send_to_obsidian}</select> </div> <button class="h2m-close">X</button> </div> </div> `) $modal.find('.h2m-obsidian-select').append($('<option>').val('').text(`${translate.send_to_obsidian}`)) for (const vault in obsidianConfig) { for (const path of obsidianConfig[vault]) { // 插入元素 const $option = $('<option>') .val(`obsidian://advanced-uri?vault=${vault}&filepath=${path}`) .text(`${vault}: ${path}`) $modal.find('.h2m-obsidian-select').append($option) } } $modal.find('textarea').on('input', function () { // console.log("Input event triggered"); var markdown = $(this).val() var html = marked.parse(markdown) // console.log("Markdown:", markdown); // console.log("HTML:", html); $modal.find('.h2m-preview').html(html) }) $modal.on('keydown', function (e) { if (e.key === 'Escape') { $modal.remove() } }) $modal.find('.h2m-copy').on('click', function () { // 复制到剪贴板 GM_setClipboard($modal.find('textarea').val()) $modal.find('.h2m-copy').text(`${translate.copied}`) setTimeout(() => { $modal.find('.h2m-copy').text(`${translate.copy}`) }, 1000) }) $modal.find('.h2m-github').on('click', function () { if (!github_token || !github_owner || !github_repo) { alert(`${translate.configure}`) return } const labels = ['web-clipper']//标签,可多项 const markdown = $modal.find('textarea').val() const title = markdown.split('\n')[0] const body = markdown createIssue(github_token, github_owner, github_repo, title, body, labels) }) $modal.find('.h2m-download').on('click', function () { // 下载 var markdown = $modal.find('textarea').val() var blob = new Blob([markdown], { type: 'text/markdown' }) var url = URL.createObjectURL(blob) var a = document.createElement('a') a.href = url // 当前页面标题 + 时间 a.download = `${document.title}-${new Date().toISOString().replace(/:/g, '-')}.md` a.click() }) $modal.find('.h2m-obsidian-select').on('change', function () { // 发送到 Obsidian const val = $(this).val() if (!val) return const markdown = $modal.find('textarea').val() GM_setClipboard(markdown) const title = document.title.replaceAll(/[\\/:*?"<>|]/g, '_') // File name cannot contain any of the following characters: * " \ / < > : | ? const url = `${val}${title}.md&clipboard=true` window.open(url) }) $modal.find('.h2m-close').on('click', function () { // 关闭按钮 X $modal.remove() }) // 同步滚动 // 获取两个元素 var $textarea = $modal.find('textarea') var $preview = $modal.find('.h2m-preview') var isScrolling = false // 当 textarea 滚动时,设置 preview 的滚动位置 $textarea.on('scroll', function () { if (isScrolling) { isScrolling = false return } var scrollPercentage = this.scrollTop / (this.scrollHeight - this.offsetHeight) $preview[0].scrollTop = scrollPercentage * ($preview[0].scrollHeight - $preview[0].offsetHeight) isScrolling = true }) // 当 preview 滚动时,设置 textarea 的滚动位置 $preview.on('scroll', function () { if (isScrolling) { isScrolling = false return } var scrollPercentage = this.scrollTop / (this.scrollHeight - this.offsetHeight) $textarea[0].scrollTop = scrollPercentage * ($textarea[0].scrollHeight - $textarea[0].offsetHeight) isScrolling = true }) $(document).on('keydown', function (e) { if (e.key === 'Escape' && $('.h2m-modal-overlay').length > 0) { $('.h2m-modal-overlay').remove() } }) $('body').append($modal) } // 开始选择 function startSelecting() { $('body').addClass('h2m-no-scroll') // 防止页面滚动 isSelecting = true // 操作指南 tip(marked.parse(translate.guide)) } // 结束选择 function endSelecting() { isSelecting = false $('.h2m-selection-box').removeClass('h2m-selection-box') $('body').removeClass('h2m-no-scroll') $('.h2m-tip').remove() } function tip(message, timeout = null) { var $tipElement = $('<div>') .addClass('h2m-tip') .html(message) .appendTo('body') .hide() .fadeIn(200) if (timeout === null) { return } setTimeout(function () { $tipElement.fadeOut(200, function () { $tipElement.remove() }) }, timeout) } // Turndown 配置 var turndownPluginGfm = TurndownPluginGfmService var turndownService = new TurndownService({ codeBlockStyle: 'fenced' }) turndownPluginGfm.gfm(turndownService) // 引入全部插件 // turndownService.addRule('strikethrough', { // filter: ['del', 's', 'strike'], // replacement: function (content) { // return '~' + content + '~' // } // }); // turndownService.addRule('latex', { // filter: ['mjx-container'], // replacement: function (content, node) { // const text = node.querySelector('img')?.title; // const isInline = !node.getAttribute('display'); // if (text) { // if (isInline) { // return '$' + text + '$' // } // else { // return '$$' + text + '$$' // } // } // return ''; // } // }); // 添加CSS样式 GM_addStyle(` .h2m-selection-box { border: 2px dashed #f00; background-color: rgba(255, 0, 0, 0.2); } .h2m-no-scroll { overflow: hidden; z-index: 9997; } .h2m-modal { position: fixed; top: 50%; left: 50%; transform: translate(-50%, -50%); width: 80%; height: 80%; background: white; border-radius: 10px; display: flex; flex-direction: row; z-index: 9999; } .h2m-modal-overlay { position: fixed; top: 0; left: 0; width: 100%; height: 100%; background: rgba(0, 0, 0, 0.5); z-index: 9998; } .h2m-modal textarea, .h2m-modal .h2m-preview { width: 50%; height: 100%; padding: 20px; box-sizing: border-box; overflow-y: auto; } .h2m-modal .h2m-buttons { position: absolute; bottom: 10px; right: 10px; } .h2m-modal .h2m-buttons button, .h2m-modal .h2m-obsidian-select { margin-left: 10px; background-color: #4CAF50; /* Green */ border: none; color: white; padding: 13px 16px; border-radius: 10px; text-align: center; text-decoration: none; display: inline-block; font-size: 16px; transition-duration: 0.4s; cursor: pointer; } .h2m-modal .h2m-buttons button:hover, .h2m-modal .h2m-obsidian-select:hover { background-color: #45a049; } .h2m-modal .h2m-close { position: absolute; top: 10px; right: 10px; cursor: pointer; width: 25px; height: 25px; background-color: #f44336; color: white; font-size: 16px; border-radius: 50%; display: flex; justify-content: center; align-items: center; } .h2m-tip { position: fixed; top: 22%; left: 82%; transform: translate(-50%, -50%); background-color: white; border: 1px solid black; padding: 8px; z-index: 9999; border-radius: 10px; box-shadow: 5px 5px 10px rgba(0, 0, 0, 0.5); background-color: rgba(255, 255, 255, 0.7); } `) // 注册触发器 shortCutConfig = shortCutConfig ? shortCutConfig : { 'Shift': false, 'Ctrl': true, 'Alt': false, 'Key': 'm' } $(document).on('keydown', function (e) { if (e.ctrlKey === shortCutConfig['Ctrl'] && e.altKey === shortCutConfig['Alt'] && e.shiftKey === shortCutConfig['Shift'] && e.key.toUpperCase() === shortCutConfig['Key'].toUpperCase()) { e.preventDefault() startSelecting() } // else { // console.log(e.ctrlKey, e.altKey, e.shiftKey, e.key.toUpperCase()); // } }) // $(document).on('keydown', function (e) { // if (e.ctrlKey && e.key === 'm') { // e.preventDefault(); // startSelecting() // } // }); GM_registerMenuCommand(`${translate.menu}`, function () { startSelecting() }) $(document).on('mouseover', function (e) { // 开始选择 if (isSelecting) { $(selectedElement).removeClass('h2m-selection-box') selectedElement = e.target $(selectedElement).addClass('h2m-selection-box') } }).on('wheel', function (e) { // 滚轮事件 if (isSelecting) { e.preventDefault() if (e.originalEvent.deltaY < 0) { selectedElement = selectedElement.parentElement ? selectedElement.parentElement : selectedElement // 扩大 if (selectedElement.tagName === 'HTML' || selectedElement.tagName === 'BODY') { selectedElement = selectedElement.firstElementChild } } else { selectedElement = selectedElement.firstElementChild ? selectedElement.firstElementChild : selectedElement // 缩小 } $('.h2m-selection-box').removeClass('h2m-selection-box') $(selectedElement).addClass('h2m-selection-box') } }).on('keydown', function (e) { // 键盘事件 if (isSelecting) { e.preventDefault() if (e.key === 'Escape') { endSelecting() return } switch (e.key) { // 方向键:上下左右 case 'ArrowUp': selectedElement = selectedElement.parentElement ? selectedElement.parentElement : selectedElement // 扩大 if (selectedElement.tagName === 'HTML' || selectedElement.tagName === 'BODY') { // 排除HTML 和 BODY selectedElement = selectedElement.firstElementChild } break case 'ArrowDown': selectedElement = selectedElement.firstElementChild ? selectedElement.firstElementChild : selectedElement // 缩小 break case 'ArrowLeft': // 寻找上一个元素,若是最后一个子元素则选择父元素的下一个兄弟元素,直到找到一个元素 var prev = selectedElement.previousElementSibling while (prev === null && selectedElement.parentElement !== null) { selectedElement = selectedElement.parentElement prev = selectedElement.previousElementSibling ? selectedElement.previousElementSibling.lastChild : null } if (prev !== null) { if (selectedElement.tagName === 'HTML' || selectedElement.tagName === 'BODY') { selectedElement = selectedElement.firstElementChild } selectedElement = prev } break case 'ArrowRight': var next = selectedElement.nextElementSibling while (next === null && selectedElement.parentElement !== null) { selectedElement = selectedElement.parentElement next = selectedElement.nextElementSibling ? selectedElement.nextElementSibling.firstElementChild : null } if (next !== null) { if (selectedElement.tagName === 'HTML' || selectedElement.tagName === 'BODY') { selectedElement = selectedElement.firstElementChild } selectedElement = next } break } $('.h2m-selection-box').removeClass('h2m-selection-box') $(selectedElement).addClass('h2m-selection-box') // 更新选中元素的样式 } } ).on('mousedown', function (e) { // 鼠标事件,选择 mousedown 是因为防止点击元素后触发其他事件 if (isSelecting) { e.preventDefault() var markdown = convertToMarkdown(selectedElement) showMarkdownModal(markdown) endSelecting() } }) function createIssue(token, owner, repo, title, body, labels = []) { const url = `https://api.github.com/repos/${owner}/${repo}/issues` const issueData = { title: title, body: body, labels: labels } const xhr = new XMLHttpRequest() xhr.open('POST', url, true) xhr.setRequestHeader('Authorization', `token ${token}`) xhr.setRequestHeader('Accept', 'application/vnd.github+json') xhr.setRequestHeader('Content-Type', 'application/json') xhr.onreadystatechange = function () { if (xhr.readyState === 4) { if (xhr.status >= 200 && xhr.status < 300) { const response = JSON.parse(xhr.responseText) alert(`${translate.github_success}\n ${response.html_url}`) } else { alert(`${translate.github_failed}\n ${xhr.status}\n ${xhr.statusText}\n ${xhr.responseText}`) console.error(`${translate.github_failed}`, xhr.status, xhr.statusText, xhr.responseText) } } } xhr.send(JSON.stringify(issueData)) } })()