"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(); } }