mirror of
https://github.com/Vendicated/Vencord.git
synced 2025-01-25 08:46:25 +00:00
feat(plugin): PasswordProtect
This commit is contained in:
parent
748a456cfb
commit
8bf688218c
9 changed files with 414 additions and 0 deletions
5
src/plugins/passwordProtect/README.md
Normal file
5
src/plugins/passwordProtect/README.md
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
# PasswordProtect
|
||||||
|
|
||||||
|
Allows you to password protect your dms and channels!
|
||||||
|
|
||||||
|
![A popup warning you that a channel is password protected](https://github.com/Vendicated/Vencord/assets/44179559/2424085e-3090-4310-9d56-62667941c57c)
|
22
src/plugins/passwordProtect/components/accessModal.tsx
Normal file
22
src/plugins/passwordProtect/components/accessModal.tsx
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { openModalLazy } from "@utils/modal";
|
||||||
|
|
||||||
|
import { checkPassword } from "../data";
|
||||||
|
import { ModalType, PasswordModal } from "./modal";
|
||||||
|
|
||||||
|
export async function openAccessModal(channelId: string, cb: (success: boolean) => void) {
|
||||||
|
await openModalLazy(async () => {
|
||||||
|
return modalProps => <PasswordModal modalProps={modalProps} channelId={channelId} type={ModalType.Access} callback={async password => {
|
||||||
|
if (password) {
|
||||||
|
cb(await checkPassword(password, channelId));
|
||||||
|
} else {
|
||||||
|
cb(false);
|
||||||
|
}
|
||||||
|
}} />;
|
||||||
|
});
|
||||||
|
}
|
71
src/plugins/passwordProtect/components/contextMenus.tsx
Normal file
71
src/plugins/passwordProtect/components/contextMenus.tsx
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { findGroupChildrenByChildId, NavContextMenuPatchCallback } from "@api/ContextMenu";
|
||||||
|
import { Menu } from "@webpack/common";
|
||||||
|
|
||||||
|
import { isPasswordProtected } from "../data";
|
||||||
|
import { openLockModal } from "./lockModal";
|
||||||
|
import { openUnlockModal } from "./unlockModal";
|
||||||
|
|
||||||
|
function createPasswordItem(channelId: string) {
|
||||||
|
const isProtected = isPasswordProtected(channelId);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<Menu.MenuItem
|
||||||
|
id="password-protect"
|
||||||
|
label="Password Protect"
|
||||||
|
>
|
||||||
|
{!isProtected && (
|
||||||
|
<>
|
||||||
|
<Menu.MenuItem
|
||||||
|
id="vc-password-protect-lock"
|
||||||
|
label="Lock"
|
||||||
|
color="brand"
|
||||||
|
action={() => openLockModal(channelId)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
{isProtected && (
|
||||||
|
<>
|
||||||
|
<Menu.MenuItem
|
||||||
|
id="vc-password-protect-unlock"
|
||||||
|
label="Unlock"
|
||||||
|
color="danger"
|
||||||
|
action={() => openUnlockModal(channelId)}
|
||||||
|
/>
|
||||||
|
</>
|
||||||
|
)}
|
||||||
|
|
||||||
|
</Menu.MenuItem>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
const GroupDMContext: NavContextMenuPatchCallback = (children, props) => {
|
||||||
|
const container = findGroupChildrenByChildId("leave-channel", children);
|
||||||
|
container?.unshift(createPasswordItem(props.channel.id));
|
||||||
|
};
|
||||||
|
|
||||||
|
const UserContext: NavContextMenuPatchCallback = (children, props) => {
|
||||||
|
const container = findGroupChildrenByChildId("close-dm", children);
|
||||||
|
if (container) {
|
||||||
|
const idx = container.findIndex(c => c?.props?.id === "close-dm");
|
||||||
|
container.splice(idx, 0, createPasswordItem(props.channel.id));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const ChannelContect: NavContextMenuPatchCallback = (children, props) => {
|
||||||
|
const container = findGroupChildrenByChildId(["mute-channel", "unmute-channel"], children);
|
||||||
|
container?.unshift(createPasswordItem(props.channel.id));
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
export const contextMenus = {
|
||||||
|
"gdm-context": GroupDMContext,
|
||||||
|
"user-context": UserContext,
|
||||||
|
"channel-context": ChannelContect
|
||||||
|
};
|
24
src/plugins/passwordProtect/components/lockModal.tsx
Normal file
24
src/plugins/passwordProtect/components/lockModal.tsx
Normal file
|
@ -0,0 +1,24 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { openModalLazy } from "@utils/modal";
|
||||||
|
|
||||||
|
import { setPassword } from "../data";
|
||||||
|
import { isChannelCurrent, reloadChannel } from "../utils";
|
||||||
|
import { ModalType, PasswordModal } from "./modal";
|
||||||
|
|
||||||
|
export async function openLockModal(channelId: string) {
|
||||||
|
await openModalLazy(async () => {
|
||||||
|
return modalProps => <PasswordModal modalProps={modalProps} channelId={channelId} type={ModalType.Lock} callback={password => {
|
||||||
|
if (password) {
|
||||||
|
setPassword(channelId, password);
|
||||||
|
if (isChannelCurrent(channelId)) {
|
||||||
|
reloadChannel();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}} />;
|
||||||
|
});
|
||||||
|
}
|
91
src/plugins/passwordProtect/components/modal.tsx
Normal file
91
src/plugins/passwordProtect/components/modal.tsx
Normal file
|
@ -0,0 +1,91 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { classNameFactory } from "@api/Styles";
|
||||||
|
import { ModalContent, ModalFooter, ModalHeader, ModalProps, ModalRoot } from "@utils/modal";
|
||||||
|
import { Button, Forms, Text, TextInput, useEffect, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
import { checkPassword } from "../data";
|
||||||
|
|
||||||
|
export enum ModalType {
|
||||||
|
Lock = "Lock",
|
||||||
|
Unlock = "Unlock",
|
||||||
|
Access = "Access"
|
||||||
|
}
|
||||||
|
|
||||||
|
interface Props {
|
||||||
|
channelId: string;
|
||||||
|
type: ModalType;
|
||||||
|
callback: (password?: string) => void;
|
||||||
|
modalProps: ModalProps;
|
||||||
|
}
|
||||||
|
|
||||||
|
const cl = classNameFactory("vc-password-modal-");
|
||||||
|
|
||||||
|
export function PasswordModal({ channelId, type, callback, modalProps }: Props) {
|
||||||
|
|
||||||
|
const [password, setPassword] = useState("");
|
||||||
|
const [confirmPasswordValue, setConfirmPasswordValue] = useState("");
|
||||||
|
const [error, setError] = useState("");
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
(async () => {
|
||||||
|
if (password !== confirmPasswordValue && [ModalType.Lock, ModalType.Unlock].includes(type)) {
|
||||||
|
setError("Passwords do not match");
|
||||||
|
} else if (password === "") {
|
||||||
|
setError("Please enter a password");
|
||||||
|
}
|
||||||
|
else if (type !== ModalType.Lock && !(await checkPassword(password, channelId))) {
|
||||||
|
setError("Incorrect Password");
|
||||||
|
} else {
|
||||||
|
setError("");
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
}, [password, confirmPasswordValue]);
|
||||||
|
|
||||||
|
const onSubmit = async (e: React.FormEvent<HTMLFormElement> | React.MouseEvent<HTMLButtonElement, MouseEvent>) => {
|
||||||
|
e.preventDefault();
|
||||||
|
if (error) return callback();
|
||||||
|
modalProps.onClose();
|
||||||
|
callback(password);
|
||||||
|
};
|
||||||
|
return (
|
||||||
|
<ModalRoot {...modalProps}>
|
||||||
|
<ModalHeader>
|
||||||
|
<Text variant="heading-lg/semibold" style={{ flexGrow: 1 }}>{type}</Text>
|
||||||
|
</ModalHeader>
|
||||||
|
|
||||||
|
{/* form is here so when you press enter while in the text input it submits */}
|
||||||
|
<form onSubmit={onSubmit}>
|
||||||
|
<ModalContent className={cl("content")}>
|
||||||
|
<Forms.FormSection>
|
||||||
|
<Forms.FormTitle>Password</Forms.FormTitle>
|
||||||
|
<TextInput
|
||||||
|
type="password"
|
||||||
|
value={password}
|
||||||
|
onChange={e => setPassword(e)}
|
||||||
|
/>
|
||||||
|
</Forms.FormSection>
|
||||||
|
{[ModalType.Lock, ModalType.Unlock].includes(type) && (
|
||||||
|
<Forms.FormSection>
|
||||||
|
<Forms.FormTitle>Confirm Password</Forms.FormTitle>
|
||||||
|
<TextInput
|
||||||
|
type="password"
|
||||||
|
value={confirmPasswordValue}
|
||||||
|
onChange={e => setConfirmPasswordValue(e)}
|
||||||
|
/>
|
||||||
|
</Forms.FormSection>
|
||||||
|
)}
|
||||||
|
<Forms.FormDivider />
|
||||||
|
{error && <Text color="text-danger">{error}</Text>}
|
||||||
|
</ModalContent>
|
||||||
|
<ModalFooter>
|
||||||
|
<Button type="submit" onClick={onSubmit} disabled={!!error}>{type}</Button>
|
||||||
|
</ModalFooter>
|
||||||
|
</form>
|
||||||
|
</ModalRoot>
|
||||||
|
);
|
||||||
|
}
|
26
src/plugins/passwordProtect/components/unlockModal.tsx
Normal file
26
src/plugins/passwordProtect/components/unlockModal.tsx
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { openModalLazy } from "@utils/modal";
|
||||||
|
|
||||||
|
import { checkPassword, removePassword } from "../data";
|
||||||
|
import { isChannelCurrent, reloadChannel } from "../utils";
|
||||||
|
import { ModalType, PasswordModal } from "./modal";
|
||||||
|
|
||||||
|
export async function openUnlockModal(channelId: string) {
|
||||||
|
await openModalLazy(async () => {
|
||||||
|
return modalProps => <PasswordModal modalProps={modalProps} channelId={channelId} type={ModalType.Unlock} callback={async password => {
|
||||||
|
if (password) {
|
||||||
|
if (await checkPassword(password, channelId)) {
|
||||||
|
removePassword(channelId);
|
||||||
|
if (isChannelCurrent(channelId)) {
|
||||||
|
reloadChannel();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}} />;
|
||||||
|
});
|
||||||
|
}
|
60
src/plugins/passwordProtect/data.ts
Normal file
60
src/plugins/passwordProtect/data.ts
Normal file
|
@ -0,0 +1,60 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { DataStore } from "@api/index";
|
||||||
|
import { Channel } from "discord-types/general";
|
||||||
|
|
||||||
|
import { isChannelCurrent, reloadChannel, sha256 } from "./utils";
|
||||||
|
|
||||||
|
let data: Record<string, string> = {};
|
||||||
|
const accessedChannels: string[] = [];
|
||||||
|
|
||||||
|
export async function initData() {
|
||||||
|
const newData = await DataStore.get("passwordProtect");
|
||||||
|
if (newData) {
|
||||||
|
data = newData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function saveData() {
|
||||||
|
await DataStore.set("passwordProtect", data);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isLocked(channelId: string) {
|
||||||
|
if (accessedChannels.includes(channelId)) return false;
|
||||||
|
return isPasswordProtected(channelId);
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isPasswordProtected(channelId: string) {
|
||||||
|
return data?.[channelId] !== undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getPasswordHash(channelId: string) {
|
||||||
|
return data?.[channelId];
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function setPassword(channelId: string, password: string) {
|
||||||
|
data![channelId] = await sha256(password);
|
||||||
|
await saveData();
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function removePassword(channelId: string) {
|
||||||
|
delete data![channelId];
|
||||||
|
await saveData();
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function checkPassword(input: string, channelId: string) {
|
||||||
|
return await sha256(input) === getPasswordHash(channelId);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
export function accessChannel(channel: Channel) {
|
||||||
|
accessedChannels.push(channel.id);
|
||||||
|
if (isChannelCurrent(channel.id)) reloadChannel();
|
||||||
|
setTimeout(() => {
|
||||||
|
accessedChannels.splice(accessedChannels.indexOf(channel.id), 1);
|
||||||
|
}, 1000);
|
||||||
|
}
|
89
src/plugins/passwordProtect/index.tsx
Normal file
89
src/plugins/passwordProtect/index.tsx
Normal file
|
@ -0,0 +1,89 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import { getCurrentChannel } from "@utils/discord";
|
||||||
|
import definePlugin from "@utils/types";
|
||||||
|
|
||||||
|
import { openAccessModal } from "./components/accessModal";
|
||||||
|
import { contextMenus } from "./components/contextMenus";
|
||||||
|
import { accessChannel, initData, isLocked, isPasswordProtected, saveData } from "./data";
|
||||||
|
|
||||||
|
interface NSFWBlockProps {
|
||||||
|
title: string;
|
||||||
|
description: string;
|
||||||
|
agreement: string;
|
||||||
|
disagreement: string;
|
||||||
|
onAgree: () => void;
|
||||||
|
onDisagree: () => void;
|
||||||
|
}
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "PasswordProtect",
|
||||||
|
description: "Passcode protect servers, channels, and dms",
|
||||||
|
authors: [Devs.ImLvna],
|
||||||
|
|
||||||
|
contextMenus: contextMenus,
|
||||||
|
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: "GuildNSFWAgreeStore",
|
||||||
|
replacement: {
|
||||||
|
match: /didAgree\((\i)\){/,
|
||||||
|
replace: "$&if($self.isLocked($1)) return false; if($self.isPasswordProtected($1)) return true;"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
find: "return this.nsfw",
|
||||||
|
replacement: {
|
||||||
|
match: /return this.nsfw/,
|
||||||
|
replace: "if($self.isLocked(this.id)) return true;$&"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
find: ".gatedContent,",
|
||||||
|
replacement: {
|
||||||
|
match: /this.props/,
|
||||||
|
replace: "$self.patchProps($&)"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
patchProps(props: NSFWBlockProps) {
|
||||||
|
const channel = getCurrentChannel();
|
||||||
|
if (!isPasswordProtected(channel.id)) return props;
|
||||||
|
props.title = "This channel is password protected";
|
||||||
|
props.description = "This channel is password protected. Please enter the password to view the content.";
|
||||||
|
props.agreement = "Enter password";
|
||||||
|
props.disagreement = "Cancel";
|
||||||
|
const oldOnAgree = props.onAgree;
|
||||||
|
props.onAgree = () => {
|
||||||
|
openAccessModal(channel.id, async success => {
|
||||||
|
console.log(success);
|
||||||
|
if (success) {
|
||||||
|
accessChannel(channel);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
);
|
||||||
|
};
|
||||||
|
return props;
|
||||||
|
},
|
||||||
|
|
||||||
|
isLocked,
|
||||||
|
|
||||||
|
isPasswordProtected,
|
||||||
|
|
||||||
|
start() {
|
||||||
|
initData();
|
||||||
|
},
|
||||||
|
stop() {
|
||||||
|
saveData();
|
||||||
|
},
|
||||||
|
|
||||||
|
// navigate(href: string): boolean {
|
||||||
|
// return Math.random() < 0.5;
|
||||||
|
// },
|
||||||
|
});
|
26
src/plugins/passwordProtect/utils.ts
Normal file
26
src/plugins/passwordProtect/utils.ts
Normal file
|
@ -0,0 +1,26 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a Discord client mod
|
||||||
|
* Copyright (c) 2024 Vendicated and contributors
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*/
|
||||||
|
|
||||||
|
import { getCurrentChannel } from "@utils/discord";
|
||||||
|
import { NavigationRouter } from "@webpack/common";
|
||||||
|
|
||||||
|
export async function sha256(message) {
|
||||||
|
const buf = await crypto.subtle.digest("SHA-256", new TextEncoder().encode(message));
|
||||||
|
const array = Array.from(new Uint8Array(buf));
|
||||||
|
const str = array.map(b => b.toString(16).padStart(2, "0")).join("");
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function isChannelCurrent(channelId: string) {
|
||||||
|
return getCurrentChannel()?.id === channelId;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function reloadChannel() {
|
||||||
|
const channel = getCurrentChannel();
|
||||||
|
NavigationRouter.transitionTo("/channels/@me");
|
||||||
|
await new Promise(r => setTimeout(r, 0));
|
||||||
|
NavigationRouter.transitionTo(`/channels/${channel.guild_id || "@me"}/${channel.id}`);
|
||||||
|
}
|
Loading…
Reference in a new issue