dynamize projects

This commit is contained in:
isUnknown 2024-09-04 11:28:12 +02:00
parent 3f9c8bbebf
commit edd9e66efb
10 changed files with 417 additions and 291 deletions

154
src/components/Project.vue Normal file
View file

@ -0,0 +1,154 @@
<template>
<article class="project-item | flex | rounded-lg | px-2xl py-2xl">
<hgroup>
<h3>{{ project.title }}</h3>
<p>
Dernière mise à jour le
<time :datetime="project.modified">{{ frenchFormattedModified }}</time>
</p>
</hgroup>
<img src="" alt="Logo" class="project-logo | rounded-sm" />
<ol class="project-steps" data-steps="1">
<li class="project-step" data-status="in-progress">
<span class="pill" data-icon="search">Votre Brief</span>
</li>
</ol>
</article>
</template>
<script setup>
import dayjs from "dayjs";
import "dayjs/locale/fr";
const { project } = defineProps({ project: Array });
dayjs.locale("fr");
const frenchFormattedModified = dayjs(project.modified).format(
"dddd D MMMM YYYY"
);
</script>
<style scoped>
.project-item {
--wrap: no-wrap;
background: var(--color-background);
}
.project-item hgroup {
flex: 1 1 0%;
}
.project-item h3 {
font-family: var(--font-serif);
font-size: var(--text-lg);
margin-bottom: var(--space-sm);
}
.project-item p {
font-size: var(--text-sm);
line-height: var(--leading-sm);
}
.project-logo {
order: -1;
background: var(--color-grey-50);
color: var(--color-grey-400);
text-align: center;
line-height: 4.5rem;
width: 4.5rem;
height: 4.5rem;
}
.project-steps {
--color: var(--color-primary-100);
flex: 1 1 0%;
display: flex;
gap: var(--space-lg);
margin-top: -2.75rem;
position: relative;
}
.project-steps[data-steps="1"]::after {
content: "étapes à venir";
font-size: var(--text-sm);
font-weight: 500;
width: 8rem;
position: absolute;
text-align: center;
color: var(--color-grey-700);
background: var(--color-background);
bottom: -2rem;
left: 50%;
transform: translate(-50%, -0.2em);
}
.project-step {
--color: var(--color-white);
position: relative;
flex: 1 1 0%;
text-align: center;
}
.project-step:last-child {
text-align: right;
}
.project-step:only-child,
.project-step:first-child {
text-align: left;
}
.project-step[data-status="in-progress"]:only-child::before,
.project-step[data-status="in-progress"]:only-child::after {
content: "";
display: inline-block;
height: 1.25rem;
position: absolute;
right: 3.75rem;
bottom: -2rem;
}
.project-step[data-status="in-progress"]:only-child::before {
height: 1rem;
color: red;
background-repeat: repeat;
background-position: left center;
background-image: url("data:image/svg+xml,%3Csvg width='20' height='20' viewBox='0 0 20 20' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='50%' cy='50%' r='4' opacity='0.15' fill='black'/%3E%3C/svg%3E%0A");
background-size: 0.75rem;
left: 5.5rem;
right: 5.5rem;
bottom: -1.875rem;
}
.project-step[data-status="in-progress"]:only-child::after {
--icon-size: 1.25rem;
width: var(--icon-size);
background: var(--color-grey-300);
mask-repeat: no-repeat;
mask-position: center;
mask-size: var(--icon-size);
mask-image: var(--icon-point);
right: 3.75rem;
bottom: -2rem;
}
.project-step .pill::after {
content: "";
display: inline-block;
width: var(--icon-size, 1.25rem);
height: var(--icon-size, 1.25rem);
background-color: var(--icon-color, var(--color-primary-100));
mask-repeat: no-repeat;
mask-position: center;
mask-size: var(--size, 1.25rem);
mask-image: var(--icon, var(--icon-point));
position: absolute;
bottom: -2rem;
}
.project-step[data-status="done"] > .pill,
.project-step[data-status="uncompleted"] > .pill {
--background: transparent;
}
.project-step[data-status="done"] .pill::after {
mask-image: var(--icon-check-3);
}
.project-step[data-status="in-progress"] .pill::after {
mask-image: var(--icon-point-active);
}
.project-step[data-status="uncompleted"] .pill::after {
--icon-color: var(--color-grey-300);
}
</style>

View file

