SCBOY论坛自建表情包

允许你创建自己的表情包。从论坛里面收集图片,拖到饥饿的兔子头像上,即可保存。

当前为 2023-09-27 提交的版本,查看 最新版本

// ==UserScript==
// @name         SCBOY论坛自建表情包
// @namespace    *://www.scboy.cc/
// @version      1.1.0
// @description  允许你创建自己的表情包。从论坛里面收集图片,拖到饥饿的兔子头像上,即可保存。
// @author       RustyHare
// @match        *://www.scboy.cc/*
// @exclude      https://www.scboy.cc/plugin/xn_ueditor/ueditor/dialogs/emotion/emotion.html?v1.2
// @icon         
// @grant        GM_getValue
// @grant        GM_setValue
// @grant        GM_registerMenuCommand
// @require      https://cdn.bootcdn.net/ajax/libs/Sortable/1.15.0/Sortable.min.js
// @license      MIT
// ==/UserScript==


// feature: 分类;
// feature: 导入与导出;
// feature: 拖拽整理(我得加一个暂存区,像微信那样子,还得有分类能力)

// feature: 统一图片大小;更大的预览(做完了)
// bug: 回车会变成换行,是input和textarea的问题(修好了)
// bug: iframe内也会出现。不能这样子。(修好了)
// lookup: IOS Stay app 能装用户脚本但是格式不对(这我咋办,我又没有苹果)(整好了,是换行的问题)(传了个新版本)

// 我的背好疼。  

// 给读这个代码的人:注意我的变量作用域!别用错作用域了!  
unsafeWindow.imageSetPopper = {};
unsafeWindow.imageSetFloatingIcon = {};
unsafeWindow.imageSetTooltip = {};
unsafeWindow.rhImagePreview = {};
// 怎么存呢?
// 首先开一个表来表示所有的分类,自增主键(bushi)
var imageSet = GM_getValue("imageSet");
if (typeof (imageSet) == "undefined") {
    imageSet = {
        primaryKey: 0,
        tags:{
            "0":{
                name:"默认",
                data:[]
            }
        },
    };
    GM_setValue("imageSet", imageSet);
}else if(typeof(imageSet[0]) != "undefined"){
// 从1.0.x的数据迁移到1.1.x,打个备份先
    GM_setValue("backupImageSet", imageSet);
    let newImageSet = {
        primaryKey: 1,
        tags:{
            "0":{
                name:"默认",
                data:imageSet
            },
            "1":{
                name:"土豆",
                data:[]
            }
        }
    };
    GM_setValue("imageSet", newImageSet);
    imageSet = newImageSet;
}



function attachFloatingIcon() {

    let floatingIcon = document.createElement("DIV");
    floatingIcon.id = "rhPluginEntry";
    floatingIcon.style.position = "fixed";
    floatingIcon.style.top = "8%";
    floatingIcon.style.right = "20px";
    floatingIcon.style.width = "48px";
    floatingIcon.style.height = "48px";
    floatingIcon.innerHTML = '<img class="avatar-4" style="width:48px;height:48px;" src="https://img.scboy.cc/upload/avatar/000/35548.png?1677763023">';
    floatingIcon.ondragover = function (event) {
        event.preventDefault();
    }
    floatingIcon.ondrop = function (event) {
        let targetTagid = document.getElementById("imageSetTooltipTags").dataset.active;
        event.preventDefault();
        let data = event.dataTransfer.getData("text/uri-list").toLowerCase();
        try {
            let u = new URL(data);
        } catch (e) {
            // -1: not a good url
            return "-1: 这不是一个好的URL";
        }
        if (data.indexOf("jpg") == -1 && data.indexOf("png") == -1 && data.indexOf("gif") == -1 && data.indexOf("webp") == -1 && data.indexOf("jpeg") == -1 && data.indexOf("bmp") == -1) {
            // -2: not an image (perhaps)
            return "-2: 这也许不是一个图片的URL。请尝试把它上传到论坛,存为有常见扩展名的图片再添加";
        }
        // -3: already exists
        if (imageSet.tags[targetTagid].data.indexOf(data) != -1) {
            return "-3: 已有这张图片";
        }
        imageSet.tags[targetTagid].data.push(data);
        GM_setValue("imageSet", imageSet);
        imageSetTooltip.showImagesByTag();
        return "0: 添加成功";
    }
    floatingIcon.onclick = function (event) {
        console.log(imageSet);
        if (!unsafeWindow.imageSetPopper.element) {
            unsafeWindow.imageSetPopper.element = new Popper(unsafeWindow.imageSetFloatingIcon.element, unsafeWindow.imageSetTooltip.element, { placement: 'left-start' })
        }
        unsafeWindow.imageSetTooltip.element.style.display = (unsafeWindow.imageSetTooltip.element.style.display == "block" ? "none" : "block");
        unsafeWindow.imageSetTooltip.showTooltip();
    }
    document.body.appendChild(floatingIcon);
    unsafeWindow.imageSetFloatingIcon.element = floatingIcon;
};

