// 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; byId("styleScaleBarBackgroundFilter").innerHTML = allOptions; } // store some style inputs as options styleElements.on("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]; } function getColor(value, scheme = getColorScheme("bright")) { return scheme(1 - (value < 20 ? value - 5 : value) / 100); } // Toggle style sections on element select styleElementSelect.on("change", selectStyleElement); function selectStyleElement() { const styleElement = styleElementSelect.value; let el = d3.select("#" + styleElement); styleElements.querySelectorAll("tbody").forEach(e => (e.style.display = "none")); // hide all sections // show alert line if layer is not visible const isLayerOff = styleElement !== "ocean" && (el.style("display") === "none" || !el.selectAll("*").size()); styleIsOff.style.display = isLayerOff ? "block" : "none"; // active group element if (["routes", "labels", "coastline", "lakes", "anchors", "burgIcons", "borders", "terrs"].includes(styleElement)) { const group = styleGroupSelect.value; const defaultGroupSelector = styleElement === "terrs" ? "#landHeights" : "g"; el = group && el.select("#" + group).size() ? el.select("#" + group) : el.select(defaultGroupSelector); } // opacity if (!["landmass", "ocean", "regions", "legend"].includes(styleElement)) { styleOpacity.style.display = "block"; styleOpacityInput.value = el.attr("opacity") || 1; } // filter if (!["landmass", "legend", "regions", "scaleBar"].includes(styleElement)) { styleFilter.style.display = "block"; styleFilterInput.value = el.attr("filter") || ""; } // fill if (["rivers", "lakes", "landmass", "prec", "ice", "fogging", "scaleBar", "vignette"].includes(styleElement)) { styleFill.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill"); } // stroke color and width if ( [ "armies", "biomes", "borders", "cells", "coastline", "coordinates", "cults", "gridOverlay", "ice", "icons", "lakes", "prec", "relig", "routes", "zones" ].includes(styleElement) ) { styleStroke.style.display = "block"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke"); styleStrokeWidth.style.display = "block"; styleStrokeWidthInput.value = el.attr("stroke-width") || 0; } // stroke dash if ( [ "borders", "cells", "coordinates", "gridOverlay", "legend", "population", "routes", "temperature", "zones" ].includes(styleElement) ) { styleStrokeDash.style.display = "block"; styleStrokeDasharrayInput.value = el.attr("stroke-dasharray") || ""; styleStrokeLinecapInput.value = el.attr("stroke-linecap") || "inherit"; } // clipping if ( [ "biomes", "cells", "compass", "coordinates", "gridOverlay", "population", "prec", "routes", "temperature", "terrain", "texture", "zones" ].includes(styleElement) ) { styleClipping.style.display = "block"; styleClippingInput.value = el.attr("mask") || ""; } // show specific sections if (styleElement === "texture") { styleTexture.style.display = "block"; styleTextureShiftX.value = el.attr("data-x") || 0; styleTextureShiftY.value = el.attr("data-y") || 0; updateTextureSelectValue(el.attr("data-href")); } if (styleElement === "terrs") { styleHeightmap.style.display = "block"; styleHeightmapRenderOceanOption.style.display = el.attr("id") === "oceanHeights" ? "block" : "none"; styleHeightmapRenderOcean.checked = +el.attr("data-render"); styleHeightmapScheme.value = el.attr("scheme"); styleHeightmapTerracing.value = el.attr("terracing"); styleHeightmapSkip.value = el.attr("skip"); styleHeightmapSimplification.value = el.attr("relax"); styleHeightmapCurve.value = el.attr("curve"); } if (styleElement === "markers") { styleMarkers.style.display = "block"; styleRescaleMarkers.checked = +markers.attr("rescale"); } if (styleElement === "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 (styleElement === "compass") { styleCompass.style.display = "block"; const tr = parseTransform(compass.select("use").attr("transform")); styleCompassShiftX.value = tr[0]; styleCompassShiftY.value = tr[1]; styleCompassSizeInput.value = tr[2]; } if (styleElement === "terrain") { styleRelief.style.display = "block"; styleReliefSize.value = terrain.attr("size") || 1; styleReliefDensity.value = terrain.attr("density") || 0.4; styleReliefSet.value = terrain.attr("set"); } if (styleElement === "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 = el.attr("stroke-width") || 0; } if (styleElement === "regions") { styleStates.style.display = "block"; styleStatesBodyOpacity.value = statesBody.attr("opacity") || 1; styleStatesBodyFilter.value = statesBody.attr("filter") || ""; styleStatesHaloWidth.value = statesHalo.attr("data-width") || 10; styleStatesHaloOpacity.value = statesHalo.attr("opacity") || 1; styleStatesHaloBlur.value = parseFloat(statesHalo.attr("filter")?.match(/blur\(([^)]+)\)/)?.[1]) || 0; } if (styleElement === "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("font-size"); } if (styleElement === "labels") { styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleLetterSpacing.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 = el.attr("stroke-width") || 0; styleLetterSpacingInput.value = el.attr("letter-spacing") || 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 (el.node().parentNode.id === "burgLabels") { styleFontShift.style.display = "block"; styleFontShiftX.value = el.attr("data-dx") || 0; styleFontShiftY.value = el.attr("data-dy") || 0; } } if (styleElement == "burgIcons") { styleBurgIcons.style.display = "block"; styleBurgIconsIcon.value = el.attr("data-icon"); styleBurgIconsIconSize.value = el.attr("font-size"); styleBurgIconsStrokeLinejoin.value = el.attr("stroke-linejoin"); styleBurgIconsFillOpacity.value = el.attr("fill-opacity"); styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleStrokeDash.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#ffffff"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#3e3e4b"; styleStrokeWidthInput.value = el.attr("stroke-width") || 0.24; styleStrokeDasharrayInput.value = el.attr("stroke-dasharray") || ""; styleStrokeLinecapInput.value = el.attr("stroke-linecap") || "inherit"; } if (styleElement == "anchors") { styleFill.style.display = "block"; styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleFillInput.value = styleFillOutput.value = el.attr("fill") || "#ffffff"; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#3e3e4b"; styleStrokeWidthInput.value = el.attr("stroke-width") || 0.24; } if (styleElement === "legend") { styleStroke.style.display = "block"; styleStrokeWidth.style.display = "block"; styleSize.style.display = "block"; styleLegend.style.display = "block"; styleLegendColItems.value = el.attr("data-columns"); const legendBox = el.select("#legendBox"); styleLegendBack.value = styleLegendBackOutput.value = legendBox.size() ? legendBox.attr("fill") : "#ffffff"; styleLegendOpacity.value = legendBox.size() ? legendBox.attr("fill-opacity") : 1; styleStrokeInput.value = styleStrokeOutput.value = el.attr("stroke") || "#111111"; styleStrokeWidthInput.value = el.attr("stroke-width") || 0.5; styleFont.style.display = "block"; styleSelectFont.value = el.attr("font-family"); styleFontSize.value = el.attr("data-size"); } if (styleElement === "ocean") { styleOcean.style.display = "block"; styleOceanFill.value = styleOceanFillOutput.value = oceanLayers.select("#oceanBase").attr("fill"); styleOceanPattern.value = byId("oceanicPattern")?.getAttribute("href"); styleOceanPatternOpacity.value = byId("oceanicPattern").getAttribute("opacity") || 1; outlineLayers.value = oceanLayers.attr("layers"); } if (styleElement === "temperature") { styleStrokeWidth.style.display = "block"; styleTemperature.style.display = "block"; styleStrokeWidthInput.value = el.attr("stroke-width") || ""; styleTemperatureFillOpacityInput.value = el.attr("fill-opacity") || 0.1; styleTemperatureFillInput.value = styleTemperatureFillOutput.value = el.attr("fill") || "#000"; styleTemperatureFontSizeInput.value = el.attr("font-size") || "8px"; } if (styleElement === "coordinates") { styleSize.style.display = "block"; styleFontSize.value = el.attr("data-size"); } if (styleElement === "armies") { styleArmies.style.display = "block"; styleArmiesFillOpacity.value = el.attr("fill-opacity"); styleArmiesSize.value = el.attr("box-size"); } if (styleElement === "emblems") { styleEmblems.style.display = "block"; styleStrokeWidth.style.display = "block"; styleStrokeWidthInput.value = el.attr("stroke-width") || 1; emblemsStateSizeInput.value = emblems.select("#stateEmblems").attr("data-size") || 1; emblemsProvinceSizeInput.value = emblems.select("#provinceEmblems").attr("data-size") || 1; emblemsBurgSizeInput.value = emblems.select("#burgEmblems").attr("data-size") || 1; } // update group options styleGroupSelect.options.length = 0; // remove all options if (["routes", "labels", "coastline", "lakes", "anchors", "burgIcons", "borders", "terrs"].includes(styleElement)) { const groups = byId(styleElement).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(styleElement, styleElement, false, true)); styleGroup.style.display = "none"; } if (styleElement === "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"; } if (styleElement === "scaleBar") { styleScaleBar.style.display = "block"; styleScaleBarSize.value = el.attr("data-bar-size"); styleScaleBarFontSize.value = el.attr("font-size"); styleScaleBarPositionX.value = el.attr("data-x") || "99"; styleScaleBarPositionY.value = el.attr("data-y") || "99"; styleScaleBarLabel.value = el.attr("data-label") || ""; const scaleBarBack = el.select("#scaleBarBack"); if (scaleBarBack.size()) { styleScaleBarBackgroundOpacity.value = scaleBarBack.attr("opacity"); styleScaleBarBackgroundFill.value = styleScaleBarBackgroundFillOutput.value = scaleBarBack.attr("fill"); styleScaleBarBackgroundStroke.value = styleScaleBarBackgroundStrokeOutput.value = scaleBarBack.attr("stroke"); styleScaleBarBackgroundStrokeWidth.value = scaleBarBack.attr("stroke-width"); styleScaleBarBackgroundFilter.value = scaleBarBack.attr("filter"); styleScaleBarBackgroundPaddingTop.value = scaleBarBack.attr("data-top"); styleScaleBarBackgroundPaddingRight.value = scaleBarBack.attr("data-right"); styleScaleBarBackgroundPaddingBottom.value = scaleBarBack.attr("data-bottom"); styleScaleBarBackgroundPaddingLeft.value = scaleBarBack.attr("data-left"); } } if (styleElement === "vignette") { styleVignette.style.display = "block"; const maskRect = byId("vignette-rect"); if (maskRect) { const digit = str => str.replace(/[^\d.]/g, ""); styleVignetteX.value = digit(maskRect.getAttribute("x")); styleVignetteY.value = digit(maskRect.getAttribute("y")); styleVignetteWidth.value = digit(maskRect.getAttribute("width")); styleVignetteHeight.value = digit(maskRect.getAttribute("height")); styleVignetteRx.value = digit(maskRect.getAttribute("rx")); styleVignetteRy.value = digit(maskRect.getAttribute("ry")); styleVignetteBlur.value = digit(maskRect.getAttribute("filter")); } } } // Handle style inputs change styleGroupSelect.on("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.on("input", function () { styleFillOutput.value = this.value; getEl().attr("fill", this.value); }); styleStrokeInput.on("input", function () { styleStrokeOutput.value = this.value; getEl().attr("stroke", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleStrokeWidthInput.on("input", e => { getEl().attr("stroke-width", e.target.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleLetterSpacingInput.on("input", e => { getEl().attr("letter-spacing", e.target.value); }); styleStrokeDasharrayInput.on("input", function () { getEl().attr("stroke-dasharray", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleStrokeLinecapInput.on("change", function () { getEl().attr("stroke-linecap", this.value); if (styleElementSelect.value === "gridOverlay" && layerIsOn("toggleGrid")) drawGrid(); }); styleOpacityInput.on("input", e => { getEl().attr("opacity", e.target.value); }); styleFilterInput.on("change", function () { if (styleGroupSelect.value === "ocean") return oceanLayers.attr("filter", this.value); getEl().attr("filter", this.value); }); styleTextureInput.on("change", function () { changeTexture(this.value); }); function changeTexture(href) { texture.attr("data-href", href); texture.select("image").attr("href", href); } function updateTextureSelectValue(href) { const isAdded = Array.from(styleTextureInput.options).some(option => option.value === href); if (isAdded) { styleTextureInput.value = href; } else { const name = href.split("/").pop().slice(0, 20); styleTextureInput.add(new Option(name, href, false, true)); } } styleTextureShiftX.on("input", function () { texture.attr("data-x", this.value); texture .select("image") .attr("x", this.value) .attr("width", graphWidth - this.valueAsNumber); }); styleTextureShiftY.on("input", function () { texture.attr("data-y", this.value); texture .select("image") .attr("y", this.value) .attr("height", graphHeight - this.valueAsNumber); }); styleClippingInput.on("change", function () { getEl().attr("mask", this.value); }); styleGridType.on("change", function () { getEl().attr("type", this.value); if (layerIsOn("toggleGrid")) drawGrid(); calculateFriendlyGridSize(); }); styleGridScale.on("input", function () { getEl().attr("scale", this.value); if (layerIsOn("toggleGrid")) drawGrid(); calculateFriendlyGridSize(); }); function calculateFriendlyGridSize() { const size = styleGridScale.value * 25; const friendly = `${rn(size * distanceScale, 2)} ${distanceUnitInput.value}`; styleGridSizeFriendly.value = friendly; } styleGridShiftX.on("input", function () { getEl().attr("dx", this.value); if (layerIsOn("toggleGrid")) drawGrid(); }); styleGridShiftY.on("input", function () { getEl().attr("dy", this.value); if (layerIsOn("toggleGrid")) drawGrid(); }); styleRescaleMarkers.on("change", function () { markers.attr("rescale", +this.checked); invokeActiveZooming(); }); styleCoastlineAuto.on("change", function () { coastline.select("#sea_island").attr("auto-filter", +this.checked); styleFilter.style.display = this.checked ? "none" : "block"; invokeActiveZooming(); }); styleOceanFill.on("input", function () { oceanLayers.select("rect").attr("fill", this.value); styleOceanFillOutput.value = this.value; }); styleOceanPattern.on("change", function () { byId("oceanicPattern")?.setAttribute("href", this.value); }); styleOceanPatternOpacity.on("input", e => { byId("oceanicPattern").setAttribute("opacity", e.target.value); }); outlineLayers.on("change", function () { oceanLayers.selectAll("path").remove(); oceanLayers.attr("layers", this.value); OceanLayers(); }); styleHeightmapScheme.on("change", function () { getEl().attr("scheme", this.value); drawHeightmap(); }); openCreateHeightmapSchemeButton.on("click", function () { // start with current scheme const scheme = getEl().attr("scheme"); this.dataset.stops = scheme.startsWith("#") ? scheme : (() => [0, 0.25, 0.5, 0.75, 1].map(heightmapColorSchemes[scheme]).map(toHEX).join(","))(); // render dialog base structure alertMessage.innerHTML = /* html */ `
Define heightmap gradient colors from high to low altitude heightmap preview
`; renderPreview(); renderStops(); renderGradient(); function renderPreview() { const stops = openCreateHeightmapSchemeButton.dataset.stops.split(","); const scheme = d3.scaleSequential(d3.interpolateRgbBasis(stops)); const preview = drawHeights({ heights: grid.cells.h, width: grid.cellsX, height: grid.cellsY, scheme, renderOcean: false }); byId("heightmapSchemePreview").src = preview; } function renderStops() { const stops = openCreateHeightmapSchemeButton.dataset.stops.split(","); const colorInput = color => ``; const removeStopButton = index => ``; const addStopButton = () => ``; const container = byId("heightmapSchemeStops"); container.innerHTML = stops .map( (stop, index) => `${colorInput(stop)} ${index && index < stops.length - 1 ? removeStopButton(index) : ""}` ) .join(addStopButton()); Array.from(container.querySelectorAll("input.stop")).forEach( (input, index) => (input.oninput = function () { stops[index] = this.value; openCreateHeightmapSchemeButton.dataset.stops = stops.join(","); renderPreview(); renderGradient(); }) ); Array.from(container.querySelectorAll("button.remove")).forEach( button => (button.onclick = function () { const index = +this.dataset.index; stops.splice(index, 1); openCreateHeightmapSchemeButton.dataset.stops = stops.join(","); renderPreview(); renderStops(); renderGradient(); }) ); Array.from(container.querySelectorAll("button.add")).forEach( (button, index) => (button.onclick = function () { const middleColor = d3.interpolateRgb(stops[index], stops[index + 1])(0.5); stops.splice(index + 1, 0, toHEX(middleColor)); openCreateHeightmapSchemeButton.dataset.stops = stops.join(","); renderPreview(); renderStops(); renderGradient(); }) ); } function renderGradient() { const stops = openCreateHeightmapSchemeButton.dataset.stops; byId("heightmapSchemeGradient").style.background = `linear-gradient(to right, ${stops})`; } function handleCreate() { const stops = openCreateHeightmapSchemeButton.dataset.stops; if (stops in heightmapColorSchemes) return tip("This scheme already exists", false, "error"); addCustomColorScheme(stops); getEl().attr("scheme", stops); drawHeightmap(); handleClose(); } function handleClose() { $("#alert").dialog("close"); } $("#alert").dialog({ resizable: false, title: "Create heightmap color scheme", width: "28em", buttons: { Create: handleCreate, Cancel: handleClose }, position: {my: "center top+150", at: "center top", of: "svg"} }); }); styleHeightmapRenderOcean.on("change", e => { const checked = +e.target.checked; getEl().attr("data-render", checked); drawHeightmap(); }); styleHeightmapTerracing.on("input", e => { getEl().attr("terracing", e.target.value); drawHeightmap(); }); styleHeightmapSkip.on("input", e => { getEl().attr("skip", e.target.value); drawHeightmap(); }); styleHeightmapSimplification.on("input", e => { getEl().attr("relax", e.target.value); drawHeightmap(); }); styleHeightmapCurve.on("change", e => { getEl().attr("curve", e.target.value); drawHeightmap(); }); styleReliefSet.on("change", e => { terrain.attr("set", e.target.value); ReliefIcons.draw(); if (!layerIsOn("toggleRelief")) toggleRelief(); }); styleReliefSize.on("change", e => { terrain.attr("size", e.target.value); ReliefIcons.draw(); if (!layerIsOn("toggleRelief")) toggleRelief(); }); styleReliefDensity.on("change", e => { terrain.attr("density", e.target.value); ReliefIcons.draw(); if (!layerIsOn("toggleRelief")) toggleRelief(); }); styleTemperatureFillOpacityInput.on("input", e => { temperature.attr("fill-opacity", e.target.value); }); styleTemperatureFontSizeInput.on("input", e => { temperature.attr("font-size", e.target.value + "px"); }); styleTemperatureFillInput.on("input", e => { temperature.attr("fill", e.target.value); styleTemperatureFillOutput.value = e.target.value; }); stylePopulationRuralStrokeInput.on("input", e => { population.select("#rural").attr("stroke", e.target.value); stylePopulationRuralStrokeOutput.value = e.target.value; }); stylePopulationUrbanStrokeInput.on("input", e => { population.select("#urban").attr("stroke", e.target.value); stylePopulationUrbanStrokeOutput.value = e.target.value; }); styleBurgIconsIcon.on("change", e => { getEl() .attr("data-icon", e.target.value) .selectAll("use") .attr("href", d => e.target.value); }); styleBurgIconsIconSize.on("input", e => { getEl().attr("font-size", e.target.value); }); styleBurgIconsStrokeLinejoin.on("change", e => { getEl().attr("stroke-linejoin", e.target.value); }); styleBurgIconsFillOpacity.on("input", e => { getEl().attr("fill-opacity", e.target.value); }); styleCompassSizeInput.on("input", shiftCompass); styleCompassShiftX.on("input", shiftCompass); styleCompassShiftY.on("input", shiftCompass); function shiftCompass() { const tr = `translate(${styleCompassShiftX.value} ${styleCompassShiftY.value}) scale(${styleCompassSizeInput.value})`; compass.select("use").attr("transform", tr); } styleLegendColItems.on("input", e => { legend.select("#legendBox").attr("data-columns", e.target.value); redrawLegend(); }); styleLegendBack.on("input", e => { styleLegendBackOutput.value = e.target.value; legend.select("#legendBox").attr("fill", e.target.value); }); styleLegendOpacity.on("input", e => { legend.select("#legendBox").attr("fill-opacity", e.target.value); }); styleSelectFont.on("change", changeFont); function changeFont() { const family = styleSelectFont.value; getEl().attr("font-family", family); if (styleElementSelect.value === "legend") redrawLegend(); } styleShadowInput.on("input", function () { getEl().style("text-shadow", this.value); }); styleFontAdd.on("click", function () { addFontNameInput.value = ""; addFontURLInput.value = ""; $("#addFontDialog").dialog({ title: "Add custom font", width: "26em", position: {my: "center", at: "center", of: "svg"}, buttons: { Add: function () { const family = addFontNameInput.value; const src = addFontURLInput.value; const method = addFontMethod.value; if (!family) return tip("Please provide a font name", false, "error"); const existingFont = method === "fontURL" ? fonts.find(font => font.family === family && font.src === src) : fonts.find(font => font.family === family); if (existingFont) return tip("The font is already added", false, "error"); if (method === "fontURL") addWebFont(family, src); else if (method === "googleFont") addGoogleFont(family); else if (method === "localFont") addLocalFont(family); addFontNameInput.value = ""; addFontURLInput.value = ""; $(this).dialog("close"); }, Cancel: function () { $(this).dialog("close"); } } }); }); addFontMethod.on("change", function () { addFontURLInput.style.display = this.value === "fontURL" ? "inline" : "none"; }); styleFontSize.on("change", function () { changeFontSize(getEl(), +this.value); }); styleFontPlus.on("click", function () { const current = +styleFontSize.value || 12; changeFontSize(getEl(), Math.min(current + 1, 999)); }); styleFontMinus.on("click", function () { const current = +styleFontSize.value || 12; changeFontSize(getEl(), Math.max(current - 1, 1)); }); function changeFontSize(el, size) { styleFontSize.value = size; const getSizeOnScale = element => { // some labels are rescaled on zoom if (element === "labels") return Math.max(rn((size + size / scale) / 2, 2), 1); if (element === "coordinates") return rn(size / scale ** 0.8, 2); // other has the same size return size; }; const scaleSize = getSizeOnScale(styleElementSelect.value); el.attr("data-size", size).attr("font-size", scaleSize); if (styleElementSelect.value === "legend") redrawLegend(); } styleFontShiftX.on("input", e => { getEl() .attr("data-dx", e.target.value) .selectAll("text") .attr("dx", e.target.value + "em"); }); styleFontShiftY.on("input", e => { getEl() .attr("data-dy", e.target.value) .selectAll("text") .attr("dy", e.target.value + "em"); }); styleStatesBodyOpacity.on("input", e => { statesBody.attr("opacity", e.target.value); }); styleStatesBodyFilter.on("change", function () { statesBody.attr("filter", this.value); }); styleStatesHaloWidth.on("input", e => { const value = e.target.value; statesHalo.attr("data-width", value).attr("stroke-width", value); }); styleStatesHaloOpacity.on("input", e => { statesHalo.attr("opacity", e.target.value); }); styleStatesHaloBlur.on("input", e => { const value = Number(e.target.value); const blur = value > 0 ? `blur(${value}px)` : null; statesHalo.attr("filter", blur); }); styleArmiesFillOpacity.on("input", e => { armies.attr("fill-opacity", e.target.value); }); styleArmiesSize.on("input", e => { const value = Number(e.target.value); armies.attr("box-size", value).attr("font-size", value * 2); armies.selectAll("g").remove(); // clear armies layer pack.states.forEach(s => { if (!s.i || s.removed || !s.military.length) return; drawRegiments(s.military, s.i); }); }); emblemsStateSizeInput.on("change", e => { emblems.select("#stateEmblems").attr("data-size", e.target.value); drawEmblems(); }); emblemsProvinceSizeInput.on("change", e => { emblems.select("#provinceEmblems").attr("data-size", e.target.value); drawEmblems(); }); emblemsBurgSizeInput.on("change", e => { emblems.select("#burgEmblems").attr("data-size", e.target.value); drawEmblems(); }); // request a URL to image to be used as a texture function textureProvideURL() { alertMessage.innerHTML = /* html */ `Provide a texture image URL: `; $("#alert").dialog({ resizable: false, title: "Load custom texture", width: "28em", buttons: { Apply: function () { if (!textureURL.value) return tip("Please provide a valid URL", false, "error"); changeTexture(textureURL.value); updateTextureSelectValue(textureURL.value); $(this).dialog("close"); }, Cancel: function () { $(this).dialog("close"); } } }); } function fetchTextureURL(url) { INFO && console.info("Provided URL is", url); const img = new Image(); img.onload = function () { const canvas = byId("texturePreview"); const ctx = canvas.getContext("2d"); ctx.clearRect(0, 0, canvas.width, canvas.height); ctx.drawImage(img, 0, 0, canvas.width, canvas.height); }; img.src = url; } const vignettePresets = { default: `{ "#vignette": { "opacity": 0.3, "fill": "#000000", "filter": null }, "#vignette-rect": { "x": "0.3%", "y": "0.4%", "width": "99.6%", "height": "99.2%", "rx": "5%", "ry": "5%", "filter": "blur(20px)" } }`, neon: `{ "#vignette": { "opacity": 0.5, "fill": "#7300ff", "filter": null }, "#vignette-rect": { "x": "0.3%", "y": "0.4%", "width": "99.6%", "height": "99.2%", "rx": "0%", "ry": "0%", "filter": "blur(15px)" } }`, smoke: `{ "#vignette": { "opacity": 1, "fill": "#000000", "filter": "url(#splotch)" }, "#vignette-rect": { "x": "3%", "y": "5%", "width": "96%", "height": "90%", "rx": "10%", "ry": "10%", "filter": "blur(100px)" } }`, wound: `{ "#vignette": { "opacity": 0.8, "fill": "#ff0000", "filter": "url(#paper)"}, "#vignette-rect": {"x": "0.5%", "y": "1%", "width": "99%", "height": "98%", "rx": "5%", "ry": "5%", "filter": "blur(50px)" } }`, paper: `{ "#vignette": { "opacity": 1, "fill": "#000000", "filter": "url(#paper)" }, "#vignette-rect": { "x": "0.3%", "y": "0.4%", "width": "99.6%", "height": "99.2%", "rx": "20%", "ry": "20%", "filter": "blur(150px)" } }`, granite: `{ "#vignette": { "opacity": 0.95, "fill": "#231b1b", "filter": "url(#crumpled)" }, "#vignette-rect": { "x": "3%", "y": "5%", "width": "94%", "height": "90%", "rx": "20%", "ry": "20%", "filter": "blur(150px)" } }`, spotlight: `{ "#vignette": { "opacity": 0.96, "fill": "#000000", "filter": null }, "#vignette-rect": { "x": "20%", "y": "30%", "width": "24%", "height": "30%", "rx": "50%", "ry": "50%", "filter": "blur(30px) "} }` }; Object.keys(vignettePresets).forEach(preset => { styleVignettePreset.options.add(new Option(preset, preset, false, false)); }); styleVignettePreset.on("change", function () { const attributes = JSON.parse(vignettePresets[this.value]); for (const selector in attributes) { const el = document.querySelector(selector); if (!el) continue; for (const attr in attributes[selector]) { const value = attributes[selector][attr]; el.setAttribute(attr, value); } } const vignette = byId("vignette"); if (vignette) { styleOpacityInput.value = vignette.getAttribute("opacity"); styleFillInput.value = styleFillOutput.value = vignette.getAttribute("fill"); styleFilterInput.value = vignette.getAttribute("filter"); } const maskRect = byId("vignette-rect"); if (maskRect) { const digit = str => str.replace(/[^\d.]/g, ""); styleVignetteX.value = digit(maskRect.getAttribute("x")); styleVignetteY.value = digit(maskRect.getAttribute("y")); styleVignetteWidth.value = digit(maskRect.getAttribute("width")); styleVignetteHeight.value = digit(maskRect.getAttribute("height")); styleVignetteRx.value = digit(maskRect.getAttribute("rx")); styleVignetteRy.value = digit(maskRect.getAttribute("ry")); styleVignetteBlur.value = digit(maskRect.getAttribute("filter")); } }); styleVignetteX.on("input", e => { byId("vignette-rect")?.setAttribute("x", `${e.target.value}%`); }); styleVignetteWidth.on("input", e => { byId("vignette-rect")?.setAttribute("width", `${e.target.value}%`); }); styleVignetteY.on("input", e => { byId("vignette-rect")?.setAttribute("y", `${e.target.value}%`); }); styleVignetteHeight.on("input", e => { byId("vignette-rect")?.setAttribute("height", `${e.target.value}%`); }); styleVignetteRx.on("input", e => { byId("vignette-rect")?.setAttribute("rx", `${e.target.value}%`); }); styleVignetteRy.on("input", e => { byId("vignette-rect")?.setAttribute("ry", `${e.target.value}%`); }); styleVignetteBlur.on("input", e => { byId("vignette-rect")?.setAttribute("filter", `blur(${e.target.value}px)`); }); styleScaleBar.on("input", function (event) { const scaleBarBack = scaleBar.select("#scaleBarBack"); if (!scaleBarBack.size()) return; const {id, value} = event.target; if (id === "styleScaleBarSize") scaleBar.attr("data-bar-size", value); else if (id === "styleScaleBarFontSize") scaleBar.attr("font-size", value); else if (id === "styleScaleBarPositionX") scaleBar.attr("data-x", value); else if (id === "styleScaleBarPositionY") scaleBar.attr("data-y", value); else if (id === "styleScaleBarLabel") scaleBar.attr("data-label", value); else if (id === "styleScaleBarBackgroundOpacity") scaleBarBack.attr("opacity", value); else if (id === "styleScaleBarBackgroundFill") scaleBarBack.attr("fill", value); else if (id === "styleScaleBarBackgroundStroke") scaleBarBack.attr("stroke", value); else if (id === "styleScaleBarBackgroundStrokeWidth") scaleBarBack.attr("stroke-width", value); else if (id === "styleScaleBarBackgroundFilter") scaleBarBack.attr("filter", value); else if (id === "styleScaleBarBackgroundPaddingTop") scaleBarBack.attr("data-top", value); else if (id === "styleScaleBarBackgroundPaddingRight") scaleBarBack.attr("data-right", value); else if (id === "styleScaleBarBackgroundPaddingBottom") scaleBarBack.attr("data-bottom", value); else if (id === "styleScaleBarBackgroundPaddingLeft") scaleBarBack.attr("data-left", value); if ( [ "styleScaleBarSize", "styleScaleBarPositionX", "styleScaleBarPositionY", "styleScaleBarLabel", "styleScaleBarBackgroundPaddingLeft", "styleScaleBarBackgroundPaddingTop", "styleScaleBarBackgroundPaddingRight", "styleScaleBarBackgroundPaddingBottom" ].includes(id) ) { drawScaleBar(scaleBar, scale); fitScaleBar(scaleBar, svgWidth, svgHeight); } }); function updateElements() { if (layerIsOn("toggleHeight")) drawHeightmap(); if (legend.selectAll("*").size() && window.redrawLegend) redrawLegend(); oceanLayers.selectAll("path").remove(); OceanLayers(); invokeActiveZooming(); } // GLOBAL FILTERS mapFilters.on("click", applyMapFilter); function applyMapFilter(event) { if (event.target.tagName !== "BUTTON") return; const button = event.target; svg.attr("data-filter", null).attr("filter", null); if (button.classList.contains("pressed")) return button.classList.remove("pressed"); mapFilters.querySelectorAll(".pressed").forEach(button => button.classList.remove("pressed")); button.classList.add("pressed"); svg.attr("data-filter", button.id).attr("filter", "url(#filter-" + button.id + ")"); }