var starred = [] var knownStations = [] /** * @type {'unavailable' | 'notRequested' | 'waiting' | 'gotData'} */ var nearbyStatus = 'notRequested' var nearbyStations = [] function goToStation(stationId) { var url = new URL(window.location.href) url.pathname = 'view-station.html' url.searchParams.set('stationId', stationId) url.searchParams.set('date', new Date().toISOString()) window.location.href = url.toString() } function searchNormalize(str) { return str .toLowerCase() .replaceAll('ă', 'a') .replaceAll('â', 'a') .replaceAll('î', 'i') .replaceAll('ș', 's') .replaceAll('ț', 't') } var focusedElement = null var _rebuildDebounce = null var _rebuildRequested = false function createSuggestion(suggestion, index) { delete suggestion['products'] var suggestionDiv = document.createElement('div') suggestionDiv.classList.add('suggestion') var suggestionLi = document.createElement('li') suggestionDiv.appendChild(suggestionLi) suggestionLi.classList.add('items') if (index) { suggestionLi.tabIndex = index + 1 } suggestionLi.style.padding = '2px 0' function onAction(e) { goToStation(JSON.stringify(suggestion)) } suggestionLi.addEventListener('click', onAction) suggestionLi.addEventListener('keypress', function (e) { if (e.key == 'Enter') { onAction(e) } }) suggestionLi.addEventListener('focus', function (e) { focusedElement = suggestionLi }) var stationNameP = document.createElement('p') suggestionLi.appendChild(stationNameP) var stationNameSpan = document.createElement('span') stationNameP.append(stationNameSpan) stationNameSpan.textContent = suggestion.name || suggestion.address stationNameSpan.classList.add('pri', 'stationName') if (suggestion.distance) { stationNameP.append(' ') var stationDistanceSpan = document.createElement('span') stationNameP.append(stationDistanceSpan) stationDistanceSpan.append('(', suggestion.distance.toString(), ' m)') stationDistanceSpan.classList.add('stationDistance') } if (window.localStorage) { var suggestionDistance = suggestion['distance'] delete suggestion['distance'] var suggestionLink = document.createElement('a') suggestionLink.classList.add('no-custom-a') var suggestionStar = document.createElement('object') suggestionLink.appendChild(suggestionStar) suggestionStar.classList.add('star') suggestionStar.type = 'image/svg+xml' function setStar() { if (starred.includes(JSON.stringify(suggestion))) { suggestionStar.data = '/icons/star_full.svg' suggestionStar.classList.add('checked') } else { suggestionStar.data = '/icons/star_empty.svg' suggestionStar.classList.remove('checked') } } suggestionLink.addEventListener('click', function (event) { event.preventDefault() if (starred.includes(JSON.stringify(suggestion))) { starred = starred.filter(function (s) { return s !== suggestion }) } else { starred.push(JSON.stringify(suggestion)) } setStar() localStorage.setItem('stations/starred', JSON.stringify(starred)) }) setStar() suggestionDiv.appendChild(suggestionLink) suggestion['distance'] = suggestionDistance } // var trainCompanyP = document.createElement('p') // suggestionLi.appendChild(trainCompanyP) // trainCompanyP.textContent = suggestion.company // trainCompanyP.classList.add('thi') return suggestionDiv } function rebuildSuggestions() { if (_rebuildDebounce !== null) { _rebuildRequested = true return } _rebuildRequested = false _rebuildDebounce = 123 var suggestionsArea = document.getElementById('suggestionsArea') while (suggestionsArea.childNodes.length > 0) { suggestionsArea.childNodes[0].remove() } var suggestions = knownStations.slice() if (suggestions.length === 0) { suggestions = starred.map(function (s) { return JSON.parse(s) }) } suggestions.forEach(function (suggestion, index) { suggestionsArea.append(createSuggestion(suggestion, index)) }) if (nearbyStatus !== 'unavailable') { suggestionsArea.appendChild(h4('Nearby stations')) if (nearbyStatus === 'notRequested') { suggestionsArea.appendChild(a('', 'Load nearby stations').event$('click', function (event) { event.preventDefault() var latitude = 0 var longitude = 0 var watchId = navigator.geolocation.watchPosition( function (data) { if (data.coords.latitude !== latitude || data.coords.longitude !== longitude) { latitude = data.coords.latitude longitude = data.coords.longitude var geoUrl = new URL('https://v6.db.transport.rest/locations/nearby') geoUrl.searchParams.append('latitude', data.coords.latitude.toString()) geoUrl.searchParams.append('longitude', data.coords.longitude.toString()) geoUrl.searchParams.append('results', '10') fetch(geoUrl) .then(function (response) { return response.json() }) .then(function (data) { nearbyStatus = 'gotData' nearbyStations = data rebuildSuggestions() }) .catch(function () { nearbyStatus = 'unavailable' rebuildSuggestions() }) } }, function (error) { if (nearbyStations.length === 0) { nearbyStatus = 'unavailable' rebuildSuggestions() } navigator.geolocation.clearWatch(watchId) }, { enableHighAccuracy: true, }, ) nearbyStatus = 'waiting' rebuildSuggestions() })) } else if (nearbyStatus === 'waiting') { var waitingP = document.createElement('p') suggestionsArea.append(waitingP) waitingP.append('Loading...') waitingP.classList.add('pri') } else if (nearbyStatus === 'gotData') { nearbyStations.forEach(function (suggestion, index) { suggestionsArea.appendChild(createSuggestion(suggestion, suggestions.length + index)) }) } } setTimeout(function () { _rebuildDebounce = null if (_rebuildRequested) { rebuildSuggestions() } }, 500) } var fetchAbortController = new AbortController() function reloadSuggestions() { var stationNameInput = document.getElementById('stationName') var stationName = searchNormalize(stationNameInput.value.trim()) var locationsUrl = new URL('https://v6.db.transport.rest/locations') locationsUrl.searchParams.set('query', stationName) locationsUrl.searchParams.set('limit', '25') locationsUrl.searchParams.set('fuzzy', 'true') locationsUrl.searchParams.set('stops', 'true') fetchAbortController.abort() fetchAbortController = new AbortController() fetch(locationsUrl.toString(), { signal: fetchAbortController.signal }) .then(function (response) { if (response.ok) { return response.json() } else { return {} } }) .then(function (data) { if (data) { knownStations = Object.values(data) rebuildSuggestions() } }) } function lsk() { document.getElementById('stationName').focus() } function csk() { if (focusedElement == null) { return } if (focusedElement.id === 'stationName') { goToTrain(document.activeElement.value.trim()) } else { focusedElement.click() } } window.addEventListener('load', function (e) { if (window.localStorage) { var maybeStarred = JSON.parse(localStorage.getItem('stations/starred')) if (maybeStarred) { starred = maybeStarred } } var stationName = document.getElementById('stationName') stationName.addEventListener('input', function (e) { reloadSuggestions() }) stationName.addEventListener('focus', function (e) { focusedElement = stationName document.getElementsByClassName('lsk')[0].textContent = '' document.getElementsByClassName('csk')[0].textContent = 'Search' }) stationName.addEventListener('blur', function (e) { document.getElementsByClassName('lsk')[0].textContent = 'Search' document.getElementsByClassName('csk')[0].textContent = 'Select' }) stationName.addEventListener('keypress', function (e) { if (e.key == 'Enter') { goToStation(stationName.value.trim()) } }) document.querySelectorAll('.lsk').forEach(function (lskElem) { lskElem.addEventListener('click', function (e) { lsk() }) }) document.querySelectorAll('.csk').forEach(function (cskElem) { cskElem.addEventListener('click', function (e) { csk() }) }) document.body.addEventListener('keydown', function (e) { if (e.key == 'SoftLeft') { lsk() } else if (e.key == 'Enter') { csk() } }) reloadSuggestions() })