Acfun直播观众助手

Acfun直播间自动点赞(可设置下次进入直播间就直接开始自动点赞),自动发送弹幕(弹幕抽奖好帮手);此脚本仅限个人学习交流,若侵犯到您的权益请联系删除。

// ==UserScript==
// @name         Acfun直播观众助手
// @namespace    http://tampermonkey.net/
// @version      1.0
// @description  Acfun直播间自动点赞(可设置下次进入直播间就直接开始自动点赞),自动发送弹幕(弹幕抽奖好帮手);此脚本仅限个人学习交流,若侵犯到您的权益请联系删除。
// @author       czhen
// @license      MIT
// @match        https://live.acfun.cn/live/*
// @icon         https://cdn.aixifan.com/ico/favicon.ico
// @require      https://libs.baidu.com/jquery/1.11.1/jquery.min.js
// @grant        GM_setValue
// @grant        GM_getValue
// @grant        GM_deleteValue
// @grant        GM_listValues
// @grant        GM_notification
// @grant        GM_info
// @grant        GM_registerMenuCommand
// ==/UserScript==

(function () {
    'use strict';
    //格式化字符
    const tempName = 'acLive';
    //用户uid对应的cookie名
    const userCookie = 'auth_key';
    //显示的base64图片
    const imgBase64 = '';
    //js脚本操作实体
    let JSEntity = {
        //脚本是否可用
        isEnable: true,
        //添加样式
        addStyle: function () {
            let style = document.createElement('style');
            style.type = 'text/css';
            style.textContent = `
            .${tempName}-dom {
              position: absolute;
              z-index: 100;
              top: 108px;
              left: 50%;
            }
            #${tempName}-click {
             width: 50px;
             height: 50px;
             background-image:url(${imgBase64});
             background-size:100% 100%;
             background-position: center center;
             background-repeat:no-repeat;
             background-color: #fd4c5c;
             cursor: pointer;
             border:1px solid red;
             border-radius: 10px;
            }
            #${tempName}-click:hover {
              background-size:122% 122%;
            }
            #${tempName}-menu {
             z-index: 200;
             width: 400px;
             height: 160px;
             display:none;
             border-radius: 10px;
             background-color: #fd4c5c;
             color:#fff;
            }
            #${tempName}-menu dt {
              font-size:16px;
              font-weight:700;
              margin-bottom:5px;
              color:#ffefa2;
            }
            #${tempName}-menu label { cursor: pointer; }
            #${tempName}-menu input[type="checkbox"] {
              vertical-align:middle;
              width:16px;
              height:16px;
              cursor: pointer;
              margin-left:0;
              margin-right:3px;
            }
            #${tempName}-menu input[type="number"] {
              vertical-align:middle;
              width:30%;
              height:18px;
              font-size:16px;
              padding-top:2px;
              padding-left:5px;
              border:1px solid #767676;
              border-radius: 5px;
              margin-left:0;
              margin-right:3px;
            }
            #${tempName}-menu > .menu-box {
              height:72%;
              display: flex;
              flex-wrap: wrap;
              justify-content: space-between;
              align-content: stretch;
            }
            #${tempName}-menu > .menu-box > .menu-item {
              box-sizing: border-box;
              width: 50%;
              padding:2%;
              border: 1px dashed #ffbc00;
              border-radius:10px;
            }
            #${tempName}-menu button {
              cursor: pointer;
            }
            #${tempName}-menu dd.action-btns {
              margin-top: 5px;
              text-align:center;
            }
            #${tempName}-menu > .menu-btns {
              margin-top: 12px;
              text-align:center;
            }
            input.intervalTime::-webkit-outer-spin-button {
              -webkit-appearance: none;
            }
            input.intervalTime::-webkit-inner-spin-button {
              -webkit-appearance: none;
            }
            input.intervalTime {
              -moz-appearance: textfield;
            }
            `;
            document.head.appendChild(style);
        },
        //添加dom节点
        addDom: function () {
            let addHtml = `
        <div class="${tempName}-dom" id="${tempName}-click" title="点击我打开菜单栏"></div>
        <div class="${tempName}-dom" id="${tempName}-menu">
            <div class="menu-box">
                <div class="menu-item" id="${tempName}-likeItem">
                    <dl>
                        <dt>自动点赞</dt>
                        <dd title="每隔多少秒执行一次自动点赞"><input type="number" value="10" class="intervalTime" />间隔秒数</dd>
                        <dd title="勾选表示下次进入当前直播间后就会直接开始自动点赞"><label><input type="checkbox" class="isAuto" />下次进入自动点赞</label></dd>
                        <dd class="action-btns">
                            <button type="button" data-type="likestart">开始点赞</button>
                            <button type="button" data-type="likeend">取消点赞</button>
                        </dd>
                    </dl>
                </div>
                <div class="menu-item" id="${tempName}-commentItem">
                    <dl>
                        <dt>自动弹幕</dt>
                        <dd title="每隔多少秒发送一次弹幕"><input type="number" value="10" class="intervalTime" />间隔秒数</dd>
                        <dd class="action-btns">
                            <button type="button" data-type="commentstart">开始弹幕</button>
                            <button type="button" data-type="commentend">取消弹幕</button>
                        </dd>
                    </dl>
                </div>
            </div>
            <div class="menu-btns">
                <button type="button" data-type="close">关闭弹窗</button>
            </div>
        </div>
`;
            //添加操作菜单
            $('body').append(addHtml);

            //添加点赞次数
            $('.author-info-detail').append(`<div style="color:red;font-size:18px;margin-left:15px;" title="已点赞次数">+<b id="${tempName}-likeCount">0</b></div>`);
        },
        //点赞相关功能
        likeUnits: {
            //点赞按钮
            btn: null,
            //点赞次数
            count: 0,
            //点赞计时器
            timer: null,
            //执行点赞的方法
            exeFn: function () {
                let _this = this;
                _this.btn.click();
                _this.count++;
                $(`#${tempName}-likeCount`).text(_this.count);
            },
            //自动开始点赞
            autoStart: function (num) {
                let _this = this;
                let _p = JSEntity.publicUnits;

                _this.btn = $('.author-interactive-area >.like-btn');

                //未找到点赞按钮,直接返回
                if (!_this.btn) {
                    //发送失败通知
                    _p.showPageNotify('自动点赞功能 开启失败!\nX﹏X');
                    return false;
                };

                //发送通知
                _p.showPageNotify('开启 自动点赞功能!\n(づ ̄3 ̄)づ╭❤~');

                //直接先触发一次点赞事件
                _this.exeFn();

                //初始化计时器开始连续点赞
                _this.timer = setInterval(function () {
                    _this.exeFn();
                }, num * 1000);

                //发送弹幕
                JSEntity.commentUnits.inputVal('请组织放心,我已开启自动点赞!为' + (_p.upNickName) + '疯狂打CALL!!!');
                $('.wrap-bottom-area > .send-btn').click();
            },
            //开始点赞
            start: function () {
                let _this = this;
                let _p = JSEntity.publicUnits;
                _this.btn = $('.author-interactive-area >.like-btn');
                //未找到点赞按钮,直接返回
                if (!_this.btn) {
                    //发送失败通知
                    _p.showPageNotify('自动点赞功能 开启失败!\nX﹏X');
                    return false;
                };
                //隐藏菜单
                _p.toggleMenu(0);

                //发送通知
                _p.showPageNotify('开启 自动点赞功能!\n(づ ̄3 ̄)づ╭❤~');

                //直接先触发一次点赞事件
                _this.exeFn();

                //获取计时器时间间隔
                let val = $(`#${tempName}-likeItem`).find('input.intervalTime').val();
                let num = Number(val) || 10;
                if (num < 1) num = 1;

                //先清除原有计时器
                _this.timer && clearInterval(_this.timer);
                //初始化计时器开始连续点赞
                _this.timer = setInterval(function () {
                    _this.exeFn();
                }, num * 1000);

                //是否下次进入直播间就自动开始点赞
                let isAuto = $(`#${tempName}-likeItem`).find('input.isAuto').prop('checked');

                //存储本地数据
                _p.setValue('like', { intervalTime: num, isAuto: isAuto });
            },
            //取消点赞
            end: function () {
                let _this = this;
                let _p = JSEntity.publicUnits;
                //隐藏菜单
                _p.toggleMenu(0);

                //发送通知
                _p.showPageNotify('关闭 自动点赞功能!\nヾ( ̄▽ ̄)Bye~Bye~');

                //清除计时器
                if (_this.timer) {
                    clearInterval(_this.timer);
                    _this.timer = null;
                };

                //还原本地存储为默认数据
                let def = _p.defStorageOptions.like;
                _p.setValue('like', def);
            }
        },
        //发送弹幕相关功能
        commentUnits: {
            //弹幕发送按钮
            btn: null,
            //弹幕内容文本框
            textarea: null,
            //输入的弹幕内容
            content: null,
            //自动发送弹幕次数
            count: 0,
            //发送弹幕计时器
            timer: null,
            //填入弹幕内容
            inputVal: function (val) {
                let _this = this;
                if (!_this.textarea) _this.textarea = $('.danmaku-input-wrap >textarea');
                _this.textarea.val(val);
                let event = new Event('input', { bubbles: true });
                let tracker = _this.textarea[0]._valueTracker;
                if (tracker) tracker.setValue('');
                _this.textarea[0].dispatchEvent(event);
            },
            //执行发送弹幕的方法
            exeFn: function () {
                let _this = this;
                _this.btn.click();
                _this.count++;
                _this.inputVal(_this.content);
            },
            //开始发送弹幕
            start: function () {
                let _this = this;
                let _p = JSEntity.publicUnits;
                _this.btn = $('.wrap-bottom-area > .send-btn');
                //未找到发送弹幕按钮,直接返回
                if (!_this.btn) {
                    //发送失败通知
                    _p.showPageNotify('自动发送弹幕功能 开启失败!\nX﹏X');
                    return false;
                };
                _this.textarea = $('.danmaku-input-wrap >textarea');
                _this.content = _this.textarea.val();
                //没有要发送的弹幕内容,直接返回
                if (!_this.btn.hasClass('enable') || !_this.content) {
                    //发送失败通知
                    _p.showPageNotify('没有装填弹幕,这让我很难搞啊!\n请先在右下角评论框中输入弹幕!\n ┗( T﹏T )┛');
                    _this.textarea.focus();
                    return false;
                };

                //隐藏菜单
                _p.toggleMenu(0);

                //发送通知
                _p.showPageNotify('开启 自动发送弹幕功能!\n(o゜▽゜)o☆');

                //直接先触发一次弹幕发送事件
                _this.exeFn();

                //获取计时器时间间隔
                let val = $(`#${tempName}-commentItem`).find('input.intervalTime').val();
                let num = (Number(val) || 10);
                if (num < 0.1) num = 0.1;

                //先清除原有计时器
                _this.timer && clearInterval(_this.timer);
                //初始化计时器开始连续发送弹幕
                _this.timer = setInterval(function () {
                    _this.exeFn();
                }, num * 1000);

                //存储本地数据
                _p.setValue('comment', { intervalTime: num });
            },
            //取消发送弹幕
            end: function () {
                let _this = this;
                let _p = JSEntity.publicUnits;
                //隐藏菜单
                _p.toggleMenu(0);

                //发送通知
                _p.showPageNotify('关闭 自动发送弹幕功能!\nヾ( ̄▽ ̄)Bye~Bye~');

                //清除计时器
                if (_this.timer) {
                    clearInterval(_this.timer);
                    _this.timer = null;
                };

                //还原本地存储为默认数据
                let def = _p.defStorageOptions.comment;
                _p.setValue('comment', def);
            }
        },
        //公用相关功能
        publicUnits: {
            //脚本名称
            scriptName: null,
            //主播uid
            upUid: null,
            //主播昵称
            upNickName: null,
            //主播头像
            upPhoto: null,
            //当前登录用户uid
            userUid: null,
            //当前直播间的本地存储名称
            storageName: null,
            //桌面通知默认配置
            defNotifyOptions: {
                title: 'ACFUN',
                text: '通知信息',
                image: 'https://cdn.aixifan.com/ico/favicon.ico',
                highlight: true,
                timeout: 60000
            },
            //本地存储默认配置
            defStorageOptions: {
                like: {
                    intervalTime: 10,
                    isAuto: false
                },
                comment: {
                    intervalTime: 10
                }
            },
            //显示/隐藏菜单
            toggleMenu: function (type) {
                let $menu = $(`#${tempName}-menu`);
                if (type === 1) {
                    $menu.show();
                } else {
                    $menu.hide();
                }
            },
            //显示桌面通知基础方法
            showNotify: function (options) {
                if (!options) return false;
                if (typeof options === 'string') options = { text: options };
                let config = $.extend({}, JSEntity.publicUnits.defNotifyOptions, options);
                GM_notification(config);
            },
            //显示当前页面的桌面通知(带着主播头像和昵称)
            showPageNotify: function (text) {
                let _this = this;
                _this.showNotify({
                    title: `${_this.upNickName}的直播间`,
                    text: text,
                    image: `${_this.upPhoto}`
                });
            },
            //设置本地存储的数据
            setValue: function (type, value) {
                let _this = this;
                let typeData = {};
                typeData[type] = value;

                let oldValue = _this.getValue();
                let newValue = $.extend(true, {}, oldValue, typeData);
                GM_setValue(_this.storageName, newValue);
            },
            //获取本地存储
            getValue: function () {
                let _this = this;
                return GM_getValue(_this.storageName, _this.defStorageOptions);
            },
            //删除本地存储
            deleteValue: function (name) {
                let _this = this;
                GM_deleteValue(name || _this.storageName);
            },
            //清空本地存储
            clearValue: function () {
                let _this = this;
                let list = GM_listValues();
                if (list.length > 0) {
                    list.forEach(function (item) {
                        _this.deleteValue(item);
                    });
                };
            }
        },
        //绑定事件
        bindEvents: function () {
            //显示菜单
            $(`#${tempName}-click`).click(function () {
                JSEntity.publicUnits.toggleMenu(1);
            });
            //菜单按钮点击
            $(`#${tempName}-menu button`).click(function () {
                let $btn = $(this),
                    type = $btn.data('type');
                switch (type) {
                    case 'likestart':
                        JSEntity.likeUnits.start();
                        break;
                    case 'likeend':
                        JSEntity.likeUnits.end();
                        break;
                    case 'commentstart':
                        JSEntity.commentUnits.start();
                        break;
                    case 'commentend':
                        JSEntity.commentUnits.end();
                        break;
                    default:
                        JSEntity.publicUnits.toggleMenu(0);
                        break;
                }
            });
        },
        initData: function () {
            let publicUnits = JSEntity.publicUnits;
            //脚本名称
            publicUnits.scriptName = publicUnits.defNotifyOptions.title = GM_info.script.name;

            //登录用户uid
            let userUid = document.cookie.match(`[;\s+]?${userCookie}=([^;]*)`)?.pop();
            if (!userUid) {
                publicUnits.showNotify('没登录还想调戏我! ╮(╯_╰)╭');
                JSEntity.isEnable = false;
                return false;
            };
            publicUnits.userUid = userUid;

            //主播uid
            publicUnits.upUid = window.location.href.split('/live/')[1];
            //主播昵称
            publicUnits.upNickName = $('.author-info-detail').find('a.up-name').text();
            //主播头像
            publicUnits.upPhoto = $('.live-author-avatar').find('img.live-author-avatar-img').attr('src');

            //存储名称
            publicUnits.storageName = `${tempName}_${publicUnits.upUid}`;
        },
        //初始化本地存储
        initStorage: function () {
            //获取当前直播间的本地存储数据
            let storage = JSEntity.publicUnits.getValue();
            //根据本地存储的数据设置页面元素的值
            for (let key in storage) {
                let json = storage[key];
                if (key === 'like') {
                    let $item = $(`#${tempName}-likeItem`);
                    $('input.intervalTime', $item).val(json.intervalTime);
                    if (json.isAuto) {
                        //如果勾选了下次自动点赞,直接开始执行自动点赞事件
                        $('input.isAuto', $item).prop('checked', true);
                        //在页面全部加载完成之后再开始点赞,一开始点赞按钮还没加载
                        $(window).load(function () {
                            JSEntity.likeUnits.autoStart(json.intervalTime);
                        });
                    }
                } else {
                    let $item = $(`#${tempName}-commentItem`);
                    $('input.intervalTime', $item).val(json.intervalTime);
                }
            };
        },
        //注册脚本菜单
        registerMenuCmd: function () {
            let _this = this;
            GM_registerMenuCommand("清空本地存储", function () {
                let publicUnits = _this.publicUnits;
                publicUnits.clearValue();
                publicUnits.showNotify('本地存储已清空!\n所有直播间的设置都没了~\n (。﹏。*)');
            });
            GM_registerMenuCommand("帮助中心", function () {
                alert('脚本用法及介绍请至A站文章区搜索“' + (JSEntity.publicUnits.scriptName) + '”\r\n联系作者:wx784354310(微信)');
            });
        },
        //初始化所有数据
        initAll: function () {
            this.initData();
            if (!this.isEnable) return false;
            this.addStyle();
            this.addDom();
            this.initStorage();
            this.bindEvents();
            this.registerMenuCmd();
        }
    };
    JSEntity.initAll();

})();