mirror of
https://github.com/Azgaar/Fantasy-Map-Generator.git
synced 2025-12-16 17:31:24 +01:00
187 lines
No EOL
7.5 KiB
JavaScript
187 lines
No EOL
7.5 KiB
JavaScript
"use strict";
|
|
function editCoastline(node = d3.event.target) {
|
|
if (customization) return;
|
|
closeDialogs(".stable");
|
|
if (layerIsOn("toggleCells")) toggleCells();
|
|
|
|
$("#coastlineEditor").dialog({
|
|
title: "Edit Coastline", resizable: false,
|
|
position: {my: "center top+20", at: "top", of: d3.event, collision: "fit"},
|
|
close: closeCoastlineEditor
|
|
});
|
|
|
|
debug.append("g").attr("id", "vertices");
|
|
elSelected = d3.select(node);
|
|
selectCoastlineGroup(node);
|
|
drawCoastlineVertices();
|
|
viewbox.on("touchmove mousemove", null);
|
|
|
|
if (modules.editCoastline) return;
|
|
modules.editCoastline = true;
|
|
|
|
// add listeners
|
|
document.getElementById("coastlineGroupsShow").addEventListener("click", showGroupSection);
|
|
document.getElementById("coastlineGroup").addEventListener("change", changeCoastlineGroup);
|
|
document.getElementById("coastlineGroupAdd").addEventListener("click", toggleNewGroupInput);
|
|
document.getElementById("coastlineGroupName").addEventListener("change", createNewGroup);
|
|
document.getElementById("coastlineGroupRemove").addEventListener("click", removeCoastlineGroup);
|
|
document.getElementById("coastlineGroupsHide").addEventListener("click", hideGroupSection);
|
|
document.getElementById("coastlineEditStyle").addEventListener("click", editGroupStyle);
|
|
|
|
function drawCoastlineVertices() {
|
|
const f = +elSelected.attr("data-f"); // feature id
|
|
const v = pack.features[f].vertices; // coastline outer vertices
|
|
|
|
const l = pack.cells.i.length;
|
|
const c = [... new Set(v.map(v => pack.vertices.c[v]).flat())].filter(c => c < l);
|
|
debug.select("#vertices").selectAll("polygon").data(c).enter().append("polygon")
|
|
.attr("points", d => getPackPolygon(d)).attr("data-c", d => d);
|
|
|
|
debug.select("#vertices").selectAll("circle").data(v).enter().append("circle")
|
|
.attr("cx", d => pack.vertices.p[d][0]).attr("cy", d => pack.vertices.p[d][1])
|
|
.attr("r", .4).attr("data-v", d => d).call(d3.drag().on("drag", dragVertex))
|
|
.on("mousemove", () => tip("Drag to move the vertex, please use for fine-tuning only. Edit heightmap to change actual cell heights"));
|
|
|
|
const unit = areaUnit.value === "square" ? " " + distanceUnitInput.value + "²" : " " + areaUnit.value;
|
|
const area = pack.features[f].area;
|
|
coastlineArea.innerHTML = si(area * distanceScaleInput.value ** 2) + unit;
|
|
}
|
|
|
|
function dragVertex() {
|
|
const x = rn(d3.event.x, 2), y = rn(d3.event.y, 2);
|
|
this.setAttribute("cx", x);
|
|
this.setAttribute("cy", y);
|
|
const v = +this.dataset.v;
|
|
pack.vertices.p[v] = [x, y];
|
|
debug.select("#vertices").selectAll("polygon").attr("points", d => getPackPolygon(d));
|
|
redrawCoastline();
|
|
}
|
|
|
|
function redrawCoastline() {
|
|
lineGen.curve(d3.curveBasisClosed);
|
|
const f = +elSelected.attr("data-f");
|
|
const vertices = pack.features[f].vertices;
|
|
const points = vertices.map(v => pack.vertices.p[v]);
|
|
const d = round(lineGen(points));
|
|
elSelected.attr("d", d);
|
|
defs.select("mask#land > path#land_"+f).attr("d", d); // update land mask
|
|
defs.select("mask#water > path#water_"+f).attr("d", d); // update water mask
|
|
|
|
const unit = areaUnit.value === "square" ? " " + distanceUnitInput.value + "²" : " " + areaUnit.value;
|
|
const area = Math.abs(d3.polygonArea(points));
|
|
coastlineArea.innerHTML = si(area * distanceScaleInput.value ** 2) + unit;
|
|
}
|
|
|
|
function showGroupSection() {
|
|
document.querySelectorAll("#coastlineEditor > button").forEach(el => el.style.display = "none");
|
|
document.getElementById("coastlineGroupsSelection").style.display = "inline-block";
|
|
}
|
|
|
|
function hideGroupSection() {
|
|
document.querySelectorAll("#coastlineEditor > button").forEach(el => el.style.display = "inline-block");
|
|
document.getElementById("coastlineGroupsSelection").style.display = "none";
|
|
document.getElementById("coastlineGroupName").style.display = "none";
|
|
document.getElementById("coastlineGroupName").value = "";
|
|
document.getElementById("coastlineGroup").style.display = "inline-block";
|
|
}
|
|
|
|
function selectCoastlineGroup(node) {
|
|
const group = node.parentNode.id;
|
|
const select = document.getElementById("coastlineGroup");
|
|
select.options.length = 0; // remove all options
|
|
|
|
coastline.selectAll("g").each(function() {
|
|
select.options.add(new Option(this.id, this.id, false, this.id === group));
|
|
});
|
|
}
|
|
|
|
function changeCoastlineGroup() {
|
|
document.getElementById(this.value).appendChild(elSelected.node());
|
|
}
|
|
|
|
function toggleNewGroupInput() {
|
|
if (coastlineGroupName.style.display === "none") {
|
|
coastlineGroupName.style.display = "inline-block";
|
|
coastlineGroupName.focus();
|
|
coastlineGroup.style.display = "none";
|
|
} else {
|
|
coastlineGroupName.style.display = "none";
|
|
coastlineGroup.style.display = "inline-block";
|
|
}
|
|
}
|
|
|
|
function createNewGroup() {
|
|
if (!this.value) {tip("Please provide a valid group name"); return;}
|
|
const group = this.value.toLowerCase().replace(/ /g, "_").replace(/[^\w\s]/gi, "");
|
|
|
|
if (document.getElementById(group)) {
|
|
tip("Element with this id already exists. Please provide a unique name", false, "error");
|
|
return;
|
|
}
|
|
|
|
if (Number.isFinite(+group.charAt(0))) {
|
|
tip("Group name should start with a letter", false, "error");
|
|
return;
|
|
}
|
|
|
|
// just rename if only 1 element left
|
|
const oldGroup = elSelected.node().parentNode;
|
|
const basic = ["sea_island", "lake_island"].includes(oldGroup.id);
|
|
if (!basic && oldGroup.childElementCount === 1) {
|
|
document.getElementById("coastlineGroup").selectedOptions[0].remove();
|
|
document.getElementById("coastlineGroup").options.add(new Option(group, group, false, true));
|
|
oldGroup.id = group;
|
|
toggleNewGroupInput();
|
|
document.getElementById("coastlineGroupName").value = "";
|
|
return;
|
|
}
|
|
|
|
// create a new group
|
|
const newGroup = elSelected.node().parentNode.cloneNode(false);
|
|
document.getElementById("coastline").appendChild(newGroup);
|
|
newGroup.id = group;
|
|
document.getElementById("coastlineGroup").options.add(new Option(group, group, false, true));
|
|
document.getElementById(group).appendChild(elSelected.node());
|
|
|
|
toggleNewGroupInput();
|
|
document.getElementById("coastlineGroupName").value = "";
|
|
}
|
|
|
|
function removeCoastlineGroup() {
|
|
const group = elSelected.node().parentNode.id;
|
|
if (["sea_island", "lake_island"].includes(group)) {
|
|
tip("This is one of the default groups, it cannot be removed", false, "error");
|
|
return;
|
|
}
|
|
|
|
const count = elSelected.node().parentNode.childElementCount;
|
|
alertMessage.innerHTML = `Are you sure you want to remove the group?
|
|
All coastline elements of the group (${count}) will be moved under <i>sea_island</i> group`;
|
|
$("#alert").dialog({resizable: false, title: "Remove coastline group", width:"26em",
|
|
buttons: {
|
|
Remove: function() {
|
|
$(this).dialog("close");
|
|
const sea = document.getElementById("sea_island");
|
|
const groupEl = document.getElementById(group);
|
|
while (groupEl.childNodes.length) {
|
|
sea.appendChild(groupEl.childNodes[0]);
|
|
}
|
|
groupEl.remove();
|
|
document.getElementById("coastlineGroup").selectedOptions[0].remove();
|
|
document.getElementById("coastlineGroup").value = "sea_island";
|
|
},
|
|
Cancel: function() {$(this).dialog("close");}
|
|
}
|
|
});
|
|
}
|
|
|
|
function editGroupStyle() {
|
|
const g = elSelected.node().parentNode.id;
|
|
editStyle("coastline", g);
|
|
}
|
|
|
|
function closeCoastlineEditor() {
|
|
debug.select("#vertices").remove();
|
|
unselect();
|
|
}
|
|
} |