1531 lines
78 KiB
JavaScript
1531 lines
78 KiB
JavaScript
/**
|
||
* @name ImageUtilities
|
||
* @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/ImageUtilities
|
||
* @source https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/ImageUtilities/ImageUtilities.plugin.js
|
||
* @updateUrl https://raw.githubusercontent.com/mwittrien/BetterDiscordAddons/master/Plugins/ImageUtilities/ImageUtilities.plugin.js
|
||
*/
|
||
|
||
module.exports = (_ => {
|
||
const config = {
|
||
"info": {
|
||
"name": "ImageUtilities",
|
||
"author": "DevilBro",
|
||
"version": "4.2.4",
|
||
"description": "Add a handful of options for images/emotes/avatars (direct download, reverse image search, zoom, copy image link, copy image to clipboard, gallery mode)"
|
||
},
|
||
"changeLog": {
|
||
"improved": {
|
||
"Download locations": "Merged 'save' and 'save as' entry, clicking it opens the save as modal, and hovering over it opens a submenu for preset paths, you can add your own download locations in the plugin settings",
|
||
"Toggle": "You can now toggle the download locations"
|
||
},
|
||
"fixed": {
|
||
"Slow modal transitions": "Fixed issue where the css of the plugin would cause render slow downds for modals",
|
||
"New download locations": "Fixed issue where you needed to restart plugin for new locations to be editable"
|
||
}
|
||
}
|
||
};
|
||
|
||
return !window.BDFDB_Global || (!window.BDFDB_Global.loaded && !window.BDFDB_Global.started) ? class {
|
||
getName () {return config.info.name;}
|
||
getAuthor () {return config.info.author;}
|
||
getVersion () {return config.info.version;}
|
||
getDescription () {return `The Library Plugin needed for ${config.info.name} is missing. Open the Plugin Settings to download it.\n\n${config.info.description}`;}
|
||
|
||
load () {
|
||
if (!window.BDFDB_Global || !Array.isArray(window.BDFDB_Global.pluginQueue)) window.BDFDB_Global = Object.assign({}, window.BDFDB_Global, {pluginQueue: []});
|
||
if (!window.BDFDB_Global.downloadModal) {
|
||
window.BDFDB_Global.downloadModal = true;
|
||
BdApi.showConfirmationModal("Library Missing", `The Library Plugin needed for ${config.info.name} is missing. Please click "Download Now" to install it.`, {
|
||
confirmText: "Download Now",
|
||
cancelText: "Cancel",
|
||
onCancel: _ => {delete window.BDFDB_Global.downloadModal;},
|
||
onConfirm: _ => {
|
||
delete window.BDFDB_Global.downloadModal;
|
||
require("request").get("https://mwittrien.github.io/BetterDiscordAddons/Library/0BDFDB.plugin.js", (e, r, b) => {
|
||
if (!e && b && b.indexOf(`* @name BDFDB`) > -1) require("fs").writeFile(require("path").join(BdApi.Plugins.folder, "0BDFDB.plugin.js"), b, _ => {});
|
||
else BdApi.alert("Error", "Could not download BDFDB Library Plugin, try again later or download it manually from GitHub: https://github.com/mwittrien/BetterDiscordAddons/tree/master/Library/");
|
||
});
|
||
}
|
||
});
|
||
}
|
||
if (!window.BDFDB_Global.pluginQueue.includes(config.info.name)) window.BDFDB_Global.pluginQueue.push(config.info.name);
|
||
}
|
||
start () {this.load();}
|
||
stop () {}
|
||
getSettingsPanel () {
|
||
let template = document.createElement("template");
|
||
template.innerHTML = `<div style="color: var(--header-primary); font-size: 16px; font-weight: 300; white-space: pre; line-height: 22px;">The Library Plugin needed for ${config.info.name} is missing.\nPlease click <a style="font-weight: 500;">Download Now</a> to install it.</div>`;
|
||
template.content.firstElementChild.querySelector("a").addEventListener("click", _ => {
|
||
require("request").get("https://mwittrien.github.io/BetterDiscordAddons/Library/0BDFDB.plugin.js", (e, r, b) => {
|
||
if (!e && b && b.indexOf(`* @name BDFDB`) > -1) require("fs").writeFile(require("path").join(BdApi.Plugins.folder, "0BDFDB.plugin.js"), b, _ => {});
|
||
else BdApi.alert("Error", "Could not download BDFDB Library Plugin, try again later or download it manually from GitHub: https://github.com/mwittrien/BetterDiscordAddons/tree/master/Library/");
|
||
});
|
||
});
|
||
return template.content.firstElementChild;
|
||
}
|
||
} : (([Plugin, BDFDB]) => {
|
||
const imgUrlReplaceString = "DEVILBRO_BD_REVERSEIMAGESEARCH_REPLACE_IMAGEURL";
|
||
var firedEvents = [], clickedImage;
|
||
var settings = {}, amounts = {}, zoomSettings = {}, engines = {}, enabledEngines = {}, ownLocations = {}, downloadsFolder;
|
||
|
||
const ImageDetails = class ImageDetails extends BdApi.React.Component {
|
||
componentDidMount () {
|
||
this.props.attachment = BDFDB.ReactUtils.findValue(BDFDB.ObjectUtils.get(this, `${BDFDB.ReactUtils.instanceKey}.return`), "attachment", {up: true});
|
||
BDFDB.ReactUtils.forceUpdate(this);
|
||
}
|
||
componentDidUpdate () {
|
||
if ((!this.props.attachment || !this.props.attachment.size) && !this.props.loaded) {
|
||
this.props.loaded = true;
|
||
this.props.attachment = BDFDB.ReactUtils.findValue(BDFDB.ObjectUtils.get(this, `${BDFDB.ReactUtils.instanceKey}.return`), "attachment", {up: true});
|
||
BDFDB.ReactUtils.forceUpdate(this);
|
||
}
|
||
}
|
||
render () {
|
||
return !this.props.attachment ? null : BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
|
||
className: BDFDB.disCN._imageutilitiesimagedetails,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Anchor, {
|
||
title: this.props.original,
|
||
href: this.props.original,
|
||
children: this.props.attachment.filename,
|
||
onClick: event => {
|
||
BDFDB.ListenerUtils.stopEvent(event);
|
||
BDFDB.DiscordUtils.openLink(this.props.original);
|
||
}
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
|
||
children: BDFDB.NumberUtils.formatBytes(this.props.attachment.size)
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
|
||
children: `${this.props.attachment.width}x${this.props.attachment.height}px`
|
||
})
|
||
})
|
||
]
|
||
});
|
||
}
|
||
};
|
||
|
||
return class ImageUtilities extends Plugin {
|
||
onLoad () {
|
||
firedEvents = [];
|
||
clickedImage = null;
|
||
|
||
this.defaults = {
|
||
settings: {
|
||
resizeImage: {value: true, inner: false, description: "Always resize Image to fit the whole Image Modal"},
|
||
addDetails: {value: true, inner: false, description: "Add Image Details (Name, Size, Amount) in the Image Modal"},
|
||
showAsHeader: {value: true, inner: false, description: "Show Image Details as a Details Header above the Image in the Chat"},
|
||
showOnHover: {value: false, inner: false, description: "Show Image Details as Tooltip in the Chat"},
|
||
enableGallery: {value: true, inner: false, description: "Display previous/next Images in the same message in the Image Modal"},
|
||
enableZoom: {value: true, inner: false, description: "Create a Zoom Lens if you press down on an Image in the Image Modal"},
|
||
enableCopyImg: {value: true, inner: false, description: "Add a copy Image option in the Image Modal"},
|
||
enableSaveImg: {value: true, inner: false, description: "Add a save Image as option in the Image Modal"},
|
||
addUserAvatarEntry: {value: true, inner: true, description: "User Avatars"},
|
||
addGuildIconEntry: {value: true, inner: true, description: "Server Icons"},
|
||
addEmojiEntry: {value: true, inner: true, description: "Custom Emojis/Emotes"}
|
||
},
|
||
amounts: {
|
||
hoverDelay: {value: 0, min: 0, description: "Image Tooltip Delay (in ms)"}
|
||
},
|
||
zoomSettings: {
|
||
zoomlevel: {value: 2, digits: 1, minValue: 1, maxValue: 20, unit: "x", label: "ACCESSIBILITY_ZOOM_LEVEL_LABEL"},
|
||
lensesize: {value: 200, digits: 0, minValue: 50, maxValue: 5000, unit: "px", label: "context_lenssize"}
|
||
},
|
||
engines: {
|
||
_all: {value: true, name: BDFDB.LanguageUtils.LanguageStrings.FORM_LABEL_ALL, url: null},
|
||
Baidu: {value: true, name: "Baidu", url: "http://image.baidu.com/pcdutu?queryImageUrl=" + imgUrlReplaceString},
|
||
Bing: {value: true, name: "Bing", url: "https://www.bing.com/images/search?q=imgurl: " + imgUrlReplaceString + "&view=detailv2&iss=sbi&FORM=IRSBIQ"},
|
||
Google: {value: true, name: "Google", url: "https://images.google.com/searchbyimage?image_url=" + imgUrlReplaceString},
|
||
IQDB: {value: true, name: "IQDB", url: "https://iqdb.org/?url=" + imgUrlReplaceString},
|
||
Reddit: {value: true, name: "Reddit", url: "http://karmadecay.com/search?q=" + imgUrlReplaceString},
|
||
SauceNAO: {value: true, name: "SauceNAO", url: "https://saucenao.com/search.php?db=999&url=" + imgUrlReplaceString},
|
||
Sogou: {value: true, name: "Sogou", url: "http://pic.sogou.com/ris?flag=1&drag=0&query=" + imgUrlReplaceString + "&flag=1"},
|
||
TinEye: {value: true, name: "TinEye", url: "https://tineye.com/search?url=" + imgUrlReplaceString},
|
||
WhatAnime: {value: true, name: "WhatAnime", url: "https://trace.moe/?url=" + imgUrlReplaceString},
|
||
Yandex: {value: true, name: "Yandex", url: "https://yandex.com/images/search?url=" + imgUrlReplaceString + "&rpt=imageview"}
|
||
}
|
||
};
|
||
|
||
this.patchedModules = {
|
||
before: {
|
||
LazyImage: "render"
|
||
},
|
||
after: {
|
||
ImageModal: ["render", "componentDidMount"],
|
||
LazyImage: ["render", "componentDidMount"]
|
||
}
|
||
};
|
||
|
||
this.css = `
|
||
${BDFDB.dotCN._imageutilitiesimagedetails} {
|
||
margin: 5px 0;
|
||
}
|
||
${BDFDB.dotCNS.spoilerhidden + BDFDB.dotCN._imageutilitiesimagedetails} {
|
||
visibility: hidden;
|
||
max-width: 1px;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesgallery},
|
||
${BDFDB.dotCN._imageutilitiesdetailsadded} {
|
||
transform: unset !important;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiessibling} {
|
||
display: flex;
|
||
align-items: center;
|
||
position: fixed;
|
||
top: 50%;
|
||
bottom: 50%;
|
||
cursor: pointer;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesprevious} {
|
||
justify-content: flex-end;
|
||
right: 90%;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesnext} {
|
||
justify-content: flex-start;
|
||
left: 90%;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesswitchicon} {
|
||
position: absolute;
|
||
background: rgba(0, 0, 0, 0.3);
|
||
border-radius: 50%;
|
||
padding: 15px;
|
||
transition: all 0.3s ease;
|
||
}
|
||
${BDFDB.dotCNS._imageutilitiesprevious + BDFDB.dotCN._imageutilitiesswitchicon} {
|
||
right: 10px;
|
||
}
|
||
${BDFDB.dotCNS._imageutilitiesnext + BDFDB.dotCN._imageutilitiesswitchicon} {
|
||
left: 10px;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiessibling}:hover ${BDFDB.dotCN._imageutilitiesswitchicon} {
|
||
background: rgba(0, 0, 0, 0.5);
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesdetailswrapper} {
|
||
position: fixed;
|
||
bottom: 10px;
|
||
left: 15px;
|
||
right: 15px;
|
||
pointer-events: none;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesdetails} {
|
||
color: #dcddde;
|
||
margin-top: 5px;
|
||
font-size: 14px;
|
||
font-weight: 500;
|
||
white-space: nowrap;
|
||
overflow: hidden;
|
||
text-overflow: ellipsis;
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesdetailslabel} {
|
||
font-weight: 600;
|
||
}
|
||
${BDFDB.dotCN._imageutilitieslense} {
|
||
border: 2px solid rgb(114, 137, 218);
|
||
}
|
||
${BDFDB.dotCN._imageutilitiesoperations} {
|
||
position: absolute;
|
||
display: flex;
|
||
}
|
||
${BDFDB.dotCNS._imageutilitiesoperations + BDFDB.dotCN.downloadlink} {
|
||
position: relative !important;
|
||
white-space: nowrap !important;
|
||
}
|
||
${BDFDB.dotCNS._imageutilitiesoperations + BDFDB.dotCN.anchor + BDFDB.dotCN.downloadlink} {
|
||
margin: 0 !important;
|
||
}
|
||
`;
|
||
}
|
||
|
||
onStart () {
|
||
// REMOVE 16.12.2020
|
||
let oL = BDFDB.DataUtils.load(this, "ownLocations"), c = false;
|
||
for (let i in oL) if (!BDFDB.ObjectUtils.is(oL[i])) {
|
||
oL[i] = {enabled: true, location: oL[i]};
|
||
c = true;
|
||
}
|
||
if (c) BDFDB.DataUtils.save(oL, this, "ownLocations")
|
||
|
||
BDFDB.ListenerUtils.add(this, document.body, "click", BDFDB.dotCNS.message + BDFDB.dotCNS.imagewrapper + "img", e => {
|
||
clickedImage = e.target;
|
||
BDFDB.TimeUtils.timeout(_ => {clickedImage = null;});
|
||
});
|
||
|
||
BDFDB.PatchUtils.patch(this, (BDFDB.ModuleUtils.findByName("renderImageComponent", false).exports || {}), "renderImageComponent", {after: e => {
|
||
if (e.returnValue && e.returnValue.type && (e.returnValue.type.displayName == "LazyImageZoomable" || e.returnValue.type.displayName == "LazyImage") && e.methodArguments[0].original && e.methodArguments[0].src.indexOf("https://media.discordapp.net/attachments") == 0 && (e.methodArguments[0].className || "").indexOf(BDFDB.disCN.embedthumbnail) == -1) return this.injectImageDetails(e.methodArguments[0], e.returnValue);
|
||
}});
|
||
|
||
this.forceUpdateAll();
|
||
}
|
||
|
||
onStop () {
|
||
this.cleanupListeners("Gallery");
|
||
this.cleanupListeners("Zoom");
|
||
|
||
this.forceUpdateAll();
|
||
}
|
||
|
||
getSettingsPanel (collapseStates = {}) {
|
||
let settingsPanel;
|
||
|
||
return settingsPanel = BDFDB.PluginUtils.createSettingsPanel(this, {
|
||
collapseStates: collapseStates,
|
||
children: _ => {
|
||
let settingsItems = [];
|
||
|
||
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
|
||
title: "Settings",
|
||
collapseStates: collapseStates,
|
||
children: Object.keys(settings).map(key => !this.defaults.settings[key].inner && BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
|
||
type: "Switch",
|
||
plugin: this,
|
||
keys: ["settings", key],
|
||
label: this.defaults.settings[key].description,
|
||
value: settings[key]
|
||
})).concat(Object.keys(amounts).map(key => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
|
||
type: "TextInput",
|
||
plugin: this,
|
||
keys: ["amounts", key],
|
||
label: this.defaults.amounts[key].description,
|
||
basis: "50%",
|
||
childProps: {type: "number"},
|
||
min: this.defaults.amounts[key].min,
|
||
max: this.defaults.amounts[key].max,
|
||
value: amounts[key]
|
||
}))).filter(n => n)
|
||
}));
|
||
|
||
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
|
||
title: "Download Locations",
|
||
collapseStates: collapseStates,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormTitle, {
|
||
className: BDFDB.disCN.marginbottom4,
|
||
tag: BDFDB.LibraryComponents.FormComponents.FormTitle.Tags.H3,
|
||
children: "Add additional Download Locations: "
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex, {
|
||
className: BDFDB.disCN.marginbottom8,
|
||
align: BDFDB.LibraryComponents.Flex.Align.END,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormItem, {
|
||
title: "Name:",
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextInput, {
|
||
className: "input-newlocation input-name",
|
||
value: "",
|
||
placeholder: "Name"
|
||
})
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormItem, {
|
||
title: "Location:",
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextInput, {
|
||
className: "input-newlocation input-location",
|
||
value: "",
|
||
placeholder: "Location"
|
||
})
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Button, {
|
||
style: {marginBottom: 1},
|
||
onClick: _ => {
|
||
for (let input of settingsPanel.props._node.querySelectorAll(".input-newlocation " + BDFDB.dotCN.input)) if (!input.value || input.value.length == 0 || input.value.trim().length == 0) return BDFDB.NotificationUtils.toast("Fill out all fields to add a new location.", {type: "danger"});
|
||
let name = settingsPanel.props._node.querySelector(".input-name " + BDFDB.dotCN.input).value.trim();
|
||
let location = settingsPanel.props._node.querySelector(".input-location " + BDFDB.dotCN.input).value.trim();
|
||
if (ownLocations[name] || name == "Downloads") return BDFDB.NotificationUtils.toast("A location with the choosen name already exists, please choose another name", {type: "danger"});
|
||
else if (!BDFDB.LibraryRequires.fs.existsSync(location)) return BDFDB.NotificationUtils.toast("The choosen download location is not a valid path to a folder", {type: "danger"});
|
||
else {
|
||
ownLocations[name] = {enabled: true, location: location};
|
||
BDFDB.DataUtils.save(ownLocations, this, "ownLocations");
|
||
BDFDB.PluginUtils.refreshSettingsPanel(this, settingsPanel, collapseStates);
|
||
}
|
||
},
|
||
children: BDFDB.LanguageUtils.LanguageStrings.ADD
|
||
})
|
||
]
|
||
})
|
||
].concat(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsPanelList, {
|
||
title: "Your own Download Locations:",
|
||
dividerTop: true,
|
||
children: Object.keys(ownLocations).map(name => {
|
||
let editable = name != "Downloads";
|
||
return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Card, {
|
||
horizontal: true,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
grow: 0,
|
||
basis: "180px",
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextInput, {
|
||
value: name,
|
||
placeholder: name,
|
||
size: BDFDB.LibraryComponents.TextInput.Sizes.MINI,
|
||
maxLength: 100000000000000000000,
|
||
style: {marginRight: 6},
|
||
disabled: !editable,
|
||
onChange: !editable ? null : value => {
|
||
ownLocations[value] = ownLocations[name];
|
||
delete ownLocations[name];
|
||
BDFDB.DataUtils.save(ownLocations, this, "ownLocations");
|
||
}
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextInput, {
|
||
value: ownLocations[name].location,
|
||
placeholder: ownLocations[name].location,
|
||
size: BDFDB.LibraryComponents.TextInput.Sizes.MINI,
|
||
maxLength: 100000000000000000000,
|
||
style: {marginRight: 10},
|
||
disabled: !editable,
|
||
onChange: !editable ? null : value => {
|
||
ownLocations[name].location = value;
|
||
BDFDB.DataUtils.save(ownLocations, this, "ownLocations");
|
||
}
|
||
})
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Flex.Child, {
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Switch, {
|
||
value: ownLocations[name].enabled,
|
||
size: BDFDB.LibraryComponents.Switch.Sizes.MINI,
|
||
onChange: value => {
|
||
ownLocations[name].enabled = value;
|
||
BDFDB.DataUtils.save(ownLocations, this, "ownLocations");
|
||
}
|
||
})
|
||
})
|
||
],
|
||
noRemove: !editable,
|
||
onRemove: !editable ? null : _ => {
|
||
delete ownLocations[name];
|
||
BDFDB.DataUtils.save(ownLocations, this, "ownLocations");
|
||
BDFDB.PluginUtils.refreshSettingsPanel(this, settingsPanel);
|
||
}
|
||
});
|
||
})
|
||
})).filter(n => n)
|
||
}));
|
||
|
||
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
|
||
title: "Context Menu Entries",
|
||
collapseStates: collapseStates,
|
||
children: [BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.FormComponents.FormTitle, {
|
||
className: BDFDB.disCN.marginbottom4,
|
||
tag: BDFDB.LibraryComponents.FormComponents.FormTitle.Tags.H3,
|
||
children: "Add additional Context Menu Entry for: "
|
||
})].concat(Object.keys(settings).map(key => this.defaults.settings[key].inner && BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
|
||
type: "Switch",
|
||
plugin: this,
|
||
keys: ["settings", key],
|
||
label: this.defaults.settings[key].description,
|
||
value: settings[key]
|
||
})))
|
||
}));
|
||
|
||
settingsItems.push(BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.CollapseContainer, {
|
||
title: "Search Engines",
|
||
collapseStates: collapseStates,
|
||
children: Object.keys(engines).filter(n => n && n != "_all").map(key => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SettingsSaveItem, {
|
||
type: "Switch",
|
||
plugin: this,
|
||
keys: ["engines", key],
|
||
label: this.defaults.engines[key].name,
|
||
value: engines[key]
|
||
}))
|
||
}));
|
||
|
||
|
||
|
||
return settingsItems;
|
||
}
|
||
});
|
||
}
|
||
|
||
onSettingsClosed () {
|
||
if (this.SettingsUpdated) {
|
||
delete this.SettingsUpdated;
|
||
this.forceUpdateAll();
|
||
}
|
||
}
|
||
|
||
forceUpdateAll () {
|
||
settings = BDFDB.DataUtils.get(this, "settings");
|
||
amounts = BDFDB.DataUtils.get(this, "amounts");
|
||
zoomSettings = BDFDB.DataUtils.get(this, "zoomSettings");
|
||
engines = BDFDB.DataUtils.get(this, "engines");
|
||
enabledEngines = BDFDB.ObjectUtils.filter(engines, n => n);
|
||
ownLocations = Object.assign({"Downloads": {enabled:true, location: this.getDownloadLocation()}}, BDFDB.DataUtils.load(this, "ownLocations"));
|
||
|
||
BDFDB.PatchUtils.forceAllUpdates(this);
|
||
BDFDB.MessageUtils.rerenderAll();
|
||
}
|
||
|
||
onGuildContextMenu (e) {
|
||
if (e.instance.props.guild && settings.addGuildIconEntry) {
|
||
let banner = BDFDB.DOMUtils.getParent(BDFDB.dotCN.guildheader, e.instance.props.target) || BDFDB.DOMUtils.getParent(BDFDB.dotCN.guildchannels, e.instance.props.target) && !e.instance.props.target.className && e.instance.props.target.parentElement.firstElementChild == e.instance.props.target;
|
||
if (banner) {
|
||
if (e.instance.props.guild.banner) this.injectItem(e, BDFDB.LibraryModules.IconUtils.getGuildBannerURL(e.instance.props.guild));
|
||
}
|
||
else if (e.type != "GuildChannelListContextMenu") this.injectItem(e, e.instance.props.guild.getIconURL("png"), BDFDB.LibraryModules.IconUtils.hasAnimatedGuildIcon(e.instance.props.guild) && e.instance.props.guild.getIconURL("gif"));
|
||
}
|
||
}
|
||
|
||
onUserContextMenu (e) {
|
||
if (e.instance.props.user && settings.addUserAvatarEntry) this.injectItem(e, e.instance.props.user.getAvatarURL("png"), BDFDB.LibraryModules.IconUtils.hasAnimatedAvatar(e.instance.props.user) && e.instance.props.user.getAvatarURL("gif"))
|
||
}
|
||
|
||
onNativeContextMenu (e) {
|
||
if (e.type == "NativeImageContextMenu" && (e.instance.props.href || e.instance.props.src)) {
|
||
this.injectItem(e, e.instance.props.href || e.instance.props.src);
|
||
}
|
||
}
|
||
|
||
onMessageContextMenu (e) {
|
||
if (e.instance.props.message && e.instance.props.channel && e.instance.props.target) {
|
||
if (e.instance.props.attachment) this.injectItem(e, e.instance.props.attachment.url);
|
||
else if (e.instance.props.target.tagName == "A" && e.instance.props.message.embeds && e.instance.props.message.embeds[0] && e.instance.props.message.embeds[0].type == "image") this.injectItem(e, e.instance.props.target.href);
|
||
else if (e.instance.props.target.tagName == "IMG") {
|
||
if (BDFDB.DOMUtils.containsClass(e.instance.props.target.parentElement, BDFDB.disCN.imagewrapper)) this.injectItem(e, e.instance.props.target.src);
|
||
else if (BDFDB.DOMUtils.containsClass(e.instance.props.target, BDFDB.disCN.embedauthoricon) && settings.addUserAvatarEntry) this.injectItem(e, e.instance.props.target.src);
|
||
else if (BDFDB.DOMUtils.containsClass(e.instance.props.target, BDFDB.disCN.emojiold, "emote", false) && settings.addEmojiEntry) this.injectItem(e, e.instance.props.target.src);
|
||
}
|
||
else {
|
||
let reaction = BDFDB.DOMUtils.getParent(BDFDB.dotCN.messagereaction, e.instance.props.target);
|
||
if (reaction && settings.addEmojiEntry) this.injectItem(e, reaction.querySelector(BDFDB.dotCN.emojiold).src);
|
||
}
|
||
}
|
||
}
|
||
|
||
injectItem (e, ...urls) {
|
||
let types = [];
|
||
let validUrls = urls.filter(n => this.isValidImg(n)).map(n => {
|
||
let url = n.replace(/^url\(|\)$|"|'/g, "").replace(/\?size\=\d+$/, "?size=4096").replace(/[\?\&](height|width)=\d+/g, "");
|
||
if (url.indexOf("https://images-ext-1.discordapp.net/external/") > -1) {
|
||
if (url.split("/https/").length > 1) url = "https://" + url.split("/https/").pop();
|
||
else if (url.split("/http/").length > 1) url = "http://" + url.split("/http/").pop();
|
||
}
|
||
const file = url && (BDFDB.LibraryModules.URLParser.parse(url).pathname || "").toLowerCase();
|
||
const type = file && file.split(".").pop();
|
||
return url && type && !types.includes(type) && types.push(type) && {url, type};
|
||
}).filter(n => n);
|
||
if (!validUrls.length) return;
|
||
let [children, index] = BDFDB.ContextMenuUtils.findItem(e.returnvalue, {id: "devmode-copy-id", group: true});
|
||
children.splice(index > -1 ? index : children.length, 0, BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuGroup, {
|
||
children: BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: BDFDB.LanguageUtils.LanguageStrings.IMAGE + " " + BDFDB.LanguageUtils.LanguageStrings.ACTIONS,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "main-subitem"),
|
||
children: validUrls.length == 1 ? this.createUrlMenu(e, validUrls[0].url) : validUrls.map((urlData, i) => BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: urlData.type.toUpperCase(),
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "subitem", i),
|
||
children: this.createUrlMenu(e, urlData.url)
|
||
}))
|
||
})
|
||
}));
|
||
}
|
||
|
||
createUrlMenu (e, url) {
|
||
let enginesWithoutAll = BDFDB.ObjectUtils.filter(enabledEngines, n => n != "_all", true);
|
||
let engineKeys = Object.keys(enginesWithoutAll);
|
||
let locations = Object.keys(ownLocations).filter(n => ownLocations[n].enabled);
|
||
return [
|
||
BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_viewimage,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "view-image"),
|
||
action: _ => {
|
||
let img = new Image();
|
||
img.onload = function() {
|
||
BDFDB.LibraryModules.ModalUtils.openModal(modalData => {
|
||
return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ModalComponents.ModalRoot, Object.assign({
|
||
className: BDFDB.disCN.imagemodal
|
||
}, modalData, {
|
||
size: BDFDB.LibraryComponents.ModalComponents.ModalSize.DYNAMIC,
|
||
"aria-label": BDFDB.LanguageUtils.LanguageStrings.IMAGE,
|
||
children: BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.ImageModal, {
|
||
src: url,
|
||
original: url,
|
||
width: this.width,
|
||
height: this.height,
|
||
className: BDFDB.disCN.imagemodalimage,
|
||
shouldAnimate: true,
|
||
renderLinkComponent: props => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Anchor, props)
|
||
})
|
||
}), true);
|
||
});
|
||
};
|
||
img.src = url;
|
||
}
|
||
}),
|
||
BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_saveimageas,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "download-image-as"),
|
||
action: _ => {
|
||
this.downloadImageAs(url);
|
||
},
|
||
children: locations.length && BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuGroup, {
|
||
children: locations.map((name, i) => BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "download", name, i),
|
||
label: name,
|
||
action: _ => {
|
||
this.downloadImage(url, ownLocations[name].location);
|
||
}
|
||
}))
|
||
})
|
||
}),
|
||
!this.isCopyable(url) ? null : BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_copyimage,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "copy-image"),
|
||
action: _ => {
|
||
this.copyImage(url);
|
||
}
|
||
}),
|
||
BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_copyimagelink,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "copy-src"),
|
||
action: _ => {
|
||
BDFDB.LibraryRequires.electron.clipboard.write({text: url});
|
||
BDFDB.NotificationUtils.toast(this.labels.toast_copyimagelink_success, {type: "success"});
|
||
}
|
||
}),
|
||
!this.isSearchable(url) ? null : engineKeys.length == 1 ? BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_reverseimagesearch.replace("...", this.defaults.engines[engineKeys[0]].name),
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "single-search"),
|
||
persisting: true,
|
||
action: event => {
|
||
if (!event.shiftKey) BDFDB.ContextMenuUtils.close(e.instance);
|
||
BDFDB.DiscordUtils.openLink(this.defaults.engines[engineKeys[0]].url.replace(imgUrlReplaceString, encodeURIComponent(url)), {
|
||
minimized: event.shiftKey
|
||
});
|
||
}
|
||
}) : BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.context_reverseimagesearch,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "submenu-search"),
|
||
children: !engineKeys.length ? BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.labels.submenu_disabled,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "disabled"),
|
||
disabled: true
|
||
}) : Object.keys(enabledEngines).map(key => BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
label: this.defaults.engines[key].name,
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "search", key),
|
||
color: key == "_all" ? BDFDB.LibraryComponents.MenuItems.Colors.DANGER : BDFDB.LibraryComponents.MenuItems.Colors.DEFAULT,
|
||
persisting: true,
|
||
action: event => {
|
||
if (!event.shiftKey) BDFDB.ContextMenuUtils.close(e.instance);
|
||
if (key == "_all") {
|
||
for (let key2 in enginesWithoutAll) BDFDB.DiscordUtils.openLink(this.defaults.engines[key2].url.replace(imgUrlReplaceString, encodeURIComponent(url)), {
|
||
minimized: event.shiftKey
|
||
});
|
||
}
|
||
else BDFDB.DiscordUtils.openLink(this.defaults.engines[key].url.replace(imgUrlReplaceString, encodeURIComponent(url)), {
|
||
minimized: event.shiftKey
|
||
});
|
||
}
|
||
}))
|
||
})
|
||
].filter(n => n);
|
||
}
|
||
|
||
processImageModal (e) {
|
||
if (clickedImage) e.instance.props.cachedImage = clickedImage;
|
||
let url = e.instance.props.cachedImage && e.instance.props.cachedImage.src ? e.instance.props.cachedImage : e.instance.props.src;
|
||
url = url.src || url;
|
||
let messages = this.getMessageGroupOfImage(url);
|
||
if (e.returnvalue) {
|
||
let [children, index] = BDFDB.ReactUtils.findParent(e.returnvalue, {props: [["className", BDFDB.disCN.downloadlink]]});
|
||
if (index > -1) {
|
||
children[index].props.onClick = event => {
|
||
return event.shiftKey;
|
||
};
|
||
let openContext = event => {
|
||
BDFDB.ContextMenuUtils.open(this, event, BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuGroup, {
|
||
children: Object.keys(zoomSettings).map(type => BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuSliderItem, Object.assign({
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, type),
|
||
value: zoomSettings[type],
|
||
renderLabel: value => {
|
||
return (this.labels[this.defaults.zoomSettings[type].label] || BDFDB.LanguageUtils.LanguageStrings[this.defaults.zoomSettings[type].label]) + ": " + value + this.defaults.zoomSettings[type].unit;
|
||
},
|
||
onValueRender: value => {
|
||
return value + this.defaults.zoomSettings[type].unit;
|
||
},
|
||
onValueChange: value => {
|
||
zoomSettings[type] = value;
|
||
BDFDB.DataUtils.save(zoomSettings, this, "zoomSettings");
|
||
}
|
||
}, BDFDB.ObjectUtils.extract(this.defaults.zoomSettings[type], "digits", "minValue", "maxValue"))))
|
||
}));
|
||
};
|
||
let isVideo = (typeof e.instance.props.children == "function" && e.instance.props.children(Object.assign({}, e.instance.props, {size: e.instance.props})) || {type: {}}).type.displayName == "Video";
|
||
children[index] = BDFDB.ReactUtils.createElement("span", {
|
||
className: BDFDB.disCN._imageutilitiesoperations,
|
||
children: [
|
||
children[index],
|
||
settings.enableSaveImg && !isVideo && [
|
||
BDFDB.ReactUtils.createElement("span", {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: "|",
|
||
style: {margin: "0 5px"}
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Anchor, {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: this.labels.context_saveimageas,
|
||
onClick: event => {
|
||
BDFDB.ListenerUtils.stopEvent(event);
|
||
this.downloadImageAs(url);
|
||
},
|
||
onContextMenu: event => {
|
||
let locations = Object.keys(ownLocations).filter(n => ownLocations[n].enabled);
|
||
if (locations.length) BDFDB.ContextMenuUtils.open(this, event, BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuGroup, {
|
||
children: Object.keys(locations).map((name, i) => BDFDB.ContextMenuUtils.createItem(BDFDB.LibraryComponents.MenuItems.MenuItem, {
|
||
id: BDFDB.ContextMenuUtils.createItemId(this.name, "download", name, i),
|
||
label: name,
|
||
action: _ => {
|
||
this.downloadImage(url, ownLocations[name].location);
|
||
}
|
||
}))
|
||
}));
|
||
}
|
||
})
|
||
],
|
||
settings.enableCopyImg && this.isCopyable(url) && !isVideo && [
|
||
BDFDB.ReactUtils.createElement("span", {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: "|",
|
||
style: {margin: "0 5px"}
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Anchor, {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: this.labels.context_copyimage,
|
||
onClick: event => {
|
||
BDFDB.ListenerUtils.stopEvent(event);
|
||
this.copyImage(url);
|
||
}
|
||
})
|
||
],
|
||
settings.enableZoom && !isVideo && [
|
||
BDFDB.ReactUtils.createElement("span", {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: "|",
|
||
style: {margin: "0 5px"}
|
||
}),
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.Anchor, {
|
||
className: BDFDB.disCN.downloadlink,
|
||
children: `Zoom ${BDFDB.LanguageUtils.LanguageStrings.SETTINGS}`,
|
||
onClick: openContext,
|
||
onContextMenu: openContext
|
||
})
|
||
]
|
||
].flat(10).filter(n => n)
|
||
});
|
||
}
|
||
let imageIndex = 0, amount = 1;
|
||
if (messages.length) {
|
||
let data = this.getSiblingsAndPosition(url, messages);
|
||
imageIndex = data.index;
|
||
amount = data.amount;
|
||
if (data.previous) {
|
||
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, data.previous, "previous");
|
||
}
|
||
if (data.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, data.next, "next");
|
||
}
|
||
}
|
||
if (settings.addDetails) e.returnvalue.props.children.push(BDFDB.ReactUtils.createElement("div", {
|
||
className: BDFDB.disCN._imageutilitiesdetailswrapper,
|
||
children: [
|
||
{label: "Source", text: e.instance.props.src},
|
||
{label: "Size", text: `${e.instance.props.width} x ${e.instance.props.height}px`},
|
||
{label: "Image", text: `${imageIndex + 1} of ${amount}`}
|
||
].map(data => BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TextElement, {
|
||
className: BDFDB.disCN._imageutilitiesdetails,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement("div", {
|
||
className: BDFDB.disCN._imageutilitiesdetailslabel,
|
||
children: data.label + ":"
|
||
}),
|
||
data.text
|
||
]
|
||
}))
|
||
}));
|
||
}
|
||
if (e.node) {
|
||
let modal = BDFDB.DOMUtils.getParent(BDFDB.dotCNC.modal + BDFDB.dotCN.layermodal, e.node);
|
||
if (modal) {
|
||
modal.className = BDFDB.DOMUtils.formatClassName(modal.className, messages.length && BDFDB.disCN._imageutilitiesgallery, settings.addDetails && BDFDB.disCN._imageutilitiesdetailsadded);
|
||
this.cleanupListeners("Gallery");
|
||
if (messages.length) {
|
||
document.keydownImageUtilitiesGalleryListener = event => {
|
||
if (!document.contains(e.node)) this.cleanupListeners("Gallery");
|
||
else if (!firedEvents.includes("Gallery")) {
|
||
firedEvents.push("Gallery");
|
||
if (event.keyCode == 37) this.switchImages(e.instance, "previous");
|
||
else if (event.keyCode == 39) this.switchImages(e.instance, "next");
|
||
}
|
||
};
|
||
document.keyupImageUtilitiesGalleryListener = _ => {
|
||
BDFDB.ArrayUtils.remove(firedEvents, "Gallery", true);
|
||
if (!document.contains(e.node)) this.cleanupListeners("Gallery");
|
||
};
|
||
document.addEventListener("keydown", document.keydownImageUtilitiesGalleryListener);
|
||
document.addEventListener("keyup", document.keyupImageUtilitiesGalleryListener);
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
processLazyImage (e) {
|
||
if (e.node) {
|
||
if (e.instance.props.resized) e.instance.state.readyState = BDFDB.LibraryComponents.Image.ImageReadyStates.READY;
|
||
if (settings.enableZoom && !e.node.querySelector("video") && !BDFDB.DOMUtils.containsClass(e.node.parentElement, BDFDB.disCN._imageutilitiessibling) && BDFDB.DOMUtils.getParent(BDFDB.dotCN.imagemodal, e.node)) {
|
||
e.node.addEventListener("mousedown", event => {
|
||
if (event.which != 1) return;
|
||
BDFDB.ListenerUtils.stopEvent(event);
|
||
|
||
let imgRects = BDFDB.DOMUtils.getRects(e.node.firstElementChild);
|
||
|
||
let lens = BDFDB.DOMUtils.create(`<div class="${BDFDB.disCN._imageutilitieslense}" style="border-radius: 50% !important; pointer-events: none !important; z-index: 10000 !important; width: ${zoomSettings.lensesize}px !important; height: ${zoomSettings.lensesize}px !important; position: fixed !important;"><div style="position: absolute !important; top: 0 !important; right: 0 !important; bottom: 0 !important; left: 0 !important;"><${e.node.firstElementChild.tagName} src="${e.instance.props.src}" style="width: ${imgRects.width * zoomSettings.zoomlevel}px; height: ${imgRects.height * zoomSettings.zoomlevel}px; position: fixed !important;"${e.node.firstElementChild.tagName == "VIDEO" ? " loop autoplay" : ""}></${e.node.firstElementChild.tagName}></div></div>`);
|
||
let pane = lens.firstElementChild.firstElementChild;
|
||
let backdrop = BDFDB.DOMUtils.create(`<div class="${BDFDB.disCN._imageutilitieslensebackdrop}" style="background: rgba(0, 0, 0, 0.3) !important; position: absolute !important; top: 0 !important; right: 0 !important; bottom: 0 !important; left: 0 !important; pointer-events: none !important; z-index: 8000 !important;"></div>`);
|
||
let appMount = document.querySelector(BDFDB.dotCN.appmount);
|
||
appMount.appendChild(lens);
|
||
appMount.appendChild(backdrop);
|
||
|
||
let lensRects = BDFDB.DOMUtils.getRects(lens);
|
||
|
||
let halfW = lensRects.width / 2, halfH = lensRects.height / 2;
|
||
let minX = imgRects.left, maxX = minX + imgRects.width;
|
||
let minY = imgRects.top, maxY = minY + imgRects.height;
|
||
|
||
lens.update = _ => {
|
||
let x = event.clientX > maxX ? maxX - halfW : event.clientX < minX ? minX - halfW : event.clientX - halfW;
|
||
let y = event.clientY > maxY ? maxY - halfH : event.clientY < minY ? minY - halfH : event.clientY - halfH;
|
||
lens.style.setProperty("left", x + "px", "important");
|
||
lens.style.setProperty("top", y + "px", "important");
|
||
lens.style.setProperty("width", zoomSettings.lensesize + "px", "important");
|
||
lens.style.setProperty("height", zoomSettings.lensesize + "px", "important");
|
||
lens.style.setProperty("clip-path", `circle(${(zoomSettings.lensesize/2) + 2}px at center)`, "important");
|
||
lens.firstElementChild.style.setProperty("clip-path", `circle(${zoomSettings.lensesize/2}px at center)`, "important");
|
||
pane.style.setProperty("left", imgRects.left + ((zoomSettings.zoomlevel - 1) * (imgRects.left - x - halfW)) + "px", "important");
|
||
pane.style.setProperty("top", imgRects.top + ((zoomSettings.zoomlevel - 1) * (imgRects.top - y - halfH)) + "px", "important");
|
||
pane.style.setProperty("width", imgRects.width * zoomSettings.zoomlevel + "px", "important");
|
||
pane.style.setProperty("height", imgRects.height * zoomSettings.zoomlevel + "px", "important");
|
||
};
|
||
lens.update();
|
||
|
||
let dragging = event2 => {
|
||
event = event2;
|
||
lens.update();
|
||
};
|
||
let releasing = _ => {
|
||
this.cleanupListeners("Zoom");
|
||
document.removeEventListener("mousemove", dragging);
|
||
document.removeEventListener("mouseup", releasing);
|
||
if (document.removeImageUtilitiesZoomObserver) {
|
||
document.removeImageUtilitiesZoomObserver.disconnect();
|
||
delete document.removeImageUtilitiesZoomObserver;
|
||
}
|
||
BDFDB.DOMUtils.remove(lens, backdrop);
|
||
BDFDB.DataUtils.save(zoomSettings, this, "zoomSettings");
|
||
};
|
||
document.addEventListener("mousemove", dragging);
|
||
document.addEventListener("mouseup", releasing);
|
||
|
||
this.cleanupListeners("Zoom");
|
||
document.wheelImageUtilitiesZoomListener = event2 => {
|
||
if (!document.contains(e.node)) this.cleanupListeners("Zoom");
|
||
else {
|
||
if (event2.deltaY < 0 && (zoomSettings.zoomlevel + 0.1) <= this.defaults.zoomSettings.zoomlevel.maxValue) {
|
||
zoomSettings.zoomlevel += 0.1;
|
||
lens.update();
|
||
}
|
||
else if (event2.deltaY > 0 && (zoomSettings.zoomlevel - 0.1) >= this.defaults.zoomSettings.zoomlevel.minValue) {
|
||
zoomSettings.zoomlevel -= 0.1;
|
||
lens.update();
|
||
}
|
||
}
|
||
};
|
||
document.keydownImageUtilitiesZoomListener = event2 => {
|
||
if (!document.contains(e.node)) this.cleanupListeners("Zoom");
|
||
else if (!firedEvents.includes("Zoom")) {
|
||
firedEvents.push("Zoom");
|
||
if (event2.keyCode == 187 && (zoomSettings.zoomlevel + 0.5) <= this.defaults.zoomSettings.zoomlevel.maxValue) {
|
||
zoomSettings.zoomlevel += 0.5;
|
||
lens.update();
|
||
}
|
||
else if (event2.keyCode == 189 && (zoomSettings.zoomlevel - 0.5) >= this.defaults.zoomSettings.zoomlevel.minValue) {
|
||
zoomSettings.zoomlevel -= 0.5;
|
||
lens.update();
|
||
}
|
||
}
|
||
};
|
||
document.keyupImageUtilitiesZoomListener = _ => {
|
||
BDFDB.ArrayUtils.remove(firedEvents, "Zoom", true);
|
||
if (!document.contains(e.node)) this.cleanupListeners("Zoom");
|
||
};
|
||
document.addEventListener("wheel", document.wheelImageUtilitiesZoomListener);
|
||
document.addEventListener("keydown", document.keydownImageUtilitiesZoomListener);
|
||
document.addEventListener("keyup", document.keyupImageUtilitiesZoomListener);
|
||
|
||
document.removeImageUtilitiesZoomObserver = new MutationObserver(changes => changes.forEach(change => {
|
||
let nodes = Array.from(change.removedNodes);
|
||
if (nodes.indexOf(appMount) > -1 || nodes.some(n => n.contains(appMount)) || nodes.indexOf(e.node) > -1 || nodes.some(n => n.contains(e.node))) {
|
||
releasing();
|
||
}
|
||
}));
|
||
document.removeImageUtilitiesZoomObserver.observe(document.body, {subtree: true, childList: true});
|
||
});
|
||
}
|
||
}
|
||
else if (e.returnvalue) {
|
||
if (settings.showOnHover && e.instance.props.original && e.instance.props.src.indexOf("https://media.discordapp.net/attachments") == 0 && typeof e.returnvalue.props.children == "function") {
|
||
let attachment = BDFDB.ReactUtils.findValue(e.instance, "attachment", {up: true});
|
||
if (attachment) {
|
||
let renderChildren = e.returnvalue.props.children;
|
||
e.returnvalue.props.children = (...args) => {
|
||
return BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.TooltipContainer, {
|
||
text: `${attachment.filename}\n${BDFDB.NumberUtils.formatBytes(attachment.size)}\n${attachment.width}x${attachment.height}px`,
|
||
tooltipConfig: {
|
||
type: "right",
|
||
delay: amounts.hoverDelay
|
||
},
|
||
children: renderChildren(...args)
|
||
});
|
||
};
|
||
}
|
||
}
|
||
}
|
||
else {
|
||
if (settings.resizeImage && e.instance.props.className && e.instance.props.className.indexOf(BDFDB.disCN.imagemodalimage) > -1 && BDFDB.ReactUtils.findOwner(BDFDB.ObjectUtils.get(e, `instance.${BDFDB.ReactUtils.instanceKey}`), {name: "ImageModal", up: true})) {
|
||
let data = settings.enableGallery ? this.getSiblingsAndPosition(e.instance.props.src, this.getMessageGroupOfImage(e.instance.props.src)) : {};
|
||
let aRects = BDFDB.DOMUtils.getRects(document.querySelector(BDFDB.dotCN.appmount));
|
||
let ratio = Math.min((aRects.width * (data.previous || data.next ? 0.8 : 1) - 20) / e.instance.props.width, (aRects.height - (settings.addDetails ? 310 : 100)) / e.instance.props.height);
|
||
let width = Math.round(ratio * e.instance.props.width);
|
||
let height = Math.round(ratio * e.instance.props.height);
|
||
e.instance.props.width = width;
|
||
e.instance.props.maxWidth = width;
|
||
e.instance.props.height = height;
|
||
e.instance.props.maxHeight = height;
|
||
e.instance.props.src = e.instance.props.src.replace(/width=\d+/, `width=${width}`).replace(/height=\d+/, `height=${height}`);
|
||
e.instance.props.resized = true;
|
||
}
|
||
}
|
||
}
|
||
|
||
injectImageDetails (props, child) {
|
||
if (settings.showAsHeader) {
|
||
props.detailsAdded = true;
|
||
return BDFDB.ReactUtils.createElement("div", {
|
||
className: BDFDB.disCN.embedwrapper,
|
||
children: [
|
||
BDFDB.ReactUtils.createElement(ImageDetails, {
|
||
original: props.original,
|
||
attachment: {
|
||
height: 0,
|
||
width: 0,
|
||
filename: "unknown.png"
|
||
}
|
||
}),
|
||
child
|
||
]
|
||
});
|
||
}
|
||
return child;
|
||
}
|
||
|
||
isValidImg (url) {
|
||
const file = url && (BDFDB.LibraryModules.URLParser.parse(url).pathname || "").toLowerCase();
|
||
return file && (url.startsWith("https://images-ext-2.discordapp.net/") || file.endsWith(".jpg") || file.endsWith(".jpeg") || file.endsWith(".png") || file.endsWith(".gif") || file.endsWith(".apng") || file.endsWith(".webp") || file.endsWith(".svg"));
|
||
}
|
||
|
||
isCopyable (url) {
|
||
const file = url && (BDFDB.LibraryModules.URLParser.parse(url).pathname || "").toLowerCase();
|
||
return file && (url.startsWith("https://images-ext-2.discordapp.net/") || file.endsWith(".jpg") || file.endsWith(".jpeg") || file.endsWith(".png"));
|
||
}
|
||
|
||
isSearchable (url) {
|
||
const file = url && (BDFDB.LibraryModules.URLParser.parse(url).pathname || "").toLowerCase();
|
||
return file && (url.startsWith("https://images-ext-2.discordapp.net/") || file.endsWith(".jpg") || file.endsWith(".jpeg") || file.endsWith(".png") || file.endsWith(".gif") || file.endsWith(".apng") || file.endsWith(".webp"));
|
||
}
|
||
|
||
downloadImage (url, path) {
|
||
url = url.startsWith("/assets") ? (window.location.origin + url) : url;
|
||
BDFDB.LibraryRequires.request(url, {encoding: null}, (error, response, body) => {
|
||
if (error) BDFDB.NotificationUtils.toast(this.labels.toast_saveimage_failed.replace("{{path}}", path), {type: "error"});
|
||
else {
|
||
BDFDB.LibraryRequires.fs.writeFile(this.getFileName(path, url.split("/").pop().split(".").slice(0, -1).join("."), response.headers["content-type"].split("/").pop().split("+")[0], 0), body, error => {
|
||
if (error) BDFDB.NotificationUtils.toast(this.labels.toast_saveimage_failed.replace("{{path}}", path), {type: "error"});
|
||
else BDFDB.NotificationUtils.toast(this.labels.toast_saveimage_success.replace("{{path}}", path), {type: "success"});
|
||
});
|
||
}
|
||
});
|
||
}
|
||
|
||
downloadImageAs (url) {
|
||
url = url.startsWith("/assets") ? (window.location.origin + url) : url;
|
||
BDFDB.LibraryRequires.request(url, {encoding: null}, (error, response, body) => {
|
||
let hrefURL = window.URL.createObjectURL(new Blob([body]));
|
||
let tempLink = document.createElement("a");
|
||
tempLink.href = hrefURL;
|
||
tempLink.download = `${url.split("/").pop().split(".").slice(0, -1).join(".")}.${response.headers["content-type"].split("/").pop().split("+")[0]}`;
|
||
tempLink.click();
|
||
window.URL.revokeObjectURL(hrefURL);
|
||
});
|
||
}
|
||
|
||
copyImage (url) {
|
||
BDFDB.LibraryRequires.request(url, {encoding: null}, (error, response, buffer) => {
|
||
if (error) BDFDB.NotificationUtils.toast(this.labels.toast_copyimage_failed, {type: "error"});
|
||
else if (buffer) {
|
||
if (BDFDB.LibraryRequires.process.platform === "win32" || BDFDB.LibraryRequires.process.platform === "darwin") {
|
||
BDFDB.LibraryRequires.electron.clipboard.write({image: BDFDB.LibraryRequires.electron.nativeImage.createFromBuffer(buffer)});
|
||
}
|
||
else {
|
||
let file = BDFDB.LibraryRequires.path.join(BDFDB.LibraryRequires.process.env.HOME, "imageutilstempimg.png");
|
||
BDFDB.LibraryRequires.fs.writeFileSync(file, buffer, {encoding: null});
|
||
BDFDB.LibraryRequires.electron.clipboard.write({image: file});
|
||
BDFDB.LibraryRequires.fs.unlinkSync(file);
|
||
}
|
||
BDFDB.NotificationUtils.toast(this.labels.toast_copyimage_success, {type: "success"});
|
||
}
|
||
});
|
||
}
|
||
|
||
getDownloadLocation () {
|
||
if (downloadsFolder && BDFDB.LibraryRequires.fs.existsSync(downloadsFolder)) return downloadsFolder;
|
||
let homePath = BDFDB.LibraryRequires.process.env.USERPROFILE || BDFDB.LibraryRequires.process.env.HOMEPATH || BDFDB.LibraryRequires.process.env.HOME;
|
||
let downloadPath = homePath && BDFDB.LibraryRequires.path.join(homePath, "Downloads");
|
||
if (downloadPath && BDFDB.LibraryRequires.fs.existsSync(downloadPath)) return downloadsFolder = downloadPath;
|
||
return downloadsFolder = BDFDB.BDUtils.getPluginsFolder();
|
||
}
|
||
|
||
getFileName (path, fileName, extension, i) {
|
||
let wholePath = BDFDB.LibraryRequires.path.join(path, i ? `${fileName} (${i}).${extension}` : `${fileName}.${extension}`);
|
||
if (BDFDB.LibraryRequires.fs.existsSync(wholePath)) return this.getFileName(path, fileName, extension, i + 1);
|
||
else return wholePath;
|
||
}
|
||
|
||
getMessageGroupOfImage (src) {
|
||
if (src && settings.enableGallery) for (let message of document.querySelectorAll(BDFDB.dotCN.message)) for (let img of message.querySelectorAll(BDFDB.dotCNS.imagewrapper + "img")) if (this.isSameImage(src, img)) {
|
||
let previousSiblings = [], nextSiblings = [];
|
||
let previousSibling = message.previousSibling, nextSibling = message.nextSibling;
|
||
if (!BDFDB.DOMUtils.containsClass(message, BDFDB.disCN.messagegroupstart)) while (previousSibling) {
|
||
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;
|
||
}
|
||
else nextSibling = null;
|
||
}
|
||
return [].concat(previousSiblings.reverse(), message, nextSiblings).filter(n => n && BDFDB.DOMUtils.containsClass(n, BDFDB.disCN.message));
|
||
}
|
||
return [];
|
||
}
|
||
|
||
getSiblingsAndPosition (url, messages) {
|
||
let images = messages.map(n => Array.from(n.querySelectorAll(BDFDB.dotCNS.imagewrapper + "img"))).flat().filter(img => !BDFDB.DOMUtils.getParent(BDFDB.dotCN.spoilerhidden, img));
|
||
let next, previous, index = 0, amount = images.length;
|
||
for (let i = 0; i < amount; i++) if (this.isSameImage(url, images[i])) {
|
||
index = i;
|
||
previous = images[i-1];
|
||
next = images[i+1];
|
||
break;
|
||
}
|
||
return {next, previous, index, amount};
|
||
}
|
||
|
||
isSameImage (src, img) {
|
||
return img.src && (Node.prototype.isPrototypeOf(src) && img == src || !Node.prototype.isPrototypeOf(src) && this.getImageSrc(img) == this.getImageSrc(src));
|
||
}
|
||
|
||
getImageSrc (img) {
|
||
if (!img) return null;
|
||
return (typeof img == "string" ? img : (img.src || (img.querySelector("canvas") ? img.querySelector("canvas").src : ""))).split("?width=")[0];
|
||
}
|
||
|
||
createImageWrapper (instance, imgRef, type, svgIcon) {
|
||
return BDFDB.ReactUtils.createElement("div", {
|
||
className: BDFDB.disCNS._imageutilitiessibling + BDFDB.disCN[`_imageutilities${type}`],
|
||
onClick: _ => {this.switchImages(instance, type);},
|
||
children: [
|
||
imgRef,
|
||
BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.SvgIcon, {
|
||
className: BDFDB.disCNS._imageutilitiesswitchicon + BDFDB.disCN.svgicon,
|
||
name: svgIcon
|
||
})
|
||
]
|
||
});
|
||
}
|
||
|
||
loadImage (instance, img, type) {
|
||
let imageThrowaway = document.createElement("img");
|
||
let src = this.getImageSrc(img);
|
||
imageThrowaway.src = src;
|
||
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;
|
||
instance[type + "Img"] = img;
|
||
instance[type + "Ref"] = BDFDB.ReactUtils.createElement(BDFDB.LibraryComponents.LazyImage, {
|
||
src: src,
|
||
height: imageThrowaway.naturalHeight,
|
||
width: imageThrowaway.naturalWidth,
|
||
maxHeight: newHeight,
|
||
maxWidth: newWidth,
|
||
});
|
||
BDFDB.ReactUtils.forceUpdate(instance);
|
||
};
|
||
}
|
||
|
||
switchImages (instance, type) {
|
||
let img = instance[type + "Img"];
|
||
let imgRef = instance[type + "Ref"];
|
||
if (!img || !imgRef) return;
|
||
delete instance.previousRef;
|
||
delete instance.nextRef;
|
||
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;
|
||
BDFDB.ReactUtils.forceUpdate(instance);
|
||
}
|
||
|
||
cleanupListeners (type) {
|
||
if (!type) return;
|
||
for (let eventType of ["wheel", "keydown", "keyup"]) {
|
||
document.removeEventListener("wheel", document[`${eventType}ImageUtilities${type}Listener`]);
|
||
delete document[`${eventType}ImageUtilities${type}Listener`];
|
||
}
|
||
}
|
||
|
||
setLabelsByLanguage () {
|
||
switch (BDFDB.LanguageUtils.getLanguage().id) {
|
||
case "bg": // Bulgarian
|
||
return {
|
||
context_copyimage: "Копирай изображение",
|
||
context_copyimagelink: "Копирайте Image Link",
|
||
context_lenssize: "Размер на обектива",
|
||
context_reverseimagesearch: "Търсене на изображение с ...",
|
||
context_saveimageas: "Запазете изображението като ...",
|
||
context_viewimage: "Преглед на изображението",
|
||
submenu_disabled: "Всички инвалиди",
|
||
toast_copyimage_failed: "Копирането на изображението в клипборда не бе успешно",
|
||
toast_copyimage_success: "Копирано изображение в клипборда",
|
||
toast_copyimagelink_success: "Копирана връзка към изображение в клипборда",
|
||
toast_saveimage_failed: "Запазването на изображението в '{{path}}' не бе успешно",
|
||
toast_saveimage_success: "Запазено изображение в '{{path}}'"
|
||
};
|
||
case "da": // Danish
|
||
return {
|
||
context_copyimage: "Kopier billede",
|
||
context_copyimagelink: "Kopier billedlink",
|
||
context_lenssize: "Objektivstørrelse",
|
||
context_reverseimagesearch: "Søg i billede med ...",
|
||
context_saveimageas: "Gem billede som ...",
|
||
context_viewimage: "Se billede",
|
||
submenu_disabled: "Alle handicappede",
|
||
toast_copyimage_failed: "Kunne ikke kopiere billedet til udklipsholderen",
|
||
toast_copyimage_success: "Kopieret billede til udklipsholderen",
|
||
toast_copyimagelink_success: "Kopieret billedlink til udklipsholder",
|
||
toast_saveimage_failed: "Kunne ikke gemme billedet i '{{path}}'",
|
||
toast_saveimage_success: "Gemt billede i '{{path}}'"
|
||
};
|
||
case "de": // German
|
||
return {
|
||
context_copyimage: "Bild kopieren",
|
||
context_copyimagelink: "Bildadresse kopieren",
|
||
context_lenssize: "Linsengröße",
|
||
context_reverseimagesearch: "Bild suchen mit ...",
|
||
context_saveimageas: "Bild speichern unter ...",
|
||
context_viewimage: "Bild ansehen",
|
||
submenu_disabled: "Alle deaktiviert",
|
||
toast_copyimage_failed: "Bild konnte nicht in die Zwischenablage kopiert werden",
|
||
toast_copyimage_success: "Kopiertes Bild in die Zwischenablage",
|
||
toast_copyimagelink_success: "Bildadresse in die Zwischenablage kopieren",
|
||
toast_saveimage_failed: "Bild konnte nicht in '{{path}}' gespeichert werden",
|
||
toast_saveimage_success: "Bild wurde in '{{path}}' gespeichert"
|
||
};
|
||
case "el": // Greek
|
||
return {
|
||
context_copyimage: "Αντιγραφή εικόνας",
|
||
context_copyimagelink: "Αντιγραφή συνδέσμου εικόνας",
|
||
context_lenssize: "Μέγεθος φακού",
|
||
context_reverseimagesearch: "Αναζήτηση εικόνας με ...",
|
||
context_saveimageas: "Αποθήκευση εικόνας ως ...",
|
||
context_viewimage: "Προβολή εικόνας",
|
||
submenu_disabled: "Όλα τα άτομα με ειδικές ανάγκες",
|
||
toast_copyimage_failed: "Αποτυχία αντιγραφής εικόνας στο Πρόχειρο",
|
||
toast_copyimage_success: "Αντιγράφηκε η εικόνα στο Πρόχειρο",
|
||
toast_copyimagelink_success: "Αντιγράφηκε σύνδεσμος εικόνας στο Πρόχειρο",
|
||
toast_saveimage_failed: "Αποτυχία αποθήκευσης εικόνας στο '{{path}}'",
|
||
toast_saveimage_success: "Αποθηκευμένη εικόνα στο '{{path}}'"
|
||
};
|
||
case "es": // Spanish
|
||
return {
|
||
context_copyimage: "Copiar imagen",
|
||
context_copyimagelink: "Copiar enlace de imagen",
|
||
context_lenssize: "Tamaño de la lente",
|
||
context_reverseimagesearch: "Buscar imagen con ...",
|
||
context_saveimageas: "Guardar imagen como ...",
|
||
context_viewimage: "Ver imagen",
|
||
submenu_disabled: "Todos discapacitados",
|
||
toast_copyimage_failed: "No se pudo copiar la imagen al portapapeles",
|
||
toast_copyimage_success: "Imagen copiada al portapapeles",
|
||
toast_copyimagelink_success: "Enlace de imagen copiado al portapapeles",
|
||
toast_saveimage_failed: "No se pudo guardar la imagen en '{{path}}'",
|
||
toast_saveimage_success: "Imagen guardada en '{{path}}'"
|
||
};
|
||
case "fi": // Finnish
|
||
return {
|
||
context_copyimage: "Kopioi kuva",
|
||
context_copyimagelink: "Kopioi kuvalinkki",
|
||
context_lenssize: "Linssin koko",
|
||
context_reverseimagesearch: "Hae kuvaa ...",
|
||
context_saveimageas: "Tallenna kuva nimellä ...",
|
||
context_viewimage: "Näytä kuva",
|
||
submenu_disabled: "Kaikki vammaiset",
|
||
toast_copyimage_failed: "Kuvan kopioiminen leikepöydälle epäonnistui",
|
||
toast_copyimage_success: "Kopioitu kuva leikepöydälle",
|
||
toast_copyimagelink_success: "Kopioitu kuvalinkki leikepöydälle",
|
||
toast_saveimage_failed: "Kuvan tallentaminen kohteeseen '{{path}}' epäonnistui",
|
||
toast_saveimage_success: "Tallennettu kuva kansioon '{{path}}'"
|
||
};
|
||
case "fr": // French
|
||
return {
|
||
context_copyimage: "Copier l'image",
|
||
context_copyimagelink: "Copier le lien de l'image",
|
||
context_lenssize: "Taille de l'objectif",
|
||
context_reverseimagesearch: "Rechercher une image avec ...",
|
||
context_saveimageas: "Enregistrer l'image sous ...",
|
||
context_viewimage: "Voir l'image",
|
||
submenu_disabled: "Tout désactivé",
|
||
toast_copyimage_failed: "Échec de la copie de l'image dans le presse-papiers",
|
||
toast_copyimage_success: "Image copiée dans le presse-papiers",
|
||
toast_copyimagelink_success: "Lien d'image copié vers le presse-papiers",
|
||
toast_saveimage_failed: "Échec de l'enregistrement de l'image dans '{{path}}'",
|
||
toast_saveimage_success: "Image enregistrée dans '{{path}}'"
|
||
};
|
||
case "hr": // Croatian
|
||
return {
|
||
context_copyimage: "Kopiraj sliku",
|
||
context_copyimagelink: "Kopiraj vezu slike",
|
||
context_lenssize: "Veličina leće",
|
||
context_reverseimagesearch: "Traži sliku pomoću ...",
|
||
context_saveimageas: "Spremi sliku kao ...",
|
||
context_viewimage: "Pogledati sliku",
|
||
submenu_disabled: "Svi invalidi",
|
||
toast_copyimage_failed: "Kopiranje slike u međuspremnik nije uspjelo",
|
||
toast_copyimage_success: "Kopirana slika u međuspremnik",
|
||
toast_copyimagelink_success: "Veza slike kopirana je u međuspremnik",
|
||
toast_saveimage_failed: "Spremanje slike u '{{path}}' nije uspjelo",
|
||
toast_saveimage_success: "Spremljena slika na '{{path}}'"
|
||
};
|
||
case "hu": // Hungarian
|
||
return {
|
||
context_copyimage: "Képmásolat",
|
||
context_copyimagelink: "Képlink másolása",
|
||
context_lenssize: "Lencse mérete",
|
||
context_reverseimagesearch: "Kép keresése a következővel:",
|
||
context_saveimageas: "Kép mentése másként ...",
|
||
context_viewimage: "Kép megtekintése",
|
||
submenu_disabled: "Minden fogyatékkal él",
|
||
toast_copyimage_failed: "Nem sikerült másolni a képet a vágólapra",
|
||
toast_copyimage_success: "Kép másolása a vágólapra",
|
||
toast_copyimagelink_success: "Képlink linkre másolva a vágólapra",
|
||
toast_saveimage_failed: "Nem sikerült menteni a képet a (z) '{{path}}” mappába",
|
||
toast_saveimage_success: "Mentett kép itt: '{{path}}”"
|
||
};
|
||
case "it": // Italian
|
||
return {
|
||
context_copyimage: "Copia l'immagine",
|
||
context_copyimagelink: "Copia link immagine",
|
||
context_lenssize: "Dimensione della lente",
|
||
context_reverseimagesearch: "Cerca immagine con ...",
|
||
context_saveimageas: "Salva l'immagine come ...",
|
||
context_viewimage: "Guarda l'immagine",
|
||
submenu_disabled: "Tutti disabilitati",
|
||
toast_copyimage_failed: "Impossibile copiare l'immagine negli Appunti",
|
||
toast_copyimage_success: "Immagine copiata negli Appunti",
|
||
toast_copyimagelink_success: "Collegamento immagine copiato negli Appunti",
|
||
toast_saveimage_failed: "Impossibile salvare l'immagine in '{{path}}'",
|
||
toast_saveimage_success: "Immagine salvata in '{{path}}'"
|
||
};
|
||
case "ja": // Japanese
|
||
return {
|
||
context_copyimage: "画像をコピーする",
|
||
context_copyimagelink: "画像リンクをコピー",
|
||
context_lenssize: "レンズサイズ",
|
||
context_reverseimagesearch: "で画像を検索...",
|
||
context_saveimageas: "画像を保存します ...",
|
||
context_viewimage: "画像を見る",
|
||
submenu_disabled: "すべて無効",
|
||
toast_copyimage_failed: "画像をクリップボードにコピーできませんでした",
|
||
toast_copyimage_success: "画像をクリップボードにコピー",
|
||
toast_copyimagelink_success: "クリップボードへのコピーされた画像リンク",
|
||
toast_saveimage_failed: "'{{path}}'に画像を保存できませんでした",
|
||
toast_saveimage_success: "'{{path}}'に保存された画像"
|
||
};
|
||
case "ko": // Korean
|
||
return {
|
||
context_copyimage: "복사 이미지",
|
||
context_copyimagelink: "이미지 링크 복사",
|
||
context_lenssize: "렌즈 크기",
|
||
context_reverseimagesearch: "이미지 검색 ...",
|
||
context_saveimageas: "다른 이름으로 이미지 저장 ...",
|
||
context_viewimage: "이미지보기",
|
||
submenu_disabled: "모두 비활성화 됨",
|
||
toast_copyimage_failed: "이미지를 클립 보드로 복사하지 못했습니다.",
|
||
toast_copyimage_success: "클립 보드에 복사 된 이미지",
|
||
toast_copyimagelink_success: "클립 보드에 복사 된 이미지 링크",
|
||
toast_saveimage_failed: "'{{path}}'에 이미지를 저장하지 못했습니다.",
|
||
toast_saveimage_success: "'{{path}}'에 저장된 이미지"
|
||
};
|
||
case "lt": // Lithuanian
|
||
return {
|
||
context_copyimage: "Kopijuoti paveiksliuką",
|
||
context_copyimagelink: "Kopijuoti vaizdo nuorodą",
|
||
context_lenssize: "Objektyvo dydis",
|
||
context_reverseimagesearch: "Ieškoti vaizde su ...",
|
||
context_saveimageas: "Išsaugoti paveikslėlį kaip ...",
|
||
context_viewimage: "Peržiūrėti Nuotrauka",
|
||
submenu_disabled: "Visi neįgalūs",
|
||
toast_copyimage_failed: "Nepavyko nukopijuoti vaizdo į iškarpinę",
|
||
toast_copyimage_success: "Nukopijuotas vaizdas į mainų sritį",
|
||
toast_copyimagelink_success: "Nukopijuota vaizdo nuoroda į iškarpinę",
|
||
toast_saveimage_failed: "Nepavyko išsaugoti vaizdo '{{path}}'",
|
||
toast_saveimage_success: "Išsaugotas vaizdas '{{path}}'"
|
||
};
|
||
case "nl": // Dutch
|
||
return {
|
||
context_copyimage: "Kopieer afbeelding",
|
||
context_copyimagelink: "Kopieer afbeeldingslink",
|
||
context_lenssize: "Lens Maat",
|
||
context_reverseimagesearch: "Zoek afbeelding met ...",
|
||
context_saveimageas: "Sla afbeelding op als ...",
|
||
context_viewimage: "Bekijk afbeelding",
|
||
submenu_disabled: "Allemaal uitgeschakeld",
|
||
toast_copyimage_failed: "Het kopiëren van de afbeelding naar het klembord is mislukt",
|
||
toast_copyimage_success: "Gekopieerde afbeelding naar klembord",
|
||
toast_copyimagelink_success: "Gekopieerde afbeeldingslink naar het klembord",
|
||
toast_saveimage_failed: "Kan afbeelding niet opslaan in '{{path}}'",
|
||
toast_saveimage_success: "Opgeslagen afbeelding in '{{path}}'"
|
||
};
|
||
case "no": // Norwegian
|
||
return {
|
||
context_copyimage: "Kopier bilde",
|
||
context_copyimagelink: "Kopier bildelink",
|
||
context_lenssize: "Linsestørrelse",
|
||
context_reverseimagesearch: "Søk i bilde med ...",
|
||
context_saveimageas: "Lagre bildet som ...",
|
||
context_viewimage: "Vis bilde",
|
||
submenu_disabled: "Alle funksjonshemmede",
|
||
toast_copyimage_failed: "Kunne ikke kopiere bildet til utklippstavlen",
|
||
toast_copyimage_success: "Kopiert bilde til utklippstavlen",
|
||
toast_copyimagelink_success: "Kopiert bildekobling til utklippstavlen",
|
||
toast_saveimage_failed: "Kunne ikke lagre bildet i '{{path}}'",
|
||
toast_saveimage_success: "Lagret bilde i '{{path}}'"
|
||
};
|
||
case "pl": // Polish
|
||
return {
|
||
context_copyimage: "Skopiuj obraz",
|
||
context_copyimagelink: "Kopiuj łącze do obrazu",
|
||
context_lenssize: "Rozmiar soczewki",
|
||
context_reverseimagesearch: "Szukaj obrazu za pomocą ...",
|
||
context_saveimageas: "Zapisz obraz jako ...",
|
||
context_viewimage: "Zobacz obrazek",
|
||
submenu_disabled: "Wszystkie wyłączone",
|
||
toast_copyimage_failed: "Nie udało się skopiować obrazu do schowka",
|
||
toast_copyimage_success: "Skopiowany obraz do schowka",
|
||
toast_copyimagelink_success: "Link do skopiowanego obrazu do schowka",
|
||
toast_saveimage_failed: "Nie udało się zapisać obrazu w '{{path}}”",
|
||
toast_saveimage_success: "Zapisany obraz w '{{path}}”"
|
||
};
|
||
case "pt-BR": // Portuguese (Brazil)
|
||
return {
|
||
context_copyimage: "Copiar imagem",
|
||
context_copyimagelink: "Copiar link da imagem",
|
||
context_lenssize: "Tamanho da lente",
|
||
context_reverseimagesearch: "Pesquisar imagem com ...",
|
||
context_saveimageas: "Salvar imagem como ...",
|
||
context_viewimage: "Ver imagem",
|
||
submenu_disabled: "Todos desativados",
|
||
toast_copyimage_failed: "Falha ao copiar a imagem para a área de transferência",
|
||
toast_copyimage_success: "Imagem copiada para a área de transferência",
|
||
toast_copyimagelink_success: "Link da imagem copiada para a área de transferência",
|
||
toast_saveimage_failed: "Falha ao salvar a imagem em '{{path}}'",
|
||
toast_saveimage_success: "Imagem salva em '{{path}}'"
|
||
};
|
||
case "ro": // Romanian
|
||
return {
|
||
context_copyimage: "Copiază imaginea",
|
||
context_copyimagelink: "Copiați linkul de imagine",
|
||
context_lenssize: "Dimensiunea obiectivului",
|
||
context_reverseimagesearch: "Căutați imaginea cu ...",
|
||
context_saveimageas: "Salveaza imaginea ca ...",
|
||
context_viewimage: "Vezi imaginea",
|
||
submenu_disabled: "Toate sunt dezactivate",
|
||
toast_copyimage_failed: "Copierea imaginii în clipboard nu a reușit",
|
||
toast_copyimage_success: "Copiat imaginea în Clipboard",
|
||
toast_copyimagelink_success: "Link copiat pentru imagine în Clipboard",
|
||
toast_saveimage_failed: "Salvarea imaginii în '{{path}}” nu a reușit",
|
||
toast_saveimage_success: "Imagine salvată în '{{path}}”"
|
||
};
|
||
case "ru": // Russian
|
||
return {
|
||
context_copyimage: "Копировать изображение",
|
||
context_copyimagelink: "Копировать ссылку на изображение",
|
||
context_lenssize: "Размер линзы",
|
||
context_reverseimagesearch: "Искать изображение с помощью ...",
|
||
context_saveimageas: "Сохранить изображение как ...",
|
||
context_viewimage: "Просмотр изображения",
|
||
submenu_disabled: "Все отключены",
|
||
toast_copyimage_failed: "Не удалось скопировать изображение в буфер обмена",
|
||
toast_copyimage_success: "Изображение скопировано в буфер обмена",
|
||
toast_copyimagelink_success: "Ссылка на скопированное изображение в буфер обмена",
|
||
toast_saveimage_failed: "Не удалось сохранить изображение в '{{path}}'",
|
||
toast_saveimage_success: "Сохраненное изображение в '{{path}}'"
|
||
};
|
||
case "sv": // Swedish
|
||
return {
|
||
context_copyimage: "Kopiera bild",
|
||
context_copyimagelink: "Kopiera bildlänk",
|
||
context_lenssize: "Linsstorlek",
|
||
context_reverseimagesearch: "Sök bild med ...",
|
||
context_saveimageas: "Spara bild som ...",
|
||
context_viewimage: "Se bild",
|
||
submenu_disabled: "Alla funktionshindrade",
|
||
toast_copyimage_failed: "Det gick inte att kopiera bilden till Urklipp",
|
||
toast_copyimage_success: "Kopierad bild till Urklipp",
|
||
toast_copyimagelink_success: "Kopierad bildlänk till Urklipp",
|
||
toast_saveimage_failed: "Det gick inte att spara bilden i '{{path}}'",
|
||
toast_saveimage_success: "Sparad bild i '{{path}}'"
|
||
};
|
||
case "th": // Thai
|
||
return {
|
||
context_copyimage: "คัดลอกรูปภาพ",
|
||
context_copyimagelink: "คัดลอกลิงค์รูปภาพ",
|
||
context_lenssize: "ขนาดเลนส์",
|
||
context_reverseimagesearch: "ค้นหาภาพด้วย ...",
|
||
context_saveimageas: "บันทึกภาพเป็น ...",
|
||
context_viewimage: "ดูภาพ",
|
||
submenu_disabled: "ปิดใช้งานทั้งหมด",
|
||
toast_copyimage_failed: "คัดลอกรูปภาพไปยังคลิปบอร์ดไม่สำเร็จ",
|
||
toast_copyimage_success: "คัดลอกรูปภาพไปยังคลิปบอร์ดแล้ว",
|
||
toast_copyimagelink_success: "ลิงก์รูปภาพที่คัดลอกไปยังคลิปบอร์ด",
|
||
toast_saveimage_failed: "บันทึกภาพใน '{{path}}' ไม่สำเร็จ",
|
||
toast_saveimage_success: "ภาพที่บันทึกไว้ใน '{{path}}'"
|
||
};
|
||
case "tr": // Turkish
|
||
return {
|
||
context_copyimage: "Resmi kopyala",
|
||
context_copyimagelink: "Resim Bağlantısını Kopyala",
|
||
context_lenssize: "Lens Boyutu",
|
||
context_reverseimagesearch: "Şununla Resim Ara ...",
|
||
context_saveimageas: "Resmi farklı kaydet ...",
|
||
context_viewimage: "Görseli göster",
|
||
submenu_disabled: "Hepsi devre dışı",
|
||
toast_copyimage_failed: "Görüntü Panoya kopyalanamadı",
|
||
toast_copyimage_success: "Görüntü panoya kopyalandı",
|
||
toast_copyimagelink_success: "Görüntü bağlantısı panoya kopyalandı",
|
||
toast_saveimage_failed: "Resim '{{yol}}' içine kaydedilemedi",
|
||
toast_saveimage_success: "Resim '{{yol}}' içine kaydedildi"
|
||
};
|
||
case "uk": // Ukrainian
|
||
return {
|
||
context_copyimage: "Скопіювати зображення",
|
||
context_copyimagelink: "Скопіювати посилання на зображення",
|
||
context_lenssize: "Розмір лінзи",
|
||
context_reverseimagesearch: "Шукати зображення за допомогою ...",
|
||
context_saveimageas: "Зберегти зображення як ...",
|
||
context_viewimage: "Переглянути зображення",
|
||
submenu_disabled: "Всі інваліди",
|
||
toast_copyimage_failed: "Не вдалося скопіювати зображення в буфер обміну",
|
||
toast_copyimage_success: "Скопійовано зображення в буфер обміну",
|
||
toast_copyimagelink_success: "Скопійовано посилання на зображення в буфер обміну",
|
||
toast_saveimage_failed: "Не вдалося зберегти зображення у '{{path}}'",
|
||
toast_saveimage_success: "Збережене зображення у '{{path}}'"
|
||
};
|
||
case "vi": // Vietnamese
|
||
return {
|
||
context_copyimage: "Sao chép hình ảnh",
|
||
context_copyimagelink: "Sao chép liên kết hình ảnh",
|
||
context_lenssize: "Kích thước ống kính",
|
||
context_reverseimagesearch: "Tìm kiếm Hình ảnh bằng ...",
|
||
context_saveimageas: "Lưu ảnh dưới dạng ...",
|
||
context_viewimage: "Xem hình ảnh",
|
||
submenu_disabled: "Tất cả đã bị vô hiệu hóa",
|
||
toast_copyimage_failed: "Không sao chép được hình ảnh vào Clipboard",
|
||
toast_copyimage_success: "Đã sao chép hình ảnh vào Clipboard",
|
||
toast_copyimagelink_success: "Liên kết hình ảnh được sao chép vào Clipboard",
|
||
toast_saveimage_failed: "Không lưu được Hình ảnh trong '{{path}}'",
|
||
toast_saveimage_success: "Hình ảnh đã Lưu trong '{{path}}'"
|
||
};
|
||
case "zh": // Chinese
|
||
return {
|
||
context_copyimage: "复制图片",
|
||
context_copyimagelink: "复制图像链接",
|
||
context_lenssize: "镜片尺寸",
|
||
context_reverseimagesearch: "用...搜索图像",
|
||
context_saveimageas: "将图像另存为...",
|
||
context_viewimage: "看图",
|
||
submenu_disabled: "全部禁用",
|
||
toast_copyimage_failed: "无法将图像复制到剪贴板",
|
||
toast_copyimage_success: "复制到剪贴板的图像",
|
||
toast_copyimagelink_success: "复制的图像链接到剪贴板",
|
||
toast_saveimage_failed: "无法将图片保存到'{{path}}'中",
|
||
toast_saveimage_success: "已将图像保存在'{{path}}'中"
|
||
};
|
||
case "zh-TW": // Chinese (Traditional)
|
||
return {
|
||
context_copyimage: "複製圖片",
|
||
context_copyimagelink: "複製圖像鏈接",
|
||
context_lenssize: "鏡片尺寸",
|
||
context_reverseimagesearch: "用...搜索圖像",
|
||
context_saveimageas: "將圖像另存為...",
|
||
context_viewimage: "看圖",
|
||
submenu_disabled: "全部禁用",
|
||
toast_copyimage_failed: "無法將圖像複製到剪貼板",
|
||
toast_copyimage_success: "複製到剪貼板的圖像",
|
||
toast_copyimagelink_success: "複製的圖像鏈接到剪貼板",
|
||
toast_saveimage_failed: "無法將圖片保存到'{{path}}'中",
|
||
toast_saveimage_success: "已將圖像保存在'{{path}}'中"
|
||
};
|
||
default: // English
|
||
return {
|
||
context_copyimage: "Copy Image",
|
||
context_copyimagelink: "Copy Image Link",
|
||
context_lenssize: "Lens Size",
|
||
context_reverseimagesearch: "Search Image with ...",
|
||
context_saveimageas: "Save Image as ...",
|
||
context_viewimage: "View Image",
|
||
submenu_disabled: "All disabled",
|
||
toast_copyimage_failed: "Failed to copy image to Clipboard",
|
||
toast_copyimage_success: "Copied image to Clipboard",
|
||
toast_copyimagelink_success: "Copied image link to Clipboard",
|
||
toast_saveimage_failed: "Failed to save Image in '{{path}}'",
|
||
toast_saveimage_success: "Saved Image in '{{path}}'"
|
||
};
|
||
}
|
||
}
|
||
};
|
||
})(window.BDFDB_Global.PluginUtils.buildPlugin(config));
|
||
})();
|