designtopack/src/components/Tabs.vue

119 lines
2.7 KiB
Vue
Raw Normal View History

2024-09-04 11:28:12 +02:00
<template>
<div role="tablist" aria-labelledby="tablist">
<!-- TODO: aria-selected et tabindex ne fonctionnent pas -->
2024-09-04 11:28:12 +02:00
<button
v-for="tab in tabs"
:key="tab.label"
2024-11-21 16:58:19 +01:00
:id="slugify(tab.label, { lower: true }) + '-label'"
2024-09-04 11:28:12 +02:00
type="button"
role="tab"
:data-icon="tab.icon ?? null"
2024-09-04 11:28:12 +02:00
:aria-selected="tab.isActive"
2024-11-21 16:58:19 +01:00
:aria-controls="slugify(tab.label, { lower: true })"
:tabindex="tab.isActive ? '-1' : null"
2024-09-04 11:28:12 +02:00
@click="changeTab(tab.id)"
>
<span class="label">{{ tab.label }}</span>
<span class="count">{{ tab.count }}</span>
</button>
</div>
2024-09-04 11:28:12 +02:00
</template>
<script setup>
2024-09-10 09:12:52 +02:00
import { computed } from "vue";
2024-09-04 11:28:12 +02:00
import slugify from "slugify";
2024-09-10 10:50:51 +02:00
const { tabs } = defineProps({
tabs: Array,
2024-09-10 09:12:52 +02:00
});
2024-09-04 11:28:12 +02:00
const emit = defineEmits(["update:currentTab"]);
function changeTab(tabId) {
emit("update:currentTab", tabId);
}
</script>
<style scoped>
[role="tablist"] {
--tabs: 2;
--tab-w: 15rem; /* 240px */
width: 100%;
max-width: calc(var(--tabs) * var(--tab-w));
2024-09-04 11:28:12 +02:00
display: flex;
align-items: flex-start;
justify-content: center;
margin: 0 auto;
2024-09-10 18:07:43 +02:00
border-radius: var(--rounded-md);
background-color: var(--color-grey-200);
flex-shrink: 1;
2024-09-04 11:28:12 +02:00
}
[role="tab"] {
--tab-h: 2.5rem;
--tab-py: var(--space-8);
--tab-px: var(--space-12);
--tab-border-w: 4px;
2024-09-04 11:28:12 +02:00
position: relative;
display: flex;
2024-09-04 11:28:12 +02:00
align-items: center;
font-size: var(--text-md);
text-align: left;
border-radius: var(--rounded-md);
2024-09-04 11:28:12 +02:00
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-16);
width: 100%;
height: var(--tab-h);
border: var(--tab-border-w) solid var(--color-grey-200);
flex-shrink: 1;
2024-09-04 11:28:12 +02:00
}
[role="tab"]:focus-visible {
z-index: 20;
2024-09-04 11:28:12 +02:00
}
[role="tab"] .label {
flex-grow: 1;
font-family: var(--font-serif);
margin-top: 0.1em;
2024-09-04 11:28:12 +02:00
}
[role="tab"] .count {
font-size: var(--text-sm);
font-weight: 500;
margin-top: -0.1em;
}
[role="tab"] + [role="tab"] {
margin-left: calc(var(--tab-border-w) / -2);
2024-09-04 11:28:12 +02:00
}
[role="tab"][aria-selected="true"] {
--background: var(--color-background);
z-index: 10;
}
[role="tab"][aria-selected="true"]:hover {
background-color: var(--color-white-100);
}
2024-09-04 11:28:12 +02:00
[role="tab"][aria-selected="false"] {
--background: var(--color-grey-200);
2024-09-12 19:09:41 +02:00
}
[role="tab"][aria-selected="false"]:hover {
--background: var(--color-white-50);
2024-09-04 11:28:12 +02:00
}
[role="tab"][data-icon="favorite"]::before {
--icon-color: var(--color-grey-400);
2024-09-04 11:28:12 +02:00
}
[role="tab"][data-icon="favorite"][aria-selected="true"]::before {
--icon-color: var(--color-brand);
}
2024-09-04 11:28:12 +02:00
[role="tabpanel"] {
width: 100%;
overflow: auto;
}
</style>