diff --git a/src/modules/leaflet-control-map-settings.js b/src/modules/leaflet-control-map-settings.js index b82d7b8cf..56d66bce3 100644 --- a/src/modules/leaflet-control-map-settings.js +++ b/src/modules/leaflet-control-map-settings.js @@ -6,6 +6,7 @@ L.Control.MapSettings = L.Control.extend({ activeTasksLabel: "Only show markers for active tasks", expandMapLegendLabel: "Don't collapse layers control", expandMapLegendChecked: false, + alwaysShowSnipers: true, }, onAdd: function (map) { const className = "leaflet-control-map-settings"; @@ -104,6 +105,24 @@ L.Control.MapSettings = L.Control.extend({ const activeQuestMarkersLabelContent = L.DomUtil.create("span", undefined, activeQuestMarkersLabel); activeQuestMarkersLabelContent.textContent = this.options.activeTasksLabel; + // always show snipers setting + const alwaysShowSnipersDiv = L.DomUtil.create("div", `${className}-setting-container`, form); + + const alwaysShowSnipersLabel = L.DomUtil.create("label", undefined, alwaysShowSnipersDiv); + alwaysShowSnipersLabel.setAttribute("for", "alwaysShowSnipers"); + + const alwaysShowSnipersCheckbox = L.DomUtil.create("input", undefined, alwaysShowSnipersLabel); + alwaysShowSnipersCheckbox.id = "alwaysShowSnipers"; + alwaysShowSnipersCheckbox.setAttribute("type", "checkbox"); + if (!!this.options.alwaysShowSnipers) { + alwaysShowSnipersCheckbox.setAttribute("checked", !!this.options.alwaysShowSnipers); + alwaysShowSnipersCheckbox.checked = true; + } + L.DomEvent.on(alwaysShowSnipersCheckbox, "click", this._onSettingChanged, this); + + const alwaysShowSnipersLabelContent = L.DomUtil.create("span", undefined, alwaysShowSnipersLabel); + alwaysShowSnipersLabelContent.textContent = this.options.alwaysShowSnipersLabel; + L.DomUtil.create("div", `${className}-separator player-location-help-separator`, form); // show location labels setting diff --git a/src/pages/map/index.css b/src/pages/map/index.css index 7c94a5cd8..623cff70b 100644 --- a/src/pages/map/index.css +++ b/src/pages/map/index.css @@ -20,14 +20,14 @@ .leaflet-popup-content-wrapper a { color: var(--color-gold-one); font-family: - 'bender', + "bender", -apple-system, system-ui, BlinkMacSystemFont, - 'Segoe UI', - 'Roboto', - 'Helvetica Neue', - 'Arial', + "Segoe UI", + "Roboto", + "Helvetica Neue", + "Arial", sans-serif; font-style: normal; @@ -108,14 +108,14 @@ } .leaflet-control-layers-group-expand-default::before { - content: '+'; + content: "+"; width: 12px; display: inline-block; text-align: center; } .leaflet-control-layers-group-collapse-default::before { - content: '-'; + content: "-"; width: 12px; display: inline-block; text-align: center; @@ -146,6 +146,10 @@ div.leaflet-pane.leaflet-marker-pane > .off-level { z-index: -9999 !important; } +.leaflet-container.always-show-snipers div.leaflet-pane.leaflet-marker-pane > .off-level.sniper-spawn { + opacity: 1; +} + div.leaflet-overlay-pane > svg.leaflet-zoom-animated > g > path.off-level { stroke-opacity: 0.2; fill-opacity: 0.02; @@ -163,7 +167,7 @@ div.leaflet-overlay-pane > svg.leaflet-zoom-animated > g > path.off-level { .btr-stop-name { font-weight: 800; font-size: 18px; - font-family: 'Arial', 'Helvetica', sans-serif; + font-family: "Arial", "Helvetica", sans-serif; color: var(--color-white); white-space: nowrap; @@ -218,7 +222,8 @@ img.popup-item { max-width: 24px; max-height: 24px; vertical-align: middle; - filter: drop-shadow(0.01em 0.01em white) drop-shadow(-0.01em -0.01em white) drop-shadow(-0.01em 0.01em white) drop-shadow(0.01em -0.01em white); + filter: drop-shadow(0.01em 0.01em white) drop-shadow(-0.01em -0.01em white) drop-shadow(-0.01em 0.01em white) + drop-shadow(0.01em -0.01em white); } .poi-link { @@ -230,14 +235,14 @@ img.popup-item { background-color: rgb(from var(--color-black) r g b / 0.75); color: var(--color-gold-one); font-family: - 'bender', + "bender", -apple-system, system-ui, BlinkMacSystemFont, - 'Segoe UI', - 'Roboto', - 'Helvetica Neue', - 'Arial', + "Segoe UI", + "Roboto", + "Helvetica Neue", + "Arial", sans-serif; font-style: normal; font-weight: 400; @@ -248,7 +253,7 @@ img.popup-item { border-color: var(--color-gold-two); } -.leaflet-control input[type='checkbox'] { +.leaflet-control input[type="checkbox"] { /* Add if not using autoprefixer */ appearance: none; @@ -265,8 +270,8 @@ img.popup-item { transform: translateY(-0.075em); } -.leaflet-control input[type='checkbox']::before { - content: ''; +.leaflet-control input[type="checkbox"]::before { + content: ""; width: 0.65em; height: 0.65em; transform: scale(0); @@ -280,11 +285,11 @@ img.popup-item { margin-top: 2px; } -.leaflet-control input[type='checkbox']:checked::before { +.leaflet-control input[type="checkbox"]:checked::before { transform: scale(1); } -.leaflet-control input[type='radio'] { +.leaflet-control input[type="radio"] { /* Add if not using autoprefixer */ appearance: none; @@ -301,8 +306,8 @@ img.popup-item { transform: translateY(-0.075em); } -.leaflet-control input[type='radio']::before { - content: ''; +.leaflet-control input[type="radio"]::before { + content: ""; width: 0.65em; height: 0.65em; border-radius: 0.5em; @@ -315,7 +320,7 @@ img.popup-item { margin-top: 0.1em; } -.leaflet-control input[type='radio']:checked::before { +.leaflet-control input[type="radio"]:checked::before { transform: scale(1); } @@ -335,7 +340,7 @@ img.popup-item { /* fullscreen control */ .leaflet-control-fullscreen a { background-color: rgb(from var(--color-black) r g b / 0.75); - background-image: url(''); + background-image: url(""); } .leaflet-control-fullscreen a:hover { @@ -384,14 +389,14 @@ img.popup-item { /* Area labels */ .leaflet-marker-icon.map-area-label { font-family: - 'bender', + "bender", -apple-system, system-ui, BlinkMacSystemFont, - 'Segoe UI', - 'Roboto', - 'Helvetica Neue', - 'Arial', + "Segoe UI", + "Roboto", + "Helvetica Neue", + "Arial", sans-serif; font-weight: 800; font-size: 20px; @@ -415,7 +420,8 @@ img.popup-item { } .leaflet-container .leaflet-marker-pane img.loot-outline { - filter: drop-shadow(0.01em 0.01em white) drop-shadow(-0.01em -0.01em white) drop-shadow(-0.01em 0.01em white) drop-shadow(0.01em -0.01em white); + filter: drop-shadow(0.01em 0.01em white) drop-shadow(-0.01em -0.01em white) drop-shadow(-0.01em 0.01em white) + drop-shadow(0.01em -0.01em white); } /* active level badge for grouped layer */ @@ -444,10 +450,10 @@ img.popup-item { white-space: nowrap; } -.leaflet-control-container [data-badge^='-']::after, -.leaflet-control-container [data-badge='0']::after, -.leaflet-control-container [data-badge='']::after, -.leaflet-control-container [data-badge='undefined']::after { +.leaflet-control-container [data-badge^="-"]::after, +.leaflet-control-container [data-badge="0"]::after, +.leaflet-control-container [data-badge=""]::after, +.leaflet-control-container [data-badge="undefined"]::after { display: none; } diff --git a/src/pages/map/index.jsx b/src/pages/map/index.jsx index 906b9c86f..413d90d26 100644 --- a/src/pages/map/index.jsx +++ b/src/pages/map/index.jsx @@ -303,6 +303,7 @@ function Map() { showOnlyActiveTasks: false, expandMapLegend: false, expandSearch: false, + alwaysShowSnipers: true, }); const mapSettingsRef = useRef(savedMapSettings); @@ -525,6 +526,8 @@ function Map() { expandSearchChecked: mapSettingsRef.current.expandSearch, expandSearchLabel: tMaps("Don't collapse search control"), playerLocationLabel: tMaps("Use TarkovMonitor to show your position"), + alwaysShowSnipers: mapSettingsRef.current.alwaysShowSnipers ?? true, + alwaysShowSnipersLabel: tMaps("Always show snipers"), collapsed: true, }) .addTo(map); @@ -542,6 +545,13 @@ function Map() { if (e.settingName === "expandSearch") { map.searchControl.setCollapse(!e.settingValue); } + if (e.settingName === "alwaysShowSnipers") { + if (e.settingValue) { + map._container.classList.add("always-show-snipers"); + } else { + map._container.classList.remove("always-show-snipers"); + } + } mapSettingsRef.current[e.settingName] = e.settingValue; updateSavedMapSettings(); }); @@ -953,6 +963,7 @@ function Map() { } } // if baseLayer._image is set, it's an svg map + // since we're adding a height layer, we set base layer to off level if (baseLayer._image && !layer.show) { baseLayer._image.classList.add("off-level"); } else if (baseLayer._container && !layer.show) { @@ -962,36 +973,48 @@ function Map() { // remove the hidden-layer class from the added layer // we wrap it in the svg loading promsise to make sure the svg file has finished loading svgLoaded.finally(() => { - const layerGroup = [...baseLayer._image.children[0]?.children].find( - (c) => c.id === layer.svgLayer, - ); - layerGroup?.classList.remove("hidden-layer"); + if (!baseLayer._image.children[0]?.children) { + return; + } + for (const layerGroup of baseLayer._image.children[0].children) { + if (layerGroup.id !== layer.svgLayer) { + // hide all layers that weren't just added + layerGroup?.classList.add("hidden-layer"); + continue; + } + // un-hide added layer + layerGroup?.classList.remove("hidden-layer"); + } }); } map.layerControl.updateBadge(tMaps(layer.name)); }); heightLayer.on("remove", () => { - const heightLayer = Object.values(map._layers).findLast((l) => l.options?.extents); - if (heightLayer) { - for (const marker of Object.values(map._layers)) { - checkMarkerForActiveLayers(marker); - } - const layers = Object.values(map._layers).filter((l) => l.options.type === "map-layer"); - if (layers.length !== 1) { - return; - } - map.layerControl.updateBadge(); - if (baseLayer._image) { - baseLayer._image.classList.remove("off-level"); - } else if (baseLayer._container) { - baseLayer._container.classList.remove("off-level"); - } - if (baseLayer._image?.children[0]) { - // add the hidden-layer class to the removed layer - const layerGroup = [...baseLayer._image.children[0].children].find( - (c) => c.id === layer.svgLayer, - ); - layerGroup?.classList.add("hidden-layer"); + /*const heightLayer = Object.values(map._layers).findLast((l) => l.options?.extents); + if (!heightLayer) { + return; + }*/ + for (const marker of Object.values(map._layers)) { + checkMarkerForActiveLayers(marker); + } + const layers = Object.values(map._layers).filter((l) => l.options.type === "map-layer"); + if (layers.length !== 1) { + return; + } + map.layerControl.updateBadge(); + if (baseLayer._image) { + baseLayer._image.classList.remove("off-level"); + } else if (baseLayer._container) { + baseLayer._container.classList.remove("off-level"); + } + if (baseLayer._image?.children[0]) { + // add the hidden-layer class to the removed layer + for (const layerGroup of baseLayer._image.children[0].children) { + if (layerGroup.id !== layer.svgLayer) { + continue; + } + layerGroup.classList.add("hidden-layer"); + break; } } }); @@ -1175,6 +1198,7 @@ function Map() { } let spawnType = ""; let bosses = []; + let markerClass; if (spawn.categories.includes("boss")) { bosses = mapData.bosses.filter((boss) => @@ -1206,6 +1230,7 @@ function Map() { } } else if (spawn.categories.includes("sniper")) { spawnType = "sniper_scav"; + markerClass = "sniper-spawn"; } else if (spawn.sides.includes("scav")) { if (spawn.categories.includes("bot") || spawn.categories.includes("all")) { spawnType = "scav"; @@ -1222,6 +1247,7 @@ function Map() { iconUrl: `${process.env.PUBLIC_URL}/maps/interactive/spawn_${spawnType}.png`, iconSize: [24, 24], popupAnchor: [0, -12], + className: markerClass, }); if (spawnType === "pmc") { @@ -2116,7 +2142,7 @@ function Map() { key="seo-wrapper" />,