mirror of
https://github.com/Azgaar/Fantasy-Map-Generator.git
synced 2026-02-04 09:31:23 +01:00
* chore: add npm + vite for progressive enhancement * fix: update Dockerfile to copy only the dist folder contents * fix: update Dockerfile to use multi-stage build for optimized production image * fix: correct nginx config file copy command in Dockerfile * chore: add netlify configuration for build and redirects * fix: add NODE_VERSION to environment in Netlify configuration * remove wrong dist folder * Update package.json Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * chore: split public and src * migrating all util files from js to ts * feat: Implement HeightmapGenerator and Voronoi module - Added HeightmapGenerator class for generating heightmaps with various tools (Hill, Pit, Range, Trough, Strait, etc.). - Introduced Voronoi class for creating Voronoi diagrams using Delaunator. - Updated index.html to include new modules. - Created index.ts to manage module imports. - Enhanced arrayUtils and graphUtils with type definitions and improved functionality. - Added utility functions for generating grids and calculating Voronoi cells. * chore: add GitHub Actions workflow for deploying to GitHub Pages * fix: update branch name in GitHub Actions workflow from 'main' to 'master' * chore: update package.json to specify Node.js engine version and remove unused launch.json * Initial plan * Update copilot guidelines to reflect NPM/Vite/TypeScript migration Co-authored-by: Azgaar <26469650+Azgaar@users.noreply.github.com> * Update src/modules/heightmap-generator.ts Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update src/utils/graphUtils.ts Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * Update src/modules/heightmap-generator.ts Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> * feat: Add TIME and ERROR variables to global scope in HeightmapGenerator * fix: Update base path in vite.config.ts for Netlify deployment * fix: Update Node.js version in Dockerfile to 24-alpine --------- Co-authored-by: Marc Emmanuel <marc.emmanuel@tado.com> Co-authored-by: Marc Emmanuel <marcwissler@gmail.com> Co-authored-by: Copilot <175728472+Copilot@users.noreply.github.com> Co-authored-by: copilot-swe-agent[bot] <198982749+Copilot@users.noreply.github.com> Co-authored-by: Azgaar <26469650+Azgaar@users.noreply.github.com>
144 lines
4.3 KiB
JavaScript
144 lines
4.3 KiB
JavaScript
"use strict";
|
|
function createRiver() {
|
|
if (customization) return;
|
|
closeDialogs();
|
|
if (!layerIsOn("toggleRivers")) toggleRivers();
|
|
|
|
document.getElementById("toggleCells").dataset.forced = +!layerIsOn("toggleCells");
|
|
if (!layerIsOn("toggleCells")) toggleCells();
|
|
|
|
tip("Click to add river point, click again to remove", true);
|
|
debug.append("g").attr("id", "controlCells");
|
|
viewbox.style("cursor", "crosshair").on("click", onCellClick);
|
|
|
|
createRiver.cells = [];
|
|
const body = document.getElementById("riverCreatorBody");
|
|
|
|
$("#riverCreator").dialog({
|
|
title: "Create River",
|
|
resizable: false,
|
|
position: {my: "left top", at: "left+10 top+10", of: "#map"},
|
|
close: closeRiverCreator
|
|
});
|
|
|
|
if (modules.createRiver) return;
|
|
modules.createRiver = true;
|
|
|
|
// add listeners
|
|
document.getElementById("riverCreatorComplete").addEventListener("click", addRiver);
|
|
document.getElementById("riverCreatorCancel").addEventListener("click", () => $("#riverCreator").dialog("close"));
|
|
body.addEventListener("click", function (ev) {
|
|
const el = ev.target;
|
|
const cl = el.classList;
|
|
const cell = +el.parentNode.dataset.cell;
|
|
if (cl.contains("editFlux")) pack.cells.fl[cell] = +el.value;
|
|
else if (cl.contains("icon-trash-empty")) removeCell(cell);
|
|
});
|
|
|
|
function onCellClick() {
|
|
const cell = findCell(...d3.mouse(this));
|
|
|
|
if (createRiver.cells.includes(cell)) removeCell(cell);
|
|
else addCell(cell);
|
|
}
|
|
|
|
function addCell(cell) {
|
|
createRiver.cells.push(cell);
|
|
drawCells(createRiver.cells);
|
|
|
|
const flux = pack.cells.fl[cell];
|
|
const line = `<div class="editorLine" data-cell="${cell}">
|
|
<span>Cell ${cell}</span>
|
|
<span data-tip="Set flux affects river width" style="margin-left: 0.4em">Flux</span>
|
|
<input type="number" min=0 value="${flux}" class="editFlux" style="width: 5em"/>
|
|
<span data-tip="Remove the cell" class="icon-trash-empty pointer"></span>
|
|
</div>`;
|
|
body.innerHTML += line;
|
|
}
|
|
|
|
function removeCell(cell) {
|
|
createRiver.cells = createRiver.cells.filter(c => c !== cell);
|
|
drawCells(createRiver.cells);
|
|
body.querySelector(`div[data-cell='${cell}']`)?.remove();
|
|
}
|
|
|
|
function drawCells(cells) {
|
|
debug
|
|
.select("#controlCells")
|
|
.selectAll(`polygon`)
|
|
.data(cells)
|
|
.join("polygon")
|
|
.attr("points", d => getPackPolygon(d))
|
|
.attr("class", "current");
|
|
}
|
|
|
|
function addRiver() {
|
|
const {rivers, cells} = pack;
|
|
const riverCells = createRiver.cells;
|
|
if (riverCells.length < 2) return tip("Add at least 2 cells", false, "error");
|
|
|
|
const riverId = Rivers.getNextId(rivers);
|
|
const parent = cells.r[last(riverCells)] || riverId;
|
|
|
|
riverCells.forEach(cell => {
|
|
if (!cells.r[cell]) cells.r[cell] = riverId;
|
|
});
|
|
|
|
const source = riverCells[0];
|
|
const mouth = parent === riverId ? last(riverCells) : riverCells[riverCells.length - 2];
|
|
const sourceWidth = Rivers.getSourceWidth(cells.fl[source]);
|
|
const defaultWidthFactor = rn(1 / (pointsInput.dataset.cells / 10000) ** 0.25, 2);
|
|
const widthFactor = 1.2 * defaultWidthFactor;
|
|
|
|
const meanderedPoints = Rivers.addMeandering(riverCells);
|
|
|
|
const discharge = cells.fl[mouth]; // m3 in second
|
|
const length = Rivers.getApproximateLength(meanderedPoints);
|
|
const width = Rivers.getWidth(
|
|
Rivers.getOffset({
|
|
flux: discharge,
|
|
pointIndex: meanderedPoints.length,
|
|
widthFactor,
|
|
startingWidth: sourceWidth
|
|
})
|
|
);
|
|
const name = Rivers.getName(mouth);
|
|
const basin = Rivers.getBasin(parent);
|
|
|
|
rivers.push({
|
|
i: riverId,
|
|
source,
|
|
mouth,
|
|
discharge,
|
|
length,
|
|
width,
|
|
widthFactor,
|
|
sourceWidth,
|
|
parent,
|
|
cells: riverCells,
|
|
basin,
|
|
name,
|
|
type: "River"
|
|
});
|
|
const id = "river" + riverId;
|
|
|
|
viewbox
|
|
.select("#rivers")
|
|
.append("path")
|
|
.attr("id", id)
|
|
.attr("d", Rivers.getRiverPath(meanderedPoints, widthFactor, sourceWidth));
|
|
|
|
editRiver(id);
|
|
}
|
|
|
|
function closeRiverCreator() {
|
|
body.innerHTML = "";
|
|
debug.select("#controlCells").remove();
|
|
restoreDefaultEvents();
|
|
clearMainTip();
|
|
|
|
const forced = +document.getElementById("toggleCells").dataset.forced;
|
|
document.getElementById("toggleCells").dataset.forced = 0;
|
|
if (forced && layerIsOn("toggleCells")) toggleCells();
|
|
}
|
|
}
|