Greasy Fork

Greasy Fork is available in English.

TikTok 小助手

获取ttk数据!

目前为 2024-10-10 提交的版本,查看 最新版本

// ==UserScript==
// @name         TikTok 小助手
// @namespace    http://tampermonkey.net/
// @version      5.13
// @description  获取ttk数据!
// @author       Belugu
// @match        https://www.tiktok.com/*
// @grant        none
// @icon         https://iili.io/dy5xjOg.jpg
// ==/UserScript==

(function() {
    'use strict';

    let currentUrl = window.location.href;
    let retryCount = 0;

    // Create a button to toggle display of the data panel
    function createButton(parsedData) {
        const existingButton = document.querySelector('#tiktokDataButton');
        if (existingButton) {
            existingButton.remove();
        }

        const button = document.createElement('button');
        button.id = 'tiktokDataButton';
        button.innerHTML = '🤓';
        button.style.position = 'fixed';
        button.style.top = '10px';
        button.style.right = '200px'; // Move the button to the left
        button.style.zIndex = '10001';
        button.style.padding = '5px';
        button.style.backgroundColor = '#D5E1EC';
        button.style.color = '#ffffff';
        button.style.border = 'none';
        button.style.borderRadius = '12px';
        button.style.cursor = 'pointer';
        button.style.fontSize = '20px';
        button.style.boxShadow = '0 4px #999';
        button.style.transition = 'all 0.1s ease-in-out';
        button.style.userSelect = 'none'; // Prevent emoji from being selected

        button.addEventListener('mousedown', () => {
            button.style.transform = 'translateY(4px)';
            button.style.boxShadow = '0 2px #666';
        });

        button.addEventListener('mouseup', () => {
            button.style.transform = 'translateY(0)';
            button.style.boxShadow = '0 4px #999';
        });

        button.addEventListener('click', () => {
            toggleDataDisplay(parsedData);
        });

        document.body.appendChild(button);
        console.log('Button created and appended to the page.');

        createRefreshButton();
    }

    // Create a button to manually refresh data extraction
    function createRefreshButton() {
        const existingRefreshButton = document.querySelector('#tiktokRefreshButton');
        if (existingRefreshButton) {
            existingRefreshButton.remove();
        }

        const refreshButton = document.createElement('button');
        refreshButton.id = 'tiktokRefreshButton';
        refreshButton.innerHTML = '🔄 刷新数据';
        refreshButton.style.position = 'fixed';
        refreshButton.style.top = '10px';
        refreshButton.style.right = '270px'; // Placed next to the main button
        refreshButton.style.zIndex = '10001';
        refreshButton.style.padding = '5px';
        refreshButton.style.backgroundColor = '#4CAF50';
        refreshButton.style.color = '#ffffff';
        refreshButton.style.border = 'none';
        refreshButton.style.borderRadius = '12px';
        refreshButton.style.cursor = 'pointer';
        refreshButton.style.fontSize = '20px';
        refreshButton.style.boxShadow = '0 4px #006400';
        refreshButton.style.transition = 'all 0.1s ease-in-out';
        refreshButton.style.userSelect = 'none';

        refreshButton.addEventListener('mousedown', () => {
            refreshButton.style.transform = 'translateY(4px)';
            refreshButton.style.boxShadow = '0 2px #006400';
        });


        refreshButton.addEventListener('mouseup', () => {
            refreshButton.style.transform = 'translateY(0)';
            refreshButton.style.boxShadow = '0 4px #006400'; // 改为深绿色
        });


        refreshButton.addEventListener('click', () => {
            console.log('Manual refresh button clicked.');
            retryCount = 0;
            currentUrl = window.location.href; // Update to current URL
            extractStats(true);
        });

        document.body.appendChild(refreshButton);
        console.log('Refresh button created and appended to the page.');
    }

    // Toggle the display of the data panel on the main page
    function toggleDataDisplay(parsedData) {
        let dataContainer = document.querySelector('#tiktokDataContainer');
        if (dataContainer) {
            dataContainer.style.transform = 'translateX(100%)';
            dataContainer.style.opacity = '0';
            setTimeout(() => {
                dataContainer.remove();
            }, 500);
            return;
        }

        dataContainer = document.createElement('div');
        dataContainer.id = 'tiktokDataContainer';
        dataContainer.style.transition = 'transform 0.5s ease-in-out, opacity 0.5s ease-in-out';
        dataContainer.style.transform = 'translateX(100%)'; // 初始位置完全在屏幕外
        dataContainer.style.opacity = '0';
        dataContainer.style.position = 'fixed';
        dataContainer.style.top = '60px';
        dataContainer.style.right = '20px'; // 设置数据面板初始在屏幕右侧
        dataContainer.style.width = '300px';
        dataContainer.style.maxHeight = '400px';
        dataContainer.style.overflowY = 'auto';
        dataContainer.style.backgroundColor = '#ffffff';
        dataContainer.style.border = '1px solid #ccc';
        dataContainer.style.borderRadius = '8px';
        dataContainer.style.boxShadow = '0px 0px 10px rgba(0, 0, 0, 0.1)';
        dataContainer.style.padding = '15px';
        dataContainer.style.zIndex = '10000';

        const title = document.createElement('h4');
        title.textContent = '好!发现了';
        title.style.color = '#000000';
        title.style.marginBottom = '10px';
        dataContainer.appendChild(title);

        const image = document.createElement('img');
        image.src = '';
        image.style.width = '50%'; // 设置宽度
        image.style.borderRadius = '4px'; // 可选:添加圆角
        dataContainer.appendChild(image);

function createJsonElement(data, container) {
    const fields = ['diggCount', 'playCount', 'commentCount', 'shareCount', 'collectCount', 'createTime'];

    // 提取账户名,去掉 @ 符号
    const accountName = window.location.pathname.split('/')[1].replace('@', '');

    // Base64 编码的复制图标
    const base64CopyIcon = " ";

    // 创建账户名和复制图标
    const accountRow = document.createElement('div');
    accountRow.style.display = 'flex';
    accountRow.style.alignItems = 'center';
    accountRow.style.marginBottom = '5px';

    const accountNameElement = document.createElement('div');
    accountNameElement.style.fontWeight = 'bold';
    accountNameElement.fontSize = '20px'
    accountNameElement.textContent = `${accountName}`;

    const copyAccountIcon = document.createElement('img');
    copyAccountIcon.src = base64CopyIcon; // 使用 Base64 编码的图标
    copyAccountIcon.style.cursor = 'pointer';
    copyAccountIcon.style.width = '20px'; // 设置图标宽度
    copyAccountIcon.style.marginLeft = '10px';

    copyAccountIcon.addEventListener('click', (event) => {
        event.preventDefault();
        navigator.clipboard.writeText(accountName).then(() => {
            showNotification('已复制到剪贴板: ' + accountName);
        }).catch(err => {
            console.error('复制失败: ', err);
        });
    });

    accountRow.appendChild(accountNameElement);
    accountRow.appendChild(copyAccountIcon);
    container.appendChild(accountRow);

    // 创建粉丝数和复制图标
    const followerCountRow = document.createElement('div');
    followerCountRow.style.display = 'flex';
    followerCountRow.style.alignItems = 'center';
    followerCountRow.style.marginBottom = '10px';

    const followerCountText = document.createElement('div');
    followerCountText.textContent = `粉丝数: ${data.followerCount || '未知'}`;

    const copyFollowerIcon = document.createElement('img');
    copyFollowerIcon.src = base64CopyIcon; // 使用 Base64 编码的图标
    copyFollowerIcon.style.cursor = 'pointer';
    copyFollowerIcon.style.width = '20px'; // 设置图标宽度
    copyFollowerIcon.style.marginLeft = '10px';

    copyFollowerIcon.addEventListener('click', (event) => {
        event.preventDefault();
        navigator.clipboard.writeText(data.followerCount).then(() => {
            showNotification('已复制到剪贴板: ' + followerCountText.textContent);
        }).catch(err => {
            console.error('复制失败: ', err);
        });
    });

    followerCountRow.appendChild(followerCountText);
    followerCountRow.appendChild(copyFollowerIcon);
    container.appendChild(followerCountRow);

    // 创建其他统计信息
    fields.forEach(field => {
        if (data.hasOwnProperty(field)) {
            if (field === 'createTime' && data[field] === 0) {
                return; // Skip if createTime is 0
            }

            const item = document.createElement('div');
            item.style.marginBottom = '10px';
            item.style.display = 'flex';
            item.style.alignItems = 'center';

            let text = '';
            if (field === 'diggCount') {
                text = `点赞数: ${data[field]}`;
            } else if (field === 'playCount') {
                text = `播放数: ${data[field]}`;
            } else if (field === 'commentCount') {
                text = `评论数: ${data[field]}`;
            } else if (field === 'shareCount') {
                text = `分享数: ${data[field]}`;
            } else if (field === 'collectCount') {
                text = `收藏数: ${data[field]}`;
            } else if (field === 'createTime') {
                const date = new Date(data[field] * 1000);
                text = `创建时间: ${date.toLocaleString()}`;
            }

            const textElement = document.createElement('span');
            textElement.textContent = text;
            textElement.style.color = '#000000';
            item.appendChild(textElement);

            const copyButtonIcon = document.createElement('img');
            copyButtonIcon.src = base64CopyIcon; // 使用 Base64 编码的图标
            copyButtonIcon.style.cursor = 'pointer';
            copyButtonIcon.style.width = '20px'; // 设置图标宽度
            copyButtonIcon.style.marginLeft = '10px';

            copyButtonIcon.addEventListener('click', (event) => {
                event.preventDefault();
                const date = new Date(data[field] * 1000);
                const formattedDate = date.toISOString().slice(0, 19).replace('T', ' '); // 格式化为 YYYY-MM-DD HH:mm:ss
                navigator.clipboard.writeText(formattedDate).then(() => {
                    showNotification('已复制到剪贴板: ' + formattedDate);
                }).catch(err => {
                    console.error('复制失败: ', err);
                });
            });


            item.appendChild(copyButtonIcon);
            container.appendChild(item);
        }
    });
}










        createJsonElement(parsedData, dataContainer);
        document.body.appendChild(dataContainer);
        setTimeout(() => {
            dataContainer.style.transform = 'translateX(0)'; // 从右向左进入
            dataContainer.style.opacity = '1'; // 渐显
        }, 10);
    }

    // Extract video stats by parsing the script tag with id "__UNIVERSAL_DATA_FOR_REHYDRATION__"
    function extractStats(isManual = false) {
        fetch(window.location.href)
            .then(response => response.text())
            .then(responseText => {
                const scriptMatch = responseText.match(/<script id="__UNIVERSAL_DATA_FOR_REHYDRATION__" type="application\/json">([\s\S]*?)<\/script>/);
                if (scriptMatch) {
                    try {
                        const jsonData = JSON.parse(scriptMatch[1]);
                        console.log('Attempting to extract data from script tag:', jsonData);
                        const stats = findStats(jsonData);
                        if (stats) {
                            console.log('Video stats found:', stats);
                            extractFollowerCount(stats);
                            if (isManual) {
                                showNotification('数据已成功刷新');
                            }
                        } else {
                            console.warn('No relevant stats found in the script tag.');
                        }
                    } catch (e) {
                        console.error('Error parsing script tag:', e);
                    }
                } else {
                    console.warn('Script tag "__UNIVERSAL_DATA_FOR_REHYDRATION__" not found.');
                    if (!isManual) {
                        retryExtractStats(); // Retry if data not available immediately
                    }
                }
            });
    }

    // Retry extractStats after a delay if data is not immediately available
    function retryExtractStats() {
        if (retryCount < 5) { // Retry up to 5 times
            setTimeout(() => {
                console.log('Retrying data extraction...');
                retryCount++;
                extractStats();
            }, 2000);
        } else {
            console.warn('Max retry attempts reached. Data extraction failed.');
        }
    }

    // Extract follower count from the user profile page using fetch API
    function extractFollowerCount(stats) {
        const userUrl = `https://www.tiktok.com/${window.location.pathname.split('/')[1]}`;

        fetch(userUrl)
            .then(response => response.text())
            .then(responseText => {
                const scriptMatch = responseText.match(/<script id="__UNIVERSAL_DATA_FOR_REHYDRATION__" type="application\/json">([\s\S]*?)<\/script>/);
                if (scriptMatch) {
                    try {
                        const obj = JSON.parse(scriptMatch[1]);
                        const followerCount = findFollowerCount(obj);
                        if (followerCount !== null) {
                            stats.followerCount = followerCount;
                            createButton(stats);
                            toggleDataDisplay(stats); // Ensure data panel shows when page loads
                        } else {
                            console.warn('未找到粉丝计数。');
                        }
                    } catch (error) {
                        console.error('解析 JSON 时出错:', error);
                    }
                } else {
                    console.log('未找到包含页面数据的 <script> 标签。');
                }
            })
            .catch(error => {
                console.error('请求用户页面时出错:', error);
            });
    }

    // Run extractStats after the page is fully loaded
    window.addEventListener('load', () => {
        console.log('Page fully loaded, attempting to extract stats.');
        extractStats();
    });

    // Listen for URL changes and re-run extractStats if necessary
    setInterval(() => {
        if (currentUrl !== window.location.href) {
            console.log('URL changed, attempting to extract stats again.');
            currentUrl = window.location.href;
            retryCount = 0; // Reset retry count on URL change
            extractStats();
        }
    }, 1000);

    // Extract video stats directly from the JSON data
    function findStats(jsonData) {
        let result = null;
        function recursiveSearch(obj) {
            for (const key in obj) {
                if (typeof obj[key] === 'object' && obj[key] !== null) {
                    recursiveSearch(obj[key]);
                } else if ((key === 'diggCount' || key === 'playCount' || key === 'commentCount' || key === 'shareCount' || key === 'collectCount' || key === 'createTime') && obj[key] !== 0) {
                    if (!result) {
                        result = {};
                    }
                    result[key] = obj[key];
                }
            }
        }
        recursiveSearch(jsonData);
        return result;
    }

    // Recursively search for follower count in the JSON data
    function findFollowerCount(jsonData) {
        let followerCount = null;
        function recursiveSearch(obj) {
            for (const key in obj) {
                if (key === 'followerCount') {
                    followerCount = obj[key];
                    return;
                }
                if (typeof obj[key] === 'object' && obj[key] !== null) {
                    recursiveSearch(obj[key]);
                }
            }
        }
        recursiveSearch(jsonData);
        return followerCount;
    }

    // Function to show notification
    function showNotification(message) {
        // Create notification container
        const notificationContainer = document.createElement('div');
        notificationContainer.className = 'notificationContainer';
        notificationContainer.style.position = 'fixed';
        notificationContainer.style.bottom = '20px';
        notificationContainer.style.right = '-100%'; // Start from outside the right edge
        notificationContainer.style.width = 'auto';
        notificationContainer.style.padding = '10px 20px';
        notificationContainer.style.backgroundColor = getRandomColor();
        notificationContainer.style.color = '#fff';
        notificationContainer.style.borderRadius = '5px';
        notificationContainer.style.boxShadow = '0px 0px 10px rgba(0, 0, 0, 0.3)';
        notificationContainer.style.zIndex = '10000';
        notificationContainer.style.opacity = '0';
        notificationContainer.style.transition = 'transform 0.5s ease-in-out, opacity 0.5s ease-in-out';
        notificationContainer.textContent = message;

        document.body.appendChild(notificationContainer);

        // Push up old notifications
        const existingNotifications = document.querySelectorAll('.notificationContainer');
        existingNotifications.forEach((container, index) => {
            container.style.bottom = `${20 + (index + 1) * 50}px`;
        });

        // Make notification slide in from the right
        setTimeout(() => {
            notificationContainer.style.right = '20px';
            notificationContainer.style.opacity = '1';
        }, 10);

        // Hide the notification after 3 seconds
        setTimeout(() => {
            notificationContainer.style.opacity = '0';
            notificationContainer.style.right = '-100%';
            setTimeout(() => {
                notificationContainer.remove();
            }, 500);
        }, 3000);
    }

    // Function to get a random background color for the notification
    function getRandomColor() {
        const colors = ['#FF5733', '#33FF57', '#3357FF', '#FF33A1', '#A133FF'];
        return colors[Math.floor(Math.random() * colors.length)];
    }
})();