This commit is contained in:
thororen1234 2024-07-22 00:28:27 -04:00
parent a0f3747187
commit a0eb38934c
10 changed files with 10984 additions and 0 deletions

View file

@ -0,0 +1,124 @@
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
import { definePluginSettings } from "@api/Settings";
import { EquicordDevs } from "@utils/constants";
import definePlugin, { OptionType } from "@utils/types";
import { findByProps } from "@webpack";
import { Button, Forms, i18n, Menu, TabBar } from "@webpack/common";
import { ReactElement } from "react";
import { cl, QrCodeCameraIcon } from "./ui";
import openQrModal from "./ui/modals/QrModal";
export let jsQR: (img: Uint8ClampedArray, width: number, height: number) => {
binaryData: number[];
data: string;
chunks: any[];
version: number;
location: object;
};
import("./jsQR.js").then(x => (jsQR = x.default));
export default definePlugin({
name: "LoginWithQR",
description: "Allows you to login to another device by scanning a login QR code, just like on mobile!",
authors: [EquicordDevs.nexpid],
settings: definePluginSettings({
scanQr: {
type: OptionType.COMPONENT,
description: "Scan a QR code",
component() {
if (!Vencord.Plugins.plugins.LoginWithQR.started)
return (
<Forms.FormText>
Scan a login QR code
</Forms.FormText>
);
return (
<Button size={Button.Sizes.SMALL} onClick={openQrModal}>
{i18n.Messages.USER_SETTINGS_SCAN_QR_CODE}
</Button>
);
},
},
}),
patches: [
// Prevent paste event from firing when the QRModal is open
{
find: ".clipboardData&&(",
replacement: {
// Find the handleGlobalPaste & handlePaste functions and prevent
// them from firing when the modal is open. Does this have any
// side effects? Maybe
match: /handle(Global)?Paste:(\i)(}|,)/g,
replace: "handle$1Paste:(...args)=>!$self.qrModalOpen&&$2(...args)$3",
},
},
// Insert a Scan QR Code button in the My Account tab
{
find: "UserSettingsAccountProfileCard",
replacement: {
// Find the Edit User Profile button and insert our custom button.
// A bit jank, but whatever
match: /,(.{11}\.Button,.{58}\.USER_SETTINGS_EDIT_USER_PROFILE}\))/,
replace: ",$self.insertScanQrButton($1)",
},
},
// Insert a Scan QR Code MenuItem in the simplified user popout
{
find: "Messages.MULTI_ACCOUNT_MENU_LABEL",
replacement: {
// Insert our own MenuItem before the Switch Account button
match: /children:\[(.{54}id:"switch-account")/,
replace: "children:[$self.ScanQrMenuItem,$1",
},
},
// Add a Scan QR entry to the settings TabBar
{
find: ".BILLING_SETTINGS,",
replacement: {
match: /((\i\.settings)\.forEach.+?(\i).push\(.+}\)}\))/,
replace: (_, original, settings, elements) =>
`${original},${settings}?.[0]=="ACCOUNT"` +
`&&${elements}.push({section:"CUSTOM",element:$self.ScanQrTabBarComponent})`,
},
},
],
qrModalOpen: false,
insertScanQrButton: (button: ReactElement) => (
<div className={cl("settings-btns")}>
<Button size={Button.Sizes.SMALL} onClick={openQrModal}>
{i18n.Messages.USER_SETTINGS_SCAN_QR_CODE}
</Button>
{button}
</div>
),
get ScanQrMenuItem() {
const { menuItemFocused, subMenuIcon } = findByProps("menuItemFocused") as Record<string, string>;
return (
<Menu.MenuGroup>
<Menu.MenuItem
id="scan-qr"
label={i18n.Messages.USER_SETTINGS_SCAN_QR_CODE}
icon={QrCodeCameraIcon}
action={openQrModal}
showIconFirst
focusedClassName={menuItemFocused}
subMenuIconClassName={subMenuIcon}
/>
</Menu.MenuGroup>
);
},
ScanQrTabBarComponent: () => (
<TabBar.Item id="Scan QR Code" onClick={openQrModal}>
{i18n.Messages.USER_SETTINGS_SCAN_QR_CODE}
</TabBar.Item>
),
});

File diff suppressed because it is too large Load diff

View file