@ -0,0 +1,24 @@
<template>
<template v-if="data">
<Tabs :projects="data.children" @update:currentTab="changeTab" />
<TabPanel :projects="data.children" :currentTab="currentTab" />
</template>
</template>
<script setup>
import Tabs from "./Tabs.vue";
import TabPanel from "./TabPanel.vue";
import { useApiStore } from "../stores/api";
import { ref } from "vue";
const data = ref(null);
const currentTab = ref("current");
const api = useApiStore();
api.fetchPageData("projects").then((res) => (data.value = res));
function changeTab(newValue) {
currentTab.value = newValue;
}
</script>
<style scoped></style>

View file

@ -0,0 +1,47 @@
<template>
<section
v-if="currentTab === 'current'"
id="projets-en-cours"
role="tabpanel"
tabindex="0"
aria-labelledby="projets-en-cours-label"
class="flow"
>
<Project
v-for="project in currentProjects"
:key="project.id"
:project="project"
/>
</section>
<section
v-else
id="projets-archives"
role="tabpanel"
tabindex="0"
aria-labelledby="projet-archives-label"
>
<Project
v-for="project in archivedProjects"
:key="project.id"
:project="project"
/>
</section>
</template>
<script setup>
import Project from "./Project.vue";
import { computed } from "vue";
const { projects, currentTab } = defineProps({
projects: Array,
currentTab: String,
});
const currentProjects = computed(() => {
return projects.filter((child) => child.status === "listed");
});
const archivedProjects = computed(() => {
return projects.filter((child) => child.status === "unlisted");
});
</script>

116
src/components/Tabs.vue Normal file
View file

@ -0,0 +1,116 @@
<template>
<h2 id="tabslist" class="sr-only">Projets</h2>
<header role="tablist" aria-labelledby="tablist">
<button
v-for="tab in tabs"
:key="tab.label"
:id="slugify(tab.label) + '-label'"
type="button"
role="tab"
:aria-selected="tab.isActive"
:aria-controls="slugify(tab.label)"
:tabindex="tab.isActive ? '-1' : false"
@click="changeTab(tab.id)"
>
<span class="label">{{ tab.label }}</span>
<span class="count">{{ tab.count }}</span>
</button>
</header>
</template>
<script setup>
import { ref } from "vue";
import slugify from "slugify";
const { projects } = defineProps({
projects: Array,
});
const tabs = ref([
{
label: "Projets en cours",
id: "current",
count: projects.filter((project) => project.status === "listed").length,
isActive: true,
},
{
label: "Projets archivés",
id: "archived",
count: projects.filter((project) => project.status === "unlisted").length,
isActive: false,
},
]);
const emit = defineEmits(["update:currentTab"]);
function changeTab(tabId) {
emit("update:currentTab", tabId);
}
</script>
<style scoped>
[role="tablist"] {
width: fit-content;
display: flex;
align-items: flex-start;
justify-content: center;
margin: 0 auto var(--space-md);
border-radius: var(--rounded-full);
min-height: 60px;
}
[role="tab"] {
--tab-height: 2.5rem;
--tab-py: var(--space-md);
--tab-px: var(--space-lg);
position: relative;
display: inline-flex;
align-items: center;
font-size: var(--text-md);
border-radius: var(--rounded-full);
background-color: var(--background, var(--color-background));
color: var(--color, var(--color-text));
z-index: 2;
padding: var(--tab-py) var(--tab-px);
margin: 0;
cursor: pointer;
gap: var(--space-lg);
height: var(--tab-height);
}
[role="tab"]:focus,
[role="tab"]:hover {
}
[role="tab"] .label {
flex-grow: 1;
font-family: var(--font-serif);
}
[role="tab"] .count {
font-size: var(--text-sm);
font-weight: 500;
}
[role="tab"][aria-selected="true"] {
--background: var(--color-background);
z-index: 10;
}
[role="tab"][aria-selected="false"] {
--background: var(--color-grey-200);
--color: var(--color-grey-700);
}
[role="tab"][aria-selected="true"] + [aria-selected="false"]::before {
content: "";
display: inline-block;
height: var(--tab-height);
width: calc(var(--tab-px) * 2);
position: absolute;
left: calc(var(--tab-px) * -1);
background-color: var(--color-grey-200);
z-index: -1;
}
[role="tabpanel"] {
width: 100%;
overflow: auto;
}
</style>