Skip to content

Commit

Permalink
feat: beta live announcements feature
Browse files Browse the repository at this point in the history
  • Loading branch information
davwheat committed Nov 3, 2023
1 parent 8cde024 commit ac65678
Show file tree
Hide file tree
Showing 4 changed files with 239 additions and 84 deletions.
1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -43,6 +43,7 @@
"react": "^18.2.0",
"react-beautiful-dnd": "^13.1.1",
"react-dom": "^18.2.0",
"react-fullscreen-crossbrowser": "^1.1.3",
"react-head": "^3.4.2",
"react-select": "^5.7.7",
"react-tabs": "^6.0.2",
Expand Down
Original file line number Diff line number Diff line change
@@ -1,10 +1,12 @@
import React, { useEffect, useMemo, useState } from 'react'
import StationAnnouncementSystem from '@announcement-data/StationAnnouncementSystem'
import CallingAtSelector, { CallingAtPoint } from '@components/CallingAtSelector'
import CustomAnnouncementPane, { ICustomAnnouncementPreset } from '@components/PanelPanes/CustomAnnouncementPane'
import CustomAnnouncementPane, { ICustomAnnouncementPaneProps, ICustomAnnouncementPreset } from '@components/PanelPanes/CustomAnnouncementPane'
import CustomButtonPane from '@components/PanelPanes/CustomButtonPane'
import { AllStationsTitleValueMap } from '@data/StationManipulators'
import crsToStationItemMapper, { stationItemCompleter } from '@helpers/crsToStationItemMapper'
import { AudioItem, CustomAnnouncementTab } from '../../AnnouncementSystem'
import FullscreenIcon from 'mdi-react/FullscreenIcon'

type ChimeType = /*'3' |*/ 'four' | 'none'

Expand Down Expand Up @@ -249,7 +251,7 @@ export default class KeTechPhil extends StationAnnouncementSystem {
}
}

private AVAILABLE_TOCS = [
readonly AVAILABLE_TOCS = [
'a replacement bus',
'additional',
'additional Chiltern Railways',
Expand Down Expand Up @@ -890,8 +892,8 @@ export default class KeTechPhil extends StationAnnouncementSystem {
await this.playAudioFiles(files, download)
}

private platforms = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12].flatMap(x => [`${x}`, `${x}a`, `${x}b`, `${x}c`, `${x}d`]).concat(['a', 'b'])
private stations = [
readonly platforms = [1, 2, 3, 4, 5, 6, 7, 8, 9, 10, 11, 12].flatMap(x => [`${x}`, `${x}a`, `${x}b`, `${x}c`, `${x}d`]).concat(['a', 'b'])
readonly stations = [
'AAP',
'AAT',
'ABA',
Expand Down Expand Up @@ -3609,85 +3611,14 @@ export default class KeTechPhil extends StationAnnouncementSystem {
},
},
},
// standingTrain: {
// name: 'Standing train',
// component: CustomAnnouncementPane,
// props: {
// // presets: AnnouncementPresets.standingTrain,
// playHandler: this.playStandingTrainAnnouncement.bind(this),
// options: {
// chime: {
// name: 'Chime',
// type: 'select',
// default: 'none',
// options: [
// { title: '3 chimes', value: '3' },
// { title: '4 chimes', value: '4' },
// { title: 'No chime', value: 'none' },
// ],
// },
// currentStation: {
// name: 'Current station',
// default: AVAILABLE_station.m.filter(x => AVAILABLE_station.e.includes(x as any))[0],
// options: AllStationsTitleValueMap.filter(
// s => AVAILABLE_station.e.includes(s.value as any) && AVAILABLE_station.m.includes(s.value as any),
// ),
// type: 'select',
// },
// platform: {
// name: 'Platform',
// default: AVAILABLE_ALPHANUMBERS[0],
// options: AVAILABLE_ALPHANUMBERS.map(p => ({ title: `Platform ${p}`, value: p })),
// type: 'select',
// },
// hour: {
// name: 'Hour',
// default: AVAILABLE_HOURS[0],
// options: AVAILABLE_HOURS.map(h => ({ title: h, value: h })),
// type: 'select',
// },
// min: {
// name: 'Minute',
// default: AVAILABLE_MINUTES[0],
// options: AVAILABLE_MINUTES.map(m => ({ title: m, value: m })),
// type: 'select',
// },
// toc: {
// name: 'TOC',
// default: this.AVAILABLE_TOCS[0].toLowerCase(),
// options: this.AVAILABLE_TOCS.map(m => ({ title: m, value: m.toLowerCase() })),
// type: 'select',
// },
// terminatingStationCode: {
// name: 'Terminating station',
// default: this.stations[1],
// options: AllStationsTitleValueMap.filter(s => this.station.includes(s.value)),
// type: 'select',
// },
// via: {
// name: 'Via... (optional)',
// default: 'none',
// options: [{ title: 'NONE', value: 'none' }, ...AllStationsTitleValueMap.filter(s => this.station.includes(s.value))],
// type: 'select',
// },
// callingAt: {
// name: '',
// type: 'custom',
// component: CallingAtSelector,
// props: {
// availableStations: AVAILABLE_station.high,
// },
// default: [],
// },
// coaches: {
// name: 'Coach count',
// default: AVAILABLE_NUMBERS.filter(x => parseInt(x) > 1)[0],
// options: AVAILABLE_NUMBERS.filter(x => parseInt(x) > 1).map(c => ({ title: c, value: c })),
// type: 'select',
// },
// },
// },
// },
liveTrains: {
name: 'Live trains (beta)',
component: LiveTrainAnnouncements,
props: {
nextTrainHandler: this.playNextTrainAnnouncement.bind(this),
system: this,
},
},
announcementButtons: {
name: 'Announcement buttons',
component: CustomButtonPane,
Expand Down Expand Up @@ -3753,3 +3684,218 @@ export default class KeTechPhil extends StationAnnouncementSystem {
},
}
}