@ -0,0 +1,47 @@
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
import "./style.css";
import { classNameFactory } from "@api/Styles";
import { proxyLazy } from "@utils/lazy";
import { findByPropsLazy } from "@webpack";
import { Forms } from "@webpack/common";
import { ComponentType, HTMLAttributes } from "react";
export enum SpinnerTypes {
WANDERING_CUBES = "wanderingCubes",
CHASING_DOTS = "chasingDots",
PULSING_ELLIPSIS = "pulsingEllipsis",
SPINNING_CIRCLE = "spinningCircle",
SPINNING_CIRCLE_SIMPLE = "spinningCircleSimple",
LOW_MOTION = "lowMotion",
}
type Spinner = ComponentType<Omit<HTMLAttributes<HTMLDivElement>, "children"> & {
type?: SpinnerTypes;
animated?: boolean;
className?: string;
itemClassName?: string;
"aria-label"?: string;
}> & {
Type: typeof SpinnerTypes;
};
// https://github.com/Kyuuhachi/VencordPlugins/blob/main/MessageLinkTooltip/index.tsx#L11-L33
export const { Spinner } = proxyLazy(() => Forms as any as {
Spinner: Spinner,
SpinnerTypes: typeof SpinnerTypes;
});
export const { QrCodeCameraIcon } = findByPropsLazy("QrCodeCameraIcon") as {
QrCodeCameraIcon: ComponentType<{
size: number;
}>;
};
export const cl = classNameFactory("qrlogin-");

View file

