BetterDiscordAddons/Plugins/ImageGallery/ImageGallery.plugin.js

313 lines
12 KiB
JavaScript
Raw Normal View History

2020-02-27 08:44:03 +01:00
//META{"name":"ImageGallery","authorId":"278543574059057154","invite":"Jx3TjNS","donate":"https://www.paypal.me/MircoWittrien","patreon":"https://www.patreon.com/MircoWittrien","website":"https://github.com/mwittrien/BetterDiscordAddons/tree/master/Plugins/ImageGallery","source":"https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/ImageGallery/ImageGallery.plugin.js"}*//
2018-10-11 10:21:26 +02:00
2020-02-04 08:20:40 +01:00
var ImageGallery = (_ => {
2020-02-09 18:16:04 +01:00
var eventFired, clickedImage;
2020-02-04 08:20:40 +01:00
return class ImageGallery {
getName () {return "ImageGallery";}
2019-01-11 21:57:59 +01:00
2020-02-11 09:46:53 +01:00
getVersion () {return "1.6.6";}
2019-01-11 21:57:59 +01:00
2020-02-04 08:20:40 +01:00
getAuthor () {return "DevilBro";}
2019-01-11 21:57:59 +01:00
2020-02-04 08:20:40 +01:00
getDescription () {return "Allows the user to browse through images sent inside the same message.";}
2019-01-26 22:45:19 +01:00
2020-02-04 08:20:40 +01:00
constructor () {
this.changelog = {
2020-02-11 09:46:53 +01:00
"added":[["Details","Added some details about the current image and amount of images, can be turned off in settings now"]],
"fixed":[["Size","Images in the gallery no longer get resized to a thumbnail size"]],
2020-02-04 08:20:40 +01:00
"improved":[["New Library Structure & React","Restructured my Library and switched to React rendering instead of DOM manipulation"]]
};
2019-09-04 12:34:02 +02:00
2020-02-04 08:20:40 +01:00
this.patchedModules = {
after: {
ImageModal: ["render", "componentDidMount"]
}
};
}
2019-09-04 12:34:02 +02:00
2020-02-04 08:20:40 +01:00
initConstructor () {
2020-02-11 09:46:53 +01:00
this.defaults = {
settings: {
addDetails: {value:true, description:"Adds details (name, size, amount) to the Image Modal"}
}
};
2020-02-04 08:20:40 +01:00
this.css = `
2020-02-09 18:16:04 +01:00
${BDFDB.dotCN._imagegallerysibling} {
display: flex;
align-items: center;
2020-02-04 08:20:40 +01:00
position: fixed;
z-index: -1;
}
2020-02-09 18:16:04 +01:00
${BDFDB.dotCN._imagegalleryprevious} {
justify-content: flex-end;
2020-02-04 08:20:40 +01:00
right: 90%;
}
2020-02-09 18:16:04 +01:00
${BDFDB.dotCN._imagegallerynext} {
justify-content: flex-start;
2020-02-04 08:20:40 +01:00
left: 90%;
2020-02-09 18:16:04 +01:00
}
${BDFDB.dotCN._imagegalleryicon} {
position: absolute;
background: rgba(0, 0, 0, 0.3);
border-radius: 50%;
padding: 15px;
transition: all 0.3s ease;
}
${BDFDB.dotCNS._imagegalleryprevious + BDFDB.dotCN._imagegalleryicon} {
right: 10px;
}
${BDFDB.dotCNS._imagegallerynext + BDFDB.dotCN._imagegalleryicon} {
left: 10px;
}
${BDFDB.dotCN._imagegallerysibling}:hover ${BDFDB.dotCN._imagegalleryicon} {
background: rgba(0, 0, 0, 0.5);
}
2020-02-09 18:16:42 +01:00
${BDFDB.dotCN._imagegallerydetailswrapper} {
position: fixed;
bottom: 10px;
left: 15px;
right: 15px;
2020-02-11 10:03:11 +01:00
pointer-events: none;
2020-02-09 18:16:42 +01:00
}
${BDFDB.dotCN._imagegallerydetails} {
margin-top: 5px;
font-size: 14px;
font-weight: 500;
}
${BDFDB.dotCN._imagegallerydetailslabel} {
font-weight: 600;
}
2020-02-09 18:16:04 +01:00
`;
2020-02-04 08:20:40 +01:00
}
2019-01-26 22:45:19 +01:00
2020-02-11 09:46:53 +01:00
getSettingsPanel () {
if (!window.BDFDB || typeof BDFDB != "object" || !BDFDB.loaded || !this.started) return;
let settings = BDFDB.DataUtils.get(this, "settings");
let settingspanel, settingsitems = [];
for (let key in settings) settingsitems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
className: BDFDB.disCN.marginbottom8,
type: "Switch",
plugin: this,
keys: ["settings", key],
label: this.defaults.settings[key].description,
value: settings[key]
}));
return settingspanel = BDFDB.PluginUtils.createSettingsPanel(this, settingsitems);
}
2020-02-04 08:20:40 +01:00
//legacy
load () {}
start () {
if (!window.BDFDB) window.BDFDB = {myPlugins:{}};
if (window.BDFDB && window.BDFDB.myPlugins && typeof window.BDFDB.myPlugins == "object") window.BDFDB.myPlugins[this.getName()] = this;
let libraryScript = document.querySelector("head script#BDFDBLibraryScript");
if (!libraryScript || (performance.now() - libraryScript.getAttribute("date")) > 600000) {
if (libraryScript) libraryScript.remove();
libraryScript = document.createElement("script");
libraryScript.setAttribute("id", "BDFDBLibraryScript");
libraryScript.setAttribute("type", "text/javascript");
libraryScript.setAttribute("src", "https://mwittrien.github.io/BetterDiscordAddons/Plugins/BDFDB.min.js");
libraryScript.setAttribute("date", performance.now());
libraryScript.addEventListener("load", _ => {this.initialize();});
document.head.appendChild(libraryScript);
2019-11-24 09:18:13 +01:00
}
2020-02-04 08:20:40 +01:00
else if (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) this.initialize();
this.startTimeout = setTimeout(_ => {
try {return this.initialize();}
catch (err) {console.error(`%c[${this.getName()}]%c`, "color: #3a71c1; font-weight: 700;", "", "Fatal Error: Could not initiate plugin! " + err);}
}, 30000);
2018-10-11 10:21:26 +02:00
}
2020-02-04 08:20:40 +01:00
initialize () {
if (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) {
if (this.started) return;
BDFDB.PluginUtils.init(this);
eventFired = false;
2020-02-09 18:16:04 +01:00
clickedImage = null;
BDFDB.ListenerUtils.add(this, document.body, "click", BDFDB.dotCNS.message + BDFDB.dotCNS.imagewrapper + "img", e => {
clickedImage = e.target;
BDFDB.TimeUtils.timeout(_ => {clickedImage = null;});
});
2019-01-26 22:45:19 +01:00
2020-02-04 08:20:40 +01:00
BDFDB.ModuleUtils.forceAllUpdates(this);
}
else console.error(`%c[${this.getName()}]%c`, "color: #3a71c1; font-weight: 700;", "", "Fatal Error: Could not load BD functions!");
2018-10-11 10:21:26 +02:00
}
2020-02-04 08:20:40 +01:00
stop () {
if (window.BDFDB && typeof BDFDB === "object" && BDFDB.loaded) {
this.stopping = true;
2019-10-22 11:37:23 +02:00
2020-02-04 08:20:40 +01:00
BDFDB.ModuleUtils.forceAllUpdates(this);
2019-01-26 22:45:19 +01:00
2020-02-04 08:20:40 +01:00
BDFDB.PluginUtils.clear(this);
}
2018-10-11 10:21:26 +02:00
}
2019-01-26 22:45:19 +01:00
2020-02-04 08:20:40 +01:00
// begin of own functions
2019-01-26 22:45:19 +01:00
2020-02-04 08:20:40 +01:00
processImageModal (e) {
2020-02-09 18:16:04 +01:00
if (clickedImage) e.instance.props.cachedImage = clickedImage;
let src = e.instance.props.cachedImage && e.instance.props.cachedImage.src ? e.instance.props.cachedImage : e.instance.props.src;
let messages = this.getMessageGroupOfImage(src);
2020-02-04 08:20:40 +01:00
if (messages.length) {
if (e.returnvalue) {
let images = messages.map(n => Array.from(n.querySelectorAll(BDFDB.dotCNS.imagewrapper + "img"))).flat().filter(img => !BDFDB.DOMUtils.getParent(BDFDB.dotCN.spoilerhidden, img));
2020-02-09 18:16:04 +01:00
let next, previous, index = 0, amount = images.length;
for (let i = 0; i < amount; i++) if (this.isSameImage(src, images[i])) {
index = i;
previous = images[i-1];
next = images[i+1];
2020-02-04 08:20:40 +01:00
break;
}
if (previous) {
2020-02-09 18:16:04 +01:00
if (e.instance.previousRef) e.returnvalue.props.children.push(this.createImageWrapper(e.instance, e.instance.previousRef, "previous", BDFDB.LibraryComponents.SvgIcon.Names.LEFT_CARET));
else this.loadImage(e.instance, previous, "previous");
}
if (next) {
if (e.instance.nextRef) e.returnvalue.props.children.splice(1, 0, this.createImageWrapper(e.instance, e.instance.nextRef, "next", BDFDB.LibraryComponents.SvgIcon.Names.RIGHT_CARET));
else this.loadImage(e.instance, next, "next");
2020-02-04 08:20:40 +01:00
}
2020-02-11 09:46:53 +01:00
if (BDFDB.DataUtils.get(this, "settings", "addDetails")) e.returnvalue.props.children.push(BDFDB.ReactUtils.createElement("div", {
2020-02-09 18:16:42 +01:00
className: BDFDB.disCN._imagegallerydetailswrapper,
2020-02-09 18:16:04 +01:00
children: [
{label: "Source", text: e.instance.props.src},
{label: "Size", text: `${e.instance.props.width} x ${e.instance.props.height}px`},
{label: "Image", text: `${index+1} of ${amount}`}
].map(data => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
2020-02-09 18:16:42 +01:00
className: BDFDB.disCNS._imagegallerydetails + BDFDB.disCN.primary,
2020-02-09 18:16:04 +01:00
children: [
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
grow: 0,
shrink: 0,
basis: 100,
2020-02-09 18:16:42 +01:00
className: BDFDB.disCN._imagegallerydetailslabel,
2020-02-09 18:16:04 +01:00
children: data.label + ":"
}),
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextScroller, {
children: data.text
})
]
}))
}));
2019-01-11 21:57:59 +01:00
}
2020-02-04 08:20:40 +01:00
if (e.node) {
BDFDB.DOMUtils.addClass(BDFDB.DOMUtils.getParent(BDFDB.dotCN.modal, e.node), BDFDB.disCN._imagegallerygallery);
this.cleanUpListeners();
document.keydownImageGalleryListener = event => {
if (!document.contains(e.node)) this.cleanUpListeners();
else if (!eventFired) {
eventFired = true;
if (event.keyCode == 37) this.switchImages(e.instance, "previous");
else if (event.keyCode == 39) this.switchImages(e.instance, "next");
}
};
document.keyupImageGalleryListener = _ => {
eventFired = false;
if (!document.contains(e.node)) this.cleanUpListeners();
};
document.addEventListener("keydown", document.keydownImageGalleryListener);
document.addEventListener("keyup", document.keyupImageGalleryListener);
2018-10-11 10:21:26 +02:00
}
}
2020-02-04 08:20:40 +01:00
}
getMessageGroupOfImage (src) {
2020-02-09 18:16:04 +01:00
if (src) for (let message of document.querySelectorAll(BDFDB.dotCN.message)) for (let img of message.querySelectorAll(BDFDB.dotCNS.imagewrapper + "img")) if (this.isSameImage(src, img)) {
2020-02-04 08:20:40 +01:00
let previousSiblings = [], nextSiblings = [];
let previousSibling = message.previousSibling, nextSibling = message.nextSibling;
2020-02-06 23:04:41 +01:00
if (!BDFDB.DOMUtils.containsClass(message, BDFDB.disCN.messagegroupstart)) while (previousSibling) {
2020-02-04 08:20:40 +01:00
previousSiblings.push(previousSibling);
if (BDFDB.DOMUtils.containsClass(previousSibling, BDFDB.disCN.messagegroupstart)) previousSibling = null;
else previousSibling = previousSibling.previousSibling;
}
while (nextSibling) {
if (!BDFDB.DOMUtils.containsClass(nextSibling, BDFDB.disCN.messagegroupstart)) {
nextSiblings.push(nextSibling);
nextSibling = nextSibling.nextSibling;
2019-11-14 23:15:44 +01:00
}
2020-02-04 08:20:40 +01:00
else nextSibling = null;
}
return [].concat(previousSiblings.reverse(), message, nextSiblings).filter(n => n && BDFDB.DOMUtils.containsClass(n, BDFDB.disCN.message));
2018-10-11 10:21:26 +02:00
}
2020-02-04 08:20:40 +01:00
return [];
2018-10-11 10:21:26 +02:00
}
2020-02-09 18:16:04 +01:00
isSameImage (src, img) {
2020-02-11 09:46:53 +01:00
return img.src && (Node.prototype.isPrototypeOf(src) && img == src || !Node.prototype.isPrototypeOf(src) && this.getImageSrc(img) == this.getImageSrc(src));
2020-02-09 18:16:04 +01:00
}
2019-01-26 22:45:19 +01:00
2020-02-11 09:46:53 +01:00
getImageSrc (img) {
2020-02-04 08:20:40 +01:00
if (!img) return null;
return (typeof img == "string" ? img : (img.src || (img.querySelector("canvas") ? img.querySelector("canvas").src : ""))).split("?width=")[0];
}
2020-02-09 18:16:04 +01:00
createImageWrapper (instance, imgRef, type, svgIcon) {
return BDFDB.ReactUtils.createElement("div", {
className: BDFDB.disCNS.imagewrapper + BDFDB.disCNS.imageclickable + BDFDB.disCNS._imagegallerysibling + BDFDB.disCN[`_imagegallery${type}`],
onClick: _ => {this.switchImages(instance, type);},
children: [
imgRef,
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, {
className: BDFDB.disCNS._imagegalleryicon + BDFDB.disCN.svgicon,
name: svgIcon
})
]
});
}
loadImage (instance, img, type) {
2020-02-04 08:20:40 +01:00
let imagethrowaway = document.createElement("img");
2020-02-11 09:46:53 +01:00
let src = this.getImageSrc(img);
imagethrowaway.src = src;
2020-02-04 08:20:40 +01:00
imagethrowaway.onload = _ => {
let arects = BDFDB.DOMUtils.getRects(document.querySelector(BDFDB.dotCN.appmount));
let resizeY = (arects.height/imagethrowaway.naturalHeight) * 0.65, resizeX = (arects.width/imagethrowaway.naturalWidth) * 0.8;
let resize = resizeX < resizeY ? resizeX : resizeY;
let newHeight = imagethrowaway.naturalHeight * resize;
let newWidth = imagethrowaway.naturalWidth * resize;
2020-02-09 18:16:04 +01:00
instance[type + "Img"] = img;
2020-02-04 08:20:40 +01:00
instance[type + "Ref"] = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.LazyImage, {
2020-02-11 09:46:53 +01:00
src: src,
2020-02-04 08:20:40 +01:00
height: imagethrowaway.naturalHeight,
width: imagethrowaway.naturalWidth,
maxHeight: newHeight,
maxWidth: newWidth,
});
BDFDB.ReactUtils.forceUpdate(instance);
};
}
switchImages (instance, type) {
2020-02-09 18:16:04 +01:00
let img = instance[type + "Img"];
let imgRef = instance[type + "Ref"];
if (!img || !imgRef) return;
2020-02-04 08:20:40 +01:00
delete instance.previousRef;
delete instance.nextRef;
2020-02-09 18:16:04 +01:00
delete instance.previousImg;
delete instance.nextImg;
instance.props.original = imgRef.props.src;
instance.props.placeholder = imgRef.props.src;
instance.props.src = imgRef.props.src;
instance.props.height = imgRef.props.height;
instance.props.width = imgRef.props.width;
instance.props.cachedImage = img;
2019-11-14 23:15:44 +01:00
BDFDB.ReactUtils.forceUpdate(instance);
2020-02-04 08:20:40 +01:00
}
cleanUpListeners () {
document.removeEventListener("keydown", document.keydownImageGalleryListener);
document.removeEventListener("keyup", document.keyupImageGalleryListener);
delete document.keydownImageGalleryListener;
delete document.keyupImageGalleryListener;
}
2018-10-11 10:21:26 +02:00
}
2020-02-04 08:20:40 +01:00
})();