feat(plugin): Image Zoom (#510)
Co-authored-by: Nuckyz <61953774+Nuckyz@users.noreply.github.com> Co-authored-by: Ven <vendicated@riseup.net>
This commit is contained in:
parent
2e6c5eacf7
commit
df7357b357
5 changed files with 504 additions and 0 deletions
198
src/plugins/imageZoom/components/Magnifier.tsx
Normal file
198
src/plugins/imageZoom/components/Magnifier.tsx
Normal file
|
@ -0,0 +1,198 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 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 { FluxDispatcher, React, useRef, useState } from "@webpack/common";
|
||||||
|
|
||||||
|
import { ELEMENT_ID } from "../constants";
|
||||||
|
import { settings } from "../index";
|
||||||
|
import { waitFor } from "../utils/waitFor";
|
||||||
|
|
||||||
|
interface Vec2 {
|
||||||
|
x: number,
|
||||||
|
y: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface MagnifierProps {
|
||||||
|
zoom: number;
|
||||||
|
size: number,
|
||||||
|
instance: any;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const Magnifier: React.FC<MagnifierProps> = ({ instance, size: initialSize, zoom: initalZoom }) => {
|
||||||
|
const [ready, setReady] = useState(false);
|
||||||
|
|
||||||
|
|
||||||
|
const [lensPosition, setLensPosition] = useState<Vec2>({ x: 0, y: 0 });
|
||||||
|
const [imagePosition, setImagePosition] = useState<Vec2>({ x: 0, y: 0 });
|
||||||
|
const [opacity, setOpacity] = useState(0);
|
||||||
|
|
||||||
|
const isShiftDown = useRef(false);
|
||||||
|
|
||||||
|
const zoom = useRef(initalZoom);
|
||||||
|
const size = useRef(initialSize);
|
||||||
|
|
||||||
|
const element = useRef<HTMLDivElement | null>(null);
|
||||||
|
const currentVideoElementRef = useRef<HTMLVideoElement | null>(null);
|
||||||
|
const originalVideoElementRef = useRef<HTMLVideoElement | null>(null);
|
||||||
|
const imageRef = useRef<HTMLImageElement | null>(null);
|
||||||
|
|
||||||
|
// since we accessing document im gonna use useLayoutEffect
|
||||||
|
React.useLayoutEffect(() => {
|
||||||
|
const onKeyDown = (e: KeyboardEvent) => {
|
||||||
|
if (e.key === "Shift") {
|
||||||
|
isShiftDown.current = true;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const onKeyUp = (e: KeyboardEvent) => {
|
||||||
|
if (e.key === "Shift") {
|
||||||
|
isShiftDown.current = false;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const syncVideos = () => {
|
||||||
|
currentVideoElementRef.current!.currentTime = originalVideoElementRef.current!.currentTime;
|
||||||
|
};
|
||||||
|
|
||||||
|
const updateMousePosition = (e: MouseEvent) => {
|
||||||
|
if (instance.state.mouseOver && instance.state.mouseDown) {
|
||||||
|
const offset = size.current / 2;
|
||||||
|
const pos = { x: e.pageX, y: e.pageY };
|
||||||
|
const x = -((pos.x - element.current!.getBoundingClientRect().left) * zoom.current - offset);
|
||||||
|
const y = -((pos.y - element.current!.getBoundingClientRect().top) * zoom.current - offset);
|
||||||
|
setLensPosition({ x: e.x - offset, y: e.y - offset });
|
||||||
|
setImagePosition({ x, y });
|
||||||
|
setOpacity(1);
|
||||||
|
} else {
|
||||||
|
setOpacity(0);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
const onMouseDown = (e: MouseEvent) => {
|
||||||
|
if (instance.state.mouseOver && e.button === 0 /* left click */) {
|
||||||
|
zoom.current = settings.store.zoom;
|
||||||
|
size.current = settings.store.size;
|
||||||
|
|
||||||
|
// close context menu if open
|
||||||
|
if (document.getElementById("image-context")) {
|
||||||
|
FluxDispatcher.dispatch({ type: "CONTEXT_MENU_CLOSE" });
|
||||||
|
}
|
||||||
|
|
||||||
|
updateMousePosition(e);
|
||||||
|
setOpacity(1);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onMouseUp = () => {
|
||||||
|
setOpacity(0);
|
||||||
|
if (settings.store.saveZoomValues) {
|
||||||
|
settings.store.zoom = zoom.current;
|
||||||
|
settings.store.size = size.current;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const onWheel = async (e: WheelEvent) => {
|
||||||
|
if (instance.state.mouseOver && instance.state.mouseDown && !isShiftDown.current) {
|
||||||
|
const val = zoom.current + ((e.deltaY / 100) * (settings.store.invertScroll ? -1 : 1)) * settings.store.zoomSpeed;
|
||||||
|
zoom.current = val <= 1 ? 1 : val;
|
||||||
|
updateMousePosition(e);
|
||||||
|
}
|
||||||
|
if (instance.state.mouseOver && instance.state.mouseDown && isShiftDown.current) {
|
||||||
|
const val = size.current + (e.deltaY * (settings.store.invertScroll ? -1 : 1)) * settings.store.zoomSpeed;
|
||||||
|
size.current = val <= 50 ? 50 : val;
|
||||||
|
updateMousePosition(e);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
waitFor(() => instance.state.readyState === "READY", () => {
|
||||||
|
const elem = document.getElementById(ELEMENT_ID) as HTMLDivElement;
|
||||||
|
element.current = elem;
|
||||||
|
elem.firstElementChild!.setAttribute("draggable", "false");
|
||||||
|
if (instance.props.animated) {
|
||||||
|
originalVideoElementRef.current = elem!.querySelector("video")!;
|
||||||
|
originalVideoElementRef.current.addEventListener("timeupdate", syncVideos);
|
||||||
|
setReady(true);
|
||||||
|
} else {
|
||||||
|
setReady(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
document.addEventListener("keydown", onKeyDown);
|
||||||
|
document.addEventListener("keyup", onKeyUp);
|
||||||
|
document.addEventListener("mousemove", updateMousePosition);
|
||||||
|
document.addEventListener("mousedown", onMouseDown);
|
||||||
|
document.addEventListener("mouseup", onMouseUp);
|
||||||
|
document.addEventListener("wheel", onWheel);
|
||||||
|
return () => {
|
||||||
|
document.removeEventListener("keydown", onKeyDown);
|
||||||
|
document.removeEventListener("keyup", onKeyUp);
|
||||||
|
document.removeEventListener("mousemove", updateMousePosition);
|
||||||
|
document.removeEventListener("mousedown", onMouseDown);
|
||||||
|
document.removeEventListener("mouseup", onMouseUp);
|
||||||
|
document.removeEventListener("wheel", onWheel);
|
||||||
|
|
||||||
|
if (settings.store.saveZoomValues) {
|
||||||
|
settings.store.zoom = zoom.current;
|
||||||
|
settings.store.size = size.current;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, []);
|
||||||
|
|
||||||
|
if (!ready) return null;
|
||||||
|
|
||||||
|
const box = element.current!.getBoundingClientRect();
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
className="lens"
|
||||||
|
style={{
|
||||||
|
opacity,
|
||||||
|
width: size.current + "px",
|
||||||
|
height: size.current + "px",
|
||||||
|
transform: `translate(${lensPosition.x}px, ${lensPosition.y}px)`,
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
{instance.props.animated ?
|
||||||
|
(
|
||||||
|
<video
|
||||||
|
ref={currentVideoElementRef}
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
left: `${imagePosition.x}px`,
|
||||||
|
top: `${imagePosition.y}px`
|
||||||
|
}}
|
||||||
|
width={`${box.width * zoom.current}px`}
|
||||||
|
height={`${box.height * zoom.current}px`}
|
||||||
|
poster={instance.props.src}
|
||||||
|
src={originalVideoElementRef.current?.src ?? instance.props.src}
|
||||||
|
autoPlay
|
||||||
|
loop
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<img
|
||||||
|
ref={imageRef}
|
||||||
|
style={{
|
||||||
|
position: "absolute",
|
||||||
|
transform: `translate(${imagePosition.x}px, ${imagePosition.y}px)`
|
||||||
|
}}
|
||||||
|
width={`${box.width * zoom.current}px`}
|
||||||
|
height={`${box.height * zoom.current}px`}
|
||||||
|
src={instance.props.src} alt=""
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
19
src/plugins/imageZoom/constants.ts
Normal file
19
src/plugins/imageZoom/constants.ts
Normal file
|
@ -0,0 +1,19 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 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 const ELEMENT_ID = "magnify-modal";
|
234
src/plugins/imageZoom/index.tsx
Normal file
234
src/plugins/imageZoom/index.tsx
Normal file
|
@ -0,0 +1,234 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 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 "./styles.css";
|
||||||
|
|
||||||
|
import { addContextMenuPatch, NavContextMenuPatchCallback, removeContextMenuPatch } from "@api/ContextMenu";
|
||||||
|
import { definePluginSettings } from "@api/settings";
|
||||||
|
import { makeRange } from "@components/PluginSettings/components";
|
||||||
|
import { Devs } from "@utils/constants";
|
||||||
|
import { debounce } from "@utils/debounce";
|
||||||
|
import definePlugin, { OptionType } from "@utils/types";
|
||||||
|
import { Menu, React, ReactDOM } from "@webpack/common";
|
||||||
|
import type { Root } from "react-dom/client";
|
||||||
|
|
||||||
|
import { Magnifier, MagnifierProps } from "./components/Magnifier";
|
||||||
|
import { ELEMENT_ID } from "./constants";
|
||||||
|
|
||||||
|
export const settings = definePluginSettings({
|
||||||
|
saveZoomValues: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Whether to save zoom and lens size values",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
preventCarouselFromClosingOnClick: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
// Thanks chat gpt
|
||||||
|
description: "Allow the image modal in the image slideshow thing / carousel to remain open when clicking on the image",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
invertScroll: {
|
||||||
|
type: OptionType.BOOLEAN,
|
||||||
|
description: "Invert scroll",
|
||||||
|
default: true,
|
||||||
|
},
|
||||||
|
|
||||||
|
zoom: {
|
||||||
|
description: "Zoom of the lens",
|
||||||
|
type: OptionType.SLIDER,
|
||||||
|
markers: makeRange(1, 50, 4),
|
||||||
|
default: 2,
|
||||||
|
stickToMarkers: false,
|
||||||
|
},
|
||||||
|
size: {
|
||||||
|
description: "Radius / Size of the lens",
|
||||||
|
type: OptionType.SLIDER,
|
||||||
|
markers: makeRange(50, 1000, 50),
|
||||||
|
default: 100,
|
||||||
|
stickToMarkers: false,
|
||||||
|
},
|
||||||
|
|
||||||
|
zoomSpeed: {
|
||||||
|
description: "How fast the zoom / lens size changes",
|
||||||
|
type: OptionType.SLIDER,
|
||||||
|
markers: makeRange(0.1, 5, 0.2),
|
||||||
|
default: 0.5,
|
||||||
|
stickToMarkers: false,
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
const imageContextMenuPatch: NavContextMenuPatchCallback = (children, _) => {
|
||||||
|
if (!children.some(child => child?.props?.id === "image-zoom")) {
|
||||||
|
children.push(
|
||||||
|
<Menu.MenuGroup id="image-zoom">
|
||||||
|
{/* thanks SpotifyControls */}
|
||||||
|
<Menu.MenuControlItem
|
||||||
|
id="zoom"
|
||||||
|
label="Zoom"
|
||||||
|
control={(props, ref) => (
|
||||||
|
<Menu.MenuSliderControl
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
minValue={1}
|
||||||
|
maxValue={50}
|
||||||
|
value={settings.store.zoom}
|
||||||
|
onChange={debounce((value: number) => settings.store.zoom = value, 100)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Menu.MenuControlItem
|
||||||
|
id="size"
|
||||||
|
label="Lens Size"
|
||||||
|
control={(props, ref) => (
|
||||||
|
<Menu.MenuSliderControl
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
minValue={50}
|
||||||
|
maxValue={1000}
|
||||||
|
value={settings.store.size}
|
||||||
|
onChange={debounce((value: number) => settings.store.size = value, 100)}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
<Menu.MenuControlItem
|
||||||
|
id="zoom-speed"
|
||||||
|
label="Zoom Speed"
|
||||||
|
control={(props, ref) => (
|
||||||
|
<Menu.MenuSliderControl
|
||||||
|
ref={ref}
|
||||||
|
{...props}
|
||||||
|
minValue={0.1}
|
||||||
|
maxValue={5}
|
||||||
|
value={settings.store.zoomSpeed}
|
||||||
|
onChange={debounce((value: number) => settings.store.zoomSpeed = value, 100)}
|
||||||
|
renderValue={(value: number) => `${value.toFixed(3)}x`}
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
/>
|
||||||
|
</Menu.MenuGroup>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export default definePlugin({
|
||||||
|
name: "ImageZoom",
|
||||||
|
description: "Lets you zoom in to images and gifs. Use scroll wheel to zoom in and shift + scroll wheel to increase lens radius / size",
|
||||||
|
authors: [Devs.Aria],
|
||||||
|
patches: [
|
||||||
|
{
|
||||||
|
find: '"renderLinkComponent","maxWidth"',
|
||||||
|
replacement: {
|
||||||
|
match: /(return\(.{1,100}\(\)\.wrapper.{1,100})(src)/,
|
||||||
|
replace: `$1id: '${ELEMENT_ID}',$2`
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
find: "handleImageLoad=",
|
||||||
|
replacement: [
|
||||||
|
{
|
||||||
|
match: /(render=function\(\){.{1,500}limitResponsiveWidth.{1,600})onMouseEnter:/,
|
||||||
|
replace: "$1...$self.makeProps(this),onMouseEnter:"
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
match: /componentDidMount=function\(\){/,
|
||||||
|
replace: "$&$self.renderMagnifier(this);",
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
match: /componentWillUnmount=function\(\){/,
|
||||||
|
replace: "$&$self.unMountMagnifier();"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
|
||||||
|
{
|
||||||
|
find: ".carouselModal,",
|
||||||
|
replacement: {
|
||||||
|
match: /onClick:(\i),/,
|
||||||
|
replace: "onClick:$self.settings.store.preventCarouselFromClosingOnClick ? () => {} : $1,"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
|
||||||
|
settings,
|
||||||
|
|
||||||
|
// to stop from rendering twice /shrug
|
||||||
|
currentMagnifierElement: null as React.FunctionComponentElement<MagnifierProps & JSX.IntrinsicAttributes> | null,
|
||||||
|
element: null as HTMLDivElement | null,
|
||||||
|
|
||||||
|
Magnifier,
|
||||||
|
root: null as Root | null,
|
||||||
|
makeProps(instance) {
|
||||||
|
return {
|
||||||
|
onMouseOver: () => this.onMouseOver(instance),
|
||||||
|
onMouseOut: () => this.onMouseOut(instance),
|
||||||
|
onMouseDown: (e: React.MouseEvent) => this.onMouseDown(e, instance),
|
||||||
|
onMouseUp: () => this.onMouseUp(instance),
|
||||||
|
id: instance.props.id,
|
||||||
|
};
|
||||||
|
},
|
||||||
|
|
||||||
|
renderMagnifier(instance) {
|
||||||
|
if (instance.props.id === ELEMENT_ID) {
|
||||||
|
if (!this.currentMagnifierElement) {
|
||||||
|
this.currentMagnifierElement = <Magnifier size={settings.store.size} zoom={settings.store.zoom} instance={instance} />;
|
||||||
|
this.root = ReactDOM.createRoot(this.element!);
|
||||||
|
this.root.render(this.currentMagnifierElement);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
unMountMagnifier() {
|
||||||
|
this.root?.unmount();
|
||||||
|
this.currentMagnifierElement = null;
|
||||||
|
this.root = null;
|
||||||
|
},
|
||||||
|
|
||||||
|
onMouseOver(instance) {
|
||||||
|
instance.setState((state: any) => ({ ...state, mouseOver: true }));
|
||||||
|
},
|
||||||
|
onMouseOut(instance) {
|
||||||
|
instance.setState((state: any) => ({ ...state, mouseOver: false }));
|
||||||
|
},
|
||||||
|
onMouseDown(e: React.MouseEvent, instance) {
|
||||||
|
if (e.button === 0 /* left */)
|
||||||
|
instance.setState((state: any) => ({ ...state, mouseDown: true }));
|
||||||
|
},
|
||||||
|
onMouseUp(instance) {
|
||||||
|
instance.setState((state: any) => ({ ...state, mouseDown: false }));
|
||||||
|
},
|
||||||
|
|
||||||
|
start() {
|
||||||
|
addContextMenuPatch("image-context", imageContextMenuPatch);
|
||||||
|
this.element = document.createElement("div");
|
||||||
|
this.element.classList.add("MagnifierContainer");
|
||||||
|
document.body.appendChild(this.element);
|
||||||
|
},
|
||||||
|
|
||||||
|
stop() {
|
||||||
|
// so componenetWillUnMount gets called if Magnifier component is still alive
|
||||||
|
this.root && this.root.unmount();
|
||||||
|
this.element?.remove();
|
||||||
|
removeContextMenuPatch("image-context", imageContextMenuPatch);
|
||||||
|
}
|
||||||
|
});
|
31
src/plugins/imageZoom/styles.css
Normal file
31
src/plugins/imageZoom/styles.css
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
.lens {
|
||||||
|
position: absolute;
|
||||||
|
inset: 0;
|
||||||
|
z-index: 9999;
|
||||||
|
border: 2px solid grey;
|
||||||
|
border-radius: 50%;
|
||||||
|
overflow: hidden;
|
||||||
|
cursor: none;
|
||||||
|
box-shadow: inset 0 0 10px 2px grey;
|
||||||
|
filter: drop-shadow(0 0 2px grey);
|
||||||
|
pointer-events: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.zoom img {
|
||||||
|
position: absolute;
|
||||||
|
top: 50%;
|
||||||
|
left: 50%;
|
||||||
|
transform: translate(-50%, -50%);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* make the carousel take up less space so we can click the backdrop and exit out of it */
|
||||||
|
[class^="focusLock"] > [class^="carouselModal"] {
|
||||||
|
height: fit-content;
|
||||||
|
box-shadow: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
[class^="focusLock"] > [class^="carouselModal"] > div {
|
||||||
|
height: fit-content;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
}
|
22
src/plugins/imageZoom/utils/waitFor.ts
Normal file
22
src/plugins/imageZoom/utils/waitFor.ts
Normal file
|
@ -0,0 +1,22 @@
|
||||||
|
/*
|
||||||
|
* Vencord, a modification for Discord's desktop app
|
||||||
|
* Copyright (c) 2023 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 waitFor(condition: () => boolean, cb: () => void) {
|
||||||
|
if (condition()) cb();
|
||||||
|
else requestAnimationFrame(() => waitFor(condition, cb));
|
||||||
|
}
|
Loading…
Reference in a new issue