[buyi] 抖音工具

抖音工具

// ==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或关注我们的公众号极客氢云获取最新地址