// ==UserScript== // @name 复制标题和地址 // @namespace http://tampermonkey.net/ // @version 0.7 // @description 一键复制标题和地址为Markdown格式并带上当前时间(myFirstScript) // @author LiarCoder // @match *://*/* // @grant GM_addStyle // @icon  // @downloadURL https://update.greasyfork.icu/scripts/483589/%E5%A4%8D%E5%88%B6%E6%A0%87%E9%A2%98%E5%92%8C%E5%9C%B0%E5%9D%80.user.js // @updateURL https://update.greasyfork.icu/scripts/483589/%E5%A4%8D%E5%88%B6%E6%A0%87%E9%A2%98%E5%92%8C%E5%9C%B0%E5%9D%80.meta.js // ==/UserScript== (function () { 'use strict'; /** * 创建DOM元素 * @param {string} eleName - 元素标签名 * @param {string} text - 元素的文本内容 * @param {Object} attrs - 元素的属性键值对 * @returns {HTMLElement} 创建的DOM元素 */ function createEle(eleName, text, attrs) { const ele = document.createElement(eleName); ele.innerText = text; for (const k in attrs) { ele.setAttribute(k, attrs[k]); } return ele; } // 添加提示框样式 const btnStyle = ` #copy-title-and-location { position: fixed; top: 100px; left: -95px; opacity: 0.3; z-index: 2147483647; background-image: none; cursor: pointer; color: #fff; background-color: #0084ff !important; margin: 5px 0px; width: auto; border-radius: 3px; border: #0084ff; outline: none; padding: 3px 6px; height: 26px; font-family: Arial, sans-serif; font-size: 12px; transition: left, 0.5s; } #copy-title-and-location:hover { left: 0px; opacity: 1; } #copy-title-and-location svg { width: auto; vertical-align: middle; margin-left: 10px; border-style: none; text-align: center; display: inline-block !important; margin-bottom: 2px; } `; // 将按钮图标由原来的img改为了svg,以增强适应性,同时也将对svg的样式设置移到了上面的 btnStyle 中 const iconSVG = ''; const btn = createEle("button", "", { id: "copy-title-and-location" }); btn.innerHTML = "复制标题和地址" + iconSVG; const date = new Date(); const timeStamp = `更新:${date .toLocaleDateString() .replace("/", "年") .replace("/", "月")}日${date.toLocaleTimeString("chinese", { hour12: false, })}`; /** * 复制文本到剪贴板 * @param {string} text - 要复制的文本内容 */ const copyToClipboard = async (text) => { try { await navigator.clipboard.writeText(text); } catch (err) { console.log("尝试使用备用复制方法:" + err); try { // 创建临时textarea元素用于复制 const textarea = document.createElement("textarea"); textarea.style.cssText = 'position:fixed;top:-999px;left:-999px;'; textarea.value = text; document.body.appendChild(textarea); textarea.select(); document.execCommand("copy"); document.body.removeChild(textarea); } catch (err) { console.error("复制失败:", err); } } }; /** * 获取当前页面的引用地址 * @param {boolean} hasQuote - 是否添加引用符号(>) * @returns {string} 格式化后的引用地址 */ const getAddress = (hasQuote = true) => { const titleInfo = document.title; let address = `参考:[${titleInfo}](${location})`; // 针对不同网站的特殊处理规则 const siteHandlers = { 'mp.weixin.qq.com': () => { const officialAccount = document.getElementById("js_name"); const publishDate = document.getElementById("publish_time"); if (officialAccount && publishDate) { publishDate.click(); return `参考:[【微信公众号:${officialAccount.innerText}${publishDate.innerText}】${titleInfo}](${location})`; } return address; } // 可在此处添加其他网站的特殊处理规则 }; const domain = location.hostname; for (const site in siteHandlers) { if (domain.includes(site)) { address = siteHandlers[site](); break; } } return hasQuote ? `\n> ${address}` : address; }; // 注册按钮事件 btn.addEventListener("click", async (e) => { await copyToClipboard(timeStamp + getAddress()); }); btn.addEventListener("contextmenu", async (e) => { e.preventDefault(); await copyToClipboard(getAddress(false)); }); // document.body.appendChild(style); // 这种写法会导致脚本在