選択できるのは25トピックまでです。 トピックは、先頭が英数字で、英数字とダッシュ('-')を使用した35文字以内のものにしてください。
 
 
 
 
 

140 行
3.3 KiB

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