2
0
Fork 0
mirror of https://github.com/koel/koel synced 2025-01-03 00:08:45 +00:00
koel/resources/assets/js/components/screens/VisualizerScreen.vue

100 lines
3 KiB
Vue
Raw Normal View History

2022-11-06 17:09:06 +00:00
<template>
2024-03-16 18:11:08 +00:00
<section id="vizContainer" ref="container" :class="{ fullscreen: isFullscreen }" @dblclick.prevent="toggleFullscreen">
2024-04-04 22:20:42 +00:00
<div
class="absolute z-[1] w-full h-full top-0 left-0 opacity-0 transition-opacity
duration-300 ease-in-out hover:opacity-100"
>
<div
v-if="selectedVisualizer"
class="absolute bottom-8 left-8 px-6 py-4 bg-black/30 rounded-md"
>
<h3 class="text-lg mb-2">{{ selectedVisualizer.name }}</h3>
<p v-if="selectedVisualizer.credits" class="text-k-text-secondary">
2022-11-06 17:09:06 +00:00
by {{ selectedVisualizer.credits.author }}
2024-04-04 22:20:42 +00:00
<a :href="selectedVisualizer.credits.url" class="ml-2" target="_blank">
2023-11-10 13:16:06 +00:00
<Icon :icon="faUpRightFromSquare" />
2022-11-06 17:09:06 +00:00
</a>
</p>
</div>
2024-04-04 22:20:42 +00:00
<div class="absolute bottom-8 right-8 border border-white/30 rounded-md">
2024-04-18 21:51:06 +00:00
<SelectBox v-model="selectedId" class="!bg-black/20 !text-white block">
2024-04-04 22:20:42 +00:00
<option disabled value="-1">Pick a visualizer</option>
<option v-for="v in visualizers" :key="v.id" :value="v.id">{{ v.name }}</option>
</SelectBox>
</div>
2022-11-06 17:09:06 +00:00
</div>
2024-04-04 22:20:42 +00:00
<div ref="el" class="viz h-full w-full absolute z-0" />
2022-11-06 17:09:06 +00:00
</section>
</template>
<script lang="ts" setup>
import { faUpRightFromSquare } from '@fortawesome/free-solid-svg-icons'
import { nextTick, onBeforeUnmount, onMounted, ref, watch } from 'vue'
2024-03-16 18:11:08 +00:00
import { useFullscreen } from '@vueuse/core'
2022-11-06 17:09:06 +00:00
import { logger } from '@/utils'
import { preferenceStore as preferences, visualizerStore } from '@/stores'
2024-04-04 22:20:42 +00:00
import SelectBox from '@/components/ui/form/SelectBox.vue'
2022-11-06 17:09:06 +00:00
const visualizers = visualizerStore.all
let destroyVisualizer: () => void
2024-03-16 18:11:08 +00:00
const el = ref<HTMLElement | null>(null)
const container = ref<HTMLElement | null>(null)
2022-11-06 17:09:06 +00:00
const selectedId = ref<Visualizer['id']>()
2024-03-16 18:11:08 +00:00
const { isFullscreen, toggle: toggleFullscreen } = useFullscreen(container)
2022-11-06 17:09:06 +00:00
const render = async (viz: Visualizer) => {
if (!el.value) {
await nextTick()
await render(viz)
}
freeUp()
try {
destroyVisualizer = await viz.init(el.value!)
} catch (error: unknown) {
2022-11-06 17:09:06 +00:00
// in e.g., DOM testing, the call will fail due to the lack of proper API support
logger.warn('Failed to initialize visualizer', error)
2022-11-06 17:09:06 +00:00
}
}
const selectedVisualizer = ref<Visualizer>()
watch(selectedId, id => {
preferences.visualizer = id
selectedVisualizer.value = visualizerStore.getVisualizerById(id || 'default')!
render(selectedVisualizer.value)
})
onMounted(() => {
selectedId.value = preferences.visualizer || 'default'
if (!visualizerStore.getVisualizerById(selectedId.value)) {
selectedId.value = 'default'
}
})
const freeUp = () => {
destroyVisualizer?.()
el.value && (el.value.innerHTML = '')
}
onBeforeUnmount(() => freeUp())
</script>
2024-04-04 22:20:42 +00:00
<style lang="postcss" scoped>
:deep(canvas) {
@apply transition-opacity duration-300 h-full w-full;
}
2022-11-06 17:09:06 +00:00
2024-04-04 22:20:42 +00:00
.fullscreen {
/* :fullscreen pseudo support is kind of buggy, so we use a class instead */
@apply bg-k-bg-primary;
2022-11-06 17:09:06 +00:00
}
</style>