Skip to content

Instantly share code, notes, and snippets.

@littensy
Last active January 11, 2025 22:48
Show Gist options
  • Save littensy/519d6b88a8aa9c9aace4da947a5fb1d8 to your computer and use it in GitHub Desktop.
Save littensy/519d6b88a8aa9c9aace4da947a5fb1d8 to your computer and use it in GitHub Desktop.
Roblox-TS React hook for responsive sizes
import { useCamera, useDebounceState, useEventListener } from "@rbxts/pretty-react-hooks";
import { useMemo } from "@rbxts/roact";
interface ScaleFunction {
/**
* Scales `pixels` based on the current viewport size and rounds the result.
*/
(pixels: number): number;
/**
* Scales `pixels` and rounds the result to the nearest even number.
*/
even: (pixels: number) => number;
/**
* Scales a number based on the current viewport size without rounding.
*/
scale: (percent: number) => number;
/**
* Scales `pixels` and rounds the result down.
*/
floor: (pixels: number) => number;
/**
* Scales `pixels` and rounds the result up.
*/
ceil: (pixels: number) => number;
}
const BASE_RESOLUTION = new Vector2(1280, 832);
const MIN_SCALE = 0.75;
const DOMINANT_AXIS = 0.5;
/**
* @see https://discord.com/channels/476080952636997633/476080952636997635/1146857136358432900
*/
function calculateScale(viewport: Vector2) {
const width = math.log(viewport.X / BASE_RESOLUTION.X, 2);
const height = math.log(viewport.Y / BASE_RESOLUTION.Y, 2);
const centered = width + (height - width) * DOMINANT_AXIS;
return math.max(2 ** centered, MIN_SCALE);
}
export function usePx(): ScaleFunction {
const camera = useCamera();
const [scale, setScale] = useDebounceState(calculateScale(camera.ViewportSize), {
wait: 0.2,
leading: true,
});
useEventListener(camera.GetPropertyChangedSignal("ViewportSize"), () => {
setScale(calculateScale(camera.ViewportSize));
});
return useMemo(() => {
const api = {
even: (value: number) => math.round(value * scale * 0.5) * 2,
scale: (value: number) => value * scale,
floor: (value: number) => math.floor(value * scale),
ceil: (value: number) => math.ceil(value * scale),
};
setmetatable(api, {
__call: (_, value) => math.round((value as number) * scale),
});
return api as ScaleFunction;
}, [scale]);
}
@flowers3176
Copy link

Maybe a fromUDim2

import { useCamera, useDebounceState, useEventListener } from "@rbxts/pretty-react-hooks";
import { useMemo } from "@rbxts/react";

interface ScaleFunction {
	/**
	 * Scales `pixels` based on the current viewport size and rounds the result.
	 */
	(pixels: number): number;
	/**
	 * Scales `pixels` and rounds the result to the nearest even number.
	 */
	even: (pixels: number) => number;
	/**
	 * Scales a number based on the current viewport size without rounding.
	 */
	scale: (percent: number) => number;
	/**
	 * Scales `pixels` and rounds the result down.
	 */
	floor: (pixels: number) => number;
	/**
	 * Scales `pixels` and rounds the result up.
	 */
	ceil: (pixels: number) => number;
	/**
	 * Scales `x` and `y` `pixels` and gives back an UDim2
	 */
	fromUDim2: (xPixels?: number, yPixels?: number) => UDim2;
}

const BASE_RESOLUTION = new Vector2(1280, 832);
const MIN_SCALE = 0.75;
const DOMINANT_AXIS = 0.5;

/**
 * @see https://discord.com/channels/476080952636997633/476080952636997635/1146857136358432900
 */
function calculateScale(viewport: Vector2) {
	const width = math.log(viewport.X / BASE_RESOLUTION.X, 2);
	const height = math.log(viewport.Y / BASE_RESOLUTION.Y, 2);
	const centered = width + (height - width) * DOMINANT_AXIS;

	return math.max(2 ** centered, MIN_SCALE);
}

export function usePx(): ScaleFunction {
	const camera = useCamera();

	const [scale, setScale] = useDebounceState(calculateScale(camera.ViewportSize), {
		wait: 0.2,
		leading: true,
	});

	useEventListener(camera.GetPropertyChangedSignal("ViewportSize"), () => {
		setScale(calculateScale(camera.ViewportSize));
	});

	return useMemo(() => {
		const api = {
			even: (value: number) => math.round(value * scale * 0.5) * 2,
			scale: (value: number) => value * scale,
			floor: (value: number) => math.floor(value * scale),
			ceil: (value: number) => math.ceil(value * scale),
			fromUDim2: (xPixels: number, yPixels?: number) =>
				UDim2.fromOffset(
					xPixels !== undefined ? xPixels * scale : 0,
					yPixels !== undefined ? yPixels * scale : 0,
				),
		};

		setmetatable(api, {
			__call: (_, value) => math.round((value as number) * scale),
		});

		return api as ScaleFunction;
	}, [scale]);
}

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment