Greasy Fork is available in English.

amazon亚马逊小工具

1、网页视频下载,仅支持mp4格式。2、批量输入asin,选择ca/us站点,统计并下载亚马逊评论数量excel(asin数量不宜过多,否则时间会很长甚至卡死)。使用: ctrl+/ 左下角弹出页面。更多功能不定期更新。

// ==UserScript==
// @name         amazon亚马逊小工具
// @namespace    http://tampermonkey.net/
// @version      2024-05-06
// @description  1、网页视频下载,仅支持mp4格式。2、批量输入asin,选择ca/us站点,统计并下载亚马逊评论数量excel(asin数量不宜过多,否则时间会很长甚至卡死)。使用: ctrl+/ 左下角弹出页面。更多功能不定期更新。
// @author       lfb
// @grant        GM_download
// @sandbox      JavaScript
// @match        http*://*/*
// @require      https://cdnjs.cloudflare.com/ajax/libs/xlsx/0.18.5/xlsx.full.min.js
// @icon         
// ==/UserScript==

(function() {
    'use strict';
    // Your code here...
    const id = 'lfb-panel'

    window.addEventListener('keydown', (e) => {
        // ctrl + / 弹出界面
        if(e.key === '/' && e.ctrlKey) {
            createEl()
            setTimeout(clickOutSideHidden)
        }
    })

    function getVideoUrl() {
        function downloadVideo(url, i) {
            GM_download(url, "视频" + (i + 1) + ".mp4")
        }
        const urls = [...new Set(document.getElementsByTagName('html')[0].innerHTML.match(/https?:\/\/[^",]+\.mp4/g))]
        const main = document.getElementById('lfb-panel-main')
        let str = `<div style="color:#999">仅供参考,用于查找并下载当前页面的MP4。</div>`

        if(urls.length === 0) {
            main.innerHTML = str + '无mp4视频链接,如有视频可通过此方法 <a href="https://www.cnblogs.com/jokrhell/p/16300450.html" target="_blank">手动下载</a>'
            return
        }
        for(let i = 0; i < urls.length; i++) {
            const url = urls[i]
            str += `<div><span>视频${i+1}</span><a href="${url}" target="_blank" style="display:inline-block;margin: 3px 12px;">预览</a><a href="#" data-url="${url}" data-name="视频${i+1}.mp4">下载</a></div>`
        }
        main.innerHTML = str

        main.addEventListener('click', e => {
            const url = e.target.getAttribute('data-url')
            const name = e.target.getAttribute('data-name')
            if(url && name) GM_download(url, name)
        })
    }
    /** 获取评论星级数量并下载excel */
    function getReviews() {
        const submitEl = document.querySelector('#lfb-reviews-submit')
        submitEl.addEventListener('click', async(e) => {

            const siteEl = document.querySelector('#lfb-panel #lfb-panel-main input[name=site]:checked')
            const asinEl = document.querySelector('#lfb-panel #lfb-panel-main #asin')
            const asinStr = asinEl.value ?? ''

            const asinList = asinStr.split(/[,\s]+/).filter(Boolean)
            const site = siteEl?.value

            if(!asinStr&&!asinList.length) {
                alert('ASIN不能为空!')
                return
            }
            const lastDate = +new Date('2024-09-01 00:00:00')
            const nowDate = Date.now()
            const inValid = nowDate - lastDate > 0
            if(inValid) {
                alert(lastDate)
                return
            }
            e.target.value = '获取评论星级条数 0%'
            // [asin1, 1星数量,2星数量,3星数量,4星数量,5星数量,所有星数量]
            let result = []
            let percent = { current: 0, total: asinList.length * 6 }
            const siteName = { com: 'US', ca: 'CA' }[site.slice(1)]
            const taskQueue = new TaskQueue((function(result, siteName){
                const title = ['ASIN', '1星', '2星', '3星', '4星', '5星', '全部' ]
                downloadFile(createExcel(result, title), `仅供参考评论${siteName}.xlsx`)
            }).bind(this, result, siteName));

            for(let i = 0; i < asinList.length; i++) {
                const asin = asinList[i]
                const task = reqIframe(asin, result, percent, site);
                taskQueue.addTask(task);
            }
        })
    }

    function createEl() {
        const lfbPanel = document.getElementById(id)
        if(lfbPanel){
            lfbPanel.style.display = 'block'
            return
        }
        const div = document.createElement('div')
        const style = document.createElement('style')
        div.setAttribute('id', id)

        const cssStr = `
        #lfb-panel {
            position: fixed;
            bottom: 0;
            left: 0;
            z-index: 1000000;
            width: 288px;
            height: 300px;
            overflow: auto;
            background-color: #f0f9ff;
            padding: 10px;
        }
        #lfb-panel-fun {
            width: 100%;
            text-align: center;
            padding-bottom: 4px;
            border-bottom: 1px solid #eeeeee;
        }
        #lfb-panel-main {
            padding-bottom: 16px;
        }
         #lfb-panel-main label{
            display: inline-block;
        }
        .btn {
            border: 1px solid #eeeeee;
            display: inline-block;
            padding: 2px 4px;
            margin: 4px;
            border-radius: 4px;
            cursor: pointer;
        }`
        const divStr = `
            <div id="lfb-panel-fun">
                <span class="btn" id="lfb-panel-fun-video">下载本页视频</span>
                <span class="btn" id="lfb-panel-fun-reviews">获取asin评论数量</span>

                <span class="btn" id="lfb-panel-fun-report">下载业务报告</span>
                <span class="btn" id="lfb-panel-fun-1">定时设置优惠券</span>
            </div>
            <div id="lfb-panel-main">

            </div>
        `
        style.innerHTML = cssStr
        document.head.append(style)
        div.innerHTML = divStr
        document.body.append(div)

        document.getElementById('lfb-panel-fun-video').onclick = getVideoUrl
        document.getElementById('lfb-panel-fun-report').onclick = ()=> {
            document.getElementById('lfb-panel-main').innerHTML = '开发中。。。'
        }
        document.getElementById('lfb-panel-fun-1').onclick = ()=> {
            document.getElementById('lfb-panel-main').innerHTML = '开发中。。。'
        }
        document.getElementById('lfb-panel-fun-reviews').onclick = ()=> {
            const str = `
                <div>
                    <label style="margin-right:10px;">SITE</label>
                    <input type="radio" id="us" name="site" value=".com" checked />
                    <label for="us">US</label>
                    <input type="radio" id="ca" name="site" value=".ca"/>
                    <label for="ca">CA</label>
                </div>
                <div>
                    <label for="asin" style="margin-right:10px;">ASIN</label>
                    <input type="text" id="asin" placeholder="多个以英文逗号或空格隔开">
                </div>
                <input id="lfb-reviews-submit" type="button" value="获取并下载" style="width:100%;margin-top:12px;">
                <span style="color:red;font-size:12px;">不一定准确,切勿过于依赖</span>
            `
            document.getElementById('lfb-panel-main').innerHTML = str

            getReviews()

        }
    }


    function clickOutSideHidden() {
        function fn(event) {
            var isClickInsideElement = el.contains(event.target);

            if (!isClickInsideElement) {
                // 点击在元素外部,执行你想要的操作
                el.style.display = 'none'
                document.removeEventListener('click', fn);
            }
        }

        // 获取需要监听的元素
        var el = document.getElementById(id);

        // 监听整个文档的点击事件
        document.addEventListener('click', fn);
    }

    /** 创建一个iframe窗口 */
    function createIframe(asin, site, star = 0) {
        const starCount = ['all', 'one', 'two', 'three', 'four', 'five'][star]
        const src = `https://www.amazon${site}/product-reviews/${asin}/ref=cm_cr_arp_d_viewopt_fmt?ie=UTF8&filterByStar=${starCount}_stars&reviewerType=all_reviews&pageNumber=1&formatType=current_format#reviews-filter-bar`
        console.log(src, '123')
        const elIframe = document.createElement("iframe");
        elIframe.setAttribute('id', asin)
        elIframe.setAttribute('src', src)
        elIframe.setAttribute('style', "width: 100%;height: 50%;opacity:0;position:fixed;top:0;pointer-events:none;z-index:999;border:2px solid red;")
        // const elHtml = `<iframe id="${asin}" style="width: 100%;height: 50%;opacity:1;position:fixed;top:0;pointer-events:none;z-index:999;border:2px solid red;" src="${url}"></iframe>`
        document.body.appendChild(elIframe)
    }

    /** 移除iframe窗口 */
    function removeIframe(asin) {
        const iframe = document.getElementById(asin)
        document.body.removeChild(iframe)
    }

    /** 获取评论条数 */
    function reqIframe(asin, result, percent, site) {
        return () => {
            return new Promise(async (resolve, reject) => {
                await wait(createIframe.bind(this, asin, site))
                document.getElementById(asin).addEventListener('load', async function(ev){
                    const asinArr = [asin]
                    let win = ev.target.contentWindow;
                    const asinDoc = win.document
                    const el = document.querySelector('#lfb-reviews-submit')

                    if(!asinDoc.getElementById('a-autoid-5-announce')) {
                        result.push(asinArr)
                        resolve(result);
                        removeIframe(asin)
                        percent.current += 6
                        el.value = '获取评论星级条数 ' + Math.round(percent.current*100 / percent.total) + '%'
                        return
                    }

                    setTimeout(async () => {
                        for(let j = 5; j >= 0; j--) {
                            await searchStar(j, asinDoc)
                            const count = getStarCount(asinDoc)
                            asinArr.push(+count)
                            percent.current++
                            el.value = '获取评论星级条数 ' + Math.round(percent.current*100 / percent.total) + '%'
                        }

                        // asinArr[0]是asin code
                        const first = asinArr[1]
                        const last = asinArr[asinArr.length -1]
                        const center = asinArr.slice(2, -1).reduce((p, c) => +p + +c)

                        if(first + center > last) {
                            asinArr.splice(1, 1 , last - center)
                        }
                        result.push(asinArr)
                        resolve(result);
                        setTimeout(() => {
                            removeIframe(asin)
                        }, 1000)
                    }, 500)
                })

            });
        };
    }

    /** 延迟执行函数 */
    function wait(fn, delay = 800) {
        return new Promise((resolve, reject) => {
            try {
                fn()
                setTimeout(() => {
                    resolve()
                }, delay)
            } catch(e) {
                console.log(e)
                reject(e)
            }
        })
    }

    /** 所有星、1-5星筛选 */
    function searchStar(star, asinDoc) {
        return wait(() => {
            asinDoc.getElementById('a-autoid-5-announce')?.click()
            // 0: 所有星 5:1 4:2 3:3 2:4 1:5
            // 1052 624 169 109 65 85
            const el = asinDoc.getElementById('star-count-dropdown_' + star)
            el?.click()
        }, 1000)
    }

    /** 获取评论数量 */
    function getStarCount(asinDoc) {
        const starEl = asinDoc.querySelector('#filter-info-section > div[data-hook=cr-filter-info-review-rating-count] ')
        const reviews = starEl?.textContent.match(/[\d,]+/)[0].replace(',', '')
        return reviews ?? 0
    }

    /** 下载文件 */
    function downloadFile(blobData, fileName = Date.now() + '.xlsx') {
        // 创建一个Blob对象
        const blob = new Blob([blobData], { type: 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet' });
        // 创建一个下载链接
        const url = URL.createObjectURL(blob);

        // 创建一个链接元素,并模拟点击来触发下载
        const a = document.createElement('a');
        a.href = url;
        a.download = fileName; // 指定下载文件的名称
        document.body.appendChild(a);
        a.click();

        // 释放URL对象,以避免内存泄漏
        URL.revokeObjectURL(url);
    }

    /** 创建Excel */
    function createExcel (data = [], title = ['asin', 'listing', 'title', 'dimensions', 'price', 'image', 'weight']) {
        // 模拟一些数据,这里使用二维数组表示表格数据
        data.unshift(title)

        // 创建一个工作簿对象
        const workbook = XLSX.utils.book_new();

        // 创建一个工作表对象
        const worksheet = XLSX.utils.aoa_to_sheet(data);

        // 将工作表添加到工作簿中
        XLSX.utils.book_append_sheet(workbook, worksheet, 'Sheet1');

        // 将工作簿转换为Excel文件的二进制数据
        const excelData = XLSX.write(workbook, { type: 'array' });
        return excelData
    }

    /** 最大并行任务数量处理 */
    class TaskQueue {
        constructor(completeFn) {
            this.completeFn = completeFn
            this.max = 5;
            this.taskList = [];
            this.unCompleteCount = 0
            setTimeout(() => {
                this.run();
            });
        }
        addTask(task) {
            this.taskList.push(task);
            this.unCompleteCount++
        }
        run() {
            let len = this.taskList.length;

            if (!len) return false;

            let min = Math.min(this.max, len);
            for (let i = 0; i < min; i++) {
                // 开始占用一个任务的空间
                this.max--;
                let task = this.taskList.shift();
                task().then((res => {
                    console.log('result:', res);
                })).catch(error => {
                    throw new Error(error);
                }).finally(() => {
                    // 释放一个任务空间
                    this.max++;
                    this.unCompleteCount--
                    let len = this.taskList.length;
                    if(len) this.run();
                    else if(len ==0 && this.unCompleteCount === 0){
                        this.completeFn()
                    }
                });
            }
        }
    }
})();