interface LiveTrainAnnouncementsProps extends ICustomAnnouncementPaneProps {
system: KeTechPhil
nextTrainHandler: (options: INextTrainAnnouncementOptions) => Promise<void>
}

import FullScreen from 'react-fullscreen-crossbrowser'
import { Option } from '@helpers/createOptionField'
import Select from 'react-select'
import { makeStyles } from '@material-ui/styles'

const useLiveTrainsStyles = makeStyles({
fullscreenButton: {
display: 'flex',
alignItems: 'center',
marginBottom: 16,
},
iframe: {
border: 'none',
width: '100%',
height: 400,

':fullscreen &': {
height: '100%',
},
},
})

function LiveTrainAnnouncements({ nextTrainHandler, system }: LiveTrainAnnouncementsProps) {
const classes = useLiveTrainsStyles()

const supportedStations: Option[] = useMemo(
() =>
system.stations.map(s => {
const r = crsToStationItemMapper(s)

return {
value: r.crsCode,
label: r.name,
}
}),
[system.stations],
)

const [isFullscreen, setFullscreen] = useState(false)
const [selectedCrs, setselectedCrs] = useState('ECR')
const [hasEnabledFeature, setHasEnabledFeature] = useState(false)
const [isPlaying, setIsPlaying] = useState(false)

const [nextTrainAnnounced, setNextTrainAnnounced] = useState<Record<string, number>>({})

function removeOldIds() {
const now = Date.now()

setNextTrainAnnounced(prev => {
const newIds = Object.fromEntries(Object.entries(prev).filter(([_, v]) => now - v < 1000 * 60 * 60))

return newIds
})
}

function markTrainIdAnnounced(id: string) {
setNextTrainAnnounced(prev => ({
...prev,
[id]: Date.now(),
}))
}

useEffect(() => {
const key = setInterval(removeOldIds, 1000 * 60 * 5)

return () => {
clearInterval(key)
}
}, [removeOldIds])

useEffect(() => {
if (!hasEnabledFeature) return

const checkAndPlay = async () => {
if (isPlaying) return

const resp = await fetch(
`https://national-rail-api.davwheat.dev/departures/${selectedCrs}?expand=true&numServices=3&timeOffset=0&timeWindow=10`,
)

if (!resp.ok) return

let services

try {
const data = await resp.json()
services = data.trainServices
} catch {
return
}

if (!services) return

const firstUnannounced = services.find(
s => !nextTrainAnnounced[s.rsid] && !s.isCancelled && s.etd !== 'Delayed' && s.platform !== null && !!s.length,
)
if (!firstUnannounced) return

console.log(firstUnannounced)

markTrainIdAnnounced(firstUnannounced.rsid)

const h = firstUnannounced.std.split(':')[0]
const m = firstUnannounced.std.split(':')[1]

const options: INextTrainAnnouncementOptions = {
chime: 'four',
hour: h === '00' ? '00 - midnight' : h,
min: m === '00' ? '00 - hundred' : m,
toc: system.AVAILABLE_TOCS.find(t => t.toLowerCase() === firstUnannounced.operator.toLowerCase()) ?? '',
coaches: `${firstUnannounced.length} coaches`,
platform: system.platforms.includes(firstUnannounced.platform.toLowerCase()) ? firstUnannounced.platform.toLowerCase() : '1',
terminatingStationCode: firstUnannounced.destination[0].crs,
vias: [],
callingAt: firstUnannounced.subsequentCallingPoints[0].callingPoint
.map((p): CallingAtPoint | null => {
if (p.isCancelled || p.et === 'Cancelled') return null
if (!system.stations.includes(p.crs)) return null

return {
crsCode: p.crs,
name: '',
randomId: '',
}
})
.filter(x => !!x),
}

console.log(options)

setIsPlaying(true)
try {
console.log('PLAYING')
await nextTrainHandler(options)
} catch (e) {
console.log('FAILED')
console.error(e)
setIsPlaying(false)
}
console.log('COMPLETE')
setIsPlaying(false)
}

const refreshInterval = setInterval(checkAndPlay, 30_000)
checkAndPlay()

return () => {
clearInterval(refreshInterval)
}
}, [hasEnabledFeature, nextTrainAnnounced, markTrainIdAnnounced, system, nextTrainHandler, selectedCrs, isPlaying, setIsPlaying])

return (
<div>
<label className="option-select" htmlFor="station-select">
Station
<Select<Option, false>
id="station-select"
value={{ value: selectedCrs, label: supportedStations.find(option => option.value === selectedCrs)?.label || '' }}
onChange={val => {
setselectedCrs(val!!.value)
}}
options={supportedStations}
/>
</label>

<p style={{ margin: '16px 0' }}>
This is a beta feature, and isn't complete or fully functional. Please report any issues you face on GitHub.
</p>
<p style={{ margin: '16px 0' }}>
This page will auto-announce all departures in the next 10 minutes from the selected station. Departures outside this timeframe will
appear on the board below, but won't be announced until closer to the time.
</p>
<p style={{ margin: '16px 0' }}>
At the moment, we also won't announce services which:
<ul className="list">
<li>have no platform allocated in data feeds (common at larger stations, even at the time of departure)</li>
<li>have no train formation info (num of coaches)</li>
<li>are marked as cancelled or have an estimated time of "delayed"</li>
<li>have already been announced by the system in the last hour (only affects services which suddenly get delayed)</li>
</ul>
</p>
<p>
We also can't handle splits (we'll only announce the main portion), request stops, short platforms, delays (e.g., "for the delayed") and
many more features. As I said, it's a beta!
</p>

{!hasEnabledFeature ? (
<>
<button className={classes.fullscreenButton} onClick={() => setHasEnabledFeature(true)}>
Enable live trains
</button>
</>
) : (
<>
<button className={classes.fullscreenButton} onClick={() => setFullscreen(true)}>
<FullscreenIcon style={{ marginRight: 4 }} /> Fullscreen
</button>

<FullScreen enabled={isFullscreen} onChange={setFullscreen}>
<iframe
className={classes.iframe}
src={`https://raildotmatrix.davwheat.dev/board/?type=gtr-new&station=${selectedCrs}&noBg=1&hideSettings=1`}
/>
</FullScreen>
</>
)}
</div>
)
}
2 changes: 1 addition & 1 deletion src/helpers/createOptionField.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -12,7 +12,7 @@ interface OptionFieldOptions {
activeState?: Record<string, unknown>
}

interface Option {
export interface Option {
readonly label: string
readonly value: string
}
Expand Down
8 changes: 8 additions & 0 deletions yarn.lock
Original file line number Diff line number Diff line change
Expand Up @@ -12328,6 +12328,7 @@ __metadata:
react: ^18.2.0
react-beautiful-dnd: ^13.1.1
react-dom: ^18.2.0
react-fullscreen-crossbrowser: ^1.1.3
react-head: ^3.4.2
react-select: ^5.7.7
react-tabs: ^6.0.2
Expand Down Expand Up @@ -12473,6 +12474,13 @@ __metadata:
languageName: node
linkType: hard

"react-fullscreen-crossbrowser@npm:^1.1.3":
version: 1.1.3
resolution: "react-fullscreen-crossbrowser@npm:1.1.3"
checksum: bbf3ae87e22d3049d3605155d33b46632680b9a76d0911433ce245bb3636f52cfc99c0c0e8a1eb251fbcd0f63bdffae90cc3d447f74f1b616ac6e1f147c0b527
languageName: node
linkType: hard

"react-head@npm:^3.4.2":
version: 3.4.2
resolution: "react-head@npm:3.4.2"
Expand Down

0 comments on commit ac65678

Please sign in to comment.