/* * 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 . */ import { addPreEditListener, addPreSendListener, removePreEditListener, removePreSendListener } from "@api/MessageEvents"; import { definePluginSettings, Settings } from "@api/Settings"; import { Devs } from "@utils/constants"; import { ApngBlendOp, ApngDisposeOp, importApngJs } from "@utils/dependencies"; import { getCurrentGuild } from "@utils/discord"; import { Logger } from "@utils/Logger"; import definePlugin, { OptionType } from "@utils/types"; import { findByPropsLazy, findStoreLazy, proxyLazyWebpack } from "@webpack"; import { ChannelStore, EmojiStore, FluxDispatcher, lodash, Parser, PermissionStore, UploadHandler, UserSettingsActionCreators, UserStore } from "@webpack/common"; import type { Message } from "discord-types/general"; import { applyPalette, GIFEncoder, quantize } from "gifenc"; import type { ReactElement, ReactNode } from "react"; const DRAFT_TYPE = 0; const StickerStore = findStoreLazy("StickersStore") as { getPremiumPacks(): StickerPack[]; getAllGuildStickers(): Map; getStickerById(id: string): Sticker | undefined; }; const UserSettingsProtoStore = findStoreLazy("UserSettingsProtoStore"); const ProtoUtils = findByPropsLazy("BINARY_READ_OPTIONS"); function searchProtoClassField(localName: string, protoClass: any) { const field = protoClass?.fields?.find((field: any) => field.localName === localName); if (!field) return; const fieldGetter = Object.values(field).find(value => typeof value === "function") as any; return fieldGetter?.(); } const PreloadedUserSettingsActionCreators = proxyLazyWebpack(() => UserSettingsActionCreators.PreloadedUserSettingsActionCreators); const AppearanceSettingsActionCreators = proxyLazyWebpack(() => searchProtoClassField("appearance", PreloadedUserSettingsActionCreators.ProtoClass)); const ClientThemeSettingsActionsCreators = proxyLazyWebpack(() => searchProtoClassField("clientThemeSettings", AppearanceSettingsActionCreators)); const USE_EXTERNAL_EMOJIS = 1n << 18n; const USE_EXTERNAL_STICKERS = 1n << 37n; const enum EmojiIntentions { REACTION = 0, STATUS = 1, COMMUNITY_CONTENT = 2, CHAT = 3, GUILD_STICKER_RELATED_EMOJI = 4, GUILD_ROLE_BENEFIT_EMOJI = 5, COMMUNITY_CONTENT_ONLY = 6, SOUNDBOARD = 7 } const enum StickerType { PNG = 1, APNG = 2, LOTTIE = 3, // don't think you can even have gif stickers but the docs have it GIF = 4 } interface BaseSticker { available: boolean; description: string; format_type: number; id: string; name: string; tags: string; type: number; } interface GuildSticker extends BaseSticker { guild_id: string; } interface DiscordSticker extends BaseSticker { pack_id: string; } type Sticker = GuildSticker | DiscordSticker; interface StickerPack { id: string; name: string; sku_id: string; description: string; cover_sticker_id: string; banner_asset_id: string; stickers: Sticker[]; } const enum FakeNoticeType { Sticker, Emoji } const fakeNitroEmojiRegex = /\/emojis\/(\d+?)\.(png|webp|gif)/; const fakeNitroStickerRegex = /\/stickers\/(\d+?)\./; const fakeNitroGifStickerRegex = /\/attachments\/\d+?\/\d+?\/(\d+?)\.gif/; const hyperLinkRegex = /\[.+?\]\((https?:\/\/.+?)\)/; const settings = definePluginSettings({ enableEmojiBypass: { description: "Allow sending fake emojis", type: OptionType.BOOLEAN, default: true, restartNeeded: true }, emojiSize: { description: "Size of the emojis when sending", type: OptionType.SLIDER, default: 48, markers: [32, 48, 64, 128, 160, 256, 512] }, transformEmojis: { description: "Whether to transform fake emojis into real ones", type: OptionType.BOOLEAN, default: true, restartNeeded: true }, enableStickerBypass: { description: "Allow sending fake stickers", type: OptionType.BOOLEAN, default: true, restartNeeded: true }, stickerSize: { description: "Size of the stickers when sending", type: OptionType.SLIDER, default: 160, markers: [32, 64, 128, 160, 256, 512] }, transformStickers: { description: "Whether to transform fake stickers into real ones", type: OptionType.BOOLEAN, default: true, restartNeeded: true }, transformCompoundSentence: { description: "Whether to transform fake stickers and emojis in compound sentences (sentences with more content than just the fake emoji or sticker link)", type: OptionType.BOOLEAN, default: false }, enableStreamQualityBypass: { description: "Allow streaming in nitro quality", type: OptionType.BOOLEAN, default: true, restartNeeded: true }, useHyperLinks: { description: "Whether to use hyperlinks when sending fake emojis and stickers", type: OptionType.BOOLEAN, default: true } }); export default definePlugin({ name: "FakeNitro", authors: [Devs.Arjix, Devs.D3SOX, Devs.Ven, Devs.obscurity, Devs.captain, Devs.Nuckyz, Devs.AutumnVN], description: "Allows you to stream in nitro quality, send fake emojis/stickers and use client themes.", dependencies: ["MessageEventsAPI"], settings, patches: [ { find: ".PREMIUM_LOCKED;", predicate: () => settings.store.enableEmojiBypass, replacement: [ { // Create a variable for the intention of listing the emoji match: /(?<=,intention:(\i).+?;)/, replace: (_, intention) => `let fakeNitroIntention=${intention};` }, { // Send the intention of listing the emoji to the nitro permission check functions match: /\.(?:canUseEmojisEverywhere|canUseAnimatedEmojis)\(\i(?=\))/g, replace: '$&,typeof fakeNitroIntention!=="undefined"?fakeNitroIntention:void 0' }, { // Disallow the emoji if the intention doesn't allow it match: /(&&!\i&&)!(\i)(?=\)return \i\.\i\.DISALLOW_EXTERNAL;)/, replace: (_, rest, canUseExternal) => `${rest}(!${canUseExternal}&&(typeof fakeNitroIntention==="undefined"||![${EmojiIntentions.CHAT},${EmojiIntentions.GUILD_STICKER_RELATED_EMOJI}].includes(fakeNitroIntention)))` }, { // Make the emoji always available if the intention allows it match: /if\(!\i\.available/, replace: m => `${m}&&(typeof fakeNitroIntention==="undefined"||![${EmojiIntentions.CHAT},${EmojiIntentions.GUILD_STICKER_RELATED_EMOJI}].includes(fakeNitroIntention))` } ] }, // Allow emojis and animated emojis to be sent everywhere { find: "canUseAnimatedEmojis:function", predicate: () => settings.store.enableEmojiBypass, replacement: { match: /((?:canUseEmojisEverywhere|canUseAnimatedEmojis):function\(\i)\){(.+?\))(?=})/g, replace: (_, rest, premiumCheck) => `${rest},fakeNitroIntention){${premiumCheck}||fakeNitroIntention==null||[${EmojiIntentions.CHAT},${EmojiIntentions.GUILD_STICKER_RELATED_EMOJI}].includes(fakeNitroIntention)` } }, // Allow stickers to be sent everywhere { find: "canUseCustomStickersEverywhere:function", predicate: () => settings.store.enableStickerBypass, replacement: { match: /canUseCustomStickersEverywhere:function\(\i\){/, replace: "$&return true;" }, }, // Make stickers always available { find: "\"SENDABLE\"", predicate: () => settings.store.enableStickerBypass, replacement: { match: /(\w+)\.available\?/, replace: "true?" } }, // Allow streaming with high quality { find: "canUseHighVideoUploadQuality:function", predicate: () => settings.store.enableStreamQualityBypass, replacement: [ "canUseHighVideoUploadQuality", "canStreamQuality", ].map(func => { return { match: new RegExp(`${func}:function\\(\\i(?:,\\i)?\\){`, "g"), replace: "$&return true;" }; }) }, // Remove boost requirements to stream with high quality { find: "STREAM_FPS_OPTION.format", predicate: () => settings.store.enableStreamQualityBypass, replacement: { match: /guildPremiumTier:\i\.\i\.TIER_\d,?/g, replace: "" } }, // Allow client themes to be changeable { find: "canUseClientThemes:function", replacement: { match: /canUseClientThemes:function\(\i\){/, replace: "$&return true;" } }, { find: '.displayName="UserSettingsProtoStore"', replacement: [ { // Overwrite incoming connection settings proto with our local settings match: /CONNECTION_OPEN:function\((\i)\){/, replace: (m, props) => `${m}$self.handleProtoChange(${props}.userSettingsProto,${props}.user);` }, { // Overwrite non local proto changes with our local settings match: /let{settings:/, replace: "arguments[0].local||$self.handleProtoChange(arguments[0].settings.proto);$&" } ] }, // Call our function to handle changing the gradient theme when selecting a new one { find: ",updateTheme(", replacement: { match: /(function \i\(\i\){let{backgroundGradientPresetId:(\i).+?)(\i\.\i\.updateAsync.+?theme=(.+?),.+?},\i\))/, replace: (_, rest, backgroundGradientPresetId, originalCall, theme) => `${rest}$self.handleGradientThemeSelect(${backgroundGradientPresetId},${theme},()=>${originalCall});` } }, { find: '["strong","em","u","text","inlineCode","s","spoiler"]', replacement: [ { // Call our function to decide whether the emoji link should be kept or not predicate: () => settings.store.transformEmojis, match: /1!==(\i)\.length\|\|1!==\i\.length/, replace: (m, content) => `${m}||$self.shouldKeepEmojiLink(${content}[0])` }, { // Patch the rendered message content to add fake nitro emojis or remove sticker links predicate: () => settings.store.transformEmojis || settings.store.transformStickers, match: /(?=return{hasSpoilerEmbeds:\i,content:(\i)})/, replace: (_, content) => `${content}=$self.patchFakeNitroEmojisOrRemoveStickersLinks(${content},arguments[2]?.formatInline);` } ] }, { find: "renderEmbeds(", replacement: [ { // Call our function to decide whether the embed should be ignored or not predicate: () => settings.store.transformEmojis || settings.store.transformStickers, match: /(renderEmbeds\((\i)\){)(.+?embeds\.map\((\i)=>{)/, replace: (_, rest1, message, rest2, embed) => `${rest1}const fakeNitroMessage=${message};${rest2}if($self.shouldIgnoreEmbed(${embed},fakeNitroMessage))return null;` }, { // Patch the stickers array to add fake nitro stickers predicate: () => settings.store.transformStickers, match: /(?<=renderStickersAccessories\((\i)\){let (\i)=\(0,\i\.\i\)\(\i\).+?;)/, replace: (_, message, stickers) => `${stickers}=$self.patchFakeNitroStickers(${stickers},${message});` }, { // Filter attachments to remove fake nitro stickers or emojis predicate: () => settings.store.transformStickers, match: /renderAttachments\(\i\){let{attachments:(\i).+?;/, replace: (m, attachments) => `${m}${attachments}=$self.filterAttachments(${attachments});` } ] }, { find: ".Messages.STICKER_POPOUT_UNJOINED_PRIVATE_GUILD_DESCRIPTION.format", predicate: () => settings.store.transformStickers, replacement: [ { // Export the renderable sticker to be used in the fake nitro sticker notice match: /let{renderableSticker:(\i).{0,250}isGuildSticker.+?channel:\i,/, replace: (m, renderableSticker) => `${m}fakeNitroRenderableSticker:${renderableSticker},` }, { // Add the fake nitro sticker notice match: /(let \i,{sticker:\i,channel:\i,closePopout:\i.+?}=(\i).+?;)(.+?description:)(\i)(?=,sticker:\i)/, replace: (_, rest, props, rest2, reactNode) => `${rest}let{fakeNitroRenderableSticker}=${props};${rest2}$self.addFakeNotice(${FakeNoticeType.Sticker},${reactNode},!!fakeNitroRenderableSticker?.fake)` } ] }, { find: ".EMOJI_UPSELL_POPOUT_MORE_EMOJIS_OPENED,", predicate: () => settings.store.transformEmojis, replacement: { // Export the emoji node to be used in the fake nitro emoji notice match: /isDiscoverable:\i,shouldHideRoleSubscriptionCTA:\i,(?<={node:(\i),.+?)/, replace: (m, node) => `${m}fakeNitroNode:${node},` } }, { find: ".Messages.EMOJI_POPOUT_UNJOINED_DISCOVERABLE_GUILD_DESCRIPTION", predicate: () => settings.store.transformEmojis, replacement: { // Add the fake nitro emoji notice match: /(?<=isDiscoverable:\i,emojiComesFromCurrentGuild:\i,.+?}=(\i).+?;)(.*?return )(.{0,1000}\.Messages\.EMOJI_POPOUT_UNJOINED_DISCOVERABLE_GUILD_DESCRIPTION.+?)(?=},)/, replace: (_, props, rest, reactNode) => `let{fakeNitroNode}=${props};${rest}$self.addFakeNotice(${FakeNoticeType.Emoji},${reactNode},!!fakeNitroNode?.fake)` } }, // Allow using custom app icons { find: "canUsePremiumAppIcons:function", replacement: { match: /canUsePremiumAppIcons:function\(\i\){/, replace: "$&return true;" } }, // Separate patch for allowing using custom app icons { find: ".FreemiumAppIconIds.DEFAULT&&(", replacement: { match: /\i\.\i\.isPremium\(\i\.\i\.getCurrentUser\(\)\)/, replace: "true" } } ], get guildId() { return getCurrentGuild()?.id; }, get canUseEmotes() { return (UserStore.getCurrentUser().premiumType ?? 0) > 0; }, get canUseStickers() { return (UserStore.getCurrentUser().premiumType ?? 0) > 1; }, handleProtoChange(proto: any, user: any) { if (proto == null || typeof proto === "string" || !UserSettingsProtoStore || !PreloadedUserSettingsActionCreators || !AppearanceSettingsActionCreators || !ClientThemeSettingsActionsCreators) return; const premiumType: number = user?.premium_type ?? UserStore?.getCurrentUser()?.premiumType ?? 0; if (premiumType !== 2) { proto.appearance ??= AppearanceSettingsActionCreators.create(); if (UserSettingsProtoStore.settings.appearance?.theme != null) { const appearanceSettingsDummy = AppearanceSettingsActionCreators.create({ theme: UserSettingsProtoStore.settings.appearance.theme }); proto.appearance.theme = appearanceSettingsDummy.theme; } if (UserSettingsProtoStore.settings.appearance?.clientThemeSettings?.backgroundGradientPresetId?.value != null) { const clientThemeSettingsDummy = ClientThemeSettingsActionsCreators.create({ backgroundGradientPresetId: { value: UserSettingsProtoStore.settings.appearance.clientThemeSettings.backgroundGradientPresetId.value } }); proto.appearance.clientThemeSettings ??= clientThemeSettingsDummy; proto.appearance.clientThemeSettings.backgroundGradientPresetId = clientThemeSettingsDummy.backgroundGradientPresetId; } } }, handleGradientThemeSelect(backgroundGradientPresetId: number | undefined, theme: number, original: () => void) { const premiumType = UserStore?.getCurrentUser()?.premiumType ?? 0; if (premiumType === 2 || backgroundGradientPresetId == null) return original(); if (!PreloadedUserSettingsActionCreators || !AppearanceSettingsActionCreators || !ClientThemeSettingsActionsCreators || !ProtoUtils) return; const currentAppearanceSettings = PreloadedUserSettingsActionCreators.getCurrentValue().appearance; const newAppearanceProto = currentAppearanceSettings != null ? AppearanceSettingsActionCreators.fromBinary(AppearanceSettingsActionCreators.toBinary(currentAppearanceSettings), ProtoUtils.BINARY_READ_OPTIONS) : AppearanceSettingsActionCreators.create(); newAppearanceProto.theme = theme; const clientThemeSettingsDummy = ClientThemeSettingsActionsCreators.create({ backgroundGradientPresetId: { value: backgroundGradientPresetId } }); newAppearanceProto.clientThemeSettings ??= clientThemeSettingsDummy; newAppearanceProto.clientThemeSettings.backgroundGradientPresetId = clientThemeSettingsDummy.backgroundGradientPresetId; const proto = PreloadedUserSettingsActionCreators.ProtoClass.create(); proto.appearance = newAppearanceProto; FluxDispatcher.dispatch({ type: "USER_SETTINGS_PROTO_UPDATE", local: true, partial: true, settings: { type: 1, proto } }); }, trimContent(content: Array) { const firstContent = content[0]; if (typeof firstContent === "string") { content[0] = firstContent.trimStart(); content[0] || content.shift(); } else if (firstContent?.type === "span") { firstContent.props.children = firstContent.props.children.trimStart(); firstContent.props.children || content.shift(); } const lastIndex = content.length - 1; const lastContent = content[lastIndex]; if (typeof lastContent === "string") { content[lastIndex] = lastContent.trimEnd(); content[lastIndex] || content.pop(); } else if (lastContent?.type === "span") { lastContent.props.children = lastContent.props.children.trimEnd(); lastContent.props.children || content.pop(); } }, clearEmptyArrayItems(array: Array) { return array.filter(item => item != null); }, ensureChildrenIsArray(child: ReactElement) { if (!Array.isArray(child.props.children)) child.props.children = [child.props.children]; }, patchFakeNitroEmojisOrRemoveStickersLinks(content: Array, inline: boolean) { // If content has more than one child or it's a single ReactElement like a header, list or span if ((content.length > 1 || typeof content[0]?.type === "string") && !settings.store.transformCompoundSentence) return content; let nextIndex = content.length; const transformLinkChild = (child: ReactElement) => { if (settings.store.transformEmojis) { const fakeNitroMatch = child.props.href.match(fakeNitroEmojiRegex); if (fakeNitroMatch) { let url: URL | null = null; try { url = new URL(child.props.href); } catch { } const emojiName = EmojiStore.getCustomEmojiById(fakeNitroMatch[1])?.name ?? url?.searchParams.get("name") ?? "FakeNitroEmoji"; return Parser.defaultRules.customEmoji.react({ jumboable: !inline && content.length === 1 && typeof content[0].type !== "string", animated: fakeNitroMatch[2] === "gif", emojiId: fakeNitroMatch[1], name: emojiName, fake: true }, void 0, { key: String(nextIndex++) }); } } if (settings.store.transformStickers) { if (fakeNitroStickerRegex.test(child.props.href)) return null; const gifMatch = child.props.href.match(fakeNitroGifStickerRegex); if (gifMatch) { // There is no way to differentiate a regular gif attachment from a fake nitro animated sticker, so we check if the StickerStore contains the id of the fake sticker if (StickerStore.getStickerById(gifMatch[1])) return null; } } return child; }; const transformChild = (child: ReactElement) => { if (child?.props?.trusted != null) return transformLinkChild(child); if (child?.props?.children != null) { if (!Array.isArray(child.props.children)) { child.props.children = modifyChild(child.props.children); return child; } child.props.children = modifyChildren(child.props.children); if (child.props.children.length === 0) return null; return child; } return child; }; const modifyChild = (child: ReactElement) => { const newChild = transformChild(child); if (newChild?.type === "ul" || newChild?.type === "ol") { this.ensureChildrenIsArray(newChild); if (newChild.props.children.length === 0) return null; let listHasAnItem = false; for (const [index, child] of newChild.props.children.entries()) { if (child == null) { delete newChild.props.children[index]; continue; } this.ensureChildrenIsArray(child); if (child.props.children.length > 0) listHasAnItem = true; else delete newChild.props.children[index]; } if (!listHasAnItem) return null; newChild.props.children = this.clearEmptyArrayItems(newChild.props.children); } return newChild; }; const modifyChildren = (children: Array) => { for (const [index, child] of children.entries()) children[index] = modifyChild(child); children = this.clearEmptyArrayItems(children); this.trimContent(children); return children; }; try { return modifyChildren(lodash.cloneDeep(content)); } catch (err) { new Logger("FakeNitro").error(err); return content; } }, patchFakeNitroStickers(stickers: Array, message: Message) { const itemsToMaybePush: Array = []; const contentItems = message.content.split(/\s/); if (settings.store.transformCompoundSentence) itemsToMaybePush.push(...contentItems); else if (contentItems.length === 1) itemsToMaybePush.push(contentItems[0]); itemsToMaybePush.push(...message.attachments.filter(attachment => attachment.content_type === "image/gif").map(attachment => attachment.url)); for (const item of itemsToMaybePush) { if (!settings.store.transformCompoundSentence && !item.startsWith("http") && !hyperLinkRegex.test(item)) continue; const imgMatch = item.match(fakeNitroStickerRegex); if (imgMatch) { let url: URL | null = null; try { url = new URL(item); } catch { } const stickerName = StickerStore.getStickerById(imgMatch[1])?.name ?? url?.searchParams.get("name") ?? "FakeNitroSticker"; stickers.push({ format_type: 1, id: imgMatch[1], name: stickerName, fake: true }); continue; } const gifMatch = item.match(fakeNitroGifStickerRegex); if (gifMatch) { if (!StickerStore.getStickerById(gifMatch[1])) continue; const stickerName = StickerStore.getStickerById(gifMatch[1])?.name ?? "FakeNitroSticker"; stickers.push({ format_type: 2, id: gifMatch[1], name: stickerName, fake: true }); } } return stickers; }, shouldIgnoreEmbed(embed: Message["embeds"][number], message: Message) { const contentItems = message.content.split(/\s/); if (contentItems.length > 1 && !settings.store.transformCompoundSentence) return false; switch (embed.type) { case "image": { if ( !settings.store.transformCompoundSentence && !contentItems.some(item => item === embed.url! || item.match(hyperLinkRegex)?.[1] === embed.url!) ) return false; if (settings.store.transformEmojis) { if (fakeNitroEmojiRegex.test(embed.url!)) return true; } if (settings.store.transformStickers) { if (fakeNitroStickerRegex.test(embed.url!)) return true; const gifMatch = embed.url!.match(fakeNitroGifStickerRegex); if (gifMatch) { // There is no way to differentiate a regular gif attachment from a fake nitro animated sticker, so we check if the StickerStore contains the id of the fake sticker if (StickerStore.getStickerById(gifMatch[1])) return true; } } break; } } return false; }, filterAttachments(attachments: Message["attachments"]) { return attachments.filter(attachment => { if (attachment.content_type !== "image/gif") return true; const match = attachment.url.match(fakeNitroGifStickerRegex); if (match) { // There is no way to differentiate a regular gif attachment from a fake nitro animated sticker, so we check if the StickerStore contains the id of the fake sticker if (StickerStore.getStickerById(match[1])) return false; } return true; }); }, shouldKeepEmojiLink(link: any) { return link.target && fakeNitroEmojiRegex.test(link.target); }, addFakeNotice(type: FakeNoticeType, node: Array, fake: boolean) { if (!fake) return node; node = Array.isArray(node) ? node : [node]; switch (type) { case FakeNoticeType.Sticker: { node.push(" This is a FakeNitro sticker and renders like a real sticker only for you. Appears as a link to non-plugin users."); return node; } case FakeNoticeType.Emoji: { node.push(" This is a FakeNitro emoji and renders like a real emoji only for you. Appears as a link to non-plugin users."); return node; } } }, hasPermissionToUseExternalEmojis(channelId: string): boolean { const channel = ChannelStore.getChannel(channelId); if (!channel || channel.isDM() || channel.isGroupDM() || channel.isMultiUserDM()) return true; return PermissionStore.can(USE_EXTERNAL_EMOJIS, channel); }, hasPermissionToUseExternalStickers(channelId: string) { const channel = ChannelStore.getChannel(channelId); if (!channel || channel.isDM() || channel.isGroupDM() || channel.isMultiUserDM()) return true; return PermissionStore.can(USE_EXTERNAL_STICKERS, channel); }, getStickerLink(stickerId: string) { return `https://media.discordapp.net/stickers/${stickerId}.png?size=${settings.store.stickerSize}`; }, async sendAnimatedSticker(stickerLink: string, stickerId: string, channelId: string) { const { parseURL } = importApngJs(); const { frames, width, height } = await parseURL(stickerLink); const gif = GIFEncoder(); const resolution = Settings.plugins.FakeNitro.stickerSize; const canvas = document.createElement("canvas"); canvas.width = resolution; canvas.height = resolution; const ctx = canvas.getContext("2d", { willReadFrequently: true })!; const scale = resolution / Math.max(width, height); ctx.scale(scale, scale); let previousFrameData: ImageData; for (const frame of frames) { const { left, top, width, height, img, delay, blendOp, disposeOp } = frame; previousFrameData = ctx.getImageData(left, top, width, height); if (blendOp === ApngBlendOp.SOURCE) { ctx.clearRect(left, top, width, height); } ctx.drawImage(img, left, top, width, height); const { data } = ctx.getImageData(0, 0, resolution, resolution); const palette = quantize(data, 256); const index = applyPalette(data, palette); gif.writeFrame(index, resolution, resolution, { transparent: true, palette, delay }); if (disposeOp === ApngDisposeOp.BACKGROUND) { ctx.clearRect(left, top, width, height); } else if (disposeOp === ApngDisposeOp.PREVIOUS) { ctx.putImageData(previousFrameData, left, top); } } gif.finish(); const file = new File([gif.bytesView()], `${stickerId}.gif`, { type: "image/gif" }); UploadHandler.promptToUpload([file], ChannelStore.getChannel(channelId), DRAFT_TYPE); }, start() { const s = settings.store; if (!s.enableEmojiBypass && !s.enableStickerBypass) { return; } function getWordBoundary(origStr: string, offset: number) { return (!origStr[offset] || /\s/.test(origStr[offset])) ? "" : " "; } this.preSend = addPreSendListener((channelId, messageObj, extra) => { const { guildId } = this; stickerBypass: { if (!s.enableStickerBypass) break stickerBypass; const sticker = StickerStore.getStickerById(extra.stickers?.[0]!); if (!sticker) break stickerBypass; // Discord Stickers are now free yayyy!! :D if ("pack_id" in sticker) break stickerBypass; const canUseStickers = this.canUseStickers && this.hasPermissionToUseExternalStickers(channelId); if (sticker.available !== false && (canUseStickers || sticker.guild_id === guildId)) break stickerBypass; // [12/12/2023] // Work around an annoying bug where getStickerLink will return StickerType.GIF, // but will give us a normal non animated png for no reason // TODO: Remove this workaround when it's not needed anymore let link = this.getStickerLink(sticker.id); if (sticker.format_type === StickerType.GIF && link.includes(".png")) { link = link.replace(".png", ".gif"); } if (sticker.format_type === StickerType.APNG) { this.sendAnimatedSticker(link, sticker.id, channelId); return { cancel: true }; } else { const url = new URL(link); url.searchParams.set("name", sticker.name); messageObj.content += `${getWordBoundary(messageObj.content, messageObj.content.length - 1)}${s.useHyperLinks ? `[${sticker.name}](${url})` : url}`; extra.stickers!.length = 0; } } if (s.enableEmojiBypass) { const canUseEmotes = this.canUseEmotes && this.hasPermissionToUseExternalEmojis(channelId); for (const emoji of messageObj.validNonShortcutEmojis) { if (!emoji.require_colons) continue; if (emoji.available !== false && canUseEmotes) continue; if (emoji.guildId === guildId && !emoji.animated) continue; const emojiString = `<${emoji.animated ? "a" : ""}:${emoji.originalName || emoji.name}:${emoji.id}>`; const url = new URL(emoji.url); url.searchParams.set("size", s.emojiSize.toString()); url.searchParams.set("name", emoji.name); messageObj.content = messageObj.content.replace(emojiString, (match, offset, origStr) => { return `${getWordBoundary(origStr, offset - 1)}${s.useHyperLinks ? `[${emoji.name}](${url})` : url}${getWordBoundary(origStr, offset + match.length)}`; }); } } return { cancel: false }; }); this.preEdit = addPreEditListener((channelId, __, messageObj) => { if (!s.enableEmojiBypass) return; const canUseEmotes = this.canUseEmotes && this.hasPermissionToUseExternalEmojis(channelId); const { guildId } = this; messageObj.content = messageObj.content.replace(/(?/ig, (emojiStr, emojiId, offset, origStr) => { const emoji = EmojiStore.getCustomEmojiById(emojiId); if (emoji == null) return emojiStr; if (!emoji.require_colons) return emojiStr; if (emoji.available !== false && canUseEmotes) return emojiStr; if (emoji.guildId === guildId && !emoji.animated) return emojiStr; const url = new URL(emoji.url); url.searchParams.set("size", s.emojiSize.toString()); url.searchParams.set("name", emoji.name); return `${getWordBoundary(origStr, offset - 1)}${s.useHyperLinks ? `[${emoji.name}](${url})` : url}${getWordBoundary(origStr, offset + emojiStr.length)}`; }); }); }, stop() { removePreSendListener(this.preSend); removePreEditListener(this.preEdit); } });