@ -0,0 +1,351 @@
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
import {
ModalContent,
ModalHeader,
ModalProps,
ModalRoot,
ModalSize,
openModal,
} from "@utils/modal";
import { findByPropsLazy } from "@webpack";
import {
Button,
i18n,
RestAPI,
Text,
useEffect,
useRef,
useState,
} from "@webpack/common";
import { jsQR } from "../../";
import { cl, Spinner, SpinnerTypes } from "..";
import openVerifyModal from "./VerifyModal";
enum LoginStateType {
Idle,
Loading,
Camera,
}
const { getVideoDeviceId } = findByPropsLazy("getVideoDeviceId");
const tokenRegex = /^https:\/\/discord\.com\/ra\/(.+)$/;
const verifyUrl = async (
token: string,
exit: (err: string | null) => void,
closeMain: () => void
) => {
// yay
let handshake: string | null = null;
try {
const res = await RestAPI.post({
url: "/users/@me/remote-auth",
body: { fingerprint: token },
});
if (res.ok && res.status === 200) handshake = res.body?.handshake_token;
} catch { }
openVerifyModal(
handshake,
() => {
exit(null);
RestAPI.post({
url: "/users/@me/remote-auth/cancel",
body: { handshake_token: handshake },
});
},
closeMain
);
};
const handleProcessImage = (
file: File,
exit: (err: string | null) => void,
closeMain: () => void
) => {
const reader = new FileReader();
reader.addEventListener("load", () => {
if (!reader.result) return;
const img = new Image();
img.addEventListener("load", () => {
const canvas = document.createElement("canvas");
canvas.width = img.width;
canvas.height = img.height;
const ctx = canvas.getContext("2d")!;
ctx.drawImage(img, 0, 0);
const { data, width, height } = ctx.getImageData(
0,
0,
canvas.width,
canvas.height
);
const code = jsQR(data, width, height);
const token = code?.data.match(tokenRegex)?.[1];
if (token)
verifyUrl(token, exit, closeMain).catch(e =>
exit(e?.message)
);
else exit(null);
canvas.remove();
});
img.src = reader.result as string;
});
reader.readAsDataURL(file);
};
function QrModal(props: ModalProps) {
const [state, setState] = useState(LoginStateType.Idle);
const inputRef = useRef<HTMLInputElement>(null);
const error = useRef<string | null>(null);
useEffect(() => {
const plugin = Vencord.Plugins.plugins.LoginWithQR as any;
plugin.qrModalOpen = true;
return () => void (plugin.qrModalOpen = false);
}, []);
useEffect(() => {
const callback = (e: ClipboardEvent) => {
e.preventDefault();
if (state !== LoginStateType.Idle || !e.clipboardData) return;
for (const item of e.clipboardData.items) {
if (item.kind === "file" && item.type.startsWith("image/")) {
setState(LoginStateType.Loading);
handleProcessImage(
item.getAsFile()!,
err => (
(error.current = err), setState(LoginStateType.Idle)
),
props.onClose
);
break;
}
}
};
if (state === LoginStateType.Idle)
document.addEventListener("paste", callback);
return () => document.removeEventListener("paste", callback);
}, [state]);
useEffect(() => {
if (state !== LoginStateType.Camera) return;
const exit = (err: string | null) => (
(error.current = err), setState(LoginStateType.Idle)
);
const video = document.createElement("video");
const canvas = document.createElement("canvas");
canvas.width = 1280;
canvas.height = 720;
const ctx = canvas.getContext("2d")!;
let stream: MediaStream;
let snapshotTimeout: number;
let stopped = false;
const stop = (stream: MediaStream) => (
stream.getTracks().forEach(track => track.stop()),
setState(LoginStateType.Idle)
);
const snapshot = () => {
ctx.drawImage(video, 0, 0, canvas.width, canvas.height);
const { data, width, height } = ctx.getImageData(
0,
0,
canvas.width,
canvas.height
);
const code = jsQR(data, width, height);
const token = code?.data.match(tokenRegex)?.[1];
if (token) {
setState(LoginStateType.Loading);
verifyUrl(token, exit, props.onClose).catch(e =>
exit(e?.message)
);
} else snapshotTimeout = setTimeout(snapshot, 1000) as any;
};
navigator.mediaDevices
.getUserMedia({
audio: false,
video: {
deviceId: getVideoDeviceId(),
width: canvas.width,
height: canvas.height,
frameRate: 5
},
})
.then(str => {
if (stopped) return stop(str);
stream = str;
video.srcObject = str;
video.addEventListener("loadedmetadata", () => {
video.play();
snapshot();
});
})
.catch(() => setState(LoginStateType.Idle));
return () => {
stopped = true;
clearTimeout(snapshotTimeout);
if (stream) stop(stream);
video.remove();
canvas.remove();
};
}, [state]);
return (
<ModalRoot size={ModalSize.DYNAMIC} {...props}>
<ModalHeader separator={false} className={cl("modal-header")}>
<Text
color="header-primary"
variant="heading-lg/semibold"
tag="h1"
style={{ flexGrow: 1 }}
>
{i18n.Messages.USER_SETTINGS_SCAN_QR_CODE}
</Text>
</ModalHeader>
<ModalContent scrollbarType="none">
<div
className={cl(
"modal-filepaste",
state === LoginStateType.Camera &&
"modal-filepaste-disabled"
)}
onClick={() =>
state === LoginStateType.Idle &&
inputRef.current?.click()
}
onDragEnter={e =>
e.currentTarget.classList.add(
cl("modal-filepaste-drop")
)
}
onDragLeave={e =>
e.currentTarget.classList.remove(
cl("modal-filepaste-drop")
)
}
onDrop={e => {
e.preventDefault();
e.currentTarget.classList.remove(
cl("modal-filepaste-drop")
);
if (state !== LoginStateType.Idle) return;
for (const item of e.dataTransfer.files) {
if (item.type.startsWith("image/")) {
setState(LoginStateType.Loading);
handleProcessImage(
item,
err => (
(error.current = err),
setState(LoginStateType.Idle)
),
props.onClose
);
break;
}
}
}}
role="button"
>
{state === LoginStateType.Loading ? (
<Spinner type={SpinnerTypes.WANDERING_CUBES} />
) : error.current ? (
<Text color="text-danger" variant="heading-md/semibold">
{error.current}
</Text>
) : state === LoginStateType.Camera ? (
<Text
color="header-primary"
variant="heading-md/semibold"
>
Scanning...
</Text>
) : (
<>
<Text
color="header-primary"
variant="heading-md/semibold"
>
Drag and drop an image here, or click to select
an image
</Text>
<Text
color="text-muted"
variant="heading-sm/medium"
>
Or paste an image from your clipboard!
</Text>
</>
)}
</div>
<input
type="file"
accept="image/*"
onChange={e => {
if (!e.target.files || state !== LoginStateType.Idle)
return;
for (const item of e.target.files) {
if (item.type.startsWith("image/")) {
setState(LoginStateType.Loading);
handleProcessImage(
item,
err => (
(error.current = err),
setState(LoginStateType.Idle)
),
props.onClose
);
break;
}
}
}}
ref={inputRef}
style={{ display: "none" }}
/>
<Button
size={Button.Sizes.MEDIUM}
className={cl("modal-button")}
disabled={state === LoginStateType.Loading}
onClick={() => {
if (state === LoginStateType.Idle)
setState(LoginStateType.Camera);
else if (state === LoginStateType.Camera)
setState(LoginStateType.Idle);
}}
>
{state === LoginStateType.Camera
? "Stop scanning"
: "Scan using webcamera"}
</Button>
</ModalContent>
</ModalRoot>
);
}
export default function openQrModal() {
return openModal(props => <QrModal {...props} />);
}