function attachTooltip() {

    let tooltip = document.createElement("DIV");
    tooltip.style.display = "none";
    tooltip.style.width = "440px";
    tooltip.style.height = "600px";
    tooltip.style.backgroundColor = "#F5F5F5";
    tooltip.style.zIndex = 65535;
    tooltip.style.borderRadius = "5px";
    tooltip.ondragover = function (event) {
        event.preventDefault();
    }
    tooltip.ondrop = function (event) {
        event.preventDefault();
        let targetTagid;
        if(event.target.dataset.tagid != undefined){
            targetTagid = event.target.dataset.tagid;
        }else{
            targetTagid = document.getElementById("imageSetTooltipTags").dataset.active;
        }
        console.log(targetTagid);
        let data = event.dataTransfer.getData("text/uri-list").toLowerCase();
        try {
            let u = new URL(data);
        } catch (e) {
            return -1;
        }
        if (data.indexOf("jpg") == -1 && data.indexOf("png") == -1 && data.indexOf("gif") == -1 && data.indexOf("webp") == -1 && data.indexOf("jpeg") == -1 && data.indexOf("bmp") == -1) {
            return -2;
        }
        if (imageSet.tags[targetTagid].data.indexOf(data) != -1) {
            return -3;
        }
        imageSet.tags[targetTagid].data.push(data);
        GM_setValue("imageSet", imageSet);
        imageSetTooltip.showImagesByTag();
        return 0;
    }
    document.body.appendChild(tooltip);
    unsafeWindow.imageSetTooltip.element = tooltip;
    // 改写:现在将tooltip的框架改为单独的函数,直接往里塞。
    unsafeWindow.imageSetTooltip.showTooltip = function(event){
        unsafeWindow.imageSetTooltip.element.innerHTML = `
                <div class="container" style="height:100%;">
                    <div class="row" style="height:40px;background-color: #CFCFCF;">
                        <div class="col-lg-12" style="height:40px;">
                            <div style="font-size:6px;">
                                这是一个带有拖拽和标签的预览版awa原来的帖子里有使用方法awa而且还没做完但是很开心就给推了release
                            </div>
                        </div>
                    </div>
                    <div class="row" style="height:560px;">
                        <div class="col-lg-2" style="overflow:hidden;height:560px;background-color: #CFCFCF; padding-left: 5px;" >
                            <table style="width: 72px; overflow-x: hidden;overflow-y: scroll; padding-top: 10px; padding-left: 0px; border-collapse: separate; border-spacing: 0px 10px; table-layout: fixed;">
                                <thead>
                                    <tr>
                                        <th></th>
                                    </tr>
                                </thead>
                                <tbody id="imageSetTooltipTags" data-active="`+ ((!!document.getElementById("imageSetTooltipTags")) ? document.getElementById("imageSetTooltipTags").dataset.active : "0") +`">
                                </tbody>
                            </table>
                        </div>
                        <div class="col-lg-10" style="overflow:hidden;height:560px;">
                            <div style="display:flex;flex-wrap:wrap;width:370px;height:540px;overflow-x:hidden;overflow-y:scroll;align-content:flex-start;margin-top: 10px;" id="imageSetTooltipScrollArea">

                            </div>
                        </div>
                    </div>
                </div>
            `;
        // 塞完了框架,往里塞标签
        unsafeWindow.imageSetTooltip.showTags();
        unsafeWindow.imageSetTooltip.showImagesByTag();
    }
    // 需要添加:渲染所有标签的函数,这玩意儿不管图片,只管tag
    unsafeWindow.imageSetTooltip.showTags = function(_event = {}){
        let imageSetTooltipTags = document.getElementById("imageSetTooltipTags");
        imageSetTooltipTags.innerHTML = "";
        for (t in imageSet.tags){
            let tag = imageSet.tags[t];
            console.log(tag)
            let tagRow = document.createElement("TR")
            tagRow.style.height = "16px";
            tagRow.style.backgroundColor = "#FFEB3B";
            tagRow.style.cursor = "pointer";
            tagRow.dataset.tagid = t;
            tagRow.onclick = unsafeWindow.imageSetTooltip.switchTag;
            tagRow.innerHTML = `<th style="overflow: hidden; text-overflow: ellipsis; white-space: nowrap; box-shadow: -2px 2px 2px 1px #9C9C9C;" data-tagid="` + t + `">` + tag.name + `</th>`
            imageSetTooltipTags.appendChild(tagRow);
        }
        let tagRow = document.createElement("TR")
        tagRow.style.height = "16px";
        tagRow.style.backgroundColor = "#FFEB3B";
        tagRow.style.cursor = "pointer";
        tagRow.onclick = unsafeWindow.imageSetTooltip.addTag;
        tagRow.innerHTML = `<th style="overflow: hidden; text-overflow: ellipsis; white-space: nowrap; box-shadow: -2px 2px 2px 1px #9C9C9C;">` + "  +  " + `</th>`
        imageSetTooltipTags.appendChild(tagRow);
    }

    unsafeWindow.imageSetTooltip.addTag = function(){
        imageSet["primaryKey"] += 1;
        let newTagObject = {
            name:imageSet["primaryKey"].toString(),
            data:[]
        }
        imageSet.tags[imageSet["primaryKey"].toString()] = newTagObject;
        unsafeWindow.imageSetTooltip.showTags();
        GM_setValue("imageSet", imageSet);
    }

    // 点击标签的时候,切换框体中的图片
    unsafeWindow.imageSetTooltip.switchTag = function(event){
        console.log(event);
        document.getElementById("imageSetTooltipTags").dataset.active = event.target.dataset.tagid;
        unsafeWindow.imageSetTooltip.showImagesByTag({});
    }

    // 需要添加:标签点击的时候,更改imageSetTooltip.element.dataset.activeTag的值,并根据此值来改变  
    // 别学我这写法啊  
    unsafeWindow.imageSetTooltip.showImagesByTag = function (event) {
        let activeTag = document.getElementById("imageSetTooltipTags").dataset.active;
        let data = imageSet.tags[activeTag].data;
        let scrollArea = document.getElementById("imageSetTooltipScrollArea");
        scrollArea.innerHTML = "";
        for (let i = 0; i < data.length; i++) {
            let imgFrm = document.createElement("DIV")
            imgFrm.style.display = "inline";
            imgFrm.style.position = "relative";
            imgFrm.style.lineHeight = "85px";
            imgFrm.style.width = "85px";
            imgFrm.style.height = "85px";
            imgFrm.style.border = "1px solid #DCDCDC";
            imgFrm.style.textAlign = "center";
            imgFrm.onmouseenter = rhEnterPreview;
            imgFrm.onmouseleave = rhLeavePreview;
            imgFrm.onclick = addAWAImage;
            imgFrm.dataset.imgSrc = data[i];
            imgFrm.dataset.tagid = activeTag;
            imgFrm.dataset.imgid = i.toString();
            imgFrm.innerHTML = `
                    <h6 style="display:inline">
                        <span class="label label-info" data-url="` + data[i] + `"onclick="removeImageByURL(event)" style="position: absolute;top: 5px;right: 0px;line-height: 0;cursor: pointer;">x</span>
                    </h6>
                    <img src=` + data[i] + ` data-img-src="` + data[i] + `" style="padding:8px;vertical-align:middle;max-width:100%;max-height:100%;" data-imgid="` + i.toString() + `" data-tagid="` + activeTag + `">
                `
            scrollArea.appendChild(imgFrm)
        }

        unsafeWindow.imageSetTooltip.sortable = new Sortable(imageSetTooltipScrollArea, {
            animation: 100,
            onEnd(_event){
                let activeTag = document.getElementById("imageSetTooltipTags").dataset.active;
                let scrollArea = document.getElementById("imageSetTooltipScrollArea");
                let sortedImgs = [];
                for(let i=0;i<scrollArea.children.length;i++){
                    sortedImgs.push(scrollArea.children[i].dataset.imgSrc);
                    scrollArea.children[i].dataset.imgid = i;
                    scrollArea.children[i].children[1].dataset.imgid = i;
                }
                imageSet.tags[activeTag].data = sortedImgs;
                GM_setValue("imageSet", imageSet);
            }
        })
    }



};

