import { HookContainer } from "./hook_container.js";
import { Vector3, Box3, dirs, dirKeys, dirAngles, normalizedDirs } from "./geometry.js";
import { asyncFrom, mod, merge } from "./utils.js";
import { DeleteBrush, AddBrush, SelectBrush, DistancePegBrush, AreaBrush } from "./brushes/index.js";
import { PanEvent } from "./drag_events/index.js";
import { Selection } from "./selection.js";
import { ChangeNameAction, MergeAction } from "./actions/index.js";
import { Brushbar } from "./brushbar.js";
import { ExportUI } from "./export_ui.js";
import { MegaTile, megaTileSize } from "./mega_tile.js";
import { NodeRender, tileSize } from "./node_render.js";
import { style } from "./style.js";
import { version } from "./version.js";
/** A render context of a mapper into a specific element.
* Handles keeping the UI connected to an element on a page.
* See Mapper.render() for instantiation.
* Call disconnect() on a render context once the element is no longer being used for a specific Mapper to close event listeners.
*/
class RenderContext {
/** Construct the render context for the specified mapper in a specific parent element.
* Will set up event listeners and build the initial UI.
*/
constructor(parent, mapper, options) {
this.parent = parent;
this.mapper = mapper;
this.options = merge({
mode: "normal",
}, options);
this.alive = true;
this.hooks = new HookContainer();
this.keyboardShortcuts = [];
this.wantRedraw = true;
this.recalculateViewport = true;
this.recalculateUpdate = [];
this.recalculateRemoved = [];
this.recalculateTranslated = [];
this.infoMessages = [];
this.infoMessageTimeout = 5000;
this.wantRecheckSelection = true;
this.wantUpdateSelection = true;
this.undoStack = [];
this.redoStack = [];
this.nodeRenders = {};
this.megaTiles = {};
this.nodeIdsToMegatiles = {};
this.drawnNodeIds = {};
this.labelPositions = {};
this.backgroundColor = "#997";
this.pressedKeys = {};
this.mouseDragEvents = {};
this.oldMousePosition = Vector3.ZERO;
this.mousePosition = Vector3.ZERO;
this.debugMode = false;
this.scrollDelta = 0;
this.scrollOffset = Vector3.ZERO;
this.defaultZoom = 5;
this.zoom = this.defaultZoom;
this.requestedZoom = this.zoom;
this.lastZoomRequest = 0;
this.zoomRequestTimeout = 1000;
this.drawSelectionCanvas = true;
this.selectionCanvasToggleTime = 300;
this.altitudeIncrement = this.mapper.metersToUnits(5);
this.distanceMarkers = {};
this.brushes = {
add: new AddBrush(this, false),
extend: new AddBrush(this, true),
select: new SelectBrush(this),
"delete": new DeleteBrush(this),
"area": new AreaBrush(this),
"peg1": new DistancePegBrush(this, 1),
"peg2": new DistancePegBrush(this, 2),
};
this.brush = this.brushes.add;
this.defaultLayer = this.mapper.backend.layerRegistry.getDefault();
this.currentLayer = this.defaultLayer;
this.hoverSelection = new Selection(this, []);
this.selection = new Selection(this, []);
this.backgroundNodeCache = {};
this.styleElement = style();
document.head.appendChild(this.styleElement);
// The UI is just a canvas.
// We will keep its size filling the parent element.
this.canvas = document.createElement("canvas");
this.selectionCanvas = document.createElement("canvas");
this.selectionCanvasScroll = Vector3.ZERO;
this.canvas.tabIndex = 1;
this.parent.appendChild(this.canvas);
// The canvas has no extra size.
this.canvas.style.padding = "0";
this.canvas.style.margin = "0";
this.canvas.style.border = "0";
this.mapper.hooks.add("updateNode", (nodeRef) => this.recalculateNodeUpdate(nodeRef));
this.mapper.hooks.add("removeNodes", (nodeRefs) => this.recalculateNodesRemove(nodeRefs));
this.mapper.hooks.add("translateNodes", (nodeRefs) => this.recalculateNodesTranslate(nodeRefs));
this.mapper.hooks.add("update", this.requestUpdateSelection.bind(this));
this.wasActivity = false;
this.lastActivity = performance.now();
this.lastActivityTimeout = 300;
this.brushbar = new Brushbar(this);
this.canvas.addEventListener("contextmenu", event => {
event.preventDefault();
});
this.canvas.addEventListener("mousedown", async (event) => {
if(this.requestedZoom !== this.zoom) {
// Forcibly apply the last zoom request.
this.lastZoomRequest = 0;
return;
}
if(this.mouseDragEvents[event.button] === undefined) {
const where = this.mouseEventToCanvasPoint(event);
if(event.button === 0) {
const dragEvent = await this.brush.activate(where);
if(dragEvent) {
this.mouseDragEvents[event.button] = dragEvent;
}
}
else if(event.button === 2) {
if(this.mouseDragEvents[0] !== undefined) {
this.cancelMouseButtonPress(0);
}
else {
this.mouseDragEvents[event.button] = new PanEvent(this, where);
}
}
}
});
this.canvas.addEventListener("mouseup", async (event) => {
const where = this.mouseEventToCanvasPoint(event);
this.endMouseButtonPress(event.button, where);
this.requestRedraw();
});
this.canvas.addEventListener("mousemove", async (event) => {
this.oldMousePosition = this.mousePosition;
this.mousePosition = this.mouseEventToCanvasPoint(event);
for(const button in this.mouseDragEvents) {
const mouseDragEvent = this.mouseDragEvents[button];
await mouseDragEvent.next(this.mousePosition);
}
this.requestRecheckSelection();
this.requestRedraw();
});
this.canvas.addEventListener("mouseout", (event) => {
event;
this.cancelMouseButtonPresses();
});
this.canvas.addEventListener("keydown", async (event) => {
this.pressedKeys[event.key] = true;
for(const shortcut of this.keyboardShortcuts) {
if(shortcut.filter(this, event)) {
if(await shortcut.handler() !== true) {
// Shortcuts can do anything, so let's forget about keeping track of which keys are down.
// It's possible that a dialog might absorb the keyup event, leaving modifiers stuck in our pressed keys ledger.
this.clearKeysDown();
return;
}
}
}
const handlePanKeys = async () => {
if(event.key === "ArrowUp") {
this.setScrollOffset(this.scrollOffset.subtract(new Vector3(0, this.screenSize().y / 3, 0)).round());
return true;
}
else if(event.key === "ArrowDown") {
this.setScrollOffset(this.scrollOffset.add(new Vector3(0, this.screenSize().y / 3, 0)).round());
return true;
}
else if(event.key === "ArrowLeft") {
this.setScrollOffset(this.scrollOffset.subtract(new Vector3(this.screenSize().x / 3, 0, 0)).round());
return true;
}
else if(event.key === "ArrowRight") {
this.setScrollOffset(this.scrollOffset.add(new Vector3(this.screenSize().x / 3, 0, 0)).round());
return true;
}
};
const handleOrientationKeys = async () => {
if(this.isKeyDown("Control")) {
if(event.key === "c") {
await this.resetOrientation();
return true;
}
else if(event.key === "=" || event.key === "+") {
this.requestZoomChangeDelta(-1);
event.preventDefault();
return true;
}
else if(event.key === "-") {
this.requestZoomChangeDelta(1);
event.preventDefault();
return true;
}
}
};
if(this.inPreviewMode()) {
if(await handlePanKeys());
else if(await handleOrientationKeys());
}
else if(this.inNormalMode()) {
if(await handlePanKeys());
else if(await handleOrientationKeys());
else if(this.isKeyDown("Control") && event.key === "z") {
await this.undo();
}
else if(this.isKeyDown("Control") && event.key === "y") {
await this.redo();
}
else if(this.isKeyDown("Control") && event.key === "e") {
this.openExportModal();
}
else if(event.key === "1") {
this.changeBrush(this.brushes.peg1);
}
else if(event.key === "2") {
this.changeBrush(this.brushes.peg2);
}
else if(event.key === "d") {
this.changeBrush(this.brushes["delete"]);
}
else if(event.key === "a") {
this.changeBrush(this.brushes.add);
}
else if(event.key === "e") {
this.changeBrush(this.brushes.extend);
}
else if(event.key === "s") {
this.changeBrush(this.brushes.select);
}
else if(event.key === "c") {
this.changeBrush(this.brushes.area);
}
else if(event.key === "C") {
if(this.brush === this.brushes.area) {
this.brushes.area.reset();
}
}
else if(event.key === "m") {
await this.performAction(new MergeAction(this, {nodeRefs: Array.from(this.selection.getOrigins())}), true);
}
else if(event.key === "l") {
const layerArray = Array.from(this.mapper.backend.layerRegistry.getLayers());
const layerIdArray = layerArray.map(layer => layer.id);
this.setCurrentLayer(layerArray[(layerIdArray.indexOf(this.getCurrentLayer().id) + 1) % layerArray.length]);
}
else if(event.key === "`") {
this.debugMode = !this.debugMode;
}
else if(event.key === "n") {
const nodeRef = await this.hoverSelection.getParent();
if(nodeRef) {
const where = (await this.getNamePosition(nodeRef)).center.subtract(this.scrollOffset);
const input = document.createElement("input");
input.value = (await nodeRef.getPString("name")) || "";
input.style.position = "absolute";
input.style.left = `${where.x}px`;
input.style.top = `${where.y}px`;
input.style.fontSize = "16px";
const cancel = () => {
input.removeEventListener("blur", cancel);
input.remove();
this.focus();
};
const submit = async () => {
this.performAction(new ChangeNameAction(this, {nodeRef: nodeRef, name: input.value}), true);
cancel();
};
input.addEventListener("blur", cancel);
input.addEventListener("keyup", (event) => {
if(event.key === "Escape") {
cancel();
}
else if(event.key === "Enter") {
submit();
}
event.preventDefault();
});
this.parent.appendChild(input);
input.focus();
event.preventDefault();
}
}
}
this.requestRedraw();
});
this.canvas.addEventListener("keyup", (event) => {
this.pressedKeys[event.key] = false;
this.requestRedraw();
});
this.canvas.addEventListener("wheel", (event) => {
event.preventDefault();
this.scrollDelta = this.scrollDelta + event.deltaY;
const delta = this.scrollDelta;
if(Math.abs(delta) >= 100) {
if(this.isKeyDown("q")) {
if(delta < 0) {
this.brush.increment();
}
else {
this.brush.decrement();
}
}
else if(this.isKeyDown("w")) {
if(delta < 0) {
this.brush.enlarge();
}
else {
this.brush.shrink();
}
}
else {
this.requestZoomChangeDelta((delta < 0 ? -1 : 1));
}
this.scrollDelta = 0;
}
this.requestRedraw();
});
// Watch the parent resize so we can keep our canvas filling the whole thing.
this.parentObserver = new ResizeObserver(() => this.recalculateSize());
this.parentObserver.observe(this.parent);
this.hooks.add("", async (hookName, ...args) => {
await this.brush.hooks.call("context_" + hookName, ...args);
});
this.mapper.hooks.add("", async (hookName, ...args) => {
await this.brush.hooks.call("mapper_" + hookName, ...args);
});
this.recalculateSize();
window.requestAnimationFrame(this.redrawLoop.bind(this));
setTimeout(this.recalculateLoop.bind(this), 10);
if(this.inNormalMode()) {
setTimeout(this.recalculateSelection.bind(this), 10);
setTimeout(this.toggleSelectionCanvas.bind(this), this.selectionCanvasToggleTime);
}
this.changeBrush(this.brushes.add);
this.setCurrentLayer(this.getCurrentLayer());
}
mouseEventToCanvasPoint(event) {
return new Vector3(event.offsetX, event.offsetY, 0);
}
canvasOffset() {
return new Vector3(this.canvas.offsetLeft, this.canvas.offsetTop, 0);
}
inNormalMode() {
return this.options.mode === "normal";
}
inExportMode() {
return this.options.mode === "export";
}
inPreviewMode() {
return this.options.mode === "preview";
}
inControlledMode() {
return this.options.mode === "normal" || this.options.mode === "preview";
}
async openExportModal() {
this.canvas.style.display = "none";
this.brushbar.element.style.display = "none";
const exportUI = new ExportUI(this);
await exportUI.show();
this.brushbar.element.style.display = "";
this.canvas.style.display = "";
this.clearKeysDown();
this.focus();
}
toggleSelectionCanvas() {
this.drawSelectionCanvas = !this.drawSelectionCanvas;
this.requestRedraw();
if(this.alive) {
setTimeout(this.toggleSelectionCanvas.bind(this), this.selectionCanvasToggleTime);
}
}
pushInfoMessage(message) {
this.infoMessages.push({
message: message,
when: performance.now(),
});
while(this.infoMessages.length > 10) {
this.infoMessages.pop(0);
}
this.requestRedraw();
}
async undo() {
const undo = this.undoStack.pop();
if(undo !== undefined) {
this.redoStack.push(await this.performAction(undo, false));
this.hooks.call("undid");
}
}
async redo() {
const redo = this.redoStack.pop();
if(redo !== undefined) {
this.pushUndo(await this.performAction(redo, false), true);
this.hooks.call("redid");
}
}
msSinceLastZoomRequest() {
return performance.now() - this.lastZoomRequest;
}
async getNamePosition(nodeRef) {
if(!(await nodeRef.getType()).getScale() === "explicit") {
const labelPositions = this.labelPositions[this.zoom];
if(labelPositions !== undefined) {
const labelPositionOnCanvas = labelPositions[nodeRef.id];
if(labelPositionOnCanvas !== undefined) {
return labelPositionOnCanvas;
}
}
}
return {
center: this.mapPointToAbsoluteCanvas(await nodeRef.getCenter()),
size: 24,
};
}
async resetOrientation() {
await this.forceZoom(this.defaultZoom);
this.setScrollOffset(Vector3.ZERO);
}
getCurrentLayer() {
return this.currentLayer;
}
setCurrentLayer(layer) {
this.currentLayer = layer;
this.hooks.call("current_layer_change", layer);
this.requestRecheckSelection();
}
isPanning() {
return this.mouseDragEvents[2] instanceof PanEvent;
}
isCalculatingDistance() {
return this.brush instanceof DistancePegBrush;
}
setScrollOffset(value) {
this.scrollOffset = value;
this.lastActivity = performance.now();
this.recalculateEntireViewport();
}
registerKeyboardShortcut(filter, handler) {
this.keyboardShortcuts.push({
filter: filter,
handler: handler,
});
}
changeBrush(brush) {
this.brush = brush;
this.brush.switchTo();
this.hooks.call("changed_brush", brush);
this.requestRecheckSelection();
this.redrawSelection();
this.requestRedraw();
}
requestZoomChange(zoom) {
if(this.requestedZoom !== zoom) {
this.requestedZoom = Math.max(1, zoom);
this.lastZoomRequest = performance.now();
this.requestRedraw();
this.hooks.call("requested_zoom", zoom);
}
}
requestZoomChangeDelta(zoomDelta) {
this.requestZoomChange(this.requestedZoom + zoomDelta);
}
forceZoom(zoom) {
return new Promise((resolve) => {
if(this.zoom === zoom) {
resolve(this.zoom);
}
else {
let f;
f = () => {
resolve(this.zoom);
this.hooks.remove("changed_zoom", f);
};
this.hooks.add("changed_zoom", f);
this.requestZoomChange(zoom);
this.lastZoomRequest = 0;
}
});
}
async redrawLoop() {
if(this.wantRedraw) {
this.wantRedraw = false;
await this.redraw();
}
if(this.alive) {
window.requestAnimationFrame(this.redrawLoop.bind(this));
}
}
async redrawSelection() {
const sc = this.selectionCanvas.getContext("2d");
sc.clearRect(0, 0, this.selectionCanvas.width, this.selectionCanvas.height);
const hoverPatternImage = document.createElement("canvas");
hoverPatternImage.width = hoverPatternImage.height = tileSize;
const hoverPatternContext = hoverPatternImage.getContext("2d");
hoverPatternContext.strokeStyle = "black";
hoverPatternContext.moveTo(0, 0);
hoverPatternContext.lineTo(tileSize, tileSize);
hoverPatternContext.stroke();
const hoverPattern = sc.createPattern(hoverPatternImage, "repeat");
const selectPatternImage = document.createElement("canvas");
selectPatternImage.width = selectPatternImage.height = tileSize;
const selectPatternContext = selectPatternImage.getContext("2d");
selectPatternContext.strokeStyle = "black";
selectPatternContext.moveTo(tileSize, 0);
selectPatternContext.lineTo(0, tileSize);
selectPatternContext.stroke();
const selectPattern = sc.createPattern(selectPatternImage, "repeat");
const megaTiles = this.megaTiles[this.zoom];
if(megaTiles !== undefined) {
const screenBoxInMegaTiles = this.absoluteScreenBox().map(v => v.divideScalar(megaTileSize).map(Math.floor));
for(let x = screenBoxInMegaTiles.a.x; x <= screenBoxInMegaTiles.b.x; x++) {
const megaTileX = megaTiles[x];
if(megaTileX !== undefined) {
for(let y = screenBoxInMegaTiles.a.y; y <= screenBoxInMegaTiles.b.y; y++) {
const megaTile = megaTileX[y];
if(megaTile !== undefined) {
for(const part of megaTile.parts) {
const nodeRef = part.nodeRef;
const point = part.absolutePoint.subtract(this.scrollOffset);
if(this.selection.hasNodeRef(nodeRef) && this.brush.usesSelection()) {
sc.fillStyle = selectPattern;
sc.beginPath();
sc.arc(point.x, point.y, part.radius, 0, 2 * Math.PI, false);
sc.fill();
}
if(this.hoverSelection.hasNodeRef(nodeRef) && this.brush.usesHover()) {
sc.fillStyle = hoverPattern;
sc.beginPath();
sc.arc(point.x, point.y, part.radius, 0, 2 * Math.PI, false);
sc.fill();
}
}
}
}
}
}
}
this.selectionCanvasScroll = this.scrollOffset;
}
async updateSelection(newSelection) {
this.selection = newSelection;
this.hooks.call("selection_change", newSelection);
}
activity() {
return this.isAnyMouseButtonDown() || (performance.now() - this.lastActivity < this.lastActivityTimeout);
}
async recalculateSelection() {
const oldHoverSelection = this.hoverSelection;
const oldSelection = this.selection;
if(this.wantRecheckSelection && !this.activity()) {
this.wantRecheckSelection = false;
const closestNodePart = await this.getDrawnNodePartAtCanvasPoint(this.mousePosition, this.getCurrentLayer());
if(closestNodePart) {
this.hoverSelection = await Selection.fromNodeRefs(this, [closestNodePart.nodeRef]);
}
else {
this.hoverSelection = new Selection(this, []);
}
}
if(this.wantUpdateSelection) {
this.wantUpdateSelection = false;
this.hoverSelection = await this.hoverSelection.updated();
this.selection = await this.selection.updated();
this.requestRedraw();
}
if(!oldHoverSelection.equals(this.hoverSelection) || !oldSelection.equals(this.selection)) {
await this.redrawSelection();
}
if(this.alive) {
setTimeout(this.recalculateSelection.bind(this), 100);
}
}
/** Get the altitude of the map object pointed to by the cursor at the point pointed to.
* @returns {number} The Z coordinate of that point on the map.
*/
async getCursorAltitude() {
// Just return the first Z coordinate of whatever we're hovering over.
for (const origin of this.hoverSelection.getOrigins()) {
return (await origin.getCenter()).z;
}
// Hovering over nothing, use default value.
return 0;
}
/** Get the node drawn at a specific canvas point in the specified layer.
* @param point {Vector3}
* @param layer {Layer}
* @returns {part|null}
*/
async getDrawnNodePartAtCanvasPoint(point, layer) {
const absolutePoint = point.add(this.scrollOffset);
const absoluteMegaTile = absolutePoint.divideScalar(megaTileSize).map(Math.floor);
const megaTiles = this.megaTiles[this.zoom];
if(megaTiles !== undefined) {
const megaTileX = megaTiles[absoluteMegaTile.x];
if(megaTileX !== undefined) {
const megaTile = megaTileX[absoluteMegaTile.y];
if(megaTile !== undefined) {
return megaTile.getDrawnNodePartAtPoint(absolutePoint, layer);
}
}
}
return null;
}
async getDrawnNodePartAtAbsoluteCanvasPointTileAligned(absolutePoint, layer) {
const absoluteMegaTile = absolutePoint.divideScalar(megaTileSize).map(Math.floor);
const megaTiles = this.megaTiles[this.zoom];
if(megaTiles !== undefined) {
const megaTileX = megaTiles[absoluteMegaTile.x];
if(megaTileX !== undefined) {
const megaTile = megaTileX[absoluteMegaTile.y];
if(megaTile !== undefined) {
return megaTile.getDrawnNodePartAtPointTileAligned(absolutePoint, layer);
}
}
}
return null;
}
async getBackgroundNode(nodeRef) {
let backgroundNode = this.backgroundNodeCache[nodeRef.id];
if(backgroundNode === undefined) {
const parent = await nodeRef.getParent();
if(parent && this.backgroundNodeCache[parent.id] === undefined) {
await this.buildBackgroundNodeCache(parent);
}
else {
await this.buildBackgroundNodeCache(nodeRef);
}
backgroundNode = this.backgroundNodeCache[nodeRef.id];
}
return backgroundNode;
}
async buildBackgroundNodeCache(nodeRef) {
const nodeType = await nodeRef.getType();
const nodePosition = await nodeRef.getCenter();
// We'll search for potential background nodes in a box around the node.
const box = Box3.fromRadius(nodePosition, await nodeRef.getRadius());
box.a.z = -Infinity;
box.b.z = Infinity;
const candidates = [];
const layer = await nodeRef.getLayer();
for await(const candidateNodeRef of this.mapper.getNodesTouchingArea(box, 0)) {
if(await candidateNodeRef.getNodeType() !== "point") {
continue;
}
const candidateType = await candidateNodeRef.getType();
// Only nodes of a different type than the original node can provide a background, and they must have a background to provide.
if(candidateType.id !== nodeType.id && candidateType.hasBackground()) {
const candidateLayer = await candidateNodeRef.getLayer();
if(candidateLayer.id === layer.id) {
const center = (await candidateNodeRef.getEffectiveCenter());
candidates.push({
nodeRef: candidateNodeRef,
point: center.noZ(),
z: center.z,
radius: await candidateNodeRef.getRadius(),
givesBackground: candidateType.givesBackground(),
});
}
}
}
// Loop through the original NodeRef as well as all it's children.
for(const iterable of [[nodeRef], await asyncFrom(nodeRef.getChildren())]) {
for(const tryNodeRef of iterable) {
const tryCenter = (await tryNodeRef.getEffectiveCenter()).noZ();
let best = null;
/* For this node, go through every candidate and select the "best".
*
* The best candidate to provide a background is, in order of importance:
* 1. Of a node type that explicitly gives a background.
* 2. Of a high z-level (i.e. on top).
*
* This allows for situations where no node explicitly gives a background for nodes on top to still inheirit background color,
* and for stacks of nodes to always inheirit from the top-most node that does explicitly give a background.
*/
for(const candidate of candidates) {
if(candidate.point.subtract(tryCenter).length() <= candidate.radius) {
if(!best || (candidate.z >= best.z && (candidate.givesBackground || !best.givesBackground)) || (!best.givesBackground && candidate.givesBackground)) {
best = candidate;
}
}
}
this.backgroundNodeCache[tryNodeRef.id] = best ? best.nodeRef : null;
}
}
}
async recalculateLoop() {
// Change the zoom level if requested.
// We do this in the same async loop method as recalculating the renderings so that the rendering is never out of sync between zoom levels.
if(this.zoom !== this.requestedZoom && this.msSinceLastZoomRequest() > this.zoomRequestTimeout) {
const oldLandmark = this.canvasPointToMap(this.mousePosition);
this.zoom = this.requestedZoom;
const newLandmark = this.canvasPointToMap(this.mousePosition);
this.scrollOffset = this.scrollOffset.add(this.mapPointToCanvas(oldLandmark).subtract(this.mapPointToCanvas(newLandmark)));
await this.hooks.call("changed_zoom", this.zoom);
this.recalculateEntireViewport();
}
const activity = this.activity();
if(this.wasActivity !== activity) {
if(!activity) {
this.recalculateEntireViewport();
}
this.wasActivity = activity;
}
const oldLength = this.infoMessages.length;
this.infoMessages = this.infoMessages.filter(m => performance.now() - m.when < this.infoMessageTimeout);
if(this.infoMessages.length > 0 || this.infoMessages.length !== oldLength) {
this.requestRedraw();
}
// If anything's changed on the map, try to recalculate the renderings.
if(this.recalculateViewport || this.recalculateUpdate.length > 0 || this.recalculateRemoved.length > 0 || this.recalculateTranslated.length > 0) {
await this.recalculate(this.recalculateViewport, this.recalculateUpdate.splice(0, this.recalculateUpdate.length), this.recalculateRemoved.splice(0, this.recalculateRemoved.length), this.recalculateTranslated.splice(0, this.recalculateTranslated.length));
await this.redrawSelection();
this.recalculateViewport = false;
}
if(this.alive) {
setTimeout(this.recalculateLoop.bind(this), 50);
}
}
async performAction(action, addToUndoStack) {
const undo = await action.perform();
if(addToUndoStack) {
this.pushUndo(undo);
}
await this.hooks.call("action", action, undo, addToUndoStack);
return undo;
}
async stripDoStack(filter) {
this.undoStack = this.undoStack.filter(action => !filter(action));
this.redoStack = this.redoStack.filter(action => !filter(action));
await this.hooks.call("do_stripped");
}
hoveringOverSelection() {
return this.selection.exists() && this.hoverSelection.exists() && this.selection.contains(this.hoverSelection);
}
pushUndo(action, fromRedo) {
if(!action.empty()) {
this.undoStack.push(action);
if(!fromRedo) {
this.redoStack = [];
}
}
this.hooks.call("undo_pushed", action, fromRedo);
}
requestRecheckSelection() {
this.wantRecheckSelection = true;
}
requestUpdateSelection() {
this.wantUpdateSelection = true;
}
requestRedraw() {
this.wantRedraw = true;
}
focus() {
this.canvas.focus();
}
isKeyDown(key) {
return !!this.pressedKeys[key];
}
clearKeysDown() {
this.pressedKeys = {};
}
isAnyMouseButtonDown() {
for(const button in this.mouseDragEvents) {
return true;
}
return false;
}
isMouseButtonDown(button) {
return !!this.mouseDragEvents[button];
}
endMouseButtonPress(button, where) {
if(this.mouseDragEvents[button] !== undefined) {
this.mouseDragEvents[button].end(where);
delete this.mouseDragEvents[button];
}
}
cancelMouseButtonPress(button) {
if(this.mouseDragEvents[button] !== undefined) {
this.mouseDragEvents[button].cancel();
delete this.mouseDragEvents[button];
this.requestRedraw();
}
}
cancelMouseButtonPresses() {
for(const button in this.mouseDragEvents) {
this.cancelMouseButtonPress(button);
}
}
getBrush() {
return this.brush;
}
canvasPointToMap(v) {
return new Vector3(v.x, v.y, 0).add(this.scrollOffset).map((a) => this.pixelsToUnits(a));
}
mapPointToCanvas(v) {
return new Vector3(v.x, v.y, 0).map((a) => this.unitsToPixels(a)).subtract(this.scrollOffset);
}
mapPointToAbsoluteCanvas(v) {
return new Vector3(v.x, v.y, 0).map((a) => this.unitsToPixels(a));
}
canvasPathToMap(path) {
return path.mapOrigin((origin) => this.canvasPointToMap(origin)).mapLines((v) => v.map((a) => this.pixelsToUnits(a)));
}
/**
* Get the zoom factor based on a specific zoom level.
* @param zoom {number}
* @returns {number} Zoom factor, multiply number of pixels by this factor to get map units.
*/
zoomFactor(zoom) {
return zoom / (1 + 20 / zoom);
}
unitsPerPixelToZoom(unitsPerPixel) {
return Math.ceil((unitsPerPixel + Math.sqrt((unitsPerPixel ** 2) + 80 * unitsPerPixel)) / 2);
}
pixelsToUnits(pixels) {
return pixels * this.zoomFactor(this.zoom);
}
unitsToPixels(units) {
return units / this.pixelsToUnits(1);
}
screenSize() {
return new Vector3(this.canvas.width, this.canvas.height, 0);
}
screenBox() {
return new Box3(Vector3.ZERO, this.screenSize());
}
absoluteScreenBox() {
return new Box3(this.scrollOffset, this.screenSize().add(this.scrollOffset));
}
/** Recalculate the UI size based on the parent.
* This requires a full redraw.
*/
recalculateSize() {
if(this.inExportMode()) {
this.canvas.width = this.options.exportBox.size().x;
this.canvas.height = this.options.exportBox.size().y;
}
else {
// Keep the canvas matching the parent size.
this.canvas.width = this.parent.clientWidth;
this.canvas.height = this.parent.clientHeight;
}
this.selectionCanvas.width = this.canvas.width;
this.selectionCanvas.height = this.canvas.height;
this.hooks.call("size_change");
this.recalculateEntireViewport();
}
getNodeRender(nodeRef) {
let nodeRender = this.nodeRenders[nodeRef.id];
if(nodeRender === undefined) {
this.nodeRenders[nodeRef.id] = nodeRender = new NodeRender(this, nodeRef);
}
return nodeRender;
}
invalidateNodeRender(nodeRef) {
this.removeNodeRender(nodeRef);
}
removeNodeRender(nodeRef) {
delete this.nodeRenders[nodeRef.id];
}
recalculateEntireViewport() {
this.recalculateViewport = true;
}
async objectNode(nodeRef) {
if(await nodeRef.getNodeType() === "object")
return nodeRef;
else
return await nodeRef.getParent();
}
recalculateNodeUpdate(nodeRef) {
this.recalculateUpdate.push(nodeRef);
}
recalculateNodesRemove(nodeRefs) {
this.recalculateRemoved.push(...nodeRefs);
}
recalculateNodesTranslate(nodeRefs) {
this.recalculateTranslated.push(...nodeRefs);
}
async recalculate(viewport, updatedNodeRefs, removedNodeRefs, translatedNodeRefs) {
const redrawNodeIds = new Set();
const updateNodeIds = new Set();
let drawnNodeIds = this.drawnNodeIds[this.zoom];
if(drawnNodeIds === undefined) {
drawnNodeIds = this.drawnNodeIds[this.zoom] = new Set();
}
let labelPositions = this.labelPositions[this.zoom];
if(labelPositions === undefined) {
labelPositions = this.labelPositions[this.zoom] = {};
}
const visibleNodeIds = new Set(await asyncFrom(this.visibleObjectNodes(), nodeRef => nodeRef.id));
for(const visibleNodeId of visibleNodeIds) {
if(!drawnNodeIds.has(visibleNodeId)) {
redrawNodeIds.add(visibleNodeId);
updateNodeIds.add(visibleNodeId);
}
else if(viewport) {
redrawNodeIds.add(visibleNodeId);
}
}
for(const nodeRef of updatedNodeRefs) {
const actualNodeRef = await this.objectNode(nodeRef);
this.invalidateNodeRender(actualNodeRef);
redrawNodeIds.add(actualNodeRef.id);
updateNodeIds.add(actualNodeRef.id);
}
for(const nodeRef of removedNodeRefs) {
const actualNodeRef = await this.objectNode(nodeRef);
this.removeNodeRender(actualNodeRef);
redrawNodeIds.add(actualNodeRef.id);
drawnNodeIds.delete(actualNodeRef.id);
delete labelPositions[actualNodeRef.id];
}
for(const nodeRef of translatedNodeRefs) {
const actualNodeRef = await this.objectNode(nodeRef);
this.invalidateNodeRender(actualNodeRef);
redrawNodeIds.add(actualNodeRef.id);
}
const redrawMegaTiles = new Set();
for(const nodeId of redrawNodeIds) {
const megaTilesByNode = this.nodeIdsToMegatiles[nodeId];
if(megaTilesByNode !== undefined) {
for(const megaTile of megaTilesByNode) {
const tilePosition = megaTile.tileCorner;
for(const nodeId of megaTile.nodeIds) {
updateNodeIds.add(nodeId);
}
delete this.megaTiles[megaTile.oneUnitInPixels][tilePosition.x][tilePosition.y];
}
delete this.nodeIdsToMegatiles[nodeId];
}
}
for(const nodeRef of removedNodeRefs) {
const actualNodeRef = await this.objectNode(nodeRef);
delete this.nodeIdsToMegatiles[actualNodeRef.id];
}
const screenBox = this.absoluteScreenBox();
const screenBoxInTiles = this.absoluteScreenBox().map(v => v.divideScalar(tileSize).map(Math.floor));
const screenBoxInMegaTiles = this.absoluteScreenBox().map(v => v.divideScalar(megaTileSize).map(Math.floor));
let megaTiles = this.megaTiles[this.zoom];
if(megaTiles === undefined) {
megaTiles = this.megaTiles[this.zoom] = {};
}
const drewToMegaTiles = new Set();
// Sort all layers by Z order.
const nodeLayers = Array.from(this.mapper.backend.layerRegistry.getLayers());
nodeLayers.sort((a, b) => a.getZ() - b.getZ());
// A list of filters in order; nodes matching each filter will be rendered on the same Z level.
const filters = [];
// Add a filter for each layer in order.
for(const layer of nodeLayers) {
if(layer.id === "geographical") {
// If this is the geographical layer, render terrain objects before explicit objects.
filters.push(async nodeRef => (await nodeRef.getLayer()).id === layer.id && (await nodeRef.getType()).getScale() === "terrain");
filters.push(async nodeRef => (await nodeRef.getLayer()).id === layer.id && (await nodeRef.getType()).getScale() === "explicit");
}
else {
filters.push(async nodeRef => (await nodeRef.getLayer()).id === layer.id);
}
}
for(const filter of filters) {
const focusTiles = {};
const drawLayer = async (layer, drawAgainIds, callbacks) => {
const nodeId = layer.nodeRender.nodeRef.id;
const absoluteLayerBox = Box3.fromOffset(layer.corner, new Vector3(layer.width, layer.height, 0));
const layerBoxInMegaTiles = absoluteLayerBox.map(v => v.divideScalar(megaTileSize).map(Math.floor));
for(let x = Math.max(layerBoxInMegaTiles.a.x, screenBoxInMegaTiles.a.x); x <= Math.min(layerBoxInMegaTiles.b.x, screenBoxInMegaTiles.b.x); x++) {
let megaTileX = megaTiles[x];
if(megaTileX === undefined) {
megaTileX = megaTiles[x] = {};
}
for(let y = Math.max(layerBoxInMegaTiles.a.y, screenBoxInMegaTiles.a.y); y <= Math.min(layerBoxInMegaTiles.b.y, screenBoxInMegaTiles.b.y); y++) {
const megaTilePoint = new Vector3(x, y, 0);
let megaTile = megaTileX[y];
if(megaTile === undefined) {
megaTile = megaTileX[y] = new MegaTile(this, this.zoom, megaTilePoint);
redrawMegaTiles.add(megaTile);
}
const firstAppearanceInMegaTile = !megaTile.nodeIds.has(nodeId);
if(redrawMegaTiles.has(megaTile) || firstAppearanceInMegaTile) {
const pointOnLayer = megaTilePoint.multiplyScalar(megaTileSize).subtract(absoluteLayerBox.a);
const realPointOnLayer = pointOnLayer.map(c => Math.max(c, 0));
const pointOnMegaTile = realPointOnLayer.subtract(pointOnLayer);
callbacks.push({
callback: async () => {
megaTile.context.drawImage(await layer.canvas(), realPointOnLayer.x, realPointOnLayer.y, megaTileSize, megaTileSize, pointOnMegaTile.x, pointOnMegaTile.y, megaTileSize, megaTileSize);
this.nodeIdsToMegatiles[nodeId].add(megaTile);
megaTile.nodeIds.add(nodeId);
megaTile.addParts(layer.parts);
drewToMegaTiles.add(megaTile);
if(!layer.zWait) {
let averagePartPoint = Vector3.ZERO;
for(const part of layer.parts) {
averagePartPoint = averagePartPoint.add(part.absolutePoint);
}
labelPositions[nodeId] = {
center: Vector3.max(Vector3.min(averagePartPoint.divideScalar(layer.parts.length), screenBox.b), screenBox.a),
size: Math.min(24, Math.ceil(this.unitsToPixels(await this.mapper.backend.getNodeRef(nodeId).getRadius()) / 4)),
};
}
},
z: layer.z,
});
}
if(firstAppearanceInMegaTile && drawAgainIds) {
for(const otherNodeId of megaTile.nodeIds) {
drawAgainIds.add(otherNodeId);
}
}
}
}
};
const waitLayers = new Set();
const drawNodeIds = async (nodeIds, drawAgainIds) => {
const layers = [];
const focusTileLists = new Set();
for(const nodeId of nodeIds) {
const nodeRef = this.mapper.backend.getNodeRef(nodeId);
// Only render valid nodes in the current filter.
if(!await filter(nodeRef) || !(await nodeRef.valid()))
continue;
drawnNodeIds.add(nodeRef.id);
const nodeRender = this.getNodeRender(nodeRef);
for(const layer of await nodeRender.getLayers(this.zoom)) {
layers.push(layer);
if(!this.activity()) {
focusTileLists.add(layer.focusTiles);
}
}
if(this.nodeIdsToMegatiles[nodeId] === undefined)
this.nodeIdsToMegatiles[nodeId] = new Set();
}
const callbacks = [];
for(const layer of layers) {
if(layer.zWait) {
waitLayers.add(layer);
}
else {
await drawLayer(layer, drawAgainIds, callbacks);
}
}
callbacks.sort((a, b) => a.z - b.z);
for(const callback of callbacks) {
await callback.callback();
}
for(const subFocusTiles of focusTileLists) {
for(const tX in subFocusTiles) {
const subFocusTilesX = subFocusTiles[tX];
let focusTilesX = focusTiles[tX];
if(focusTilesX === undefined) {
focusTilesX = focusTiles[tX] = {};
}
for(const tY in subFocusTilesX) {
focusTilesX[tY] = subFocusTilesX[tY];
}
}
}
};
const secondPassNodeIds = new Set();
await drawNodeIds(updateNodeIds, secondPassNodeIds);
await drawNodeIds(secondPassNodeIds);
for(let tX in focusTiles) {
tX = +tX;
if(tX >= screenBoxInTiles.a.x && tX <= screenBoxInTiles.b.x) {
const megaTilePointX = Math.floor(tX * tileSize / megaTileSize);
const megaTileX = megaTiles[megaTilePointX];
if(megaTileX !== undefined) {
const focusTilesX = focusTiles[tX];
for(let tY in focusTilesX) {
tY = +tY;
if(tY >= screenBoxInTiles.a.y && tY <= screenBoxInTiles.b.y) {
const megaTilePointY = Math.floor(tY * tileSize / megaTileSize);
const megaTile = megaTileX[megaTilePointY];
if(drewToMegaTiles.has(megaTile)) {
const tile = focusTilesX[tY];
const center = tile.centerPoint;
const drawPoint = center.subtract(megaTile.corner);
const neighbors = [];
for(const dirKey of dirKeys) {
const tileDir = dirs[dirKey].multiplyScalar(tileSize);
const neighborPoint = center.add(tileDir.divideScalar(2)).map(c => Math.floor(c)).map(c => c - c % 16);
const neighborNodePart = await this.getDrawnNodePartAtAbsoluteCanvasPointTileAligned(neighborPoint, tile.layer);
if(neighborNodePart) {
neighbors.push({
nodeRef: neighborNodePart.nodeRef,
part: neighborNodePart,
angle: dirAngles[dirKey],
normalizedDir: normalizedDirs[dirKey],
});
}
}
const c = megaTile.context;
for(const neighbor of neighbors) {
c.fillStyle = neighbor.part.fillStyle;
c.globalAlpha = 0.5;
const angle = neighbor.angle;
const arcPoint = drawPoint.add(neighbor.normalizedDir.multiplyScalar(tileSize / 2));
c.beginPath();
c.arc(arcPoint.x, arcPoint.y, tileSize / 2, angle - Math.PI / 2, angle + Math.PI / 2, false);
c.fill();
}
c.globalAlpha = 1;
}
}
}
}
}
}
const callbacks = [];
for(const layer of waitLayers) {
drawLayer(layer, undefined, callbacks);
}
callbacks.sort((a, b) => a.z - b.z);
for(const callback of callbacks) {
await callback.callback();
}
}
this.hooks.call("calculated");
this.requestRedraw();
}
async drawBrush() {
await this.brush.draw(this.canvas.getContext("2d"), this.mousePosition);
}
async clearCanvas() {
const c = this.canvas.getContext("2d");
c.beginPath();
c.rect(0, 0, this.canvas.width, this.canvas.height);
c.fillStyle = this.backgroundColor;
c.fill();
}
async drawHelp() {
const c = this.canvas.getContext("2d");
c.textBaseline = "top";
c.font = "18px sans";
let infoLineY = 9;
function infoLine(l) {
const measure = c.measureText(l);
c.globalAlpha = 0.25;
c.fillStyle = "black";
c.fillRect(18, infoLineY - 2, measure.width, Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent) + 4);
c.globalAlpha = 1;
c.fillStyle = "white";
c.fillText(l, 18, infoLineY);
infoLineY += 24;
}
// Debug help
if(this.inNormalMode()) {
infoLine("Press 'n' to set or edit an object's name. ` to toggle debug mode.");
}
if(this.brush instanceof AddBrush) {
infoLine("Click to add terrain");
}
else if(this.brush instanceof SelectBrush) {
infoLine("Click to select, drag to move. Hold Control and click to select multiply objects.");
}
else if(this.brush instanceof DeleteBrush) {
infoLine("Click to delete an area. Hold Shift and click to delete an entire object.");
}
else if(this.brush instanceof AreaBrush) {
infoLine("Click to select an area. Hold Shift and click to delete part of that area.");
}
infoLine("Right click or arrow keys to move map.");
}
async drawDebug() {
const c = this.canvas.getContext("2d");
const drawn = new Set();
c.setLineDash([]);
const drawNodePoint = async (nodeRef) => {
if(!drawn.has(nodeRef.id)) {
drawn.add(nodeRef.id);
const position = this.mapPointToCanvas(await nodeRef.getCenter());
c.beginPath();
c.arc(position.x, position.y, 4, 0, 2 * Math.PI, false);
c.strokeStyle = "white";
c.stroke();
// Draw edges.
for await (const dirEdgeRef of nodeRef.getEdges()) {
if(!drawn.has(dirEdgeRef.id)) {
drawn.add(dirEdgeRef.id);
const otherNodeRef = await dirEdgeRef.getDirOtherNode();
const otherPosition = this.mapPointToCanvas(await otherNodeRef.getCenter());
c.strokeStyle = "white";
c.beginPath();
c.moveTo(position.x, position.y);
c.lineTo(otherPosition.x, otherPosition.y);
c.stroke();
}
}
// Draw effective bounding radius.
const effectivePosition = this.mapPointToCanvas(await nodeRef.getEffectiveCenter());
c.beginPath();
c.arc(effectivePosition.x, effectivePosition.y, this.unitsToPixels(await nodeRef.getRadius()), 0, 2 * Math.PI, false);
c.strokeStyle = "gray";
c.stroke();
}
};
for await (const nodeRef of this.drawnNodes()) {
// Draw center.
await drawNodePoint(nodeRef);
// Draw border path.
for await (const child of nodeRef.getChildren()) {
await drawNodePoint(child);
}
// Draw bounding radius.
const position = this.mapPointToCanvas(await nodeRef.getCenter());
c.beginPath();
c.arc(position.x, position.y, this.unitsToPixels(await nodeRef.getRadius()), 0, 2 * Math.PI, false);
c.strokeStyle = "gray";
c.stroke();
}
c.strokeStyle = "black";
const screenBoxInMegaTiles = this.absoluteScreenBox().map(v => v.divideScalar(megaTileSize).map(Math.floor));
for(let x = screenBoxInMegaTiles.a.x; x <= screenBoxInMegaTiles.b.x; x++) {
for(let y = screenBoxInMegaTiles.a.y; y <= screenBoxInMegaTiles.b.y; y++) {
const point = new Vector3(x, y, 0).multiplyScalar(megaTileSize).subtract(this.scrollOffset);
c.strokeRect(point.x, point.y, megaTileSize, megaTileSize);
c.strokeText(`${x}, ${y}`, point.x, point.y);
}
}
}
async drawScale() {
const c = this.canvas.getContext("2d");
const barHeight = 10;
const barWidth = this.canvas.width / 5 - mod(this.canvas.width / 5, this.unitsToPixels(this.mapper.metersToUnits(10 ** Math.ceil(Math.log10(this.zoom * 5)))));
const barX = 10;
const label2Y = this.canvas.height - barHeight;
const barY = label2Y - barHeight - 15;
const labelY = barY - barHeight / 2;
c.textBaseline = "alphabetic";
c.fillStyle = "black";
c.fillRect(barX, barY, barWidth, barHeight);
c.font = "16px mono";
c.fillStyle = "white";
for(let point = 0; point < 6; point++) {
const y = (point % 2 === 0) ? labelY : label2Y;
const pixel = barWidth * point / 5;
c.fillText(`${Math.floor(this.mapper.unitsToMeters(this.pixelsToUnits(pixel)) + 0.5)}m`, barX + pixel, y);
c.fillRect(barX + pixel, barY, 2, barHeight);
}
}
async drawVersion() {
const c = this.canvas.getContext("2d");
c.textBaseline = "top";
c.font = "14px sans";
const text = `v${version}`;
const measure = c.measureText(text);
const x = this.screenBox().b.x - measure.width;
const height = Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent);
c.globalAlpha = 0.25;
c.fillStyle = "black";
c.fillRect(x, 0, measure.width, height);
c.fillStyle = "white";
c.globalAlpha = 1;
c.fillText(text, x, 0);
}
async drawPegs() {
const c = this.canvas.getContext("2d");
const colors = {
1: "red",
2: "blue",
};
const positions = {};
for(const distanceMarkerN in this.distanceMarkers) {
const distanceMarker = this.distanceMarkers[distanceMarkerN];
const position = this.mapPointToCanvas(distanceMarker);
positions[distanceMarkerN] = position;
c.beginPath();
c.arc(position.x, position.y, 4, 0, 2 * Math.PI, false);
c.fillStyle = colors[distanceMarkerN] || "black";
c.fill();
c.fillStyle = "white";
c.textBaseline = "alphabetic";
c.font = "16px mono";
const worldPosition = this.canvasPointToMap(position).map(c => this.mapper.unitsToMeters(c)).round();
const text = `${worldPosition.x}m, ${worldPosition.y}m, ${worldPosition.z}m`;
c.fillText(text, position.x - c.measureText(text).width / 2, position.y - 16);
}
if(positions[1] && positions[2]) {
c.lineWidth = 3;
c.setLineDash([5, 15]);
c.strokeStyle = "black";
c.beginPath();
c.moveTo(positions[1].x, positions[1].y);
c.lineTo(positions[2].x, positions[2].y);
c.stroke();
c.setLineDash([11, 22]);
c.strokeStyle = "white";
c.beginPath();
c.moveTo(positions[1].x, positions[1].y);
c.lineTo(positions[2].x, positions[2].y);
c.stroke();
c.setLineDash([]);
c.lineWidth = 1;
const meters = this.mapper.unitsToMeters(this.distanceMarkers[1].subtract(this.distanceMarkers[2]).length());
c.textBaseline = "top";
c.font = "16px mono";
const position = this.mapPointToCanvas(positions[1].add(positions[2]).divideScalar(2).round());
const text = `Distance between markers: ${Math.floor(meters + 0.5)}m (${Math.floor(meters / 1000 + 0.5)}km)`;
const measure = c.measureText(text);
const height = Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent);
c.globalAlpha = 0.25;
c.fillStyle = "black";
c.fillRect(position.x - measure.width / 2, position.y, measure.width, height);
c.globalAlpha = 1;
c.fillStyle = "white";
c.fillText(text, position.x - measure.width / 2, position.y);
}
}
async drawNodes() {
const c = this.canvas.getContext("2d");
const megaTiles = this.megaTiles[this.zoom];
if(megaTiles !== undefined) {
const screenBoxInMegaTiles = this.absoluteScreenBox().map(v => v.divideScalar(megaTileSize).map(Math.floor));
for(let x = screenBoxInMegaTiles.a.x; x <= screenBoxInMegaTiles.b.x; x++) {
const megaTileX = megaTiles[x];
if(megaTileX !== undefined) {
for(let y = screenBoxInMegaTiles.a.y; y <= screenBoxInMegaTiles.b.y; y++) {
const megaTile = megaTileX[y];
if(megaTile !== undefined) {
const point = megaTile.corner.subtract(this.scrollOffset);
c.drawImage(megaTile.canvas, point.x, point.y);
}
}
}
}
}
if(this.drawSelectionCanvas && (this.brush.usesHover() || this.brush.usesSelection())) {
c.globalAlpha = 0.25;
const offset = this.selectionCanvasScroll.subtract(this.scrollOffset);
c.drawImage(this.selectionCanvas, offset.x, offset.y);
c.globalAlpha = 1;
}
}
async drawLabels() {
const c = this.canvas.getContext("2d");
c.textBaseline = "top";
const currentLayer = this.getCurrentLayer();
const seenBoxes = [];
const collides = (box) => {
for(const seenBox of seenBoxes) {
if(box.collides(seenBox)) {
return true;
}
}
return false;
};
for await (const nodeRef of this.drawnNodes()) {
const labelText = await nodeRef.getPString("name");
if(labelText !== undefined && labelText.length > 0) {
const labelPositionOnCanvas = await this.getNamePosition(nodeRef);
const layer = (await nodeRef.getType()).getLayer();
const layerSelected = layer.id === currentLayer.id;
const selected = (this.selection.hasNodeRef(nodeRef) || this.hoverSelection.hasNodeRef(nodeRef));
const fontSize = (selected ? 24 : labelPositionOnCanvas.size) * (layerSelected ? 1 : 0.5);
const font = layerSelected ? "serif" : "sans";
c.font = selected ? `bold ${fontSize}px ${font}` : `${fontSize}px ${font}`;
const measure = c.measureText(labelText);
const height = Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent);
const originalBox = Box3.fromOffset(labelPositionOnCanvas.center.subtract(this.scrollOffset).subtract(new Vector3(measure.width / 2, height / 2, 0, 0)), new Vector3(measure, height, 0)).map(v => v.noZ());
let box = originalBox;
let amount = 12;
let arc = 0;
while(collides(box)) {
if(arc > Math.PI * 2) {
arc = 0;
amount = amount + 12;
continue;
}
box = originalBox.map(v => v.add((new Vector3(Math.cos(arc), Math.sin(arc), 0)).multiplyScalar(amount)));
arc = arc + 8 / Math.PI;
}
seenBoxes.push(box);
const where = box.a;
c.globalAlpha = 0.25;
c.fillStyle = "black";
c.fillRect(where.x, where.y, measure.width, height);
c.globalAlpha = 1;
c.fillStyle = "white";
c.fillText(labelText, where.x, where.y);
}
}
}
async drawZoom() {
const timeoutFraction = Math.max(0, this.msSinceLastZoomRequest() / this.zoomRequestTimeout);
const pixelToMeters = this.mapper.unitsToMeters(this.zoomFactor(this.requestedZoom));
const lines = [
`Zoom ${this.requestedZoom}`,
`1px = ${pixelToMeters.toFixed(2)}m`,
`Brush diameter ${(pixelToMeters * this.brush.getRadius()).toFixed(2)}m`,
`Screen diagonal ${(this.mapper.unitsToMeters(this.zoomFactor(this.requestedZoom) * (new Vector3(0, 0, 0)).subtract(this.screenSize()).length()) / 1000).toFixed(2)}km`,
"Click to apply",
];
const c = this.canvas.getContext("2d");
c.textBaseline = "top";
c.font = "16px mono";
let width = 0;
let height = 0;
for(const text of lines) {
const measure = c.measureText(text);
height = Math.max(height, Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent));
width = Math.max(width, measure.width);
}
const totalHeight = height * lines.length;
const screenCenter = this.screenSize().divideScalar(2).round();
const radius = Math.ceil(Math.max(width, totalHeight) / 2);
const where = new Vector3(screenCenter.x - width / 2, screenCenter.y - totalHeight / 2, 0);
c.fillStyle = "black";
c.globalAlpha = 0.5;
c.beginPath();
c.arc(screenCenter.x, screenCenter.y, radius * Math.sqrt(2), 0, 2 * Math.PI, false);
c.fill();
c.globalAlpha = 1;
c.lineWidth = 2;
c.strokeStyle = "white";
c.beginPath();
c.arc(screenCenter.x, screenCenter.y, radius * Math.sqrt(2) + 2, 0, (1 - timeoutFraction) * 2 * Math.PI, false);
c.stroke();
for(let i = 0; i < lines.length; i++) {
const text = lines[i];
c.fillStyle = "white";
c.fillText(text, where.x, where.y + height * i);
}
}
async drawInfoMessages() {
const c = this.canvas.getContext("2d");
c.textBaseline = "top";
c.font = "24px mono";
const f = (message) => {
return Math.ceil(Math.max(0, 1 - (performance.now() - message.when) / this.infoMessageTimeout) * 0.5 * 24 + 24 * 0.5);
};
let width = 0;
let height = 0;
for(const message of this.infoMessages) {
const text = message.message;
const measure = c.measureText(text);
height = Math.max(height, Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent));
width = Math.max(width, measure.width);
}
const totalHeight = height * this.infoMessages.length;
const screenCenter = this.screenSize().divideScalar(2).round();
const where = new Vector3(screenCenter.x - width / 2, screenCenter.y - totalHeight / 2, 0);
c.fillStyle = "black";
c.globalAlpha = 0.5;
c.beginPath();
c.fillRect(where.x, where.y, width, totalHeight);
c.fill();
c.globalAlpha = 1;
for(let i = 0; i < this.infoMessages.length; i++) {
const message = this.infoMessages[i];
c.fillStyle = "white";
c.font = `${f(message)}px mono`;
const text = message.message;
const measure = c.measureText(text);
const actualHeight = Math.abs(measure.actualBoundingBoxAscent) + Math.abs(measure.actualBoundingBoxDescent);
c.fillText(text, where.x + Math.floor((width - measure.width) / 2), where.y + height * i + Math.floor((height - actualHeight) / 2));
}
}
/** Completely redraw the displayed UI. */
async redraw() {
await this.clearCanvas();
await this.drawNodes();
await this.drawLabels();
if(this.inControlledMode()) {
if(this.isCalculatingDistance()) {
await this.drawPegs();
}
await this.drawBrush();
if(this.msSinceLastZoomRequest() < this.zoomRequestTimeout) {
await this.drawZoom();
this.requestRedraw();
}
await this.drawHelp();
await this.drawScale();
await this.drawInfoMessages();
if(this.debugMode) {
await this.drawDebug();
}
await this.drawVersion();
}
await this.hooks.call("drawn");
}
async * visibleObjectNodes() {
yield* this.getObjectNodesInRelativeBox(this.screenBox());
}
async * getObjectNodesInAbsoluteBox(box) {
yield* this.getObjectNodesInBox(box.map(v => v.map(c => this.pixelsToUnits(c))));
}
async * getObjectNodesInRelativeBox(box) {
yield* this.getObjectNodesInBox(box.map((v) => this.canvasPointToMap(v)));
}
async * getObjectNodesInBox(box) {
const mapBox = box.map(v => v);
mapBox.a.z = -Infinity;
mapBox.b.z = Infinity;
yield* this.mapper.getObjectNodesTouchingArea(mapBox, this.pixelsToUnits(1));
}
async * drawnNodes() {
const drawnNodeIds = this.drawnNodeIds[this.zoom];
if(drawnNodeIds !== undefined) {
for(const nodeId of drawnNodeIds) {
yield this.mapper.backend.getNodeRef(nodeId);
}
}
}
async * allDrawnNodes() {
yield* this.drawnNodes();
const megaTiles = this.megaTiles[this.zoom];
if(megaTiles !== undefined) {
const screenBoxInMegaTiles = this.absoluteScreenBox().map(v => v.divideScalar(megaTileSize).map(Math.floor));
for(let x = screenBoxInMegaTiles.a.x; x <= screenBoxInMegaTiles.b.x; x++) {
const megaTileX = megaTiles[x];
if(megaTileX !== undefined) {
for(let y = screenBoxInMegaTiles.a.y; y <= screenBoxInMegaTiles.b.y; y++) {
const megaTile = megaTileX[y];
if(megaTile !== undefined) {
for(const part of megaTile.parts) {
yield part.nodeRef;
}
}
}
}
}
}
}
/** Disconnect the render context from the page and clean up listeners. */
disconnect() {
this.alive = false;
this.hooks.call("disconnect").then(() => {
this.styleElement.remove();
this.parentObserver.disconnect();
this.canvas.remove();
});
}
}
/** Mapper interface
* A connection to a database and mapper UI.
* Instantiate Mapper and then call the render() method to insert the UI into a div element.
*/
class Mapper {
/* Set the backend for the mapper, i.e. the map it is presenting.
* See: backend.js
*/
constructor(backend) {
this.backend = backend;
this.hooks = new HookContainer();
this.backend.hooks.add("load", async () => await this.hooks.call("update"));
this.hooks.add("updateNode", async () => await this.hooks.call("update"));
this.hooks.add("insertNode", async (nodeRef) => await this.hooks.call("updateNode", nodeRef));
this.hooks.add("removeNodes", async () => await this.hooks.call("update"));
this.hooks.add("translateNodes", async () => await this.hooks.call("update"));
this.hooks.add("update", () => { this.declareUnsavedChanges(); });
this.options = {
blendDistance: 400,
cleanNormalDistance: 0.5,
};
this.unsavedChanges = false;
}
unitsToMeters(units) {
return units * 2;
}
metersToUnits(meters) {
return meters / this.unitsToMeters(1);
}
clearUnsavedChangeState() {
this.unsavedChanges = false;
this.hooks.call("unsavedStateChange", false);
}
declareUnsavedChanges() {
this.unsavedChanges = true;
this.hooks.call("unsavedStateChange", true);
}
hasUnsavedChanges() {
return this.unsavedChanges;
}
/** Get all nodes in or near a spatial box (according to their radii).
* @param box {Box3}
* @param minRadius {number}
* @returns {AsyncIterable.<NodeRef>}
*/
async * getNodesTouchingArea(box, minRadius) {
yield* this.backend.getNodesTouchingArea(box, minRadius);
}
/** Get all nodes in or near a spatial box (according to their radii).
* @param box {Box3}
* @param minRadius {number}
* @returns {AsyncIterable.<NodeRef>}
*/
async * getObjectNodesTouchingArea(box, minRadius) {
yield* this.backend.getObjectNodesTouchingArea(box, minRadius);
}
/** Get all edges attached to the specified node.
* @param nodeRef {NodeRef}
* @returns {AsyncIterable.<DirEdgeRef>} the edges coming from the specified node
*/
async * getNodeEdges(nodeRef) {
yield* this.backend.getNodeEdges(nodeRef.id);
}
/** Render Mapper into a div element
* @returns {RenderContext}
* Example: const renderContext = mapper.render(document.getElementById("mapper_div"))
*/
render(element, options={}) {
return new RenderContext(element, this, options);
}
async insertNode(point, nodeType, options) {
const nodeRef = await this.backend.createNode(options.parent ? options.parent.id : null, nodeType);
await nodeRef.setCenter(point);
await nodeRef.setEffectiveCenter(point);
await nodeRef.setType(options.type);
await nodeRef.setLayer(this.backend.layerRegistry.get(options.type.getLayer()));
await nodeRef.setRadius(options.radius);
await this.hooks.call("insertNode", nodeRef);
return nodeRef;
}
async translateNode(originNodeRef, offset) {
const nodeRefs = await asyncFrom(originNodeRef.getSelfAndAllDescendants());
for(const nodeRef of nodeRefs) {
await nodeRef.setCenter((await nodeRef.getCenter()).add(offset));
await nodeRef.setEffectiveCenter((await nodeRef.getEffectiveCenter()).add(offset));
}
await this.hooks.call("translateNodes", nodeRefs);
}
async removeNodes(nodeRefs) {
let nodeIds = new Set(nodeRefs.map((nodeRef) => nodeRef.id));
for(const nodeRef of nodeRefs) {
for await (const childNodeRef of nodeRef.getAllDescendants()) {
nodeIds.add(childNodeRef.id);
}
}
const nodeRefsWithChildren = Array.from(nodeIds, (nodeId) => this.backend.getNodeRef(nodeId));
const parentNodeIds = new Set();
for(const nodeRef of nodeRefsWithChildren) {
const parent = await nodeRef.getParent();
if(parent && !nodeIds.has(parent.id)) {
parentNodeIds.add(parent.id);
}
await nodeRef.remove();
}
for(const nodeId of parentNodeIds) {
const nodeRef = this.backend.getNodeRef(nodeId);
if(!(await nodeRef.hasChildren())) {
await nodeRef.remove();
nodeRefsWithChildren.push(nodeRef);
}
}
await this.hooks.call("removeNodes", nodeRefsWithChildren);
return nodeRefsWithChildren;
}
async unremoveNodes(nodeRefs) {
for(const nodeRef of nodeRefs) {
await nodeRef.unremove();
await this.hooks.call("insertNode", nodeRef);
}
}
async removeEdges(edgeRefs) {
for(const edgeRef of edgeRefs) {
for await (const nodeRef of edgeRef.getNodes()) {
await this.hooks.call("updateNode", nodeRef);
}
await edgeRef.remove();
}
}
async unremoveEdges(edgeRefs) {
for(const edgeRef of edgeRefs) {
await edgeRef.unremove();
for await (const nodeRef of edgeRef.getNodes()) {
await this.hooks.call("updateNode", nodeRef);
}
}
}
}
export { Mapper };