const $body = insertEditorHtml(); addListeners(); export function open() { closeDialogs("#statesEditor, .stable"); if (!layerIsOn("toggleStates")) toggleStates(); if (!layerIsOn("toggleBorders")) toggleBorders(); if (layerIsOn("toggleCultures")) toggleCultures(); if (layerIsOn("toggleBiomes")) toggleBiomes(); if (layerIsOn("toggleReligions")) toggleReligions(); refreshStatesEditor(); $("#statesEditor").dialog({ title: "States Editor", resizable: false, close: closeStatesEditor, position: {my: "right top", at: "right-10 top+10", of: "svg", collision: "fit"} }); } function insertEditorHtml() { const editorHtml = /* html */ `
State 
Form 
Capital 
Culture 
Burgs 
Area 
Population 
States: 0
Cells: 0
Burgs: 0
Land Area: 0
Population: 0
`; byId("dialogs").insertAdjacentHTML("beforeend", editorHtml); return byId("statesBodySection"); } function addListeners() { applySortingByHeader("statesHeader"); byId("statesEditorRefresh").on("click", refreshStatesEditor); byId("statesEditStyle").on("click", () => editStyle("regions")); byId("statesLegend").on("click", toggleLegend); byId("statesPercentage").on("click", togglePercentageMode); byId("statesChart").on("click", showStatesChart); byId("statesRegenerate").on("click", openRegenerationMenu); byId("statesRegenerateBack").on("click", exitRegenerationMenu); byId("statesRecalculate").on("click", () => recalculateStates(true)); byId("statesRandomize").on("click", randomizeStatesExpansion); byId("statesNeutral").on("input", changeStatesGrowthRate); byId("statesNeutralNumber").on("change", changeStatesGrowthRate); byId("statesManually").on("click", enterStatesManualAssignent); byId("statesManuallyApply").on("click", applyStatesManualAssignent); byId("statesManuallyCancel").on("click", () => exitStatesManualAssignment(false)); byId("statesAdd").on("click", enterAddStateMode); byId("statesExport").on("click", downloadStatesCsv); $body.on("click", event => { const $element = event.target; const classList = $element.classList; const stateId = +$element.parentNode?.dataset?.id; if ($element.tagName === "FILL-BOX") stateChangeFill($element); else if (classList.contains("name")) editStateName(stateId); else if (classList.contains("coaIcon")) editEmblem("state", "stateCOA" + stateId, pack.states[stateId]); else if (classList.contains("icon-star-empty")) stateCapitalZoomIn(stateId); else if (classList.contains("statePopulation")) changePopulation(stateId); else if (classList.contains("icon-pin")) toggleFog(stateId, classList); else if (classList.contains("icon-trash-empty")) stateRemovePrompt(stateId); }); $body.on("input", function (ev) { const $element = ev.target; const classList = $element.classList; const line = $element.parentNode; const state = +line.dataset.id; if (classList.contains("stateCapital")) stateChangeCapitalName(state, line, $element.value); else if (classList.contains("cultureType")) stateChangeType(state, line, $element.value); else if (classList.contains("statePower")) stateChangeExpansionism(state, line, $element.value); }); $body.on("change", function (ev) { const $element = ev.target; const classList = $element.classList; const line = $element.parentNode; const state = +line.dataset.id; if (classList.contains("stateCulture")) stateChangeCulture(state, line, $element.value); }); } function refreshStatesEditor() { BurgsAndStates.collectStatistics(); statesEditorAddLines(); } // add line for each state function statesEditorAddLines() { const unit = getAreaUnit(); const hidden = byId("statesRegenerateButtons").style.display === "block" ? "" : "hidden"; // toggle regenerate columns let lines = ""; let totalArea = 0; let totalPopulation = 0; let totalBurgs = 0; for (const s of pack.states) { if (s.removed) continue; const area = getArea(s.area); const rural = s.rural * populationRate; const urban = s.urban * populationRate * urbanization; const population = rn(rural + urban); const populationTip = `Total population: ${si(population)}; Rural population: ${si(rural)}; Urban population: ${si( urban )}. Click to change`; totalArea += area; totalPopulation += population; totalBurgs += s.burgs; const focused = defs.select("#fog #focusState" + s.i).size(); if (!s.i) { // Neutral line lines += /* html */ `
${s.burgs}
${si(area)} ${unit}
${si(population)}
${s.cells}
`; continue; } const capital = pack.burgs[s.capital].name; COArenderer.trigger("stateCOA" + s.i, s.coa); lines += /* html */ `
${s.burgs}
${si(area)} ${unit}
${si(population)}
${s.cells}
`; } $body.innerHTML = lines; // update footer byId("statesFooterStates").innerHTML = pack.states.filter(s => s.i && !s.removed).length; byId("statesFooterCells").innerHTML = pack.cells.h.filter(h => h >= 20).length; byId("statesFooterBurgs").innerHTML = totalBurgs; byId("statesFooterArea").innerHTML = si(totalArea) + unit; byId("statesFooterArea").dataset.area = totalArea; byId("statesFooterPopulation").innerHTML = si(totalPopulation); byId("statesFooterPopulation").dataset.population = totalPopulation; // add listeners $body.querySelectorAll(":scope > div").forEach($line => { $line.on("mouseenter", stateHighlightOn); $line.on("mouseleave", stateHighlightOff); $line.on("click", selectStateOnLineClick); }); if ($body.dataset.type === "percentage") { $body.dataset.type = "absolute"; togglePercentageMode(); } applySorting(statesHeader); $("#statesEditor").dialog({width: fitContent()}); } function getCultureOptions(culture) { let options = ""; pack.cultures.forEach(c => { if (!c.removed) { options += ``; } }); return options; } function getTypeOptions(type) { let options = ""; const types = ["Generic", "River", "Lake", "Naval", "Nomadic", "Hunting", "Highland"]; types.forEach(t => (options += ``)); return options; } function stateHighlightOn(event) { if (!layerIsOn("toggleStates")) return; if (defs.select("#fog path").size()) return; const state = +event.target.dataset.id; if (customization || !state) return; const d = regions.select("#state" + state).attr("d"); const path = debug .append("path") .attr("class", "highlight") .attr("d", d) .attr("fill", "none") .attr("stroke", "red") .attr("stroke-width", 1) .attr("opacity", 1) .attr("filter", "url(#blur1)"); const totalLength = path.node().getTotalLength(); const duration = (totalLength + 5000) / 2; const interpolate = d3.interpolateString(`0, ${totalLength}`, `${totalLength}, ${totalLength}`); path .transition() .duration(duration) .attrTween("stroke-dasharray", () => interpolate); } function stateHighlightOff() { debug.selectAll(".highlight").each(function () { d3.select(this).transition().duration(1000).attr("opacity", 0).remove(); }); } function stateChangeFill(el) { const currentFill = el.getAttribute("fill"); const state = +el.parentNode.dataset.id; const callback = function (newFill) { el.fill = newFill; pack.states[state].color = newFill; statesBody.select("#state" + state).attr("fill", newFill); statesBody.select("#state-gap" + state).attr("stroke", newFill); const halo = d3.color(newFill) ? d3.color(newFill).darker().hex() : "#666666"; statesHalo.select("#state-border" + state).attr("stroke", halo); // recolor regiments const solidColor = newFill[0] === "#" ? newFill : "#999"; const darkerColor = d3.color(solidColor).darker().hex(); armies.select("#army" + state).attr("fill", solidColor); armies .select("#army" + state) .selectAll("g > rect:nth-of-type(2)") .attr("fill", darkerColor); }; openPicker(currentFill, callback); } function editStateName(state) { // reset input value and close add mode stateNameEditorCustomForm.value = ""; const addModeActive = stateNameEditorCustomForm.style.display === "inline-block"; if (addModeActive) { stateNameEditorCustomForm.style.display = "none"; stateNameEditorSelectForm.style.display = "inline-block"; } const s = pack.states[state]; byId("stateNameEditor").dataset.state = state; byId("stateNameEditorShort").value = s.name || ""; applyOption(stateNameEditorSelectForm, s.formName); byId("stateNameEditorFull").value = s.fullName || ""; $("#stateNameEditor").dialog({ resizable: false, title: "Change state name", buttons: { Apply: function () { applyNameChange(s); $(this).dialog("close"); }, Cancel: function () { $(this).dialog("close"); } }, position: {my: "center", at: "center", of: "svg"} }); if (modules.editStateName) return; modules.editStateName = true; // add listeners byId("stateNameEditorShortCulture").on("click", regenerateShortNameCuture); byId("stateNameEditorShortRandom").on("click", regenerateShortNameRandom); byId("stateNameEditorAddForm").on("click", addCustomForm); byId("stateNameEditorCustomForm").on("change", addCustomForm); byId("stateNameEditorFullRegenerate").on("click", regenerateFullName); function regenerateShortNameCuture() { const state = +stateNameEditor.dataset.state; const culture = pack.states[state].culture; const name = Names.getState(Names.getCultureShort(culture), culture); byId("stateNameEditorShort").value = name; } function regenerateShortNameRandom() { const base = rand(nameBases.length - 1); const name = Names.getState(Names.getBase(base), undefined, base); byId("stateNameEditorShort").value = name; } function addCustomForm() { const value = stateNameEditorCustomForm.value; const addModeActive = stateNameEditorCustomForm.style.display === "inline-block"; stateNameEditorCustomForm.style.display = addModeActive ? "none" : "inline-block"; stateNameEditorSelectForm.style.display = addModeActive ? "inline-block" : "none"; if (value && addModeActive) applyOption(stateNameEditorSelectForm, value); stateNameEditorCustomForm.value = ""; } function regenerateFullName() { const short = byId("stateNameEditorShort").value; const form = byId("stateNameEditorSelectForm").value; byId("stateNameEditorFull").value = getFullName(); function getFullName() { if (!form) return short; if (!short && form) return "The " + form; const tick = +stateNameEditorFullRegenerate.dataset.tick; stateNameEditorFullRegenerate.dataset.tick = tick + 1; return tick % 2 ? getAdjective(short) + " " + form : form + " of " + short; } } function applyNameChange(s) { const nameInput = byId("stateNameEditorShort"); const formSelect = byId("stateNameEditorSelectForm"); const fullNameInput = byId("stateNameEditorFull"); const nameChanged = nameInput.value !== s.name; const formChanged = formSelect.value !== s.formName; const fullNameChanged = fullNameInput.value !== s.fullName; const changed = nameChanged || formChanged || fullNameChanged; if (formChanged) { const selected = formSelect.selectedOptions[0]; const form = selected.parentElement.label || null; if (form) s.form = form; } s.name = nameInput.value; s.formName = formSelect.value; s.fullName = fullNameInput.value; if (changed && stateNameEditorUpdateLabel.checked) BurgsAndStates.drawStateLabels([s.i]); refreshStatesEditor(); } } function stateChangeCapitalName(state, line, value) { line.dataset.capital = value; const capital = pack.states[state].capital; if (!capital) return; pack.burgs[capital].name = value; document.querySelector("#burgLabel" + capital).textContent = value; } function changePopulation(stateId) { const state = pack.states[stateId]; if (!state.cells) return tip("State does not have any cells, cannot change population", false, "error"); const rural = rn(state.rural * populationRate); const urban = rn(state.urban * populationRate * urbanization); const total = rural + urban; const format = n => Number(n).toLocaleString(); alertMessage.innerHTML = /* html */ `
Change population of all cells assigned to the state
Rural: Urban:
Total population: ${format(total)} ⇒ ${format(total)} (100%)
`; const update = function () { const totalNew = ruralPop.valueAsNumber + urbanPop.valueAsNumber; if (isNaN(totalNew)) return; totalPop.innerHTML = format(totalNew); totalPopPerc.innerHTML = rn((totalNew / total) * 100); }; ruralPop.oninput = () => update(); urbanPop.oninput = () => update(); $("#alert").dialog({ resizable: false, title: "Change state population", width: "24em", buttons: { Apply: function () { applyPopulationChange(); $(this).dialog("close"); }, Cancel: function () { $(this).dialog("close"); } }, position: {my: "center", at: "center", of: "svg"} }); function applyPopulationChange() { const ruralChange = ruralPop.value / rural; if (isFinite(ruralChange) && ruralChange !== 1) { const cells = pack.cells.i.filter(i => pack.cells.state[i] === stateId); cells.forEach(i => (pack.cells.pop[i] *= ruralChange)); } if (!isFinite(ruralChange) && +ruralPop.value > 0) { const points = ruralPop.value / populationRate; const cells = pack.cells.i.filter(i => pack.cells.state[i] === stateId); const pop = points / cells.length; cells.forEach(i => (pack.cells.pop[i] = pop)); } const urbanChange = urbanPop.value / urban; if (isFinite(urbanChange) && urbanChange !== 1) { const burgs = pack.burgs.filter(b => !b.removed && b.state === stateId); burgs.forEach(b => (b.population = rn(b.population * urbanChange, 4))); } if (!isFinite(urbanChange) && +urbanPop.value > 0) { const points = urbanPop.value / populationRate / urbanization; const burgs = pack.burgs.filter(b => !b.removed && b.state === stateId); const population = rn(points / burgs.length, 4); burgs.forEach(b => (b.population = population)); } refreshStatesEditor(); } } function stateCapitalZoomIn(state) { const capital = pack.states[state].capital; const l = burgLabels.select("[data-id='" + capital + "']"); const x = +l.attr("x"), y = +l.attr("y"); zoomTo(x, y, 8, 2000); } function stateChangeCulture(state, line, value) { line.dataset.base = pack.states[state].culture = +value; } function stateChangeType(state, line, value) { line.dataset.type = pack.states[state].type = value; recalculateStates(); } function stateChangeExpansionism(state, line, value) { line.dataset.expansionism = pack.states[state].expansionism = value; recalculateStates(); } function toggleFog(state, cl) { if (customization) return; const path = statesBody.select("#state" + state).attr("d"), id = "focusState" + state; cl.contains("inactive") ? fog(id, path) : unfog(id); cl.toggle("inactive"); } function stateRemovePrompt(state) { if (customization) return; confirmationDialog({ title: "Remove state", message: "Are you sure you want to remove the state?
This action cannot be reverted", confirm: "Remove", onConfirm: () => stateRemove(state) }); } function stateRemove(state) { statesBody.select("#state" + state).remove(); statesBody.select("#state-gap" + state).remove(); statesHalo.select("#state-border" + state).remove(); labels.select("#stateLabel" + state).remove(); defs.select("#textPath_stateLabel" + state).remove(); unfog("focusState" + state); pack.burgs.forEach(b => { if (b.state === state) b.state = 0; }); pack.cells.state.forEach((s, i) => { if (s === state) pack.cells.state[i] = 0; }); // remove emblem const coaId = "stateCOA" + state; byId(coaId).remove(); emblems.select(`#stateEmblems > use[data-i='${state}']`).remove(); // remove provinces pack.states[state].provinces.forEach(p => { pack.provinces[p] = {i: p, removed: true}; pack.cells.province.forEach((pr, i) => { if (pr === p) pack.cells.province[i] = 0; }); const coaId = "provinceCOA" + p; if (byId(coaId)) byId(coaId).remove(); emblems.select(`#provinceEmblems > use[data-i='${p}']`).remove(); const g = provs.select("#provincesBody"); g.select("#province" + p).remove(); g.select("#province-gap" + p).remove(); }); // remove military pack.states[state].military.forEach(m => { const id = `regiment${state}-${m.i}`; const index = notes.findIndex(n => n.id === id); if (index != -1) notes.splice(index, 1); }); armies.select("g#army" + state).remove(); const capital = pack.states[state].capital; pack.burgs[capital].capital = 0; pack.burgs[capital].state = 0; moveBurgToGroup(capital, "towns"); pack.states[state] = {i: state, removed: true}; debug.selectAll(".highlight").remove(); if (!layerIsOn("toggleStates")) toggleStates(); else drawStates(); if (!layerIsOn("toggleBorders")) toggleBorders(); else drawBorders(); if (layerIsOn("toggleProvinces")) drawProvinces(); refreshStatesEditor(); } function toggleLegend() { if (legend.selectAll("*").size()) return clearLegend(); // hide legend const data = pack.states .filter(s => s.i && !s.removed && s.cells) .sort((a, b) => b.area - a.area) .map(s => [s.i, s.color, s.name]); drawLegend("States", data); } function togglePercentageMode() { if ($body.dataset.type === "absolute") { $body.dataset.type = "percentage"; const totalCells = +byId("statesFooterCells").innerText; const totalBurgs = +byId("statesFooterBurgs").innerText; const totalArea = +byId("statesFooterArea").dataset.area; const totalPopulation = +byId("statesFooterPopulation").dataset.population; $body.querySelectorAll(":scope > div").forEach(function (el) { const {cells, burgs, area, population} = el.dataset; el.querySelector(".stateCells").innerText = rn((+cells / totalCells) * 100) + "%"; el.querySelector(".stateBurgs").innerText = rn((+burgs / totalBurgs) * 100) + "%"; el.querySelector(".stateArea").innerText = rn((+area / totalArea) * 100) + "%"; el.querySelector(".statePopulation").innerText = rn((+population / totalPopulation) * 100) + "%"; }); } else { $body.dataset.type = "absolute"; statesEditorAddLines(); } } function showStatesChart() { // build hierarchy tree const statesData = pack.states.filter(s => !s.removed); if (statesData.length < 2) return tip("There are no states to show", false, "error"); const root = d3 .stratify() .id(d => d.i) .parentId(d => (d.i ? 0 : null))(statesData) .sum(d => d.area) .sort((a, b) => b.value - a.value); const size = 150 + 200 * uiSizeOutput.value; const margin = {top: 0, right: -50, bottom: 0, left: -50}; const w = size - margin.left - margin.right; const h = size - margin.top - margin.bottom; const treeLayout = d3.pack().size([w, h]).padding(3); // prepare svg alertMessage.innerHTML = /* html */ ``; alertMessage.innerHTML += `
`; const svg = d3 .select("#alertMessage") .insert("svg", "#statesInfo") .attr("id", "statesTree") .attr("width", size) .attr("height", size) .style("font-family", "Almendra SC") .attr("text-anchor", "middle") .attr("dominant-baseline", "central"); const graph = svg.append("g").attr("transform", `translate(-50, 0)`); byId("statesTreeType").on("change", updateChart); treeLayout(root); const node = graph .selectAll("g") .data(root.leaves()) .enter() .append("g") .attr("transform", d => `translate(${d.x},${d.y})`) .attr("data-id", d => d.data.i) .on("mouseenter", d => showInfo(event, d)) .on("mouseleave", d => hideInfo(event, d)); node .append("circle") .attr("fill", d => d.data.color) .attr("r", d => d.r); const exp = /(?=[A-Z][^A-Z])/g; const lp = n => d3.max(n.split(exp).map(p => p.length)) + 1; // longest name part + 1 node .append("text") .style("font-size", d => rn((d.r ** 0.97 * 4) / lp(d.data.name), 2) + "px") .selectAll("tspan") .data(d => d.data.name.split(exp)) .join("tspan") .attr("x", 0) .text(d => d) .attr("dy", (d, i, n) => `${i ? 1 : (n.length - 1) / -2}em`); function showInfo(ev, d) { d3.select(ev.target).select("circle").classed("selected", 1); const state = d.data.fullName; const area = getArea(d.data.area) + " " + getAreaUnit(); const rural = rn(d.data.rural * populationRate); const urban = rn(d.data.urban * populationRate * urbanization); const option = statesTreeType.value; const value = option === "area" ? "Area: " + area : option === "rural" ? "Rural population: " + si(rural) : option === "urban" ? "Urban population: " + si(urban) : option === "burgs" ? "Burgs number: " + d.data.burgs : "Population: " + si(rural + urban); statesInfo.innerHTML = /* html */ `${state}. ${value}`; stateHighlightOn(ev); } function hideInfo(ev) { stateHighlightOff(ev); if (!byId("statesInfo")) return; statesInfo.innerHTML = "‍"; d3.select(ev.target).select("circle").classed("selected", 0); } function updateChart() { const value = this.value === "area" ? d => d.area : this.value === "rural" ? d => d.rural : this.value === "urban" ? d => d.urban : this.value === "burgs" ? d => d.burgs : d => d.rural + d.urban; root.sum(value); node.data(treeLayout(root).leaves()); node .transition() .duration(1500) .attr("transform", d => `translate(${d.x},${d.y})`); node .select("circle") .transition() .duration(1500) .attr("r", d => d.r); node .select("text") .transition() .duration(1500) .style("font-size", d => rn((d.r ** 0.97 * 4) / lp(d.data.name), 2) + "px"); } $("#alert").dialog({ title: "States bubble chart", width: fitContent(), position: {my: "left bottom", at: "left+10 bottom-10", of: "svg"}, buttons: {}, close: () => { alertMessage.innerHTML = ""; } }); } function openRegenerationMenu() { byId("statesBottom") .querySelectorAll(":scope > button") .forEach(el => (el.style.display = "none")); byId("statesRegenerateButtons").style.display = "block"; byId("statesEditor") .querySelectorAll(".show") .forEach(el => el.classList.remove("hidden")); $("#statesEditor").dialog({position: {my: "right top", at: "right-10 top+10", of: "svg", collision: "fit"}}); } function recalculateStates(must) { if (!must && !statesAutoChange.checked) return; BurgsAndStates.expandStates(); BurgsAndStates.generateProvinces(); if (!layerIsOn("toggleStates")) toggleStates(); else drawStates(); if (!layerIsOn("toggleBorders")) toggleBorders(); else drawBorders(); if (layerIsOn("toggleProvinces")) drawProvinces(); if (adjustLabels.checked) BurgsAndStates.drawStateLabels(); refreshStatesEditor(); } function changeStatesGrowthRate() { const growthRate = +this.value; byId("statesNeutral").value = growthRate; byId("statesNeutralNumber").value = growthRate; statesNeutral = growthRate; tip("Growth rate: " + growthRate); recalculateStates(false); } function randomizeStatesExpansion() { pack.states.forEach(s => { if (!s.i || s.removed) return; const expansionism = rn(Math.random() * 4 + 1, 1); s.expansionism = expansionism; $body.querySelector("div.states[data-id='" + s.i + "'] > input.statePower").value = expansionism; }); recalculateStates(true, true); } function exitRegenerationMenu() { byId("statesBottom") .querySelectorAll(":scope > button") .forEach(el => (el.style.display = "inline-block")); byId("statesRegenerateButtons").style.display = "none"; byId("statesEditor") .querySelectorAll(".show") .forEach(el => el.classList.add("hidden")); $("#statesEditor").dialog({position: {my: "right top", at: "right-10 top+10", of: "svg", collision: "fit"}}); } function enterStatesManualAssignent() { if (!layerIsOn("toggleStates")) toggleStates(); customization = 2; statesBody.append("g").attr("id", "temp"); document.querySelectorAll("#statesBottom > button").forEach(el => (el.style.display = "none")); byId("statesManuallyButtons").style.display = "inline-block"; byId("statesHalo").style.display = "none"; byId("statesEditor") .querySelectorAll(".hide") .forEach(el => el.classList.add("hidden")); statesFooter.style.display = "none"; $body.querySelectorAll("div > input, select, span, svg").forEach(e => (e.style.pointerEvents = "none")); $("#statesEditor").dialog({position: {my: "right top", at: "right-10 top+10", of: "svg", collision: "fit"}}); tip("Click on state to select, drag the circle to change state", true); viewbox .style("cursor", "crosshair") .on("click", selectStateOnMapClick) .call(d3.drag().on("start", dragStateBrush)) .on("touchmove mousemove", moveStateBrush); $body.querySelector("div").classList.add("selected"); } function selectStateOnLineClick() { if (customization !== 2) return; if (this.parentNode.id !== "statesBodySection") return; $body.querySelector("div.selected").classList.remove("selected"); this.classList.add("selected"); } function selectStateOnMapClick() { const point = d3.mouse(this); const i = findCell(point[0], point[1]); if (pack.cells.h[i] < 20) return; const assigned = statesBody.select("#temp").select("polygon[data-cell='" + i + "']"); const state = assigned.size() ? +assigned.attr("data-state") : pack.cells.state[i]; $body.querySelector("div.selected").classList.remove("selected"); $body.querySelector("div[data-id='" + state + "']").classList.add("selected"); } function dragStateBrush() { const r = +statesManuallyBrush.value; d3.event.on("drag", () => { if (!d3.event.dx && !d3.event.dy) return; const p = d3.mouse(this); moveCircle(p[0], p[1], r); const found = r > 5 ? findAll(p[0], p[1], r) : [findCell(p[0], p[1], r)]; const selection = found.filter(isLand); if (selection) changeStateForSelection(selection); }); } // change state within selection function changeStateForSelection(selection) { const temp = statesBody.select("#temp"); const $selected = $body.querySelector("div.selected"); const stateNew = +$selected.dataset.id; const color = pack.states[stateNew].color || "#ffffff"; selection.forEach(function (i) { const exists = temp.select("polygon[data-cell='" + i + "']"); const stateOld = exists.size() ? +exists.attr("data-state") : pack.cells.state[i]; if (stateNew === stateOld) return; if (i === pack.states[stateOld].center) return; // change of append new element if (exists.size()) exists.attr("data-state", stateNew).attr("fill", color).attr("stroke", color); else temp .append("polygon") .attr("data-cell", i) .attr("data-state", stateNew) .attr("points", getPackPolygon(i)) .attr("fill", color) .attr("stroke", color); }); } function moveStateBrush() { showMainTip(); const point = d3.mouse(this); const radius = +statesManuallyBrush.value; moveCircle(point[0], point[1], radius); } function applyStatesManualAssignent() { const {cells} = pack; const affectedStates = []; const affectedProvinces = []; statesBody .select("#temp") .selectAll("polygon") .each(function () { const i = +this.dataset.cell; const c = +this.dataset.state; affectedStates.push(cells.state[i], c); affectedProvinces.push(cells.province[i]); cells.state[i] = c; if (cells.burg[i]) pack.burgs[cells.burg[i]].state = c; }); if (affectedStates.length) { refreshStatesEditor(); layerIsOn("toggleStates") ? drawStates() : toggleStates(); if (adjustLabels.checked) BurgsAndStates.drawStateLabels([...new Set(affectedStates)]); adjustProvinces([...new Set(affectedProvinces)]); layerIsOn("toggleBorders") ? drawBorders() : toggleBorders(); if (layerIsOn("toggleProvinces")) drawProvinces(); } exitStatesManualAssignment(false); } function adjustProvinces(affectedProvinces) { const {cells, provinces, states, burgs} = pack; affectedProvinces.forEach(provinceId => { if (!provinces[provinceId]) return; // lands without province captured => do nothing // find states owning at least 1 province cell const provCells = cells.i.filter(i => cells.province[i] === provinceId); const provStates = [...new Set(provCells.map(i => cells.state[i]))]; // province is captured completely => change owner or remove if (provinceId && provStates.length === 1) return changeProvinceOwner(provinceId, provStates[0], provCells); // province is captured partially => split province splitProvince(provinceId, provStates, provCells); }); function changeProvinceOwner(provinceId, newOwnerId, provinceCells) { const province = provinces[provinceId]; const prevOwner = states[province.state]; // remove province from old owner list prevOwner.provinces = prevOwner.provinces.filter(province => province !== provinceId); if (newOwnerId) { // new owner is a state => change owner province.state = newOwnerId; states[newOwnerId].provinces.push(provinceId); } else { // new owner is neutral => remove province provinces[provinceId] = {i: provinceId, removed: true}; provinceCells.forEach(i => { cells.province[i] = 0; }); } } function splitProvince(provinceId, provinceStates, provinceCells) { const province = provinces[provinceId]; const prevOwner = states[province.state]; const provinceCenterOwner = cells.state[province.center]; provinceStates.forEach(stateId => { const stateProvinceCells = provinceCells.filter(i => cells.state[i] === stateId); if (stateId === provinceCenterOwner) { // province center is owned by the same state => do nothing for this state if (stateId === prevOwner.i) return; // province center is captured by neutrals => remove province if (!stateId) { provinces[provinceId] = {i: provinceId, removed: true}; stateProvinceCells.forEach(i => { cells.province[i] = 0; }); return; } // reassign province ownership to province center owner prevOwner.provinces = prevOwner.provinces.filter(province => province !== provinceId); province.state = stateId; province.color = getMixedColor(states[stateId].color); states[stateId].provinces.push(provinceId); return; } // province cells captured by neutrals => remove captured cells from province if (!stateId) { stateProvinceCells.forEach(i => { cells.province[i] = 0; }); return; } // a few province cells owned by state => add to closes province if (stateProvinceCells.length < 20) { const closestProvince = findClosestProvince(provinceId, stateId, stateProvinceCells); if (closestProvince) { stateProvinceCells.forEach(i => { cells.province[i] = closestProvince; }); return; } } // some province cells owned by state => create new province createProvince(province, stateId, stateProvinceCells); }); } function createProvince(oldProvince, stateId, provinceCells) { const newProvinceId = provinces.length; const burgCell = provinceCells.find(i => cells.burg[i]); const center = burgCell ? burgCell : provinceCells[0]; const burgId = burgCell ? cells.burg[burgCell] : 0; const burg = burgId ? burgs[burgId] : null; const culture = cells.culture[center]; const nameByBurg = burgCell && P(0.5); const name = nameByBurg ? burg.name : oldProvince.name || Names.getState(Names.getCultureShort(culture), culture); const formOptions = ["Zone", "Area", "Territory", "Province"]; const formName = burgCell && oldProvince.formName ? oldProvince.formName : ra(formOptions); const color = getMixedColor(states[stateId].color); const kinship = nameByBurg ? 0.8 : 0.4; const type = BurgsAndStates.getType(center, burg?.port); const coa = COA.generate(burg?.coa || states[stateId].coa, kinship, burg ? null : 0.9, type); coa.shield = COA.getShield(culture, stateId); provinces.push({ i: newProvinceId, state: stateId, center, burg: burgId, name, formName, fullName: `${name} ${formName}`, color, coa }); provinceCells.forEach(i => { cells.province[i] = newProvinceId; }); states[stateId].provinces.push(newProvinceId); } function findClosestProvince(provinceId, stateId, sourceCells) { const borderCell = sourceCells.find(i => cells.c[i].some(c => { return cells.state[c] === stateId && cells.province[c] && cells.province[c] !== provinceId; }) ); const closesProvince = borderCell && cells.c[borderCell].map(c => cells.province[c]).find(province => province && province !== provinceId); return closesProvince; } } function exitStatesManualAssignment(close) { customization = 0; statesBody.select("#temp").remove(); removeCircle(); document.querySelectorAll("#statesBottom > button").forEach(el => (el.style.display = "inline-block")); byId("statesManuallyButtons").style.display = "none"; byId("statesHalo").style.display = "block"; byId("statesEditor") .querySelectorAll(".hide:not(.show)") .forEach(el => el.classList.remove("hidden")); statesFooter.style.display = "block"; $body.querySelectorAll("div > input, select, span, svg").forEach(e => (e.style.pointerEvents = "all")); if (!close) $("#statesEditor").dialog({position: {my: "right top", at: "right-10 top+10", of: "svg", collision: "fit"}}); restoreDefaultEvents(); clearMainTip(); const selected = $body.querySelector("div.selected"); if (selected) selected.classList.remove("selected"); } function enterAddStateMode() { if (this.classList.contains("pressed")) { exitAddStateMode(); return; } customization = 3; this.classList.add("pressed"); tip("Click on the map to create a new capital or promote an existing burg", true); viewbox.style("cursor", "crosshair").on("click", addState); $body.querySelectorAll("div > input, select, span, svg").forEach(e => (e.style.pointerEvents = "none")); } function addState() { const {cells, states, burgs} = pack; const point = d3.mouse(this); const center = findCell(point[0], point[1]); if (cells.h[center] < 20) return tip("You cannot place state into the water. Please click on a land cell", false, "error"); let burg = cells.burg[center]; if (burg && burgs[burg].capital) return tip("Existing capital cannot be selected as a new state capital! Select other cell", false, "error"); if (!burg) burg = addBurg(point); // add new burg const oldState = cells.state[center]; const newState = states.length; // turn burg into a capital burgs[burg].capital = 1; burgs[burg].state = newState; moveBurgToGroup(burg, "cities"); if (d3.event.shiftKey === false) exitAddStateMode(); const culture = cells.culture[center]; const basename = center % 5 === 0 ? burgs[burg].name : Names.getCulture(culture); const name = Names.getState(basename, culture); const color = getRandomColor(); const pole = cells.p[center]; // generate emblem const cultureType = pack.cultures[culture].type; const coa = COA.generate(burgs[burg].coa, 0.4, null, cultureType); coa.shield = COA.getShield(culture, null); // update diplomacy and reverse relations const diplomacy = states.map(s => { if (!s.i || s.removed) return "x"; if (!oldState) { s.diplomacy.push("Neutral"); return "Neutral"; } let relations = states[oldState].diplomacy[s.i]; // relations between Nth state and old overlord if (s.i === oldState) relations = "Enemy"; // new state is Enemy to its old overlord else if (relations === "Ally") relations = "Suspicion"; else if (relations === "Friendly") relations = "Suspicion"; else if (relations === "Suspicion") relations = "Neutral"; else if (relations === "Enemy") relations = "Friendly"; else if (relations === "Rival") relations = "Friendly"; else if (relations === "Vassal") relations = "Suspicion"; else if (relations === "Suzerain") relations = "Enemy"; s.diplomacy.push(relations); return relations; }); diplomacy.push("x"); states[0].diplomacy.push([ `Independance declaration`, `${name} declared its independance from ${states[oldState].name}` ]); cells.state[center] = newState; cells.province[center] = 0; states.push({ i: newState, name, diplomacy, provinces: [], color, expansionism: 0.5, capital: burg, type: "Generic", center, culture, military: [], alert: 1, coa, pole }); BurgsAndStates.collectStatistics(); BurgsAndStates.defineStateForms([newState]); adjustProvinces([cells.province[center]]); if (layerIsOn("toggleProvinces")) toggleProvinces(); if (!layerIsOn("toggleStates")) toggleStates(); else drawStates(); if (!layerIsOn("toggleBorders")) toggleBorders(); else drawBorders(); // add label defs .select("#textPaths") .append("path") .attr("d", `M${pole[0] - 50},${pole[1] + 6}h${100}`) .attr("id", "textPath_stateLabel" + newState); labels .select("#states") .append("text") .attr("id", "stateLabel" + newState) .append("textPath") .attr("xlink:href", "#textPath_stateLabel" + newState) .attr("startOffset", "50%") .attr("font-size", "50%") .append("tspan") .attr("x", name.length * -3) .text(name); COArenderer.add("state", newState, coa, states[newState].pole[0], states[newState].pole[1]); statesEditorAddLines(); } function exitAddStateMode() { customization = 0; restoreDefaultEvents(); clearMainTip(); $body.querySelectorAll("div > input, select, span, svg").forEach(e => (e.style.pointerEvents = "all")); if (statesAdd.classList.contains("pressed")) statesAdd.classList.remove("pressed"); } function downloadStatesCsv() { const unit = getAreaUnit("2"); const headers = `Id,State,Full Name,Form,Color,Capital,Culture,Type,Expansionism,Cells,Burgs,Area ${unit},Total Population,Rural Population,Urban Population`; const lines = Array.from($body.querySelectorAll(":scope > div")); const data = lines.map($line => { const {id, name, form, color, capital, culture, type, expansionism, cells, burgs, area, population} = $line.dataset; const {fullName = "", rural, urban} = pack.states[+id]; const ruralPopulation = Math.round(rural * populationRate); const urbanPopulation = Math.round(urban * populationRate * urbanization); return [ id, name, fullName, form, color, capital, culture, type, expansionism, cells, burgs, area, population, ruralPopulation, urbanPopulation ].join(","); }); const csvData = [headers].concat(data).join("\n"); const name = getFileName("States") + ".csv"; downloadFile(csvData, name); } function closeStatesEditor() { if (customization === 2) exitStatesManualAssignment(true); if (customization === 3) exitAddStateMode(); debug.selectAll(".highlight").remove(); $body.innerHTML = ""; }