// 0920:添加预览功能
function attachImagePreview() {
    unsafeWindow.rhImagePreview.element = document.createElement("IMG");
    unsafeWindow.rhImagePreview.element.style.display = "none";
    unsafeWindow.rhImagePreview.element.id = "rhImagePreview";
    unsafeWindow.rhImagePreview.element.style.zIndex = 65537;
    unsafeWindow.rhImagePreview.element.style.width = "30%";
    unsafeWindow.rhImagePreview.element.style.maxHeight = "80%";
    document.body.appendChild(unsafeWindow.rhImagePreview.element)
};


// 这个方法会修改原有的快速回帖槽位
function injectReplyForm() {
    let form = document.getElementById("quick_reply_form")
    if (!!form) {
        // 我假定[0]是doctype的那个Input
        form.children[0].value = 0;
        // 把原来那个改掉
        let originalTextarea = document.getElementById("message");
        originalTextarea.name = "fakeMessage";

        replacedTextarea = document.createElement("input");
        replacedTextarea.type = "hidden";
        replacedTextarea.name = "message";
        replacedTextarea.id = "replacedTextarea";
        form.appendChild(replacedTextarea);
        // 狸猫换太子!
        originalTextarea.oninput = function (event) {
            document.getElementById("replacedTextarea").value = parseEmojiMark(document.getElementById("message").value);
        }
        form.onmousemove = function (event) {
            document.getElementById("replacedTextarea").value = parseEmojiMark(document.getElementById("message").value);
        }
    }
};

