mirror of
https://github.com/ArthurDanjou/website.git
synced 2026-01-30 11:47:53 +01:00
working on talents actualisation
This commit is contained in:
@@ -1,7 +1,23 @@
|
|||||||
|
import { useTalentsStore } from '~/store/talents'
|
||||||
|
|
||||||
export async function useTalents() {
|
export async function useTalents() {
|
||||||
const { $trpc } = useNuxtApp()
|
const { $trpc } = useNuxtApp()
|
||||||
|
const { setCategory, setFavorite, getCategory, isFavorite } = useTalentsStore()
|
||||||
|
|
||||||
const getTalents = async (category: string, favorite: boolean) => await $trpc.talents.getTalents.query({ favorite, category })
|
const {
|
||||||
|
data: talents,
|
||||||
|
refresh: refreshTalents,
|
||||||
|
} = await useAsyncData('talents:talents', async () => await $trpc.talents.getTalents.query({ favorite: isFavorite.value, category: getCategory.value }))
|
||||||
|
|
||||||
|
async function switchCategory(category: string) {
|
||||||
|
setCategory(category)
|
||||||
|
await refreshTalents()
|
||||||
|
}
|
||||||
|
|
||||||
|
async function toggleFavorite() {
|
||||||
|
setFavorite()
|
||||||
|
await refreshTalents()
|
||||||
|
}
|
||||||
|
|
||||||
const {
|
const {
|
||||||
data: getCategories,
|
data: getCategories,
|
||||||
@@ -12,8 +28,11 @@ export async function useTalents() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
return {
|
return {
|
||||||
getTalents,
|
talents,
|
||||||
getCategories,
|
getCategories,
|
||||||
getCategoryById,
|
getCategoryById,
|
||||||
|
isFavorite,
|
||||||
|
switchCategory,
|
||||||
|
toggleFavorite,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,31 +1,12 @@
|
|||||||
<script lang="ts" setup>
|
<script lang="ts" setup>
|
||||||
import type { Talent } from '@prisma/client'
|
|
||||||
|
|
||||||
useHead({
|
useHead({
|
||||||
title: 'Discover new talents • Arthur Danjou',
|
title: 'Discover new talents • Arthur Danjou',
|
||||||
})
|
})
|
||||||
|
|
||||||
const talents = ref<Talent[]>()
|
const categories = ref<Array<{ label: string, slug: string }>>([{ label: 'All', slug: 'all' }])
|
||||||
|
const { getCategories, talents, getCategoryById, isFavorite, toggleFavorite, switchCategory } = await useTalents()
|
||||||
|
|
||||||
const currentFavorite = ref<boolean>(false)
|
getCategories.value?.forEach(category => categories.value.push({ label: category.name, slug: category.slug }))
|
||||||
const currentCategory = ref('all')
|
|
||||||
const { getCategories, getTalents, getCategoryById } = await useTalents()
|
|
||||||
|
|
||||||
talents.value = await getTalents(currentCategory.value, currentFavorite.value)
|
|
||||||
|
|
||||||
watch(currentFavorite, async () => {
|
|
||||||
talents.value = await getTalents(currentCategory.value, currentFavorite.value)
|
|
||||||
})
|
|
||||||
|
|
||||||
watch(currentCategory, async () => {
|
|
||||||
talents.value = await getTalents(currentCategory.value, currentFavorite.value)
|
|
||||||
})
|
|
||||||
|
|
||||||
const setCategory = (category: string) => currentCategory.value = category
|
|
||||||
|
|
||||||
function isCategory(category: string) {
|
|
||||||
return currentCategory.value === category
|
|
||||||
}
|
|
||||||
|
|
||||||
const appConfig = useAppConfig()
|
const appConfig = useAppConfig()
|
||||||
function getColor() {
|
function getColor() {
|
||||||
@@ -59,54 +40,45 @@ function getColor() {
|
|||||||
<UButton label="Join the talent's list" color="primary" />
|
<UButton label="Join the talent's list" color="primary" />
|
||||||
</NuxtLink>
|
</NuxtLink>
|
||||||
</div>
|
</div>
|
||||||
<div v-if="getCategories" class="flex gap-2 justify-between border-b border-zinc-100 dark:border-zinc-700/40 mb-4">
|
<div v-if="getCategories" class="flex gap-2 items-center justify-between border-b border-zinc-100 dark:border-zinc-700/40 mb-4">
|
||||||
<div class="flex gap-4 overflow-x-scroll sm:overflow-x-hidden">
|
<div class="flex gap-4 overflow-x-scroll sm:overflow-x-hidden">
|
||||||
<div
|
<UTabs :items="categories">
|
||||||
class="category"
|
<template #default="{ item }">
|
||||||
:class="{ 'current-category': isCategory('all') }"
|
<div class="flex items-center gap-2 relative w-full" @click="switchCategory(item.slug)">
|
||||||
@click.prevent="setCategory('all')"
|
<div class="w-full">{{ item.label }}</div >
|
||||||
>
|
</div>
|
||||||
All
|
</template>
|
||||||
</div>
|
</UTabs>
|
||||||
<div
|
|
||||||
v-for="category in getCategories"
|
|
||||||
:key="category.slug"
|
|
||||||
class="category"
|
|
||||||
:class="{ 'current-category': isCategory(category.slug) }"
|
|
||||||
@click.prevent="setCategory(category.slug)"
|
|
||||||
>
|
|
||||||
<p class="w-full">
|
|
||||||
{{ category.name }}
|
|
||||||
</p>
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
<UPopover>
|
<UPopover>
|
||||||
<UButton
|
<UButton
|
||||||
:icon="currentFavorite ? 'i-mdi-filter-variant-remove' : 'i-mdi-filter-variant'"
|
:icon="isFavorite ? 'i-mdi-filter-variant-remove' : 'i-mdi-filter-variant'"
|
||||||
color="primary"
|
color="primary"
|
||||||
variant="ghost"
|
variant="soft"
|
||||||
|
size="lg"
|
||||||
/>
|
/>
|
||||||
<template #panel>
|
<template #panel>
|
||||||
<div
|
<div
|
||||||
class="flex p-2 gap-2 items-center cursor-pointer select-none text-subtitle"
|
class="flex p-2 gap-2 items-center cursor-pointer select-none text-subtitle"
|
||||||
@click.prevent="currentFavorite = !currentFavorite"
|
@click.prevent="toggleFavorite()"
|
||||||
>
|
>
|
||||||
<UIcon v-if="currentFavorite" name="i-material-symbols-check-box-outline-rounded" />
|
<UIcon v-if="isFavorite" name="i-material-symbols-check-box-outline-rounded" />
|
||||||
<UIcon v-else name="i-material-symbols-check-box-outline-blank" />
|
<UIcon v-else name="i-material-symbols-check-box-outline-blank" />
|
||||||
<p>Show favorites only</p>
|
<p>Show favorites only</p>
|
||||||
</div>
|
</div>
|
||||||
</template>
|
</template>
|
||||||
</UPopover>
|
</UPopover>
|
||||||
</div>
|
</div>
|
||||||
<div class="mt-16 md:mt-20">
|
<div v-if="talents && getCategories" class="mt-16 md:mt-20">
|
||||||
<div v-if="talents && getCategories" class="grid grid-cols-1 gap-x-12 gap-y-16 sm:grid-cols-2 lg:grid-cols-3">
|
<div v-if="talents.length > 0" class="grid grid-cols-1 gap-x-12 gap-y-16 sm:grid-cols-2 lg:grid-cols-3">
|
||||||
<div
|
<div
|
||||||
v-for="talent in talents"
|
v-for="talent in talents"
|
||||||
:key="talent.name.toLowerCase().trim()"
|
:key="talent.name.toLowerCase().trim()"
|
||||||
class="group relative flex flex-col justify-between"
|
class="group relative flex flex-col justify-between"
|
||||||
>
|
>
|
||||||
<div class="flex gap-6 items-center">
|
<div class="flex ">
|
||||||
<img :src="talent.logo" class="z-20 h-8 w-8 rounded-md">
|
<div class="flex gap-6 items-center">
|
||||||
|
<img :src="talent.logo" class="z-20 h-12 w-12 rounded-md">
|
||||||
<div>
|
<div>
|
||||||
<h2 class="text-base font-semibold text-zinc-800 dark:text-zinc-100">
|
<h2 class="text-base font-semibold text-zinc-800 dark:text-zinc-100">
|
||||||
<div class="absolute -inset-y-6 -inset-x-4 z-0 scale-95 bg-zinc-50 opacity-0 transition group-hover:scale-100 group-hover:opacity-100 dark:bg-zinc-800/50 sm:-inset-x-6 sm:rounded-2xl" />
|
<div class="absolute -inset-y-6 -inset-x-4 z-0 scale-95 bg-zinc-50 opacity-0 transition group-hover:scale-100 group-hover:opacity-100 dark:bg-zinc-800/50 sm:-inset-x-6 sm:rounded-2xl" />
|
||||||
@@ -125,13 +97,14 @@ function getColor() {
|
|||||||
<p class="relative z-10 my-2 text-sm text-zinc-600 dark:text-zinc-400">
|
<p class="relative z-10 my-2 text-sm text-zinc-600 dark:text-zinc-400">
|
||||||
{{ talent.work }}
|
{{ talent.work }}
|
||||||
</p>
|
</p>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="flex gap-2 z-10">
|
<div class="flex gap-2 z-10">
|
||||||
<span
|
<span
|
||||||
v-for="category in talent.categories"
|
v-for="category in talent.categories"
|
||||||
:key="category"
|
:key="category.id"
|
||||||
class="text-[.7rem] px-0.5 text-subtitle rounded-md ring-1 ring-zinc-900/5 dark:border dark:border-zinc-700/50 dark:bg-zinc-800 dark:ring-0"
|
class="text-[.6rem] px-1 text-subtitle rounded-md ring-1 ring-zinc-900/5 dark:border dark:border-zinc-700/50 dark:bg-zinc-800 dark:ring-0"
|
||||||
>{{ getCategoryById(category.categoryId) }}</span>
|
>{{ getCategoryById(category.categoryId) }}</span>
|
||||||
</div>
|
</div>
|
||||||
<p class="relative z-10 mt-4 flex text-sm font-medium items-center" :class="getColor()">
|
<p class="relative z-10 mt-4 flex text-sm font-medium items-center" :class="getColor()">
|
||||||
@@ -140,18 +113,12 @@ function getColor() {
|
|||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<p v-else class="my-16 text-subtitle">
|
<p v-else-if="talents?.length === 0" class="my-8 text-subtitle">
|
||||||
|
There are no talents here. Maybe soon...
|
||||||
|
</p>
|
||||||
|
<p v-else class="my-8 text-subtitle">
|
||||||
The talents are loading...
|
The talents are loading...
|
||||||
</p>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
</section>
|
</section>
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<style lang="scss">
|
|
||||||
.category {
|
|
||||||
@apply min-w-fit cursor-pointer py-2 text-gray-500 dark:text-gray-400 duration-300 transition-colors hover:text-black dark:hover:text-white
|
|
||||||
}
|
|
||||||
.current-category {
|
|
||||||
@apply text-black dark:text-white border-b-2 border-black dark:border-white
|
|
||||||
}
|
|
||||||
</style>
|
|
||||||
|
|||||||
@@ -11,7 +11,7 @@ export const useColorStore = defineStore(
|
|||||||
appConfig.ui.primary = newColor
|
appConfig.ui.primary = newColor
|
||||||
}, { immediate: true })
|
}, { immediate: true })
|
||||||
|
|
||||||
const setColor = (color: string) => {
|
function setColor(color: string) {
|
||||||
colorCookie.value = color as ColorsTheme
|
colorCookie.value = color as ColorsTheme
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
30
src/store/talents.ts
Normal file
30
src/store/talents.ts
Normal file
@@ -0,0 +1,30 @@
|
|||||||
|
import { defineStore } from 'pinia'
|
||||||
|
|
||||||
|
export const useTalentsStore = defineStore(
|
||||||
|
'talents',
|
||||||
|
() => {
|
||||||
|
const currentCategory = ref<string>('all')
|
||||||
|
const currentFavorite = ref<boolean>(false)
|
||||||
|
|
||||||
|
const getCategory = computed(() => currentCategory)
|
||||||
|
const isFavorite = computed(() => currentFavorite)
|
||||||
|
|
||||||
|
function setCategory(category: string) {
|
||||||
|
currentCategory.value = category
|
||||||
|
}
|
||||||
|
|
||||||
|
function setFavorite() {
|
||||||
|
currentFavorite.value = !currentFavorite.value
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
getCategory,
|
||||||
|
setCategory,
|
||||||
|
setFavorite,
|
||||||
|
isFavorite,
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
persist: true,
|
||||||
|
}
|
||||||
|
)
|
||||||
Reference in New Issue
Block a user