-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathGeolocationButton.client.ts
59 lines (49 loc) · 2.13 KB
/
GeolocationButton.client.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
export type GeolocationData = { address: { postcode: string } }
import { GeolocationPermission } from '../utils/GeolocationPermission.ts'
const geolocationPermission = new GeolocationPermission()
export class GeolocationButton extends HTMLElement {
button: HTMLButtonElement | undefined
deniedMessage: HTMLParagraphElement | undefined
buttonText = ''
isLoading = false
connectedCallback() {
this.button = this.querySelector('button')!
this.buttonText = this.button.textContent ?? ''
this.deniedMessage = this.querySelector<HTMLParagraphElement>('.blocked-msg')!
this.button.addEventListener('click', (evt) => this.handleClick(evt))
geolocationPermission.addEventListener('change', () => this.updateDeniedState())
this.updateDeniedState()
}
updateDeniedState() {
this.deniedMessage!.style.display = geolocationPermission.state === 'denied' ? '' : 'none'
if (geolocationPermission.state === 'denied') {
this.button!.setAttribute('aria-disabled', String((this.button!.disabled = true)))
} else if (this.getAttribute('data-loading') !== 'true') {
this.button!.setAttribute('aria-disabled', String((this.button!.disabled = false)))
}
}
setLoadingState(state: boolean) {
this.setAttribute('data-loading', String(state))
this.button!.setAttribute(
'aria-disabled',
String((this.button!.disabled = state || geolocationPermission.state === 'denied')),
)
this.button!.textContent = state ? 'Getting location...' : this.buttonText
}
handleClick(evt: Event) {
if (this.button!.disabled) return
evt.preventDefault()
this.setLoadingState(true)
navigator.geolocation.getCurrentPosition(
({ coords: { latitude: lat, longitude: lon } }) => {
void fetch(`https://nominatim.openstreetmap.org/reverse?lat=${lat}&lon=${lon}&format=json`)
.then((response) => response.json())
.then((result: GeolocationData) => {
this.dispatchEvent(new CustomEvent('geolocation', { bubbles: true, detail: result }))
})
.finally(() => this.setLoadingState(false))
},
() => this.setLoadingState(false),
)
}
}