refactor: generation script

This commit is contained in:
Azgaar 2022-07-13 01:53:06 +03:00
parent c0f6ce00ef
commit 87d8c1024d
31 changed files with 364 additions and 324 deletions

View file

@ -1869,15 +1869,10 @@
<button <button
id="configureWorld" id="configureWorld"
data-tip="Click to open world configurator to setup map position on Globe and World climate" data-tip="Click to open world configurator to setup map position on Globe and World climate"
onclick="editWorld()"
> >
Configure World Configure World
</button> </button>
<button <button id="optionsReset" data-tip="Click to restore default options (page will be reloaded)">
id="optionsReset"
data-tip="Click to restore default options (page will be reloaded)"
onclick="restoreDefaultOptions()"
>
Reset to defaults Reset to defaults
</button> </button>
</div> </div>
@ -5547,8 +5542,8 @@
<div id="options3dBottom" style="margin-top: 0.2em"> <div id="options3dBottom" style="margin-top: 0.2em">
<button id="options3dUpdate" data-tip="Update the scene" class="icon-cw"></button> <button id="options3dUpdate" data-tip="Update the scene" class="icon-cw"></button>
<button <button
id="options3dConfigureWorld"
data-tip="Configure world and map size and climate settings" data-tip="Configure world and map size and climate settings"
onclick="editWorld()"
class="icon-globe" class="icon-globe"
></button> ></button>
<button id="options3dSave" data-tip="Save screenshot of the 3d scene" class="icon-button-screenshot"></button> <button id="options3dSave" data-tip="Save screenshot of the 3d scene" class="icon-button-screenshot"></button>

View file

@ -3,7 +3,7 @@
"display": "standalone", "display": "standalone",
"orientation": "any", "orientation": "any",
"name": "Azgaar's Fantansy Map Generator", "name": "Azgaar's Fantansy Map Generator",
"short_name": "Azgaar's Fantansy Map Generator", "short_name": "Azgaar's Fantasy Map Generator",
"description": "Web application generating interactive and highly customizable maps", "description": "Web application generating interactive and highly customizable maps",
"scope": "/Fantasy-Map-Generator/", "scope": "/Fantasy-Map-Generator/",
"start_url": "/Fantasy-Map-Generator/?source=pwa", "start_url": "/Fantasy-Map-Generator/?source=pwa",

View file

