mirror of
https://github.com/Vendicated/Vencord.git
synced 2025-02-24 15:35:11 +00:00
Merge changes from 'main'
This commit is contained in:
parent
6cccb54ffc
commit
e4947793bb
4 changed files with 237 additions and 19 deletions
|
@ -20,13 +20,17 @@ import "./styles.css";
|
||||||
|
|
||||||
import { get, set } from "@api/DataStore";
|
import { get, set } from "@api/DataStore";
|
||||||
import { updateMessage } from "@api/MessageUpdater";
|
import { updateMessage } from "@api/MessageUpdater";
|
||||||
import { migratePluginSettings } from "@api/Settings";
|
import { definePluginSettings, migratePluginSettings } from "@api/Settings";
|
||||||
import { ImageInvisible, ImageVisible } from "@components/Icons";
|
import { ImageInvisible, ImageVisible } from "@components/Icons";
|
||||||
import { Devs } from "@utils/constants";
|
import { Devs } from "@utils/constants";
|
||||||
import { classes } from "@utils/misc";
|
import { classes } from "@utils/misc";
|
||||||
import definePlugin from "@utils/types";
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
import { ChannelStore } from "@webpack/common";
|
import { ChannelStore } from "@webpack/common";
|
||||||
import { MessageSnapshot } from "@webpack/types";
|
import { MessageSnapshot } from "@webpack/types";
|
||||||
|
import { Embed } from "discord-types/general";
|
||||||
|
|
||||||
|
import { ILoadMessagesSuccessPayload, IMessage, IMessageCreatePayload, IMessageUpdatePayload } from "./types";
|
||||||
|
import { isStringEmpty } from "./utils";
|
||||||
|
|
||||||
const KEY = "HideAttachments_HiddenIds";
|
const KEY = "HideAttachments_HiddenIds";
|
||||||
|
|
||||||
|
@ -41,10 +45,145 @@ const saveHiddenMessages = (ids: Set<string>) => set(KEY, ids);
|
||||||
|
|
||||||
migratePluginSettings("HideMedia", "HideAttachments");
|
migratePluginSettings("HideMedia", "HideAttachments");
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Toggle attachment/embed hiding
|
||||||
|
*/
|
||||||
|
const toggleHide = async (channelId: string, messageId: string): Promise<void> => {
|
||||||
|
const ids = await getHiddenMessages();
|
||||||
|
if (!ids.delete(messageId))
|
||||||
|
ids.add(messageId);
|
||||||
|
|
||||||
|
await saveHiddenMessages(ids);
|
||||||
|
updateMessage(channelId, messageId);
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if the message should be blocked according to user ID filter
|
||||||
|
* @param {Message} payload The message to be checked
|
||||||
|
* @param {string[]} userFilters List of user IDs to be checked
|
||||||
|
* @returns {boolean}
|
||||||
|
*/
|
||||||
|
const shouldHideByUserIdFilter = (payload: IMessage, userFilters: string[]): boolean => {
|
||||||
|
for (const id of userFilters) {
|
||||||
|
if (payload.author.id === id) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
const shouldHideEmbed = (embeds: Embed[], domainList: string[]): boolean => {
|
||||||
|
for (const embed of embeds) {
|
||||||
|
if (!embed.url) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const domain of domainList) {
|
||||||
|
const host = URL.parse(embed.url)?.hostname ?? "";
|
||||||
|
if (host.indexOf(domain) >= 0) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Determine if the message should be blocked according to domain list filter
|
||||||
|
* @param {Message} payload The message to be checked
|
||||||
|
* @param {string[]} domainList List of domains to be checked
|
||||||
|
* @returns {boolean}
|
||||||
|
*/
|
||||||
|
const shouldHideByDomainListFilter = (payload: IMessage, domainList: string[]): boolean => {
|
||||||
|
if (payload.embeds.length <= 0) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (shouldHideEmbed(payload.embeds, domainList)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check embeds from the forwarded messages
|
||||||
|
const hasReference = payload.message_reference && Array.isArray(payload.message_snapshots);
|
||||||
|
if (!hasReference) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const snapshot of payload.message_snapshots!) {
|
||||||
|
if (shouldHideEmbed(snapshot.message.embeds, domainList)) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks and hides the attachment/embed
|
||||||
|
* @param {Message} message The message to check
|
||||||
|
* @param {object} store The configuration values
|
||||||
|
*/
|
||||||
|
const checkAndHide = async (message: IMessage, store: typeof settings.store): Promise<void> => {
|
||||||
|
if (!store.enableAutoHideAttachments) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (hiddenMessages.has(message.id)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const userFilters = isStringEmpty(store.filterUserList)
|
||||||
|
? []
|
||||||
|
: store.filterUserList.split(",");
|
||||||
|
if (shouldHideByUserIdFilter(message, userFilters)) {
|
||||||
|
await toggleHide(message.channel_id, message.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const domainFilters = isStringEmpty(store.filterDomainList)
|
||||||
|
? []
|
||||||
|
: store.filterDomainList.split(",");
|
||||||
|
if (shouldHideByDomainListFilter(message, domainFilters)) {
|
||||||
|
await toggleHide(message.channel_id, message.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forwarded messages
|
||||||
|
const hasReference = message.message_reference && Array.isArray(message.message_snapshots);
|
||||||
|
if (hasReference) {
|
||||||
|
for (const snapshot of message.message_snapshots!) {
|
||||||
|
if (shouldHideByDomainListFilter(snapshot.message, domainFilters)) {
|
||||||
|
await toggleHide(message.channel_id, message.id);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const settings = definePluginSettings({
|
||||||
|
enableAutoHideAttachments: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Enable auto hide attachments",
|
||||||
|
default: false,
|
||||||
|
},
|
||||||
|
filterUserList: {
|
||||||
|
type: OptionType.STRING,
|
||||||
|
description: "Comma separated list of User IDs to automatically hide their attachments/embeds. (Requires auto hide to be ON)",
|
||||||
|
default: "",
|
||||||
|
},
|
||||||
|
filterDomainList: {
|
||||||
|
type: OptionType.STRING,
|
||||||
|
description: "Comma separated list of domains to automatically hide their embeds. (Requires auto hide to be ON)",
|
||||||
|
default: "",
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
export default definePlugin({
|
export default definePlugin({
|
||||||
name: "HideMedia",
|
name: "HideAttachments",
|
||||||
description: "Hide attachments and embeds for individual messages via hover button",
|
description: "Hide attachments and Embeds for individual messages via hover button",
|
||||||
authors: [Devs.Ven],
|
authors: [Devs.Ven, Devs.aiko],
|
||||||
dependencies: ["MessageUpdaterAPI"],
|
dependencies: ["MessageUpdaterAPI"],
|
||||||
|
|
||||||
patches: [{
|
patches: [{
|
||||||
|
@ -55,13 +194,15 @@ export default definePlugin({
|
||||||
}
|
}
|
||||||
}],
|
}],
|
||||||
|
|
||||||
renderMessagePopoverButton(msg) {
|
settings,
|
||||||
|
|
||||||
|
renderMessagePopoverButton(msg: IMessage) {
|
||||||
// @ts-ignore - discord-types lags behind discord.
|
// @ts-ignore - discord-types lags behind discord.
|
||||||
const hasAttachmentsInShapshots = msg.messageSnapshots.some(
|
const hasAttachmentsInSnapshots = msg.messageSnapshots.some(
|
||||||
(snapshot: MessageSnapshot) => snapshot?.message.attachments.length
|
(snapshot: MessageSnapshot) => snapshot?.message.attachments.length || snapshot?.message.embeds.length
|
||||||
);
|
);
|
||||||
|
|
||||||
if (!msg.attachments.length && !msg.embeds.length && !msg.stickerItems.length && !hasAttachmentsInShapshots) return null;
|
if (!msg.attachments.length && !msg.embeds.length && !msg.stickerItems.length && !hasAttachmentsInSnapshots) return null;
|
||||||
|
|
||||||
const isHidden = hiddenMessages.has(msg.id);
|
const isHidden = hiddenMessages.has(msg.id);
|
||||||
|
|
||||||
|
@ -70,7 +211,7 @@ export default definePlugin({
|
||||||
icon: isHidden ? ImageVisible : ImageInvisible,
|
icon: isHidden ? ImageVisible : ImageInvisible,
|
||||||
message: msg,
|
message: msg,
|
||||||
channel: ChannelStore.getChannel(msg.channel_id),
|
channel: ChannelStore.getChannel(msg.channel_id),
|
||||||
onClick: () => this.toggleHide(msg.channel_id, msg.id)
|
onClick: () => toggleHide(msg.channel_id, msg.id)
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
|
||||||
|
@ -84,6 +225,22 @@ export default definePlugin({
|
||||||
);
|
);
|
||||||
},
|
},
|
||||||
|
|
||||||
|
flux: {
|
||||||
|
async LOAD_MESSAGES_SUCCESS(payload: ILoadMessagesSuccessPayload) {
|
||||||
|
for (const message of payload.messages) {
|
||||||
|
await checkAndHide(message, settings.store);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
async MESSAGE_CREATE({ message }: IMessageCreatePayload) {
|
||||||
|
await checkAndHide(message, settings.store);
|
||||||
|
},
|
||||||
|
|
||||||
|
async MESSAGE_UPDATE({ message }: IMessageUpdatePayload) {
|
||||||
|
await checkAndHide(message, settings.store);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
async start() {
|
async start() {
|
||||||
await getHiddenMessages();
|
await getHiddenMessages();
|
||||||
},
|
},
|
||||||
|
@ -94,14 +251,5 @@ export default definePlugin({
|
||||||
|
|
||||||
shouldHide(messageId: string) {
|
shouldHide(messageId: string) {
|
||||||
return hiddenMessages.has(messageId);
|
return hiddenMessages.has(messageId);
|
||||||
},
|
|
||||||
|
|
||||||
async toggleHide(channelId: string, messageId: string) {
|
|
||||||
const ids = await getHiddenMessages();
|
|
||||||
if (!ids.delete(messageId))
|
|
||||||
ids.add(messageId);
|
|
||||||
|
|
||||||
await saveHiddenMessages(ids);
|
|
||||||
updateMessage(channelId, messageId);
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
44
src/plugins/hideAttachments/types.ts
Normal file
44
src/plugins/hideAttachments/types.ts
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
* 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 { Message } from "discord-types/general";
|
||||||
|
|
||||||
|
export interface ILoadMessagesSuccessPayload {
|
||||||
|
channelId: string;
|
||||||
|
messages: Array<Message>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IMessage extends Message {
|
||||||
|
message_reference?: {
|
||||||
|
type: number;
|
||||||
|
channel_id: string;
|
||||||
|
message_id: string;
|
||||||
|
guild_id: string;
|
||||||
|
},
|
||||||
|
message_snapshots?: {
|
||||||
|
message: Message;
|
||||||
|
}[]
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IMessageCreatePayload {
|
||||||
|
message: IMessage;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface IMessageUpdatePayload {
|
||||||
|
message: IMessage;
|
||||||
|
}
|
22
src/plugins/hideAttachments/utils.ts
Normal file
22
src/plugins/hideAttachments/utils.ts
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
/*
|
||||||
|
* 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/>.
|
||||||
|
*/
|
||||||
|
|
||||||
|
export function isStringEmpty (str: string) {
|
||||||
|
if (!str) return false;
|
||||||
|
return str.trim().length === 0;
|
||||||
|
}
|
|
@ -579,6 +579,10 @@ export const Devs = /* #__PURE__*/ Object.freeze({
|
||||||
name: "jamesbt365",
|
name: "jamesbt365",
|
||||||
id: 158567567487795200n,
|
id: 158567567487795200n,
|
||||||
},
|
},
|
||||||
|
aiko: {
|
||||||
|
name: "kima_riiiiiii",
|
||||||
|
id: 366434327761911808n
|
||||||
|
}
|
||||||
} satisfies Record<string, Dev>);
|
} satisfies Record<string, Dev>);
|
||||||
|
|
||||||
// iife so #__PURE__ works correctly
|
// iife so #__PURE__ works correctly
|
||||||
|
|
Loading…
Add table
Reference in a new issue