// remove in release  
unsafeWindow.clearImg = function () {
    GM_setValue("imageSet", [])
};

// click handler  
unsafeWindow.removeImageByURL = function (e) {
    e.stopPropagation();
    let activeTag = document.getElementById("imageSetTooltipTags").dataset.active;
    imageSet.tags[activeTag].data.splice(imageSet.tags[activeTag].data.indexOf(e.target.dataset.url), 1);
    GM_setValue("imageSet", imageSet);
    imageSetTooltip.showImagesByTag();
};

// quick reply handler  
unsafeWindow.parseEmojiMark = function (message) {
    message = message.replaceAll("\n", "<br>");
    // 0924: 我终于决定使用regexp了。  
    let re = /\[awa_\d+_\d+\]/;
    let res;
    let mark;
    let pattern;
    while((res = re.exec(message)) !== null){
        mark = res[0].split("_");
        pattern = `<img src=` + imageSet.tags[mark[1]].data[parseInt(mark[2])] + ` style="max-height:240px; min-height: 50px;" />`
        message = message.replace(res[0], pattern);
    }
    return message;
};

// dynamically add image to the input in one single function
unsafeWindow.addAWAImage = function (e) {
    if (typeof (UE) != "undefined") {
        let awa = UE.getEditor('msg_container');
        let ins = `<img src="` + e.target.dataset.imgSrc + `" style="max-height:240px; min-height: 50px;" />`;
        awa.execCommand('inserthtml', ins);
    } else {
        console.log(e.target.dataset)
        $('#message').insertAtCaret('[awa_' + e.target.dataset.tagid + '_' + e.target.dataset.imgid + ']');
        $('#message').trigger("input");
    }
};

unsafeWindow.rhEnterPreview = function (e) {
    unsafeWindow.rhImagePreview.popper = new Popper(unsafeWindow.imageSetTooltip.element, unsafeWindow.rhImagePreview.element, { placement: 'left-start' });
    unsafeWindow.rhImagePreview.popper.reference = unsafeWindow.imageSetTooltip.element;
    unsafeWindow.rhImagePreview.element.style.display = "block";
    unsafeWindow.rhImagePreview.element.src = e.target.dataset.imgSrc;
};

unsafeWindow.rhLeavePreview = function (e) {
    unsafeWindow.rhImagePreview.element.style.display = "none";
};

function main() {
    attachFloatingIcon();
    attachTooltip();
    attachImagePreview();
    injectReplyForm();
}


if (window.addEventListener != null) {
    window.addEventListener("load", main, false);
} else if (window.attachEvent != null) {
    window.attachEvent("onload", main);
}

QingJ © 2025

镜像随时可能失效,请加Q群300939539或关注我们的公众号极客氢云获取最新地址