mirror of
https://github.com/koel/koel
synced 2025-01-11 20:28:49 +00:00
128 lines
4.7 KiB
Vue
128 lines
4.7 KiB
Vue
<template>
|
|
<a
|
|
data-testid="episode-item"
|
|
class="group relative flex flex-col md:flex-row gap-4 px-6 py-5 !text-k-text-primary hover:bg-white/10 duration-200"
|
|
:class="isCurrentEpisode && 'current'"
|
|
:href="`/#/episodes/${episode.id}`"
|
|
@contextmenu.prevent="requestContextMenu"
|
|
@dragstart="onDragStart"
|
|
>
|
|
<Icon :icon="faBookmark" size="xl" class="absolute -top-1 right-3 text-k-accent" v-if="isCurrentEpisode" />
|
|
<button
|
|
class="hidden md:block md:flex-[0_0_128px] relative overflow-hidden rounded-lg active:scale-95"
|
|
role="button"
|
|
@click.prevent="playOrPause"
|
|
>
|
|
<img
|
|
:src="episode.episode_image"
|
|
alt="Episode thumbnail"
|
|
class="w-[128px] aspect-square object-cover"
|
|
loading="lazy"
|
|
/>
|
|
<span class="absolute top-0 left-0 w-full h-full group-hover:bg-black/40 z-10" />
|
|
<span
|
|
class="absolute flex opacity-0 items-center justify-center w-[48px] aspect-square rounded-full top-1/2
|
|
left-1/2 -translate-x-1/2 -translate-y-1/2 bg-k-highlight group-hover:opacity-100 duration-500 transition z-20"
|
|
>
|
|
<Icon v-if="isPlaying" :icon="faPause" class="text-white" size="2xl" />
|
|
<Icon v-else :icon="faPlay" class="ml-1 text-white" size="2xl" />
|
|
</span>
|
|
</button>
|
|
<div class="flex-1">
|
|
<time
|
|
:datetime="episode.created_at"
|
|
:title="episode.created_at"
|
|
class="block uppercase text-sm mb-1 text-k-text-secondary"
|
|
>
|
|
{{ publicationDateForHumans }}
|
|
</time>
|
|
|
|
<h3 class="text-xl" :title="episode.title">{{ episode.title }}</h3>
|
|
<div class="description text-k-text-secondary mt-3 line-clamp-3" v-html="description" />
|
|
</div>
|
|
<div class="md:flex-[0_0_122px] text-sm text-k-text-secondary flex md:flex-col items-center justify-center w-full">
|
|
<span class="block md:mb-2">{{ timeLeft ? timeLeft : 'Played' }}</span>
|
|
<div class="px-4 flex-1 md:flex-grow-0 md:w-full">
|
|
<EpisodeProgress v-if="shouldShowProgress" :episode="episode" :position="currentPosition" />
|
|
</div>
|
|
</div>
|
|
</a>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import DOMPurify from 'dompurify'
|
|
import { orderBy } from 'lodash'
|
|
import { faBookmark, faPause, faPlay } from '@fortawesome/free-solid-svg-icons'
|
|
import { computed, defineAsyncComponent, toRefs } from 'vue'
|
|
import { eventBus, secondsToHumanReadable } from '@/utils'
|
|
import { useDraggable } from '@/composables'
|
|
import { formatTimeAgo } from '@vueuse/core'
|
|
import { playbackService } from '@/services'
|
|
import { preferenceStore as preferences, queueStore, songStore as episodeStore } from '@/stores'
|
|
|
|
const EpisodeProgress = defineAsyncComponent(() => import('@/components/podcast/EpisodeProgress.vue'))
|
|
|
|
const props = defineProps<{ episode: Episode, podcast: Podcast }>()
|
|
const { episode, podcast } = toRefs(props)
|
|
|
|
const { startDragging } = useDraggable('playables')
|
|
|
|
const publicationDateForHumans = computed(() => {
|
|
const publishedAt = new Date(episode.value.created_at)
|
|
|
|
if ((Date.now() - publishedAt.getTime()) / (1000 * 60 * 60 * 24) < 31) {
|
|
return formatTimeAgo(publishedAt)
|
|
}
|
|
|
|
return publishedAt.toLocaleDateString(undefined, {
|
|
year: 'numeric',
|
|
month: 'short',
|
|
day: 'numeric'
|
|
})
|
|
})
|
|
|
|
const currentPosition = computed(() => podcast.value.state.progresses[episode.value.id] || 0)
|
|
|
|
const timeLeft = computed(() => {
|
|
if (currentPosition.value === 0) return secondsToHumanReadable(episode.value.length)
|
|
const secondsLeft = episode.value.length - currentPosition.value
|
|
return secondsLeft === 0 ? 0 : `${secondsToHumanReadable(secondsLeft)} left`
|
|
})
|
|
|
|
const shouldShowProgress = computed(() => timeLeft.value !== 0 && episode.value.length && currentPosition.value)
|
|
const isCurrentEpisode = computed(() => podcast.value.state.current_episode === episode.value.id)
|
|
const description = computed(() => DOMPurify.sanitize(episode.value.episode_description))
|
|
|
|
const onDragStart = (event: DragEvent) => startDragging(event, episode.value)
|
|
const requestContextMenu = (event: MouseEvent) => eventBus.emit('PLAYABLE_CONTEXT_MENU_REQUESTED', event, episode.value)
|
|
|
|
const isPlaying = computed(() => episode.value.playback_state === 'Playing')
|
|
|
|
const playOrPause = async () => {
|
|
if (isPlaying.value) {
|
|
return playbackService.pause()
|
|
}
|
|
|
|
if (episode.value.playback_state === 'Paused') {
|
|
return playbackService.resume()
|
|
}
|
|
|
|
if (preferences.continuous_playback) {
|
|
queueStore.replaceQueueWith(orderBy(await episodeStore.fetchForPodcast(podcast.value.id), 'created_at'))
|
|
}
|
|
|
|
await playbackService.play(episode.value, currentPosition.value >= episode.value.length ? 0 : currentPosition.value)
|
|
}
|
|
</script>
|
|
|
|
<style scoped lang="postcss">
|
|
.description {
|
|
:deep(p) {
|
|
@apply mb-3;
|
|
}
|
|
|
|
:deep(a) {
|
|
@apply text-k-text-primary hover:text-k-accent;
|
|
}
|
|
}
|
|
</style>
|