View file

@ -0,0 +1,240 @@
/*
* Vencord, a Discord client mod
* Copyright (c) 2024 Vendicated and contributors
* SPDX-License-Identifier: GPL-3.0-or-later
*/
import {
ModalContent,
ModalFooter,
ModalProps,
ModalRoot,
ModalSize,
openModal,
} from "@utils/modal";
import { findByPropsLazy } from "@webpack";
import {
Button,
i18n,
RestAPI,
Text,
useEffect,
useRef,
useState,
} from "@webpack/common";
import { cl } from "..";
const { Controller } = findByPropsLazy("Controller");
enum VerifyState {
Verifying,
LoggedIn,
NotFound,
}
export enum AbortStatus {
Abort,
Fail,
Success,
}
function VerifyModal({
token,
onAbort,
closeMain,
...props
}: {
token: string | null;
onAbort: () => void;
closeMain: () => void;
} & ModalProps) {
const [state, setState] = useState(
!token ? VerifyState.NotFound : VerifyState.Verifying
);
useEffect(() => () => void (state !== VerifyState.LoggedIn && onAbort()), []);
const [inProgress, setInProgress] = useState(false);
const buttonRef = useRef<HTMLButtonElement>();
const controllerRef = useRef(new Controller({ progress: "0%" })).current;
const holdDuration = 1000;
let timeout: any;
const startInput = () => {
if (!buttonRef.current) return;
controllerRef.start({
progress: "100%",
config: {
duration: holdDuration,
// https://easings.net/#easeInOutSine
easing: (t: number) => -(Math.cos(Math.PI * t) - 1) / 2,
},
});
timeout = setTimeout(() => {
if (state !== VerifyState.Verifying) return;
setInProgress(true);
RestAPI.post({
url: "/users/@me/remote-auth/finish",
body: {
handshake_token: token,
},
})
.then(() => {
closeMain();
setState(VerifyState.LoggedIn);
})
.catch(() => setState(VerifyState.NotFound))
.finally(() => setInProgress(false));
}, holdDuration + 250);
};
const endInput = () => {
if (!buttonRef.current) return;
controllerRef.start({
progress: "0%",
config: {
duration: 696,
// https://easings.net/#easeOutCubic
easing: (t: number) => 1 - Math.pow(1 - t, 3),
},
});
clearTimeout(timeout);
};
useEffect(() => {
let frame: number;
const update = () => {
buttonRef.current?.style.setProperty(
"--progress",
controllerRef.get().progress
);
frame = requestAnimationFrame(update);
};
if (state === VerifyState.Verifying) requestAnimationFrame(update);
return () => cancelAnimationFrame(frame);
}, [state]);
return (
<ModalRoot size={ModalSize.DYNAMIC} {...props}>
<ModalContent scrollbarType="none" className={cl("device-content")}>
{state === VerifyState.LoggedIn ? (
<>
<img
className={cl("device-image")}
src="https://github.com/nexpid/Themelings/raw/data/icons/images/native/img_remote_auth_succeeded.png"
key="img-success"
draggable={false}
/>
<Text
variant="heading-xl/bold"
color="header-primary"
tag="h1"
className={cl("device-header")}
>
{i18n.Messages.QR_CODE_LOGIN_SUCCESS}
</Text>
<Text
variant="text-md/semibold"
color="text-normal"
style={{ width: "30rem" }}
>
{i18n.Messages.QR_CODE_LOGIN_SUCCESS_FLAVOR}
</Text>
</>
) : state === VerifyState.NotFound ? (
<>
<img
className={cl("device-image")}
src="https://github.com/nexpid/Themelings/raw/data/icons/images/native/img_remote_auth_not_found.png"
key="img-not_found"
draggable={false}
/>
<Text
variant="heading-xl/bold"
color="header-primary"
tag="h1"
className={cl("device-header")}
>
{i18n.Messages.QR_CODE_NOT_FOUND}
</Text>
<Text
variant="text-md/semibold"
color="text-normal"
style={{ width: "30rem" }}
>
{i18n.Messages.QR_CODE_NOT_FOUND_DESCRIPTION}
</Text>
</>
) : (
<>
<img
className={cl("device-image")}
src="https://github.com/nexpid/Themelings/raw/data/icons/images/native/img_remote_auth_loaded.png"
key="img-loaded"
draggable={false}
/>
<Text
variant="heading-xl/bold"
color="header-primary"
tag="h1"
className={cl("device-header")}
>
{i18n.Messages.QR_CODE_LOGIN_CONFIRM}
</Text>
<Text variant="text-md/semibold" color="text-danger">
Never scan a login QR code from another user or
application.
</Text>
<Button
size={Button.Sizes.LARGE}
color={Button.Colors.RED}
className={cl("device-confirm")}
style={{
["--duration" as any]: `${holdDuration}ms`,
}}
onPointerDown={startInput}
onPointerUp={endInput}
// @ts-expect-error stop whining about refs
buttonRef={buttonRef}
disabled={inProgress}
>
Hold to confirm login
</Button>
</>
)}
</ModalContent>
<ModalFooter className={cl("device-footer")}>
{state === VerifyState.LoggedIn ? (
<Button onClick={props.onClose}>
{i18n.Messages.QR_CODE_LOGIN_FINISH_BUTTON}
</Button>
) : (
<Button
color={Button.Colors.LINK}
look={Button.Looks.LINK}
onClick={props.onClose}
>
{state === VerifyState.NotFound
? i18n.Messages.CLOSE
: i18n.Messages.CANCEL}
</Button>
)}
</ModalFooter>
</ModalRoot>
);
}
export default function openVerifyModal(
token: string | null,
onAbort: () => void,
closeMain: () => void
) {
return openModal(props => (
<VerifyModal {...props} token={token} onAbort={onAbort} closeMain={closeMain} />
));
}

