diff --git a/packages/client/src/components/Settings/TemplateSettings.tsx b/packages/client/src/components/Settings/TemplateSettings.tsx index 0bc7f01..3bd7ae1 100644 --- a/packages/client/src/components/Settings/TemplateSettings.tsx +++ b/packages/client/src/components/Settings/TemplateSettings.tsx @@ -1,5 +1,5 @@ import { useTemplateContext } from "../../contexts/TemplateContext"; -import { Input, Slider, Switch } from "@nextui-org/react"; +import { Input, Select, SelectItem, Slider, Switch } from "@nextui-org/react"; export const TemplateSettings = () => { const { @@ -15,6 +15,8 @@ export const TemplateSettings = () => { setY, opacity, setOpacity, + style, + setStyle, showMobileTools, setShowMobileTools, } = useTemplateContext(); @@ -70,6 +72,29 @@ export const TemplateSettings = () => { onChange={(v) => setOpacity(v as number)} getValue={(v) => v + "%"} /> + + {style !== "ONE_TO_ONE" && ( +
+ Warning: Template color picking only +
+ works with one-to-one template style +
+ )} { const { config } = useAppContext(); - const { enable, url, width, setWidth, x, y, opacity, setX, setY } = + const { enable, url, width, setWidth, x, y, opacity, setX, setY, style } = useTemplateContext(); const templateHolder = useRef(null); const instance = useRef(); @@ -87,7 +87,9 @@ export const Template = () => { useEffect(() => { if (!instance.current) { - console.warn("Received template enable but no instance exists"); + console.warn( + "[Template] Received template enable but no instance exists" + ); return; } @@ -95,7 +97,7 @@ export const Template = () => { if (enable && url) { instance.current.loadImage(url).then(() => { - console.log("enable: load image finished"); + console.log("[Template] enable: load image finished"); }); } }, [enable]); @@ -103,29 +105,29 @@ export const Template = () => { useEffect(() => { if (!instance.current) { console.warn( - "recieved template url update but no template instance exists" + "[Template] Recieved template url update but no template instance exists" ); return; } if (!url) { - console.warn("received template url blank"); + console.warn("[Template] Received template url blank"); return; } if (!enable) { - console.info("Got template URL but not enabled, ignoring"); + console.info("[Template] Got template URL but not enabled, ignoring"); return; } instance.current.loadImage(url).then(() => { - console.log("template loader finished"); + console.log("[Template] Template loader finished"); }); }, [url]); useEffect(() => { if (!instance.current) { - console.warn("received template width with no instance"); + console.warn("[Template] Received template width with no instance"); return; } @@ -133,6 +135,15 @@ export const Template = () => { instance.current.rasterizeTemplate(); }, [width]); + useEffect(() => { + if (!instance.current) { + console.warn("[Template] Received style update with no instance"); + return; + } + + instance.current.setOption("style", style); + }, [style]); + return (
{ const [x, setX] = useState(routerData.template?.x || 0); const [y, setY] = useState(routerData.template?.y || 0); const [opacity, setOpacity] = useState(100); + const [style, setStyle] = useState( + routerData.template?.style || "ONE_TO_ONE" + ); const [showMobileTools, setShowMobileTools] = useState(true); const initAt = useRef(); @@ -70,6 +76,7 @@ export const TemplateContext = ({ children }: PropsWithChildren) => { setWidth(data.template.width); setX(data.template.x || 0); setY(data.template.y || 0); + setStyle(data.template.style || "ONE_TO_ONE"); } else { setEnable(false); } @@ -89,7 +96,7 @@ export const TemplateContext = ({ children }: PropsWithChildren) => { }, []); useEffect(() => { - Router.setTemplate({ enabled: enable, width, x, y, url }); + Router.setTemplate({ enabled: enable, width, x, y, url, style }); if (!initAt.current) { console.debug("TemplateContext updating router but no initAt"); @@ -102,7 +109,7 @@ export const TemplateContext = ({ children }: PropsWithChildren) => { if (initAt.current && Date.now() - initAt.current > 2 * 1000) Router.queueUpdate(); - }, [enable, width, x, y, url]); + }, [enable, width, x, y, url, style]); return ( { setY, opacity, setOpacity, + style, + setStyle, showMobileTools, setShowMobileTools, }} diff --git a/packages/client/src/lib/router.ts b/packages/client/src/lib/router.ts index ad84582..79497d7 100644 --- a/packages/client/src/lib/router.ts +++ b/packages/client/src/lib/router.ts @@ -2,6 +2,7 @@ import { PanZoom } from "@sc07-canvas/lib/src/renderer/PanZoom"; import { Canvas } from "./canvas"; import throttle from "lodash.throttle"; import EventEmitter from "eventemitter3"; +import { TemplateStyle, TemplateStyles } from "./template"; const CLIENT_PARAMS = { canvas_x: "x", @@ -11,6 +12,7 @@ const CLIENT_PARAMS = { template_width: "tw", template_x: "tx", template_y: "ty", + template_style: "ts", }; export interface IRouterData { @@ -24,6 +26,7 @@ export interface IRouterData { width?: number; x?: number; y?: number; + style?: TemplateStyle; }; } @@ -41,10 +44,12 @@ class _Router extends EventEmitter { x: number; y: number; url?: string; + style: TemplateStyle; } = { enabled: false, x: 0, y: 0, + style: "ONE_TO_ONE", }; constructor() { @@ -105,6 +110,8 @@ class _Router extends EventEmitter { params.set(CLIENT_PARAMS.template_width, this.templateState.width + ""); params.set(CLIENT_PARAMS.template_x, this.templateState.x + ""); params.set(CLIENT_PARAMS.template_y, this.templateState.y + ""); + if (this.templateState.style) + params.set(CLIENT_PARAMS.template_style, this.templateState.style + ""); } return ( @@ -161,6 +168,7 @@ class _Router extends EventEmitter { width?: number; x?: number; y?: number; + style?: TemplateStyle; } | undefined = undefined; @@ -190,6 +198,14 @@ class _Router extends EventEmitter { template.y = y; } } + + if (params.has(CLIENT_PARAMS.template_style)) { + let style = params.get(CLIENT_PARAMS.template_style); + + if (style && TemplateStyles.indexOf(style) > -1) { + template.style = style as any; + } + } } return { @@ -208,6 +224,7 @@ class _Router extends EventEmitter { x: number; y: number; url?: string; + style: TemplateStyle; }) { this.templateState = args; } diff --git a/packages/client/src/lib/template.ts b/packages/client/src/lib/template.ts index 3dd392e..4770fdd 100644 --- a/packages/client/src/lib/template.ts +++ b/packages/client/src/lib/template.ts @@ -15,18 +15,30 @@ interface TemplateEvents { interface ITemplateOptions { enable: boolean; width?: number; - style: TemplateStyle; + style: keyof typeof TemplateStyle; } -enum TemplateStyle { - SOURCE = "", - ONE_TO_ONE = "", - ONE_TO_ONE_INCORRECT = "", - DOTTED_SMALL = "", - DOTTED_BIG = "", - SYMBOLS = "", - NUMBERS = "", -} +const TemplateStyle = { + SOURCE: "", + ONE_TO_ONE: + "", + ONE_TO_ONE_INCORRECT: + "", + DOTTED_SMALL: + "", + DOTTED_BIG: + "", + SYMBOLS: + "", + NUMBERS: + "", +}; + +export type TemplateStyle = keyof typeof TemplateStyle; +export const TemplateStyles = Object.keys(TemplateStyle); + +const STYLES_Y = 16; +const STYLES_X = 16; export class Template extends EventEmitter { static instance: Template; @@ -41,7 +53,7 @@ export class Template extends EventEmitter { options: ITemplateOptions = { enable: false, - style: TemplateStyle.ONE_TO_ONE, + style: "ONE_TO_ONE", }; constructor(config: ClientConfig, templateHolder: HTMLDivElement) { @@ -67,7 +79,11 @@ export class Template extends EventEmitter { this.$style = document.createElement("img"); this.$style.setAttribute("crossorigin", ""); - this.$style.setAttribute("src", this.options!.style); + this.$style.setAttribute("src", TemplateStyle[this.options!.style]); + this.$style.addEventListener("load", () => { + console.log("[Template] Style loaded"); + this.loadStyle(); + }); this.$canvas = document.createElement("canvas"); @@ -108,6 +124,18 @@ export class Template extends EventEmitter { case "enable": this.setElementVisible([this.$canvas], !!value); break; + case "style": + if ((value as keyof typeof TemplateStyle) in TemplateStyle) { + const key = value as keyof typeof TemplateStyle; + + this.$style.setAttribute("src", TemplateStyle[key]); + this.$imageLoader.style.display = key === "SOURCE" ? "block" : "none"; + + if (key === "SOURCE") { + this.stylizeTemplate(); + } + } + break; } this.emit("option", key, value); @@ -208,15 +236,27 @@ export class Template extends EventEmitter { width: this.$imageLoader.naturalWidth, height: this.$imageLoader.naturalHeight, }; + let style = { + width: this.$style.naturalWidth / STYLES_X, + height: this.$style.naturalHeight / STYLES_Y, + }; let aspectRatio = source.height / source.width; + let display = { + width: Math.round(this.options?.width || source.width), + height: Math.round((this.options?.width || source.width) * aspectRatio), + }; + let internal = { + width: display.width * style.width, + height: display.height * style.height, + }; + return { source, - display: { - width: Math.round(this.options?.width || source.width), - height: Math.round((this.options?.width || source.width) * aspectRatio), - }, + style, + display, + internal, aspectRatio, }; } @@ -240,15 +280,14 @@ export class Template extends EventEmitter { } = { downscaling: {} } as any; updateSize() { - const { - display: { width, height }, - } = this.getDimentions(); + const { display, internal } = this.getDimentions(); - this.$wrapper.style.width = width + "px"; - this.$imageLoader.style.width = width + "px"; + this.$wrapper.style.width = display.width + "px"; + this.$imageLoader.style.width = display.width + "px"; + this.$canvas.style.width = display.width + "px"; - this.$canvas.width = width; - this.$canvas.height = height; + this.$canvas.width = internal.width; + this.$canvas.height = internal.height; } /** @@ -261,8 +300,6 @@ export class Template extends EventEmitter { const palette: { value: string }[] = this.config.pallete.colors.map( (color) => ({ value: color.hex }) ); - const STYLES_Y = 16; - const STYLES_X = 16; const context = this.$canvas.getContext("webgl", { premultipliedAlpha: true, @@ -589,11 +626,9 @@ export class Template extends EventEmitter { stylizeTemplate() { this.updateSize(); - const { - display: { width, height }, - } = this.getDimentions(); + const { internal, display } = this.getDimentions(); - if (this.context == null || width === 0 || height === 0) { + if (this.context == null || internal.width === 0 || internal.height === 0) { return; } @@ -602,14 +637,14 @@ export class Template extends EventEmitter { this.framebuffers.main ); this.context.clear(this.context.COLOR_BUFFER_BIT); - this.context.viewport(0, 0, width, height); + this.context.viewport(0, 0, internal.width, internal.height); this.context.useProgram(this.programs.stylize); this.context.uniform2f( this.context.getUniformLocation(this.programs.stylize, "u_TexelSize"), - 1 / width, - 1 / height + 1 / display.width, + 1 / display.height ); this.context.activeTexture(this.context.TEXTURE0);