Amelioration de la navigation sur la carte
This commit is contained in:
parent
aa8ea624c7
commit
799dd817ae
5 changed files with 171 additions and 46 deletions
|
|
@ -88,8 +88,53 @@ body > hr {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.highlight-point-icon > * {
|
#map .highlight-point-icon > * {
|
||||||
|
width: fit-content;
|
||||||
|
height: fit-content;
|
||||||
transform: translateX(-50%) translateY(-50%);
|
transform: translateX(-50%) translateY(-50%);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
#map .highlight-point-icon > a {
|
||||||
|
text-decoration: none;
|
||||||
|
color: inherit;
|
||||||
|
}
|
||||||
|
|
||||||
|
#map .highlight-point-icon h3 {
|
||||||
|
width: max-content;
|
||||||
|
max-width: min(25vw, 100px);
|
||||||
|
text-align: center;
|
||||||
|
margin: 0;
|
||||||
|
color: white;
|
||||||
|
font-size: 1em;
|
||||||
|
line-height: 1;
|
||||||
|
--outline-size: 1px;
|
||||||
|
text-shadow: 0px var(--outline-size) 0px var(--symbol-border-color),
|
||||||
|
var(--outline-size) 0px 0px var(--symbol-border-color),
|
||||||
|
var(--outline-size) var(--outline-size) 0px var(--symbol-border-color),
|
||||||
|
calc(var(--outline-size) * -1) calc(var(--outline-size) * -1) 0px var(--symbol-border-color),
|
||||||
|
0px calc(var(--outline-size) * -1) 0px var(--symbol-border-color),
|
||||||
|
calc(var(--outline-size) * -1) 0px 0px var(--symbol-border-color)
|
||||||
|
;
|
||||||
|
opacity: clamp(0, calc( var(--zoom-level) - 18 ), 1) ;
|
||||||
|
}
|
||||||
|
|
||||||
|
#map .highlight-point-icon img ~ h3 {
|
||||||
|
position: absolute;
|
||||||
|
bottom: 0;
|
||||||
|
left: 50%;
|
||||||
|
transform: translateX(-50%) translateY(100%);
|
||||||
|
margin-top: 1ex;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
#map .highlight-point-icon:not(.active) {
|
||||||
|
opacity: 0;
|
||||||
|
pointer-events: none;
|
||||||
}
|
}
|
||||||
|
|
||||||
.map-amenity-icon .map-amenity-icon-container {
|
.map-amenity-icon .map-amenity-icon-container {
|
||||||
|
|
@ -135,10 +180,6 @@ body > hr {
|
||||||
opacity: 0;
|
opacity: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#map .highlight-point-icon:not(.active) {
|
|
||||||
opacity: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* SAERCH FORM */
|
/* SAERCH FORM */
|
||||||
|
|
||||||
#search-section {
|
#search-section {
|
||||||
|
|
|
||||||
|
|
@ -209,7 +209,7 @@ export class BidiPanelElement extends HTMLElement {
|
||||||
requestDispatchChangeEvent(){
|
requestDispatchChangeEvent(){
|
||||||
let currentPanel = this.activeChildren
|
let currentPanel = this.activeChildren
|
||||||
if(currentPanel != this.#lastActive){
|
if(currentPanel != this.#lastActive){
|
||||||
this.dispatchEvent(new ActivePanelChangeEvent("activePanelChange", this.#currentIndex))
|
this.dispatchEvent(new ActivePanelChangeEvent("activePanelChange", this.#currentIndex || this.activeChildrenIndex))
|
||||||
this.#lastActive = currentPanel
|
this.#lastActive = currentPanel
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
||||||
60
js/index.js
60
js/index.js
|
|
@ -28,11 +28,42 @@ search_form.elements["query"].addEventListener("input", () => search_form.reques
|
||||||
window.addEventListener("hashchange", () => {
|
window.addEventListener("hashchange", () => {
|
||||||
let place_id = decodeURIComponent(location.hash.substring(1))
|
let place_id = decodeURIComponent(location.hash.substring(1))
|
||||||
if (place_id) {
|
if (place_id) {
|
||||||
|
let feature = places.getFeatureById(place_id);
|
||||||
|
if (feature) {
|
||||||
MAP.highlight(place_id)
|
MAP.highlight(place_id)
|
||||||
|
if(document.getElementById("search-result")) {
|
||||||
|
openSearchResultItem(feature)
|
||||||
|
} else {
|
||||||
|
let foundIndex = null;
|
||||||
|
let panelChildren = document.getElementById("result-panel").children
|
||||||
|
for(let i = 0; i<panelChildren.length; i++) {
|
||||||
|
let feature_el = panelChildren[i];
|
||||||
|
if(feature_el instanceof FeatureElement && feature_el.feature.id == feature.id){
|
||||||
|
foundIndex = i;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if(foundIndex != null){
|
||||||
|
document.getElementById("result-panel").setActiveChildrenIndex(foundIndex, {behavior: "instant"})
|
||||||
|
} else {
|
||||||
|
openFeature(feature)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for(let feature_el of document.getElementById("result-panel").children) {
|
||||||
|
if(feature_el instanceof FeatureElement){
|
||||||
|
MAP.show(feature_el.feature)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
MAP.unhighlight_all()
|
MAP.unhighlight_all()
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
if(location.hash){
|
||||||
|
window.dispatchEvent(new Event("hashchange"))
|
||||||
|
}
|
||||||
|
|
||||||
search_form.addEventListener("submit", e => {
|
search_form.addEventListener("submit", e => {
|
||||||
e.preventDefault()
|
e.preventDefault()
|
||||||
|
|
@ -44,23 +75,23 @@ search_form.addEventListener("submit", e => {
|
||||||
|
|
||||||
let search_results = places.search(data.get("query"))
|
let search_results = places.search(data.get("query"))
|
||||||
|
|
||||||
|
MAP.unhighlight_all();
|
||||||
|
|
||||||
for(let result_item of search_results){
|
for(let result_item of search_results){
|
||||||
let el = document.createElement("li")
|
let el = document.createElement("li")
|
||||||
let a = document.createElement("a")
|
let a = document.createElement("a")
|
||||||
el.append(a)
|
el.append(a)
|
||||||
a.href = "#"+encodeURIComponent(result_item.ref)
|
a.href = "#"+encodeURIComponent(result_item.ref)
|
||||||
a.addEventListener("click", e => {
|
|
||||||
e.preventDefault()
|
|
||||||
openSearchResultItem(result_item.feature)
|
|
||||||
})
|
|
||||||
let header = document.createElement("camp-feature-short-header")
|
let header = document.createElement("camp-feature-short-header")
|
||||||
header.feature = result_item.feature
|
header.feature = result_item.feature
|
||||||
a.append(header)
|
a.append(header)
|
||||||
resultElements.push(el)
|
resultElements.push(el)
|
||||||
|
MAP.show(result_item.feature)
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!document.getElementById("search-result")){
|
if(!document.getElementById("search-result")){
|
||||||
let el = document.createElement("ol")
|
let el = document.createElement("ol")
|
||||||
|
|
||||||
el.id = "search-result"
|
el.id = "search-result"
|
||||||
document.getElementById("result-panel").replaceChildren(el)
|
document.getElementById("result-panel").replaceChildren(el)
|
||||||
}
|
}
|
||||||
|
|
@ -71,7 +102,6 @@ search_form.addEventListener("submit", e => {
|
||||||
document.getElementById("search-result").replaceChildren(document.createTextNode("Pas de resultat"))
|
document.getElementById("search-result").replaceChildren(document.createTextNode("Pas de resultat"))
|
||||||
}
|
}
|
||||||
|
|
||||||
document.getElementById("search-result").children[0]?.scrollIntoView()
|
|
||||||
} else {
|
} else {
|
||||||
document.getElementById("search-result")?.remove()
|
document.getElementById("search-result")?.remove()
|
||||||
}
|
}
|
||||||
|
|
@ -112,25 +142,39 @@ function openSearchResultItem(feature){
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
function openFeature(feature){
|
||||||
|
let panel = document.getElementById("result-panel")
|
||||||
|
let root = document.createElement("camp-feature")
|
||||||
|
root.feature = feature
|
||||||
|
panel.replaceChildren(root)
|
||||||
|
requestAnimationFrame(() => {
|
||||||
|
panel.setActiveChildrenIndex(0, {behavior: "instant"})
|
||||||
|
})
|
||||||
|
updateActiveFeature(feature)
|
||||||
|
}
|
||||||
|
|
||||||
function updateActiveFeature(feature_or_featureid){
|
function updateActiveFeature(feature_or_featureid){
|
||||||
let newUrl = new URL(window.location)
|
let newUrl = new URL(window.location)
|
||||||
if(feature_or_featureid){
|
if(feature_or_featureid){
|
||||||
newUrl.hash = encodeURIComponent(feature_or_featureid.id || feature_or_featureid)
|
newUrl.hash = encodeURIComponent(feature_or_featureid.id || feature_or_featureid)
|
||||||
} else {
|
} else {
|
||||||
delete newUrl.hash;
|
newUrl.hash = "";
|
||||||
}
|
}
|
||||||
|
if(newUrl.toString() != location.toString()){
|
||||||
window.history.replaceState(null, "", newUrl.toString())
|
window.history.replaceState(null, "", newUrl.toString())
|
||||||
window.dispatchEvent(new Event("hashchange"))
|
window.dispatchEvent(new Event("hashchange"))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
document.getElementById("result-panel").addEventListener("activePanelChange", e => {
|
document.getElementById("result-panel").addEventListener("activePanelChange", e => {
|
||||||
if(e.activePanelIndex){
|
console.log("Panel changed", e.activePanelIndex)
|
||||||
|
if(e.activePanelIndex || e.activePanelIndex === 0){
|
||||||
let activeElement = e.target.children[e.activePanelIndex]
|
let activeElement = e.target.children[e.activePanelIndex]
|
||||||
if(activeElement instanceof FeatureElement){
|
if(activeElement instanceof FeatureElement){
|
||||||
let feature = activeElement.feature
|
let feature = activeElement.feature
|
||||||
updateActiveFeature(feature)
|
updateActiveFeature(feature)
|
||||||
} else {
|
} else {
|
||||||
MAP.unhighlight_all()
|
updateActiveFeature(null)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
|
||||||
52
js/map.js
52
js/map.js
|
|
@ -107,22 +107,38 @@ export async function init_places(places_db){
|
||||||
// La couche des zones disponibles
|
// La couche des zones disponibles
|
||||||
const area_highlight = L.geoJSON(null, {
|
const area_highlight = L.geoJSON(null, {
|
||||||
pointToLayer: function(feature, latlng) {
|
pointToLayer: function(feature, latlng) {
|
||||||
|
let contentEl = document.createElement("a")
|
||||||
|
|
||||||
|
if(feature.id){
|
||||||
|
contentEl.href = `#${encodeURIComponent(feature.id)}`
|
||||||
|
} else if(feature.parentFeature?.id){
|
||||||
|
contentEl.href = `#${encodeURIComponent(feature.parentFeature.id)}`
|
||||||
|
}
|
||||||
|
|
||||||
let symbol = feature.mapSymbol;
|
let symbol = feature.mapSymbol;
|
||||||
if(symbol.markerUrl) {
|
if(symbol.markerUrl) {
|
||||||
|
let iconEl = document.createElement("img")
|
||||||
|
iconEl.src = symbol.markerUrl
|
||||||
|
contentEl.append(iconEl)
|
||||||
|
}
|
||||||
|
|
||||||
let iconEl = document.createElement("img");
|
if(symbol.borderColor && symbol.borderColor != "white"){
|
||||||
iconEl.src = symbol.markerUrl;
|
contentEl.style.setProperty("--symbol-border-color", symbol.borderColor);
|
||||||
|
}
|
||||||
|
|
||||||
|
if(feature.properties.name){
|
||||||
|
let nameEl = document.createElement("h3")
|
||||||
|
nameEl.textContent = feature.properties.name
|
||||||
|
contentEl.append(nameEl)
|
||||||
|
}
|
||||||
|
|
||||||
if(iconEl){
|
|
||||||
return L.marker(latlng, {
|
return L.marker(latlng, {
|
||||||
icon: L.divIcon({
|
icon: L.divIcon({
|
||||||
className: "highlight-point-icon",
|
className: "highlight-point-icon",
|
||||||
html: iconEl,
|
html: contentEl,
|
||||||
iconSize: [0, 0]
|
iconSize: [0, 0]
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
style: function(feature){
|
style: function(feature){
|
||||||
let symbol = feature.mapSymbol;
|
let symbol = feature.mapSymbol;
|
||||||
|
|
@ -171,6 +187,24 @@ export function highlight(place_or_placeid){
|
||||||
place = place_or_placeid
|
place = place_or_placeid
|
||||||
}
|
}
|
||||||
|
|
||||||
|
show(place_or_placeid)
|
||||||
|
|
||||||
|
let centroid = turf.centroid(place)
|
||||||
|
map.panTo(L.latLng(
|
||||||
|
centroid.geometry.coordinates[1],
|
||||||
|
centroid.geometry.coordinates[0]
|
||||||
|
))
|
||||||
|
}
|
||||||
|
|
||||||
|
export function show(place_or_placeid){
|
||||||
|
let place;
|
||||||
|
|
||||||
|
if(typeof place_or_placeid == "string"){
|
||||||
|
place = places.getFeatureById(place_or_placeid)
|
||||||
|
} else {
|
||||||
|
place = place_or_placeid
|
||||||
|
}
|
||||||
|
|
||||||
if(place[HIGHLIGHT_LAYER]){
|
if(place[HIGHLIGHT_LAYER]){
|
||||||
place[HIGHLIGHT_LAYER].getElement()
|
place[HIGHLIGHT_LAYER].getElement()
|
||||||
.classList.add("active")
|
.classList.add("active")
|
||||||
|
|
@ -181,10 +215,4 @@ export function highlight(place_or_placeid){
|
||||||
point_layer.getElement()
|
point_layer.getElement()
|
||||||
.classList.add("active")
|
.classList.add("active")
|
||||||
}
|
}
|
||||||
|
|
||||||
let centroid = turf.centroid(place)
|
|
||||||
map.panTo(L.latLng(
|
|
||||||
centroid.geometry.coordinates[1],
|
|
||||||
centroid.geometry.coordinates[0]
|
|
||||||
))
|
|
||||||
}
|
}
|
||||||
|
|
@ -103,8 +103,14 @@ function getBaseSymbolForFeature(feature){
|
||||||
return BUILDING_SYMBOL
|
return BUILDING_SYMBOL
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(feature.parentFeature && feature.parentFeature.geometry.type != "Point"){
|
||||||
|
return DEFAULT_AREA_SYMBOL
|
||||||
|
} else if (feature.geometry.type != "Point") {
|
||||||
|
return DEFAULT_AREA_SYMBOL
|
||||||
|
} else {
|
||||||
return DEFAULT_SYMBOL
|
return DEFAULT_SYMBOL
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// PREDEFINED Symbols
|
// PREDEFINED Symbols
|
||||||
|
|
||||||
|
|
@ -115,6 +121,12 @@ export const DEFAULT_SYMBOL = new MapSymbol()
|
||||||
DEFAULT_SYMBOL.borderColor = "white"
|
DEFAULT_SYMBOL.borderColor = "white"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export const DEFAULT_AREA_SYMBOL = new MapSymbol()
|
||||||
|
{
|
||||||
|
DEFAULT_AREA_SYMBOL.backgroundColor = "white"
|
||||||
|
DEFAULT_AREA_SYMBOL.borderColor = "white"
|
||||||
|
}
|
||||||
|
|
||||||
export const SLEEPING_SYMBOL = new MapSymbol()
|
export const SLEEPING_SYMBOL = new MapSymbol()
|
||||||
{
|
{
|
||||||
SLEEPING_SYMBOL.markerUrl = new URL("../icons/dortoir.svg", import.meta.url)
|
SLEEPING_SYMBOL.markerUrl = new URL("../icons/dortoir.svg", import.meta.url)
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue