mirror of
https://github.com/Azgaar/Fantasy-Map-Generator.git
synced 2025-12-16 17:31:24 +01:00
* refactor: submap - start * refactor: submap - continue * Merge branch 'master' of https://github.com/Azgaar/Fantasy-Map-Generator into submap-refactoring * refactor: submap - relocate burgs * refactor: submap - restore routes * refactor: submap - restore lake names * refactor: submap - UI update * refactor: submap - restore river and biome data * refactor: submap - simplify options * refactor: submap - restore rivers * refactor: submap - recalculateMapSize * refactor: submap - add middle points * refactor: submap - don't add middle points, unified findPath fn * chore: update version * feat: submap - relocate out of map regiments * feat: submap - fix route gen * feat: submap - allow custom number of cells * feat: submap - add checkbox submapRescaleBurgStyles * feat: submap - update version hash * chore: supporters update --------- Co-authored-by: Azgaar <azgaar.fmg@yandex.com>
285 lines
8.2 KiB
JavaScript
285 lines
8.2 KiB
JavaScript
"use strict";
|
|
function editRiver(id) {
|
|
if (customization) return;
|
|
if (elSelected && id === elSelected.attr("id")) return;
|
|
closeDialogs(".stable");
|
|
if (!layerIsOn("toggleRivers")) toggleRivers();
|
|
|
|
byId("toggleCells").dataset.forced = +!layerIsOn("toggleCells");
|
|
if (!layerIsOn("toggleCells")) toggleCells();
|
|
|
|
elSelected = d3.select("#" + id).on("click", addControlPoint);
|
|
|
|
tip(
|
|
"Drag control points to change the river course. Click on point to remove it. Click on river to add additional control point. For major changes please create a new river instead",
|
|
true
|
|
);
|
|
debug.append("g").attr("id", "controlCells");
|
|
debug.append("g").attr("id", "controlPoints");
|
|
|
|
updateRiverData();
|
|
|
|
const river = getRiver();
|
|
const {cells, points} = river;
|
|
const riverPoints = Rivers.getRiverPoints(cells, points);
|
|
drawControlPoints(riverPoints);
|
|
drawCells(cells);
|
|
|
|
$("#riverEditor").dialog({
|
|
title: "Edit River",
|
|
resizable: false,
|
|
position: {my: "left top", at: "left+10 top+10", of: "#map"},
|
|
close: closeRiverEditor
|
|
});
|
|
|
|
if (modules.editRiver) return;
|
|
modules.editRiver = true;
|
|
|
|
// add listeners
|
|
byId("riverCreateSelectingCells").on("click", createRiver);
|
|
byId("riverEditStyle").on("click", () => editStyle("rivers"));
|
|
byId("riverElevationProfile").on("click", showRiverElevationProfile);
|
|
byId("riverLegend").on("click", editRiverLegend);
|
|
byId("riverRemove").on("click", removeRiver);
|
|
byId("riverName").on("input", changeName);
|
|
byId("riverType").on("input", changeType);
|
|
byId("riverNameCulture").on("click", generateNameCulture);
|
|
byId("riverNameRandom").on("click", generateNameRandom);
|
|
byId("riverMainstem").on("change", changeParent);
|
|
byId("riverSourceWidth").on("input", changeSourceWidth);
|
|
byId("riverWidthFactor").on("input", changeWidthFactor);
|
|
|
|
function getRiver() {
|
|
const riverId = +elSelected.attr("id").slice(5);
|
|
const river = pack.rivers.find(r => r.i === riverId);
|
|
return river;
|
|
}
|
|
|
|
function updateRiverData() {
|
|
const r = getRiver();
|
|
|
|
byId("riverName").value = r.name;
|
|
byId("riverType").value = r.type;
|
|
|
|
const parentSelect = byId("riverMainstem");
|
|
parentSelect.options.length = 0;
|
|
const parent = r.parent || r.i;
|
|
const sortedRivers = pack.rivers.slice().sort((a, b) => (a.name > b.name ? 1 : -1));
|
|
sortedRivers.forEach(river => {
|
|
const opt = new Option(river.name, river.i, false, river.i === parent);
|
|
parentSelect.options.add(opt);
|
|
});
|
|
byId("riverBasin").value = pack.rivers.find(river => river.i === r.basin).name;
|
|
|
|
byId("riverDischarge").value = r.discharge + " m³/s";
|
|
byId("riverSourceWidth").value = r.sourceWidth;
|
|
byId("riverWidthFactor").value = r.widthFactor;
|
|
|
|
updateRiverLength(r);
|
|
updateRiverWidth(r);
|
|
}
|
|
|
|
function updateRiverLength(river) {
|
|
river.length = rn(elSelected.node().getTotalLength() / 2, 2);
|
|
const lengthUI = `${rn(river.length * distanceScale)} ${distanceUnitInput.value}`;
|
|
byId("riverLength").value = lengthUI;
|
|
}
|
|
|
|
function updateRiverWidth(river) {
|
|
const {cells, discharge, widthFactor, sourceWidth} = river;
|
|
const meanderedPoints = Rivers.addMeandering(cells);
|
|
river.width = Rivers.getWidth(
|
|
Rivers.getOffset({
|
|
flux: discharge,
|
|
pointIndex: meanderedPoints.length,
|
|
widthFactor,
|
|
startingWidth: sourceWidth
|
|
})
|
|
);
|
|
|
|
const width = `${rn(river.width * distanceScale, 3)} ${distanceUnitInput.value}`;
|
|
byId("riverWidth").value = width;
|
|
}
|
|
|
|
function drawControlPoints(points) {
|
|
debug
|
|
.select("#controlPoints")
|
|
.selectAll("circle")
|
|
.data(points)
|
|
.join("circle")
|
|
.attr("cx", d => d[0])
|
|
.attr("cy", d => d[1])
|
|
.attr("r", 0.6)
|
|
.call(d3.drag().on("start", dragControlPoint))
|
|
.on("click", removeControlPoint);
|
|
}
|
|
|
|
function drawCells(cells) {
|
|
const validCells = [...new Set(cells)].filter(i => pack.cells.i[i]);
|
|
debug
|
|
.select("#controlCells")
|
|
.selectAll(`polygon`)
|
|
.data(validCells)
|
|
.join("polygon")
|
|
.attr("points", d => getPackPolygon(d));
|
|
}
|
|
|
|
function dragControlPoint() {
|
|
const {r, fl} = pack.cells;
|
|
const river = getRiver();
|
|
|
|
const {x: x0, y: y0} = d3.event;
|
|
const initCell = findCell(x0, y0);
|
|
|
|
let movedToCell = null;
|
|
|
|
d3.event.on("drag", function () {
|
|
const {x, y} = d3.event;
|
|
const currentCell = findCell(x, y);
|
|
|
|
movedToCell = initCell !== currentCell ? currentCell : null;
|
|
|
|
this.setAttribute("cx", x);
|
|
this.setAttribute("cy", y);
|
|
this.__data__ = [rn(x, 1), rn(y, 1)];
|
|
redrawRiver();
|
|
drawCells(river.cells);
|
|
});
|
|
|
|
d3.event.on("end", () => {
|
|
if (movedToCell && !r[movedToCell]) {
|
|
// swap river data
|
|
r[initCell] = 0;
|
|
r[movedToCell] = river.i;
|
|
const sourceFlux = fl[initCell];
|
|
fl[initCell] = fl[movedToCell];
|
|
fl[movedToCell] = sourceFlux;
|
|
redrawRiver();
|
|
}
|
|
});
|
|
}
|
|
|
|
function redrawRiver() {
|
|
const river = getRiver();
|
|
river.points = debug.selectAll("#controlPoints > *").data();
|
|
river.cells = river.points.map(([x, y]) => findCell(x, y));
|
|
|
|
lineGen.curve(d3.curveCatmullRom.alpha(0.1));
|
|
const meanderedPoints = Rivers.addMeandering(river.cells, river.points);
|
|
const path = Rivers.getRiverPath(meanderedPoints, river.widthFactor, river.sourceWidth);
|
|
elSelected.attr("d", path);
|
|
|
|
updateRiverLength(river);
|
|
if (byId("elevationProfile").offsetParent) showRiverElevationProfile();
|
|
}
|
|
|
|
function addControlPoint() {
|
|
const [x, y] = d3.mouse(this);
|
|
const point = [rn(x, 1), rn(y, 1)];
|
|
|
|
const river = getRiver();
|
|
if (!river.points) river.points = debug.selectAll("#controlPoints > *").data();
|
|
|
|
const index = getSegmentId(river.points, point, 2);
|
|
river.points.splice(index, 0, point);
|
|
drawControlPoints(river.points);
|
|
redrawRiver();
|
|
}
|
|
|
|
function removeControlPoint() {
|
|
this.remove();
|
|
redrawRiver();
|
|
|
|
const {cells} = getRiver();
|
|
drawCells(cells);
|
|
}
|
|
|
|
function changeName() {
|
|
getRiver().name = this.value;
|
|
}
|
|
|
|
function changeType() {
|
|
getRiver().type = this.value;
|
|
}
|
|
|
|
function generateNameCulture() {
|
|
const r = getRiver();
|
|
r.name = riverName.value = Rivers.getName(r.mouth);
|
|
}
|
|
|
|
function generateNameRandom() {
|
|
const r = getRiver();
|
|
if (r) r.name = riverName.value = Names.getBase(rand(nameBases.length - 1));
|
|
}
|
|
|
|
function changeParent() {
|
|
const r = getRiver();
|
|
r.parent = +this.value;
|
|
r.basin = pack.rivers.find(river => river.i === r.parent).basin;
|
|
byId("riverBasin").value = pack.rivers.find(river => river.i === r.basin).name;
|
|
}
|
|
|
|
function changeSourceWidth() {
|
|
const river = getRiver();
|
|
river.sourceWidth = +this.value;
|
|
updateRiverWidth(river);
|
|
redrawRiver();
|
|
}
|
|
|
|
function changeWidthFactor() {
|
|
const river = getRiver();
|
|
river.widthFactor = +this.value;
|
|
updateRiverWidth(river);
|
|
redrawRiver();
|
|
}
|
|
|
|
function showRiverElevationProfile() {
|
|
const points = debug
|
|
.selectAll("#controlPoints > *")
|
|
.data()
|
|
.map(([x, y]) => findCell(x, y));
|
|
const river = getRiver();
|
|
const riverLen = rn(river.length * distanceScale);
|
|
showElevationProfile(points, riverLen, true);
|
|
}
|
|
|
|
function editRiverLegend() {
|
|
const id = elSelected.attr("id");
|
|
const river = getRiver();
|
|
editNotes(id, river.name + " " + river.type);
|
|
}
|
|
|
|
function removeRiver() {
|
|
alertMessage.innerHTML = "Are you sure you want to remove the river and all its tributaries";
|
|
$("#alert").dialog({
|
|
resizable: false,
|
|
width: "22em",
|
|
title: "Remove river and tributaries",
|
|
buttons: {
|
|
Remove: function () {
|
|
$(this).dialog("close");
|
|
const river = +elSelected.attr("id").slice(5);
|
|
Rivers.remove(river);
|
|
elSelected.remove();
|
|
$("#riverEditor").dialog("close");
|
|
},
|
|
Cancel: function () {
|
|
$(this).dialog("close");
|
|
}
|
|
}
|
|
});
|
|
}
|
|
|
|
function closeRiverEditor() {
|
|
debug.select("#controlPoints").remove();
|
|
debug.select("#controlCells").remove();
|
|
|
|
elSelected.on("click", null);
|
|
unselect();
|
|
clearMainTip();
|
|
|
|
const forced = +byId("toggleCells").dataset.forced;
|
|
byId("toggleCells").dataset.forced = 0;
|
|
if (forced && layerIsOn("toggleCells")) toggleCells();
|
|
}
|
|
}
|