View file

@ -0,0 +1,88 @@
.qrlogin-modal-header {
padding: 16px;
}
.qrlogin-modal-filepaste {
width: 30rem;
height: 13rem;
border-radius: 1rem;
background-color: var(--background-secondary);
border-style: dashed;
border-color: var(--background-modifier-accent);
border-width: 4px;
display: flex;
flex-direction: column;
align-items: center;
justify-content: center;
cursor: pointer;
overflow: hidden;
transition: 200ms background-color ease-in-out, 200ms border-color ease-in-out,
200ms opacity ease-in-out;
}
.qrlogin-modal-filepaste * {
pointer-events: none;
}
.qrlogin-modal-filepaste-drop {
background-color: var(--mention-background);
}
.qrlogin-modal-filepaste-disabled {
background-color: var(--background-secondary) !important;
opacity: 0.5;
cursor: unset;
}
.qrlogin-modal-button {
margin: 1rem 0;
width: 100%;
}
.qrlogin-device-content {
padding: 24px;
display: flex;
flex-direction: column;
justify-content: center;
align-items: center;
gap: 2px;
}
.qrlogin-device-image {
height: 8rem;
user-select: none;
}
.qrlogin-device-header {
margin: 6px 0 12px;
}
.qrlogin-device-confirm {
margin-top: 18px;
width: 20rem;
background: linear-gradient(
to right,
var(--button-danger-background) calc(var(--progress) - 1%),
var(--button-danger-background-active) var(--progress)
) !important;
--progress: 0%;
}
.qrlogin-device-footer {
justify-content: space-between;
}
.qrlogin-settings-btns {
display: flex;
align-items: center;
gap: 8px;
}
[data-tab-id="Scan QR Code"]::before {
/* stylelint-disable-next-line property-no-vendor-prefix */
-webkit-mask: var(--si-scan-qr-code) center/contain no-repeat !important;
mask: var(--si-scan-qr-code) center/contain no-repeat !important;
--si-scan-qr-code: url("")
}