mirror of
https://github.com/Vendicated/Vencord.git
synced 2025-04-07 15:49:30 +00:00
83 lines
3.3 KiB
TypeScript
83 lines
3.3 KiB
TypeScript
/*
|
|
* 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 { runtimeHashMessageKey } from "./intlHash";
|
|
import { Patch, PatchReplacement, ReplaceFn } from "./types";
|
|
|
|
export function canonicalizeMatch<T extends RegExp | string>(match: T): T {
|
|
let partialCanon = typeof match === "string" ? match : match.source;
|
|
partialCanon = partialCanon.replaceAll(/#{intl::([\w$+/]*)(?:::(\w+))?}/g, (_, key, modifier) => {
|
|
const hashed = modifier === "raw" ? key : runtimeHashMessageKey(key);
|
|
|
|
const isString = typeof match === "string";
|
|
const hasSpecialChars = !Number.isNaN(Number(hashed[0])) || hashed.includes("+") || hashed.includes("/");
|
|
|
|
if (hasSpecialChars) {
|
|
return isString
|
|
? `["${hashed}"]`
|
|
: String.raw`(?:\["${hashed}"\])`.replaceAll("+", "\\+");
|
|
}
|
|
|
|
return isString ? `.${hashed}` : String.raw`(?:\.${hashed})`;
|
|
});
|
|
|
|
if (typeof match === "string") {
|
|
return partialCanon as T;
|
|
}
|
|
|
|
const canonSource = partialCanon.replaceAll("\\i", String.raw`(?:[A-Za-z_$][\w$]*)`);
|
|
const canonRegex = new RegExp(canonSource, match.flags);
|
|
canonRegex.toString = match.toString.bind(match);
|
|
|
|
return canonRegex as T;
|
|
}
|
|
|
|
export function canonicalizeReplace<T extends string | ReplaceFn>(replace: T, pluginPath: string): T {
|
|
if (typeof replace !== "function")
|
|
return replace.replaceAll("$self", pluginPath) as T;
|
|
|
|
return ((...args) => replace(...args).replaceAll("$self", pluginPath)) as T;
|
|
}
|
|
|
|
export function canonicalizeDescriptor<T>(descriptor: TypedPropertyDescriptor<T>, canonicalize: (value: T) => T) {
|
|
if (descriptor.get) {
|
|
const original = descriptor.get;
|
|
descriptor.get = function () {
|
|
return canonicalize(original.call(this));
|
|
};
|
|
} else if (descriptor.value) {
|
|
descriptor.value = canonicalize(descriptor.value);
|
|
}
|
|
return descriptor;
|
|
}
|
|
|
|
export function canonicalizeReplacement(replacement: Pick<PatchReplacement, "match" | "replace">, pluginPath: string) {
|
|
const descriptors = Object.getOwnPropertyDescriptors(replacement);
|
|
descriptors.match = canonicalizeDescriptor(descriptors.match, canonicalizeMatch);
|
|
descriptors.replace = canonicalizeDescriptor(
|
|
descriptors.replace,
|
|
replace => canonicalizeReplace(replace, pluginPath),
|
|
);
|
|
Object.defineProperties(replacement, descriptors);
|
|
}
|
|
|
|
export function canonicalizeFind(patch: Patch) {
|
|
const descriptors = Object.getOwnPropertyDescriptors(patch);
|
|
descriptors.find = canonicalizeDescriptor(descriptors.find, canonicalizeMatch);
|
|
Object.defineProperties(patch, descriptors);
|
|
}
|