@@ -1,2 +0,0 @@ | |||||
- variables typées | |||||
- tests |
@@ -13,7 +13,7 @@ | |||||
</b> | </b> | ||||
</nav> | </nav> | ||||
<main class="flex max-w-screen-2xl justify-center flex-wrap m-6"> | <main class="flex max-w-screen-2xl justify-center flex-wrap m-6"> | ||||
<NuxtLink v-for="(film, index) in films" :key="index" :href="film.title" :to="'/details/' + film.id" | |||||
<NuxtLink v-for="(film, index) in films" :key="index" :title="film.title" :to="'/details/' + film.id" | |||||
class="hover:bg-green-500 transition flex flex-col bg-slate-400 rounded-lg w-60 m-4"> | class="hover:bg-green-500 transition flex flex-col bg-slate-400 rounded-lg w-60 m-4"> | ||||
<img @error="setPlaceholder" :src="config.public.IMG_BASE_URL + film.poster_path" :alt="film.title" | <img @error="setPlaceholder" :src="config.public.IMG_BASE_URL + film.poster_path" :alt="film.title" | ||||
class="w-full rounded-t-lg"> | class="w-full rounded-t-lg"> | ||||
@@ -37,9 +37,17 @@ | |||||
<script setup lang="ts"> | <script setup lang="ts"> | ||||
import { useDateFormat } from '@vueuse/core' | import { useDateFormat } from '@vueuse/core' | ||||
interface Film { | |||||
title: string, | |||||
id: number, | |||||
poster_path: string, | |||||
release_date: number, | |||||
vote_average: number, | |||||
} | |||||
const config = useRuntimeConfig() | const config = useRuntimeConfig() | ||||
const terms = ref('') | const terms = ref('') | ||||
const films = ref([]) | |||||
const films: Ref<Film[]> = ref([]) | |||||
const page = ref(1) | const page = ref(1) | ||||
const isLoading = ref(true) | const isLoading = ref(true) | ||||
const { onScrollEnd } = useScrollEnd() | const { onScrollEnd } = useScrollEnd() | ||||
@@ -63,8 +71,9 @@ const pressEnter = () => { | |||||
search() | search() | ||||
} | } | ||||
const setPlaceholder = (event) => { | |||||
event.target.src = 'https://via.placeholder.com/200x300' | |||||
const setPlaceholder = (event: Event) => { | |||||
const target = event.target as HTMLImageElement | |||||
target.src = 'https://via.placeholder.com/200x300' | |||||
} | } | ||||
onScrollEnd(() => { | onScrollEnd(() => { | ||||
@@ -1,6 +1,6 @@ | |||||
export function useScrollEnd() { | export function useScrollEnd() { | ||||
let callback = null | |||||
let callback: Function | null = null | |||||
const handleScroll = () => { | const handleScroll = () => { | ||||
const scrollPosition = window.innerHeight + window.scrollY | const scrollPosition = window.innerHeight + window.scrollY | ||||
@@ -11,7 +11,7 @@ export function useScrollEnd() { | |||||
} | } | ||||
} | } | ||||
const onScrollEnd = (cb) => { | |||||
const onScrollEnd = (cb: Function | null) => { | |||||
callback = cb | callback = cb | ||||
} | } | ||||
@@ -8,6 +8,7 @@ | |||||
"hasInstallScript": true, | "hasInstallScript": true, | ||||
"dependencies": { | "dependencies": { | ||||
"@pinia/nuxt": "^0.5.5", | "@pinia/nuxt": "^0.5.5", | ||||
"@types/node-fetch": "^2.6.11", | |||||
"axios": "^1.7.7", | "axios": "^1.7.7", | ||||
"node-fetch": "^3.3.2", | "node-fetch": "^3.3.2", | ||||
"nuxt": "^3.12.3", | "nuxt": "^3.12.3", | ||||
@@ -1712,6 +1713,15 @@ | |||||
"undici-types": "~5.26.4" | "undici-types": "~5.26.4" | ||||
} | } | ||||
}, | }, | ||||
"node_modules/@types/node-fetch": { | |||||
"version": "2.6.11", | |||||
"resolved": "https://registry.npmjs.org/@types/node-fetch/-/node-fetch-2.6.11.tgz", | |||||
"integrity": "sha512-24xFj9R5+rfQJLRyM56qh+wnVSYhyXC2tkoBndtY0U+vubqNsYXGjufB2nn8Q6gt0LrARwL6UBtMCSVCwl4B1g==", | |||||
"dependencies": { | |||||
"@types/node": "*", | |||||
"form-data": "^4.0.0" | |||||
} | |||||
}, | |||||
"node_modules/@types/resolve": { | "node_modules/@types/resolve": { | ||||
"version": "1.20.2", | "version": "1.20.2", | ||||
"resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.20.2.tgz", | "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-1.20.2.tgz", | ||||
@@ -4,61 +4,113 @@ | |||||
<NuxtLink class="w-full m-4 flex justify-end" :to="'/'"> | <NuxtLink class="w-full m-4 flex justify-end" :to="'/'"> | ||||
<uiClose /> | <uiClose /> | ||||
</NuxtLink> | </NuxtLink> | ||||
<img class="rounded-lg mb-6" :src="config.public.IMG_BASE_URL + film.poster_path" :alt="film.title"> | |||||
<img class="rounded-lg mb-6" :src="config.public.IMG_BASE_URL + film?.poster_path" :alt="film?.title"> | |||||
<div class="poster flex flex-col lg:flex-row w-full justify-center"> | <div class="poster flex flex-col lg:flex-row w-full justify-center"> | ||||
<span> | <span> | ||||
<h1 class="text-4xl font-bold leading-relaxed"> | <h1 class="text-4xl font-bold leading-relaxed"> | ||||
{{ film.title }} | |||||
{{ film?.title }} | |||||
</h1> | </h1> | ||||
<h2 class="text-2xl text-slate-700"> | <h2 class="text-2xl text-slate-700"> | ||||
{{ director }} | {{ director }} | ||||
</h2> | </h2> | ||||
<p class="my-8"> | <p class="my-8"> | ||||
{{ film.overview }} | |||||
{{ film?.overview }} | |||||
</p> | </p> | ||||
<p> | <p> | ||||
{{ formatPercent(film.vote_average) }} % | |||||
{{ formatPercent(film?.vote_average as number) }} % | |||||
</p> | </p> | ||||
<p> | <p> | ||||
{{ film.vote_count }} votes | |||||
{{ film?.vote_count }} votes | |||||
</p> | </p> | ||||
<div class="flex justify-center"> | <div class="flex justify-center"> | ||||
<ul class="flex items-center overflow-y-hidden overflow-x-scroll w-full max-w-lg"> | <ul class="flex items-center overflow-y-hidden overflow-x-scroll w-full max-w-lg"> | ||||
<li v-for="star in film.credits.cast" class="flex flex-col items-center p-4 bg-slate-200 rounded m-4"> | |||||
<li v-for="star in film?.credits.cast" class="flex flex-col items-center p-4 bg-slate-200 rounded m-4"> | |||||
<UiPerson class="h-10" /> | <UiPerson class="h-10" /> | ||||
{{ star.name }} | {{ star.name }} | ||||
</li> | </li> | ||||
</ul> | </ul> | ||||
</div> | </div> | ||||
<ul class="flex"> | <ul class="flex"> | ||||
<li v-for="genre in film.genres" class="p-8"> {{ genre.name }} </li> | |||||
<li v-for="genre in film?.genres" class="p-8"> {{ genre.name }} </li> | |||||
</ul> | </ul> | ||||
</span> | </span> | ||||
</div> | </div> | ||||
<div> | <div> | ||||
<CommentForm :filmId="film.id" /> | |||||
<CommentList :filmId="film.id" /> | |||||
<CommentForm :filmId="film?.id" /> | |||||
<CommentList :filmId="film?.id" /> | |||||
</div> | </div> | ||||
</div> | </div> | ||||
</section> | </section> | ||||
</template> | </template> | ||||
<script setup lang="ts"> | <script setup lang="ts"> | ||||
const router = useRouter() | |||||
const config = useRuntimeConfig() | |||||
const film = ref() | |||||
const route = useRoute() | const route = useRoute() | ||||
const filmId = ref('') | |||||
const director = ref('') | |||||
const config = useRuntimeConfig() | |||||
interface FilmDetails { | |||||
title: string, | |||||
id: number, | |||||
poster_path: string, | |||||
release_date: number, | |||||
vote_average: number, | |||||
vote_count: number, | |||||
genres: genre[], | |||||
overview: string, | |||||
credits: Credits, | |||||
} | |||||
interface genre { | |||||
id: number, | |||||
name: string, | |||||
} | |||||
interface Credits { | |||||
id: string, | |||||
cast: Cast[], | |||||
crew: Crew[], | |||||
} | |||||
interface Cast { | |||||
adult: boolean, | |||||
gender: number, | |||||
id: number, | |||||
known_for_department: string, | |||||
name: string, | |||||
original_name: string, | |||||
popularity: number, | |||||
profile_path: string, | |||||
cast_id: number, | |||||
character: string, | |||||
credit_id: string, | |||||
order: number, | |||||
} | |||||
interface Crew { | |||||
adult: boolean, | |||||
gender: number, | |||||
id: number, | |||||
known_for_department: string, | |||||
name: string, | |||||
original_name: string, | |||||
popularity: number, | |||||
profile_path: string | null, | |||||
credit_id: string, | |||||
department: string, | |||||
job: string, | |||||
} | |||||
const film: Ref<FilmDetails | null> = ref(null) | |||||
const filmId: Ref<string> = ref('') | |||||
const director: Ref<string> = ref('') | |||||
filmId.value = route.params.id | |||||
filmId.value = route.params.id as string | |||||
if (filmId.value !== '') { | if (filmId.value !== '') { | ||||
const { data } = await useFetch(`/api/details/${filmId.value}`) | const { data } = await useFetch(`/api/details/${filmId.value}`) | ||||
film.value = data.value | |||||
director.value = film.value.credits.crew.filter((member) => member.job === 'Director') | |||||
director.value = director.value[0].name | |||||
film.value = data.value as FilmDetails | |||||
const dataDirector = film.value.credits.crew.filter((member) => member.job === 'Director') | |||||
director.value = dataDirector[0].name | |||||
} | } | ||||
onMounted(() => { | onMounted(() => { | ||||
@@ -1,13 +1,18 @@ | |||||
import fetch from 'node-fetch' | import fetch from 'node-fetch' | ||||
interface Data { | |||||
} | |||||
export default eventHandler(async (req) => { | export default eventHandler(async (req) => { | ||||
const page = req.context.params?.page || 1 | const page = req.context.params?.page || 1 | ||||
const url = `https://api.themoviedb.org/3/discover/movie?include_adult=false&include_video=false&language=en-US&page=${page}&sort_by=popularity.desc` | const url = `https://api.themoviedb.org/3/discover/movie?include_adult=false&include_video=false&language=en-US&page=${page}&sort_by=popularity.desc` | ||||
const response = await fetch(url, { | |||||
const response: Response = await fetch(url, { | |||||
method: 'get', | method: 'get', | ||||
headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer eyJhbGciOiJIUzI1NiJ9.eyJhdWQiOiI2MmVmZmYzYWU2YWYyNWQ0NTY4OGY3OTkxYjgyNmNhOCIsIm5iZiI6MTcyODA1NTIwMy4wNzcyNywic3ViIjoiNjcwMDA2ZjQ5ZWJlYTE5MDA2ZjgxZmJhIiwic2NvcGVzIjpbImFwaV9yZWFkIl0sInZlcnNpb24iOjF9.XuH-_0UggmULCgQQajKc-QsmlRYW2rqSenyhguE6wRU' } | headers: { 'Content-Type': 'application/json', 'Authorization': 'Bearer eyJhbGciOiJIUzI1NiJ9.eyJhdWQiOiI2MmVmZmYzYWU2YWYyNWQ0NTY4OGY3OTkxYjgyNmNhOCIsIm5iZiI6MTcyODA1NTIwMy4wNzcyNywic3ViIjoiNjcwMDA2ZjQ5ZWJlYTE5MDA2ZjgxZmJhIiwic2NvcGVzIjpbImFwaV9yZWFkIl0sInZlcnNpb24iOjF9.XuH-_0UggmULCgQQajKc-QsmlRYW2rqSenyhguE6wRU' } | ||||
}) | }) | ||||
const data = await response.json() | const data = await response.json() | ||||
return data.results | |||||
return data | |||||
}) | }) |