Files
homebox/frontend/components/global/DetailsSection/DetailsSection.vue
Matthieu Evrin 790352da34 fix(item): remove line break in Items label in location view (#975)
fix: prevent items word wrapped in firefox

Signed-off-by: lekaf974 <matthieu.evrin@gmail.com>
2025-09-01 22:52:14 +01:00

149 lines
4.3 KiB
Vue

<template>
<div class="border-t px-4 py-5 sm:p-0">
<dl class="sm:divide-y">
<div v-for="(detail, i) in details" :key="i" class="group py-4 sm:grid sm:grid-cols-3 sm:gap-4 sm:px-6">
<dt class="text-sm font-medium">
{{ $t(detail.name) }}
</dt>
<dd class="text-start text-sm sm:col-span-2">
<slot :name="detail.slot || detail.name" v-bind="{ detail }">
<DateTime
v-if="detail.type == 'date'"
:date="detail.text"
:datetime-type="detail.date ? 'date' : 'datetime'"
/>
<Currency v-else-if="detail.type == 'currency'" :amount="detail.text" />
<template v-else-if="detail.type === 'link'">
<TooltipProvider :delay-duration="0">
<Tooltip>
<TooltipTrigger as-child>
<a
:href="detail.href"
target="_blank"
rel="noopener noreferrer"
:class="badgeVariants()"
class="gap-1"
>
<MdiOpenInNew />
{{ detail.text }}
</a>
</TooltipTrigger>
<TooltipContent>
{{ detail.href }}
</TooltipContent>
</Tooltip>
</TooltipProvider>
</template>
<template v-else-if="detail.type === 'markdown'">
<ClientOnly>
<!-- eslint-disable-next-line tailwindcss/no-custom-classname -->
<div class="markdown-container w-full overflow-hidden break-words">
<Markdown :source="detail.text" />
</div>
</ClientOnly>
</template>
<template v-else>
<!-- Fixed version with improved overflow handling -->
<span class="flex w-full items-center break-words">
<a
v-if="maybeUrl(detail.text.toString()).isUrl"
:href="maybeUrl(detail.text.toString()).url"
target="_blank"
rel="noopener noreferrer"
class="overflow-hidden break-all text-primary underline hover:text-primary/80"
>{{ detail.text }}</a
>
<span v-else class="overflow-hidden break-all">{{ detail.text }}</span>
<span
v-if="detail.copyable"
class="my-0 ml-4 shrink-0 opacity-0 transition-opacity duration-75 group-hover:opacity-100"
>
<CopyText v-if="detail.text.toString()" :text="detail.text.toString()" :icon-size="16" />
</span>
</span>
</template>
</slot>
</dd>
</div>
</dl>
</div>
</template>
<script setup lang="ts">
import type { AnyDetail, Detail } from "./types";
import MdiOpenInNew from "~icons/mdi/open-in-new";
import { badgeVariants } from "~/components/ui/badge";
import { Tooltip, TooltipContent, TooltipTrigger, TooltipProvider } from "@/components/ui/tooltip";
defineProps({
details: {
type: Object as () => (Detail | AnyDetail)[],
required: true,
},
});
</script>
<style>
/* Use :deep to target elements inside the markdown container */
:deep(.markdown-container) {
/* General container styles */
width: 100%;
overflow-wrap: break-word;
word-wrap: break-word;
/* Handle tables */
table {
table-layout: fixed;
width: 100%;
}
td,
th {
word-break: break-word;
overflow-wrap: break-word;
max-width: 100%;
}
/* Handle code blocks */
pre,
code {
white-space: pre-wrap;
word-break: break-all;
overflow-x: auto;
}
/* Handle images */
img {
max-width: 100%;
height: auto;
}
/* Handle headings */
h1,
h2,
h3,
h4,
h5,
h6 {
word-break: break-word;
overflow-wrap: break-word;
}
/* Handle blockquotes */
blockquote {
overflow-wrap: break-word;
word-break: break-word;
}
/* Handle inline elements */
a,
strong,
em {
word-break: break-all;
overflow-wrap: break-word;
}
}
</style>