您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
抖音工具
// ==UserScript== // @name [buyi] 抖音工具 // @namespace buyi // @version 1.0.8 // @description 抖音工具 // @author buyi // @match *://*.douyin.com/* // @icon  // @grant unsafeWindow // @license MIT // ==/UserScript== (function () { 'use strict'; function main() { // 标题尾部增加复制按钮 function addCopyTitleIcon() { let titleNodes = document.querySelectorAll('#video-info-wrap > div.video-info-detail.isVideoInfoOptimise > div > div.title > div > div > span > span') if (!titleNodes || titleNodes.length == 0) { log.log("未找到标题节点") return } for (let i = 0; i < titleNodes.length; i++) { let titleNode = titleNodes[i] if (!titleNode) { // log.log("未找到标题节点") continue } let titleWrapNode = titleNode.parentNode.parentNode.parentNode.parentNode // 如果已添加,则不重复添加 if (titleWrapNode.querySelector('.byjs-copyIcon')) { // log.log("已添加复制按钮") continue } titleWrapNode.style.position = "relative" // 添加样式,如果已添加就跳过 if (!document.querySelector('#byjs-iconWrapStyle')) { var style = document.createElement('style') style.id = "byjs-iconWrapStyle" style.textContent = ` .byjs-iconWrap { position: absolute; left: 100%; bottom: 2px; z-index: 999999; display: flex; align-items: center; justify-content: center; } .byjs-copyIcon { width: 20px; height: 20px; cursor: pointer; margin-left: 10px; display: inline-block; vertical-align: middle; } ` document.head.appendChild(style); } let copyTitleWithAuthor = document.createRange().createContextualFragment(`<span class="byjs-iconWrap"> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制标题</span> <img id="copyTitle" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制作者</span> <img id="copyAuthor" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制标题+作者</span> <img id="copyTitleWithAuthor" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <style> </style> </span> `) titleWrapNode.appendChild(copyTitleWithAuthor) titleWrapNode.querySelector('#copyTitle').onclick = function () { let title = titleNode.innerText navigator.clipboard.writeText(title) showToast("复制成功") log.log("复制成功: " + title) } titleWrapNode.querySelector('#copyAuthor').onclick = function () { let author = titleWrapNode.parentNode.childNodes[0].childNodes[0].innerText navigator.clipboard.writeText(author) showToast("复制成功") log.log("复制成功: " + author) } titleWrapNode.querySelector('#copyTitleWithAuthor').onclick = function () { let title = titleNode.innerText let author = titleWrapNode.parentNode.childNodes[0].childNodes[0].innerText navigator.clipboard.writeText(title + " - " + author) showToast("复制成功") log.log("复制成功: " + title + " - " + author) } } } // 详情页标题下方增加复制按钮 function addCopyTitleIconForDetail() { let titleNode = document.querySelector('#douyin-right-container > div.parent-route-container.route-scroll-container.IhmVuo1S > div > div > div.leftContainer.MRADF45Z > div.sJhfX08v > div > div.b3uZicw5.cb5piKg6 > div > h1') if (!titleNode) { log.log("未找到标题节点") return } let addPositionNode = titleNode.childNodes[0].parentNode.parentNode.parentNode.parentNode // 如果已添加,则不重复添加 if (addPositionNode.querySelector('.byjs-copyIcon')) { // log.log("已添加复制按钮") return } // 添加样式,如果已添加就跳过 if (!document.querySelector('#byjs-detail-iconWrapStyle')) { var style = document.createElement('style') style.id = "byjs-iconWrapStyle" style.textContent = ` .byjs-detail-iconWrap { z-index: 999999; } .byjs-copyIcon { width: 20px; height: 20px; cursor: pointer; margin-left: 10px; display: inline-block; vertical-align: middle; } ` document.head.appendChild(style); } let copyTitleWithAuthor = document.createRange().createContextualFragment(`<span class="byjs-detail-iconWrap"> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制标题</span> <img id="copyTitle" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制作者</span> <img id="copyAuthor" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <span class="byjs-tooltip"> <span class="byjs-tooltiptext">复制标题+作者</span> <img id="copyTitleWithAuthor" class="byjs-copyIcon" src="${iconCopyBase64}"/> </span> <style> </style> </span> `) addPositionNode.insertBefore(copyTitleWithAuthor, addPositionNode.childNodes[1]) addPositionNode.querySelector('#copyTitle').onclick = function () { let title = titleNode.innerText navigator.clipboard.writeText(title) showToast("复制成功") log.log("复制成功: " + title) } let authorSelector = '#douyin-right-container > div.parent-route-container.route-scroll-container.IhmVuo1S > div > div > div.detailPage.W_7gCbBd > div > div.cHwSTMd3 > div.OMAnlCHg > a > div > span > span' addPositionNode.querySelector('#copyAuthor').onclick = function () { let author = '@' + document.querySelector(authorSelector).innerText navigator.clipboard.writeText(author) showToast("复制成功") log.log("复制成功: " + author) } addPositionNode.querySelector('#copyTitleWithAuthor').onclick = function () { let title = titleNode.innerText let author = '@' + document.querySelector(authorSelector).innerText navigator.clipboard.writeText(title + " - " + author) showToast("复制成功") log.log("复制成功: " + title + " - " + author) } } setInterval(() => { if (RegExp(/.*douyin.com\/video\/.*/).test(window.location.href)) { log.log("setInterval 执行 addCopyTitleIconForDetail") addCopyTitleIconForDetail() } else { log.log("setInterval 执行 main") addCopyTitleIcon() } }, 2000); } // ========================================== main函数结束 ========================================== const iconCopyBase64 = '' const tool = { print(level, msg, ...args) { const now = new Date() const year = now.getFullYear() const month = (now.getMonth() + 1 < 10 ? "0" : "") + (now.getMonth() + 1) const day = (now.getDate() < 10 ? "0" : "") + now.getDate() const hour = (now.getHours() < 10 ? "0" : "") + now.getHours() const minute = (now.getMinutes() < 10 ? "0" : "") + now.getMinutes() const second = (now.getSeconds() < 10 ? "0" : "") + now.getSeconds() const timenow = "[" + year + "-" + month + "-" + day + " " + hour + ":" + minute + ":" + second + "]" const host = location.host console[level](`[🚀 ~ 抖音工具 ${host} 🎉]` + timenow + " > ", msg, ...args) } } const log = { log(msg, ...args) { tool.print("log", msg, ...args) }, info(msg, ...args) { tool.print("info", msg, ...args) }, warn(msg, ...args) { tool.print("warn", msg, ...args) }, error(msg, ...args) { tool.print("error", msg, ...args) }, debug(msg, ...args) { tool.print("debug", msg, ...args) } } // ===================================================== log end =============================================== // ========================= toast ============================= function showToast(msg) { const toast = document.createElement("div") const style = document.createElement("style") style.id = "byjs-toast-style" style.textContent = ` .byjs-toast { position: fixed; top: 10px; left: 10px; padding: 10px; background-color: rgba(0, 0, 0, 0.8); color: white; font-size: 16px; z-index: 9999; } ` toast.className = "byjs-toast" toast.innerHTML = msg toast.appendChild(style) document.body.appendChild(toast) setTimeout(() => { document.body.removeChild(toast) }, 3000) } // ========================= toast end ============================= // ========================= tooltip ================================== (function initTooltip() { const style = document.createElement("style") style.id = "byjs-tooltip-style" style.textContent = ` .byjs-tooltip { position: relative; } .byjs-tooltip .byjs-tooltiptext { visibility: hidden; width: 120px; background-color: black; color: #fff; text-align: center; border-radius: 6px; padding: 5px 0; /* 定位 */ position: absolute; top: calc(-10px - 100%); left: -50%; } .byjs-tooltip:hover .byjs-tooltiptext { visibility: visible; } ` document.body.appendChild(style) })() // ========================= tooltip end ================================== // if (!unsafeWindow.$ || !unsafeWindow.jQuery || !unsafeWindow.jQuery.fn.jquery) { // log.log("获取不到$,网站匹配失败 @buyi") // return // } // 域名匹配 if (['lf-zt.douyin.com'].includes(location.host)) { log.log(`非工作目标域名: ${location.host}`) return } log.log("启动中 @buyi") main() log.log("启动完成 @buyi") })();
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址