上一主题 下一主题
ScriptCat,新一代的脚本管理器脚本站,与全世界分享你的用户脚本油猴脚本开发指南教程目录
返回列表 发新帖

[油猴脚本开发指南]实战秒杀快手视频提取

[复制链接]
  • TA的每日心情
    擦汗
    4 天前
  • 签到天数: 194 天

    [LV.7]常住居民III

    720

    主题

    6071

    回帖

    6850

    积分

    管理员

    非物质文化遗产社会摇传承人

    积分
    6850

    荣誉开发者喜迎中秋油中2周年生态建设者

    发表于 2021-12-13 10:39:19 | 显示全部楼层 | 阅读模式

    前言

    这节课我们实战快手视频提取脚本

    地址:https://www.kuaishou.com/profile/3x72uw4df7wdq8q

    如果出现undefined多刷新基本,快手网页端做的不怎么好

    强烈要求使用chrome!

    另再次感谢cxxjackie

    开始

    我们先以一个视频为基础

    图片.png

    这个时候打印vue属性,发现不存在

    图片.png

    我们慢慢往上级找

    再多层parent找到了属性,这时候打印$el查看对应dom元素,发现class是video-card那个class类

    图片.png

    我们看看数据怎么样

    图片.png

    发现存在数据,看一下有没有视频地址,发现在photo的photoUrl是无水印视频地址

    这个时候我们还是之前的思路

    收集dom元素列表,然后开始mutationobserve监测元素改变状态,直接对抄之前的抖音脚本

    function ControlShowCheckAndNew(target){
        console.log('target',target)
    }
    let list=document.querySelectorAll('.video-card')
    const targetNode = document.querySelector('.user-photo-list')
    // 观察器的配置(需要观察什么变动)
    const config = {
    childList: true, // 观察目标子节点的变化,添加或删除
    attributes: true, // 观察属性变动
    subtree: true, //默认是false,设置为true后可观察后代节点
    };
    
    // 当观察到变动时执行的回调函数
    const callback = function(mutationsList, observer) {
    // Use traditional 'for loops' for IE 11
    console.log('mutationsList',mutationsList)
    for(let mutation of mutationsList) {
    if (mutation.type === 'childList') {
    mutation.addedNodes.forEach((item)=>{
    ControlShowCheckAndNew(item)
    })
    }
    }
    };
    // 创建一个观察器实例并传入回调函数
    const observer = new MutationObserver(callback);
    // 以上述配置开始观察目标节点
    observer.observe(targetNode, config);
    unsafeWindow.onload=()=>{
    //处理循环
    list.forEach((item)=>{
    ControlShowCheckAndNew(item)
    })
    }

    我们测试一下,发现报错了

    图片.png

    根据debugger调试,可以确定是没找到元素的事情,存在一个延时加载的问题,上去直接一个setinterval,检测出来元素之后再执行我们的逻辑

    
    function ControlShowCheckAndNew(target){
        console.log('target',target)
    }
    function wraprun(){
        let list=document.querySelectorAll('.video-card')
        const targetNode = document.querySelector('.user-photo-list')
        // 观察器的配置(需要观察什么变动)
        const config = {
            childList: true, // 观察目标子节点的变化,添加或删除
            attributes: true, // 观察属性变动
            subtree: true, //默认是false,设置为true后可观察后代节点
        };
    
        // 当观察到变动时执行的回调函数
        const callback = function(mutationsList, observer) {
            // Use traditional 'for loops' for IE 11
            console.log('mutationsList',mutationsList)
            for(let mutation of mutationsList) {
                if (mutation.type === 'childList') {
                    mutation.addedNodes.forEach((item)=>{
                        ControlShowCheckAndNew(item)
                    })
                }
            }
        };
        // 创建一个观察器实例并传入回调函数
        const observer = new MutationObserver(callback);
        // 以上述配置开始观察目标节点
        observer.observe(targetNode, config);
        unsafeWindow.onload=()=>{
            //处理循环
            list.forEach((item)=>{
                ControlShowCheckAndNew(item)
            })
        }
    }
    
    let setint=setInterval(()=>{
        let list=document.querySelectorAll('.video-card')
        if(list!=null&&list.length!==0){
            wraprun()
            clearInterval(setint)
        }
    
    },1000)

    这时候根据测试,onload回调没有执行,因为页面的图片没有加载完全,我们下拉滚动条才会开始加载视频图片,所以图片没加载完不属于onload状态,这里也没有什么特殊要求,我就直接去掉onload就好了。

    然后我们开始实现ControlShowCheckAndNew函数,直接对抄一下之前的抖音的

    let saveurl={}
    function ControlShowCheckAndNew(target){
        if(target.className.indexOf('injectvideo')!=-1){
            return true;
        }
        if(target.className.indexOf('control-pos')!=-1){
            return true;
        }
        const prop = '__vue__'
        if(prop===undefined){
            return;
        }
        let info=target[prop].data
        if(info===undefined){
            console.log('test')
        }
        let videourl=info.photo.photoUrl
        videourl='https://'+videourl.replace('https://','').replace('http://','').replace('//','')
        target.classList.add('injectvideo')
        var select=document.createElement('label')
        select.className='container control-pos'
        select.innerHTML=` <input  type="checkbox"><div class="checkmark"></div>`
        target.append(select)
        select.onclick=()=>{
            console.log('选中变化了',select.children[0].checked)
            if(select.children[0].checked){
                //选中
                saveurl[videourl]=true;
            }else{
                //未选中
                if(saveurl[videourl]){
                    delete saveurl[videourl]
                }
            }
        }
    }

    这里几乎就是对抄之前的,简单修改了一下逻辑

    之后我们之前用了样式,所以这里也要使用,简单修改一下css样式,贴进来

    let cssstyle = document.createElement("style");
    cssstyle.innerHTML =(`
    篇幅太长省略
    `);
    document.body.appendChild(cssstyle);

    测试一下发现可以看到这个了

    图片.png

    接下来开始插入按钮

    因为页面是一个延迟加载的,我们想插入到已关注的下方,所以也要判断,这里我直接加在setinterval里了

    两个条件都满足才执行

    let setint=setInterval(()=>{
        let list=document.querySelectorAll('.video-card')
        if(list!=null&&list.length!==0){
            if(document.querySelector('.user-detail')!==null){
                wraprun()
                wrapdrawbutton()
                clearInterval(setint)
            }
        }
    
    },1000)

    如果找到了wraprun就开始监听页面插入,以及绘制目前的视频的单选框

    并且wrapdrawbutton函数开始绘制按钮

    接下来我们开始写wrapdrawbutton函数

    function wrapdrawbutton(){
        let parenttagert=document.querySelector('.user-detail')
        let div=document.createElement("div");
        div.innerHTML=`<div class="follow-button user-info-follow" data-v-68cd9209="" style="margin-top: 10px;" data-v-58fa4879="">批量下载</div>`
        div.onclick=function(event){
            let size=Object.keys(saveurl)
            let text=size.join('\n')
            GM_setClipboard(text)
            alert('已设置到剪辑版共'+size.length+"个")
        };
        parenttagert.append(div);
    }

    直接复制一下抖音的,然后修改一下div内容。

    跑一下,成功!

    那我们这节课就已经讲解完毕了!图片.png

    结语

    撒花~

    混的人。
    ------------------------------------------
    進撃!永遠の帝国の破壊虎---李恒道

    入驻了爱发电https://afdian.net/a/lihengdao666
    个人宣言:この世界で私に胜てる人とコードはまだ生まれていません。死ぬのが怖くなければ来てください。
  • TA的每日心情

    2024-1-3 11:32
  • 签到天数: 18 天

    [LV.4]偶尔看看III

    108

    主题

    101

    回帖

    431

    积分

    高级工程师

    积分
    431

    油中2周年油中3周年

    发表于 2021-12-13 11:56:12 来自手机  | 显示全部楼层
    ggnb哈哈哈
    回复

    使用道具 举报

  • TA的每日心情
    擦汗
    4 天前
  • 签到天数: 194 天

    [LV.7]常住居民III

    720

    主题

    6071

    回帖

    6850

    积分

    管理员

    非物质文化遗产社会摇传承人

    积分
    6850

    荣誉开发者喜迎中秋油中2周年生态建设者

    发表于 2021-12-13 15:58:07 | 显示全部楼层

    我猜你是用手机回的
    混的人。
    ------------------------------------------
    進撃!永遠の帝国の破壊虎---李恒道

    入驻了爱发电https://afdian.net/a/lihengdao666
    个人宣言:この世界で私に胜てる人とコードはまだ生まれていません。死ぬのが怖くなければ来てください。
    回复

    使用道具 举报

  • TA的每日心情

    2024-1-3 11:32
  • 签到天数: 18 天

    [LV.4]偶尔看看III

    108

    主题

    101

    回帖

    431

    积分

    高级工程师

    积分
    431

    油中2周年油中3周年

    发表于 2021-12-13 16:00:39 来自手机  | 显示全部楼层
    李恒道 发表于 2021-12-13 15:58
    我猜你是用手机回的

    确实,手机会更方便些
    回复

    使用道具 举报

    发表回复

    本版积分规则

    快速回复 返回顶部 返回列表