Compare commits
14 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
ef217b1c40
|
|||
|
7499b80789
|
|||
|
a5b663f6fc
|
|||
|
05506704bf
|
|||
|
63188e57fd
|
|||
|
4572d30005
|
|||
|
ccc376d158
|
|||
|
7e432e9033
|
|||
|
01f58377c2
|
|||
|
6ef5dc28ed
|
|||
|
3450d70047
|
|||
|
731b9e9b1e
|
|||
|
72f07d0a50
|
|||
|
a56e8f445e
|
@@ -0,0 +1,28 @@
|
|||||||
|
name: Setup
|
||||||
|
description: Restore caches and install pnpm dependencies (run after checkout)
|
||||||
|
|
||||||
|
runs:
|
||||||
|
using: composite
|
||||||
|
steps:
|
||||||
|
- name: 💾 Setup pnpm Cache
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: .pnpm-store
|
||||||
|
key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-pnpm-
|
||||||
|
|
||||||
|
- name: 🦀 Cache Cargo
|
||||||
|
uses: actions/cache@v4
|
||||||
|
with:
|
||||||
|
path: |
|
||||||
|
~/.cargo/registry
|
||||||
|
~/.cargo/git
|
||||||
|
target
|
||||||
|
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
||||||
|
restore-keys: |
|
||||||
|
${{ runner.os }}-cargo-
|
||||||
|
|
||||||
|
- name: 📦 Install Dependencies
|
||||||
|
shell: bash
|
||||||
|
run: pnpm install --frozen-lockfile --store-dir .pnpm-store
|
||||||
@@ -12,7 +12,7 @@ env:
|
|||||||
CARGO_TARGET_DIR: target
|
CARGO_TARGET_DIR: target
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
benchmark:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
||||||
|
|
||||||
@@ -23,29 +23,10 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
token: ${{ secrets.GITEA_TOKEN }}
|
token: ${{ secrets.GITEA_TOKEN }}
|
||||||
|
|
||||||
- name: 💾 Setup pnpm Cache
|
- name: 🔧 Setup
|
||||||
uses: actions/cache@v4
|
uses: ./.gitea/actions/setup
|
||||||
with:
|
|
||||||
path: ${{ env.PNPM_CACHE_FOLDER }}
|
|
||||||
key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-pnpm-
|
|
||||||
|
|
||||||
- name: 🦀 Cache Cargo
|
- name: 🛠️ Build Nodes
|
||||||
uses: actions/cache@v4
|
|
||||||
with:
|
|
||||||
path: |
|
|
||||||
~/.cargo/registry
|
|
||||||
~/.cargo/git
|
|
||||||
target
|
|
||||||
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-cargo-
|
|
||||||
|
|
||||||
- name: 📦 Install Dependencies
|
|
||||||
run: pnpm install --frozen-lockfile --store-dir ${{ env.PNPM_CACHE_FOLDER }}
|
|
||||||
|
|
||||||
- name: 🛠️Build Nodes
|
|
||||||
run: pnpm build:nodes
|
run: pnpm build:nodes
|
||||||
|
|
||||||
- name: 🏃 Execute Runtime
|
- name: 🏃 Execute Runtime
|
||||||
@@ -56,6 +37,12 @@ jobs:
|
|||||||
mkdir -p ~/.ssh
|
mkdir -p ~/.ssh
|
||||||
echo "${{ secrets.GIT_SSH_PRIVATE_KEY }}" > ~/.ssh/id_ed25519
|
echo "${{ secrets.GIT_SSH_PRIVATE_KEY }}" > ~/.ssh/id_ed25519
|
||||||
chmod 600 ~/.ssh/id_ed25519
|
chmod 600 ~/.ssh/id_ed25519
|
||||||
|
cat >> ~/.ssh/config <<'EOF'
|
||||||
|
Host git.max-richter.dev
|
||||||
|
Port 2222
|
||||||
|
IdentityFile ~/.ssh/id_ed25519
|
||||||
|
IdentitiesOnly yes
|
||||||
|
EOF
|
||||||
ssh-keyscan -p 2222 -H git.max-richter.dev >> ~/.ssh/known_hosts
|
ssh-keyscan -p 2222 -H git.max-richter.dev >> ~/.ssh/known_hosts
|
||||||
|
|
||||||
- name: 📤 Push Results
|
- name: 📤 Push Results
|
||||||
@@ -65,22 +52,16 @@ jobs:
|
|||||||
git config --global user.name "nodarium-bot"
|
git config --global user.name "nodarium-bot"
|
||||||
git config --global user.email "nodarium-bot@max-richter.dev"
|
git config --global user.email "nodarium-bot@max-richter.dev"
|
||||||
|
|
||||||
# 2. Clone the benchmarks repo into a temp folder
|
|
||||||
git config --global core.sshCommand "ssh -p 2222 -i ~/.ssh/id_ed25519 -o IdentitiesOnly=yes"
|
|
||||||
git clone git@git.max-richter.dev:max/nodarium-benchmarks.git target_bench_repo
|
git clone git@git.max-richter.dev:max/nodarium-benchmarks.git target_bench_repo
|
||||||
|
|
||||||
# 3. Create a directory structure based on the branch
|
BRANCH="${GITHUB_HEAD_REF:-$GITHUB_REF_NAME}"
|
||||||
# This allows the UI to "switch between branches"
|
SAFE_PR_NAME=$(printf "%s" "$BRANCH" | tr '/' '-')
|
||||||
SAFE_PR_NAME=$(printf "%s" "$GITHUB_HEAD_REF" | tr '/' '-')
|
|
||||||
DEST_DIR="target_bench_repo/data/$SAFE_PR_NAME/$(date +%s)"
|
DEST_DIR="target_bench_repo/data/$SAFE_PR_NAME/$(date +%s)"
|
||||||
mkdir -p "$DEST_DIR"
|
mkdir -p "$DEST_DIR"
|
||||||
|
|
||||||
# 4. Copy the new results
|
|
||||||
# Assuming your bench tool outputs a file named 'results.json'
|
|
||||||
cp app/benchmark/out/*.json "$DEST_DIR/"
|
cp app/benchmark/out/*.json "$DEST_DIR/"
|
||||||
|
|
||||||
# 5. Commit and Push
|
|
||||||
cd target_bench_repo
|
cd target_bench_repo
|
||||||
git add .
|
git add .
|
||||||
git commit -m "Update benchmarks for $SAFE_PR_NAME: ${{ github.sha }}"
|
git commit -m "Update benchmarks for $SAFE_PR_NAME: ${{ gitea.sha }}"
|
||||||
git push origin main
|
git push origin main
|
||||||
|
|||||||
@@ -13,7 +13,7 @@ env:
|
|||||||
CARGO_TARGET_DIR: target
|
CARGO_TARGET_DIR: target
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
release:
|
quality:
|
||||||
runs-on: ubuntu-latest
|
runs-on: ubuntu-latest
|
||||||
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
||||||
|
|
||||||
@@ -24,27 +24,8 @@ jobs:
|
|||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
token: ${{ secrets.GITEA_TOKEN }}
|
token: ${{ secrets.GITEA_TOKEN }}
|
||||||
|
|
||||||
- name: 💾 Setup pnpm Cache
|
- name: 🔧 Setup
|
||||||
uses: actions/cache@v4
|
uses: ./.gitea/actions/setup
|
||||||
with:
|
|
||||||
path: ${{ env.PNPM_CACHE_FOLDER }}
|
|
||||||
key: ${{ runner.os }}-pnpm-${{ hashFiles('**/pnpm-lock.yaml') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-pnpm-
|
|
||||||
|
|
||||||
- name: 🦀 Cache Cargo
|
|
||||||
uses: actions/cache@v4
|
|
||||||
with:
|
|
||||||
path: |
|
|
||||||
~/.cargo/registry
|
|
||||||
~/.cargo/git
|
|
||||||
target
|
|
||||||
key: ${{ runner.os }}-cargo-${{ hashFiles('**/Cargo.lock') }}
|
|
||||||
restore-keys: |
|
|
||||||
${{ runner.os }}-cargo-
|
|
||||||
|
|
||||||
- name: 📦 Install Dependencies
|
|
||||||
run: pnpm install --frozen-lockfile --store-dir ${{ env.PNPM_CACHE_FOLDER }}
|
|
||||||
|
|
||||||
- name: 🧹 Quality Control
|
- name: 🧹 Quality Control
|
||||||
run: |
|
run: |
|
||||||
@@ -52,7 +33,58 @@ jobs:
|
|||||||
pnpm format:check
|
pnpm format:check
|
||||||
pnpm check
|
pnpm check
|
||||||
pnpm build
|
pnpm build
|
||||||
xvfb-run --auto-servernum --server-args="-screen 0 1280x1024x24" pnpm test
|
|
||||||
|
test-unit:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: 📑 Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
token: ${{ secrets.GITEA_TOKEN }}
|
||||||
|
|
||||||
|
- name: 🔧 Setup
|
||||||
|
uses: ./.gitea/actions/setup
|
||||||
|
|
||||||
|
- name: 🧪 Run Tests
|
||||||
|
run: xvfb-run --auto-servernum --server-args="-screen 0 1280x1024x24" pnpm test:unit
|
||||||
|
|
||||||
|
test-e2e:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: 📑 Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
token: ${{ secrets.GITEA_TOKEN }}
|
||||||
|
|
||||||
|
- name: 🔧 Setup
|
||||||
|
uses: ./.gitea/actions/setup
|
||||||
|
|
||||||
|
- name: 🧪 Run Tests
|
||||||
|
run: xvfb-run --auto-servernum --server-args="-screen 0 1280x1024x24" pnpm test:e2e
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
needs: [quality, test-e2e, test-unit]
|
||||||
|
container: git.max-richter.dev/max/nodarium-ci:a56e8f445edb6064ae7a7b3b783fb7445f1b4e69
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: 📑 Checkout Code
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
fetch-depth: 0
|
||||||
|
token: ${{ secrets.GITEA_TOKEN }}
|
||||||
|
|
||||||
|
- name: 🔧 Setup
|
||||||
|
uses: ./.gitea/actions/setup
|
||||||
|
|
||||||
|
- name: 🏗️ Build Web Assets
|
||||||
|
run: pnpm build
|
||||||
|
|
||||||
- name: 🚀 Create Release Commit
|
- name: 🚀 Create Release Commit
|
||||||
if: gitea.ref_type == 'tag'
|
if: gitea.ref_type == 'tag'
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ ENV RUSTUP_HOME=/usr/local/rustup \
|
|||||||
PATH=/usr/local/cargo/bin:$PATH
|
PATH=/usr/local/cargo/bin:$PATH
|
||||||
|
|
||||||
RUN apt-get update && apt-get install -y \
|
RUN apt-get update && apt-get install -y \
|
||||||
|
openssh-client \
|
||||||
ca-certificates=20230311+deb12u1 \
|
ca-certificates=20230311+deb12u1 \
|
||||||
gpg=2.2.40-1.1+deb12u2 \
|
gpg=2.2.40-1.1+deb12u2 \
|
||||||
gpg-agent=2.2.40-1.1+deb12u2 \
|
gpg-agent=2.2.40-1.1+deb12u2 \
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
import { describe, expect, it } from 'vitest';
|
import { assert, describe, expect, it } from 'vitest';
|
||||||
import { GraphManager } from './graph-manager.svelte';
|
import { GraphManager } from './graph-manager.svelte';
|
||||||
import {
|
import {
|
||||||
createMockNodeRegistry,
|
createMockNodeRegistry,
|
||||||
@@ -9,8 +9,151 @@ import {
|
|||||||
mockVec3OutputNode
|
mockVec3OutputNode
|
||||||
} from './test-utils';
|
} from './test-utils';
|
||||||
|
|
||||||
describe('GraphManager', () => {
|
describe('groupNodes', () => {
|
||||||
describe('getPossibleSockets', () => {
|
it('should not do anything if no nodes are selected', () => {
|
||||||
|
const registry = createMockNodeRegistry([
|
||||||
|
mockFloatOutputNode,
|
||||||
|
mockFloatInputNode,
|
||||||
|
mockGeometryOutputNode,
|
||||||
|
mockPathInputNode
|
||||||
|
]);
|
||||||
|
|
||||||
|
const manager = new GraphManager(registry);
|
||||||
|
|
||||||
|
const floatInputNode = manager.createNode({
|
||||||
|
type: 'test/node/input',
|
||||||
|
position: [100, 100],
|
||||||
|
props: {}
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.isDefined(floatInputNode);
|
||||||
|
|
||||||
|
const floatOutputNode = manager.createNode({
|
||||||
|
type: 'test/node/output',
|
||||||
|
position: [0, 0],
|
||||||
|
props: {}
|
||||||
|
});
|
||||||
|
assert.isDefined(floatOutputNode);
|
||||||
|
|
||||||
|
const edge = manager.createEdge(floatInputNode, 0, floatOutputNode, 'input');
|
||||||
|
assert.isDefined(edge);
|
||||||
|
manager.save();
|
||||||
|
|
||||||
|
manager.groupNodes([]);
|
||||||
|
|
||||||
|
const graph = manager.serialize();
|
||||||
|
expect(graph.nodes.length).toBe(2);
|
||||||
|
expect(graph.edges.length).toBe(1);
|
||||||
|
expect(graph.groups.length).toBe(0);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should group selected nodes and create a group node', () => {
|
||||||
|
const registry = createMockNodeRegistry([
|
||||||
|
mockFloatOutputNode,
|
||||||
|
mockFloatInputNode,
|
||||||
|
mockGeometryOutputNode,
|
||||||
|
mockPathInputNode
|
||||||
|
]);
|
||||||
|
|
||||||
|
const manager = new GraphManager(registry);
|
||||||
|
|
||||||
|
const floatInputNode = manager.createNode({
|
||||||
|
type: 'test/node/input',
|
||||||
|
position: [100, 100],
|
||||||
|
props: {}
|
||||||
|
});
|
||||||
|
|
||||||
|
assert.isDefined(floatInputNode);
|
||||||
|
|
||||||
|
const floatOutputNode = manager.createNode({
|
||||||
|
type: 'test/node/output',
|
||||||
|
position: [0, 0],
|
||||||
|
props: {}
|
||||||
|
});
|
||||||
|
assert.isDefined(floatOutputNode);
|
||||||
|
|
||||||
|
const edge = manager.createEdge(floatInputNode, 0, floatOutputNode, 'input');
|
||||||
|
assert.isDefined(edge);
|
||||||
|
manager.save();
|
||||||
|
|
||||||
|
const groupNode = manager.groupNodes([floatInputNode.id]);
|
||||||
|
assert.isDefined(groupNode);
|
||||||
|
|
||||||
|
const graph = manager.serialize();
|
||||||
|
|
||||||
|
expect(graph.nodes.map(n => n.id), 'graph to contain group node').to.contain(groupNode.id);
|
||||||
|
expect(graph.groups[0].nodes.map(n => n.id), 'group graph to contain float node').to.contain(
|
||||||
|
floatInputNode.id
|
||||||
|
);
|
||||||
|
expect(graph.nodes.map(n => n.id)).not.to.contain(floatInputNode.id);
|
||||||
|
|
||||||
|
expect(graph.nodes.length).toBe(2);
|
||||||
|
expect(graph.edges.length).toBe(1);
|
||||||
|
expect(graph.groups.length).toBe(1);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should rewire external edges when grouping a middle node in a chain', () => {
|
||||||
|
const registry = createMockNodeRegistry([mockFloatOutputNode, mockFloatInputNode]);
|
||||||
|
const manager = new GraphManager(registry);
|
||||||
|
|
||||||
|
// A → B → C (float chain: output → middle → input)
|
||||||
|
const nodeA = manager.createNode({ type: 'test/node/output', position: [0, 0], props: {} });
|
||||||
|
const nodeB = manager.createNode({ type: 'test/node/output', position: [100, 0], props: {} });
|
||||||
|
const nodeC = manager.createNode({ type: 'test/node/input', position: [200, 0], props: {} });
|
||||||
|
|
||||||
|
assert.isDefined(nodeA);
|
||||||
|
assert.isDefined(nodeB);
|
||||||
|
assert.isDefined(nodeC);
|
||||||
|
|
||||||
|
manager.createEdge(nodeA, 0, nodeB, 'input');
|
||||||
|
manager.createEdge(nodeB, 0, nodeC, 'value');
|
||||||
|
|
||||||
|
const groupNode = manager.groupNodes([nodeB.id]);
|
||||||
|
assert.isDefined(groupNode);
|
||||||
|
|
||||||
|
const graph = manager.serialize();
|
||||||
|
|
||||||
|
// Top-level: A, C, groupNode — B is gone
|
||||||
|
expect(graph.nodes.length, 'top-level node count').toBe(3);
|
||||||
|
const topLevelIds = graph.nodes.map(n => n.id);
|
||||||
|
expect(topLevelIds).toContain(nodeA.id);
|
||||||
|
expect(topLevelIds).toContain(nodeC.id);
|
||||||
|
expect(topLevelIds).toContain(groupNode.id);
|
||||||
|
expect(topLevelIds).not.toContain(nodeB.id);
|
||||||
|
|
||||||
|
// Both original edges survive, now routing through the group node
|
||||||
|
expect(graph.edges.length, 'edge count unchanged').toBe(2);
|
||||||
|
const edgeSources = graph.edges.map(e => e[0]);
|
||||||
|
const edgeTargets = graph.edges.map(e => e[2]);
|
||||||
|
expect(edgeTargets).toContain(groupNode.id); // A → groupNode
|
||||||
|
expect(edgeSources).toContain(groupNode.id); // groupNode → C
|
||||||
|
|
||||||
|
// One group definition was created
|
||||||
|
expect(graph.groups.length).toBe(1);
|
||||||
|
const group = graph.groups[0];
|
||||||
|
|
||||||
|
// Group contains B plus the two boundary nodes
|
||||||
|
const groupNodeIds = group.nodes.map(n => n.id);
|
||||||
|
expect(groupNodeIds).toContain(nodeB.id);
|
||||||
|
const inputBoundary = group.nodes.find(n => n.type === '__internal/group/input');
|
||||||
|
const outputBoundary = group.nodes.find(n => n.type === '__internal/group/output');
|
||||||
|
expect(inputBoundary, 'group input boundary node').toBeDefined();
|
||||||
|
expect(outputBoundary, 'group output boundary node').toBeDefined();
|
||||||
|
|
||||||
|
// Group declares one input slot and one output slot
|
||||||
|
expect(Object.keys(group.inputs ?? {}).length, 'group input count').toBe(1);
|
||||||
|
expect(group.outputs?.length, 'group output count').toBe(1);
|
||||||
|
|
||||||
|
// Internal edges wire: inputBoundary → B → outputBoundary
|
||||||
|
expect(group.edges.length, 'internal edge count').toBe(2);
|
||||||
|
const internalSources = group.edges.map(e => e[0]);
|
||||||
|
const internalTargets = group.edges.map(e => e[2]);
|
||||||
|
expect(internalTargets).toContain(nodeB.id);
|
||||||
|
expect(internalSources).toContain(nodeB.id);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
describe('getPossibleSockets', () => {
|
||||||
describe('when dragging an output socket', () => {
|
describe('when dragging an output socket', () => {
|
||||||
it('should return compatible input sockets based on type', () => {
|
it('should return compatible input sockets based on type', () => {
|
||||||
const registry = createMockNodeRegistry([
|
const registry = createMockNodeRegistry([
|
||||||
@@ -261,5 +404,4 @@ describe('GraphManager', () => {
|
|||||||
expect(possibleSocketsForFloat.map(([n]) => n.id)).toContain(floatInputNode!.id);
|
expect(possibleSocketsForFloat.map(([n]) => n.id)).toContain(floatInputNode!.id);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
|
||||||
});
|
});
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -5,7 +5,7 @@ import { SvelteMap, SvelteSet } from 'svelte/reactivity';
|
|||||||
import type { OrthographicCamera, Vector3 } from 'three';
|
import type { OrthographicCamera, Vector3 } from 'three';
|
||||||
import type { GraphManager } from './graph-manager.svelte';
|
import type { GraphManager } from './graph-manager.svelte';
|
||||||
import { ColorGenerator } from './graph/colors';
|
import { ColorGenerator } from './graph/colors';
|
||||||
import { getNodeHeight, getSocketPosition } from './helpers/nodeHelpers';
|
import { getNodeHeight, getParameterHeight } from './helpers/nodeHelpers';
|
||||||
|
|
||||||
const graphStateKey = Symbol('graph-state');
|
const graphStateKey = Symbol('graph-state');
|
||||||
export function getGraphState() {
|
export function getGraphState() {
|
||||||
@@ -99,9 +99,6 @@ export class GraphState {
|
|||||||
edges: [number, number, number, string][];
|
edges: [number, number, number, string][];
|
||||||
} = null;
|
} = null;
|
||||||
|
|
||||||
// Saved camera position per group so re-entering restores where you left off
|
|
||||||
groupCameras = new Map<string, [number, number, number]>();
|
|
||||||
|
|
||||||
cameraBounds = $derived([
|
cameraBounds = $derived([
|
||||||
this.cameraPosition[0] - this.width / this.cameraPosition[2] / 2,
|
this.cameraPosition[0] - this.width / this.cameraPosition[2] / 2,
|
||||||
this.cameraPosition[0] + this.width / this.cameraPosition[2] / 2,
|
this.cameraPosition[0] + this.width / this.cameraPosition[2] / 2,
|
||||||
@@ -155,10 +152,6 @@ export class GraphState {
|
|||||||
this.edges.delete(edgeId);
|
this.edges.delete(edgeId);
|
||||||
}
|
}
|
||||||
|
|
||||||
getEdgeData() {
|
|
||||||
return this.edges;
|
|
||||||
}
|
|
||||||
|
|
||||||
updateNodePosition(node: NodeInstance) {
|
updateNodePosition(node: NodeInstance) {
|
||||||
if (
|
if (
|
||||||
node.state.x === node.position[0]
|
node.state.x === node.position[0]
|
||||||
@@ -193,29 +186,6 @@ export class GraphState {
|
|||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
tryConnectToDebugNode(nodeId: number) {
|
|
||||||
const node = this.graph.nodes.get(nodeId);
|
|
||||||
if (!node) return;
|
|
||||||
if (node.type.endsWith('/debug')) return;
|
|
||||||
if (!node.state.type?.outputs?.length) return;
|
|
||||||
for (const _node of this.graph.nodes.values()) {
|
|
||||||
if (_node.type.endsWith('/debug')) {
|
|
||||||
this.graph.createEdge(node, 0, _node, 'input');
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const debugNode = this.graph.createNode({
|
|
||||||
type: 'max/plantarium/debug',
|
|
||||||
position: [node.position[0] + 30, node.position[1]],
|
|
||||||
props: {}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (debugNode) {
|
|
||||||
this.graph.createEdge(node, 0, debugNode, 'input');
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
copyNodes() {
|
copyNodes() {
|
||||||
if (this.activeNodeId === -1 && !this.selectedNodes?.size) {
|
if (this.activeNodeId === -1 && !this.selectedNodes?.size) {
|
||||||
return;
|
return;
|
||||||
@@ -243,6 +213,10 @@ export class GraphState {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
groupSelectedNodes() {
|
||||||
|
return this.graph.groupNodes([...this.selectedNodes.keys(), this.activeNodeId]);
|
||||||
|
}
|
||||||
|
|
||||||
centerNode(node?: NodeInstance) {
|
centerNode(node?: NodeInstance) {
|
||||||
const average = [0, 0, 4];
|
const average = [0, 0, 4];
|
||||||
if (node) {
|
if (node) {
|
||||||
@@ -304,7 +278,7 @@ export class GraphState {
|
|||||||
if (edge[3] === index) {
|
if (edge[3] === index) {
|
||||||
node = edge[0];
|
node = edge[0];
|
||||||
index = edge[1];
|
index = edge[1];
|
||||||
position = getSocketPosition(node, index);
|
position = this.getSocketPosition(node, index);
|
||||||
this.graph.removeEdge(edge);
|
this.graph.removeEdge(edge);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -324,7 +298,7 @@ export class GraphState {
|
|||||||
return {
|
return {
|
||||||
node,
|
node,
|
||||||
index,
|
index,
|
||||||
position: getSocketPosition(node, index)
|
position: this.getSocketPosition(node, index)
|
||||||
};
|
};
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -361,7 +335,8 @@ export class GraphState {
|
|||||||
for (const node of this.graph.nodes.values()) {
|
for (const node of this.graph.nodes.values()) {
|
||||||
const x = node.position[0];
|
const x = node.position[0];
|
||||||
const y = node.position[1];
|
const y = node.position[1];
|
||||||
const height = getNodeHeight(node.state.type!);
|
const nodeType = this.graph.getNodeType(node);
|
||||||
|
const height = nodeType ? getNodeHeight(nodeType) : 20;
|
||||||
if (downX > x && downX < x + 20 && downY > y && downY < y + height) {
|
if (downX > x && downX < x + 20 && downY > y && downY < y + height) {
|
||||||
clickedNodeId = node.id;
|
clickedNodeId = node.id;
|
||||||
break;
|
break;
|
||||||
@@ -373,7 +348,8 @@ export class GraphState {
|
|||||||
}
|
}
|
||||||
|
|
||||||
isNodeInView(node: NodeInstance) {
|
isNodeInView(node: NodeInstance) {
|
||||||
const height = getNodeHeight(node.state.type!);
|
if (!node) return false;
|
||||||
|
const height = getNodeHeight(this.graph.getNodeType(node)!);
|
||||||
const width = 20;
|
const width = 20;
|
||||||
return node.position[0] > this.cameraBounds[0] - width
|
return node.position[0] > this.cameraBounds[0] - width
|
||||||
&& node.position[0] < this.cameraBounds[1]
|
&& node.position[0] < this.cameraBounds[1]
|
||||||
@@ -384,4 +360,51 @@ export class GraphState {
|
|||||||
openNodePalette() {
|
openNodePalette() {
|
||||||
this.addMenuPosition = [this.mousePosition[0], this.mousePosition[1]];
|
this.addMenuPosition = [this.mousePosition[0], this.mousePosition[1]];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
enterGroupNode() {
|
||||||
|
if (this.activeNodeId === -1) return;
|
||||||
|
const node = this.graph.getNode(this.activeNodeId);
|
||||||
|
if (!node || node.type !== '__internal/group/instance') return;
|
||||||
|
const ok = this.graph.enterGroup(this.activeNodeId, [...this.cameraPosition]);
|
||||||
|
if (ok) {
|
||||||
|
this.activeNodeId = -1;
|
||||||
|
this.clearSelection();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
exitGroupNode() {
|
||||||
|
const result = this.graph.exitGroup();
|
||||||
|
if (!result) return;
|
||||||
|
this.cameraPosition = result.camera;
|
||||||
|
this.activeNodeId = -1;
|
||||||
|
this.clearSelection();
|
||||||
|
}
|
||||||
|
|
||||||
|
getSocketPosition(
|
||||||
|
node: NodeInstance,
|
||||||
|
index: string | number
|
||||||
|
): [number, number] {
|
||||||
|
if (typeof index === 'number') {
|
||||||
|
return [
|
||||||
|
(node?.state?.x ?? node.position[0]) + 20,
|
||||||
|
(node?.state?.y ?? node.position[1]) + 2.5 + 10 * index
|
||||||
|
];
|
||||||
|
} else {
|
||||||
|
let height = 5;
|
||||||
|
const nodeType = this.graph.getNodeType(node)!;
|
||||||
|
const inputs = nodeType.inputs || {};
|
||||||
|
for (const inputKey in inputs) {
|
||||||
|
const h = getParameterHeight(nodeType, inputKey) / 10;
|
||||||
|
if (inputKey === index) {
|
||||||
|
height += h / 2;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
height += h;
|
||||||
|
}
|
||||||
|
return [
|
||||||
|
node?.state?.x ?? node.position[0],
|
||||||
|
(node?.state?.y ?? node.position[1]) + height
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -11,7 +11,6 @@
|
|||||||
import Debug from '../debug/Debug.svelte';
|
import Debug from '../debug/Debug.svelte';
|
||||||
import EdgeEl from '../edges/Edge.svelte';
|
import EdgeEl from '../edges/Edge.svelte';
|
||||||
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
||||||
import { getSocketPosition } from '../helpers/nodeHelpers';
|
|
||||||
import NodeEl from '../node/Node.svelte';
|
import NodeEl from '../node/Node.svelte';
|
||||||
import { maxZoom, minZoom } from './constants';
|
import { maxZoom, minZoom } from './constants';
|
||||||
import { FileDropEventManager } from './drop.events';
|
import { FileDropEventManager } from './drop.events';
|
||||||
@@ -39,8 +38,8 @@
|
|||||||
return [0, 0, 0, 0];
|
return [0, 0, 0, 0];
|
||||||
}
|
}
|
||||||
|
|
||||||
const pos1 = getSocketPosition(fromNode, edge[1]);
|
const pos1 = graphState.getSocketPosition(fromNode, edge[1]);
|
||||||
const pos2 = getSocketPosition(toNode, edge[3]);
|
const pos2 = graphState.getSocketPosition(toNode, edge[3]);
|
||||||
return [pos1[0], pos1[1], pos2[0], pos2[1]];
|
return [pos1[0], pos1[1], pos2[0], pos2[1]];
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -97,13 +96,12 @@
|
|||||||
}
|
}
|
||||||
|
|
||||||
function getSocketType(node: NodeInstance, index: number | string): string {
|
function getSocketType(node: NodeInstance, index: number | string): string {
|
||||||
|
const nodeType = graph.getNodeType(node);
|
||||||
|
console.log({ nodeType, index });
|
||||||
if (typeof index === 'string') {
|
if (typeof index === 'string') {
|
||||||
return node.state.type?.inputs?.[index].type || 'unknown';
|
return nodeType?.inputs?.[index].type || 'unknown';
|
||||||
}
|
}
|
||||||
if (node.type === '__virtual/group/instance') {
|
return nodeType?.outputs?.[index] || 'unknown';
|
||||||
index += 1;
|
|
||||||
}
|
|
||||||
return node.state.type?.outputs?.[index] || 'unknown';
|
|
||||||
}
|
}
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
@@ -172,6 +170,14 @@
|
|||||||
{/if}
|
{/if}
|
||||||
|
|
||||||
{#if graph.status === 'idle'}
|
{#if graph.status === 'idle'}
|
||||||
|
{#if graph.isInsideGroup}
|
||||||
|
<HTML transform={false}>
|
||||||
|
<button class="exit-group" onclick={() => graphState.exitGroupNode()}>
|
||||||
|
↑ Exit Group
|
||||||
|
</button>
|
||||||
|
</HTML>
|
||||||
|
{/if}
|
||||||
|
|
||||||
{#if graphState.addMenuPosition}
|
{#if graphState.addMenuPosition}
|
||||||
<AddMenu
|
<AddMenu
|
||||||
onnode={handleNodeCreation}
|
onnode={handleNodeCreation}
|
||||||
@@ -219,10 +225,10 @@
|
|||||||
style:transform={`scale(${graphState.cameraPosition[2] * 0.1})`}
|
style:transform={`scale(${graphState.cameraPosition[2] * 0.1})`}
|
||||||
class:hovering-sockets={graphState.activeSocket}
|
class:hovering-sockets={graphState.activeSocket}
|
||||||
>
|
>
|
||||||
{#each graph.nodes.values() as node (node.id)}
|
{#each graph.nodeArray as node, index (node.id)}
|
||||||
<NodeEl
|
<NodeEl
|
||||||
{node}
|
bind:node={graph.nodeArray[index]}
|
||||||
inView={graphState.isNodeInView(node)}
|
inView={node ? graphState.isNodeInView(node) : false}
|
||||||
/>
|
/>
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
@@ -247,6 +253,26 @@
|
|||||||
height: 100%;
|
height: 100%;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
:global(.exit-group) {
|
||||||
|
position: fixed;
|
||||||
|
top: 12px;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%);
|
||||||
|
z-index: 1000;
|
||||||
|
padding: 4px 12px;
|
||||||
|
background: var(--color-layer-2);
|
||||||
|
border: 1px solid var(--stroke);
|
||||||
|
border-radius: 4px;
|
||||||
|
color: inherit;
|
||||||
|
font-size: 0.85em;
|
||||||
|
cursor: pointer;
|
||||||
|
opacity: 0.85;
|
||||||
|
}
|
||||||
|
|
||||||
|
:global(.exit-group:hover) {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
.wrapper {
|
.wrapper {
|
||||||
position: absolute;
|
position: absolute;
|
||||||
z-index: 100;
|
z-index: 100;
|
||||||
|
|||||||
@@ -1,6 +1,7 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { createKeyMap } from '$lib/helpers/createKeyMap';
|
import { createKeyMap } from '$lib/helpers/createKeyMap';
|
||||||
import type { Graph, NodeInstance, NodeRegistry } from '@nodarium/types';
|
import type { Graph, NodeInstance, NodeRegistry } from '@nodarium/types';
|
||||||
|
import { onMount } from 'svelte';
|
||||||
import { GraphManager } from '../graph-manager.svelte';
|
import { GraphManager } from '../graph-manager.svelte';
|
||||||
import { GraphState, setGraphManager, setGraphState } from '../graph-state.svelte';
|
import { GraphState, setGraphManager, setGraphState } from '../graph-state.svelte';
|
||||||
import { setupKeymaps } from '../keymaps';
|
import { setupKeymaps } from '../keymaps';
|
||||||
@@ -83,101 +84,10 @@
|
|||||||
manager.on('save', (save) => onsave?.(save));
|
manager.on('save', (save) => onsave?.(save));
|
||||||
|
|
||||||
$effect(() => {
|
$effect(() => {
|
||||||
if (graph) {
|
if (graph && (manager.status !== 'idle' || manager.graph.id !== graph.id)) {
|
||||||
manager.load(graph);
|
manager.load(graph);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
function navigateToBreadcrumb(index: number) {
|
|
||||||
const crumbs = manager.breadcrumbs;
|
|
||||||
const depth = crumbs.length - 1 - index;
|
|
||||||
let result: { camera: [number, number, number]; nodeId: number } | false = false;
|
|
||||||
for (let i = 0; i < depth; i++) {
|
|
||||||
const groupId = manager.currentGroupContext;
|
|
||||||
if (groupId) {
|
|
||||||
state.groupCameras.set(groupId, [...state.cameraPosition] as [number, number, number]);
|
|
||||||
}
|
|
||||||
result = manager.exitGroup();
|
|
||||||
}
|
|
||||||
if (result !== false) {
|
|
||||||
state.activeNodeId = result.nodeId;
|
|
||||||
state.clearSelection();
|
|
||||||
state.cameraPosition[0] = result.camera[0];
|
|
||||||
state.cameraPosition[1] = result.camera[1];
|
|
||||||
state.cameraPosition[2] = result.camera[2];
|
|
||||||
} else {
|
|
||||||
state.activeNodeId = -1;
|
|
||||||
state.clearSelection();
|
|
||||||
state.centerNode();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#if manager.isInsideGroup}
|
|
||||||
<div class="breadcrumb-bar">
|
|
||||||
{#each manager.breadcrumbs as crumb, i}
|
|
||||||
{#if i > 0}
|
|
||||||
<span class="sep">›</span>
|
|
||||||
{/if}
|
|
||||||
<button
|
|
||||||
class="crumb"
|
|
||||||
class:active={i === manager.breadcrumbs.length - 1}
|
|
||||||
onclick={() => navigateToBreadcrumb(i)}
|
|
||||||
>
|
|
||||||
{crumb.name}
|
|
||||||
</button>
|
|
||||||
{/each}
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<GraphEl {keymap} {safePadding} />
|
<GraphEl {keymap} {safePadding} />
|
||||||
|
|
||||||
<style>
|
|
||||||
.breadcrumb-bar {
|
|
||||||
position: absolute;
|
|
||||||
top: 8px;
|
|
||||||
left: 50%;
|
|
||||||
transform: translateX(-50%);
|
|
||||||
z-index: 100;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
background: rgba(10, 15, 28, 0.85);
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
|
||||||
border-radius: 6px;
|
|
||||||
padding: 4px 10px;
|
|
||||||
font-size: 12px;
|
|
||||||
pointer-events: all;
|
|
||||||
backdrop-filter: blur(8px);
|
|
||||||
}
|
|
||||||
|
|
||||||
.sep {
|
|
||||||
opacity: 0.4;
|
|
||||||
font-size: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.crumb {
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
color: rgba(255, 255, 255, 0.6);
|
|
||||||
cursor: pointer;
|
|
||||||
padding: 2px 4px;
|
|
||||||
border-radius: 4px;
|
|
||||||
font-size: 12px;
|
|
||||||
transition: color 0.15s, background 0.15s;
|
|
||||||
}
|
|
||||||
|
|
||||||
.crumb:hover {
|
|
||||||
color: white;
|
|
||||||
background: rgba(255, 255, 255, 0.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
.crumb.active {
|
|
||||||
color: white;
|
|
||||||
cursor: default;
|
|
||||||
}
|
|
||||||
|
|
||||||
.crumb.active:hover {
|
|
||||||
background: none;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
@@ -190,7 +190,7 @@ export class MouseEventManager {
|
|||||||
// if we clicked on a node
|
// if we clicked on a node
|
||||||
if (clickedNodeId !== -1) {
|
if (clickedNodeId !== -1) {
|
||||||
if (event.ctrlKey && event.shiftKey) {
|
if (event.ctrlKey && event.shiftKey) {
|
||||||
this.state.tryConnectToDebugNode(clickedNodeId);
|
this.graph.tryConnectToDebugNode(clickedNodeId);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (this.state.activeNodeId === -1) {
|
if (this.state.activeNodeId === -1) {
|
||||||
|
|||||||
@@ -3,9 +3,6 @@ import type { NodeDefinition, NodeInstance } from '@nodarium/types';
|
|||||||
export function getParameterHeight(node: NodeDefinition, inputKey: string) {
|
export function getParameterHeight(node: NodeDefinition, inputKey: string) {
|
||||||
const input = node.inputs?.[inputKey];
|
const input = node.inputs?.[inputKey];
|
||||||
if (!input) {
|
if (!input) {
|
||||||
if (inputKey.startsWith('__virtual')) {
|
|
||||||
return 50;
|
|
||||||
}
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -26,58 +23,14 @@ export function getParameterHeight(node: NodeDefinition, inputKey: string) {
|
|||||||
return 50;
|
return 50;
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getSocketPosition(
|
|
||||||
node: NodeInstance,
|
|
||||||
index: string | number
|
|
||||||
): [number, number] {
|
|
||||||
if (typeof index === 'number') {
|
|
||||||
if (node.type === '__virtual/group/input') {
|
|
||||||
const nodeType = node.state.type;
|
|
||||||
const keys = Object.keys(nodeType?.inputs || {});
|
|
||||||
let height = 5;
|
|
||||||
for (let i = 0; i < keys.length; i++) {
|
|
||||||
const h = getParameterHeight(nodeType!, keys[i]) / 10;
|
|
||||||
if (i === index) { height += h / 2; break; }
|
|
||||||
height += h;
|
|
||||||
}
|
|
||||||
return [
|
|
||||||
(node?.state?.x ?? node.position[0]) + 20,
|
|
||||||
(node?.state?.y ?? node.position[1]) + height
|
|
||||||
];
|
|
||||||
}
|
|
||||||
return [
|
|
||||||
(node?.state?.x ?? node.position[0]) + 20,
|
|
||||||
(node?.state?.y ?? node.position[1]) + 2.5 + 10 * index
|
|
||||||
];
|
|
||||||
} else {
|
|
||||||
let height = 5;
|
|
||||||
const nodeType = node.state.type!;
|
|
||||||
const inputs = nodeType.inputs || {};
|
|
||||||
for (const inputKey in inputs) {
|
|
||||||
const h = getParameterHeight(nodeType, inputKey) / 10;
|
|
||||||
if (inputKey === index) {
|
|
||||||
height += h / 2;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
height += h;
|
|
||||||
}
|
|
||||||
return [
|
|
||||||
node?.state?.x ?? node.position[0],
|
|
||||||
(node?.state?.y ?? node.position[1]) + height
|
|
||||||
];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const nodeHeightCache: Record<string, number> = {};
|
const nodeHeightCache: Record<string, number> = {};
|
||||||
export function getNodeHeight(node: NodeDefinition) {
|
export function getNodeHeight(node: NodeDefinition) {
|
||||||
// Don't cache virtual nodes — their inputs can change dynamically
|
if (!node || !('inputs' in node)) {
|
||||||
const isVirtual = (node.id as string).startsWith('__virtual/');
|
|
||||||
if (!isVirtual && node.id in nodeHeightCache) {
|
|
||||||
return nodeHeightCache[node.id];
|
|
||||||
}
|
|
||||||
if (!node?.inputs) {
|
|
||||||
return 5;
|
return 5;
|
||||||
}
|
}
|
||||||
|
if (node.id in nodeHeightCache) {
|
||||||
|
return nodeHeightCache[node.id];
|
||||||
|
}
|
||||||
let height = 5;
|
let height = 5;
|
||||||
|
|
||||||
for (const key in node.inputs) {
|
for (const key in node.inputs) {
|
||||||
@@ -85,8 +38,6 @@ export function getNodeHeight(node: NodeDefinition) {
|
|||||||
height += h;
|
height += h;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!isVirtual) {
|
|
||||||
nodeHeightCache[node.id] = height;
|
nodeHeightCache[node.id] = height;
|
||||||
}
|
|
||||||
return height;
|
return height;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -45,26 +45,12 @@ export function setupKeymaps(keymap: Keymap, graph: GraphManager, graphState: Gr
|
|||||||
|
|
||||||
keymap.addShortcut({
|
keymap.addShortcut({
|
||||||
key: 'Escape',
|
key: 'Escape',
|
||||||
description: 'Deselect nodes / Exit group',
|
description: 'Deselect nodes',
|
||||||
callback: () => {
|
callback: () => {
|
||||||
if (graph.isInsideGroup) {
|
if (graph.isInsideGroup) {
|
||||||
const groupId = graph.currentGroupContext;
|
graphState.exitGroupNode();
|
||||||
if (groupId) {
|
|
||||||
graphState.groupCameras.set(
|
|
||||||
groupId,
|
|
||||||
[...graphState.cameraPosition] as [number, number, number]
|
|
||||||
);
|
|
||||||
}
|
|
||||||
const result = graph.exitGroup();
|
|
||||||
if (result !== false) {
|
|
||||||
graphState.activeNodeId = result.nodeId;
|
|
||||||
graphState.clearSelection();
|
|
||||||
graphState.cameraPosition[0] = result.camera[0];
|
|
||||||
graphState.cameraPosition[1] = result.camera[1];
|
|
||||||
graphState.cameraPosition[2] = result.camera[2];
|
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
}
|
|
||||||
graphState.activeNodeId = -1;
|
graphState.activeNodeId = -1;
|
||||||
graphState.clearSelection();
|
graphState.clearSelection();
|
||||||
graphState.edgeEndPosition = null;
|
graphState.edgeEndPosition = null;
|
||||||
@@ -72,6 +58,21 @@ export function setupKeymaps(keymap: Keymap, graph: GraphManager, graphState: Gr
|
|||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
keymap.addShortcut({
|
||||||
|
key: 'g',
|
||||||
|
ctrl: true,
|
||||||
|
preventDefault: true,
|
||||||
|
description: 'Group selected nodes',
|
||||||
|
callback: () => graphState.groupSelectedNodes()
|
||||||
|
});
|
||||||
|
|
||||||
|
keymap.addShortcut({
|
||||||
|
key: 'Tab',
|
||||||
|
preventDefault: true,
|
||||||
|
description: 'Enter selected node group',
|
||||||
|
callback: () => graphState.enterGroupNode()
|
||||||
|
});
|
||||||
|
|
||||||
keymap.addShortcut({
|
keymap.addShortcut({
|
||||||
key: 'A',
|
key: 'A',
|
||||||
shift: true,
|
shift: true,
|
||||||
@@ -177,80 +178,4 @@ export function setupKeymaps(keymap: Keymap, graph: GraphManager, graphState: Gr
|
|||||||
if (!edge) graph.smartConnect(nodes[1], nodes[0]);
|
if (!edge) graph.smartConnect(nodes[1], nodes[0]);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
keymap.addShortcut({
|
|
||||||
key: 'g',
|
|
||||||
ctrl: true,
|
|
||||||
preventDefault: true,
|
|
||||||
description: 'Group selected nodes',
|
|
||||||
callback: () => {
|
|
||||||
if (!graphState.isBodyFocused()) return;
|
|
||||||
const nodeIds = Array.from(
|
|
||||||
new Set([
|
|
||||||
...(graphState.selectedNodes.size > 0 ? graphState.selectedNodes.values() : []),
|
|
||||||
...(graphState.activeNodeId !== -1 ? [graphState.activeNodeId] : [])
|
|
||||||
])
|
|
||||||
);
|
|
||||||
if (nodeIds.length === 0) return;
|
|
||||||
const groupNode = graph.createGroup(nodeIds);
|
|
||||||
if (groupNode) {
|
|
||||||
graphState.selectedNodes.clear();
|
|
||||||
graphState.activeNodeId = groupNode.id;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
keymap.addShortcut({
|
|
||||||
key: 'g',
|
|
||||||
alt: true,
|
|
||||||
shift: true,
|
|
||||||
preventDefault: true,
|
|
||||||
description: 'Ungroup selected node',
|
|
||||||
callback: () => {
|
|
||||||
if (!graphState.isBodyFocused()) return;
|
|
||||||
const nodeId = graphState.activeNodeId !== -1
|
|
||||||
? graphState.activeNodeId
|
|
||||||
: graphState.selectedNodes.size === 1
|
|
||||||
? [...graphState.selectedNodes.values()][0]
|
|
||||||
: -1;
|
|
||||||
if (nodeId === -1) return;
|
|
||||||
graph.ungroup(nodeId);
|
|
||||||
graphState.activeNodeId = -1;
|
|
||||||
graphState.clearSelection();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
keymap.addShortcut({
|
|
||||||
key: 'Tab',
|
|
||||||
preventDefault: true,
|
|
||||||
description: 'Enter focused group node',
|
|
||||||
callback: () => {
|
|
||||||
if (!graphState.isBodyFocused()) return;
|
|
||||||
const entered = graph.enterGroup(
|
|
||||||
graphState.activeNodeId,
|
|
||||||
[...graphState.cameraPosition] as [number, number, number]
|
|
||||||
);
|
|
||||||
if (entered) {
|
|
||||||
graphState.activeNodeId = -1;
|
|
||||||
graphState.clearSelection();
|
|
||||||
// Restore group-specific camera if we've been here before, else snap to center
|
|
||||||
const groupId = graph.currentGroupContext;
|
|
||||||
const saved = groupId ? graphState.groupCameras.get(groupId) : undefined;
|
|
||||||
if (saved) {
|
|
||||||
graphState.cameraPosition[0] = saved[0];
|
|
||||||
graphState.cameraPosition[1] = saved[1];
|
|
||||||
graphState.cameraPosition[2] = saved[2];
|
|
||||||
} else {
|
|
||||||
const nodes = [...graph.nodes.values()];
|
|
||||||
if (nodes.length) {
|
|
||||||
const avgX = nodes.reduce((s, n) => s + n.position[0], 0) / nodes.length;
|
|
||||||
const avgY = nodes.reduce((s, n) => s + n.position[1], 0) / nodes.length;
|
|
||||||
graphState.cameraPosition[0] = avgX;
|
|
||||||
graphState.cameraPosition[1] = avgY;
|
|
||||||
graphState.cameraPosition[2] = 10;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,13 +3,14 @@
|
|||||||
import type { NodeInstance } from '@nodarium/types';
|
import type { NodeInstance } from '@nodarium/types';
|
||||||
import { T } from '@threlte/core';
|
import { T } from '@threlte/core';
|
||||||
import { type Mesh } from 'three';
|
import { type Mesh } from 'three';
|
||||||
import { getGraphState } from '../graph-state.svelte';
|
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
||||||
import { colors } from '../graph/colors.svelte';
|
import { colors } from '../graph/colors.svelte';
|
||||||
import { getNodeHeight, getParameterHeight } from '../helpers/nodeHelpers';
|
import { getNodeHeight, getParameterHeight } from '../helpers/nodeHelpers';
|
||||||
import NodeFrag from './Node.frag';
|
import NodeFrag from './Node.frag';
|
||||||
import NodeVert from './Node.vert';
|
import NodeVert from './Node.vert';
|
||||||
import NodeHtml from './NodeHTML.svelte';
|
import NodeHtml from './NodeHTML.svelte';
|
||||||
|
|
||||||
|
const graph = getGraphManager();
|
||||||
const graphState = getGraphState();
|
const graphState = getGraphState();
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
@@ -18,7 +19,7 @@
|
|||||||
};
|
};
|
||||||
let { node = $bindable(), inView }: Props = $props();
|
let { node = $bindable(), inView }: Props = $props();
|
||||||
|
|
||||||
const nodeType = $derived(node.state.type!);
|
const nodeType = $derived(graph.getNodeType(node)!);
|
||||||
|
|
||||||
const isActive = $derived(graphState.activeNodeId === node.id);
|
const isActive = $derived(graphState.activeNodeId === node.id);
|
||||||
const isSelected = $derived(graphState.selectedNodes.has(node.id));
|
const isSelected = $derived(graphState.selectedNodes.has(node.id));
|
||||||
@@ -33,14 +34,14 @@
|
|||||||
|
|
||||||
const sectionHeights = $derived(
|
const sectionHeights = $derived(
|
||||||
Object
|
Object
|
||||||
.keys(nodeType.inputs || {})
|
.keys(nodeType?.inputs || {})
|
||||||
.map(key => getParameterHeight(nodeType, key) / 10)
|
.map(key => getParameterHeight(nodeType, key) / 10)
|
||||||
.filter(b => !!b)
|
.filter(b => !!b)
|
||||||
);
|
);
|
||||||
|
|
||||||
let meshRef: Mesh | undefined = $state();
|
let meshRef: Mesh | undefined = $state();
|
||||||
|
|
||||||
const height = $derived(getNodeHeight(node.state.type!));
|
const height = $derived(nodeType ? getNodeHeight(nodeType) : 20);
|
||||||
|
|
||||||
const zoom = $derived(graphState.cameraPosition[2]);
|
const zoom = $derived(graphState.cameraPosition[2]);
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1,13 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { NodeDefinition, NodeInstance } from '@nodarium/types';
|
import type { NodeInstance } from '@nodarium/types';
|
||||||
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
||||||
import NodeHeader from './NodeHeader.svelte';
|
import NodeHeader from './NodeHeader.svelte';
|
||||||
import NodeParameter from './NodeParameter.svelte';
|
import NodeParameter from './NodeParameter.svelte';
|
||||||
|
|
||||||
let ref: HTMLDivElement;
|
let ref: HTMLDivElement;
|
||||||
|
|
||||||
|
const graph = getGraphManager();
|
||||||
const graphState = getGraphState();
|
const graphState = getGraphState();
|
||||||
const manager = getGraphManager();
|
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
node: NodeInstance;
|
node: NodeInstance;
|
||||||
@@ -31,60 +31,14 @@
|
|||||||
const zOffset = Math.random() - 0.5;
|
const zOffset = Math.random() - 0.5;
|
||||||
const zLimit = 2 - zOffset;
|
const zLimit = 2 - zOffset;
|
||||||
|
|
||||||
function buildParameters(node: NodeInstance, inputs: NodeDefinition['inputs']) {
|
const nodeType = $derived(graph.getNodeType(node));
|
||||||
let parameters = Object.entries(inputs || {}).filter(
|
|
||||||
|
const parameters = $derived(
|
||||||
|
Object.entries(nodeType?.inputs || {}).filter(
|
||||||
(p) => p[1].type !== 'seed' && !('setting' in p[1]) && p[1]?.hidden !== true
|
(p) => p[1].type !== 'seed' && !('setting' in p[1]) && p[1]?.hidden !== true
|
||||||
|
) || {}
|
||||||
);
|
);
|
||||||
|
|
||||||
if (node.type === '__virtual/group/instance') {
|
|
||||||
const groupOptions = [...(manager?.groups?.entries() ?? [])].map(([id, g]) => ({
|
|
||||||
label: g.name,
|
|
||||||
value: id
|
|
||||||
}));
|
|
||||||
// Remove the static placeholder from the definition (height-only) and replace
|
|
||||||
// with a fully dynamic version that carries current names + value.
|
|
||||||
parameters = parameters.filter(([key]) => key !== '__virtual/groupId');
|
|
||||||
parameters = [['__virtual/groupId', {
|
|
||||||
type: 'select',
|
|
||||||
value: node.props?.groupId as string,
|
|
||||||
options: groupOptions
|
|
||||||
}], ...parameters];
|
|
||||||
}
|
|
||||||
|
|
||||||
return parameters;
|
|
||||||
}
|
|
||||||
|
|
||||||
$effect(() => {
|
|
||||||
const props = node.props as Record<string, unknown> | undefined;
|
|
||||||
const virtualGroupId = props?.['__virtual/groupId'] as string | undefined;
|
|
||||||
if (!virtualGroupId) return;
|
|
||||||
const activeGroupId = props?.groupId as string | undefined;
|
|
||||||
if (virtualGroupId === activeGroupId) return;
|
|
||||||
const newGroupDef = manager?.groupNodeDefinitions.get(`__virtual/group/${virtualGroupId}`);
|
|
||||||
if (!newGroupDef) return;
|
|
||||||
const { children, parents, ref } = node.state;
|
|
||||||
node.props = { ...props, groupId: virtualGroupId, '__virtual/groupId': virtualGroupId };
|
|
||||||
node.state = { type: newGroupDef, children, parents, ref };
|
|
||||||
manager?.execute();
|
|
||||||
manager?.save();
|
|
||||||
});
|
|
||||||
|
|
||||||
const parameters = $derived(buildParameters(node, node?.state?.type?.inputs || {}));
|
|
||||||
|
|
||||||
const currentGroupId = $derived((node.props?.groupId as string) ?? '');
|
|
||||||
|
|
||||||
function onGroupSelect(event: Event) {
|
|
||||||
const select = event.target as HTMLSelectElement;
|
|
||||||
const newGroupId = select.value;
|
|
||||||
if (!manager || newGroupId === currentGroupId) return;
|
|
||||||
const newGroupDef = manager.groupNodeDefinitions.get(`__virtual/group/${newGroupId}`);
|
|
||||||
if (!newGroupDef) return;
|
|
||||||
node.props = { ...(node.props ?? {}), groupId: newGroupId };
|
|
||||||
node.state = { type: newGroupDef };
|
|
||||||
manager.execute();
|
|
||||||
manager.save();
|
|
||||||
}
|
|
||||||
|
|
||||||
$effect(() => {
|
$effect(() => {
|
||||||
if ('state' in node && !node.state.ref) {
|
if ('state' in node && !node.state.ref) {
|
||||||
node.state.ref = ref;
|
node.state.ref = ref;
|
||||||
@@ -106,52 +60,17 @@
|
|||||||
>
|
>
|
||||||
<NodeHeader {node} />
|
<NodeHeader {node} />
|
||||||
|
|
||||||
{#if false && node.type === '__virtual/group/instance'}
|
|
||||||
<div class="group-param">
|
|
||||||
<select
|
|
||||||
value={currentGroupId}
|
|
||||||
onchange={onGroupSelect}
|
|
||||||
onmousedown={(e) => e.stopPropagation()}
|
|
||||||
onclick={(e) => e.stopPropagation()}
|
|
||||||
onkeydown={(e) => e.stopPropagation()}
|
|
||||||
>
|
|
||||||
{#each manager?.groups?.entries() ?? [] as [gid, gdef]}
|
|
||||||
<option value={gid}>{gdef.name}</option>
|
|
||||||
{/each}
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
{#each parameters as [key, value], i (key)}
|
{#each parameters as [key, value], i (key)}
|
||||||
<NodeParameter
|
<NodeParameter
|
||||||
bind:node
|
bind:node
|
||||||
id={key}
|
id={key}
|
||||||
input={value}
|
input={value}
|
||||||
isLast={i == parameters.length - 1}
|
isLast={i == parameters.length - 1}
|
||||||
outputIndex={node.type === '__virtual/group/input' ? i : undefined}
|
|
||||||
/>
|
/>
|
||||||
{/each}
|
{/each}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
.group-param {
|
|
||||||
padding: 5px 8px;
|
|
||||||
border-bottom: solid 1px var(--color-layer-2);
|
|
||||||
background: var(--color-layer-1);
|
|
||||||
}
|
|
||||||
|
|
||||||
.group-param select {
|
|
||||||
width: 100%;
|
|
||||||
background: var(--color-layer-2);
|
|
||||||
color: var(--color-text);
|
|
||||||
border: 1px solid rgba(255, 255, 255, 0.12);
|
|
||||||
border-radius: 4px;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-size: 0.8em;
|
|
||||||
cursor: pointer;
|
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
|
||||||
|
|
||||||
.node {
|
.node {
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
user-select: none !important;
|
user-select: none !important;
|
||||||
|
|||||||
@@ -1,11 +1,11 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import { appSettings } from '$lib/settings/app-settings.svelte';
|
import { appSettings } from '$lib/settings/app-settings.svelte';
|
||||||
import type { NodeInstance, Socket } from '@nodarium/types';
|
import type { NodeInstance, Socket } from '@nodarium/types';
|
||||||
import { getGraphState } from '../graph-state.svelte';
|
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
||||||
import { createNodePath } from '../helpers/index.js';
|
import { createNodePath } from '../helpers/index.js';
|
||||||
import { getSocketPosition } from '../helpers/nodeHelpers';
|
|
||||||
|
|
||||||
const graphState = getGraphState();
|
const graphState = getGraphState();
|
||||||
|
const graph = getGraphManager();
|
||||||
|
|
||||||
const { node }: { node: NodeInstance } = $props();
|
const { node }: { node: NodeInstance } = $props();
|
||||||
|
|
||||||
@@ -16,13 +16,14 @@
|
|||||||
graphState.setDownSocket?.({
|
graphState.setDownSocket?.({
|
||||||
node,
|
node,
|
||||||
index: 0,
|
index: 0,
|
||||||
position: getSocketPosition?.(node, 0)
|
position: graphState.getSocketPosition?.(node, 0)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const cornerTop = 10;
|
const cornerTop = 10;
|
||||||
const rightBump = $derived(!!node?.state?.type?.outputs?.length && node.type !== '__virtual/group/input');
|
const nodeType = $derived(graph.getNodeType(node));
|
||||||
|
const rightBump = $derived(!!nodeType?.outputs?.length);
|
||||||
const aspectRatio = 0.25;
|
const aspectRatio = 0.25;
|
||||||
|
|
||||||
const path = $derived(
|
const path = $derived(
|
||||||
@@ -70,9 +71,8 @@
|
|||||||
{#if appSettings.value.debug.advancedMode}
|
{#if appSettings.value.debug.advancedMode}
|
||||||
<span class="bg-white text-black! mr-2 px-1 rounded-sm opacity-30">{node.id}</span>
|
<span class="bg-white text-black! mr-2 px-1 rounded-sm opacity-30">{node.id}</span>
|
||||||
{/if}
|
{/if}
|
||||||
{node.state?.type?.meta?.title ?? node.type.split('/').pop()}
|
{node.type.split('/').pop()}
|
||||||
</div>
|
</div>
|
||||||
{#if node.type !== '__virtual/group/input'}
|
|
||||||
<div
|
<div
|
||||||
class="target"
|
class="target"
|
||||||
role="button"
|
role="button"
|
||||||
@@ -80,7 +80,6 @@
|
|||||||
onmousedown={handleMouseDown}
|
onmousedown={handleMouseDown}
|
||||||
>
|
>
|
||||||
</div>
|
</div>
|
||||||
{/if}
|
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
viewBox="0 0 100 100"
|
viewBox="0 0 100 100"
|
||||||
|
|||||||
@@ -45,7 +45,7 @@
|
|||||||
|
|
||||||
$effect(() => {
|
$effect(() => {
|
||||||
const a = $state.snapshot(value);
|
const a = $state.snapshot(value);
|
||||||
const b = $state.snapshot(node?.props?.[id]) as number | number[] | undefined;
|
const b = $state.snapshot(node?.props?.[id]);
|
||||||
const isDiff = Array.isArray(a) ? diffArray(a, b) : a !== b;
|
const isDiff = Array.isArray(a) ? diffArray(a, b) : a !== b;
|
||||||
if (value !== undefined && isDiff) {
|
if (value !== undefined && isDiff) {
|
||||||
node.props = { ...node.props, [id]: a };
|
node.props = { ...node.props, [id]: a };
|
||||||
|
|||||||
@@ -2,7 +2,7 @@
|
|||||||
import type { NodeInput, NodeInstance, Socket } from '@nodarium/types';
|
import type { NodeInput, NodeInstance, Socket } from '@nodarium/types';
|
||||||
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
import { getGraphManager, getGraphState } from '../graph-state.svelte';
|
||||||
import { createNodePath } from '../helpers';
|
import { createNodePath } from '../helpers';
|
||||||
import { getParameterHeight, getSocketPosition } from '../helpers/nodeHelpers';
|
import { getParameterHeight } from '../helpers/nodeHelpers';
|
||||||
import NodeInputEl from './NodeInput.svelte';
|
import NodeInputEl from './NodeInput.svelte';
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
@@ -10,7 +10,6 @@
|
|||||||
input: NodeInput;
|
input: NodeInput;
|
||||||
id: string;
|
id: string;
|
||||||
isLast?: boolean;
|
isLast?: boolean;
|
||||||
outputIndex?: number;
|
|
||||||
};
|
};
|
||||||
|
|
||||||
const graph = getGraphManager();
|
const graph = getGraphManager();
|
||||||
@@ -18,14 +17,13 @@
|
|||||||
const graphId = graph?.id;
|
const graphId = graph?.id;
|
||||||
const elementId = `input-${Math.random().toString(36).substring(7)}`;
|
const elementId = `input-${Math.random().toString(36).substring(7)}`;
|
||||||
|
|
||||||
let { node = $bindable(), input, id, isLast, outputIndex = undefined }: Props = $props();
|
let { node = $bindable(), input, id, isLast }: Props = $props();
|
||||||
|
|
||||||
const nodeType = $derived(node.state.type!);
|
let nodeType = $derived(graph.getNodeType(node)!);
|
||||||
|
|
||||||
const inputType = $derived(nodeType.inputs?.[id]);
|
const inputType = $derived(nodeType.inputs?.[id]);
|
||||||
|
|
||||||
const socketId = $derived(`${node.id}-${id}`);
|
const socketId = $derived(`${node.id}-${id}`);
|
||||||
const outputSocketId = $derived(outputIndex !== undefined ? `${node.id}-${outputIndex}` : '');
|
|
||||||
const height = $derived(getParameterHeight(nodeType, id));
|
const height = $derived(getParameterHeight(nodeType, id));
|
||||||
|
|
||||||
function handleMouseDown(ev: MouseEvent) {
|
function handleMouseDown(ev: MouseEvent) {
|
||||||
@@ -34,23 +32,11 @@
|
|||||||
graphState.setDownSocket({
|
graphState.setDownSocket({
|
||||||
node,
|
node,
|
||||||
index: id,
|
index: id,
|
||||||
position: getSocketPosition(node, id)
|
position: graphState.getSocketPosition(node, id)
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleOutputMouseDown(ev: MouseEvent) {
|
const leftBump = $derived(nodeType.inputs?.[id].internal !== true);
|
||||||
ev.preventDefault();
|
|
||||||
ev.stopPropagation();
|
|
||||||
if (outputIndex === undefined) return;
|
|
||||||
graphState.setDownSocket({
|
|
||||||
node,
|
|
||||||
index: outputIndex,
|
|
||||||
position: getSocketPosition(node, outputIndex)
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const leftBump = $derived(!id.startsWith('__virtual') && nodeType.inputs?.[id].internal !== true && outputIndex === undefined);
|
|
||||||
const rightBump = $derived(outputIndex !== undefined);
|
|
||||||
const cornerBottom = $derived(isLast ? 5 : 0);
|
const cornerBottom = $derived(isLast ? 5 : 0);
|
||||||
const aspectRatio = 0.5;
|
const aspectRatio = 0.5;
|
||||||
|
|
||||||
@@ -61,7 +47,6 @@
|
|||||||
y: 50.5,
|
y: 50.5,
|
||||||
cornerBottom,
|
cornerBottom,
|
||||||
leftBump,
|
leftBump,
|
||||||
rightBump,
|
|
||||||
aspectRatio
|
aspectRatio
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
@@ -72,7 +57,6 @@
|
|||||||
y: 50.5,
|
y: 50.5,
|
||||||
cornerBottom,
|
cornerBottom,
|
||||||
leftBump,
|
leftBump,
|
||||||
rightBump,
|
|
||||||
aspectRatio
|
aspectRatio
|
||||||
})
|
})
|
||||||
);
|
);
|
||||||
@@ -95,13 +79,11 @@
|
|||||||
data-node-input={id}
|
data-node-input={id}
|
||||||
style:height="{height}px"
|
style:height="{height}px"
|
||||||
style:--socket-color={hoverColor}
|
style:--socket-color={hoverColor}
|
||||||
class:possible-socket={outputIndex !== undefined
|
class:possible-socket={graphState?.possibleSocketIds.has(socketId)}
|
||||||
? graphState?.possibleSocketIds.has(outputSocketId)
|
|
||||||
: graphState?.possibleSocketIds.has(socketId)}
|
|
||||||
>
|
>
|
||||||
{#key id && graphId}
|
{#key id && graphId}
|
||||||
<div class="content" class:disabled={graph?.inputSockets?.has(socketId)}>
|
<div class="content" class:disabled={graph?.inputSockets?.has(socketId)}>
|
||||||
{#if inputType?.label !== '' && !id.startsWith('__virtual')}
|
{#if inputType?.label !== ''}
|
||||||
<label for={elementId} title={input.description}>{input.label || id}</label>
|
<label for={elementId} title={input.description}>{input.label || id}</label>
|
||||||
{/if}
|
{/if}
|
||||||
{#if inputType?.external !== true}
|
{#if inputType?.external !== true}
|
||||||
@@ -109,7 +91,7 @@
|
|||||||
{/if}
|
{/if}
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
{#if outputIndex === undefined && node?.state?.type?.inputs?.[id]?.internal !== true}
|
{#if node?.state?.type?.inputs?.[id]?.internal !== true}
|
||||||
<div
|
<div
|
||||||
data-node-socket
|
data-node-socket
|
||||||
class="target"
|
class="target"
|
||||||
@@ -121,17 +103,6 @@
|
|||||||
{/if}
|
{/if}
|
||||||
{/key}
|
{/key}
|
||||||
|
|
||||||
{#if outputIndex !== undefined}
|
|
||||||
<div
|
|
||||||
data-node-socket
|
|
||||||
class="target target-right"
|
|
||||||
onmousedown={handleOutputMouseDown}
|
|
||||||
role="button"
|
|
||||||
tabindex="0"
|
|
||||||
>
|
|
||||||
</div>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<svg
|
<svg
|
||||||
xmlns="http://www.w3.org/2000/svg"
|
xmlns="http://www.w3.org/2000/svg"
|
||||||
viewBox="0 0 100 100"
|
viewBox="0 0 100 100"
|
||||||
@@ -159,16 +130,6 @@
|
|||||||
transform: translateY(-50%) translateX(-50%);
|
transform: translateY(-50%) translateX(-50%);
|
||||||
}
|
}
|
||||||
|
|
||||||
.target-right {
|
|
||||||
right: 0;
|
|
||||||
left: auto;
|
|
||||||
transform: translateY(-50%) translateX(50%);
|
|
||||||
}
|
|
||||||
|
|
||||||
.target-right:hover ~ svg path {
|
|
||||||
d: var(--hover-path);
|
|
||||||
}
|
|
||||||
|
|
||||||
.possible-socket .target::before {
|
.possible-socket .target::before {
|
||||||
content: "";
|
content: "";
|
||||||
position: absolute;
|
position: absolute;
|
||||||
|
|||||||
@@ -23,7 +23,11 @@ export function createMockNodeRegistry(nodes: NodeDefinition[]): NodeRegistry {
|
|||||||
|
|
||||||
export const mockFloatOutputNode: NodeDefinition = {
|
export const mockFloatOutputNode: NodeDefinition = {
|
||||||
id: 'test/node/output',
|
id: 'test/node/output',
|
||||||
inputs: {},
|
inputs: {
|
||||||
|
'input': {
|
||||||
|
type: 'float'
|
||||||
|
}
|
||||||
|
},
|
||||||
outputs: ['float'],
|
outputs: ['float'],
|
||||||
meta: { title: 'Float Output' },
|
meta: { title: 'Float Output' },
|
||||||
execute: () => new Int32Array()
|
execute: () => new Int32Array()
|
||||||
@@ -32,7 +36,7 @@ export const mockFloatOutputNode: NodeDefinition = {
|
|||||||
export const mockFloatInputNode: NodeDefinition = {
|
export const mockFloatInputNode: NodeDefinition = {
|
||||||
id: 'test/node/input',
|
id: 'test/node/input',
|
||||||
inputs: { value: { type: 'float' } },
|
inputs: { value: { type: 'float' } },
|
||||||
outputs: [],
|
outputs: ['float'],
|
||||||
meta: { title: 'Float Input' },
|
meta: { title: 'Float Input' },
|
||||||
execute: () => new Int32Array()
|
execute: () => new Int32Array()
|
||||||
};
|
};
|
||||||
|
|||||||
@@ -4,7 +4,8 @@ export function grid(width: number, height: number) {
|
|||||||
const graph: Graph = {
|
const graph: Graph = {
|
||||||
id: Math.floor(Math.random() * 100000),
|
id: Math.floor(Math.random() * 100000),
|
||||||
edges: [],
|
edges: [],
|
||||||
nodes: []
|
nodes: [],
|
||||||
|
groups: []
|
||||||
};
|
};
|
||||||
|
|
||||||
const amount = width * height;
|
const amount = width * height;
|
||||||
|
|||||||
@@ -47,6 +47,7 @@ export function tree(depth: number): Graph {
|
|||||||
return {
|
return {
|
||||||
id: Math.floor(Math.random() * 100000),
|
id: Math.floor(Math.random() * 100000),
|
||||||
nodes,
|
nodes,
|
||||||
edges
|
edges,
|
||||||
|
groups: []
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,5 +1,5 @@
|
|||||||
export const debugNode = {
|
export const debugNode = {
|
||||||
id: 'max/plantarium/debug',
|
id: '__internal/debug/instance',
|
||||||
inputs: {
|
inputs: {
|
||||||
input: {
|
input: {
|
||||||
type: '*'
|
type: '*'
|
||||||
|
|||||||
@@ -0,0 +1,13 @@
|
|||||||
|
export const groupNode = {
|
||||||
|
id: '__internal/group/instance',
|
||||||
|
meta: { title: 'Group' },
|
||||||
|
inputs: {
|
||||||
|
input: {
|
||||||
|
type: 'select',
|
||||||
|
values: []
|
||||||
|
}
|
||||||
|
},
|
||||||
|
execute(_data: Int32Array): Int32Array {
|
||||||
|
return _data;
|
||||||
|
}
|
||||||
|
} as const;
|
||||||
@@ -1,25 +0,0 @@
|
|||||||
import type { NodeDefinition } from '@nodarium/types';
|
|
||||||
|
|
||||||
export const groupInputNode: NodeDefinition = {
|
|
||||||
id: '__virtual/group/input',
|
|
||||||
inputs: {},
|
|
||||||
outputs: [],
|
|
||||||
execute(_data: Int32Array): Int32Array { return _data; }
|
|
||||||
} as unknown as NodeDefinition;
|
|
||||||
|
|
||||||
export const groupOutputNode: NodeDefinition = {
|
|
||||||
id: '__virtual/group/output',
|
|
||||||
inputs: {},
|
|
||||||
outputs: [],
|
|
||||||
execute(_data: Int32Array): Int32Array { return _data; }
|
|
||||||
} as unknown as NodeDefinition;
|
|
||||||
|
|
||||||
// Stub registered in the registry so it appears in AddMenu.
|
|
||||||
// Actual inputs/outputs are resolved from props.groupId at runtime.
|
|
||||||
export const groupNode: NodeDefinition = {
|
|
||||||
id: '__virtual/group/instance',
|
|
||||||
meta: { title: 'Group' },
|
|
||||||
inputs: {},
|
|
||||||
outputs: [],
|
|
||||||
execute(_data: Int32Array): Int32Array { return _data; }
|
|
||||||
} as unknown as NodeDefinition;
|
|
||||||
@@ -88,6 +88,7 @@ export class RemoteNodeRegistry implements NodeRegistry {
|
|||||||
}
|
}
|
||||||
|
|
||||||
async fetchNodeDefinition(nodeId: `${string}/${string}/${string}`) {
|
async fetchNodeDefinition(nodeId: `${string}/${string}/${string}`) {
|
||||||
|
if (nodeId.startsWith('__internal/')) return;
|
||||||
return this.fetchJson(`nodes/${nodeId}.json`);
|
return this.fetchJson(`nodes/${nodeId}.json`);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -109,6 +110,8 @@ export class RemoteNodeRegistry implements NodeRegistry {
|
|||||||
return this.nodes.get(id)!;
|
return this.nodes.get(id)!;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (id.startsWith('__internal/')) return;
|
||||||
|
|
||||||
const wasmBuffer = await this.fetchNodeWasm(id);
|
const wasmBuffer = await this.fetchNodeWasm(id);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
|||||||
@@ -0,0 +1,154 @@
|
|||||||
|
import { describe, expect, it } from 'vitest';
|
||||||
|
import { expandGroups } from './runtime-executor';
|
||||||
|
import type { Graph } from '@nodarium/types';
|
||||||
|
|
||||||
|
// Helpers to build minimal serialized nodes/edges
|
||||||
|
function node(id: number, type: string, props?: Record<string, number>) {
|
||||||
|
return { id, type: type as Graph['nodes'][0]['type'], position: [0, 0] as [number, number], ...(props ? { props } : {}) };
|
||||||
|
}
|
||||||
|
|
||||||
|
function edge(from: number, fromSocket: number, to: number, toSocket: string): [number, number, number, string] {
|
||||||
|
return [from, fromSocket, to, toSocket];
|
||||||
|
}
|
||||||
|
|
||||||
|
describe('expandGroups', () => {
|
||||||
|
it('returns graph unchanged when there are no groups', () => {
|
||||||
|
const graph: Graph = {
|
||||||
|
id: 1,
|
||||||
|
nodes: [node(0, 'test/node/output'), node(1, 'test/node/input')],
|
||||||
|
edges: [edge(0, 0, 1, 'value')],
|
||||||
|
groups: []
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = expandGroups(graph);
|
||||||
|
|
||||||
|
expect(result.nodes.length).toBe(2);
|
||||||
|
expect(result.edges.length).toBe(1);
|
||||||
|
expect(result).toBe(graph); // same reference — no copy needed
|
||||||
|
});
|
||||||
|
|
||||||
|
it('expands a simple group: A → [B] → C rewires to A → B → C', () => {
|
||||||
|
// IDs: A=1, B=2, C=3, groupNode=4, group.id=5, inputBoundary=6, outputBoundary=7
|
||||||
|
const groupId = 5;
|
||||||
|
const groupNodeId = 4;
|
||||||
|
const remappedB = (groupNodeId + 1) * 1_000_000 + 2; // 5_000_002
|
||||||
|
|
||||||
|
const graph: Graph = {
|
||||||
|
id: 1,
|
||||||
|
nodes: [
|
||||||
|
node(1, 'test/node/output'),
|
||||||
|
node(groupNodeId, '__internal/group/instance', { groupId }),
|
||||||
|
node(3, 'test/node/input')
|
||||||
|
],
|
||||||
|
edges: [
|
||||||
|
edge(1, 0, groupNodeId, 'input_0'), // A → group
|
||||||
|
edge(groupNodeId, 0, 3, 'value') // group → C
|
||||||
|
],
|
||||||
|
groups: [{
|
||||||
|
id: groupId,
|
||||||
|
nodes: [
|
||||||
|
node(6, '__internal/group/input'),
|
||||||
|
node(2, 'test/node/output'),
|
||||||
|
node(7, '__internal/group/output')
|
||||||
|
],
|
||||||
|
edges: [
|
||||||
|
edge(6, 0, 2, 'input'), // inputBoundary → B
|
||||||
|
edge(2, 0, 7, 'Out') // B → outputBoundary
|
||||||
|
],
|
||||||
|
inputs: { input_0: { type: 'float' } },
|
||||||
|
outputs: [{ type: 'float', label: 'Output 0' }]
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = expandGroups(graph);
|
||||||
|
|
||||||
|
const ids = result.nodes.map(n => n.id);
|
||||||
|
expect(ids).not.toContain(groupNodeId);
|
||||||
|
expect(ids).toContain(remappedB);
|
||||||
|
expect(ids).toContain(1); // A
|
||||||
|
expect(ids).toContain(3); // C
|
||||||
|
expect(result.nodes.length).toBe(3); // A, B(remapped), C
|
||||||
|
|
||||||
|
expect(result.edges).toContainEqual(edge(1, 0, remappedB, 'input')); // A → B
|
||||||
|
expect(result.edges).toContainEqual(edge(remappedB, 0, 3, 'value')); // B → C
|
||||||
|
expect(result.edges.length).toBe(2);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('expands a group with two internal nodes (B→D) and preserves the internal edge', () => {
|
||||||
|
// A → [B → D] → C
|
||||||
|
const groupId = 10;
|
||||||
|
const groupNodeId = 5;
|
||||||
|
const remappedB = (groupNodeId + 1) * 1_000_000 + 1; // 6_000_001
|
||||||
|
const remappedD = (groupNodeId + 1) * 1_000_000 + 2; // 6_000_002
|
||||||
|
|
||||||
|
const graph: Graph = {
|
||||||
|
id: 1,
|
||||||
|
nodes: [
|
||||||
|
node(0, 'test/node/output'),
|
||||||
|
node(groupNodeId, '__internal/group/instance', { groupId }),
|
||||||
|
node(9, 'test/node/input')
|
||||||
|
],
|
||||||
|
edges: [
|
||||||
|
edge(0, 0, groupNodeId, 'input_0'),
|
||||||
|
edge(groupNodeId, 0, 9, 'value')
|
||||||
|
],
|
||||||
|
groups: [{
|
||||||
|
id: groupId,
|
||||||
|
nodes: [
|
||||||
|
node(3, '__internal/group/input'),
|
||||||
|
node(1, 'test/node/output'), // B
|
||||||
|
node(2, 'test/node/output'), // D
|
||||||
|
node(4, '__internal/group/output')
|
||||||
|
],
|
||||||
|
edges: [
|
||||||
|
edge(3, 0, 1, 'input'), // inputBoundary → B
|
||||||
|
edge(1, 0, 2, 'input'), // B → D (internal)
|
||||||
|
edge(2, 0, 4, 'Out') // D → outputBoundary
|
||||||
|
],
|
||||||
|
inputs: { input_0: { type: 'float' } },
|
||||||
|
outputs: [{ type: 'float' }]
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = expandGroups(graph);
|
||||||
|
|
||||||
|
expect(result.nodes.map(n => n.id)).not.toContain(groupNodeId);
|
||||||
|
expect(result.nodes.map(n => n.id)).toContain(remappedB);
|
||||||
|
expect(result.nodes.map(n => n.id)).toContain(remappedD);
|
||||||
|
|
||||||
|
expect(result.edges).toContainEqual(edge(0, 0, remappedB, 'input')); // A → B
|
||||||
|
expect(result.edges).toContainEqual(edge(remappedB, 0, remappedD, 'input')); // B → D (internal)
|
||||||
|
expect(result.edges).toContainEqual(edge(remappedD, 0, 9, 'value')); // D → C
|
||||||
|
expect(result.edges.length).toBe(3);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('expands a group with no external connections (isolated)', () => {
|
||||||
|
const groupId = 20;
|
||||||
|
const groupNodeId = 1;
|
||||||
|
const remappedB = (groupNodeId + 1) * 1_000_000 + 2; // 2_000_002
|
||||||
|
|
||||||
|
const graph: Graph = {
|
||||||
|
id: 1,
|
||||||
|
nodes: [node(groupNodeId, '__internal/group/instance', { groupId })],
|
||||||
|
edges: [],
|
||||||
|
groups: [{
|
||||||
|
id: groupId,
|
||||||
|
nodes: [
|
||||||
|
node(3, '__internal/group/input'),
|
||||||
|
node(2, 'test/node/output'),
|
||||||
|
node(4, '__internal/group/output')
|
||||||
|
],
|
||||||
|
edges: [
|
||||||
|
edge(3, 0, 2, 'input'),
|
||||||
|
edge(2, 0, 4, 'Out')
|
||||||
|
]
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
|
||||||
|
const result = expandGroups(graph);
|
||||||
|
|
||||||
|
expect(result.nodes.map(n => n.id)).not.toContain(groupNodeId);
|
||||||
|
expect(result.nodes.map(n => n.id)).toContain(remappedB);
|
||||||
|
expect(result.edges.length).toBe(0);
|
||||||
|
});
|
||||||
|
});
|
||||||
@@ -7,18 +7,11 @@ import type {
|
|||||||
SyncCache
|
SyncCache
|
||||||
} from '@nodarium/types';
|
} from '@nodarium/types';
|
||||||
|
|
||||||
function isGroupInstanceType(type: string): boolean {
|
|
||||||
return type === '__virtual/group/instance';
|
|
||||||
}
|
|
||||||
|
|
||||||
export function expandGroups(graph: Graph): Graph {
|
export function expandGroups(graph: Graph): Graph {
|
||||||
if (!graph.groups || Object.keys(graph.groups).length === 0) {
|
if (!graph.groups || graph.groups.length === 0) return graph;
|
||||||
return graph;
|
|
||||||
}
|
|
||||||
|
|
||||||
let nodes = [...graph.nodes];
|
let nodes = [...graph.nodes];
|
||||||
let edges = [...graph.edges];
|
let edges = [...graph.edges];
|
||||||
const groups = graph.groups;
|
|
||||||
|
|
||||||
let changed = true;
|
let changed = true;
|
||||||
while (changed) {
|
while (changed) {
|
||||||
@@ -26,117 +19,69 @@ export function expandGroups(graph: Graph): Graph {
|
|||||||
|
|
||||||
for (let i = 0; i < nodes.length; i++) {
|
for (let i = 0; i < nodes.length; i++) {
|
||||||
const node = nodes[i];
|
const node = nodes[i];
|
||||||
if (!isGroupInstanceType(node.type)) continue;
|
if (node.type !== '__internal/group/instance') continue;
|
||||||
|
|
||||||
const groupId = (node.props as Record<string, unknown> | undefined)?.groupId as string | undefined;
|
const groupId = node.props?.groupId as number | undefined;
|
||||||
if (!groupId) continue;
|
if (groupId === undefined) continue;
|
||||||
const group = groups[groupId];
|
|
||||||
|
const group = graph.groups.find(g => g.id === groupId);
|
||||||
if (!group) continue;
|
if (!group) continue;
|
||||||
|
|
||||||
changed = true;
|
changed = true;
|
||||||
|
|
||||||
// Recursively expand nested groups inside this group's internal graph
|
const ID_OFFSET = (node.id + 1) * 1_000_000;
|
||||||
const expandedInternal = expandGroups({
|
|
||||||
id: 0,
|
|
||||||
nodes: group.graph.nodes,
|
|
||||||
edges: group.graph.edges,
|
|
||||||
groups
|
|
||||||
});
|
|
||||||
|
|
||||||
const ID_PREFIX = node.id * 1000000;
|
|
||||||
const idMap = new Map<number, number>();
|
const idMap = new Map<number, number>();
|
||||||
|
|
||||||
const inputVirtualNode = expandedInternal.nodes.find(
|
const inputBoundary = group.nodes.find(n => n.type === '__internal/group/input');
|
||||||
n => n.type === '__virtual/group/input'
|
const outputBoundary = group.nodes.find(n => n.type === '__internal/group/output');
|
||||||
);
|
|
||||||
const outputVirtualNode = expandedInternal.nodes.find(
|
const realNodes = group.nodes.filter(
|
||||||
n => n.type === '__virtual/group/output'
|
n => n.type !== '__internal/group/input' && n.type !== '__internal/group/output'
|
||||||
);
|
);
|
||||||
|
|
||||||
const realInternalNodes = expandedInternal.nodes.filter(
|
for (const n of realNodes) idMap.set(n.id, ID_OFFSET + n.id);
|
||||||
n => n.type !== '__virtual/group/input' && n.type !== '__virtual/group/output'
|
|
||||||
);
|
|
||||||
|
|
||||||
for (const n of realInternalNodes) {
|
|
||||||
idMap.set(n.id, ID_PREFIX + n.id);
|
|
||||||
}
|
|
||||||
|
|
||||||
const parentIncomingEdges = edges.filter(e => e[2] === node.id);
|
|
||||||
const parentOutgoingEdges = edges.filter(e => e[0] === node.id);
|
|
||||||
|
|
||||||
// Edges from/to virtual nodes in the expanded internal graph
|
|
||||||
const edgesFromInput = expandedInternal.edges.filter(
|
|
||||||
e => e[0] === inputVirtualNode?.id
|
|
||||||
);
|
|
||||||
const edgesToOutput = expandedInternal.edges.filter(
|
|
||||||
e => e[2] === outputVirtualNode?.id
|
|
||||||
);
|
|
||||||
|
|
||||||
|
const incomingExternal = edges.filter(e => e[2] === node.id);
|
||||||
|
const outgoingExternal = edges.filter(e => e[0] === node.id);
|
||||||
const newEdges: Graph['edges'] = [];
|
const newEdges: Graph['edges'] = [];
|
||||||
|
|
||||||
// Short-circuit: parent source → internal target (via group input)
|
// external_source → [inputBoundary →] internal_target
|
||||||
for (const parentEdge of parentIncomingEdges) {
|
if (inputBoundary) {
|
||||||
const socketName = parentEdge[3];
|
const fromInput = group.edges.filter(e => e[0] === inputBoundary.id);
|
||||||
const socketIdx = group.inputs.findIndex(s => s.name === socketName);
|
for (const extEdge of incomingExternal) {
|
||||||
if (socketIdx === -1) continue;
|
for (const intEdge of fromInput) {
|
||||||
|
const toId = idMap.get(intEdge[2]);
|
||||||
for (const internalEdge of edgesFromInput.filter(e => e[1] === socketIdx)) {
|
if (toId !== undefined) newEdges.push([extEdge[0], extEdge[1], toId, intEdge[3]]);
|
||||||
const remappedId = idMap.get(internalEdge[2]);
|
|
||||||
if (remappedId !== undefined) {
|
|
||||||
newEdges.push([parentEdge[0], parentEdge[1], remappedId, internalEdge[3]]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Short-circuit: internal source → parent target (via group output)
|
// internal_source → [outputBoundary →] external_target
|
||||||
for (const parentEdge of parentOutgoingEdges) {
|
if (outputBoundary) {
|
||||||
const outputIdx = parentEdge[1];
|
const toOutput = group.edges.filter(e => e[2] === outputBoundary.id);
|
||||||
const outputSocketName = group.outputs[outputIdx]?.name;
|
for (const extEdge of outgoingExternal) {
|
||||||
if (!outputSocketName) continue;
|
for (const intEdge of toOutput) {
|
||||||
|
const fromId = idMap.get(intEdge[0]);
|
||||||
for (const internalEdge of edgesToOutput.filter(e => e[3] === outputSocketName)) {
|
if (fromId !== undefined) newEdges.push([fromId, intEdge[1], extEdge[2], extEdge[3]]);
|
||||||
const remappedId = idMap.get(internalEdge[0]);
|
|
||||||
if (remappedId !== undefined) {
|
|
||||||
newEdges.push([remappedId, internalEdge[1], parentEdge[2], parentEdge[3]]);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remap internal-to-internal edges
|
// internal-to-internal edges (skip boundary edges)
|
||||||
const internalEdges = expandedInternal.edges.filter(
|
for (const e of group.edges) {
|
||||||
e => e[0] !== inputVirtualNode?.id
|
if (e[0] === inputBoundary?.id || e[2] === outputBoundary?.id) continue;
|
||||||
&& e[0] !== outputVirtualNode?.id
|
|
||||||
&& e[2] !== inputVirtualNode?.id
|
|
||||||
&& e[2] !== outputVirtualNode?.id
|
|
||||||
);
|
|
||||||
|
|
||||||
for (const e of internalEdges) {
|
|
||||||
const fromId = idMap.get(e[0]);
|
const fromId = idMap.get(e[0]);
|
||||||
const toId = idMap.get(e[2]);
|
const toId = idMap.get(e[2]);
|
||||||
if (fromId !== undefined && toId !== undefined) {
|
if (fromId !== undefined && toId !== undefined) newEdges.push([fromId, e[1], toId, e[3]]);
|
||||||
newEdges.push([fromId, e[1], toId, e[3]]);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Remove the group node
|
|
||||||
nodes.splice(i, 1);
|
nodes.splice(i, 1);
|
||||||
|
for (const n of realNodes) nodes.push({ ...n, id: idMap.get(n.id)! });
|
||||||
|
|
||||||
// Add remapped internal nodes
|
edges = edges.filter(e => e[0] !== node.id && e[2] !== node.id);
|
||||||
for (const n of realInternalNodes) {
|
|
||||||
nodes.push({ ...n, id: idMap.get(n.id)! });
|
|
||||||
}
|
|
||||||
|
|
||||||
// Remove group node's edges and add short-circuit edges
|
|
||||||
const groupEdgeKeys = new Set([
|
|
||||||
...parentIncomingEdges.map(e => `${e[0]}-${e[1]}-${e[2]}-${e[3]}`),
|
|
||||||
...parentOutgoingEdges.map(e => `${e[0]}-${e[1]}-${e[2]}-${e[3]}`)
|
|
||||||
]);
|
|
||||||
edges = edges.filter(
|
|
||||||
e => !groupEdgeKeys.has(`${e[0]}-${e[1]}-${e[2]}-${e[3]}`)
|
|
||||||
);
|
|
||||||
edges.push(...newEdges);
|
edges.push(...newEdges);
|
||||||
|
|
||||||
break; // Restart loop with updated nodes array
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -214,7 +159,7 @@ export class MemoryRuntimeExecutor implements RuntimeExecutor {
|
|||||||
// Only load non-virtual types (virtual nodes are resolved locally)
|
// Only load non-virtual types (virtual nodes are resolved locally)
|
||||||
const nonVirtualTypes = graph.nodes
|
const nonVirtualTypes = graph.nodes
|
||||||
.map(node => node.type)
|
.map(node => node.type)
|
||||||
.filter(t => !t.startsWith('__virtual/'));
|
.filter(t => !t.startsWith('__internal/'));
|
||||||
await this.registry.load(nonVirtualTypes as any);
|
await this.registry.load(nonVirtualTypes as any);
|
||||||
|
|
||||||
const typeMap = new Map<string, NodeDefinition>();
|
const typeMap = new Map<string, NodeDefinition>();
|
||||||
@@ -362,7 +307,7 @@ export class MemoryRuntimeExecutor implements RuntimeExecutor {
|
|||||||
if (inputNode) {
|
if (inputNode) {
|
||||||
if (results[inputNode.id] === undefined) {
|
if (results[inputNode.id] === undefined) {
|
||||||
throw new Error(
|
throw new Error(
|
||||||
`Node ${node.type} is missing input from node ${inputNode.type}`
|
`Node ${node.type} is missing input from node ${inputNode.type}#${inputNode.id}`
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
return results[inputNode.id];
|
return results[inputNode.id];
|
||||||
|
|||||||
@@ -1,18 +1,12 @@
|
|||||||
import { debugNode } from '$lib/node-registry/debugNode';
|
import { debugNode } from '$lib/node-registry/debugNode';
|
||||||
import { groupInputNode, groupNode, groupOutputNode } from '$lib/node-registry/groupNodes';
|
|
||||||
import { IndexDBCache, RemoteNodeRegistry } from '$lib/node-registry/index';
|
import { IndexDBCache, RemoteNodeRegistry } from '$lib/node-registry/index';
|
||||||
import type { Graph } from '@nodarium/types';
|
import type { Graph } from '@nodarium/types';
|
||||||
import { createPerformanceStore } from '@nodarium/utils';
|
import { createPerformanceStore } from '@nodarium/utils';
|
||||||
import { expandGroups, MemoryRuntimeExecutor } from './runtime-executor';
|
import { MemoryRuntimeExecutor } from './runtime-executor';
|
||||||
import { MemoryRuntimeCache } from './runtime-executor-cache';
|
import { MemoryRuntimeCache } from './runtime-executor-cache';
|
||||||
|
|
||||||
const indexDbCache = new IndexDBCache('node-registry');
|
const indexDbCache = new IndexDBCache('node-registry');
|
||||||
const nodeRegistry = new RemoteNodeRegistry('', indexDbCache, [
|
const nodeRegistry = new RemoteNodeRegistry('', indexDbCache, [debugNode]);
|
||||||
debugNode,
|
|
||||||
groupInputNode,
|
|
||||||
groupOutputNode,
|
|
||||||
groupNode
|
|
||||||
]);
|
|
||||||
|
|
||||||
const cache = new MemoryRuntimeCache();
|
const cache = new MemoryRuntimeCache();
|
||||||
const executor = new MemoryRuntimeExecutor(nodeRegistry, cache);
|
const executor = new MemoryRuntimeExecutor(nodeRegistry, cache);
|
||||||
@@ -40,13 +34,7 @@ export async function executeGraph(
|
|||||||
graph: Graph,
|
graph: Graph,
|
||||||
settings: Record<string, unknown>
|
settings: Record<string, unknown>
|
||||||
): Promise<Int32Array> {
|
): Promise<Int32Array> {
|
||||||
// Expand groups before loading types so we only load real (non-virtual) node types
|
await nodeRegistry.load(graph.nodes.map((n) => n.type));
|
||||||
const expandedGraph = expandGroups(graph);
|
|
||||||
await nodeRegistry.load(
|
|
||||||
expandedGraph.nodes
|
|
||||||
.map(n => n.type)
|
|
||||||
.filter(t => !t.startsWith('__virtual/')) as any
|
|
||||||
);
|
|
||||||
performanceStore.startRun();
|
performanceStore.startRun();
|
||||||
const res = await executor.execute(graph, settings);
|
const res = await executor.execute(graph, settings);
|
||||||
performanceStore.stopRun();
|
performanceStore.stopRun();
|
||||||
|
|||||||
@@ -52,7 +52,7 @@
|
|||||||
// select input: use index into options
|
// select input: use index into options
|
||||||
if ('options' in node && Array.isArray(node.options)) {
|
if ('options' in node && Array.isArray(node.options)) {
|
||||||
if (typeof inputValue === 'string') {
|
if (typeof inputValue === 'string') {
|
||||||
return (node.options as string[]).indexOf(inputValue);
|
return node.options.indexOf(inputValue);
|
||||||
}
|
}
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -96,4 +96,6 @@
|
|||||||
bind:value={store}
|
bind:value={store}
|
||||||
type={nodeDefinition}
|
type={nodeDefinition}
|
||||||
/>
|
/>
|
||||||
|
{:else}
|
||||||
|
<p class="mx-4 mt-4">Node has no settings</p>
|
||||||
{/if}
|
{/if}
|
||||||
|
|||||||
@@ -5,27 +5,28 @@
|
|||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
manager: GraphManager;
|
manager: GraphManager;
|
||||||
node: NodeInstance;
|
node: NodeInstance | undefined;
|
||||||
};
|
};
|
||||||
|
|
||||||
let { manager, node = $bindable() }: Props = $props();
|
let { manager, node = $bindable() }: Props = $props();
|
||||||
|
|
||||||
const inputs = $derived(node?.state?.type?.inputs || {});
|
|
||||||
|
|
||||||
const hasSettings = $derived(
|
|
||||||
Object.values(inputs).find(entry => {
|
|
||||||
return entry.hidden === true;
|
|
||||||
}) !== undefined
|
|
||||||
);
|
|
||||||
|
|
||||||
$inspect({ inputs, hasSettings });
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
{#key node.id}
|
<div class='{node?"border-l-2 pl-3.5!":""} bg-layer-2 flex items-center h-[70px] border-b-1 border-l-selected border-b-outline pl-4'>
|
||||||
{#if node && hasSettings}
|
|
||||||
<div class="border-l-2 pl-3.5! bg-layer-2 flex items-center h-[70px] border-b-1 border-l-selected border-b-outline pl-4">
|
|
||||||
<h3>Node Settings</h3>
|
<h3>Node Settings</h3>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
{#if node}
|
||||||
|
{#key node.id}
|
||||||
|
{#if node}
|
||||||
<ActiveNodeSelected {manager} bind:node />
|
<ActiveNodeSelected {manager} bind:node />
|
||||||
{/if}
|
{/if}
|
||||||
{/key}
|
{/key}
|
||||||
|
{:else}
|
||||||
|
<p class="mx-4 mt-4">No node selected</p>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if manager?.graph.groups.length}
|
||||||
|
<button onclick={() => manager.removeUnusedGroups()}>
|
||||||
|
remove unused groups
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
|||||||
@@ -1,20 +1,23 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
import type { Graph } from '$lib/types';
|
import type { Graph } from '$lib/types';
|
||||||
|
import { JsonViewer } from '@nodarium/ui';
|
||||||
|
|
||||||
const { graph }: { graph?: Graph } = $props();
|
const { graph }: { graph?: Graph } = $props();
|
||||||
|
|
||||||
function convert(g: Graph): string {
|
const data = $derived(
|
||||||
return JSON.stringify(
|
graph
|
||||||
{
|
? {
|
||||||
...g,
|
...graph,
|
||||||
nodes: g.nodes.map((n: object) => ({ ...n, tmp: undefined, state: undefined }))
|
nodes: graph.nodes.map((n: object) => ({ ...n, tmp: undefined, state: undefined }))
|
||||||
},
|
|
||||||
null,
|
|
||||||
2
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
: null
|
||||||
|
);
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<pre>
|
<div class="overflow-auto p-2">
|
||||||
{graph ? convert(graph) : "No graph loaded"}
|
{#if data}
|
||||||
</pre>
|
<JsonViewer value={data} path="graph" />
|
||||||
|
{:else}
|
||||||
|
<span class="font-mono text-xs text-neutral-500">No graph loaded</span>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
|||||||
@@ -1,175 +0,0 @@
|
|||||||
<script lang="ts">
|
|
||||||
import type { GraphManager } from '$lib/graph-interface/graph-manager.svelte';
|
|
||||||
import { InputSelect } from '@nodarium/ui';
|
|
||||||
|
|
||||||
type Props = { manager: GraphManager; groupId: string };
|
|
||||||
const { manager, groupId }: Props = $props();
|
|
||||||
|
|
||||||
const group = $derived(manager.groups.get(groupId));
|
|
||||||
|
|
||||||
const COMMON_TYPES = ['plant', 'float', 'int', 'vec3', 'bool'];
|
|
||||||
let selectedTypeIdx = $state(0);
|
|
||||||
let customType = $state('');
|
|
||||||
|
|
||||||
function rename(e: Event) {
|
|
||||||
if (!group) return;
|
|
||||||
const name = (e.target as HTMLInputElement).value.trim();
|
|
||||||
if (!name) return;
|
|
||||||
group.name = name;
|
|
||||||
if (manager.graph.groups?.[groupId]) manager.graph.groups[groupId].name = name;
|
|
||||||
const def = manager.groupNodeDefinitions.get(`__virtual/group/${groupId}`);
|
|
||||||
if (def?.meta) def.meta.title = name;
|
|
||||||
manager.save();
|
|
||||||
}
|
|
||||||
|
|
||||||
function addSocket() {
|
|
||||||
const type = customType.trim() || COMMON_TYPES[selectedTypeIdx];
|
|
||||||
if (!type) return;
|
|
||||||
manager.addGroupSocket('input', type);
|
|
||||||
customType = '';
|
|
||||||
}
|
|
||||||
|
|
||||||
function removeSocket(index: number) {
|
|
||||||
manager.removeGroupSocket('input', index);
|
|
||||||
}
|
|
||||||
|
|
||||||
function prune() {
|
|
||||||
manager.pruneUnusedGroups();
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<div class="bg-layer-2 flex items-center h-[70px] border-b-1 border-outline pl-4">
|
|
||||||
<h3>Group Settings</h3>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="px-4 py-3 flex flex-col gap-4">
|
|
||||||
<div class="flex flex-col gap-1.5">
|
|
||||||
<span class="section-label">Group name</span>
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
value={group?.name ?? ''}
|
|
||||||
onchange={rename}
|
|
||||||
onkeydown={(e) => e.stopPropagation()}
|
|
||||||
placeholder="Group name"
|
|
||||||
class="bg-layer-2 text-text rounded-[5px] px-2 py-1.5 text-sm w-full box-border outline outline-1 outline-outline"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex flex-col gap-1.5">
|
|
||||||
<span class="section-label">Inputs</span>
|
|
||||||
|
|
||||||
{#if (group?.inputs?.length ?? 0) === 0}
|
|
||||||
<p class="text-sm opacity-40 italic m-0">No inputs yet</p>
|
|
||||||
{:else}
|
|
||||||
<ul class="socket-list">
|
|
||||||
{#each group?.inputs ?? [] as socket, i}
|
|
||||||
<li class="socket-item">
|
|
||||||
<span class="flex-1 opacity-80 text-sm">{socket.name}</span>
|
|
||||||
<span class="text-xs opacity-45 italic">{socket.type}</span>
|
|
||||||
<button class="remove-btn" onclick={() => removeSocket(i)} title="Remove">×</button>
|
|
||||||
</li>
|
|
||||||
{/each}
|
|
||||||
</ul>
|
|
||||||
{/if}
|
|
||||||
|
|
||||||
<div class="flex gap-1.5 items-center">
|
|
||||||
<InputSelect options={COMMON_TYPES} bind:value={selectedTypeIdx} />
|
|
||||||
<input
|
|
||||||
type="text"
|
|
||||||
placeholder="custom type…"
|
|
||||||
bind:value={customType}
|
|
||||||
onkeydown={(e) => {
|
|
||||||
e.stopPropagation();
|
|
||||||
if (e.key === 'Enter') addSocket();
|
|
||||||
}}
|
|
||||||
class="bg-layer-2 text-text rounded-[5px] px-2 py-1 text-sm flex-1 min-w-0 outline outline-1 outline-outline"
|
|
||||||
/>
|
|
||||||
<button class="add-btn" onclick={addSocket}>+ Add</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="flex flex-col gap-1.5">
|
|
||||||
<span class="section-label">Maintenance</span>
|
|
||||||
<button class="danger-btn" onclick={prune}>Prune unused groups</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<style>
|
|
||||||
.section-label {
|
|
||||||
font-size: 0.72em;
|
|
||||||
text-transform: uppercase;
|
|
||||||
letter-spacing: 0.06em;
|
|
||||||
opacity: 0.5;
|
|
||||||
}
|
|
||||||
|
|
||||||
.socket-list {
|
|
||||||
list-style: none;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.socket-item {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
background: var(--color-layer-2);
|
|
||||||
border-radius: 5px;
|
|
||||||
padding: 4px 8px;
|
|
||||||
outline: 1px solid var(--color-outline);
|
|
||||||
}
|
|
||||||
|
|
||||||
.remove-btn {
|
|
||||||
background: none;
|
|
||||||
border: none;
|
|
||||||
color: var(--color-text);
|
|
||||||
cursor: pointer;
|
|
||||||
opacity: 0.4;
|
|
||||||
padding: 0 2px;
|
|
||||||
font-size: 1.1em;
|
|
||||||
line-height: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.remove-btn:hover {
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.add-btn {
|
|
||||||
background: var(--color-layer-2);
|
|
||||||
color: var(--color-text);
|
|
||||||
border: none;
|
|
||||||
outline: 1px solid var(--color-outline);
|
|
||||||
border-radius: 5px;
|
|
||||||
padding: 0.4em 0.7em;
|
|
||||||
font-size: 0.8em;
|
|
||||||
cursor: pointer;
|
|
||||||
white-space: nowrap;
|
|
||||||
font-family: var(--font-family);
|
|
||||||
}
|
|
||||||
|
|
||||||
.add-btn:hover {
|
|
||||||
outline-color: var(--color-selected);
|
|
||||||
}
|
|
||||||
|
|
||||||
.danger-btn {
|
|
||||||
background: var(--color-layer-2);
|
|
||||||
color: var(--color-text);
|
|
||||||
border: none;
|
|
||||||
outline: 1px solid var(--color-outline);
|
|
||||||
border-radius: 5px;
|
|
||||||
padding: 0.4em 0.7em;
|
|
||||||
font-size: 0.8em;
|
|
||||||
cursor: pointer;
|
|
||||||
font-family: var(--font-family);
|
|
||||||
opacity: 0.7;
|
|
||||||
width: 100%;
|
|
||||||
text-align: left;
|
|
||||||
}
|
|
||||||
|
|
||||||
.danger-btn:hover {
|
|
||||||
outline-color: #e05050;
|
|
||||||
opacity: 1;
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
@@ -4,8 +4,7 @@
|
|||||||
import Grid from '$lib/grid';
|
import Grid from '$lib/grid';
|
||||||
import { debounceAsyncFunction } from '$lib/helpers';
|
import { debounceAsyncFunction } from '$lib/helpers';
|
||||||
import { createKeyMap } from '$lib/helpers/createKeyMap';
|
import { createKeyMap } from '$lib/helpers/createKeyMap';
|
||||||
import { debugNode } from '$lib/node-registry/debugNode.js';
|
import { debugNode } from '$lib/node-registry/debugNode';
|
||||||
import { groupInputNode, groupNode, groupOutputNode } from '$lib/node-registry/groupNodes.js';
|
|
||||||
import { IndexDBCache, RemoteNodeRegistry } from '$lib/node-registry/index';
|
import { IndexDBCache, RemoteNodeRegistry } from '$lib/node-registry/index';
|
||||||
import NodeStore from '$lib/node-store/NodeStore.svelte';
|
import NodeStore from '$lib/node-store/NodeStore.svelte';
|
||||||
import PerformanceViewer from '$lib/performance/PerformanceViewer.svelte';
|
import PerformanceViewer from '$lib/performance/PerformanceViewer.svelte';
|
||||||
@@ -22,7 +21,6 @@
|
|||||||
import Changelog from '$lib/sidebar/panels/Changelog.svelte';
|
import Changelog from '$lib/sidebar/panels/Changelog.svelte';
|
||||||
import ExportSettings from '$lib/sidebar/panels/ExportSettings.svelte';
|
import ExportSettings from '$lib/sidebar/panels/ExportSettings.svelte';
|
||||||
import GraphSource from '$lib/sidebar/panels/GraphSource.svelte';
|
import GraphSource from '$lib/sidebar/panels/GraphSource.svelte';
|
||||||
import GroupContextPanel from '$lib/sidebar/panels/GroupContextPanel.svelte';
|
|
||||||
import Keymap from '$lib/sidebar/panels/Keymap.svelte';
|
import Keymap from '$lib/sidebar/panels/Keymap.svelte';
|
||||||
import { panelState } from '$lib/sidebar/PanelState.svelte';
|
import { panelState } from '$lib/sidebar/PanelState.svelte';
|
||||||
import Sidebar from '$lib/sidebar/Sidebar.svelte';
|
import Sidebar from '$lib/sidebar/Sidebar.svelte';
|
||||||
@@ -39,12 +37,7 @@
|
|||||||
|
|
||||||
const registryCache = new IndexDBCache('node-registry');
|
const registryCache = new IndexDBCache('node-registry');
|
||||||
|
|
||||||
const nodeRegistry = new RemoteNodeRegistry('', registryCache, [
|
const nodeRegistry = new RemoteNodeRegistry('', registryCache, [debugNode]);
|
||||||
debugNode,
|
|
||||||
groupInputNode,
|
|
||||||
groupOutputNode,
|
|
||||||
groupNode
|
|
||||||
]);
|
|
||||||
const workerRuntime = new WorkerRuntimeExecutor();
|
const workerRuntime = new WorkerRuntimeExecutor();
|
||||||
const runtimeCache = new MemoryRuntimeCache();
|
const runtimeCache = new MemoryRuntimeCache();
|
||||||
const memoryRuntime = new MemoryRuntimeExecutor(nodeRegistry, runtimeCache);
|
const memoryRuntime = new MemoryRuntimeExecutor(nodeRegistry, runtimeCache);
|
||||||
@@ -328,7 +321,7 @@
|
|||||||
hidden={!appSettings.value.debug.advancedMode}
|
hidden={!appSettings.value.debug.advancedMode}
|
||||||
icon="i-[tabler--code]"
|
icon="i-[tabler--code]"
|
||||||
>
|
>
|
||||||
<GraphSource graph={pm.graph ?? manager?.serialize()} />
|
<GraphSource graph={manager?.serialize()} />
|
||||||
</Panel>
|
</Panel>
|
||||||
<Panel
|
<Panel
|
||||||
id="benchmark"
|
id="benchmark"
|
||||||
@@ -348,20 +341,7 @@
|
|||||||
type={graphSettingTypes}
|
type={graphSettingTypes}
|
||||||
bind:value={graphSettings}
|
bind:value={graphSettings}
|
||||||
/>
|
/>
|
||||||
{#if activeNode?.id}
|
|
||||||
<ActiveNodeSettings {manager} bind:node={activeNode} />
|
<ActiveNodeSettings {manager} bind:node={activeNode} />
|
||||||
{/if}
|
|
||||||
{#if manager?.isInsideGroup}
|
|
||||||
<GroupContextPanel
|
|
||||||
{manager}
|
|
||||||
groupId={manager.currentGroupContext!}
|
|
||||||
/>
|
|
||||||
{:else if activeNode?.type === '__virtual/group/instance'}
|
|
||||||
<GroupContextPanel
|
|
||||||
{manager}
|
|
||||||
groupId={activeNode?.props?.groupId as string}
|
|
||||||
/>
|
|
||||||
{/if}
|
|
||||||
</Panel>
|
</Panel>
|
||||||
<Panel
|
<Panel
|
||||||
id="changelog"
|
id="changelog"
|
||||||
|
|||||||
+171
-279
@@ -1,312 +1,204 @@
|
|||||||
# Nodarium - LLM Documentation
|
# Nodarium — LLM Reference
|
||||||
|
|
||||||
## Overview
|
## What It Is
|
||||||
|
|
||||||
Nodarium is a **WebAssembly-based visual programming language** for creating procedural 3D plants. The app features a node-based interface where users connect WASM modules to generate plant models in real-time. Currently used to develop https://nodes.max-richter.dev, a procedural modelling tool for 3D plants.
|
Nodarium is a **node-based visual programming editor**. Users wire together nodes on a 2D canvas; each node is a WebAssembly module that receives typed inputs and produces typed outputs. A live Three.js viewer renders the resulting 3D geometry/paths/instances.
|
||||||
|
|
||||||
## Architecture
|
---
|
||||||
|
|
||||||
### Core Components
|
## Repository Layout
|
||||||
|
|
||||||
#### 1. Node System (`app/static/nodes/`)
|
```
|
||||||
|
/
|
||||||
|
├── app/ # SvelteKit web app
|
||||||
|
│ └── src/
|
||||||
|
│ ├── routes/+page.svelte # App entry point
|
||||||
|
│ └── lib/
|
||||||
|
│ ├── graph-interface/ # Canvas editor (UI + state)
|
||||||
|
│ ├── runtime/ # WASM execution engine
|
||||||
|
│ ├── node-registry/ # Fetch & cache node definitions
|
||||||
|
│ ├── project-manager/ # IndexDB persistence
|
||||||
|
│ ├── result-viewer/ # Three.js 3D output
|
||||||
|
│ ├── sidebar/ # UI panels
|
||||||
|
│ └── settings/ # App + graph settings
|
||||||
|
├── packages/
|
||||||
|
│ ├── types/ # Shared TypeScript types + Zod schemas
|
||||||
|
│ ├── utils/ # Logging, hashing, WASM wrapping, perf
|
||||||
|
│ ├── ui/ # Reusable Svelte UI components
|
||||||
|
│ ├── planty/ # Tutorial system
|
||||||
|
│ └── macros/ # Build-time macros
|
||||||
|
└── docs/
|
||||||
|
```
|
||||||
|
|
||||||
WASM-based nodes that perform computations. All nodes must implement the NodeDefinition interface.
|
---
|
||||||
|
|
||||||
- **Node Storage**: `app/static/nodes/max/plantarium/`
|
## Core Architecture
|
||||||
- `box.wasm` - Box geometry node
|
|
||||||
- `branch.wasm` - Branch generation
|
|
||||||
- `float.wasm` - Float value node
|
|
||||||
- `gravity.wasm` - Gravity/physics node
|
|
||||||
- `instance.wasm` - Instance rendering
|
|
||||||
- `leaf.wasm` - Leaf geometry
|
|
||||||
- `math.wasm` - Math operations
|
|
||||||
- `noise.wasm` - Noise generation
|
|
||||||
- `output.wasm` - Output node
|
|
||||||
- `random.wasm` - Random value generation
|
|
||||||
- `rotate.wasm` - Rotation node
|
|
||||||
- `shape.wasm` - Shape geometry
|
|
||||||
- `stem.wasm` - Stem generation
|
|
||||||
- `triangle.wasm` - Triangle geometry
|
|
||||||
- `vec3.wasm` - Vector3 node
|
|
||||||
|
|
||||||
- **Node Registry**: `app/src/lib/node-registry.ts`
|
```
|
||||||
- Loads and manages WASM nodes
|
User Interaction
|
||||||
- `getNodeWasm()` - Creates WASM wrapper from bytes
|
└── GraphInterface
|
||||||
- `getNode()` - Retrieves node definition
|
├── GraphState ← UI: selection, camera, mouse, clipboard
|
||||||
|
└── GraphManager ← Logic: nodes, edges, history, serialization
|
||||||
|
├── NodeRegistry ← fetches WASM definitions (remote API + IndexDB cache)
|
||||||
|
├── HistoryManager ← undo/redo (jsondiffpatch deltas)
|
||||||
|
└── emit('result') → RuntimeExecutor
|
||||||
|
└── node.execute(Int32Array) per node
|
||||||
|
└── ResultViewer (Three.js/Threlte)
|
||||||
|
```
|
||||||
|
|
||||||
- **Debug Node**: `app/src/lib/node-registry/debugNode.js`
|
**Event flow:**
|
||||||
- Special debug node with wildcard inputs
|
1. User edits graph → GraphManager mutates state
|
||||||
- Variable-height nodes and parameters
|
2. GraphManager emits `save` → ProjectManager persists to IndexDB
|
||||||
- Quick-connect shortcut
|
3. GraphManager emits `result` → Runtime executes graph → Viewer updates
|
||||||
|
|
||||||
#### 2. Graph Interface
|
---
|
||||||
|
|
||||||
Visual node editor built with Svelte 5.
|
## Critical Files
|
||||||
|
|
||||||
- **Main Wrapper**: `app/src/lib/graph-interface/graph/Wrapper.svelte`
|
| File | Role |
|
||||||
- Entry point for graph interface
|
|------|------|
|
||||||
- Manages GraphManager and GraphState
|
| `app/src/routes/+page.svelte` | Wires all systems; creates GraphManager, runtime, registry |
|
||||||
|
| `app/src/lib/graph-interface/graph-manager.svelte.ts` | Central graph logic: createNode, createEdge, serialize, load, history |
|
||||||
|
| `app/src/lib/graph-interface/graph-state.svelte.ts` | UI state: camera, selection, mouse, clipboard, groupSelectedNodes |
|
||||||
|
| `app/src/lib/graph-interface/graph/Graph.svelte` | Canvas renderer |
|
||||||
|
| `app/src/lib/graph-interface/node/Node.svelte` | 3D mesh node (Three.js shader) |
|
||||||
|
| `app/src/lib/graph-interface/node/NodeHTML.svelte` | HTML overlay: labels + parameters |
|
||||||
|
| `app/src/lib/graph-interface/node/NodeHeader.svelte` | Node title bar |
|
||||||
|
| `app/src/lib/graph-interface/keymaps.ts` | Keyboard shortcuts |
|
||||||
|
| `app/src/lib/graph-interface/helpers/nodeHelpers.ts` | Node height calculations |
|
||||||
|
| `app/src/lib/graph-interface/graph/colors.svelte.ts` | Socket type → color mapping |
|
||||||
|
| `app/src/lib/runtime/runtime-executor.ts` | Executes nodes in DAG order; expandGroups() |
|
||||||
|
| `app/src/lib/node-registry/index.ts` | RemoteNodeRegistry entry |
|
||||||
|
| `app/src/lib/node-registry/groupNode.ts` | Built-in group node definition |
|
||||||
|
| `app/src/lib/node-registry/debugNode.ts` | Built-in debug node |
|
||||||
|
| `app/src/lib/sidebar/panels/ActiveNodeSettings.svelte` | Per-node settings panel |
|
||||||
|
| `packages/types/src/types.ts` | Graph, NodeInstance, NodeDefinition, Edge, GroupDefinition |
|
||||||
|
| `packages/types/src/inputs.ts` | NodeInput union types (float, vec3, geometry, path, …) |
|
||||||
|
| `packages/utils/src/wasm.ts` | createWasmWrapper() — wraps WASM bytes into a NodeDefinition |
|
||||||
|
|
||||||
- **GraphManager**: `app/src/lib/graph-interface/graph-manager.svelte.ts`
|
---
|
||||||
- Core entity managing the node graph
|
|
||||||
- Handles node connections and execution flow
|
|
||||||
|
|
||||||
- **GraphState**: `app/src/lib/graph-interface/graph-state.svelte.ts`
|
## Key Types
|
||||||
- Tracks UI state (selection, snapping, help, active nodes)
|
|
||||||
|
|
||||||
- **Graph Components**:
|
|
||||||
- `app/src/lib/graph-interface/graph/` - Graph rendering
|
|
||||||
- `app/src/lib/graph-interface/node/` - Node rendering
|
|
||||||
- `app/src/lib/graph-interface/edges/` - Edge rendering
|
|
||||||
- `app/src/lib/graph-interface/components/` - UI components (AddMenu, Socket, etc.)
|
|
||||||
- `app/src/lib/graph-interface/debug/` - Debug overlays
|
|
||||||
- `app/src/lib/graph-interface/background/` - Grid/dots backgrounds
|
|
||||||
|
|
||||||
- **Helpers**:
|
|
||||||
- `app/src/lib/helpers/` - Utility functions
|
|
||||||
- `app/src/lib/helpers/createKeyMap.ts` - Keyboard shortcuts
|
|
||||||
|
|
||||||
#### 3. Runtime Execution
|
|
||||||
|
|
||||||
Performs graph execution via WASM nodes.
|
|
||||||
|
|
||||||
- **Runtime Executors** (`app/src/lib/runtime/`):
|
|
||||||
- **MemoryRuntime**: Direct WASM execution in main thread
|
|
||||||
- **WorkerRuntime**: WebWorker-based execution for performance
|
|
||||||
- Both implement the RuntimeExecutor interface
|
|
||||||
|
|
||||||
- **Runtime Cache**: `app/src/lib/runtime/cache.ts`
|
|
||||||
- Memory-based caching for graph execution
|
|
||||||
|
|
||||||
- **Execution Flow**:
|
|
||||||
1. Graph serialized from graph interface
|
|
||||||
2. Runtime executes nodes in topological order
|
|
||||||
3. Results passed through connected edges
|
|
||||||
4. Final mesh output rendered
|
|
||||||
|
|
||||||
#### 4. 3D Viewer (`app/src/lib/result-viewer/`)
|
|
||||||
|
|
||||||
Three.js-based rendering for 3D output.
|
|
||||||
|
|
||||||
- **Viewer**: `app/src/lib/result-viewer/Viewer.svelte`
|
|
||||||
- Renders generated 3D meshes
|
|
||||||
- Uses @threlte/core (Svelte-Three.js wrapper)
|
|
||||||
|
|
||||||
#### 5. Application Structure (`app/src/routes/`)
|
|
||||||
|
|
||||||
SvelteKit application routing.
|
|
||||||
|
|
||||||
- **Main Page**: `app/src/routes/+page.svelte`
|
|
||||||
- Combines GraphInterface + 3D Viewer
|
|
||||||
- Manages runtime selection (memory vs worker)
|
|
||||||
- Handles settings and performance tracking
|
|
||||||
|
|
||||||
- **Layout**: `app/src/routes/+layout.svelte`
|
|
||||||
- Application shell
|
|
||||||
|
|
||||||
- **Server**: `app/src/routes/+layout.server.ts`
|
|
||||||
- Loads git metadata and changelog
|
|
||||||
|
|
||||||
#### 6. Settings System (`app/src/lib/settings/`)
|
|
||||||
|
|
||||||
Application and graph settings.
|
|
||||||
|
|
||||||
- **App Settings**: `app/src/lib/settings/app-settings.svelte.ts`
|
|
||||||
- Debug mode, themes, node interface options
|
|
||||||
|
|
||||||
- **NestedSettings**: `app/src/lib/settings/NestedSettings.svelte`
|
|
||||||
- Recursive settings UI component
|
|
||||||
|
|
||||||
#### 7. Sidebar Panels (`app/src/lib/sidebar/`)
|
|
||||||
|
|
||||||
- `app/src/lib/sidebar/Sidebar.svelte` - Main sidebar
|
|
||||||
- `app/src/lib/sidebar/panels/` - Individual panels:
|
|
||||||
- `ActiveNodeSettings.svelte` - Selected node properties
|
|
||||||
- `BenchmarkPanel.svelte` - Performance benchmarking
|
|
||||||
- `Changelog.svelte` - Version history
|
|
||||||
- `ExportSettings.svelte` - Export options
|
|
||||||
- `GraphSource.svelte` - Graph JSON view
|
|
||||||
- `Keymap.svelte` - Keyboard shortcuts
|
|
||||||
|
|
||||||
#### 8. Project Management (`app/src/lib/project-manager/`)
|
|
||||||
|
|
||||||
- `app/src/lib/project-manager/project-manager.svelte` - Project save/load
|
|
||||||
- Uses IndexedDB for persistence
|
|
||||||
|
|
||||||
#### 9. Node Store (`app/src/lib/node-store/`)
|
|
||||||
|
|
||||||
- `app/src/lib/node-store/NodeStore.svelte`
|
|
||||||
- Remote node registry management
|
|
||||||
- IndexDBCache for offline storage
|
|
||||||
|
|
||||||
#### 10. Graph Templates (`app/src/lib/graph-templates/`)
|
|
||||||
|
|
||||||
Pre-built graph templates for testing:
|
|
||||||
|
|
||||||
- Grid, Tree, LottaFaces, LottaNodes, LottaNodesAndFaces
|
|
||||||
|
|
||||||
## Key Types (`app/src/lib/types.ts`)
|
|
||||||
|
|
||||||
```typescript
|
```typescript
|
||||||
interface NodeDefinition {
|
// packages/types/src/types.ts
|
||||||
id: string;
|
|
||||||
name: string;
|
type NodeId = `${string}/${string}/${string}` // e.g. "max/plantarium/stem"
|
||||||
inputs: Socket[];
|
|
||||||
outputs: Socket[];
|
type NodeInstance = {
|
||||||
parameters: Parameter[];
|
id: number
|
||||||
execute: (inputs: any[], parameters: any[]) => any[];
|
type: NodeId
|
||||||
|
position: [number, number]
|
||||||
|
props?: Record<string, number | number[]> // current parameter values
|
||||||
|
meta?: { title?: string; lastModified?: string }
|
||||||
|
state: NodeRuntimeState // runtime-only, NOT serialized
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Socket {
|
type NodeRuntimeState = {
|
||||||
id: string;
|
type?: NodeDefinition // resolved definition
|
||||||
name: string;
|
parents?: NodeInstance[]
|
||||||
type: string; // datatype (e.g., "number", "vec3", "*")
|
children?: NodeInstance[]
|
||||||
defaultValue?: any;
|
x?: number; y?: number // interpolated position
|
||||||
optional?: boolean;
|
mesh?: Mesh // Three.js mesh reference
|
||||||
|
ref?: HTMLElement
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Parameter {
|
type NodeDefinition = {
|
||||||
id: string;
|
id: NodeId
|
||||||
name: string;
|
inputs?: Record<string, NodeInput>
|
||||||
type: string;
|
outputs?: string[] // output type names
|
||||||
defaultValue: any;
|
meta?: { title?: string; description?: string }
|
||||||
min?: number;
|
execute(input: Int32Array): Int32Array // WASM function
|
||||||
max?: number;
|
|
||||||
options?: string[];
|
|
||||||
}
|
}
|
||||||
|
|
||||||
interface Graph {
|
// Edge: [fromNode, outputIndex, toNode, inputSocketName]
|
||||||
nodes: NodeInstance[];
|
type Edge = [NodeInstance, number, NodeInstance, string]
|
||||||
edges: Edge[];
|
|
||||||
|
type Graph = {
|
||||||
|
nodes: NodeInstance[]
|
||||||
|
edges: [number, number, number, string][] // serialized (IDs, not refs)
|
||||||
|
settings: Record<string, unknown>
|
||||||
|
groups: GroupDefinition[]
|
||||||
}
|
}
|
||||||
|
|
||||||
interface NodeInstance {
|
type GroupDefinition = {
|
||||||
id: number;
|
id: number
|
||||||
nodeId: string;
|
nodes: NodeInstance[]
|
||||||
position: { x: number; y: number };
|
edges: Edge[]
|
||||||
parameters: Record<string, any>;
|
inputs?: Record<string, NodeInput>
|
||||||
}
|
outputs?: string[]
|
||||||
|
|
||||||
interface Edge {
|
|
||||||
id: number;
|
|
||||||
fromNode: number;
|
|
||||||
fromSocket: string;
|
|
||||||
toNode: number;
|
|
||||||
toSocket: string;
|
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
## Development Workflow
|
### NodeInput socket types
|
||||||
|
`float` | `integer` | `boolean` | `select` | `seed` | `vec3` | `geometry` | `path` | `shape` | `color` | `*` (wildcard)
|
||||||
|
|
||||||
### Prerequisites
|
Each input can have: `value` (default), `label`, `hidden`, `external`, `setting` (link to graph setting), `accepts` (extra compatible types).
|
||||||
|
|
||||||
- Node.js
|
---
|
||||||
- pnpm
|
|
||||||
- Rust
|
|
||||||
- wasm-pack
|
|
||||||
|
|
||||||
### Build Commands
|
## Patterns & Conventions
|
||||||
|
|
||||||
|
### Svelte 5 reactivity
|
||||||
|
The codebase uses Svelte 5 runes throughout — `$state`, `$derived`, `$effect`. Collections use `SvelteMap<K,V>` and `SvelteSet<T>` (from `svelte/reactivity`) instead of plain Map/Set so that mutations trigger reactive updates.
|
||||||
|
|
||||||
|
### Context API
|
||||||
|
`GraphManager` and `GraphState` are provided via Svelte context (`setContext` / `getContext`) inside `GraphInterface`. All child components (Node, Edge, etc.) consume them via context rather than props.
|
||||||
|
|
||||||
|
### Edge representation
|
||||||
|
In memory, edges are `[NodeInstance, outputIndex, NodeInstance, inputSocketName]` — direct object references for fast traversal. On serialization (`Graph.edges`), they become `[nodeId, outputIndex, nodeId, inputSocketName]` (plain IDs).
|
||||||
|
|
||||||
|
### Socket compatibility
|
||||||
|
```typescript
|
||||||
|
areSocketsCompatible(outputType: string, inputType: string | string[]): boolean
|
||||||
|
// '*' wildcard matches any type; 'geometry' accepts ['geometry', 'instances']
|
||||||
|
```
|
||||||
|
|
||||||
|
### WASM execution interface
|
||||||
|
Every node exposes a single function: `execute(input: Int32Array): Int32Array`.
|
||||||
|
Data encoding (Plantarium):
|
||||||
|
- `[0, stemDepth, ...x,y,z,thickness]` — path
|
||||||
|
- `[1, vertexCount, faceCount, ...faces, ...vertices, ...normals]` — geometry
|
||||||
|
- `[2, vertexCount, faceCount, instanceCount, stemDepth, ...]` — instances
|
||||||
|
|
||||||
|
### Event emitter
|
||||||
|
`GraphManager extends EventEmitter<{ save, result, settings }>`. Subscribe with `manager.on('result', cb)`. Used to decouple the editor UI from runtime execution and persistence.
|
||||||
|
|
||||||
|
### History
|
||||||
|
Every mutation goes through `HistoryManager`. Call `this.history.save(this.serialize())` before mutations; undo/redo replays jsondiffpatch deltas.
|
||||||
|
|
||||||
|
### Internal node IDs
|
||||||
|
Built-in nodes use the `__internal/` namespace: `__internal/group/instance`, `__internal/node/debug`. Virtual boundary nodes use `__virtual/`: `__virtual/group/input`, `__virtual/group/output`.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## In-Progress: Node Groups (`feat/group-node-own`)
|
||||||
|
|
||||||
|
Group selected nodes with **Ctrl+G**. A `GroupDefinition` is stored in `Graph.groups[]`; a group instance node (`__internal/group/instance`) referencing it by `props.groupId` replaces the selected nodes.
|
||||||
|
|
||||||
|
**Known gaps as of 2026-05-03:**
|
||||||
|
|
||||||
|
| Issue | Location |
|
||||||
|
|-------|----------|
|
||||||
|
| `createGroupNode()` called but not defined | `graph-state.svelte.ts:334` → missing in `graph-manager.svelte.ts` |
|
||||||
|
| Runtime expects `group.graph.nodes/edges`; schema has flat `nodes/edges` | `runtime-executor.ts` vs `types.ts` |
|
||||||
|
| Runtime expects `group.inputs` as array; schema defines it as `Record<string, NodeInput>` | Same mismatch |
|
||||||
|
| `enterGroupNode()` is a stub — no group navigation | `graph-state.svelte.ts` |
|
||||||
|
| `serialize()` writes parent-graph edges into group instead of group-internal edges | `graph-manager.svelte.ts` |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Dev Commands
|
||||||
|
|
||||||
|
Run from `app/`:
|
||||||
|
|
||||||
```bash
|
```bash
|
||||||
# Install dependencies
|
npm run dev # start dev server (Vite)
|
||||||
pnpm i
|
npm run build # production build
|
||||||
|
npm run check # svelte-check + tsc
|
||||||
# Build WASM nodes
|
npm run lint # eslint
|
||||||
pnpm build:nodes
|
npm run test # unit (vitest) + e2e (playwright)
|
||||||
|
npm run test:unit # vitest only
|
||||||
# Start development server
|
npm run test:e2e # playwright only
|
||||||
cd app && pnpm dev
|
npm run bench # benchmark runner
|
||||||
|
|
||||||
# Run tests
|
|
||||||
cd app && pnpm test
|
|
||||||
|
|
||||||
# Lint and typecheck
|
|
||||||
cd app && pnpm lint
|
|
||||||
cd app && pnpm check
|
|
||||||
|
|
||||||
# Format code
|
|
||||||
cd app && pnpm format
|
|
||||||
```
|
```
|
||||||
|
|
||||||
### Creating New Nodes
|
|
||||||
|
|
||||||
See `docs/DEVELOPING_NODES.md` for detailed instructions on creating custom WASM nodes.
|
|
||||||
|
|
||||||
## Features
|
|
||||||
|
|
||||||
### Current Features
|
|
||||||
|
|
||||||
- Visual node-based programming with real-time 3D preview
|
|
||||||
- WebAssembly nodes for high-performance computation
|
|
||||||
- Debug node with wildcard inputs and runtime integration
|
|
||||||
- Color-coded node sockets and edges (indicating data types)
|
|
||||||
- Variable-height nodes and parameters
|
|
||||||
- Edge dragging with valid socket highlighting
|
|
||||||
- InputNumber snapping to predefined values (Alt+click)
|
|
||||||
- Project save/load with IndexedDB
|
|
||||||
- Performance monitoring and benchmarking
|
|
||||||
- Changelog viewer
|
|
||||||
- Advanced mode settings
|
|
||||||
|
|
||||||
### UI Components
|
|
||||||
|
|
||||||
- **InputNumber**: Numeric input with arrow controls
|
|
||||||
- **InputColor**: Color picker
|
|
||||||
- **InputShape**: Shape selector with preview
|
|
||||||
- **InputSelect**: Dropdown with options
|
|
||||||
|
|
||||||
## File Structure
|
|
||||||
|
|
||||||
```
|
|
||||||
nodarium/
|
|
||||||
├── app/
|
|
||||||
│ ├── src/
|
|
||||||
│ │ ├── lib/
|
|
||||||
│ │ │ ├── config.ts
|
|
||||||
│ │ │ ├── graph-interface/ # Node editor
|
|
||||||
│ │ │ ├── graph-manager.svelte.ts
|
|
||||||
│ │ │ ├── graph-state.svelte.ts
|
|
||||||
│ │ │ ├── graph-templates/ # Test templates
|
|
||||||
│ │ │ ├── grid/
|
|
||||||
│ │ │ ├── helpers/
|
|
||||||
│ │ │ ├── node-registry.ts
|
|
||||||
│ │ │ ├── node-registry/ # Node loading
|
|
||||||
│ │ │ ├── node-store/
|
|
||||||
│ │ │ ├── performance/
|
|
||||||
│ │ │ ├── project-manager/
|
|
||||||
│ │ │ ├── result-viewer/ # 3D viewer
|
|
||||||
│ │ │ ├── runtime/ # Execution
|
|
||||||
│ │ │ ├── settings/ # App settings
|
|
||||||
│ │ │ ├── sidebar/
|
|
||||||
│ │ │ └── types.ts
|
|
||||||
│ │ └── routes/
|
|
||||||
│ │ ├── +page.svelte
|
|
||||||
│ │ └── +layout.svelte
|
|
||||||
│ ├── static/
|
|
||||||
│ │ └── nodes/
|
|
||||||
│ │ └── max/
|
|
||||||
│ │ └── plantarium/ # WASM nodes
|
|
||||||
│ └── package.json
|
|
||||||
├── docs/
|
|
||||||
│ ├── ARCHITECTURE.md
|
|
||||||
│ ├── DEVELOPING_NODES.md
|
|
||||||
│ ├── NODE_DEFINITION.md
|
|
||||||
│ └── PLANTARIUM.md
|
|
||||||
├── nodes/ # WASM node source (Rust)
|
|
||||||
└── package.json
|
|
||||||
```
|
|
||||||
|
|
||||||
## Release Process
|
|
||||||
|
|
||||||
1. Create annotated tag:
|
|
||||||
```bash
|
|
||||||
git tag -a v1.0.0 -m "Release notes"
|
|
||||||
git push origin v1.0.0
|
|
||||||
```
|
|
||||||
|
|
||||||
2. CI workflow:
|
|
||||||
- Runs lint, format check, type check
|
|
||||||
- Builds project
|
|
||||||
- Updates package.json versions
|
|
||||||
- Generates CHANGELOG.md
|
|
||||||
- Creates Gitea release
|
|
||||||
|
|||||||
+2
-1
@@ -6,7 +6,8 @@
|
|||||||
"qa": "pnpm lint && pnpm check && pnpm test",
|
"qa": "pnpm lint && pnpm check && pnpm test",
|
||||||
"format": "pnpm dprint fmt",
|
"format": "pnpm dprint fmt",
|
||||||
"format:check": "pnpm dprint check",
|
"format:check": "pnpm dprint check",
|
||||||
"test": "pnpm run -r --parallel test",
|
"test:e2e": "pnpm run -r --parallel test:e2e",
|
||||||
|
"test:unit": "pnpm run -r --parallel test:unit",
|
||||||
"check": "pnpm run -r --parallel check",
|
"check": "pnpm run -r --parallel check",
|
||||||
"build": "pnpm build:nodes && pnpm build:app",
|
"build": "pnpm build:nodes && pnpm build:app",
|
||||||
"build:app": "BASE_PATH=/ui pnpm -r --filter 'ui' build && pnpm -r --filter 'app'... build",
|
"build:app": "BASE_PATH=/ui pnpm -r --filter 'ui' build && pnpm -r --filter 'app'... build",
|
||||||
|
|||||||
@@ -4,13 +4,12 @@ export type {
|
|||||||
Box,
|
Box,
|
||||||
Edge,
|
Edge,
|
||||||
Graph,
|
Graph,
|
||||||
GroupSocket,
|
GroupDefinition,
|
||||||
NodeDefinition,
|
NodeDefinition,
|
||||||
NodeGroupDefinition,
|
|
||||||
NodeId,
|
NodeId,
|
||||||
NodeInstance,
|
NodeInstance,
|
||||||
SerializedNode,
|
SerializedNode,
|
||||||
Socket
|
Socket
|
||||||
} from './types';
|
} from './types';
|
||||||
export { GraphSchema, NodeSchema } from './types';
|
export { GraphSchema, GroupSchema, NodeSchema } from './types';
|
||||||
export { NodeDefinitionSchema } from './types';
|
export { NodeDefinitionSchema } from './types';
|
||||||
|
|||||||
@@ -61,10 +61,7 @@ export const NodeInputBooleanSchema = z.object({
|
|||||||
export const NodeInputSelectSchema = z.object({
|
export const NodeInputSelectSchema = z.object({
|
||||||
...DefaultOptionsSchema.shape,
|
...DefaultOptionsSchema.shape,
|
||||||
type: z.literal('select'),
|
type: z.literal('select'),
|
||||||
options: z.union([
|
options: z.array(z.string()).optional(),
|
||||||
z.array(z.string()),
|
|
||||||
z.array(z.object({ label: z.string(), value: z.string() }))
|
|
||||||
]).optional(),
|
|
||||||
value: z.string().optional()
|
value: z.string().optional()
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|||||||
+12
-26
@@ -51,7 +51,7 @@ export const NodeSchema = z.object({
|
|||||||
id: z.number(),
|
id: z.number(),
|
||||||
type: NodeIdSchema,
|
type: NodeIdSchema,
|
||||||
props: z
|
props: z
|
||||||
.record(z.string(), z.union([z.number(), z.array(z.number()), z.string()]))
|
.record(z.string(), z.union([z.number(), z.array(z.number())]))
|
||||||
.optional(),
|
.optional(),
|
||||||
meta: z
|
meta: z
|
||||||
.object({
|
.object({
|
||||||
@@ -76,33 +76,19 @@ export type Socket = {
|
|||||||
|
|
||||||
export type Edge = [NodeInstance, number, NodeInstance, string];
|
export type Edge = [NodeInstance, number, NodeInstance, string];
|
||||||
|
|
||||||
export type GroupSocket = {
|
export const GroupSchema = z.object({
|
||||||
name: string;
|
id: z.number(),
|
||||||
type: string;
|
|
||||||
};
|
|
||||||
|
|
||||||
export type NodeGroupDefinition = {
|
|
||||||
id: string;
|
|
||||||
name: string;
|
|
||||||
inputs: GroupSocket[];
|
|
||||||
outputs: GroupSocket[];
|
|
||||||
graph: {
|
|
||||||
nodes: SerializedNode[];
|
|
||||||
edges: [number, number, number, string][];
|
|
||||||
};
|
|
||||||
};
|
|
||||||
|
|
||||||
const NodeGroupDefinitionSchema: z.ZodType<NodeGroupDefinition> = z.object({
|
|
||||||
id: z.string(),
|
|
||||||
name: z.string(),
|
|
||||||
inputs: z.array(z.object({ name: z.string(), type: z.string() })),
|
|
||||||
outputs: z.array(z.object({ name: z.string(), type: z.string() })),
|
|
||||||
graph: z.object({
|
|
||||||
nodes: z.array(NodeSchema),
|
nodes: z.array(NodeSchema),
|
||||||
edges: z.array(z.tuple([z.number(), z.number(), z.number(), z.string()]))
|
edges: z.array(z.tuple([z.number(), z.number(), z.number(), z.string()])),
|
||||||
})
|
inputs: z.record(z.string(), NodeInputSchema).optional(),
|
||||||
|
outputs: z.array(z.object({
|
||||||
|
type: z.string(),
|
||||||
|
label: z.string().optional()
|
||||||
|
})).optional()
|
||||||
});
|
});
|
||||||
|
|
||||||
|
export type GroupDefinition = z.infer<typeof GroupSchema>;
|
||||||
|
|
||||||
export const GraphSchema = z.object({
|
export const GraphSchema = z.object({
|
||||||
id: z.number(),
|
id: z.number(),
|
||||||
meta: z
|
meta: z
|
||||||
@@ -114,7 +100,7 @@ export const GraphSchema = z.object({
|
|||||||
settings: z.record(z.string(), z.any()).optional(),
|
settings: z.record(z.string(), z.any()).optional(),
|
||||||
nodes: z.array(NodeSchema),
|
nodes: z.array(NodeSchema),
|
||||||
edges: z.array(z.tuple([z.number(), z.number(), z.number(), z.string()])),
|
edges: z.array(z.tuple([z.number(), z.number(), z.number(), z.string()])),
|
||||||
groups: z.record(z.string(), NodeGroupDefinitionSchema).optional()
|
groups: z.array(GroupSchema)
|
||||||
});
|
});
|
||||||
|
|
||||||
export type Graph = z.infer<typeof GraphSchema>;
|
export type Graph = z.infer<typeof GraphSchema>;
|
||||||
|
|||||||
@@ -40,7 +40,7 @@
|
|||||||
{:else if input.type === 'boolean'}
|
{:else if input.type === 'boolean'}
|
||||||
<InputCheckbox bind:value={value as boolean} {id} />
|
<InputCheckbox bind:value={value as boolean} {id} />
|
||||||
{:else if input.type === 'select'}
|
{:else if input.type === 'select'}
|
||||||
<InputSelect bind:value={value as number | string} options={input.options} {id} />
|
<InputSelect bind:value={value as number} options={input.options} {id} />
|
||||||
{:else if input.type === 'vec3'}
|
{:else if input.type === 'vec3'}
|
||||||
<InputVec3 bind:value={value as [number, number, number]} {id} />
|
<InputVec3 bind:value={value as [number, number, number]} {id} />
|
||||||
{/if}
|
{/if}
|
||||||
|
|||||||
@@ -0,0 +1,137 @@
|
|||||||
|
<script module>
|
||||||
|
const cache = new Map<string, Record<string, boolean>>();
|
||||||
|
|
||||||
|
function getStore(root: string): Record<string, boolean> {
|
||||||
|
if (!cache.has(root)) {
|
||||||
|
try {
|
||||||
|
const raw = localStorage.getItem(`json_viewer:${root}`);
|
||||||
|
cache.set(root, raw ? JSON.parse(raw) : {});
|
||||||
|
} catch {
|
||||||
|
cache.set(root, {});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return cache.get(root)!;
|
||||||
|
}
|
||||||
|
|
||||||
|
function readOpen(path: string, fallback: boolean): boolean {
|
||||||
|
const root = path.split('/')[0];
|
||||||
|
const store = getStore(root);
|
||||||
|
return path in store ? store[path] : fallback;
|
||||||
|
}
|
||||||
|
|
||||||
|
function writeOpen(path: string, value: boolean) {
|
||||||
|
const root = path.split('/')[0];
|
||||||
|
const store = getStore(root);
|
||||||
|
store[path] = value;
|
||||||
|
try {
|
||||||
|
localStorage.setItem(`json_viewer:${root}`, JSON.stringify(store));
|
||||||
|
} catch { /* quota exceeded etc */ }
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script lang="ts">
|
||||||
|
import { browser } from '$app/environment';
|
||||||
|
import JsonViewer from './JsonViewer.svelte';
|
||||||
|
|
||||||
|
let {
|
||||||
|
value,
|
||||||
|
key,
|
||||||
|
depth = 0,
|
||||||
|
path = ''
|
||||||
|
}: { value: unknown; key?: string; depth?: number; path?: string } = $props();
|
||||||
|
|
||||||
|
const defaultOpen = $derived(depth < 4);
|
||||||
|
let open = $derived(browser && path ? readOpen(path, defaultOpen) : defaultOpen);
|
||||||
|
let flashing = $state(false);
|
||||||
|
|
||||||
|
const isArr = $derived(Array.isArray(value));
|
||||||
|
const isExpandable = $derived(value !== null && typeof value === 'object');
|
||||||
|
const open_bracket = $derived(isArr ? '[' : '{');
|
||||||
|
const close_bracket = $derived(isArr ? ']' : '}');
|
||||||
|
const items = $derived.by(() => {
|
||||||
|
if (isArr) {
|
||||||
|
return (value as unknown[]).map((v, i) => [String(i), v] as [string, unknown]);
|
||||||
|
}
|
||||||
|
if (value !== null && typeof value === 'object') {
|
||||||
|
return Object.entries(value as Record<string, unknown>).filter(
|
||||||
|
([, v]) => v !== undefined
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return [] as [string, unknown][];
|
||||||
|
});
|
||||||
|
const showKeys = $derived(!isArr || typeof items[0]?.[1] === "object")
|
||||||
|
|
||||||
|
function toggle(next: boolean) {
|
||||||
|
open = next;
|
||||||
|
if (browser && path) writeOpen(path, next);
|
||||||
|
}
|
||||||
|
|
||||||
|
let prevJson = '';
|
||||||
|
let flashTimeout: ReturnType<typeof setTimeout> | null = null;
|
||||||
|
|
||||||
|
$effect(() => {
|
||||||
|
const json = JSON.stringify(value);
|
||||||
|
if (prevJson && json !== prevJson) {
|
||||||
|
if (flashTimeout) clearTimeout(flashTimeout);
|
||||||
|
flashing = true;
|
||||||
|
flashTimeout = setTimeout(() => {
|
||||||
|
flashing = false;
|
||||||
|
flashTimeout = null;
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
prevJson = json;
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<span
|
||||||
|
class="font-mono text-xs leading-[1.6] rounded transition-[background-color] duration-500"
|
||||||
|
class:bg-layer-3={flashing}
|
||||||
|
>
|
||||||
|
{#if key !== undefined}
|
||||||
|
<span class="text-text">{key}</span><span class="text-text/40">: </span>
|
||||||
|
{/if}
|
||||||
|
|
||||||
|
{#if isExpandable}
|
||||||
|
{#if items.length === 0}
|
||||||
|
<span class="text-text/50">{open_bracket}{close_bracket}</span>
|
||||||
|
{:else if open}
|
||||||
|
{#if depth > 0}
|
||||||
|
<button class="w-3 text-text/50 hover:text-text" onclick={() => toggle(false)}>
|
||||||
|
▼
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
<span class="text-text/50">{open_bracket}</span>
|
||||||
|
<div class="pl-4 border-l border-outline">
|
||||||
|
{#each items as [k, v], i (k)}
|
||||||
|
<div>
|
||||||
|
<JsonViewer
|
||||||
|
value={v}
|
||||||
|
key={showKeys ? k : undefined }
|
||||||
|
depth={depth + 1}
|
||||||
|
path={path ? `${path}/${k}` : k}
|
||||||
|
/>{#if i < items.length - 1}<span class="text-text/20">,</span>{/if}
|
||||||
|
</div>
|
||||||
|
{/each}
|
||||||
|
</div>
|
||||||
|
<span class="text-text/50">{close_bracket}</span>
|
||||||
|
{:else}
|
||||||
|
<button
|
||||||
|
class="inline text-text/50 hover:text-text"
|
||||||
|
onclick={() => toggle(true)}
|
||||||
|
>
|
||||||
|
<span class="w-3 inline-block">▶</span>
|
||||||
|
{open_bracket}<span class="text-text/40 mx-1">{items.length}</span>{close_bracket}
|
||||||
|
</button>
|
||||||
|
{/if}
|
||||||
|
{:else if value === null}
|
||||||
|
<span class="text-emerald-500!">null</span>
|
||||||
|
{:else if typeof value === 'boolean'}
|
||||||
|
<span class="text-blue-500!">{value}</span>
|
||||||
|
{:else if typeof value === 'number'}
|
||||||
|
<span class="text-orange-400!">{value}</span>
|
||||||
|
{:else if typeof value === 'string'}
|
||||||
|
<span class="text-emerald-500!">"{value}"</span>
|
||||||
|
{:else}
|
||||||
|
<span class="text-text/70">{String(value)}</span>
|
||||||
|
{/if}
|
||||||
|
</span>
|
||||||
@@ -7,6 +7,7 @@ export { default as InputShape } from './inputs/InputShape.svelte';
|
|||||||
export { default as InputVec3 } from './inputs/InputVec3.svelte';
|
export { default as InputVec3 } from './inputs/InputVec3.svelte';
|
||||||
|
|
||||||
export { default as Details } from './Details.svelte';
|
export { default as Details } from './Details.svelte';
|
||||||
|
export { default as JsonViewer } from './JsonViewer.svelte';
|
||||||
export { default as ShortCut } from './ShortCut.svelte';
|
export { default as ShortCut } from './ShortCut.svelte';
|
||||||
|
|
||||||
import Input from './Input.svelte';
|
import Input from './Input.svelte';
|
||||||
|
|||||||
@@ -1,28 +1,17 @@
|
|||||||
<script lang="ts">
|
<script lang="ts">
|
||||||
type StringOption = string;
|
|
||||||
type LabeledOption = { label: string; value: string };
|
|
||||||
|
|
||||||
interface Props {
|
interface Props {
|
||||||
options?: StringOption[] | LabeledOption[];
|
options?: string[];
|
||||||
value?: number | string;
|
value?: number;
|
||||||
id?: string;
|
id?: string;
|
||||||
}
|
}
|
||||||
|
|
||||||
let { options = [], value = $bindable<number | string>(0), id = '' }: Props = $props();
|
let { options = [], value = $bindable(0), id = '' }: Props = $props();
|
||||||
|
|
||||||
const isLabeled = $derived(options.length > 0 && typeof options[0] === 'object');
|
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<select {id} bind:value class="bg-layer-2 text-text">
|
<select {id} bind:value class="bg-layer-2 text-text">
|
||||||
{#if isLabeled}
|
|
||||||
{#each options as opt ((opt as LabeledOption).value)}
|
|
||||||
<option value={(opt as LabeledOption).value}>{(opt as LabeledOption).label}</option>
|
|
||||||
{/each}
|
|
||||||
{:else}
|
|
||||||
{#each options as label, i (label)}
|
{#each options as label, i (label)}
|
||||||
<option value={i}>{label as string}</option>
|
<option value={i}>{label}</option>
|
||||||
{/each}
|
{/each}
|
||||||
{/if}
|
|
||||||
</select>
|
</select>
|
||||||
|
|
||||||
<style>
|
<style>
|
||||||
|
|||||||
@@ -8,6 +8,7 @@
|
|||||||
InputSelect,
|
InputSelect,
|
||||||
InputShape,
|
InputShape,
|
||||||
InputVec3,
|
InputVec3,
|
||||||
|
JsonViewer,
|
||||||
ShortCut
|
ShortCut
|
||||||
} from '$lib';
|
} from '$lib';
|
||||||
import Section from './Section.svelte';
|
import Section from './Section.svelte';
|
||||||
@@ -25,6 +26,32 @@
|
|||||||
let colorValue = $state<[number, number, number]>([59, 130, 246]);
|
let colorValue = $state<[number, number, number]>([59, 130, 246]);
|
||||||
let mirrorShape = $state(true);
|
let mirrorShape = $state(true);
|
||||||
let detailsOpen = $state(false);
|
let detailsOpen = $state(false);
|
||||||
|
let jsonValue = $state({
|
||||||
|
id: 1,
|
||||||
|
nodes: [{ id: 0, type: 'max/test/node', position: [0, 0] }, {
|
||||||
|
id: 1,
|
||||||
|
type: 'max/test/other',
|
||||||
|
position: [100, 50]
|
||||||
|
}],
|
||||||
|
edges: [[0, 0, 1, 'input']],
|
||||||
|
groups: [],
|
||||||
|
settings: { seed: 42, enabled: true }
|
||||||
|
});
|
||||||
|
|
||||||
|
function randomlyUpdateJson() {
|
||||||
|
const rand = Math.floor(Math.random() * 5);
|
||||||
|
if (rand === 0) {
|
||||||
|
jsonValue.nodes[0].position[0] += 1;
|
||||||
|
} else if (rand === 1) {
|
||||||
|
jsonValue.nodes[0].position[1] += 1;
|
||||||
|
} else if (rand === 2) {
|
||||||
|
jsonValue.settings.seed += 1;
|
||||||
|
} else if (rand === 3) {
|
||||||
|
jsonValue.settings.enabled = !jsonValue.settings.enabled;
|
||||||
|
} else if (rand === 4) {
|
||||||
|
jsonValue.id += Math.floor(Math.random() * 10 - 5);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
let points = $state([]);
|
let points = $state([]);
|
||||||
let theme = $state('dark');
|
let theme = $state('dark');
|
||||||
@@ -56,6 +83,7 @@
|
|||||||
</Section>
|
</Section>
|
||||||
|
|
||||||
<Section title="Select" value={d}>
|
<Section title="Select" value={d}>
|
||||||
|
<i>Select with simple values</i>
|
||||||
<InputSelect bind:value={selectValue} {options} />
|
<InputSelect bind:value={selectValue} {options} />
|
||||||
</Section>
|
</Section>
|
||||||
|
|
||||||
@@ -86,6 +114,23 @@
|
|||||||
</Details>
|
</Details>
|
||||||
</Section>
|
</Section>
|
||||||
|
|
||||||
|
<Section title="JsonViewer">
|
||||||
|
{#snippet header()}
|
||||||
|
<button
|
||||||
|
onclick={() => randomlyUpdateJson()}
|
||||||
|
class="-mt-1 bg-layer-2 p-1 px-2 rounded-sm cursor-pointer"
|
||||||
|
>
|
||||||
|
update
|
||||||
|
</button>
|
||||||
|
{/snippet}
|
||||||
|
<div class="w-64 bg-layer-1 p-2 rounded">
|
||||||
|
<JsonViewer
|
||||||
|
value={jsonValue}
|
||||||
|
path="demo"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</Section>
|
||||||
|
|
||||||
<Section title="Shortcut">
|
<Section title="Shortcut">
|
||||||
<div class="flex gap-4">
|
<div class="flex gap-4">
|
||||||
<ShortCut ctrl key="S" />
|
<ShortCut ctrl key="S" />
|
||||||
|
|||||||
Generated
-6
@@ -4,12 +4,6 @@ settings:
|
|||||||
autoInstallPeers: true
|
autoInstallPeers: true
|
||||||
excludeLinksFromLockfile: false
|
excludeLinksFromLockfile: false
|
||||||
|
|
||||||
catalogs:
|
|
||||||
default:
|
|
||||||
chokidar-cli:
|
|
||||||
specifier: github:open-cli-tools/chokidar-cli#semver:v4.0.0
|
|
||||||
version: 4.0.0
|
|
||||||
|
|
||||||
importers:
|
importers:
|
||||||
|
|
||||||
.:
|
.:
|
||||||
|
|||||||
Reference in New Issue
Block a user