2022-10-21 23:17:06 +00:00
|
|
|
/*
|
|
|
|
* 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/>.
|
|
|
|
*/
|
|
|
|
|
2024-10-23 01:57:46 +00:00
|
|
|
import "./discord.css";
|
|
|
|
|
2023-05-23 01:47:09 +00:00
|
|
|
import { MessageObject } from "@api/MessageEvents";
|
2024-11-13 18:27:21 +00:00
|
|
|
import { ChannelStore, ComponentDispatch, Constants, FluxDispatcher, GuildStore, i18n, IconUtils, InviteActions, MessageActions, PrivateChannelsStore, RestAPI, SelectedChannelStore, SelectedGuildStore, UserProfileActions, UserProfileStore, UserSettingsActionCreators, UserUtils } from "@webpack/common";
|
2024-09-05 00:12:09 +00:00
|
|
|
import { Channel, Guild, Message, User } from "discord-types/general";
|
2024-10-24 11:10:29 +00:00
|
|
|
import { Except } from "type-fest";
|
2022-10-22 16:18:41 +00:00
|
|
|
|
2024-11-05 19:49:27 +00:00
|
|
|
import { runtimeHashMessageKey } from "./intlHash";
|
|
|
|
import { Logger } from "./Logger";
|
2024-10-24 11:10:29 +00:00
|
|
|
import { MediaModalItem, MediaModalProps, openMediaModal } from "./modal";
|
2023-06-16 17:35:50 +00:00
|
|
|
|
2024-11-05 19:49:27 +00:00
|
|
|
const IntlManagerLogger = new Logger("IntlManager");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get an internationalized message from a non hashed key
|
|
|
|
* @param key The plain message key
|
|
|
|
* @param values The values to interpolate, if it's a rich message
|
|
|
|
*/
|
|
|
|
export function getIntlMessage(key: string, values?: Record<PropertyKey, any>): any {
|
|
|
|
return getIntlMessageFromHash(runtimeHashMessageKey(key), values, key);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Get an internationalized message from a hashed key
|
|
|
|
* @param hashedKey The hashed message key
|
|
|
|
* @param values The values to interpolate, if it's a rich message
|
|
|
|
*/
|
|
|
|
export function getIntlMessageFromHash(hashedKey: string, values?: Record<PropertyKey, any>, originalKey?: string): any {
|
|
|
|
try {
|
|
|
|
return values == null ? i18n.intl.string(i18n.t[hashedKey]) : i18n.intl.format(i18n.t[hashedKey], values);
|
|
|
|
} catch (e) {
|
|
|
|
IntlManagerLogger.error(`Failed to get intl message for key: ${originalKey ?? hashedKey}`, e);
|
|
|
|
return originalKey ?? "";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-11-30 05:10:50 +00:00
|
|
|
/**
|
|
|
|
* Open the invite modal
|
|
|
|
* @param code The invite code
|
|
|
|
* @returns Whether the invite was accepted
|
|
|
|
*/
|
2023-11-30 02:14:52 +00:00
|
|
|
export async function openInviteModal(code: string) {
|
|
|
|
const { invite } = await InviteActions.resolveInvite(code, "Desktop Modal");
|
|
|
|
if (!invite) throw new Error("Invalid invite: " + code);
|
|
|
|
|
|
|
|
FluxDispatcher.dispatch({
|
|
|
|
type: "INVITE_MODAL_OPEN",
|
|
|
|
invite,
|
|
|
|
code,
|
|
|
|
context: "APP"
|
|
|
|
});
|
2023-11-30 05:10:50 +00:00
|
|
|
|
|
|
|
return new Promise<boolean>(r => {
|
|
|
|
let onClose: () => void, onAccept: () => void;
|
|
|
|
let inviteAccepted = false;
|
|
|
|
|
|
|
|
FluxDispatcher.subscribe("INVITE_ACCEPT", onAccept = () => {
|
|
|
|
inviteAccepted = true;
|
|
|
|
});
|
|
|
|
|
|
|
|
FluxDispatcher.subscribe("INVITE_MODAL_CLOSE", onClose = () => {
|
|
|
|
FluxDispatcher.unsubscribe("INVITE_MODAL_CLOSE", onClose);
|
|
|
|
FluxDispatcher.unsubscribe("INVITE_ACCEPT", onAccept);
|
|
|
|
r(inviteAccepted);
|
|
|
|
});
|
|
|
|
});
|
2023-11-30 02:14:52 +00:00
|
|
|
}
|
2023-04-15 02:42:18 +00:00
|
|
|
|
2024-09-05 00:12:09 +00:00
|
|
|
export function getCurrentChannel(): Channel | undefined {
|
2022-10-09 17:48:42 +00:00
|
|
|
return ChannelStore.getChannel(SelectedChannelStore.getChannelId());
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getCurrentGuild(): Guild | undefined {
|
2024-09-05 00:12:09 +00:00
|
|
|
return GuildStore.getGuild(getCurrentChannel()?.guild_id!);
|
2022-10-09 17:48:42 +00:00
|
|
|
}
|
2022-11-19 13:54:48 +00:00
|
|
|
|
|
|
|
export function openPrivateChannel(userId: string) {
|
|
|
|
PrivateChannelsStore.openPrivateChannel(userId);
|
|
|
|
}
|
2023-04-15 02:42:18 +00:00
|
|
|
|
|
|
|
export const enum Theme {
|
|
|
|
Dark = 1,
|
|
|
|
Light = 2
|
|
|
|
}
|
|
|
|
|
|
|
|
export function getTheme(): Theme {
|
2023-10-25 16:42:31 +00:00
|
|
|
return UserSettingsActionCreators.PreloadedUserSettingsActionCreators.getCurrentValue()?.appearance?.theme;
|
2023-04-15 02:42:18 +00:00
|
|
|
}
|
2023-04-18 21:12:58 +00:00
|
|
|
|
|
|
|
export function insertTextIntoChatInputBox(text: string) {
|
|
|
|
ComponentDispatch.dispatchToLastSubscribed("INSERT_TEXT", {
|
|
|
|
rawText: text,
|
|
|
|
plainText: text
|
|
|
|
});
|
|
|
|
}
|
2023-05-23 01:47:09 +00:00
|
|
|
|
|
|
|
interface MessageExtra {
|
|
|
|
messageReference: Message["messageReference"];
|
|
|
|
allowedMentions: {
|
|
|
|
parse: string[];
|
|
|
|
replied_user: boolean;
|
|
|
|
};
|
|
|
|
stickerIds: string[];
|
|
|
|
}
|
|
|
|
|
|
|
|
export function sendMessage(
|
|
|
|
channelId: string,
|
|
|
|
data: Partial<MessageObject>,
|
|
|
|
waitForChannelReady?: boolean,
|
|
|
|
extra?: Partial<MessageExtra>
|
|
|
|
) {
|
|
|
|
const messageData = {
|
|
|
|
content: "",
|
|
|
|
invalidEmojis: [],
|
|
|
|
tts: false,
|
|
|
|
validNonShortcutEmojis: [],
|
|
|
|
...data
|
|
|
|
};
|
|
|
|
|
|
|
|
return MessageActions.sendMessage(channelId, messageData, waitForChannelReady, extra);
|
|
|
|
}
|
2023-06-16 17:35:50 +00:00
|
|
|
|
2024-10-23 01:57:46 +00:00
|
|
|
/**
|
2024-10-24 11:10:29 +00:00
|
|
|
* You must specify either height or width in the item
|
2024-10-23 01:57:46 +00:00
|
|
|
*/
|
2024-10-24 11:10:29 +00:00
|
|
|
export function openImageModal(item: Except<MediaModalItem, "type">, mediaModalProps?: Omit<MediaModalProps, "items">) {
|
|
|
|
return openMediaModal({
|
|
|
|
className: "vc-image-modal",
|
|
|
|
fit: "vc-position-inherit",
|
|
|
|
shouldAnimateCarousel: true,
|
|
|
|
items: [{
|
|
|
|
type: "IMAGE",
|
|
|
|
original: item.original ?? item.url,
|
|
|
|
...item,
|
|
|
|
}],
|
|
|
|
...mediaModalProps
|
|
|
|
});
|
2023-06-16 17:35:50 +00:00
|
|
|
}
|
2023-06-23 16:09:43 +00:00
|
|
|
|
|
|
|
export async function openUserProfile(id: string) {
|
2023-10-25 14:57:50 +00:00
|
|
|
const user = await UserUtils.getUser(id);
|
2023-06-23 16:09:43 +00:00
|
|
|
if (!user) throw new Error("No such user: " + id);
|
|
|
|
|
|
|
|
const guildId = SelectedGuildStore.getGuildId();
|
2023-10-25 18:43:48 +00:00
|
|
|
UserProfileActions.openUserProfileModal({
|
2023-06-23 16:09:43 +00:00
|
|
|
userId: id,
|
|
|
|
guildId,
|
|
|
|
channelId: SelectedChannelStore.getChannelId(),
|
|
|
|
analyticsLocation: {
|
|
|
|
page: guildId ? "Guild Channel" : "DM Channel",
|
|
|
|
section: "Profile Popout"
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2023-09-08 01:42:20 +00:00
|
|
|
interface FetchUserProfileOptions {
|
|
|
|
friend_token?: string;
|
|
|
|
connections_role_id?: string;
|
|
|
|
guild_id?: string;
|
|
|
|
with_mutual_guilds?: boolean;
|
|
|
|
with_mutual_friends_count?: boolean;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Fetch a user's profile
|
|
|
|
*/
|
|
|
|
export async function fetchUserProfile(id: string, options?: FetchUserProfileOptions) {
|
|
|
|
const cached = UserProfileStore.getUserProfile(id);
|
|
|
|
if (cached) return cached;
|
|
|
|
|
|
|
|
FluxDispatcher.dispatch({ type: "USER_PROFILE_FETCH_START", userId: id });
|
|
|
|
|
|
|
|
const { body } = await RestAPI.get({
|
2024-05-14 02:00:02 +00:00
|
|
|
url: Constants.Endpoints.USER_PROFILE(id),
|
2023-09-08 01:42:20 +00:00
|
|
|
query: {
|
|
|
|
with_mutual_guilds: false,
|
|
|
|
with_mutual_friends_count: false,
|
|
|
|
...options
|
|
|
|
},
|
|
|
|
oldFormErrors: true,
|
|
|
|
});
|
|
|
|
|
|
|
|
FluxDispatcher.dispatch({ type: "USER_UPDATE", user: body.user });
|
|
|
|
await FluxDispatcher.dispatch({ type: "USER_PROFILE_FETCH_SUCCESS", ...body });
|
|
|
|
if (options?.guild_id && body.guild_member)
|
|
|
|
FluxDispatcher.dispatch({ type: "GUILD_MEMBER_PROFILE_UPDATE", guildId: options.guild_id, guildMember: body.guild_member });
|
|
|
|
|
|
|
|
return UserProfileStore.getUserProfile(id);
|
|
|
|
}
|
|
|
|
|
2023-06-23 16:09:43 +00:00
|
|
|
/**
|
|
|
|
* Get the unique username for a user. Returns user.username for pomelo people, user.tag otherwise
|
|
|
|
*/
|
|
|
|
export function getUniqueUsername(user: User) {
|
|
|
|
return user.discriminator === "0" ? user.username : user.tag;
|
|
|
|
}
|
2024-11-13 18:27:21 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Get the URL for an emoji. This function always returns a gif URL for animated emojis, instead of webp
|
|
|
|
* @param id The emoji id
|
|
|
|
* @param animated Whether the emoji is animated
|
|
|
|
* @param size The size for the emoji
|
|
|
|
*/
|
|
|
|
export function getEmojiURL(id: string, animated: boolean, size: number) {
|
|
|
|
const url = IconUtils.getEmojiURL({ id, animated, size });
|
|
|
|
return animated ? url.replace(".webp", ".gif") : url;
|
|
|
|
}
|