您需要先安装一个扩展,例如 篡改猴、Greasemonkey 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 暴力猴,之后才能安装此脚本。
您需要先安装一个扩展,例如 篡改猴 或 Userscripts ,之后才能安装此脚本。
您需要先安装一款用户脚本管理器扩展,例如 Tampermonkey,才能安装此脚本。
您需要先安装用户脚本管理器扩展后才能安装此脚本。
通过接口获取目标信息,自动跳转并点赞收藏作品
// ==UserScript== // @name 抖音自动点击 // @match https://www.douyin.com/user/* // @namespace vurses // @license Mit // @version 1.0.2 // @author layenh // @homepage https://github.com/vruses/tiktok-auto-click // @@supportURL https://github.com/vruses/tiktok-auto-click/issues // @grant none // @grant GM_setValue // @grant GM_getValue // @grant GM_xmlhttpRequest // @require https://update.gf.qytechs.cn/scripts/534265/1579489/UserPanel.js // @require https://update.gf.qytechs.cn/scripts/534382/1580036/NumberInputWidget.js // @descript 通过接口获取目标信息,自动跳转并点赞收藏作品 // @icon  // @description 通过接口获取目标信息,自动跳转并点赞收藏作品 // ==/UserScript== // 跳转频率,没有则默认为3000 const Time = GM_getValue("redirectInterval", 3) * 1000; console.log(Time); const api = GM_getValue("storedURL", ""); const baseURL = "https://www.douyin.com/user/"; let userSecUid = ""; console.log(api); // 创建设置面板加到dom中,不等window.onload也没事 const userPanel = createPanel(); // 数字输入框组件 const numberInput = createNumberInput(); numberInput.style = "display: flex;justify-content: center;"; numberInput.style.marginTop = "6px"; numberInput.addEventListener("click", function (e) { e.stopPropagation(); // 阻止事件继续冒泡 }); userPanel.shadowRoot .querySelector(".float-ball .menu-panel") .appendChild(numberInput); document.body.appendChild(userPanel); // 针对接口的简易封装 const gm_xhr = function (api, method) { return new Promise((resolve, reject) => { GM_xmlhttpRequest({ method, url: api, onload: resolve, onerror: reject, }); }).then((response) => { return response.responseText; }); }; // 如果未设置接口,则先初始化设置 if (!api) { return; // 默认先在个人主页启动 } else if (location.href.includes(baseURL + "self")) { gm_xhr(api, "GET") .then((res) => { console.log(JSON.parse(JSON.parse(res).result)); userSecUid = JSON.parse(JSON.parse(res).result).secuid; console.log(userSecUid); setTimeout(() => { location.href = baseURL + userSecUid; }, 5000); }) .catch((e) => { // 可能服务器挂了,也可能接口有问题 console.warn("接口请求出错:" + e); }); return; } // 监听 space 键按下是否需要暂停 window.addEventListener("keydown", function onKey(e) { if (e.code === "Space") { console.log("spacedown"); GM_setValue("needPause", true); // 持久化暂停 document.dispatchEvent(new Event("resumeExecution")); } }); // 如果在其他用户主页,获取下一个用户的secUid gm_xhr(api, "GET") .then((res) => { console.log(JSON.parse(JSON.parse(res).result)); userSecUid = JSON.parse(JSON.parse(res).result).secuid; console.log(userSecUid); }) .catch((e) => { location.href = baseURL + "self"; console.warn("接口请求出错:" + e); }); // webWork后台定时器管理视频切换频率 const workerJs = function () { class TimerManager { constructor() { this.timers = new Map(); } set(key, callback, delay) { this.clean(key); const id = setTimeout(() => { callback(); }, delay); this.timers.set(key, id); } clean(key) { if (this.timers.has(key)) { clearTimeout(this.timers.get(key)); this.timers.delete(key); } } cleanAll() { for (let id of this.timers.values()) { clearTimeout(id); } this.timers.clear(); } has(key) { return this.timers.has(key); } } const manager = new TimerManager(); // 只有一种倒计时,暂时先这样用着 self.addEventListener("message", function (e) { manager.set( "redirectNextUser", () => self.postMessage("redirectNextUser"), e.data ); }); }; /** * @param {string} selectorKey * @returns {Element} */ const $queryPlayerElStatus = function (selectorKey) { selector = 'div[data-e2e="video-player-key"]'.replace("key", selectorKey); const ele = document.querySelector(selector); if (ele?.dataset?.e2eState.includes("-no-")) { return ele; //现在只找第一个就行 } else { return false; } }; const pageLoadPromise = new Promise((resolve, reject) => { window.addEventListener("load", resolve); }); pageLoadPromise.then(() => { // ssr // ele具有缓存性质,需要每次重新查询 // let postList = document.querySelector("div[data-e2e=user-post-list]"); // 不要在视频页面加载 /**作品列表加载和视频加载 * @param {Element} root * @param {String} selector * @param {Function} action * @returns {Promise} * */ const loopLoad = function (root, selector, action) { return new Promise((resolve) => setTimeout(resolve, 500)) .then(() => { const el = root.querySelector(selector); if (el) { action(el); return Promise.resolve(); } else { console.log(`${selector} loading...`); return loopLoad(root, selector, action); } }) .catch((error) => { return Promise.reject("等待加载遇到错误,请联系开发者:" + error); }); }; // 通过派发自定义事件触发监听器,终止暂停 async function pauseIfNeeded() { const needPause = GM_getValue("needPause", false); if (!needPause) return Promise.resolve(); return new Promise((resolve) => { const resumeHandler = () => { document.removeEventListener("resumeExecution", resumeHandler); resolve(); }; document.addEventListener("resumeExecution", resumeHandler); }); } // 等待作品列表加载 // 等待视频加载 /** * @param {Worker} worker * */ async function firstStep(worker) { await loopLoad(document, "div[data-e2e=user-post-list] a", (el) => el.click() ); // 作品列表第一条作品 await loopLoad(document, "video", (el) => { //视频播放页 }); // 是否点赞? if (GM_getValue("likeStatus")) { const result = $queryPlayerElStatus("digg"); if (result) result.click(); } // 是否收藏? if (GM_getValue("collectStatus")) { const result = $queryPlayerElStatus("collect"); if (result) result.click(); } // 暂停直到用户按空格 await pauseIfNeeded(); // 清除状态,避免影响下一次 GM_setValue("needPause", false); // 任务结束,通知worker倒计时 worker.postMessage(Time); return Promise.resolve(); } // 后台定时器管理者实例 workerJs.toString(); const blob = new Blob([`(${workerJs})()`], { type: "application/javascript", }); const url = URL.createObjectURL(blob); const worker = new Worker(url); // 等待作品加载=>点进第一个作品=>等待视频加载=>点赞?收藏?=>定向至下一个用户主页 async function runSteps() { firstStep(worker); } worker.addEventListener("message", function (e) { location.href = baseURL + userSecUid; console.log(e.data); }); runSteps(); });
QingJ © 2025
镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址