designtopack/src/components/project/ProjectStep.vue

322 lines
9.2 KiB
Vue
Raw Normal View History

<template>
2024-11-21 11:27:14 +01:00
<section
class="flex-1"
:aria-labelledby="step.id"
2024-11-21 11:47:36 +01:00
:data-status="setStatus(page.steps, page.content.currentstep, step)"
2024-11-21 11:27:14 +01:00
>
2024-11-18 16:21:07 +01:00
<h2 :id="step.id">
<!-- ADRIEN / TIMOTHÉE : DYNAMISER L'ICONE -->
<span :data-icon="step.id">{{ step.label }}</span>
</h2>
<div class="cards | flow">
2024-11-21 18:10:24 +01:00
<template
v-if="
step.id === 'clientBrief' ||
step.files.dynamic ||
step.files.static ||
step.files.length
"
>
<article class="card" v-if="step.id !== 'proposal'">
2024-10-23 09:48:27 +02:00
<hgroup class="order-last">
2024-11-19 15:55:27 +01:00
<h3 class="card__title | font-serif | text-lg">
2024-11-21 18:10:24 +01:00
<router-link :to="'/' + step.uri" class="link-full">{{
step.label
}}</router-link>
2024-11-19 15:55:27 +01:00
</h3>
2024-10-23 09:48:27 +02:00
</hgroup>
<div class="card__meta | flex">
<time
class="card__date | text-grey-700"
2024-11-21 18:10:24 +01:00
:datetime="dayjs(step.modified).format('YYYY-M-DD')"
>{{ dayjs(step.modified).format("DD MMMM YYYY") }}</time
>
2024-10-23 09:48:27 +02:00
</div>
2024-11-21 18:10:24 +01:00
<!-- All images -->
<figure
v-if="step.id === 'clientBrief' && step.files[0].type === 'image'"
class="card__images"
:data-count="step.files.length"
:data-plus="
step.files.length > 3 ? step.files.length - 3 : undefined
"
>
<img
v-for="image in step.files.slice(0, 3)"
:key="image.uuid"
:src="image.url"
:alt="image.alt"
/>
</figure>
<figure
v-if="step.id === 'extendedBrief'"
2024-11-21 18:10:24 +01:00
class="card__images"
:data-count="step.files.dynamic.length"
:data-plus="
step.files.dynamic.length > 3
? step.files.dynamic.length - 3
: undefined
"
>
<img
v-for="image in step.files.dynamic.slice(0, 3)"
:key="image.uuid"
:src="image.url"
:alt="image.alt"
/>
</figure>
<figure
v-if="step.id === 'virtualSample'"
class="card__images"
:data-count="step.files.dynamic.length"
:data-plus="
step.files.dynamic.length > 3
? step.files.dynamic.length - 3
: undefined
"
>
<img
v-for="track in step.files.dynamic"
:key="track.files[0].slug"
:src="track.files[0].url"
:alt="track.files[0].alt"
/>
</figure>
<!-- Document -->
<div
2024-11-21 18:10:24 +01:00
v-if="step.files[0]?.type === 'document'"
class="card__images"
data-icon="document"
></div>
2024-11-20 08:39:26 +01:00
<footer
2024-11-21 18:10:24 +01:00
v-if="step?.files[0]?.comments?.length > 0"
2024-11-20 08:39:26 +01:00
class="order-last | text-sm text-primary font-medium"
>
2024-11-21 18:10:24 +01:00
{{ step.files[0].comments.length }} commentaire{{
step.files[0].comments.length > 1 ? "s" : ""
2024-11-08 12:25:00 +01:00
}}
</footer>
2024-11-18 16:21:07 +01:00
</article>
2024-11-21 18:10:24 +01:00
<template v-if="step.id == 'proposal' && step.files.length">
<article
class="card"
v-for="(file, index) in step.files"
:key="file.name"
>
<hgroup class="order-last">
<h3 class="card__title | font-serif | text-lg">
<router-link
:to="'/' + step.uri + '&fileIndex=' + index"
class="link-full"
>
{{
file.label.length
? file.label
: file.name.replace(".pdf", "")
}}
</router-link>
</h3>
</hgroup>
<div class="card__meta | flex">
<time
class="card__date | text-grey-700"
:datetime="dayjs(file.modified).format('YYYY-M-DD')"
>{{ dayjs(file.modified).format("DD MMMM YYYY") }}</time
>
</div>
<div
v-if="index === 0"
class="card__images"
data-icon="document"
></div>
<footer
v-if="file.comments?.length > 0"
class="order-last | text-sm text-primary font-medium"
>
<template v-if="step.id === 'proposal'">
<router-link
:to="'/' + step.uri + '&fileIndex=' + index + '&comments'"
>
{{ file.comments.length }} commentaire{{
file.comments.length > 1 ? "s" : ""
}}
</router-link>
</template>
<template v-else>
<router-link :to="'/' + step.uri + '&comments=true'">
{{ file.comments.length }} commentaire{{
file.comments.length > 1 ? "s" : ""
}}
</router-link>
</template>
2024-11-21 18:10:24 +01:00
</footer>
</article>
</template>
2024-11-18 16:21:07 +01:00
</template>
2024-11-21 18:10:24 +01:00
<!-- Empty state -->
2024-11-21 18:11:30 +01:00
<template v-else></template>
2024-11-18 16:21:07 +01:00
</div>
</section>
</template>
2024-09-26 19:14:20 +02:00
<script setup>
import dayjs from "dayjs";
import "dayjs/locale/fr";
2024-10-16 15:32:24 +02:00
import { usePageStore } from "../../stores/page";
import { computed } from "vue";
2024-11-21 11:27:14 +01:00
import { useProjectStore } from "../../stores/project";
2024-09-26 19:14:20 +02:00
const { step } = defineProps({
step: Object,
});
2024-10-16 15:32:24 +02:00
2024-10-23 11:32:51 +02:00
const emit = defineEmits(["update:file"]);
2024-10-16 15:32:24 +02:00
dayjs.locale("fr");
const { page } = usePageStore();
2024-11-21 11:27:14 +01:00
const { setStatus } = useProjectStore();
2024-10-16 15:32:24 +02:00
2024-10-16 17:32:15 +02:00
const steps = page.steps.map((item) => {
return item.value;
});
2024-10-16 15:32:24 +02:00
const mergedFiles = computed(() => {
if (step.id !== "virtualSample") return false;
const staticFiles = step.files?.static ?? [];
const dynamicFiles = step.files?.dynamic ?? [];
return [...staticFiles, ...dynamicFiles];
});
2024-09-26 19:14:20 +02:00
</script>
2024-11-22 17:49:06 +01:00
<style>
.kanban > section {
min-width: 20rem;
position: relative;
}
.kanban .cards {
padding-top: var(--space-16);
max-height: calc(100% - var(--header-height));
overflow-y: auto;
}
2024-11-19 15:55:27 +01:00
.kanban .card {
row-gap: 0;
}
.kanban > section h2 {
position: relative;
background-color: var(--header-bg-color);
border-radius: var(--rounded-md);
font-size: var(--text-sm);
height: var(--header-height);
}
.kanban > section h2 > span {
display: flex;
justify-content: center;
align-items: center;
gap: var(--space-8);
background-color: var(--header-title-bg-color);
color: var(--color-white);
height: 100%;
width: fit-content;
padding: 0 var(--space-12);
border-top-left-radius: inherit;
border-bottom-left-radius: inherit;
font-weight: 500;
}
.kanban > section + section h2::before {
content: "";
display: inline-block;
position: absolute;
top: calc(var(--header-height) / 2 - 1.5px);
right: 100%;
width: var(--gap);
height: 3px;
background-color: var(--color-grey-200);
z-index: -1;
}
.kanban [data-status="done"] h2::after {
content: "";
position: absolute;
top: 0;
right: var(--space-4);
bottom: 0;
display: inline-block;
width: var(--icon-size, var(--header-height));
height: var(--icon-size, var(--header-height));
background: var(--icon-color, currentColor);
mask-repeat: no-repeat;
mask-position: center;
mask-size: var(--icon-size, 1rem);
mask-image: var(--icon, var(--icon-check));
}
.kanban [data-status="in-progress"] {
--header-bg-color: var(--color-primary-20);
--header-title-bg-color: var(--color-primary);
}
.kanban [data-status="in-progress"] h2::after {
content: "";
color: var(--color-primary);
position: absolute;
top: 0;
right: var(--space-4);
bottom: 0;
display: inline-block;
width: var(--icon-size, var(--header-height));
height: var(--icon-size, var(--header-height));
background: var(--icon-color, currentColor);
mask-repeat: no-repeat;
mask-position: center;
mask-size: var(--icon-size, 1rem);
mask-image: var(--icon-point-active);
}
.kanban [data-status="in-progress"]::after {
content: "En cours";
position: absolute;
top: 0;
right: calc(var(--icon-size, var(--header-height)) + var(--space-4));
color: var(--color-primary);
font-weight: 500;
font-size: var(--text-sm);
line-height: 2.125rem;
}
.kanban [data-status="uncompleted"] h2 {
background: none;
}
.kanban [data-status="uncompleted"] h2::before {
background-color: transparent;
background-repeat: repeat-x;
background-position: left center;
background-image: url("data:image/svg+xml,%3Csvg width='8' height='8' viewBox='0 0 8 8' fill='none' xmlns='http://www.w3.org/2000/svg'%3E%3Ccircle cx='50%' cy='50%' r='2' opacity='0.15' fill='black'/%3E%3C/svg%3E%0A");
background-size: 0.5rem;
right: calc(-1 * var(--gap));
left: calc(-1 * var(--gap));
width: auto;
}
.kanban [data-status="uncompleted"] h2 > span {
border-radius: inherit;
}
.kanban [data-status="uncompleted"] .cards > * {
min-height: 10rem;
}
.kanban [data-status="uncompleted"]::after {
content: "En attente";
position: absolute;
top: 0;
right: 0;
background-color: var(--color-grey-50);
color: var(--color-grey-700);
font-weight: 500;
font-size: var(--text-sm);
line-height: 2.125rem;
padding: 0 var(--space-12);
}
</style>