massive cleanup of Graph and Editor code

This commit is contained in:
Vašek Šraier 2020-10-01 23:26:02 +02:00
parent f0e6e79364
commit ab13f0b726
13 changed files with 218 additions and 253 deletions

View file

@ -1,7 +1,7 @@
<script lang="ts"> <script lang="ts">
import Graph from "./Graph.svelte"; import Graph from "./Graph.svelte";
import { loadTasks } from "./task-loader"; import { loadTasks } from "./tasks";
import type { TasksFile, TaskDescriptor } from "./task-loader"; import type { TasksFile, TaskDescriptor } from "./tasks";
import TasksLoader from "./TasksLoader.svelte"; import TasksLoader from "./TasksLoader.svelte";
import TaskPanel from "./TaskPanel.svelte"; import TaskPanel from "./TaskPanel.svelte";
import Editor from "./Editor.svelte"; import Editor from "./Editor.svelte";

View file

@ -3,10 +3,11 @@
import Graph from "./Graph.svelte"; import Graph from "./Graph.svelte";
import { nonNull } from "./helpers"; import { nonNull } from "./helpers";
import type { TaskDescriptor, TasksFile } from "./task-loader"; import type { TaskDescriptor, TasksFile } from "./tasks";
import { saveTasks, getCategories } from "./task-loader"; import { saveTasks, getCategories } from "./tasks";
import TaskDisplay from "./TaskDisplay.svelte"; import TaskDisplay from "./TaskDisplay.svelte";
import TaskDetailEditor from "./TaskDetailEditor.svelte"; import TaskDetailEditor from "./TaskDetailEditor.svelte";
import { forceSimulation } from "./force-simulation";
export let tasks: TasksFile; export let tasks: TasksFile;
@ -46,9 +47,15 @@
} }
} }
async function saveCurrentStateWithPositions() { function runSimulation() {
tasks.positions = graph.getNodePositions(); forceSimulation(
await saveTasks(tasks); tasks,
(positions) => {
tasks.tasks.forEach((t) => (t.position = positions.get(t.id)));
tasks = tasks;
},
repulsionForce
);
} }
async function saveCurrentState() { async function saveCurrentState() {
@ -254,20 +261,18 @@
<h3>Toolbox</h3> <h3>Toolbox</h3>
<div> <div>
<button on:click={saveCurrentState}>Uložit aktuální stav</button> <button on:click={saveCurrentState}>Uložit aktuální stav</button>
<button on:click={saveCurrentStateWithPositions}>Uložit aktuální stav
včetně pozic nodů</button>
</div> </div>
<div class="gap" /> <div class="gap" />
<div> <div>
<button on:click={addTask}>Nový node</button> <button on:click={addTask}>Nový node</button>
<button <button
disabled={clicked.length == 0} disabled={clicked.length == 0}
on:click={() => removeTask(clicked[clicked.length - 1])}>Odstranit {clicked[clicked.length - 1] ?? "???"}</button> on:click={() => removeTask(clicked[clicked.length - 1])}>Odstranit {clicked[clicked.length - 1] ?? '???'}</button>
</div> </div>
<div class="gap" /> <div class="gap" />
<div> <div>
<button disabled={clicked.length <= 1} on:click={addEdge}>Přidat hranu {clicked[clicked.length - 2] ?? "???"} <button disabled={clicked.length <= 1} on:click={addEdge}>Přidat hranu {clicked[clicked.length - 2] ?? '???'}
-&gt; {clicked[clicked.length - 1] ?? "???"}</button> -&gt; {clicked[clicked.length - 1] ?? '???'}</button>
<div class="checkbox"> <div class="checkbox">
<label> <label>
<input type="checkbox" bind:checked={showHiddenEdges} /> Zobrazit skryté <input type="checkbox" bind:checked={showHiddenEdges} /> Zobrazit skryté
@ -275,10 +280,10 @@
</label> </label>
</div> </div>
</div> </div>
<div class="gap" /> <div class="gap" />
<div> <div>
<button on:click={graph.runSimulation}>Spustit simulaci</button> <button on:click={runSimulation}>Spustit simulaci</button>
<div class="checkbox"> <div class="checkbox">
<label> <label>
<input type="checkbox" bind:checked={nodeDraggingEnabled} /> Povolit <input type="checkbox" bind:checked={nodeDraggingEnabled} /> Povolit
@ -290,7 +295,7 @@
Repulsion force: <input type="number" bind:value={repulsionForce} name="repulsionForceInput" max="1000" min="-10000" /> Repulsion force: <input type="number" bind:value={repulsionForce} name="repulsionForceInput" max="1000" min="-10000" />
</div> </div>
<div class="gap" /> <div class="gap" />
{#if clicked.length > 0 && getTask(clicked[clicked.length - 1]).type == 'label'} {#if clicked.length > 0 && getTask(clicked[clicked.length - 1])?.type == 'label'}
<div> <div>
Úhel rotace: <input bind:value={angle} type="range" max="360" min="0" on:change={setAngleToTheCurrentLabel} /> Úhel rotace: <input bind:value={angle} type="range" max="360" min="0" on:change={setAngleToTheCurrentLabel} />
</div> </div>

View file

@ -3,13 +3,11 @@
import GraphEdge from "./GraphEdge.svelte"; import GraphEdge from "./GraphEdge.svelte";
import { createEventDispatcher, onMount } from "svelte"; import { createEventDispatcher, onMount } from "svelte";
import * as d3 from "d3"; import * as d3 from "d3";
import type { TasksFile, TaskDescriptor } from "./task-loader"; import type { TasksFile, TaskDescriptor } from "./tasks";
import { createNodesAndEdges } from "./graph-types"; import { createEdges } from "./tasks";
import { taskForce } from "./task-force"; import { taskStatuses } from "./task-status-cache";
import { taskStatuses } from './task-status-cache'
export let tasks: TasksFile; export let tasks: TasksFile;
export let repulsionForce: number = -1000;
export let nodeDraggingEnabled: boolean = false; export let nodeDraggingEnabled: boolean = false;
export let showHiddenEdges: boolean = false; export let showHiddenEdges: boolean = false;
@ -21,89 +19,50 @@
let clientWidth: number; let clientWidth: number;
let svgElement: SVGElement; let svgElement: SVGElement;
// this prevents svelte from updating nodes and edges $: nodes = tasks.tasks;
// when we update nodes and edges $: edges = createEdges(nodes);
let [nodes, edges] = createNodesAndEdges(tasks);
function hack() {
[nodes, edges] = createNodesAndEdges(tasks, nodes, edges);
}
$: {
tasks;
hack();
}
const eventDispatcher = createEventDispatcher(); const eventDispatcher = createEventDispatcher();
function nodeClick(task: TaskDescriptor) {
const nodeClick = (task: TaskDescriptor) => (e: CustomEvent<MouseEvent>) => { function eventHandler(e: CustomEvent<MouseEvent>) {
eventDispatcher("selectTask", task); eventDispatcher("selectTask", task);
};
const nodeDoubleClick = (task: TaskDescriptor) => (e: CustomEvent<MouseEvent>) => {
eventDispatcher("openTask", task);
};
const nodeHover = (task: TaskDescriptor) => (
hovering: CustomEvent<boolean>
) => {
if (hovering.detail) {
hoveredTask = task.id;
eventDispatcher("preSelectTask", task);
} else {
if (hoveredTask == task.id) hoveredTask = null;
eventDispatcher("unPreSelectTask", task);
}
};
export function runSimulation() {
// Let's list the force we wanna apply on the network
let simulation = d3
.forceSimulation(nodes) // Force algorithm is applied to data.nodes
.force(
"link",
d3
.forceLink() // This force provides links between nodes
.id(function (d) {
return d.id;
}) // This provide the id of a node
.links(edges) // and this the list of links
)
.force("charge", d3.forceManyBody().strength(repulsionForce)) // This adds repulsion between nodes. Play with the -400 for the repulsion strength
.force("x", d3.forceX()) // attracts elements to the zero X coord
.force("y", d3.forceY().strength(0.5)) // attracts elements to the zero Y coord
.force("dependencies", taskForce())
.on("tick", ticked)
.on("end", ticked);
// This function is run at each iteration of the force algorithm, updating the nodes position.
function ticked() {
edges = edges;
nodes = nodes;
} }
return eventHandler;
} }
function nodeDoubleClick(task: TaskDescriptor) {
export function getNodePositions(): Map<string, [number, number]> { function eventHandler(e: CustomEvent<MouseEvent>) {
let res = new Map(); eventDispatcher("openTask", task);
for (let n of nodes) { }
if (n.x != undefined && n.y != undefined) { return eventHandler;
res.set(n.id, [n.x, n.y]); }
function nodeHover(task: TaskDescriptor) {
function eventHandler(hovering: CustomEvent<boolean>) {
if (hovering.detail) {
hoveredTask = task.id;
eventDispatcher("preSelectTask", task);
} else {
if (hoveredTask == task.id) hoveredTask = null;
eventDispatcher("unPreSelectTask", task);
} }
} }
return res return eventHandler;
} }
/**
// start simulation and center view on create * Make the SVG drag&zoomable
onMount(() => { **/
// set center of the SVG at (0,0) function setupZoom() {
let svg = d3.select(svgElement).select("g")
// setup zoom
function zoomed(e) { function zoomed(e) {
let svg = d3.select(svgElement).select("g");
svg.attr("transform", e.transform); svg.attr("transform", e.transform);
} }
const zoomer = d3.zoom().scaleExtent([0.1, 2]).clickDistance(10) const zoomer = d3.zoom().scaleExtent([0.1, 2]).clickDistance(10);
zoomer.on("zoom", zoomed); zoomer.on("zoom", zoomed);
d3.select(container).call(zoomer); d3.select(container).call(zoomer);
}
onMount(() => {
setupZoom();
}); });
</script> </script>
@ -116,7 +75,7 @@
left: 0px; left: 0px;
} }
:global(#header) { :global(#header) {
z-index: 20 z-index: 20;
} }
:global(#wrapper) { :global(#wrapper) {
display: flex; display: flex;
@ -131,20 +90,22 @@
<div bind:this={container} bind:clientHeight bind:clientWidth> <div bind:this={container} bind:clientHeight bind:clientWidth>
<svg bind:this={svgElement} viewBox="{0},{0},{clientWidth},{clientHeight}"> <svg bind:this={svgElement} viewBox="{0},{0},{clientWidth},{clientHeight}">
<g> <g>
<g transform="translate({clientWidth/2}, {clientHeight/2})"> <g transform="translate({clientWidth / 2}, {clientHeight / 2})">
{#each edges as edge} {#each edges as edge}
<GraphEdge {edge} showLabelEdge={showHiddenEdges}/> <GraphEdge {edge} showLabelEdge={showHiddenEdges} />
{/each} {/each}
{#each nodes as task} {#each nodes as task}
<GraphNode <GraphNode
{task} {task}
on:taskClick on:taskClick
on:click={nodeClick(task.task)} on:click={nodeClick(task)}
on:hoveringChange={nodeHover(task.task)} on:hoveringChange={nodeHover(task)}
on:positionChange={() => { tasks = tasks; }} on:positionChange={() => {
tasks = tasks;
}}
status={$taskStatuses.get(task.id)} status={$taskStatuses.get(task.id)}
draggingEnabled={nodeDraggingEnabled} draggingEnabled={nodeDraggingEnabled}
on:dblclick={nodeDoubleClick(task.task)} /> on:dblclick={nodeDoubleClick(task)} />
{/each} {/each}
</g> </g>
</g> </g>

View file

@ -1,18 +1,15 @@
<script lang="ts"> <script lang="ts">
import type { SimulationLinkDatum } from "d3"; import type { TaskEdge } from "./graph";
import type { TaskId } from "./graph-types";
export let edge: SimulationLinkDatum<TaskId>; export let edge: TaskEdge;
export let showLabelEdge: boolean = false; export let showLabelEdge: boolean = false;
$: x1 = edge?.source?.x ?? 0; $: [x1, y1] = edge?.dependency?.position ?? [0,0];
$: y1 = edge?.source?.y ?? 0; $: [x2, y2] = edge?.dependee?.position ?? [0, 0];
$: x2 = edge?.target?.x ?? 0;
$: y2 = edge?.target?.y ?? 0;
$: dx = x1 - x2 $: dx = x1 - x2
$: dy = y1 - y2 $: dy = y1 - y2
</script> </script>
{#if showLabelEdge || (edge?.target?.task?.type ?? null) != "label"} {#if showLabelEdge || (edge?.dependee?.type ?? null) != "label"}
<path d="m {x2} {y2+0} c 0 0 {dx} {dy-40} {dx} {dy-20}" style="fill:none; stroke: #aaa; stroke-width: 3px" /> <path d="m {x2} {y2+0} c 0 0 {dx} {dy-40} {dx} {dy-20}" style="fill:none; stroke: #aaa; stroke-width: 3px" />
{/if} {/if}

View file

@ -2,10 +2,10 @@
import * as d3 from "d3"; import * as d3 from "d3";
import { createEventDispatcher, onMount } from "svelte"; import { createEventDispatcher, onMount } from "svelte";
import type { TaskId } from "./graph-types";
import type { TaskStatus } from "./ksp-task-grabber"; import type { TaskStatus } from "./ksp-task-grabber";
import type { TaskDescriptor } from "./tasks";
export let task: TaskId; export let task: TaskDescriptor;
export let draggingEnabled: boolean = false; export let draggingEnabled: boolean = false;
export let status: TaskStatus | undefined = undefined; export let status: TaskStatus | undefined = undefined;
@ -13,8 +13,7 @@
let text_element: SVGTextElement; let text_element: SVGTextElement;
let mainGroup: SVGGElement; let mainGroup: SVGGElement;
$: cx = task === undefined || task.x === undefined ? 0 : task.x; $: [cx, cy] = task.position ?? [0, 0];
$: cy = task === undefined || task.y === undefined ? 0 : task.y;
const eventDispatcher = createEventDispatcher(); const eventDispatcher = createEventDispatcher();
function enter() { function enter() {
@ -43,7 +42,7 @@
}); });
// every time after that // every time after that
$: { $: {
task.task.title; task.title;
if (text_element) if (text_element)
ensureTextFits(); ensureTextFits();
} }
@ -64,9 +63,7 @@
if (!draggingEnabled) return; if (!draggingEnabled) return;
if (!dragging) return; if (!dragging) return;
let [x, y] = d3.pointer(e, mainGroup); task.position = d3.pointer(e, mainGroup);
task.x = x;
task.y = y;
eventDispatcher("positionChange"); eventDispatcher("positionChange");
e.preventDefault(); e.preventDefault();
e.stopPropagation(); e.stopPropagation();
@ -124,8 +121,8 @@
on:click={click} on:click={click}
on:mousedown={dragStart} on:mousedown={dragStart}
on:dblclick={dblclick} on:dblclick={dblclick}
class="{status == null ? '' : status.solved ? 'solved' : status.submitted ? 'submitted' : ''} {task.task.type}"> class="{status == null ? '' : status.solved ? 'solved' : status.submitted ? 'submitted' : ''} {task.type}">
{#if task.task.type == 'label'} {#if task.type == 'label'}
{#if draggingEnabled } {#if draggingEnabled }
<ellipse rx={ellipse_rx} ry={20} {cx} {cy} /> <ellipse rx={ellipse_rx} ry={20} {cx} {cy} />
{/if} {/if}
@ -135,8 +132,8 @@
y={cy + 5} y={cy + 5}
text-anchor="middle" text-anchor="middle"
alignment-baseline="middle" alignment-baseline="middle"
transform="translate({cx}, {cy}) rotate({task.task.rotationAngle ?? 0}) translate({-cx}, {-cy})"> transform="translate({cx}, {cy}) rotate({task.rotationAngle ?? 0}) translate({-cx}, {-cy})">
{task.task.title == null ? task.id : task.task.title} {task.title == null ? task.id : task.title}
</text> </text>
{:else} {:else}
<ellipse class="taskNode" rx={ellipse_rx} ry={20} {cx} {cy} /> <ellipse class="taskNode" rx={ellipse_rx} ry={20} {cx} {cy} />
@ -146,7 +143,7 @@
y={cy + 5} y={cy + 5}
text-anchor="middle" text-anchor="middle"
alignment-baseline="middle"> alignment-baseline="middle">
{task.task.title == null ? task.id : task.task.title} {task.title == null ? task.id : task.title}
</text> </text>
{/if} {/if}
</g> </g>

View file

@ -1,10 +1,9 @@
<script lang="ts"> <script lang="ts">
import { getCategories } from "./task-loader"; import { getCategories } from "./tasks";
import type { TaskDescriptor, TasksFile } from "./task-loader"; import type { TaskDescriptor, TasksFile } from "./tasks";
import { getContext } from "svelte"; import { getContext } from "svelte";
import { onMount } from "svelte"; import { onMount } from "svelte";
import ClassicEditor from "@ckeditor/ckeditor5-build-classic"; import ClassicEditor from "@ckeditor/ckeditor5-build-classic";
import App from "./App.svelte";
const { close } = getContext("simple-modal"); const { close } = getContext("simple-modal");

View file

@ -4,7 +4,7 @@
import { nonNull } from './helpers' import { nonNull } from './helpers'
import App from "./App.svelte"; import App from "./App.svelte";
import { taskStatuses } from "./task-status-cache"; import { taskStatuses } from "./task-status-cache";
import type { TaskDescriptor } from "./task-loader"; import type { TaskDescriptor } from "./tasks";
export let task: TaskDescriptor | null | undefined export let task: TaskDescriptor | null | undefined

View file

@ -1,7 +1,5 @@
<script lang="ts"> <script lang="ts">
import { grabAssignment } from "./ksp-task-grabber"; import type { TasksFile, TaskDescriptor } from "./tasks";
import type { TaskAssignmentData } from "./ksp-task-grabber";
import type { TasksFile, TaskDescriptor } from "./task-loader";
import TaskDisplay from "./TaskDisplay.svelte"; import TaskDisplay from "./TaskDisplay.svelte";
export let tasks: TasksFile; export let tasks: TasksFile;

View file

@ -1,5 +1,5 @@
<script lang="ts"> <script lang="ts">
import type { TasksFile } from "./task-loader"; import type { TasksFile } from "./tasks";
import { refresh } from './task-status-cache' import { refresh } from './task-status-cache'
export let promise: Promise<TasksFile>; export let promise: Promise<TasksFile>;

View file

@ -0,0 +1,106 @@
import type { SimulationLinkDatum, SimulationNodeDatum } from "d3";
import * as d3 from "d3";
import { createEdges, TaskDescriptor, TasksFile } from "./tasks";
type TaskId = {
id: string;
task: TaskDescriptor;
} & SimulationNodeDatum;
function toMapById(nodes: TaskId[]): Map<string, TaskId> {
let nodeMap = new Map<string, TaskId>();
for (let task of nodes) {
if (task.id in nodeMap)
throw 'duplicate IDs';
nodeMap.set(task.id, task);
}
return nodeMap;
}
function taskForce(): d3.Force<TaskId, undefined> {
let myNodes: TaskId[] | null = null;
let deps: Map<string, number> = new Map();
let idMap: Map<string, TaskId> = new Map();
function getNumberOfDeps(task: TaskId): number {
if (deps.has(task.id)) return deps.get(task.id)!;
if (task.task.requires.length == 0) return 0;
let res = 0;
for (let r of task.task.requires) {
res += getNumberOfDeps(idMap.get(r)!) + 1;
}
deps.set(task.id, res);
return res;
}
let force: d3.Force<TaskId, undefined> = function (alpha: number) {
if (myNodes == null) throw 'nodes not initialized';
for (let task of myNodes) {
if (task.vy == null) {
task.vy = 0
}
task.vy += getNumberOfDeps(task) * 25 * alpha;
}
}
force.initialize = function (nodes: TaskId[]) {
myNodes = nodes;
idMap = toMapById(myNodes);
}
return force;
}
/**
*
* @param nodes
* @param edges
* @param ticked function that gets run every tick of a running simulation
*/
export function forceSimulation(tasks: TasksFile, ticked: (positions: Map<string, [number, number]>) => void, repulsionForce?: number) {
repulsionForce = repulsionForce ?? -1000;
let nodes: TaskId[] = tasks.tasks.map(
(t) => {
return {
id: t.id,
task: t,
x: (t.position ?? [0, 0])[0],
y: (t.position ?? [0, 0])[1]
}
})
let edges: SimulationLinkDatum<TaskId>[] = createEdges(tasks.tasks).map((e) => {
return {
// FIXME are we sure its not the other way round?
source: nodes.find((n) => n.id == e.dependee.id)!,
target: nodes.find((n) => n.id == e.dependency.id)!
}
});
function tickHandler() {
ticked(new Map(nodes.map((n) => [n.id!, [n.x!, n.y!]])))
}
// Let's list the force we wanna apply on the network
let simulation = d3
.forceSimulation(nodes) // Force algorithm is applied to data.nodes
.force(
"link",
d3
.forceLink() // This force provides links between nodes
.id(function (d) {
return d.id;
}) // This provide the id of a node
.links(edges) // and this the list of links
)
.force("charge", d3.forceManyBody().strength(repulsionForce)) // This adds repulsion between nodes. Play with the -400 for the repulsion strength
.force("x", d3.forceX()) // attracts elements to the zero X coord
.force("y", d3.forceY().strength(0.5)) // attracts elements to the zero Y coord
.force("dependencies", taskForce())
.on("tick", tickHandler)
.on("end", tickHandler);
}

View file

@ -1,64 +0,0 @@
import type { SimulationLinkDatum, SimulationNodeDatum } from "d3";
import type { TaskDescriptor, TasksFile } from "./task-loader";
import { createTaskMap } from "./task-loader";
export type TaskId = {
id: string;
task: TaskDescriptor;
} & SimulationNodeDatum;
function toMapById(nodes: TaskId[]): Map<string, TaskId> {
let nodeMap = new Map<string, TaskId>();
for (let task of nodes) {
if (task.id in nodeMap)
throw 'duplicate IDs';
nodeMap.set(task.id, task);
}
return nodeMap;
}
function createNodes(tasks: TasksFile, old?: TaskId[]): TaskId[] {
let m = (old == undefined) ? new Map<string, TaskId>() : toMapById(old);
let res: TaskId[] = tasks.tasks.map((t) => {
return { id: t.id, task: t };
});
for (let t of res) {
if (tasks.positions.has(t.id)) {
[t.x, t.y] = tasks.positions.get(t.id)!
}
if (m.has(t.id)) {
Object.assign(t, m.get(t.id))
}
}
return res;
}
export function createNodesAndEdges(tasks: TasksFile, oldNodes?: TaskId[], oldEdges?: SimulationLinkDatum<TaskId>[]): [TaskId[], SimulationLinkDatum<TaskId>[]] {
let nodes = createNodes(tasks, oldNodes);
// create mapping from ID to node
let nodeMap = toMapById(nodes);
let links: SimulationLinkDatum<TaskId>[] = [];
for (const task of tasks.tasks) {
const src = nodeMap.get(task.id)!;
for (const id of task.requires) {
const t = nodeMap.get(id);
if (t === undefined) throw `missing task with id ${id}`;
const l: SimulationLinkDatum<TaskId> =
{
source: src,
target: t
};
links.push(l);
}
}
return [nodes, links];
}

View file

@ -1,50 +0,0 @@
import type { TaskId } from "./graph-types";
/* copied from graph-types.ts */
function toMapById(nodes: TaskId[]): Map<string, TaskId> {
let nodeMap = new Map<string, TaskId>();
for (let task of nodes) {
if (task.id in nodeMap)
throw 'duplicate IDs';
nodeMap.set(task.id, task);
}
return nodeMap;
}
export function taskForce(): d3.Force<TaskId, undefined> {
let myNodes: TaskId[] | null = null;
let deps: Map<string, number> = new Map();
let idMap: Map<string, TaskId> = new Map();
function getNumberOfDeps(task: TaskId): number {
if (deps.has(task.id)) return deps.get(task.id)!;
if (task.task.requires.length == 0) return 0;
let res = 0;
for (let r of task.task.requires) {
res += getNumberOfDeps(idMap.get(r)!) + 1;
}
deps.set(task.id, res);
return res;
}
let force: d3.Force<TaskId, undefined> = function(alpha: number) {
if (myNodes == null) throw 'nodes not initialized';
for (let task of myNodes) {
if (task.vy == null) {
task.vy = 0
}
task.vy += getNumberOfDeps(task) * 25 * alpha;
}
}
force.initialize = function(nodes: TaskId[]) {
myNodes = nodes;
idMap = toMapById(myNodes);
}
return force;
}

View file

@ -5,6 +5,7 @@ export type TaskDescriptor = {
title?: string title?: string
requires: string[] requires: string[]
comment?: string comment?: string
position?: [number, number]
} & ( } & (
{ {
type: "open-data", type: "open-data",
@ -25,18 +26,38 @@ export type TaskDescriptor = {
export type TasksFile = { export type TasksFile = {
tasks: TaskDescriptor[] tasks: TaskDescriptor[]
clusters: { [name: string]: string[] } clusters: { [name: string]: string[] }
positions: Map<string, [number, number]>
} }
export type TaskMap = Map<string, TaskDescriptor>; export type TaskMap = Map<string, TaskDescriptor>;
export type TaskEdge = {
dependee: TaskDescriptor
dependency: TaskDescriptor
}
export function createEdges(nodes: TaskDescriptor[]): TaskEdge[] {
let edges: TaskEdge[] = [];
for (const n of nodes) {
for (const r of n.requires) {
const a = nodes.find((t) => t.id == r);
if (a == undefined) throw `broken dependency, missing task with ${r}`
edges.push({dependee: a, dependency: n})
}
}
return edges
}
export async function loadTasks(): Promise<TasksFile> { export async function loadTasks(): Promise<TasksFile> {
const r = await fetch("/tasks.json") const r = await fetch("/tasks.json")
const j = await r.json() const j = await r.json()
if (j.positions == null)
j.positions = new Map(); // backward compatibility
else if (j.positions != null) {
j.positions = new Map(Object.entries(j.positions)) for (const [id, pos] of Object.entries(j.positions)) {
j.tasks.find((t) => t.id == id).position = pos;
}
}
return j return j
} }
@ -47,15 +68,10 @@ function normalizeTasks(tasks: TasksFile) {
export async function saveTasks(tasks: TasksFile) { export async function saveTasks(tasks: TasksFile) {
normalizeTasks(tasks); normalizeTasks(tasks);
let p: any = {}
for (let [key, val] of tasks.positions.entries())
p[key] = val;
const data = { ...tasks, positions: p }
// request options // request options
const options = { const options = {
method: 'POST', method: 'POST',
body: JSON.stringify(data, null, 4), body: JSON.stringify(tasks, null, 4),
headers: { headers: {
'Content-Type': 'application/json' 'Content-Type': 'application/json'
} }