dml_frontend/components/Breadcrumbs.vue
2025-05-25 21:45:54 +02:00

158 lines
4.2 KiB
Vue

<template>
<nav v-if="breadcrumbs.length" class="breadcrumbs" aria-label="Breadcrumb">
<ul>
<li>
<router-link to="/" aria-label="Startseite">
<svg xmlns="http://www.w3.org/2000/svg" viewBox="0 0 576 512" aria-hidden="true" focusable="false">
<title>Startseite</title>
<path d="M575.8 255.5c0 18-15 32.1-32 32.1l-32 0 .7 160.2c0 2.7-.2 5.4-.5 8.1l0 16.2c0 22.1-17.9 40-40 40l-16 0c-1.1 0-2.2 0-3.3-.1c-1.4 .1-2.8 .1-4.2 .1L416 512l-24 0c-22.1 0-40-17.9-40-40l0-24 0-64c0-17.7-14.3-32-32-32l-64 0c-17.7 0-32 14.3-32 32l0 64 0 24c0 22.1-17.9 40-40 40l-24 0-31.9 0c-1.5 0-3-.1-4.5-.2c-1.2 .1-2.4 .2-3.6 .2l-16 0c-22.1 0-40-17.9-40-40l0-112c0-.9 0-1.9 .1-2.8l0-69.7-32 0c-18 0-32-14-32-32.1c0-9 3-17 10-24L266.4 8c7-7 15-8 22-8s15 2 21 7L564.8 231.5c8 7 12 15 11 24z"/>
</svg>
</router-link>
</li>
<li v-for="(crumb, index) in breadcrumbs" :key="index">
<router-link v-if="index < breadcrumbs.length - 1" :to="crumb.to" :title="crumb.labelFull">
{{ crumb.label }}
</router-link>
<span v-else :title="crumb.labelFull">{{ crumb.label }}</span>
</li>
</ul>
</nav>
</template>
<script setup lang="ts">
import { computed } from 'vue'
import { useI18n } from 'vue-i18n'
import { useRoute } from 'vue-router'
import { i18nPages } from '~/i18n/i18n-pages'
interface Breadcrumb {
label: string
labelFull: string
to: string
}
function formatLabel(segment: string): { label: string; labelFull: string } {
const labelFull = segment.charAt(0).toUpperCase() + segment.slice(1).replace(/-/g, ' ')
let label = labelFull
if (label.length > 25) {
label = label.slice(0, 22) + '...'
}
return { label, labelFull }
}
function buildUrl(loc: string, path: string): string {
if (loc === 'de') {
return path.startsWith('/') ? path : '/' + path
}
return `/${loc}${path.startsWith('/') ? path : '/' + path}`
}
const route = useRoute()
const { locale, t } = useI18n()
const breadcrumbs = computed<Breadcrumb[]>(() => {
const loc = locale.value
const pathWithoutLang = route.path.replace(`/${loc}`, '')
const segments = pathWithoutLang.split('/').filter(Boolean)
if (segments.length === 2 && segments[0] === 'projekt') {
const referencesPath = i18nPages.references?.[loc] || '/references'
// Übersetzung für "references" holen, Fallback zu englisch falls nicht vorhanden
const referencesLabel = t('references') || 'References'
const first = {
label: referencesLabel,
labelFull: referencesLabel,
to: buildUrl(loc, referencesPath)
}
const { label, labelFull } = formatLabel(segments[1])
const second = {
label,
labelFull,
to: route.path
}
return [first, second]
}
let path = ''
return segments.map(segment => {
path += '/' + segment
const { label, labelFull } = formatLabel(segment)
return {
label,
labelFull,
to: buildUrl(loc, path)
}
})
})
</script>
<style lang="sass" scoped>
.breadcrumbs
position: fixed
top: 22vh
left: 0
text-align: left
padding: 1rem .25rem 1rem .5rem
background-color: rgba(white, .98)
border: 1px solid darken($lightgrey, 5%)
writing-mode: vertical-rl
transform: rotate(180deg)
border-top-left-radius: .8rem
border-bottom-left-radius: .8rem
text-transform: uppercase
letter-spacing: .05rem
z-index: 22
ul
display: flex
flex-wrap: wrap
list-style: none
padding: 0
margin: 0
gap: 0.5rem
cursor: pointer
li
display: flex
align-items: center
color: darken($primaryColor, 10%)
font-size: .7rem
svg
width: .8rem
height: .8rem
transform: rotate(90deg)
margin-bottom: .35rem
transition: .3s
path
fill: darken($lightgrey, 20%)
&:hover
transform: scale(1.3) rotate(90deg)
path
fill: darken($lightgrey, 30%)
&::after
content: '>'
margin: 0 0.5rem
color: $pink
&:last-child::after
content: ''
a
color: #007BFF
text-decoration: none
&:hover
transform: scale(1.2)
span
font-weight: bold
</style>