aboutsummaryrefslogtreecommitdiff
path: root/src/plugins/reverseImageSearch
diff options
context:
space:
mode:
authorV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
committerV <vendicated@riseup.net>2023-09-24 16:02:18 +0200
commit30ac25607023752031aa98060cbf8a736109992d (patch)
tree79bb82b6634ef601db6c98e751275607ec54dbea /src/plugins/reverseImageSearch
parentd0e2a324717e600736a18b88fe89a21c640a406b (diff)
downloadVencord-30ac25607023752031aa98060cbf8a736109992d.tar.gz
Vencord-30ac25607023752031aa98060cbf8a736109992d.tar.bz2
Vencord-30ac25607023752031aa98060cbf8a736109992d.zip
migrate all plugins to folders
Diffstat (limited to 'src/plugins/reverseImageSearch')
-rw-r--r--src/plugins/reverseImageSearch/index.tsx120
1 files changed, 120 insertions, 0 deletions
diff --git a/src/plugins/reverseImageSearch/index.tsx b/src/plugins/reverseImageSearch/index.tsx
new file mode 100644
index 0000000..811e7ff
--- /dev/null
+++ b/src/plugins/reverseImageSearch/index.tsx
@@ -0,0 +1,120 @@
+/*
+ * Vencord, a modification for Discord's desktop app
+ * Copyright (c) 2022 Vendicated and contributors
+ *
+ * This program is free software: you can redistribute it and/or modify
+ * it under the terms of the GNU General Public License as published by
+ * the Free Software Foundation, either version 3 of the License, or
+ * (at your option) any later version.
+ *
+ * This program is distributed in the hope that it will be useful,
+ * but WITHOUT ANY WARRANTY; without even the implied warranty of
+ * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
+ * GNU General Public License for more details.
+ *
+ * You should have received a copy of the GNU General Public License
+ * along with this program. If not, see <https://www.gnu.org/licenses/>.
+*/
+
+import { addContextMenuPatch, findGroupChildrenByChildId, NavContextMenuPatchCallback, removeContextMenuPatch } from "@api/ContextMenu";
+import { Flex } from "@components/Flex";
+import { OpenExternalIcon } from "@components/Icons";
+import { Devs } from "@utils/constants";
+import definePlugin from "@utils/types";
+import { Menu } from "@webpack/common";
+
+const Engines = {
+ Google: "https://lens.google.com/uploadbyurl?url=",
+ Yandex: "https://yandex.com/images/search?rpt=imageview&url=",
+ SauceNAO: "https://saucenao.com/search.php?url=",
+ IQDB: "https://iqdb.org/?url=",
+ TinEye: "https://www.tineye.com/search?url=",
+ ImgOps: "https://imgops.com/start?url="
+} as const;
+
+function search(src: string, engine: string) {
+ open(engine + encodeURIComponent(src), "_blank");
+}
+
+const imageContextMenuPatch: NavContextMenuPatchCallback = (children, props) => () => {
+ if (!props) return;
+ const { reverseImageSearchType, itemHref, itemSrc } = props;
+
+ if (!reverseImageSearchType || reverseImageSearchType !== "img") return;
+
+ const src = itemHref ?? itemSrc;
+
+ const group = findGroupChildrenByChildId("copy-link", children);
+ if (group) {
+ group.push((
+ <Menu.MenuItem
+ label="Search Image"
+ key="search-image"
+ id="search-image"
+ >
+ {Object.keys(Engines).map((engine, i) => {
+ const key = "search-image-" + engine;
+ return (
+ <Menu.MenuItem
+ key={key}
+ id={key}
+ label={
+ <Flex style={{ alignItems: "center", gap: "0.5em" }}>
+ <img
+ style={{
+ borderRadius: i >= 3 // Do not round Google, Yandex & SauceNAO
+ ? "50%"
+ : void 0
+ }}
+ aria-hidden="true"
+ height={16}
+ width={16}
+ src={new URL("/favicon.ico", Engines[engine]).toString().replace("lens.", "")}
+ />
+ {engine}
+ </Flex>
+ }
+ action={() => search(src, Engines[engine])}
+ />
+ );
+ })}
+ <Menu.MenuItem
+ key="search-image-all"
+ id="search-image-all"
+ label={
+ <Flex style={{ alignItems: "center", gap: "0.5em" }}>
+ <OpenExternalIcon height={16} width={16} />
+ All
+ </Flex>
+ }
+ action={() => Object.values(Engines).forEach(e => search(src, e))}
+ />
+ </Menu.MenuItem>
+ ));
+ }
+};
+
+export default definePlugin({
+ name: "ReverseImageSearch",
+ description: "Adds ImageSearch to image context menus",
+ authors: [Devs.Ven, Devs.Nuckyz],
+ tags: ["ImageUtilities"],
+
+ patches: [
+ {
+ find: ".Messages.MESSAGE_ACTIONS_MENU_LABEL",
+ replacement: {
+ match: /favoriteableType:\i,(?<=(\i)\.getAttribute\("data-type"\).+?)/,
+ replace: (m, target) => `${m}reverseImageSearchType:${target}.getAttribute("data-role"),`
+ }
+ }
+ ],
+
+ start() {
+ addContextMenuPatch("message", imageContextMenuPatch);
+ },
+
+ stop() {
+ removeContextMenuPatch("message", imageContextMenuPatch);
+ }
+});