// ==UserScript==
// @name SCBOY论坛自建表情包
// @namespace *://www.scboy.cc/
// @version 1.1.6
// @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.jsdelivr.net/npm/sortablejs@1.15.0/Sortable.min.js
// @license MIT
// @downloadURL https://update.greasyfork.icu/scripts/475719/SCBOY%E8%AE%BA%E5%9D%9B%E8%87%AA%E5%BB%BA%E8%A1%A8%E6%83%85%E5%8C%85.user.js
// @updateURL https://update.greasyfork.icu/scripts/475719/SCBOY%E8%AE%BA%E5%9D%9B%E8%87%AA%E5%BB%BA%E8%A1%A8%E6%83%85%E5%8C%85.meta.js
// ==/UserScript==
// feature: 分类;
// feature: 导入与导出;
// feature: 拖拽整理(我得加一个暂存区,像微信那样子,还得有分类能力)
// feature: 统一图片大小;更大的预览(做完了)
// feature: 标签改名,管理
// bug: 回车会变成换行,是input和textarea的问题(修好了)
// bug: iframe内也会出现。不能这样子。(修好了)
// lookup: IOS Stay app 能装用户脚本但是格式不对(这我咋办,我又没有苹果)(整好了,是换行的问题)(传了个新版本)
// bug: 直接把图片拖进去,会导致let targetTagid = document.getElementById("imageSetTooltipTags").dataset.active;找不到元素。(修好了)
// 我的背好疼。
// 给读这个代码的人:注意我的变量作用域!别用错作用域了!
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;
}
for (let tag in imageSet.tags) {
for (let i = 0; i < imageSet.tags[tag].data.length; i++) {
imageSet.tags[tag].data[i] = imageSet.tags[tag].data[i].replace("img.scboy.cc", "www.scboy.cc");
}
}
GM_setValue("imageSet", imageSet);
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 = '
';
floatingIcon.ondragover = function (event) {
event.preventDefault();
}
floatingIcon.ondrop = function (event) {
let target = document.getElementById("imageSetTooltipTags");
let targetTagid = target ? target.dataset.active : 0;
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);
try {
imageSetTooltip.showImagesByTag();
} catch (e) {
console.log("me no want fix awa me lazy awa");
}
return "0: 添加成功";
}
floatingIcon.onclick = function (_event) {
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 = "500px";
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;
}
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 = `
`;
// 塞完了框架,往里塞标签
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];
let tagRow = document.createElement("TR")
tagRow.style.height = "16px";
tagRow.style.backgroundColor = "#FFEB3B";
tagRow.dataset.tagid = t;
tagRow.onclick = unsafeWindow.imageSetTooltip.switchTag;
tagRow.innerHTML = `
|
` + tag.name + ` | `
imageSetTooltipTags.appendChild(tagRow);
}
}
unsafeWindow.imageSetTooltip.editTags = function (_event) {
unsafeWindow.imageSetTooltip.sortable = new Sortable(imageSetTooltipScrollArea, {
animation: 100,
onEnd(event) {
let targetTagid = event.originalEvent.target.dataset.tagid;
let activeTag = document.getElementById("imageSetTooltipTags").dataset.active;
let scrollArea = document.getElementById("imageSetTooltipScrollArea");
let sortedImgs = [];
if (!!targetTagid) {
if (imageSet.tags[targetTagid].data.indexOf(event.item.dataset.imgSrc) == -1) {
imageSet.tags[targetTagid].data.push(event.item.dataset.imgSrc);
}
}
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);
}
});
let btn = document.getElementById("imageSetTooltipButtonEdit");
btn.innerText = "保存";
btn.onclick = unsafeWindow.imageSetTooltip.saveTags;
let imageSetTooltipTags = document.getElementById("imageSetTooltipTags");
imageSetTooltipTags.innerHTML = "";
for (t in imageSet.tags) {
let tag = imageSet.tags[t];
let tagRow = document.createElement("TR")
tagRow.style.height = "16px";
tagRow.style.backgroundColor = "#FFEB3B";
tagRow.dataset.tagid = t;
tagRow.onclick = unsafeWindow.imageSetTooltip.switchTag;
tagRow.innerHTML = `x | ` + `` + ` | `
imageSetTooltipTags.appendChild(tagRow);
}
let tagRow = document.createElement("TR")
tagRow.style.height = "16px";
tagRow.style.backgroundColor = "#FFEB3B";
tagRow.onclick = unsafeWindow.imageSetTooltip.addTag;
tagRow.innerHTML = ` | + | `
imageSetTooltipTags.appendChild(tagRow);
}
unsafeWindow.imageSetTooltip.changeTagName = function (event) {
let targetTagid = event.target.dataset.tagid;
let value = !event.target.value ? "标签" : event.target.value;
imageSet.tags[targetTagid].name = value;
GM_setValue("imageSet", imageSet);
}
unsafeWindow.imageSetTooltip.removeTag = function (event) {
if (parseInt(event.target.dataset.tagid)) {
delete imageSet.tags[event.target.dataset.tagid];
unsafeWindow.imageSetTooltip.editTags();
GM_setValue("imageSet", imageSet);
}
}
unsafeWindow.imageSetTooltip.saveTags = function (_event) {
unsafeWindow.imageSetTooltip.sortable = null;
document.getElementById("imageSetTooltipTags").dataset.active = "0";
unsafeWindow.imageSetTooltip.showImagesByTag();
let btn = document.getElementById("imageSetTooltipButtonEdit");
btn.innerText = "编辑";
btn.onclick = unsafeWindow.imageSetTooltip.editTags;
unsafeWindow.imageSetTooltip.showTags();
}
// 增加标签
unsafeWindow.imageSetTooltip.addTag = function () {
imageSet["primaryKey"] += 1;
let newTagObject = {
name: imageSet["primaryKey"].toString(),
data: []
}
imageSet.tags[imageSet["primaryKey"].toString()] = newTagObject;
unsafeWindow.imageSetTooltip.editTags();
GM_setValue("imageSet", imageSet);
}
// 点击标签的时候,切换框体中的图片
unsafeWindow.imageSetTooltip.switchTag = function (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 = "95px";
imgFrm.style.width = "95px";
imgFrm.style.height = "95px";
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 = `
x
`
scrollArea.appendChild(imgFrm)
}
unsafeWindow.imageSetTooltip.saveAsFile = function (_event = {}) {
let json = JSON.stringify(imageSet);
let blob = new Blob([json], { type: 'application/json' });
let url = URL.createObjectURL(blob);
let link = document.createElement('A');
link.href = url;
link.download = "ImageSetSave.awa";
link.click();
}
unsafeWindow.imageSetTooltip.loadFromFile = function (_event = {}) {
let finput = document.createElement("INPUT");
finput.type = "file";
finput.accept = ".awa";
finput.onchange = function (e) {
let fileReader = new FileReader();
fileReader.onload = function (event) {
try {
let json = JSON.parse(event.target.result);
console.log(json);
if (!!json.primaryKey && !!json.tags) {
imageSet = json;
GM_setValue("imageSet", imageSet);
}
unsafeWindow.imageSetTooltip.showTags();
unsafeWindow.imageSetTooltip.showImagesByTag();
} catch (e) {
console.log(e)
}
}
fileReader.readAsText(finput.files[0]);
}
finput.click();
}
}
};
// 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 = 65538;
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", "
");
// 0924: 我终于决定使用regexp了。
let re = /\[awa_\d+_\d+\]/;
let res;
let mark;
let pattern;
while ((res = re.exec(message)) !== null) {
mark = res[0].split("_");
pattern = `
`
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 = `
`;
awa.execCommand('inserthtml', ins);
} else {
$('#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);
}