forked from Mastodon/mastoradio-la-radio-di-mastodon
136 lines
4.0 KiB
Svelte
136 lines
4.0 KiB
Svelte
<svelte:head>
|
|
<title>{`${ $current ? `${$current.title} ∴ ` : ''}Eldritch Radio`}</title>
|
|
</svelte:head>
|
|
|
|
<div class="app container">
|
|
<Header></Header>
|
|
|
|
<section class="viewer">
|
|
<Viewer></Viewer>
|
|
</section>
|
|
|
|
<section class="queue">
|
|
<Queue></Queue>
|
|
</section>
|
|
|
|
<Footer></Footer>
|
|
</div>
|
|
|
|
<script>
|
|
import { setContext } from 'svelte'
|
|
import Header from '/components/layout/Header.svelte'
|
|
import Footer from '/components/layout/Footer.svelte'
|
|
import Queue from '/components/Queue.svelte'
|
|
import Viewer from '/components/Viewer.svelte'
|
|
import { get, writable, writableLocalStorage, derived, scan, wait, startWith, distinct } from '/services/store.js'
|
|
import { radioIterator, radioShareIterator } from '/services/radio.js'
|
|
import DeepSet from '/services/deep-set.js'
|
|
|
|
export let share
|
|
|
|
const cache = new DeepSet()
|
|
|
|
const domain = writableLocalStorage('domain', 'eldritch.cafe')
|
|
|
|
const hashtags = writableLocalStorage('hashtags', [
|
|
'np',
|
|
'nowplaying',
|
|
'tootradio',
|
|
'pouetradio'
|
|
])
|
|
|
|
const paused = writable(true)
|
|
const volume = writableLocalStorage('volume', 100)
|
|
|
|
const current = writable(null)
|
|
const enqueueing = writable(false)
|
|
const loading = writable(false)
|
|
|
|
|
|
|
|
const iterator = derived([domain, hashtags], ([$domain, $hashtags], set) => {
|
|
const iterator = share == null
|
|
? radioIterator($domain, $hashtags, cache)
|
|
// this is a bit dump because we always requeue the shared track
|
|
: radioShareIterator(share, $domain, $hashtags, cache)
|
|
|
|
set(iterator)
|
|
|
|
return () => {
|
|
iterator.return()
|
|
}
|
|
}, null)
|
|
|
|
const next = derived([iterator, current], ([$iterator, $current]) => ({ $iterator, $current }))
|
|
.pipe(scan(($nextPromise, { $iterator, $current }) => {
|
|
return $nextPromise.then($next => {
|
|
if ($next == null || $next === $current) {
|
|
enqueueing.set(true)
|
|
return $iterator.next().then(({ done, value }) => {
|
|
enqueueing.set(false)
|
|
return value
|
|
})
|
|
} else {
|
|
return $nextPromise
|
|
}
|
|
})
|
|
}, Promise.resolve(null)))
|
|
.pipe(wait(x => x))
|
|
// distinct but with strict check
|
|
.pipe(distinct())
|
|
.pipe(startWith(null))
|
|
|
|
|
|
const queue = next
|
|
.pipe(scan((a, x) => x == null ? a : [...a, x], []))
|
|
|
|
const index = derived([queue, current], ([$queue, $current]) => {
|
|
const i = $queue.indexOf($current)
|
|
return i === -1 ? null : i
|
|
})
|
|
|
|
const canPrevious = derived([queue, index], ([$queue, $index]) => $index !== null && $index > 0)
|
|
const canNext = derived([queue, index], ([$queue, $index]) => $index !== null && $index < $queue.length - 1)
|
|
|
|
|
|
const select = track => {
|
|
console.log(`Select ${track.title}`)
|
|
current.set(track)
|
|
}
|
|
|
|
const selectPrevious = () => {
|
|
if (get(canPrevious)) {
|
|
const $queue = get(queue)
|
|
const $index = get(index)
|
|
select($queue[$index - 1])
|
|
}
|
|
}
|
|
|
|
const selectNext = () => {
|
|
if (get(canNext)) {
|
|
const $queue = get(queue)
|
|
const $index = get(index)
|
|
select($queue[$index + 1])
|
|
}
|
|
}
|
|
|
|
setContext('paused', paused)
|
|
setContext('volume', volume)
|
|
setContext('domain', domain)
|
|
setContext('hashtags', hashtags)
|
|
setContext('current', current)
|
|
setContext('loading', loading)
|
|
setContext('enqueueing', enqueueing)
|
|
setContext('iterator', iterator)
|
|
setContext('next', next)
|
|
setContext('queue', queue)
|
|
setContext('canPrevious', canPrevious)
|
|
setContext('canNext', canNext)
|
|
setContext('select', select)
|
|
setContext('selectPrevious', selectPrevious)
|
|
setContext('selectNext', selectNext)
|
|
|
|
$: if ($queue.length === 1 && $next != null && $current == null) {
|
|
select($next)
|
|
}
|
|
</script> |