25'ten fazla konu seçemezsiniz Konular bir harf veya rakamla başlamalı, kısa çizgiler ('-') içerebilir ve en fazla 35 karakter uzunluğunda olabilir.
 
 
 
 
 

134 satır
3.3 KiB

  1. <template>
  2. <section class="flex flex-col items-center min-h-screen">
  3. <uiLoader v-if="isLoading" class="fixed z-50 backdrop-blur-sm p-4 rounded-full top-1/2" />
  4. <div class="container w-full p-6 max-w-6xl grow flex flex-col items-center">
  5. <nav
  6. class="w-full flex bg-slate-800 font-semibold rounded-full backdrop-blur sticky top-0 justify-center items-center hover:ring">
  7. <input ref="searchInput" v-model="terms" autofocus placeholder="search" type="search"
  8. class="rounded-full text-xl w-full text-center text-white p-4 bg-transparent focus-visible:border-none"
  9. @keypress.enter="pressEnter()">
  10. <b v-if="films && films.length"
  11. class="px-4 py-2 absolute right-2 block bg-green-400 text-slate-800 rounded-full">
  12. {{ films.length }}
  13. </b>
  14. </nav>
  15. <main class="flex max-w-screen-2xl justify-center flex-wrap m-6">
  16. <NuxtLink v-for="(film, index) in films" :key="index" :href="film.title" :to="'/details/' + film.id"
  17. class="hover:bg-green-500 transition flex flex-col bg-slate-400 rounded-lg w-60 m-4">
  18. <img @error="setPlaceholder" :src="config.public.IMG_BASE_URL + film.poster_path" :alt="film.title"
  19. class="w-full rounded-t-lg">
  20. <span class="p-4 text-white flex flex-col">
  21. <h2 class="text-xl font-bold pb-2">
  22. {{ film.title }}
  23. </h2>
  24. <span v-if="film.release_date">
  25. {{ useDateFormat(film.release_date, 'D MMM YYYY') }}
  26. </span>
  27. <span class="font-bold">
  28. {{ formatPercent(film.vote_average) }}%
  29. </span>
  30. </span>
  31. </NuxtLink>
  32. </main>
  33. </div>
  34. </section>
  35. </template>
  36. <script setup lang="ts">
  37. import { useDateFormat } from '@vueuse/core'
  38. const config = useRuntimeConfig()
  39. const terms = ref('')
  40. const films = ref([])
  41. const page = ref(1)
  42. const isLoading = ref(true)
  43. const { onScrollEnd } = useScrollEnd()
  44. const search = async () => {
  45. isLoading.value = true
  46. if (terms.value !== '') {
  47. const { data } = await useFetch(`/api/search/${terms.value}-${page.value}`)
  48. console.log(data)
  49. films.value.push(...data.value)
  50. } else {
  51. const { data } = await useFetch(`/api/explore/${page.value}`)
  52. films.value.push(...data.value)
  53. }
  54. isLoading.value = false
  55. }
  56. const pressEnter = () => {
  57. page.value = 1
  58. films.value = []
  59. search()
  60. }
  61. const setPlaceholder = (event) => {
  62. event.target.src = 'https://via.placeholder.com/200x300'
  63. }
  64. onScrollEnd(() => {
  65. page.value += 1
  66. search()
  67. })
  68. onMounted(() => {
  69. nextTick(() => {
  70. search()
  71. })
  72. })
  73. </script>
  74. <style>
  75. .title b {
  76. @apply text-green-400 capitalize;
  77. }
  78. .icon {
  79. @apply w-4 h-4 mx-2;
  80. }
  81. .icon-container {
  82. color: #8094ae;
  83. @apply flex items-center
  84. }
  85. .icon-container .icon {
  86. color: rgb(109, 109, 109);
  87. }
  88. .icon-size p {
  89. color: rgb(54, 150, 75);
  90. }
  91. .icon-seed p {
  92. color: rgb(12, 108, 233);
  93. }
  94. .loader>div {
  95. animation: loader 2.4s cubic-bezier(0, 0.2, 0.8, 1) infinite;
  96. }
  97. @keyframes loader {
  98. 0%,
  99. 100% {
  100. animation-timing-function: cubic-bezier(0.5, 0, 1, 0.5);
  101. }
  102. 0% {
  103. transform: rotateY(0deg);
  104. }
  105. 50% {
  106. transform: rotateY(1800deg);
  107. animation-timing-function: cubic-bezier(0, 0.5, 0.5, 1);
  108. }
  109. 100% {
  110. transform: rotateY(3600deg);
  111. }
  112. }
  113. </style>