@ -224,10 +224,10 @@ export function open(options) {
calculateTemperatures(grid); calculateTemperatures(grid);
generatePrecipitation(grid); generatePrecipitation(grid);
reGraph(grid); reGraph(grid);
reMarkFeatures(); reMarkFeatures(pack, newGrid);
drawCoastline(); drawCoastline(pack);
Rivers.generate(erosionAllowed); Rivers.generate(pack, grid, erosionAllowed);
if (!erosionAllowed) { if (!erosionAllowed) {
for (const i of pack.cells.i) { for (const i of pack.cells.i) {
@ -237,7 +237,7 @@ export function open(options) {
} }
} }
renderLayer("rivers"); renderLayer("rivers", pack);
Lakes.defineGroup(); Lakes.defineGroup();
Biomes.define(); Biomes.define();
rankCells(); rankCells();
@ -344,9 +344,9 @@ export function open(options) {
calculateTemperatures(grid); calculateTemperatures(grid);
generatePrecipitation(grid); generatePrecipitation(grid);
reGraph(grid); reGraph(grid);
drawCoastline(); drawCoastline(pack);
if (erosionAllowed) Rivers.generate(true); if (erosionAllowed) Rivers.generate(pack, grid, true);
// assign saved pack data from grid back to pack // assign saved pack data from grid back to pack
const n = pack.cells.i.length; const n = pack.cells.i.length;

View file

@ -2,7 +2,7 @@ import * as d3 from "d3";
import {heightmapTemplates} from "config/heightmap-templates"; import {heightmapTemplates} from "config/heightmap-templates";
import {precreatedHeightmaps} from "config/precreated-heightmaps"; import {precreatedHeightmaps} from "config/precreated-heightmaps";
import {generateGrid} from "utils/graphUtils"; import {generateGrid} from "scripts/generation/graph";
import {shouldRegenerateGridPoints} from "scripts/generation/generation"; import {shouldRegenerateGridPoints} from "scripts/generation/generation";
import {byId} from "utils/shorthands"; import {byId} from "utils/shorthands";
import {generateSeed} from "utils/probabilityUtils"; import {generateSeed} from "utils/probabilityUtils";

View file

@ -15,7 +15,8 @@ const dialogsMap = {
lakeEditor: "lake-editor", lakeEditor: "lake-editor",
religionsEditor: "religions-editor", religionsEditor: "religions-editor",
statesEditor: "states-editor", statesEditor: "states-editor",
unitsEditor: "units-editor" unitsEditor: "units-editor",
worldConfigurator: "world-configurator"
}; };
type TDialog = keyof typeof dialogsMap; type TDialog = keyof typeof dialogsMap;

View file

@ -1,7 +1,7 @@
export function drawRivers() { export function drawRivers(pack: IPack) {
rivers.selectAll("*").remove(); rivers.selectAll("*").remove();
const {addMeandering, getRiverPath} = Rivers; const {addMeandering, getRiverPath} = window.Rivers;
const riverPaths = pack.rivers.map(({cells, points, i, widthFactor, sourceWidth}) => { const riverPaths = pack.rivers.map(({cells, points, i, widthFactor, sourceWidth}) => {
if (!cells || cells.length < 2) return; if (!cells || cells.length < 2) return;
@ -12,7 +12,7 @@ export function drawRivers() {
points = undefined; points = undefined;
} }
const meanderedPoints = addMeandering(cells, points); const meanderedPoints = addMeandering(pack, cells, points);
const path = getRiverPath(meanderedPoints, widthFactor, sourceWidth); const path = getRiverPath(meanderedPoints, widthFactor, sourceWidth);
return `<path id="river${i}" d="${path}"/>`; return `<path id="river${i}" d="${path}"/>`;
}); });

View file

@ -39,9 +39,9 @@ const layerRenderersMap = {
temperature: drawTemperature temperature: drawTemperature
}; };
export function renderLayer(layerName: keyof typeof layerRenderersMap) { export function renderLayer(layerName: keyof typeof layerRenderersMap, ...args) {
const rendered = layerRenderersMap[layerName]; const renderer = layerRenderersMap[layerName];
TIME && console.time(rendered.name); TIME && console.time(renderer.name);
rendered(); renderer(...args);
TIME && console.timeEnd(rendered.name); TIME && console.timeEnd(renderer.name);
} }

View file

@ -363,7 +363,7 @@ function toggleTexture(event?: MouseEvent) {
function toggleRivers(event?: MouseEvent) { function toggleRivers(event?: MouseEvent) {
if (!layerIsOn("toggleRivers")) { if (!layerIsOn("toggleRivers")) {
turnLayerButtonOn("toggleRivers"); turnLayerButtonOn("toggleRivers");
renderLayer("rivers"); renderLayer("rivers", pack);
if (isCtrlPressed(event)) editStyle("rivers"); if (isCtrlPressed(event)) editStyle("rivers");
} else { } else {
if (isCtrlPressed(event)) return editStyle("rivers"); if (isCtrlPressed(event)) return editStyle("rivers");

View file

@ -85,7 +85,7 @@ window.Biomes = (function () {
} }
// assign biome id for each cell // assign biome id for each cell
function define() { function define(pack, grid) {
TIME && console.time("defineBiomes"); TIME && console.time("defineBiomes");
const {cells} = pack; const {cells} = pack;
const {temp, prec} = grid.cells; const {temp, prec} = grid.cells;

View file

@ -6,7 +6,7 @@ import {round} from "utils/stringUtils";
import {Ruler} from "modules/measurers"; import {Ruler} from "modules/measurers";
// Detect and draw the coastline // Detect and draw the coastline
export function drawCoastline() { export function drawCoastline(pack) {
TIME && console.time("drawCoastline"); TIME && console.time("drawCoastline");
const {cells, vertices, features} = pack; const {cells, vertices, features} = pack;

View file

@ -208,8 +208,8 @@ export function resolveVersionConflicts(version) {
coastline.selectAll("path").remove(); coastline.selectAll("path").remove();
lakes.selectAll("path").remove(); lakes.selectAll("path").remove();
reMarkFeatures(); reMarkFeatures(pack, newGrid);
drawCoastline(); drawCoastline(pack);
} }
if (version < 1.11) { if (version < 1.11) {

View file

@ -1,18 +1,19 @@
import * as d3 from "d3"; import * as d3 from "d3";
import {INFO} from "config/logging"; import {INFO} from "config/logging";
import {closeDialogs} from "dialogs/utils";
import {updatePresetInput} from "layers"; import {updatePresetInput} from "layers";
import {reMarkFeatures} from "modules/markup";
import {setDefaultEventHandlers} from "scripts/events"; import {setDefaultEventHandlers} from "scripts/events";
import {regenerateMap} from "scripts/generation/generation";
import {calculateVoronoi} from "scripts/generation/graph";
import {ldb} from "scripts/indexedDB"; import {ldb} from "scripts/indexedDB";
import {tip} from "scripts/tooltips"; import {tip} from "scripts/tooltips";
import {last} from "utils/arrayUtils"; import {last} from "utils/arrayUtils";
import {parseError} from "utils/errorUtils"; import {parseError} from "utils/errorUtils";
import {calculateVoronoi, findCell} from "utils/graphUtils"; import {findCell} from "utils/graphUtils";
import {link} from "utils/linkUtils"; import {link} from "utils/linkUtils";
import {minmax, rn} from "utils/numberUtils"; import {minmax, rn} from "utils/numberUtils";
import {regenerateMap} from "scripts/generation/generation";
import {reMarkFeatures} from "modules/markup";
import {closeDialogs} from "dialogs/utils";
// add drag to upload logic, pull request from @evyatron // add drag to upload logic, pull request from @evyatron
export function addDragToUpload() { export function addDragToUpload() {

View file

@ -2,13 +2,14 @@ import * as d3 from "d3";
import {TIME} from "config/logging"; import {TIME} from "config/logging";
import {rn} from "utils/numberUtils"; import {rn} from "utils/numberUtils";
// @ts-expect-error js module
import {aleaPRNG} from "scripts/aleaPRNG"; import {aleaPRNG} from "scripts/aleaPRNG";
import {byId} from "utils/shorthands";
import {getInputNumber, getInputValue} from "utils/nodeUtils"; import {getInputNumber, getInputValue} from "utils/nodeUtils";
import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation"; import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation";
import {byId} from "utils/shorthands";
window.Lakes = (function () { window.Lakes = (function () {
const setClimateData = function (h) { const setClimateData = function (h: Uint8Array, pack: IPack, grid: IGrid) {
const cells = pack.cells; const cells = pack.cells;
const lakeOutCells = new Uint16Array(cells.i.length); const lakeOutCells = new Uint16Array(cells.i.length);
@ -39,37 +40,39 @@ window.Lakes = (function () {
}; };
// get array of land cells aroound lake // get array of land cells aroound lake
const getShoreline = function (lake) { const getShoreline = function (lake: IPackFeatureLake, pack: IPack) {
const uniqueCells = new Set(); const uniqueCells = new Set();
lake.vertices.forEach(v => pack.vertices.c[v].forEach(c => pack.cells.h[c] >= 20 && uniqueCells.add(c))); lake.vertices.forEach(v =>
pack.vertices.c[v].forEach(c => pack.cells.h[c] >= MIN_LAND_HEIGHT && uniqueCells.add(c))
);
lake.shoreline = [...uniqueCells]; lake.shoreline = [...uniqueCells];
}; };
const prepareLakeData = h => { const prepareLakeData = (h: Uint8Array, pack: IPack) => {
const cells = pack.cells; const cells = pack.cells;
const ELEVATION_LIMIT = +document.getElementById("lakeElevationLimitOutput").value; const ELEVATION_LIMIT = getInputNumber("lakeElevationLimitOutput");
pack.features.forEach(f => { pack.features.forEach(feature => {
if (f.type !== "lake") return; if (!feature || feature.type !== "lake") return;
delete f.flux; delete feature.flux;
delete f.inlets; delete feature.inlets;
delete f.outlet; delete feature.outlet;
delete f.height; delete feature.height;
delete f.closed; delete feature.closed;
!f.shoreline && Lakes.getShoreline(f); !feature.shoreline && getShoreline(feature, pack);
// lake surface height is as lowest land cells around // lake surface height is as lowest land cells around
const min = f.shoreline.sort((a, b) => h[a] - h[b])[0]; const min = feature.shoreline.sort((a, b) => h[a] - h[b])[0];
f.height = h[min] - 0.1; feature.height = h[min] - 0.1;
// check if lake can be open (not in deep depression) // check if lake can be open (not in deep depression)
if (ELEVATION_LIMIT === 80) { if (ELEVATION_LIMIT === 80) {
f.closed = false; feature.closed = false;
return; return;
} }
let deep = true; let deep = true;
const threshold = f.height + ELEVATION_LIMIT; const threshold = feature.height + ELEVATION_LIMIT;
const queue = [min]; const queue = [min];
const checked = []; const checked = [];
checked[min] = true; checked[min] = true;
@ -84,7 +87,7 @@ window.Lakes = (function () {
if (h[n] < 20) { if (h[n] < 20) {
const nFeature = pack.features[cells.f[n]]; const nFeature = pack.features[cells.f[n]];
if (nFeature.type === "ocean" || f.height > nFeature.height) { if ((nFeature && nFeature.type === "ocean") || feature.height > nFeature.height) {
deep = false; deep = false;
break; break;
} }
@ -95,11 +98,11 @@ window.Lakes = (function () {
} }
} }
f.closed = deep; feature.closed = deep;
}); });
}; };
const cleanupLakeData = function () { const cleanupLakeData = function (pack: IPack) {
for (const feature of pack.features) { for (const feature of pack.features) {
if (feature.type !== "lake") continue; if (feature.type !== "lake") continue;
delete feature.river; delete feature.river;
@ -117,14 +120,15 @@ window.Lakes = (function () {
} }
}; };
const defineGroup = function () { const defineGroup = function (pack: IPack) {
for (const feature of pack.features) { for (const feature of pack.features) {
if (feature.type !== "lake") continue; if (feature && feature.type === "lake") {
const lakeEl = lakes.select(`[data-f="${feature.i}"]`).node(); const lakeEl = lakes.select(`[data-f="${feature.i}"]`).node();
if (!lakeEl) continue; if (!lakeEl) continue;
feature.group = getGroup(feature); feature.group = getGroup(feature);
document.getElementById(feature.group).appendChild(lakeEl); byId(feature.group)?.appendChild(lakeEl);
}
} }
}; };

View file

@ -8,7 +8,7 @@ import {rw, each} from "utils/probabilityUtils";
import {aleaPRNG} from "scripts/aleaPRNG"; import {aleaPRNG} from "scripts/aleaPRNG";
window.Rivers = (function () { window.Rivers = (function () {
const generate = function (allowErosion = true) { const generate = function (pack, grid, allowErosion = true) {
TIME && console.time("generateRivers"); TIME && console.time("generateRivers");
Math.random = aleaPRNG(seed); Math.random = aleaPRNG(seed);
const {cells, features} = pack; const {cells, features} = pack;
@ -25,14 +25,14 @@ window.Rivers = (function () {
cells.conf = new Uint8Array(cells.i.length); // confluences array cells.conf = new Uint8Array(cells.i.length); // confluences array
let riverNext = 1; // first river id is 1 let riverNext = 1; // first river id is 1
const h = alterHeights(); const h = alterHeights(pack.cells);
Lakes.prepareLakeData(h); Lakes.prepareLakeData(h, pack);
resolveDepressions(h); resolveDepressions(pack, h);
drainWater(); drainWater();
defineRivers(); defineRivers();
calculateConfluenceFlux(); calculateConfluenceFlux();
Lakes.cleanupLakeData(); Lakes.cleanupLakeData(pack);
if (allowErosion) { if (allowErosion) {
cells.h = Uint8Array.from(h); // apply gradient cells.h = Uint8Array.from(h); // apply gradient
@ -42,14 +42,12 @@ window.Rivers = (function () {
TIME && console.timeEnd("generateRivers"); TIME && console.timeEnd("generateRivers");
function drainWater() { function drainWater() {
//const MIN_FLUX_TO_FORM_RIVER = 10 * distanceScale;
const MIN_FLUX_TO_FORM_RIVER = 30; const MIN_FLUX_TO_FORM_RIVER = 30;
const cellsNumberModifier = (pointsInput.dataset.cells / 10000) ** 0.25; const cellsNumberModifier = (pointsInput.dataset.cells / 10000) ** 0.25;
const prec = grid.cells.prec; const prec = grid.cells.prec;
const area = pack.cells.area;
const land = cells.i.filter(i => h[i] >= 20).sort((a, b) => h[b] - h[a]); const land = cells.i.filter(i => h[i] >= 20).sort((a, b) => h[b] - h[a]);
const lakeOutCells = Lakes.setClimateData(h); const lakeOutCells = Lakes.setClimateData(h, pack, grid);
land.forEach(function (i) { land.forEach(function (i) {
cells.fl[i] += prec[cells.g[i]] / cellsNumberModifier; // add flux from precipitation cells.fl[i] += prec[cells.g[i]] / cellsNumberModifier; // add flux from precipitation
@ -195,7 +193,7 @@ window.Rivers = (function () {
const parent = riverParents[key] || 0; const parent = riverParents[key] || 0;
const widthFactor = !parent || parent === riverId ? mainStemWidthFactor : defaultWidthFactor; const widthFactor = !parent || parent === riverId ? mainStemWidthFactor : defaultWidthFactor;
const meanderedPoints = addMeandering(riverCells); const meanderedPoints = addMeandering(pack, riverCells);
const discharge = cells.fl[mouth]; // m3 in second const discharge = cells.fl[mouth]; // m3 in second
const length = getApproximateLength(meanderedPoints); const length = getApproximateLength(meanderedPoints);
const width = getWidth(getOffset(discharge, meanderedPoints.length, widthFactor, 0)); const width = getWidth(getOffset(discharge, meanderedPoints.length, widthFactor, 0));
@ -245,8 +243,7 @@ window.Rivers = (function () {
}; };
// add distance to water value to land cells to make map less depressed // add distance to water value to land cells to make map less depressed
const alterHeights = () => { const alterHeights = ({h, c, t}) => {
const {h, c, t} = pack.cells;
return Array.from(h).map((h, i) => { return Array.from(h).map((h, i) => {
if (h < 20 || t[i] < 1) return h; if (h < 20 || t[i] < 1) return h;
return h + t[i] / 100 + d3.mean(c[i].map(c => t[c])) / 10000; return h + t[i] / 100 + d3.mean(c[i].map(c => t[c])) / 10000;
@ -254,7 +251,7 @@ window.Rivers = (function () {
}; };
// depression filling algorithm (for a correct water flux modeling) // depression filling algorithm (for a correct water flux modeling)
const resolveDepressions = function (h) { const resolveDepressions = function (pack, h) {
const {cells, features} = pack; const {cells, features} = pack;
const maxIterations = +document.getElementById("resolveDepressionsStepsOutput").value; const maxIterations = +document.getElementById("resolveDepressionsStepsOutput").value;
const checkLakeMaxIteration = maxIterations * 0.85; const checkLakeMaxIteration = maxIterations * 0.85;
@ -272,7 +269,7 @@ window.Rivers = (function () {
for (let iteration = 0; depressions && iteration < maxIterations; iteration++) { for (let iteration = 0; depressions && iteration < maxIterations; iteration++) {
if (progress.length > 5 && d3.sum(progress) > 0) { if (progress.length > 5 && d3.sum(progress) > 0) {
// bad progress, abort and set heights back // bad progress, abort and set heights back
h = alterHeights(); h = alterHeights(pack.cells);
depressions = progress[0]; depressions = progress[0];
break; break;
} }
@ -313,11 +310,11 @@ window.Rivers = (function () {
}; };
// add points at 1/3 and 2/3 of a line between adjacents river cells // add points at 1/3 and 2/3 of a line between adjacents river cells
const addMeandering = function (riverCells, riverPoints = null, meandering = 0.5) { const addMeandering = (pack, riverCells, riverPoints = null, meandering = 0.5) => {
const {fl, conf, h} = pack.cells; const {fl, conf, h} = pack.cells;
const meandered = []; const meandered = [];
const lastStep = riverCells.length - 1; const lastStep = riverCells.length - 1;
const points = getRiverPoints(riverCells, riverPoints); const points = getRiverPoints(pack, riverCells, riverPoints);
let step = h[riverCells[0]] < 20 ? 1 : 10; let step = h[riverCells[0]] < 20 ? 1 : 10;
let fluxPrev = 0; let fluxPrev = 0;
@ -373,17 +370,17 @@ window.Rivers = (function () {
return meandered; return meandered;
}; };
const getRiverPoints = (riverCells, riverPoints) => { const getRiverPoints = (pack, riverCells, riverPoints) => {
if (riverPoints) return riverPoints; if (riverPoints) return riverPoints;
const {p} = pack.cells; const {p} = pack.cells;
return riverCells.map((cell, i) => { return riverCells.map((cell, i) => {
if (cell === -1) return getBorderPoint(riverCells[i - 1]); if (cell === -1) return getBorderPoint(pack, riverCells[i - 1]);
return p[cell]; return p[cell];
}); });
}; };
const getBorderPoint = i => { const getBorderPoint = (pack, i) => {
const [x, y] = pack.cells.p[i]; const [x, y] = pack.cells.p[i];
const min = Math.min(y, graphHeight - y, x, graphWidth - x); const min = Math.min(y, graphHeight - y, x, graphWidth - x);
if (min === y) return [x, 0]; if (min === y) return [x, 0];

View file

@ -7,6 +7,7 @@ import {rn} from "utils/numberUtils";
import {aleaPRNG} from "scripts/aleaPRNG"; import {aleaPRNG} from "scripts/aleaPRNG";
import {renderLayer} from "layers"; import {renderLayer} from "layers";
import {markupGridFeatures} from "modules/markup"; import {markupGridFeatures} from "modules/markup";
import {generateGrid} from "scripts/generation/graph";
window.Submap = (function () { window.Submap = (function () {
const isWater = (pack, id) => pack.cells.h[id] < 20; const isWater = (pack, id) => pack.cells.h[id] < 20;
@ -130,8 +131,8 @@ window.Submap = (function () {
// remove misclassified cells // remove misclassified cells
stage("Define coastline."); stage("Define coastline.");
reMarkFeatures(); reMarkFeatures(pack, newGrid);
drawCoastline(); drawCoastline(pack);
/****************************************************/ /****************************************************/
/* Packed Graph */ /* Packed Graph */
@ -210,8 +211,8 @@ window.Submap = (function () {
} }
stage("Regenerating river network."); stage("Regenerating river network.");
Rivers.generate(); Rivers.generate(pack, grid);
renderLayer("rivers"); renderLayer("rivers", pack);
Lakes.defineGroup(); Lakes.defineGroup();
// biome calculation based on (resampled) grid.cells.temp and prec // biome calculation based on (resampled) grid.cells.temp and prec

View file

@ -168,6 +168,8 @@ optionsContent.on("click", function (event) {
else if (id === "translateExtent") toggleTranslateExtent(event.target); else if (id === "translateExtent") toggleTranslateExtent(event.target);
else if (id === "speakerTest") testSpeaker(); else if (id === "speakerTest") testSpeaker();
else if (id === "themeColorRestore") restoreDefaultThemeColor(); else if (id === "themeColorRestore") restoreDefaultThemeColor();
else if (id === "configureWorld") openDialog("worldConfigurator");
else if (id === "optionsReset") restoreDefaultOptions();
}); });
function mapSizeInputChange() { function mapSizeInputChange() {
@ -1025,6 +1027,7 @@ export function toggle3dOptions() {
isLoaded = true; isLoaded = true;
byId("options3dUpdate").on("click", ThreeD.update); byId("options3dUpdate").on("click", ThreeD.update);
byId("options3dConfigureWorld").on("click", () => openDialog("worldConfigurator"));
byId("options3dSave").on("click", ThreeD.saveScreenshot); byId("options3dSave").on("click", ThreeD.saveScreenshot);
byId("options3dOBJSave").on("click", ThreeD.saveOBJ); byId("options3dOBJSave").on("click", ThreeD.saveOBJ);

View file

@ -102,7 +102,7 @@ export function createRiver() {
const defaultWidthFactor = rn(1 / (pointsInput.dataset.cells / 10000) ** 0.25, 2); const defaultWidthFactor = rn(1 / (pointsInput.dataset.cells / 10000) ** 0.25, 2);
const widthFactor = 1.2 * defaultWidthFactor; const widthFactor = 1.2 * defaultWidthFactor;
const meanderedPoints = addMeandering(riverCells); const meanderedPoints = addMeandering(pack, riverCells);
const discharge = cells.fl[mouth]; // m3 in second const discharge = cells.fl[mouth]; // m3 in second
const length = getApproximateLength(meanderedPoints); const length = getApproximateLength(meanderedPoints);

View file

@ -31,7 +31,7 @@ export function editRiver(id) {
const river = getRiver(); const river = getRiver();
const {cells, points} = river; const {cells, points} = river;
const riverPoints = Rivers.getRiverPoints(cells, points); const riverPoints = Rivers.getRiverPoints(pack, cells, points);
drawControlPoints(riverPoints); drawControlPoints(riverPoints);
drawCells(cells); drawCells(cells);
@ -98,7 +98,7 @@ export function editRiver(id) {
function updateRiverWidth(river) { function updateRiverWidth(river) {
const {addMeandering, getWidth, getOffset} = Rivers; const {addMeandering, getWidth, getOffset} = Rivers;
const {cells, discharge, widthFactor, sourceWidth} = river; const {cells, discharge, widthFactor, sourceWidth} = river;
const meanderedPoints = addMeandering(cells); const meanderedPoints = addMeandering(pack, cells);
river.width = getWidth(getOffset(discharge, meanderedPoints.length, widthFactor, sourceWidth)); river.width = getWidth(getOffset(discharge, meanderedPoints.length, widthFactor, sourceWidth));
const width = `${rn(river.width * distanceScaleInput.value, 3)} ${distanceUnitInput.value}`; const width = `${rn(river.width * distanceScaleInput.value, 3)} ${distanceUnitInput.value}`;
@ -169,7 +169,7 @@ export function editRiver(id) {
river.cells = river.points.map(([x, y]) => findCell(x, y)); river.cells = river.points.map(([x, y]) => findCell(x, y));
const {widthFactor, sourceWidth} = river; const {widthFactor, sourceWidth} = river;
const meanderedPoints = Rivers.addMeandering(river.cells, river.points); const meanderedPoints = Rivers.addMeandering(pack, river.cells, river.points);
const path = Rivers.getRiverPath(meanderedPoints, widthFactor, sourceWidth); const path = Rivers.getRiverPath(meanderedPoints, widthFactor, sourceWidth);
elSelected.attr("d", path); elSelected.attr("d", path);

View file

@ -6,6 +6,7 @@ import {debounce} from "utils/functionUtils";
import {restoreLayers} from "layers"; import {restoreLayers} from "layers";
import {undraw} from "scripts/generation/generation"; import {undraw} from "scripts/generation/generation";
import {closeDialogs} from "dialogs/utils"; import {closeDialogs} from "dialogs/utils";
import {openDialog} from "dialogs";
window.UISubmap = (function () { window.UISubmap = (function () {
byId("submapPointsInput").addEventListener("input", function () { byId("submapPointsInput").addEventListener("input", function () {
@ -312,7 +313,7 @@ window.UISubmap = (function () {
restoreLayers(); restoreLayers();
if (ThreeD.options.isOn) ThreeD.redraw(); if (ThreeD.options.isOn) ThreeD.redraw();
if ($("#worldConfigurator").is(":visible")) editWorld(); if ($("#worldConfigurator").is(":visible")) openDialog("worldConfigurator");
} }
function changeStyles(scale) { function changeStyles(scale) {

View file

@ -131,11 +131,11 @@ async function openEmblemEditor() {
} }
function regenerateRivers() { function regenerateRivers() {
Rivers.generate(); Rivers.generate(pack, grid);
Lakes.defineGroup(); Lakes.defineGroup();
Rivers.specify(); Rivers.specify();
if (!layerIsOn("toggleRivers")) toggleLayer("toggleRivers"); if (!layerIsOn("toggleRivers")) toggleLayer("toggleRivers");
else renderLayer("rivers"); else renderLayer("rivers", pack);
} }
function recalculatePopulation() { function recalculatePopulation() {
@ -588,8 +588,8 @@ function addRiverOnClick() {
const initialFlux = grid.cells.prec[cells.g[i]]; const initialFlux = grid.cells.prec[cells.g[i]];
cells.fl[i] = initialFlux; cells.fl[i] = initialFlux;
const h = alterHeights(); const h = alterHeights(pacl.cells);
resolveDepressions(h); resolveDepressions(pack, h);
while (i) { while (i) {
cells.r[i] = riverId; cells.r[i] = riverId;
@ -663,7 +663,7 @@ function addRiverOnClick() {
const defaultWidthFactor = rn(1 / (pointsInput.dataset.cells / 10000) ** 0.25, 2); const defaultWidthFactor = rn(1 / (pointsInput.dataset.cells / 10000) ** 0.25, 2);
const widthFactor = const widthFactor =
river?.widthFactor || (!parent || parent === riverId ? defaultWidthFactor * 1.2 : defaultWidthFactor); river?.widthFactor || (!parent || parent === riverId ? defaultWidthFactor * 1.2 : defaultWidthFactor);
const meanderedPoints = addMeandering(riverCells); const meanderedPoints = addMeandering(pack, riverCells);
const discharge = cells.fl[mouth]; // m3 in second const discharge = cells.fl[mouth]; // m3 in second
const length = getApproximateLength(meanderedPoints); const length = getApproximateLength(meanderedPoints);
@ -704,7 +704,7 @@ function addRiverOnClick() {
riversG.append("path").attr("id", id).attr("d", path); riversG.append("path").attr("id", id).attr("d", path);
if (d3.event.shiftKey === false) { if (d3.event.shiftKey === false) {
Lakes.cleanupLakeData(); Lakes.cleanupLakeData(pack);
unpressClickToAddButton(); unpressClickToAddButton();
document.getElementById("addNewRiver").classList.remove("pressed"); document.getElementById("addNewRiver").classList.remove("pressed");
if (addNewRiver.offsetParent) riversOverviewRefresh.click(); if (addNewRiver.offsetParent) riversOverviewRefresh.click();

View file

@ -7,8 +7,7 @@ import {renderLayer} from "layers";
let isLoaded = false; let isLoaded = false;
export function editWorld() { export function open() {
if (customization) return;
$("#worldConfigurator").dialog({ $("#worldConfigurator").dialog({
title: "Configure World", title: "Configure World",
resizable: false, resizable: false,
@ -63,7 +62,7 @@ export function editWorld() {
calculateTemperatures(grid); calculateTemperatures(grid);
generatePrecipitation(grid); generatePrecipitation(grid);
const heights = new Uint8Array(pack.cells.h); const heights = new Uint8Array(pack.cells.h);
Rivers.generate(); Rivers.generate(pack, grid);
Lakes.defineGroup(); Lakes.defineGroup();
Rivers.specify(); Rivers.specify();
pack.cells.h = new Float32Array(heights); pack.cells.h = new Float32Array(heights);
@ -73,7 +72,7 @@ export function editWorld() {
if (layerIsOn("togglePrec")) renderLayer("precipitation"); if (layerIsOn("togglePrec")) renderLayer("precipitation");
if (layerIsOn("toggleBiomes")) renderLayer("biomes"); if (layerIsOn("toggleBiomes")) renderLayer("biomes");
if (layerIsOn("toggleCoordinates")) drawCoordinates(); if (layerIsOn("toggleCoordinates")) drawCoordinates();
if (layerIsOn("toggleRivers")) renderLayer("rivers"); if (layerIsOn("toggleRivers")) renderLayer("rivers", pack);
if (document.getElementById("canvas3d")) setTimeout(ThreeD.update(), 500); if (document.getElementById("canvas3d")) setTimeout(ThreeD.update(), 500);
} }

84
src/scripts/findAll.js Normal file
View file

@ -0,0 +1,84 @@
import * as d3 from "d3";
// findAll d3.quandtree search from https://bl.ocks.org/lwthatcher/b41479725e0ff2277c7ac90df2de2b5e
export function addFindAll() {
const Quad = function (node, x0, y0, x1, y1) {
this.node = node;
this.x0 = x0;
this.y0 = y0;
this.x1 = x1;
this.y1 = y1;
};
const tree_filter = function (x, y, radius) {
var t = {x, y, x0: this._x0, y0: this._y0, x3: this._x1, y3: this._y1, quads: [], node: this._root};
if (t.node) {
t.quads.push(new Quad(t.node, t.x0, t.y0, t.x3, t.y3));
}
radiusSearchInit(t, radius);
var i = 0;
while ((t.q = t.quads.pop())) {
i++;
// Stop searching if this quadrant cant contain a closer node.
if (
!(t.node = t.q.node) ||
(t.x1 = t.q.x0) > t.x3 ||
(t.y1 = t.q.y0) > t.y3 ||
(t.x2 = t.q.x1) < t.x0 ||
(t.y2 = t.q.y1) < t.y0
)
continue;
// Bisect the current quadrant.
if (t.node.length) {
t.node.explored = true;
var xm = (t.x1 + t.x2) / 2,
ym = (t.y1 + t.y2) / 2;
t.quads.push(
new Quad(t.node[3], xm, ym, t.x2, t.y2),
new Quad(t.node[2], t.x1, ym, xm, t.y2),
new Quad(t.node[1], xm, t.y1, t.x2, ym),
new Quad(t.node[0], t.x1, t.y1, xm, ym)
);
// Visit the closest quadrant first.
if ((t.i = ((y >= ym) << 1) | (x >= xm))) {
t.q = t.quads[t.quads.length - 1];
t.quads[t.quads.length - 1] = t.quads[t.quads.length - 1 - t.i];
t.quads[t.quads.length - 1 - t.i] = t.q;
}
}
// Visit this point. (Visiting coincident points isnt necessary!)
else {
var dx = x - +this._x.call(null, t.node.data),
dy = y - +this._y.call(null, t.node.data),
d2 = dx * dx + dy * dy;
radiusSearchVisit(t, d2);
}
}
return t.result;
};
d3.quadtree.prototype.findAll = tree_filter;
var radiusSearchInit = function (t, radius) {
t.result = [];
(t.x0 = t.x - radius), (t.y0 = t.y - radius);
(t.x3 = t.x + radius), (t.y3 = t.y + radius);
t.radius = radius * radius;
};
var radiusSearchVisit = function (t, d2) {
t.node.data.scanned = true;
if (d2 < t.radius) {
do {
t.result.push(t.node.data);
t.node.data.selected = true;
} while ((t.node = t.node.next));
}
};
}

View file

@ -2,6 +2,7 @@ import * as d3 from "d3";
import {ERROR, INFO, WARN} from "config/logging"; import {ERROR, INFO, WARN} from "config/logging";
import {closeDialogs} from "dialogs/utils"; import {closeDialogs} from "dialogs/utils";
import {openDialog} from "dialogs";
import {initLayers, renderLayer, restoreLayers} from "layers"; import {initLayers, renderLayer, restoreLayers} from "layers";
// @ts-expect-error js module // @ts-expect-error js module
import {drawCoastline} from "modules/coastline"; import {drawCoastline} from "modules/coastline";
@ -29,11 +30,29 @@ import {rankCells} from "../rankCells";
import {showStatistics} from "../statistics"; import {showStatistics} from "../statistics";
import {createGrid} from "./grid"; import {createGrid} from "./grid";
import {reGraph} from "./reGraph"; import {reGraph} from "./reGraph";
import {getInputValue, setInputValue} from "utils/nodeUtils";
const {Zoom, Lakes, OceanLayers, Rivers, Biomes, Cultures, BurgsAndStates, Religions, Military, Markers, Names} = const {
window; Zoom,
Lakes,
OceanLayers,
Rivers,
Biomes,
Cultures,
BurgsAndStates,
Religions,
Military,
Markers,
Names,
ThreeD
} = window;
async function generate(options?: {seed: string; graph: IGrid}) { interface IGenerationOptions {
seed: string;
graph: IGrid;
}
async function generate(options?: IGenerationOptions) {
try { try {
const timeStart = performance.now(); const timeStart = performance.now();
const {seed: precreatedSeed, graph: precreatedGraph} = options || {}; const {seed: precreatedSeed, graph: precreatedGraph} = options || {};
@ -47,17 +66,17 @@ async function generate(options?: {seed: string; graph: IGrid}) {
randomizeOptions(); randomizeOptions();
const newGrid = await createGrid(grid, precreatedGraph); const newGrid = await createGrid(grid, precreatedGraph);
const newPack = reGraph(newGrid);
const pack = reGraph(newGrid); reMarkFeatures(newPack, newGrid);
reMarkFeatures(pack, newGrid); drawCoastline(newPack);
drawCoastline();
Rivers.generate(); Rivers.generate(newPack, newGrid);
renderLayer("rivers"); renderLayer("rivers", newPack);
Lakes.defineGroup(); Lakes.defineGroup(newPack);
Biomes.define(); Biomes.define(newPack, newGrid);
rankCells(); rankCells(newPack);
Cultures.generate(); Cultures.generate();
Cultures.expand(); Cultures.expand();
BurgsAndStates.generate(); BurgsAndStates.generate();
@ -82,6 +101,9 @@ async function generate(options?: {seed: string; graph: IGrid}) {
drawScaleBar(window.scale); drawScaleBar(window.scale);
Names.getMapName(); Names.getMapName();
// @ts-expect-error redefine global
pack = newPack;
WARN && console.warn(`TOTAL: ${rn((performance.now() - timeStart) / 1000, 2)}s`); WARN && console.warn(`TOTAL: ${rn((performance.now() - timeStart) / 1000, 2)}s`);
showStatistics(); showStatistics();
INFO && console.groupEnd(); INFO && console.groupEnd();
@ -142,10 +164,10 @@ export function undraw() {
unfog(); unfog();
} }
export const regenerateMap = debounce(async function (options) { export const regenerateMap = debounce(async function (options: IGenerationOptions) {
WARN && console.warn("Generate new random map"); WARN && console.warn("Generate new random map");
const cellsDesired = +byId("pointsInput").dataset.cells; const cellsDesired = Number(byId("pointsInput")?.dataset.cells);
const shouldShowLoading = cellsDesired > 10000; const shouldShowLoading = cellsDesired > 10000;
shouldShowLoading && showLoading(); shouldShowLoading && showLoading();
@ -156,7 +178,7 @@ export const regenerateMap = debounce(async function (options) {
await generate(options); await generate(options);
restoreLayers(); restoreLayers();
if (ThreeD.options.isOn) ThreeD.redraw(); if (ThreeD.options.isOn) ThreeD.redraw();
if ($("#worldConfigurator").is(":visible")) editWorld(); if ($("#worldConfigurator").is(":visible")) openDialog("worldConfigurator");
shouldShowLoading && hideLoading(); shouldShowLoading && hideLoading();
clearMainTip(); clearMainTip();
@ -164,14 +186,13 @@ export const regenerateMap = debounce(async function (options) {
// focus on coordinates, cell or burg provided in searchParams // focus on coordinates, cell or burg provided in searchParams
function focusOn() { function focusOn() {
const url = new URL(window.location.href); const params = new URL(window.location.href).searchParams;
const params = url.searchParams;
const fromMGCG = params.get("from") === "MFCG" && document.referrer; const fromMGCG = params.get("from") === "MFCG" && document.referrer;
if (fromMGCG) { if (fromMGCG) {
if (params.get("seed").length === 13) { if (params.get("seed")?.length === 13) {
// show back burg from MFCG // show back burg from MFCG
const burgSeed = params.get("seed").slice(-4); const burgSeed = params.get("seed")!.slice(-4);
params.set("burg", burgSeed); params.set("burg", burgSeed);
} else { } else {
// select burg for MFCG // select burg for MFCG
@ -185,10 +206,10 @@ function focusOn() {
const burgParam = params.get("burg"); const burgParam = params.get("burg");
if (scaleParam || cellParam || burgParam) { if (scaleParam || cellParam || burgParam) {
const scale = +scaleParam || 8; const scale = scaleParam ? Number(scaleParam) : 8;
if (cellParam) { if (cellParam) {
const cell = +params.get("cell"); const cell = Number(scaleParam);
const [x, y] = pack.cells.p[cell]; const [x, y] = pack.cells.p[cell];
Zoom.to(x, y, scale, 1600); Zoom.to(x, y, scale, 1600);
return; return;
@ -203,14 +224,14 @@ function focusOn() {
return; return;
} }
const x = +params.get("x") || graphWidth / 2; const x = params.get("x") ? Number(params.get("x")) : graphWidth / 2;
const y = +params.get("y") || graphHeight / 2; const y = params.get("y") ? Number(params.get("y")) : graphHeight / 2;
Zoom.to(x, y, scale, 1600); Zoom.to(x, y, scale, 1600);
} }
} }
// find burg for MFCG and focus on it // find burg for MFCG and focus on it
function findBurgForMFCG(params) { function findBurgForMFCG(params: URLSearchParams) {
const {cells, burgs} = pack; const {cells, burgs} = pack;
if (pack.burgs.length < 2) { if (pack.burgs.length < 2) {
@ -219,17 +240,17 @@ function findBurgForMFCG(params) {
} }
// used for selection // used for selection
const size = +params.get("size"); const size = params.get("size") ? Number(params.get("size")) : 10;
const coast = +params.get("coast"); const coast = Boolean(params.get("coast"));
const port = +params.get("port"); const port = Boolean(params.get("port"));
const river = +params.get("river"); const river = Boolean(params.get("river"));
let selection = defineSelection(coast, port, river); let selection = defineSelection(coast, port, river);
if (!selection.length) selection = defineSelection(coast, !port, !river); if (!selection.length) selection = defineSelection(coast, !port, !river);
if (!selection.length) selection = defineSelection(!coast, 0, !river); if (!selection.length) selection = defineSelection(!coast, false, !river);
if (!selection.length) selection = [burgs[1]]; // select first if nothing is found if (!selection.length) selection = [burgs[1]]; // select first if nothing is found
function defineSelection(coast, port, river) { function defineSelection(coast: boolean, port: boolean, river: boolean) {
if (port && river) return burgs.filter(b => b.port && cells.r[b.cell]); if (port && river) return burgs.filter(b => b.port && cells.r[b.cell]);
if (!port && coast && river) return burgs.filter(b => !b.port && cells.t[b.cell] === 1 && cells.r[b.cell]); if (!port && coast && river) return burgs.filter(b => !b.port && cells.t[b.cell] === 1 && cells.r[b.cell]);
if (!coast && !river) return burgs.filter(b => cells.t[b.cell] !== 1 && !cells.r[b.cell]); if (!coast && !river) return burgs.filter(b => cells.t[b.cell] !== 1 && !cells.r[b.cell]);
@ -240,29 +261,27 @@ function findBurgForMFCG(params) {
// select a burg with closest population from selection // select a burg with closest population from selection
const selected = d3.scan(selection, (a, b) => Math.abs(a.population - size) - Math.abs(b.population - size)); const selected = d3.scan(selection, (a, b) => Math.abs(a.population - size) - Math.abs(b.population - size));
const burgId = selection[selected].i; const burgId = selected && selection[selected].i;
if (!burgId) { if (!burgId) return ERROR && console.error("Cannot select a burg for MFCG");
ERROR && console.error("Cannot select a burg for MFCG");
return;
}
const b = burgs[burgId]; const b = burgs[burgId];
const referrer = new URL(document.referrer); const searchParams = new URL(document.referrer).searchParams;
for (let p of referrer.searchParams) { for (let [param, value] of searchParams) {
if (p[0] === "name") b.name = p[1]; if (param === "name") b.name = value;
else if (p[0] === "size") b.population = +p[1]; else if (param === "size") b.population = +value;
else if (p[0] === "seed") b.MFCG = +p[1]; else if (param === "seed") b.MFCG = +value;
else if (p[0] === "shantytown") b.shanty = +p[1]; else if (param === "shantytown") b.shanty = +value;
else b[p[0]] = +p[1]; // other parameters
} }
if (params.get("name") && params.get("name") != "null") b.name = params.get("name");
const nameParam = params.get("name");
if (nameParam && nameParam !== "null") b.name = nameParam;
const label = burgLabels.select("[data-id='" + burgId + "']"); const label = burgLabels.select("[data-id='" + burgId + "']");
if (label.size()) { if (label.size()) {
label label
.text(b.name) .text(b.name)
.classed("drag", true) .classed("drag", true)
.on("mouseover", function () { .on("mouseover", function (this: Element) {
d3.select(this).classed("drag", false); d3.select(this).classed("drag", false);
label.on("mouseover", null); label.on("mouseover", null);
}); });
@ -270,24 +289,27 @@ function findBurgForMFCG(params) {
Zoom.to(b.x, b.y, 8, 1600); Zoom.to(b.x, b.y, 8, 1600);
Zoom.invoke(); Zoom.invoke();
tip("Here stands the glorious city of " + b.name, true, "success", 15000); tip("Here stands the glorious city of " + b.name, true, "success", 15000);
} }
// set map seed (string!) // set map seed (string!)
function setSeed(precreatedSeed) { function setSeed(precreatedSeed?: string) {
if (!precreatedSeed) { if (!precreatedSeed) {
const first = !mapHistory[0]; const first = !mapHistory[0];
const url = new URL(window.location.href);
const params = url.searchParams; const params = new URL(window.location.href).searchParams;
const urlSeed = url.searchParams.get("seed"); const urlSeed = params.get("seed");
if (first && params.get("from") === "MFCG" && urlSeed.length === 13) seed = urlSeed.slice(0, -4); const optionsSeed = getInputValue("optionsSeed");
if (first && params.get("from") === "MFCG" && urlSeed?.length === 13) seed = urlSeed.slice(0, -4);
else if (first && urlSeed) seed = urlSeed; else if (first && urlSeed) seed = urlSeed;
else if (optionsSeed.value && optionsSeed.value != seed) seed = optionsSeed.value; else if (optionsSeed && optionsSeed !== seed) seed = optionsSeed;
else seed = generateSeed(); else seed = generateSeed();
} else { } else {
seed = precreatedSeed; seed = precreatedSeed;
} }
byId("optionsSeed").value = seed; setInputValue("optionsSeed", seed);
Math.random = aleaPRNG(seed); Math.random = aleaPRNG(seed);
} }

View file

@ -0,0 +1,87 @@
import Delaunator from "delaunator";
import {Voronoi} from "modules/voronoi";
import {TIME} from "config/logging";
// @ts-expect-error js module
import {aleaPRNG} from "scripts/aleaPRNG";
import {createTypedArray} from "utils/arrayUtils";
import {rn} from "utils/numberUtils";
import {byId} from "utils/shorthands";
export function generateGrid() {
Math.random = aleaPRNG(seed); // reset PRNG
const {spacing, cellsDesired, boundary, points, cellsX, cellsY} = placePoints();
const {cells, vertices} = calculateVoronoi(points, boundary);
return {spacing, cellsDesired, boundary, points, cellsX, cellsY, cells, vertices};
}
// place random points to calculate Voronoi diagram
function placePoints() {
TIME && console.time("placePoints");
const cellsDesired = Number(byId("pointsInput")?.dataset.cells);
const spacing = rn(Math.sqrt((graphWidth * graphHeight) / cellsDesired), 2); // spacing between points before jirrering
const boundary = getBoundaryPoints(graphWidth, graphHeight, spacing);
const points = getJitteredGrid(graphWidth, graphHeight, spacing); // points of jittered square grid
const cellsX = Math.floor((graphWidth + 0.5 * spacing - 1e-10) / spacing);
const cellsY = Math.floor((graphHeight + 0.5 * spacing - 1e-10) / spacing);
TIME && console.timeEnd("placePoints");
return {spacing, cellsDesired, boundary, points, cellsX, cellsY};
}
// calculate Delaunay and then Voronoi diagram
export function calculateVoronoi(points: TPoints, boundary: TPoints): IGraph {
TIME && console.time("calculateDelaunay");
const allPoints: TPoints = points.concat(boundary);
const delaunay = Delaunator.from(allPoints);
TIME && console.timeEnd("calculateDelaunay");
TIME && console.time("calculateVoronoi");
const {cells, vertices} = new Voronoi(delaunay, allPoints, points.length);
const i = createTypedArray({maxValue: points.length, length: points.length}).map((_, i) => i); // array of indexes
TIME && console.timeEnd("calculateVoronoi");
return {cells: {...cells, i}, vertices};
}
// add points along map edge to pseudo-clip voronoi cells
function getBoundaryPoints(width: number, height: number, spacing: number) {
const offset = rn(-1 * spacing);
const bSpacing = spacing * 2;
const w = width - offset * 2;
const h = height - offset * 2;
const numberX = Math.ceil(w / bSpacing) - 1;
const numberY = Math.ceil(h / bSpacing) - 1;
const points: TPoints = [];
for (let i = 0.5; i < numberX; i++) {
let x = Math.ceil((w * i) / numberX + offset);
points.push([x, offset], [x, h + offset]);
}
for (let i = 0.5; i < numberY; i++) {
let y = Math.ceil((h * i) / numberY + offset);
points.push([offset, y], [w + offset, y]);
}
return points;
}
// get points on a regular square grid and jitter them a bit
function getJitteredGrid(width: number, height: number, spacing: number) {
const radius = spacing / 2; // square radius
const jittering = radius * 0.9; // max deviation
const doubleJittering = jittering * 2;
const jitter = () => Math.random() * doubleJittering - jittering;
const points: TPoints = [];
for (let y = radius; y < height; y += spacing) {
for (let x = radius; x < width; x += spacing) {
const xj = Math.min(rn(x + jitter(), 2), width);
const yj = Math.min(rn(y + jitter(), 2), height);
points.push([xj, yj]);
}
}
return points;
}

View file

@ -1,5 +1,5 @@
import {calculateTemperatures} from "modules/temperature"; import {calculateTemperatures} from "modules/temperature";
import {generateGrid} from "utils/graphUtils"; import {generateGrid} from "scripts/generation/graph";
import {calculateMapCoordinates, defineMapSize} from "modules/coordinates"; import {calculateMapCoordinates, defineMapSize} from "modules/coordinates";
import {markupGridFeatures} from "modules/markup"; import {markupGridFeatures} from "modules/markup";
// @ts-expect-error js module // @ts-expect-error js module

View file

@ -3,7 +3,7 @@ import * as d3 from "d3";
import {TIME} from "config/logging"; import {TIME} from "config/logging";
import {UINT16_MAX} from "constants"; import {UINT16_MAX} from "constants";
import {createTypedArray} from "utils/arrayUtils"; import {createTypedArray} from "utils/arrayUtils";
import {calculateVoronoi, getPackPolygon} from "utils/graphUtils"; import {calculateVoronoi} from "scripts/generation/graph";
import {rn} from "utils/numberUtils"; import {rn} from "utils/numberUtils";
import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation"; import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation";
@ -51,20 +51,21 @@ export function reGraph(grid: IGrid): IPackBase {
newCells.h.push(height); newCells.h.push(height);
} }
const {cells, vertices} = calculateVoronoi(newCells.p, grid.boundary);
function getCellArea(i: number) { function getCellArea(i: number) {
const area = Math.abs(d3.polygonArea(getPackPolygon(i))); const polygon = cells.v[i].map(v => vertices.p[v]);
const area = Math.abs(d3.polygonArea(polygon));
return Math.min(area, UINT16_MAX); return Math.min(area, UINT16_MAX);
} }
const {cells, vertices} = calculateVoronoi(newCells.p, grid.boundary);
const pack: IPackBase = { const pack: IPackBase = {
vertices, vertices,
cells: { cells: {
...cells, ...cells,
p: newCells.p, p: newCells.p,
g: createTypedArray({maxValue: grid.points.length, from: newCells.g}), g: createTypedArray({maxValue: grid.points.length, from: newCells.g}),
q: d3.quadtree(newCells.p.map(([x, y], i) => [x, y, i])), q: d3.quadtree(newCells.p.map(([x, y], i) => [x, y, i])) as unknown as Quadtree,
h: new Uint8Array(newCells.h), h: new Uint8Array(newCells.h),
area: createTypedArray({maxValue: UINT16_MAX, from: cells.i}).map(getCellArea) area: createTypedArray({maxValue: UINT16_MAX, from: cells.i}).map(getCellArea)
} }

View file

@ -8,6 +8,8 @@ import {addResizeListener} from "modules/ui/options";
// @ts-ignore // @ts-ignore
import {addDragToUpload} from "modules/io/load"; import {addDragToUpload} from "modules/io/load";
import {addHotkeyListeners} from "scripts/hotkeys"; import {addHotkeyListeners} from "scripts/hotkeys";
// @ts-ignore
import {addFindAll} from "scripts/findAll";
export function addGlobalListeners() { export function addGlobalListeners() {
if (PRODUCTION) { if (PRODUCTION) {
@ -22,6 +24,7 @@ export function addGlobalListeners() {
addHotkeyListeners(); addHotkeyListeners();
assignSpeakerBehavior(); assignSpeakerBehavior();
addDragToUpload(); addDragToUpload();
addFindAll();
} }
function registerServiceWorker() { function registerServiceWorker() {

View file

@ -8,7 +8,7 @@ const FLUX_MAX_BONUS = 250;
const SUITABILITY_FACTOR = 5; const SUITABILITY_FACTOR = 5;
// assess cells suitability for population and rank cells for culture centers and burgs placement // assess cells suitability for population and rank cells for culture centers and burgs placement
export function rankCells() { export function rankCells(pack: IPack) {
TIME && console.time("rankCells"); TIME && console.time("rankCells");
const {cells, features} = pack; const {cells, features} = pack;

View file

@ -33,3 +33,8 @@ interface Node {
on: (name: string, fn: EventListenerOrEventListenerObject, options?: AddEventListenerOptions) => void; on: (name: string, fn: EventListenerOrEventListenerObject, options?: AddEventListenerOptions) => void;
off: (name: string, fn: EventListenerOrEventListenerObject) => void; off: (name: string, fn: EventListenerOrEventListenerObject) => void;
} }
interface Quadtree extends d3.Quadtree<Number> {
find: (x: number, y: number, radius: number) => [x: number, y: number, cellId: number];
findAll: (x: number, y: number, radius: number) => [x: number, y: number, cellId: number][];
}

6
src/types/pack.d.ts vendored
View file

@ -29,7 +29,7 @@ interface IPackCells {
burg: UintArray; burg: UintArray;
haven: UintArray; haven: UintArray;
harbor: UintArray; harbor: UintArray;
q: d3.Quadtree<number[]>; q: Quadtree;
} }
interface IPackBase extends IGraph { interface IPackBase extends IGraph {
@ -95,6 +95,9 @@ interface IBurg {
x: number; x: number;
y: number; y: number;
population: number; population: number;
port: number;
shanty: number;
MFCG?: string | number;
removed?: boolean; removed?: boolean;
} }
@ -119,4 +122,5 @@ interface IRiver {
length: number; length: number;
discharge: number; discharge: number;
cells: number[]; cells: number[];
points?: number[];
} }

View file

@ -1,92 +1,6 @@
import * as d3 from "d3";
import Delaunator from "delaunator";
import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation"; import {DISTANCE_FIELD, MIN_LAND_HEIGHT} from "config/generation";
import {Voronoi} from "modules/voronoi";
// @ts-expect-error js module // @ts-expect-error js module
import {aleaPRNG} from "scripts/aleaPRNG"; import {aleaPRNG} from "scripts/aleaPRNG";
import {TIME} from "../config/logging";
import {createTypedArray} from "./arrayUtils";
import {rn} from "./numberUtils";
import {byId} from "./shorthands";
export function generateGrid() {
Math.random = aleaPRNG(seed); // reset PRNG
const {spacing, cellsDesired, boundary, points, cellsX, cellsY} = placePoints();
const {cells, vertices} = calculateVoronoi(points, boundary);
return {spacing, cellsDesired, boundary, points, cellsX, cellsY, cells, vertices};
}
// place random points to calculate Voronoi diagram
function placePoints() {
TIME && console.time("placePoints");
const cellsDesired = Number(byId("pointsInput")?.dataset.cells);
const spacing = rn(Math.sqrt((graphWidth * graphHeight) / cellsDesired), 2); // spacing between points before jirrering
const boundary = getBoundaryPoints(graphWidth, graphHeight, spacing);
const points = getJitteredGrid(graphWidth, graphHeight, spacing); // points of jittered square grid
const cellsX = Math.floor((graphWidth + 0.5 * spacing - 1e-10) / spacing);
const cellsY = Math.floor((graphHeight + 0.5 * spacing - 1e-10) / spacing);
TIME && console.timeEnd("placePoints");
return {spacing, cellsDesired, boundary, points, cellsX, cellsY};
}
// calculate Delaunay and then Voronoi diagram
export function calculateVoronoi(points: TPoints, boundary: TPoints): IGraph {
TIME && console.time("calculateDelaunay");
const allPoints: TPoints = points.concat(boundary);
const delaunay = Delaunator.from(allPoints);
TIME && console.timeEnd("calculateDelaunay");
TIME && console.time("calculateVoronoi");
const {cells, vertices} = new Voronoi(delaunay, allPoints, points.length);
const i = createTypedArray({maxValue: points.length, length: points.length}).map((_, i) => i); // array of indexes
TIME && console.timeEnd("calculateVoronoi");
return {cells: {...cells, i}, vertices};
}
// add points along map edge to pseudo-clip voronoi cells
function getBoundaryPoints(width: number, height: number, spacing: number) {
const offset = rn(-1 * spacing);
const bSpacing = spacing * 2;
const w = width - offset * 2;
const h = height - offset * 2;
const numberX = Math.ceil(w / bSpacing) - 1;
const numberY = Math.ceil(h / bSpacing) - 1;
const points: TPoints = [];
for (let i = 0.5; i < numberX; i++) {
let x = Math.ceil((w * i) / numberX + offset);
points.push([x, offset], [x, h + offset]);
}
for (let i = 0.5; i < numberY; i++) {
let y = Math.ceil((h * i) / numberY + offset);
points.push([offset, y], [w + offset, y]);
}
return points;
}
// get points on a regular square grid and jitter them a bit
function getJitteredGrid(width: number, height: number, spacing: number) {
const radius = spacing / 2; // square radius
const jittering = radius * 0.9; // max deviation
const doubleJittering = jittering * 2;
const jitter = () => Math.random() * doubleJittering - jittering;
const points: TPoints = [];
for (let y = radius; y < height; y += spacing) {
for (let x = radius; x < width; x += spacing) {
const xj = Math.min(rn(x + jitter(), 2), width);
const yj = Math.min(rn(y + jitter(), 2), height);
points.push([xj, yj]);
}
}
return points;
}
// return cell index on a regular square grid // return cell index on a regular square grid
export function findGridCell(x: number, y: number, grid: IGrid) { export function findGridCell(x: number, y: number, grid: IGrid) {
@ -125,7 +39,7 @@ export function findGridAll(x: number, y: number, radius: number) {
// return array of cell indexes in radius // return array of cell indexes in radius
export function findAll(x: number, y: number, radius: number) { export function findAll(x: number, y: number, radius: number) {
const found = pack.cells.q.findAll(x, y, radius); const found = pack.cells.q.findAll(x, y, radius);
return found.map(r => r[2]); return found.map(data => data[2]);
} }
// get polygon points for packed cells knowing cell id // get polygon points for packed cells knowing cell id
@ -157,85 +71,3 @@ export function isWater(cellId: number) {
export function isCoastal(i: number) { export function isCoastal(i: number) {
return pack.cells.t[i] === DISTANCE_FIELD.LAND_COAST; return pack.cells.t[i] === DISTANCE_FIELD.LAND_COAST;
} }
// findAll d3.quandtree search from https://bl.ocks.org/lwthatcher/b41479725e0ff2277c7ac90df2de2b5e
function addFindAll() {
const Quad = function (node, x0, y0, x1, y1) {
this.node = node;
this.x0 = x0;
this.y0 = y0;
this.x1 = x1;
this.y1 = y1;
};
const tree_filter = function (x, y, radius) {
var t = {x, y, x0: this._x0, y0: this._y0, x3: this._x1, y3: this._y1, quads: [], node: this._root};
if (t.node) {
t.quads.push(new Quad(t.node, t.x0, t.y0, t.x3, t.y3));
}
radiusSearchInit(t, radius);
var i = 0;
while ((t.q = t.quads.pop())) {
i++;
// Stop searching if this quadrant cant contain a closer node.
if (
!(t.node = t.q.node) ||
(t.x1 = t.q.x0) > t.x3 ||
(t.y1 = t.q.y0) > t.y3 ||
(t.x2 = t.q.x1) < t.x0 ||
(t.y2 = t.q.y1) < t.y0
)
continue;
// Bisect the current quadrant.
if (t.node.length) {
t.node.explored = true;
var xm = (t.x1 + t.x2) / 2,
ym = (t.y1 + t.y2) / 2;
t.quads.push(
new Quad(t.node[3], xm, ym, t.x2, t.y2),
new Quad(t.node[2], t.x1, ym, xm, t.y2),
new Quad(t.node[1], xm, t.y1, t.x2, ym),
new Quad(t.node[0], t.x1, t.y1, xm, ym)
);
// Visit the closest quadrant first.
if ((t.i = ((y >= ym) << 1) | (x >= xm))) {
t.q = t.quads[t.quads.length - 1];
t.quads[t.quads.length - 1] = t.quads[t.quads.length - 1 - t.i];
t.quads[t.quads.length - 1 - t.i] = t.q;
}
}
// Visit this point. (Visiting coincident points isnt necessary!)
else {
var dx = x - +this._x.call(null, t.node.data),
dy = y - +this._y.call(null, t.node.data),
d2 = dx * dx + dy * dy;
radiusSearchVisit(t, d2);
}
}
return t.result;
};
d3.quadtree.prototype.findAll = tree_filter;
var radiusSearchInit = function (t, radius) {
t.result = [];
(t.x0 = t.x - radius), (t.y0 = t.y - radius);
(t.x3 = t.x + radius), (t.y3 = t.y + radius);
t.radius = radius * radius;
};
var radiusSearchVisit = function (t, d2) {
t.node.data.scanned = true;
if (d2 < t.radius) {
do {
t.result.push(t.node.data);
t.node.data.selected = true;
} while ((t.node = t.node.next));
}
};
}