diff --git a/src/plugins/favGifSearch.tsx b/src/plugins/favGifSearch.tsx new file mode 100644 index 00000000..db575a03 --- /dev/null +++ b/src/plugins/favGifSearch.tsx @@ -0,0 +1,241 @@ +/* + * Vencord, a modification for Discord's desktop app + * Copyright (c) 2023 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 . +*/ + +import { definePluginSettings } from "@api/Settings"; +import ErrorBoundary from "@components/ErrorBoundary"; +import { Devs } from "@utils/constants"; +import definePlugin, { OptionType } from "@utils/types"; +import { findByPropsLazy } from "@webpack"; +import { useCallback, useEffect, useRef, useState } from "@webpack/common"; + +interface SearchBarComponentProps { + ref?: React.MutableRefObject; + autoFocus: boolean; + className: string; + size: string; + onChange: (query: string) => void; + onClear: () => void; + query: string; + placeholder: string; +} + +type TSearchBarComponent = + React.FC & { Sizes: Record<"SMALL" | "MEDIUM" | "LARGE", string>; }; + +interface Gif { + format: number; + src: string; + width: number; + height: number; + order: number; + url: string; +} + +interface Instance { + dead?: boolean; + state: { + resultType?: string; + }; + props: { + favCopy: Gif[], + + favorites: Gif[], + }, + forceUpdate: () => void; +} + + +const containerClasses: { searchBar: string; } = findByPropsLazy("searchBar", "searchHeader", "gutterSize"); + +export const settings = definePluginSettings({ + searchOption: { + type: OptionType.SELECT, + description: "The part of the url you want to search", + options: [ + { + label: "Entire Url", + value: "url" + }, + { + label: "Path Only (/somegif.gif)", + value: "path" + }, + { + label: "Host & Path (tenor.com somgif.gif)", + value: "hostandpath", + default: true + } + ] as const + } +}); + +export default definePlugin({ + name: "FavoriteGifSearch", + authors: [Devs.Aria], + description: "Adds a search bar for favorite gifs", + + patches: [ + { + find: "renderCategoryExtras", + replacement: [ + { + // https://regex101.com/r/4uHtTE/1 + // ($1 renderHeaderContent=function { ... switch (x) ... case FAVORITES:return) ($2) ($3 case default:return r.jsx(($), {...props})) + match: /(renderHeaderContent=function.{1,150}FAVORITES:return)(.{1,150};)(case.{1,200}default:return\(0,\i\.jsx\)\((?\i\.\i))/, + replace: "$1 this.state.resultType === \"Favorites\" ? $self.renderSearchBar(this, $) : $2; $3" + }, + { + // to persist filtered favorites when component re-renders. + // when resizing the window the component rerenders and we loose the filtered favorites and have to type in the search bar to get them again + match: /(,suggestions:\i,favorites:)(\i),/, + replace: "$1$self.getFav($2),favCopy:$2," + } + + ] + } + ], + + settings, + + getTargetString, + + instance: null as Instance | null, + renderSearchBar(instance: Instance, SearchBarComponent: TSearchBarComponent) { + this.instance = instance; + return ( + + + + ); + }, + + getFav(favorites: Gif[]) { + if (!this.instance || this.instance.dead) return favorites; + const { favorites: filteredFavorites } = this.instance.props; + + return filteredFavorites != null && filteredFavorites?.length !== favorites.length ? filteredFavorites : favorites; + + } +}); + + +function SearchBar({ instance, SearchBarComponent }: { instance: Instance; SearchBarComponent: TSearchBarComponent; }) { + const [query, setQuery] = useState(""); + const ref = useRef<{ containerRef?: React.MutableRefObject; } | null>(null); + + const onChange = useCallback((searchQuery: string) => { + setQuery(searchQuery); + const { props } = instance; + + // return early + if (searchQuery === "") { + props.favorites = props.favCopy; + instance.forceUpdate(); + return; + } + + + // scroll back to top + ref.current?.containerRef?.current + .closest("#gif-picker-tab-panel") + ?.querySelector("[class|=\"content\"]") + ?.firstElementChild?.scrollTo(0, 0); + + + const result = + props.favCopy + .map(gif => ({ + score: fuzzySearch(searchQuery.toLowerCase(), getTargetString(gif.url ?? gif.src).replace(/(%20|[_-])/g, " ").toLowerCase()), + gif, + })) + .filter(m => m.score != null) as { score: number; gif: Gif; }[]; + + result.sort((a, b) => b.score - a.score); + props.favorites = result.map(e => e.gif); + + instance.forceUpdate(); + }, [instance.state]); + + useEffect(() => { + return () => { + instance.dead = true; + }; + }, []); + + return ( + { + setQuery(""); + if (instance.props.favCopy != null) { + instance.props.favorites = instance.props.favCopy; + instance.forceUpdate(); + } + }} + query={query} + placeholder="Search Favorite Gifs" + /> + ); +} + + + +export function getTargetString(urlStr: string) { + const url = new URL(urlStr); + switch (settings.store.searchOption) { + case "url": + return url.href; + case "path": + if (url.host === "media.discordapp.net" || url.host === "tenor.com") + // /attachments/899763415290097664/1095711736461537381/attachment-1.gif -> attachment-1.gif + // /view/some-gif-hi-24248063 -> some-gif-hi-24248063 + return url.pathname.split("/").at(-1) ?? url.pathname; + return url.pathname; + case "hostandpath": + if (url.host === "media.discordapp.net" || url.host === "tenor.com") + return `${url.host} ${url.pathname.split("/").at(-1) ?? url.pathname}`; + return `${url.host} ${url.pathname}`; + + default: + return ""; + } +} + +function fuzzySearch(searchQuery: string, searchString: string) { + let searchIndex = 0; + let score = 0; + + for (let i = 0; i < searchString.length; i++) { + if (searchString[i] === searchQuery[searchIndex]) { + score++; + searchIndex++; + } else { + score--; + } + + if (searchIndex === searchQuery.length) { + return score; + } + } + + return null; +}