designtopack/src/views/Kanban.vue

156 lines
3.4 KiB
Vue

<template>
<main class="flex flex-col items-stretch | w-full">
<Header :title="page.content.title" />
<DialogWrapper v-if="dialog.content" />
<div class="kanban">
<ProjectStep v-for="step in page.steps" :key="step" :step="step">
</ProjectStep>
</div>
</main>
</template>
<script setup>
import ProjectStep from "../components/project/ProjectStep.vue";
import Header from "../components/project/Header.vue";
import DialogWrapper from "../components/project/DialogWrapper.vue";
import { usePageStore } from "../stores/page";
import { storeToRefs } from "pinia";
import { watch } from "vue";
import { useDialogStore } from "../stores/dialog";
import { useRoute } from "vue-router";
const { page } = storeToRefs(usePageStore());
const dialog = useDialogStore();
const route = useRoute();
if (route.query.dialog) {
openDialog(route.query.dialog);
}
watch(
() => route.query.dialog,
(targetStepSlug) => {
if (targetStepSlug) {
const targetStep = page.value.steps.find(
(step) => step.slug === targetStepSlug
);
dialog.content = targetStep;
} else {
dialog.content = null;
}
}
);
function openDialog(targetStepSlug) {
const targetStep = page.value.steps.find(
(step) => step.slug === targetStepSlug
);
dialog.content = targetStep;
}
</script>
<style scope>
/* Porject details */
.project-details textarea {
resize: none;
max-height: 5rem;
}
.project-details__filters button:first-of-type {
margin-right: calc(var(--space-12) + 1px);
}
.project-details__filters button:first-of-type::after {
content: "";
width: 1px;
height: 1.7rem;
background-color: var(--color-black-20);
position: absolute;
right: calc(-0.75rem);
}
/* Masonry */
.masonry > * {
margin-bottom: var(--space-16);
}
.masonry > *:only-child {
column-span: all;
transform: translateY(50%);
}
@media (min-width: 40rem) {
.masonry {
column-count: 2;
}
}
@media (min-width: 65rem) {
.masonry {
column-count: 3;
}
}
@media (min-width: 90rem) {
.masonry {
column-count: 4;
}
}
/* Image */
.image {
border-radius: var(--rounded-2xl);
overflow: hidden;
position: relative;
cursor: pointer;
}
.image.has-description::after {
content: "";
display: inline-block;
width: 1rem;
height: 1rem;
background: var(--icon-color, currentColor);
mask-repeat: no-repeat;
mask-position: center;
mask-size: 1rem;
mask-image: var(--icon-description);
position: absolute;
top: 1.25rem;
right: 1.25rem;
}
.image[aria-selected="true"] {
outline: 2px solid var(--color-focus-ring);
}
.image[aria-selected="true"]::after {
content: attr(data-count);
position: absolute;
top: var(--space-12);
right: var(--space-12);
border-radius: 50%;
color: var(--color-white);
font-weight: 700;
background: var(--color-grey-800);
display: grid;
place-items: center;
text-align: center;
width: 1.5rem;
height: 1.5rem;
}
.image .tag {
position: absolute;
top: var(--space-12);
left: var(--space-12);
}
/* Kanban */
.kanban {
--header-height: 2.25rem; /* 36px */
--header-bg-color: var(--color-grey-200);
--header-title-bg-color: var(--color-grey-700);
--gap: var(--gutter);
position: relative;
display: flex;
gap: var(--gap);
overflow-x: auto;
margin: 0 calc(-1 * var(--gutter));
padding: 0 var(--gutter);
height: calc(100vh - var(--gutter) * 2 - var(--header-height) - 2.25rem);
}
</style>