Files
homebox/frontend/components/global/DetailsSection/DetailsSection.vue
Tonya cbaf483788 migrate pages to shadcn (#628)
* feat: migrate tools page and label generator to shadcn

* chore: lint issues

* feat: also do profile page

* feat: shadcn 404 page

* feat: login page shadcn

* fix: daisyui ironically breaks the z height for the login page

* feat: componentise the language selector and add it to the login page

* feat: use nuxtlink

* feat: card and table made more shadcn

* feat: shadcn statscard

* chore: lint

* feat: shadcn labelchip and locationcard

* feat: shadcn locations page

* refactor: remove unused new item page

* chore: lint

* feat: shadcn item card

* fix: wrapping of location and lint

* feat: ctrl enter in text area in form submits form

* feat: begin shadcn locations page and remove pageqrcode comp in favour of integrating it into labelmaker

* chore: lint + remove unused code

* fix: remove uneeded margin

* feat: shadcn labels page and fix some issues with location

* feat: shadcn scanner

* chore: lint

* feat: begin shadcning item pages

* feat: shadcn maintenance page

* feat: begin shadcn search page

* fix: quick switch blurry text and crashing page when switching + incorrect z height for create menu

* feat: finish shadcn search page

* chore: lint

* feat: shadcn edit item page

* fix: quickmenumodal bug

* feat: shadcn item details page

* feat: remove all non-color related daisyui classes

* fix: type error

* fix: quick menu modal again :(
2025-04-20 08:58:03 +01:00

167 lines
4.3 KiB
Vue

<template>
<div class="border-t border-gray-300 px-4 py-5 sm:p-0">
<dl class="sm:divide-y sm:divide-gray-300">
<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 text-base-content">
{{ $t(detail.name) }}
</dt>
<dd class="text-start text-sm text-base-content 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" :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">
<span 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;
}
}
/* Non-scoped styles for regular text */
.break-all {
word-break: break-all;
max-width: 100%;
}
/* Handle very long words */
pre,
code,
a,
p,
span,
div,
td,
th,
li,
blockquote,
h1,
h2,
h3,
h4,
h5,
h6 {
overflow-wrap: break-word;
word-wrap: break-word;
-ms-word-break: break-all;
word-break: break-all;
word-break: break-word;
-ms-hyphens: auto;
-moz-hyphens: auto;
-webkit-hyphens: auto;
hyphens: auto;
}
</style>