Super duper alpha.. Use at your own risk. lol
The basic idea here is a "react-like" API that will create canvas "components" such that it handles:
- Events over very specific areas of of the canvas, as defined by components.
- Communicating state changes between components.
- Ensuring that only what needs to be rendered is actually rendered.
There's a playground link here on Stackblitz.
- THERE ARE NO TESTS!! (duh, huge red flag!)
- Missing events:
- onMouseDown
- onMouseUp
- onKeyPress
- onKeyDown
- onKeyUp
- touch events?
- Have yet to figure out focus management scheme
- Screen reader updates
- Components:
- Ellipse? Circle?
- Do I want to allow transformations (scale, rotate, etc) on other existing components?
See storybook examples for usage.
Basically:
- Define a component
import { defineComp, rect, text } from 'canvas-render-components';
function MyComp(props: MyCompProps, ctx: CanvasRenderingContext2D) => {
// There are hooks like React, just prefixed with `crc` instead of use:
const [count, setCount] = crcState(0);
// You can return other crc elements, like react, but JSX is annoying to hook up
// So I don't have that in this example:
return [
rect({
x: 10,
y: 10,
width: 100,
height: 100,
fillStyle: 'blue',
onClick: () => setCount(count + 1)
}),
text({
x: 10,
y: 10,
width: 100,
text: 'Click Me',
fillStyle: 'white'
})
];
}
export const myComp = defineComp(MyComp)
- Mount the component to an existing
HTMLCanvasElement
:
import { myComp } from './MyComp';
import { crc } from 'canvas-render-components';
const canvas = document.querySelector('#my-canvas-id');
crc(canvas, myComp);
Sorry, this isn't really documentation, just the basic idea:
crc(canvasElement, crcElement)
- Mount or update an existing canvas element with a crc elementdefineComp(compFn)
- used to create a more ergonomic means of consuming crc components and returning crc elements when setting up JSX is too annoying (it's always too annoying).
path
: Renders an arbitraryPath2D
rect
: Renders a rectangletext
: Text rendering (including multiline, singleline, ellipsis overflow, etc)line
: Renders a series of coordinates as connected line segmentsverticalLine
/horizontalLine
special components for rendering vertical or horizontal line segments, which includes a bit calledalignToPixelGrid
that allows you to ensure 1px lines are really 1px. (it's a canvas quirk)svgPath
: Renders svg path data as a shapeimg
: Loads and renders an imageg
: A grouping component that allows the group application of transformations such as scale, rotation, etc.clip
: A grouping component that applies a clipping path to everything rendered in itschildren
. It ALSO will "clip" events.layer
: A component for memoizing another component as a unit of render. Basically, if the props of theCompEl
passed torender
change, or if thewidth
orheight
of the layer change (it will default to the canvas width and height), it will re-render itself. Otherwise, it will render a cached image. See the storybook for example. Note that it does a shallow, reference check on the props of the element passed to render.
crcRef
- basically a simplified version of react'suseRef
crcState
- A simplified version of react'suseState
crcMemo
- Basically react'suseMemo
. This is VERY useful for memoizingPath2D
objects that need to be passed to other hooks. Strongly recommended for that use case.crcWhenChanged
- Looks like react'suseEffect
.. it is NOT. It takes a callback that will execute SYNCHRONOUSLY when dependencies change. It also allows the return of a teardown. This is specifically for use cases where one might need to execute some logic only when some dependencies change. DO NOT USE if you need to synchronously update some _state when dependencies change, usecrcMemo
instead.crcCursor
- A hook to allow the setup of CSS cursor (pointer) changes when hovering a givenPath2D
.crcEvent
- A hook for setting up events related to a particularPath2D
(or if no path is provided, the entire canvas)crcRectPath
- A simplified hook that returns a memoizedPath2D
for a rectangle (A common task).crcLinePath
- A hook for memoizedPath2D
objects from coordinates.crcSvgPath
- A hook for memoizedPath2D
objects from svg path data strings.
If you're seeing events "bleeding through" things you've rendered over top of them, you need to use the clip
component to constrain where the event is allowed to fire. Events are registered separate from rendering anything, they operate on a 2d plain of their own and don't "know" about what pixels are rendered where. The clip
component keeps track of clipping paths in a context that it will apply to events registered underneath it. Basically, if your event is registered against a path as part of a clip
components children or descendants, for the event to fire, it must match the event path AND the clipping path. Think of the clipping path as a "mask" of where events underneath it are "allowed" to fire. (It also clips what is rendered)
The last thing in a list of children will be rendered "on top". Remember that as you're rendering things.
Events are register and/or unregistered on every render. They're associated with Path2D
objects, functions that are handlers, and probably some closure and other things. The more events you have, the slower your render will be. Full stop. So, "event delegation" can be a useful tool for you. Perhaps what you need to do is use the crcEvent
hook and register one event against a compound path of some sort. Or maybe register an event against the whole canvas by passing a undefined
path to crcEvent
, and then do some of your own math to figure out if it's a hit or not. In any case, if you're registering and unregistring 1,000 event handlers on each render, it's going to add up. Don't do that.
Another technique is to have more than one canvas, one on top of the other. In this way, you can have elements of your scene that are rarely updated rendered on the "bottom" canvas, while your more frequently updated elements are rendered on the "top" canvas. This means less code being executed on each pass.