mirror of
https://github.com/System-End/Vencord.git
synced 2026-04-19 19:45:09 +00:00
210 lines
7.8 KiB
TypeScript
210 lines
7.8 KiB
TypeScript
/*
|
|
* 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 <https://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
import { _resolveReady, filters, findByCodeLazy, findByPropsLazy, findLazy, mapMangledModuleLazy, waitFor } from "@webpack";
|
|
import type { Channel } from "discord-types/general";
|
|
|
|
import type * as t from "./types/utils";
|
|
|
|
export let FluxDispatcher: t.FluxDispatcher;
|
|
waitFor(["dispatch", "subscribe"], m => {
|
|
FluxDispatcher = m;
|
|
// Non import access to avoid circular dependency
|
|
Vencord.Plugins.subscribeAllPluginsFluxEvents(m);
|
|
|
|
const cb = () => {
|
|
m.unsubscribe("CONNECTION_OPEN", cb);
|
|
_resolveReady();
|
|
};
|
|
m.subscribe("CONNECTION_OPEN", cb);
|
|
});
|
|
|
|
export let ComponentDispatch: any;
|
|
waitFor(["dispatchToLastSubscribed"], m => ComponentDispatch = m);
|
|
|
|
export const Constants: t.Constants = mapMangledModuleLazy('ME:"/users/@me"', {
|
|
Endpoints: filters.byProps("USER", "ME"),
|
|
UserFlags: filters.byProps("STAFF", "SPAMMER"),
|
|
FriendsSections: m => m.PENDING === "PENDING" && m.ADD_FRIEND
|
|
});
|
|
|
|
export const RestAPI: t.RestAPI = findLazy(m => typeof m === "object" && m.del && m.put);
|
|
export const moment: typeof import("moment") = findByPropsLazy("parseTwoDigitYear");
|
|
|
|
export const hljs: typeof import("highlight.js").default = findByPropsLazy("highlight", "registerLanguage");
|
|
|
|
export const { match, P }: Pick<typeof import("ts-pattern"), "match" | "P"> = mapMangledModuleLazy("@ts-pattern/matcher", {
|
|
match: filters.byCode("return new"),
|
|
P: filters.byProps("when")
|
|
});
|
|
|
|
export const lodash: typeof import("lodash") = findByPropsLazy("debounce", "cloneDeep");
|
|
|
|
export const i18n = mapMangledModuleLazy('defaultLocale:"en-US"', {
|
|
t: m => m?.[Symbol.toStringTag] === "IntlMessagesProxy",
|
|
intl: m => m != null && Object.getPrototypeOf(m)?.withFormatters != null
|
|
}, true);
|
|
|
|
export let SnowflakeUtils: t.SnowflakeUtils;
|
|
waitFor(["fromTimestamp", "extractTimestamp"], m => SnowflakeUtils = m);
|
|
|
|
export let Parser: t.Parser;
|
|
waitFor("parseTopic", m => Parser = m);
|
|
export let Alerts: t.Alerts;
|
|
waitFor(["show", "close"], m => Alerts = m);
|
|
|
|
const ToastType = {
|
|
MESSAGE: "message",
|
|
SUCCESS: "success",
|
|
FAILURE: "failure",
|
|
CUSTOM: "custom",
|
|
CLIP: "clip",
|
|
LINK: "link",
|
|
FORWARD: "forward",
|
|
BOOKMARK: "bookmark",
|
|
CLOCK: "clock"
|
|
};
|
|
const ToastPosition = {
|
|
TOP: 0,
|
|
BOTTOM: 1
|
|
};
|
|
|
|
export interface ToastData {
|
|
message: string,
|
|
id: string,
|
|
/**
|
|
* Toasts.Type
|
|
*/
|
|
type: string,
|
|
options?: ToastOptions;
|
|
}
|
|
|
|
export interface ToastOptions {
|
|
/**
|
|
* Toasts.Position
|
|
*/
|
|
position?: number;
|
|
component?: React.ReactNode,
|
|
duration?: number;
|
|
}
|
|
|
|
export const Toasts = {
|
|
Type: ToastType,
|
|
Position: ToastPosition,
|
|
// what's less likely than getting 0 from Math.random()? Getting it twice in a row
|
|
genId: () => (Math.random() || Math.random()).toString(36).slice(2),
|
|
|
|
// hack to merge with the following interface, dunno if there's a better way
|
|
...{} as {
|
|
show(data: ToastData): void;
|
|
pop(): void;
|
|
create(message: string, type: string, options?: ToastOptions): ToastData;
|
|
}
|
|
};
|
|
|
|
// This is the same module but this is easier
|
|
waitFor("showToast", m => {
|
|
Toasts.show = m.showToast;
|
|
Toasts.pop = m.popToast;
|
|
Toasts.create = m.createToast;
|
|
});
|
|
|
|
|
|
/**
|
|
* Show a simple toast. If you need more options, use Toasts.show manually
|
|
*/
|
|
export function showToast(message: string, type = ToastType.MESSAGE, options?: ToastOptions) {
|
|
Toasts.show(Toasts.create(message, type, options));
|
|
}
|
|
|
|
export const UserUtils = {
|
|
getUser: findByCodeLazy(".USER(")
|
|
};
|
|
|
|
export const UploadManager = findByPropsLazy("clearAll", "addFile");
|
|
export const UploadHandler = {
|
|
promptToUpload: findByCodeLazy("#{intl::ATTACHMENT_TOO_MANY_ERROR_TITLE}") as (files: File[], channel: Channel, draftType: Number) => void
|
|
};
|
|
|
|
export const ApplicationAssetUtils = mapMangledModuleLazy("getAssetImage: size must === [", {
|
|
fetchAssetIds: filters.byCode('.startsWith("http:")', ".dispatch({"),
|
|
getAssetFromImageURL: filters.byCode("].serialize(", ',":"'),
|
|
getAssetImage: filters.byCode("getAssetImage: size must === ["),
|
|
getAssets: filters.byCode(".assets")
|
|
});
|
|
|
|
export const NavigationRouter: t.NavigationRouter = mapMangledModuleLazy("Transitioning to ", {
|
|
transitionTo: filters.byCode("transitionTo -"),
|
|
transitionToGuild: filters.byCode("transitionToGuild -"),
|
|
back: filters.byCode("goBack()"),
|
|
forward: filters.byCode("goForward()"),
|
|
});
|
|
export const ChannelRouter: t.ChannelRouter = mapMangledModuleLazy('"Thread must have a parent ID."', {
|
|
transitionToChannel: filters.byCode(".preload"),
|
|
transitionToThread: filters.byCode('"Thread must have a parent ID."')
|
|
});
|
|
|
|
export let SettingsRouter: any;
|
|
waitFor(["open", "saveAccountChanges"], m => SettingsRouter = m);
|
|
|
|
export const PermissionsBits: t.PermissionsBits = findLazy(m => typeof m.ADMINISTRATOR === "bigint");
|
|
|
|
export const { zustandCreate } = mapMangledModuleLazy(["useSyncExternalStoreWithSelector:", "Object.assign"], {
|
|
zustandCreate: filters.byCode(/=>(\i)\?\i\(\1/)
|
|
});
|
|
|
|
export const { zustandPersist } = mapMangledModuleLazy(".onRehydrateStorage)?", {
|
|
zustandPersist: filters.byCode(/(\(\i,\i\))=>.+?\i\1/)
|
|
});
|
|
|
|
export const MessageActions = findByPropsLazy("editMessage", "sendMessage");
|
|
export const MessageCache = findByPropsLazy("clearCache", "_channelMessages");
|
|
export const UserProfileActions = findByPropsLazy("openUserProfileModal", "closeUserProfileModal");
|
|
export const InviteActions = findByPropsLazy("resolveInvite");
|
|
export const ChannelActionCreators = findByPropsLazy("openPrivateChannel");
|
|
|
|
export const IconUtils: t.IconUtils = findByPropsLazy("getGuildBannerURL", "getUserAvatarURL");
|
|
|
|
export const ExpressionPickerStore: t.ExpressionPickerStore = mapMangledModuleLazy("expression-picker-last-active-view", {
|
|
openExpressionPicker: filters.byCode(/setState\({activeView:(?:(?!null)\i),activeViewType:/),
|
|
closeExpressionPicker: filters.byCode("setState({activeView:null"),
|
|
toggleMultiExpressionPicker: filters.byCode(".EMOJI,"),
|
|
toggleExpressionPicker: filters.byCode(/getState\(\)\.activeView===\i\?\i\(\):\i\(/),
|
|
setExpressionPickerView: filters.byCode(/setState\({activeView:\i,lastActiveView:/),
|
|
setSearchQuery: filters.byCode("searchQuery:"),
|
|
useExpressionPickerStore: filters.byCode(/\(\i,\i=\i\)=>/)
|
|
});
|
|
|
|
export const PopoutActions: t.PopoutActions = mapMangledModuleLazy('type:"POPOUT_WINDOW_OPEN"', {
|
|
open: filters.byCode('type:"POPOUT_WINDOW_OPEN"'),
|
|
close: filters.byCode('type:"POPOUT_WINDOW_CLOSE"'),
|
|
setAlwaysOnTop: filters.byCode('type:"POPOUT_WINDOW_SET_ALWAYS_ON_TOP"'),
|
|
});
|
|
|
|
export const UsernameUtils: t.UsernameUtils = findByPropsLazy("useName", "getGlobalName");
|
|
export const DisplayProfileUtils: t.DisplayProfileUtils = mapMangledModuleLazy(/=\i\.getUserProfile\(\i\),\i=\i\.getGuildMemberProfile\(/, {
|
|
getDisplayProfile: filters.byCode(".getGuildMemberProfile("),
|
|
useDisplayProfile: filters.byCode(/\[\i\.\i,\i\.\i],\(\)=>/)
|
|
});
|
|
|
|
export const DateUtils: t.DateUtils = mapMangledModuleLazy("millisecondsInUnit:", {
|
|
calendarFormat: filters.byCode("sameElse"),
|
|
dateFormat: filters.byCode('":'),
|
|
isSameDay: filters.byCode("Math.abs(+"),
|
|
diffAsUnits: filters.byCode("days:0", "millisecondsInUnit")
|
|
});
|