// UI module to control the style "use strict"; // add available filters to lists { const filters = Array.from(byId("filters").querySelectorAll("filter")); const emptyOption = ''; const options = filters.map(filter => { const id = filter.getAttribute("id"); const name = filter.getAttribute("name"); return ``; }); const allOptions = emptyOption + options.join(""); byId("styleFilterInput").innerHTML = allOptions; byId("styleStatesBodyFilter").innerHTML = allOptions; } // store some style inputs as options styleElements.addEventListener("change", function (ev) { if (ev.target.dataset.stored) lock(ev.target.dataset.stored); }); // select element to be edited function editStyle(element, group) { showOptions(); styleTab.click(); styleElementSelect.value = element; if (group) styleGroupSelect.options.add(new Option(group, group, true, true)); selectStyleElement(); styleElementSelect.classList.add("glow"); if (group) styleGroupSelect.classList.add("glow"); setTimeout(() => { styleElementSelect.classList.remove("glow"); if (group) styleGroupSelect.classList.remove("glow"); }, 1500); } // Color schemes const heightmapColorSchemes = { bright: d3.scaleSequential(d3.interpolateSpectral), light: d3.scaleSequential(d3.interpolateRdYlGn), natural: d3.scaleSequential(d3.interpolateRgbBasis(["white", "#EEEECC", "tan", "green", "teal"])), green: d3.scaleSequential(d3.interpolateGreens), olive: d3.scaleSequential(d3.interpolateRgbBasis(["#ffffff", "#cea48d", "#d5b085", "#0c2c19", "#151320"])), livid: d3.scaleSequential(d3.interpolateRgbBasis(["#BBBBDD", "#2A3440", "#17343B", "#0A1E24"])), monochrome: d3.scaleSequential(d3.interpolateGreys) }; // add default color schemes to the list of options byId("styleHeightmapScheme").innerHTML = Object.keys(heightmapColorSchemes) .map(scheme => ``) .join(""); function addCustomColorScheme(scheme) { const stops = scheme.split(","); heightmapColorSchemes[scheme] = d3.scaleSequential(d3.interpolateRgbBasis(stops)); byId("styleHeightmapScheme").options.add(new Option(scheme, scheme, false, true)); } function getColorScheme(scheme = "bright") { if (!(scheme in heightmapColorSchemes)) { const colors = scheme.split(","); heightmapColorSchemes[scheme] = d3.scaleSequential(d3.interpolateRgbBasis(colors)); } return heightmapColorSchemes[scheme]; } // Toggle style sections on element select styleElementSelect.addEventListener("change", selectStyleElement); function selectStyleElement() { const sel = styleElementSelect.value; let el = d3.select("#" + sel); styleElements.querySelectorAll("tbody").forEach(e => (e.style.display = "none")); // hide all sections // show alert line if layer is not visible const isLayerOff = sel !== "ocean" && (el.style("display") === "none" || !el.selectAll("*").size()); styleIsOff.style.display = isLayerOff ? "block" : "none"; // active group element const group = styleGroupSelect.value; if (["routes", "labels", "coastline", "lakes", "anchors", "burgIcons", "borders"].includes(sel)) { const gEl = group && el.select("#" + group); el = group && gEl.size() ? gEl : el.select("g"); } // opacity if (!["landmass", "ocean", "regions", "legend"].includes(sel)) { styleOpacity.style.display = "block"; styleOpacityInput.value = styleOpacityOutput.value = el.attr("opacity") || 1; } // filter if (!["landmass", "legend", "regions"].includes(sel)) { styleFilter.style.display = "block"; styleFilterInput.value = el.attr("filter") || ""; } // fill if (["rivers", "lakes", "landmass", "prec", "ice", "fogging"].includes(sel)) { styleFill.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill"); } // stroke color and width if ( [ "armies", "routes", "lakes", "borders", "cults", "relig", "cells", "coastline", "prec", "ice", "icons", "coordinates", "zones", "gridOverlay" ].includes(sel) ) { styleStroke.style.display = "block"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke"); styleStrokeWidth.style.display = "block"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || ""; } // stroke dash if ( ["routes", "borders", "temperature", "legend", "population", "coordinates", "zones", "gridOverlay"].includes(sel) ) { styleStrokeDash.style.display = "block"; styleStrokeDasharrayInput.value = el.attr("stroke-dasharray") || ""; styleStrokeLinecapInput.value = el.attr("stroke-linecap") || "inherit"; } // clipping if ( [ "cells", "gridOverlay", "coordinates", "compass", "terrain", "temperature", "routes", "texture", "biomes", "zones" ].includes(sel) ) { styleClipping.style.display = "block"; styleClippingInput.value = el.attr("mask") || ""; } // show specific sections if (sel === "texture") styleTexture.style.display = "block"; if (sel === "terrs") { styleHeightmap.style.display = "block"; styleHeightmapScheme.value = terrs.attr("scheme"); styleHeightmapTerracingInput.value = styleHeightmapTerracingOutput.value = terrs.attr("terracing"); styleHeightmapSkipInput.value = styleHeightmapSkipOutput.value = terrs.attr("skip"); styleHeightmapSimplificationInput.value = styleHeightmapSimplificationOutput.value = terrs.attr("relax"); styleHeightmapCurve.value = terrs.attr("curve"); } if (sel === "markers") { styleMarkers.style.display = "block"; styleRescaleMarkers.checked = +markers.attr("rescale"); } if (sel === "gridOverlay") { styleGrid.style.display = "block"; styleGridType.value = el.attr("type"); styleGridScale.value = el.attr("scale") || 1; styleGridShiftX.value = el.attr("dx") || 0; styleGridShiftY.value = el.attr("dy") || 0; calculateFriendlyGridSize(); } if (sel === "compass") { styleCompass.style.display = "block"; const tr = parseTransform(compass.select("use").attr("transform")); styleCompassShiftX.value = tr[0]; styleCompassShiftY.value = tr[1]; styleCompassSizeInput.value = styleCompassSizeOutput.value = tr[2]; } if (sel === "terrain") { styleRelief.style.display = "block"; styleReliefSizeOutput.innerHTML = styleReliefSizeInput.value = terrain.attr("size"); styleReliefDensityOutput.innerHTML = styleReliefDensityInput.value = terrain.attr("density"); styleReliefSet.value = terrain.attr("set"); } if (sel === "population") { stylePopulation.style.display = "block"; stylePopulationRuralStrokeInput.value = stylePopulationRuralStrokeOutput.value = population .select("#rural") .attr("stroke"); stylePopulationUrbanStrokeInput.value = stylePopulationUrbanStrokeOutput.value = population .select("#urban") .attr("stroke"); styleStrokeWidth.style.display = "block"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || ""; } if (sel === "regions") { styleStates.style.display = "block"; styleStatesBodyOpacity.value = styleStatesBodyOpacityOutput.value = statesBody.attr("opacity") || 1; styleStatesBodyFilter.value = statesBody.attr("filter") || ""; styleStatesHaloWidth.value = styleStatesHaloWidthOutput.value = statesHalo.attr("data-width") || 10; styleStatesHaloOpacity.value = styleStatesHaloOpacityOutput.value = statesHalo.attr("opacity") || 1; const blur = parseFloat(statesHalo.attr("filter")?.match(/blur\(([^)]+)\)/)?.[1]) || 0; styleStatesHaloBlur.value = styleStatesHaloBlurOutput.value = blur; } if (sel === "labels") { styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleShadow.style.display = "block"; styleSize.style.display = "block"; styleVisibility.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#3e3e4b"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#3a3a3a"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || 0; styleShadowInput.value = el.style("text-shadow") || "white 0 0 4px"; styleFont.style.display = "block"; styleSelectFont.value = el.attr("font-family"); styleFontSize.value = el.attr("data-size"); } if (sel === "provs") { styleFill.style.display = "block"; styleSize.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#111111"; styleFont.style.display = "block"; styleSelectFont.value = el.attr("font-family"); styleFontSize.value = el.attr("data-size"); } if (sel == "burgIcons") { styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleStrokeDash.style.display = "block"; styleRadius.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#ffffff"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#3e3e4b"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || 0.24; styleStrokeDasharrayInput.value = el.attr("stroke-dasharray") || ""; styleStrokeLinecapInput.value = el.attr("stroke-linecap") || "inherit"; styleRadiusInput.value = el.attr("size") || 1; } if (sel == "anchors") { styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleIconSize.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#ffffff"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#3e3e4b"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || 0.24; styleIconSizeInput.value = el.attr("size") || 2; } if (sel === "legend") { styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleSize.style.display = "block"; styleLegend.style.display = "block"; styleLegendColItemsOutput.value = styleLegendColItems.value = el.attr("data-columns"); styleLegendBackOutput.value = styleLegendBack.value = el.select("#legendBox").attr("fill"); styleLegendOpacityOutput.value = styleLegendOpacity.value = el.select("#legendBox").attr("fill-opacity"); styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#111111"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || 0.5; styleFont.style.display = "block"; styleSelectFont.value = el.attr("font-family"); styleFontSize.value = el.attr("data-size"); } if (sel === "ocean") { styleOcean.style.display = "block"; styleOceanFill.value = styleOceanFillOutput.value = oceanLayers.select("#oceanBase").attr("fill"); styleOceanPattern.value = byId("oceanicPattern")?.getAttribute("href"); styleOceanPatternOpacity.value = styleOceanPatternOpacityOutput.value = byId("oceanicPattern").getAttribute("opacity") || 1; outlineLayers.value = oceanLayers.attr("layers"); } if (sel === "temperature") { styleStrokeWidth.style.display = "block"; styleTemperature.style.display = "block"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || ""; styleTemperatureFillOpacityInput.value = styleTemperatureFillOpacityOutput.value = el.attr("fill-opacity") || 0.1; styleTemperatureFillInput.value = styleTemperatureFillOutput.value = el.attr("fill") || "#000"; styleTemperatureFontSizeInput.value = styleTemperatureFontSizeOutput.value = el.attr("font-size") || "8px"; } if (sel === "coordinates") { styleSize.style.display = "block"; styleFontSize.value = el.attr("data-size"); } if (sel === "armies") { styleArmies.style.display = "block"; styleArmiesFillOpacity.value = styleArmiesFillOpacityOutput.value = el.attr("fill-opacity"); styleArmiesSize.value = styleArmiesSizeOutput.value = el.attr("box-size"); } if (sel === "emblems") { styleEmblems.style.display = "block"; styleStrokeWidth.style.display = "block"; styleStrokeWidthInput.value = styleStrokeWidthOutput.value = el.attr("stroke-width") || 1; } // update group options styleGroupSelect.options.length = 0; // remove all options if (["routes", "labels", "coastline", "lakes", "anchors", "burgIcons", "borders"].includes(sel)) { const groups = byId(sel).querySelectorAll("g"); groups.forEach(el => { if (el.id === "burgLabels") return; const option = new Option(`${el.id} (${el.childElementCount})`, el.id, false, false); styleGroupSelect.options.add(option); }); styleGroupSelect.value = el.attr("id"); styleGroup.style.display = "block"; } else { styleGroupSelect.options.add(new Option(sel, sel, false, true)); styleGroup.style.display = "none"; } if (sel === "coastline" && styleGroupSelect.value === "sea_island") { styleCoastline.style.display = "block"; const auto = (styleCoastlineAuto.checked = coastline.select("#sea_island").attr("auto-filter")); if (auto) styleFilter.style.display = "none"; } } // Handle style inputs change styleGroupSelect.addEventListener("change", selectStyleElement); function getEl() { const el = styleElementSelect.value; const g = styleGroupSelect.value; if (g === el || g === "") return svg.select("#" + el); else return svg.select("#" + el).select("#" + g); } styleFillInput.addEventListener("input", function () { styleFillOutput.value = this.value; getEl().attr("fill", this.value); }); styleStrokeInput.addEventListener("input", function () { styleStrokeOutput.value = this.value; getEl().attr("stroke", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleStrokeWidthInput.addEventListener("input", function () { styleStrokeWidthOutput.value = this.value; getEl().attr("stroke-width", +this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleStrokeDasharrayInput.addEventListener("input", function () { getEl().attr("stroke-dasharray", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleStrokeLinecapInput.addEventListener("change", function () { getEl().attr("stroke-linecap", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleOpacityInput.addEventListener("input", function () { styleOpacityOutput.value = this.value; getEl().attr("opacity", this.value); }); styleFilterInput.addEventListener("change", function () { if (styleGroupSelect.value === "ocean") return oceanLayers.attr("filter", this.value); getEl().attr("filter", this.value); }); styleTextureInput.addEventListener("change", function () { texture.select("image").attr("src", this.value); if (layerIsOn("toggleTexture")) texture.select("image").attr("href", this.value); zoom.scaleBy(svg, 1.00001); }); styleTextureShiftX.addEventListener("input", function () { texture .select("image") .attr("x", this.value) .attr("width", graphWidth - this.valueAsNumber); }); styleTextureShiftY.addEventListener("input", function () { texture .select("image") .attr("y", this.value) .attr("height", graphHeight - this.valueAsNumber); }); styleClippingInput.addEventListener("change", function () { getEl().attr("mask", this.value); }); styleGridType.addEventListener("change", function () { getEl().attr("type", this.value); if (layerIsOn("toggleGrid")) drawGrid(); calculateFriendlyGridSize(); }); styleGridScale.addEventListener("input", function () { getEl().attr("scale", this.value); if (layerIsOn("toggleGrid")) drawGrid(); calculateFriendlyGridSize(); }); function calculateFriendlyGridSize() { const size = styleGridScale.value * 25; const friendly = `${rn(size * distanceScaleInput.value, 2)} ${distanceUnitInput.value}`; styleGridSizeFriendly.value = friendly; } styleGridShiftX.addEventListener("input", function () { getEl().attr("dx", this.value); if (layerIsOn("toggleGrid")) drawGrid(); }); styleGridShiftY.addEventListener("input", function () { getEl().attr("dy", this.value); if (layerIsOn("toggleGrid")) drawGrid(); }); styleShiftX.addEventListener("input", shiftElement); styleShiftY.addEventListener("input", shiftElement); function shiftElement() { const x = styleShiftX.value || 0; const y = styleShiftY.value || 0; getEl().attr("transform", `translate(${x},${y})`); } styleRescaleMarkers.addEventListener("change", function () { markers.attr("rescale", +this.checked); invokeActiveZooming(); }); styleCoastlineAuto.addEventListener("change", function () { coastline.select("#sea_island").attr("auto-filter", +this.checked); styleFilter.style.display = this.checked ? "none" : "block"; invokeActiveZooming(); }); styleOceanFill.addEventListener("input", function () { oceanLayers.select("rect").attr("fill", this.value); styleOceanFillOutput.value = this.value; }); styleOceanPattern.addEventListener("change", function () { byId("oceanicPattern")?.setAttribute("href", this.value); }); styleOceanPatternOpacity.addEventListener("input", function () { byId("oceanicPattern").setAttribute("opacity", this.value); styleOceanPatternOpacityOutput.value = this.value; }); outlineLayers.addEventListener("change", function () { oceanLayers.selectAll("path").remove(); oceanLayers.attr("layers", this.value); OceanLayers(); }); styleHeightmapScheme.addEventListener("change", function () { terrs.attr("scheme", this.value); drawHeightmap(); }); openCreateHeightmapSchemeButton.addEventListener("click", function () { // start with current scheme this.dataset.stops = terrs.attr("scheme").startsWith("#") ? terrs.attr("scheme") : (function () { const scheme = heightmapColorSchemes[terrs.attr("scheme")]; return [0, 0.25, 0.5, 0.75, 1].map(scheme).map(toHEX).join(","); })(); // render dialog base structure alertMessage.innerHTML = /* html */ `