Loading...
Loading...
Learn how to create an interactive, draggable DOM using a Lit web component with CSS transforms and slots, enabling you to manipulate HTML and SVG elements within a canvas-like environment.
npx skill4agent add rodydavis/skills draggable-dom-with-litnpm init @vitejs/app --template lit-tslit-draggable-domcd lit-draggable-dom
npm i lit
npm i -D @types/node
code .vite.config.tsimport { defineConfig } from "vite";
import { resolve } from "path";
export default defineConfig({
base: "/lit-draggable-dom/",
build: {
rollupOptions: {
input: {
main: resolve(__dirname, "index.html"),
},
},
},
});index.html<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/src/favicon.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Lit Draggable DOM</title>
<style>
body {
margin: 0;
padding: 0;
width: 100%;
height: 100vh;
}
</style>
<script type="module" src="/src/draggable-dom.ts"></script>
</head>
<body>
<draggable-dom>
<img
src="https://lit.dev/images/logo.svg"
alt="Lit Logo"
width="500"
height="333"
style="--dx: 59.4909px; --dy: 32.8429px"
/>
<svg width="400" height="110" style="--dx: 230.057px; --dy: 33.6257px">
<rect
width="400"
height="100"
style="fill: rgb(0, 0, 255); stroke-width: 3; stroke: rgb(0, 0, 0)"
/>
</svg>
<svg height="100" width="100">
<circle
cx="50"
cy="50"
r="40"
stroke="black"
stroke-width="3"
fill="red"
/>
</svg>
</draggable-dom>
</body>
</html>lit-element--dx--dymy-element.tsdraggable-dom.tsdraggable-dom.tsimport { html, css, LitElement } from "lit";
import { customElement, query } from "lit/decorators.js";
type DragType = "none" | "canvas" | "element";
type SupportedNode = HTMLElement | SVGElement;
@customElement("draggable-dom")
export class CSSCanvas extends LitElement {
@query("main") root!: HTMLElement;
@query("#children") container!: HTMLElement;
@query("canvas") canvas!: HTMLCanvasElement;
dragType: DragType = "none";
offset: Offset = { x: 0, y: 0 };
pointerMap: Map<number, PointerData> = new Map();
static styles = css`
:host {
--offset-x: 0;
--offset-y: 0;
--grid-background-color: white;
--grid-color: black;
--grid-size: 40px;
--grid-dot-size: 1px;
}
main {
overflow: hidden;
}
canvas {
background-size: var(--grid-size) var(--grid-size);
background-image: radial-gradient(
circle,
var(--grid-color) var(--grid-dot-size),
var(--grid-background-color) var(--grid-dot-size)
);
background-position: var(--offset-x) var(--offset-y);
z-index: 0;
}
.full-size {
width: 100%;
height: 100%;
position: fixed;
}
.child {
--dx: 0px;
--dy: 0px;
position: fixed;
flex-shrink: 1;
z-index: var(--layer, 0);
transform: translate(var(--dx), var(--dy));
}
@media (prefers-color-scheme: dark) {
main {
--grid-background-color: black;
--grid-color: grey;
}
}
`;
render() {
return html`
<main class="full-size">
<canvas class="full-size"></canvas>
<div id="children" class="full-size"></div>
</main>
`;
}
}
interface Offset {
x: number;
y: number;
}
interface PointerData {
id: number;
startPos: Offset;
currentPos: Offset;
}maincanvasdivOffsetPointerDataasync firstUpdated() {
const items = Array.from(this.childNodes);
let i = 0;
for (const node of items) {
if (node instanceof SVGElement || node instanceof HTMLElement) {
const child = node as SupportedNode;
child.classList.add("child");
child.style.setProperty("--layer", `${i}`);
this.container.append(child);
child.addEventListener("pointerdown", (e: any) => {
// Pointer Down for Child
});
child.addEventListener("pointermove", (e: any) => {
// Pointer Move for Child
});
i++;
}
}
this.requestUpdate();
this.root.addEventListener("pointerdown", (e: any) => {
// Pointer Down for Canvas
});
this.root.addEventListener("pointermove", (e: any) => {
// Pointer Move for Canvas
});
this.root.addEventListener("pointerup", (e: any) => {
// Pointer Up for Canvas
});
}--layerz-indexhandleDown(event: PointerEvent, type: DragType) {
if (this.dragType === "none") {
event.preventDefault();
this.dragType = type;
(event.target as Element).setPointerCapture(event.pointerId);
this.pointerMap.set(event.pointerId, {
id: event.pointerId,
startPos: { x: event.clientX, y: event.clientY },
currentPos: { x: event.clientX, y: event.clientY },
});
}
}
handleMove(
event: PointerEvent,
type: DragType,
onMove: (delta: Offset) => void
) {
if (this.dragType === type) {
event.preventDefault();
const saved = this.pointerMap.get(event.pointerId)!;
const current = { ...saved.currentPos };
saved.currentPos = { x: event.clientX, y: event.clientY };
const delta = {
x: saved.currentPos.x - current.x,
y: saved.currentPos.y - current.y,
};
onMove(delta);
}
}
handleUp(event: PointerEvent) {
this.dragType = "none";
(event.target as Element).releasePointerCapture(event.pointerId);
}canvaselementmoveCanvas(delta: Offset) {
this.offset.x += delta.x;
this.offset.y += delta.y;
this.root.style.setProperty("--offset-x", `${this.offset.x}px`);
this.root.style.setProperty("--offset-y", `${this.offset.y}px`);
}
moveElement(child: SupportedNode, delta: Offset) {
const getNumber = (key: "--dx" | "--dy", fallback: number) => {
const saved = child.style.getPropertyValue(key);
if (saved.length > 0) {
return parseFloat(saved.replace("px", ""));
}
return fallback;
};
const dx = getNumber("--dx", 0) + delta.x;
const dy = getNumber("--dy", 0) + delta.y;
child.style.transform = `translate(${dx}px, ${dy}px)`;
child.style.setProperty("--dx", `${dx}px`);
child.style.setProperty("--dy", `${dy}px`);
}background-positionasync firstUpdated() {
const items = Array.from(this.childNodes);
let i = 0;
for (const node of items) {
if (node instanceof SVGElement || node instanceof HTMLElement) {
const child = node as SupportedNode;
child.classList.add("child");
child.style.setProperty("--layer", `${i}`);
this.container.append(child);
child.addEventListener("pointerdown", (e: any) => {
this.handleDown(e, "element");
});
child.addEventListener("pointermove", (e: any) => {
this.handleMove(e, "element", (delta) => {
this.moveElement(child, delta);
});
});
i++;
}
}
this.requestUpdate();
this.root.addEventListener("pointerdown", (e: any) => {
this.handleDown(e, "canvas");
});
this.root.addEventListener("pointermove", (e: any) => {
this.handleMove(e, "canvas", (delta) => {
this.moveCanvas(delta);
for (const node of Array.from(this.container.children)) {
if (node instanceof SVGElement || node instanceof HTMLElement) {
this.moveElement(node, delta);
}
}
});
});
this.root.addEventListener("pointerup", (e: any) => {
this.handleUp(e);
});
}import { html, css, LitElement } from "lit";
import { customElement, query } from "lit/decorators.js";
type DragType = "none" | "canvas" | "element";
type SupportedNode = HTMLElement | SVGElement;
@customElement("draggable-dom")
export class DraggableDOM extends LitElement {
@query("main") root!: HTMLElement;
@query("#children") container!: HTMLElement;
@query("canvas") canvas!: HTMLCanvasElement;
dragType: DragType = "none";
offset: Offset = { x: 0, y: 0 };
pointerMap: Map<number, PointerData> = new Map();
static styles = css`
:host {
--offset-x: 0;
--offset-y: 0;
--grid-background-color: white;
--grid-color: black;
--grid-size: 40px;
--grid-dot-size: 1px;
}
main {
overflow: hidden;
}
canvas {
background-size: var(--grid-size) var(--grid-size);
background-image: radial-gradient(
circle,
var(--grid-color) var(--grid-dot-size),
var(--grid-background-color) var(--grid-dot-size)
);
background-position: var(--offset-x) var(--offset-y);
z-index: 0;
}
.full-size {
width: 100%;
height: 100%;
position: fixed;
}
.child {
--dx: 0px;
--dy: 0px;
position: fixed;
flex-shrink: 1;
z-index: var(--layer, 0);
transform: translate(var(--dx), var(--dy));
}
@media (prefers-color-scheme: dark) {
main {
--grid-background-color: black;
--grid-color: grey;
}
}
`;
render() {
console.log("render");
return html`
<main class="full-size">
<canvas class="full-size"></canvas>
<div id="children" class="full-size"></div>
</main>
`;
}
handleDown(event: PointerEvent, type: DragType) {
if (this.dragType === "none") {
event.preventDefault();
this.dragType = type;
(event.target as Element).setPointerCapture(event.pointerId);
this.pointerMap.set(event.pointerId, {
id: event.pointerId,
startPos: { x: event.clientX, y: event.clientY },
currentPos: { x: event.clientX, y: event.clientY },
});
}
}
handleMove(
event: PointerEvent,
type: DragType,
onMove: (delta: Offset) => void
) {
if (this.dragType === type) {
event.preventDefault();
const saved = this.pointerMap.get(event.pointerId)!;
const current = { ...saved.currentPos };
saved.currentPos = { x: event.clientX, y: event.clientY };
const delta = {
x: saved.currentPos.x - current.x,
y: saved.currentPos.y - current.y,
};
onMove(delta);
}
}
handleUp(event: PointerEvent) {
this.dragType = "none";
(event.target as Element).releasePointerCapture(event.pointerId);
}
moveCanvas(delta: Offset) {
this.offset.x += delta.x;
this.offset.y += delta.y;
this.root.style.setProperty("--offset-x", `${this.offset.x}px`);
this.root.style.setProperty("--offset-y", `${this.offset.y}px`);
}
moveElement(child: SupportedNode, delta: Offset) {
const getNumber = (key: "--dx" | "--dy", fallback: number) => {
const saved = child.style.getPropertyValue(key);
if (saved.length > 0) {
return parseFloat(saved.replace("px", ""));
}
return fallback;
};
const dx = getNumber("--dx", 0) + delta.x;
const dy = getNumber("--dy", 0) + delta.y;
child.style.transform = `translate(${dx}px, ${dy}px)`;
child.style.setProperty("--dx", `${dx}px`);
child.style.setProperty("--dy", `${dy}px`);
}
async firstUpdated() {
const items = Array.from(this.childNodes);
let i = 0;
for (const node of items) {
if (node instanceof SVGElement || node instanceof HTMLElement) {
const child = node as SupportedNode;
child.classList.add("child");
child.style.setProperty("--layer", `${i}`);
this.container.append(child);
child.addEventListener("pointerdown", (e: any) => {
this.handleDown(e, "element");
});
child.addEventListener("pointermove", (e: any) => {
this.handleMove(e, "element", (delta) => {
this.moveElement(child, delta);
});
});
child.setAttribute("draggable", "false");
i++;
}
}
this.requestUpdate();
this.root.addEventListener("pointerdown", (e: any) => {
this.handleDown(e, "canvas");
});
this.root.addEventListener("pointermove", (e: any) => {
this.handleMove(e, "canvas", (delta) => {
this.moveCanvas(delta);
for (const node of Array.from(this.container.children)) {
if (node instanceof SVGElement || node instanceof HTMLElement) {
this.moveElement(node, delta);
}
}
});
});
this.root.addEventListener("touchstart", function (e) {
e.preventDefault();
});
this.root.addEventListener("pointerup", (e: any) => {
this.handleUp(e);
});
}
}
interface Offset {
x: number;
y: number;
}
interface PointerData {
id: number;
startPos: Offset;
currentPos: Offset;
}