59 lines
1.8 KiB
Vue
59 lines
1.8 KiB
Vue
<script setup lang="ts">
|
|
import { search, type Result } from '~/openapi/search'
|
|
|
|
const route = useRoute()
|
|
const searchQuery = ref(route.query.title as string || '')
|
|
|
|
const results = ref<Result[]>([])
|
|
const isLoading = ref(false)
|
|
const error = ref<unknown>(null)
|
|
|
|
watchEffect(async () => {
|
|
if (!searchQuery.value) return
|
|
|
|
try {
|
|
isLoading.value = true
|
|
error.value = null
|
|
const response = await search({ title: searchQuery.value })
|
|
results.value = response.data.results || []
|
|
} catch (err) {
|
|
error.value = err
|
|
console.error('Search failed:', err)
|
|
} finally {
|
|
isLoading.value = false
|
|
}
|
|
})
|
|
</script>
|
|
|
|
<template>
|
|
<div>
|
|
<div v-if="isLoading">
|
|
Loading results...
|
|
</div>
|
|
|
|
<div v-if="error">
|
|
Error loading results: {{ error }}
|
|
</div>
|
|
|
|
<div v-if="results.length > 0" class="grid grid-cols-[repeat(auto-fill,32rem)] justify-around gap-8 grid-flow-row">
|
|
<div v-for="item in results" :key="item.id" class="flex w-[32rem]">
|
|
<NuxtLink :to="`/anime/${item.id}`">
|
|
<img v-if="item.material_data?.anime_poster_url" :src="item.material_data.anime_poster_url" :alt="item.title"
|
|
class="rounded-md">
|
|
<div>
|
|
<h3 className="text-2xl font-semibold tracking-tight">{{ item.title }}</h3>
|
|
<p v-if="item.material_data?.year">Year: {{ item.material_data.year }}</p>
|
|
<p v-if="item.material_data?.anime_kind">Type: {{ item.material_data.anime_kind }}</p>
|
|
<p v-if="item.material_data?.episodes_total">Episodes: {{ item.material_data.episodes_total }}
|
|
</p>
|
|
</div>
|
|
</NuxtLink>
|
|
</div>
|
|
</div>
|
|
|
|
<div v-if="!isLoading && !error && results.length === 0 && searchQuery">
|
|
No results found for "{{ searchQuery }}"
|
|
</div>
|
|
</div>
|
|
</template>
|