123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133 |
- <!-- Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/ -->
- <script setup lang="ts">
- import { computed, toRef } from 'vue'
- import { useLink } from 'vue-router'
- import { getLinkClasses } from '#shared/initializer/initializeLinkClasses.ts'
- import { useApplicationStore } from '#shared/stores/application.ts'
- import type { Link } from '#shared/types/router.ts'
- import stopEvent from '#shared/utils/events.ts'
- import type { Sizes } from './types.ts'
- export interface Props {
- link: Link
- external?: boolean
- internal?: boolean
- restApi?: boolean
- disabled?: boolean
- rel?: string
- target?: string
- openInNewTab?: boolean
- replace?: boolean
- activeClass?: string
- exactActiveClass?: string
- size?: Sizes
- }
- const props = withDefaults(defineProps<Props>(), {
- external: false,
- internal: false,
- replace: false,
- append: false,
- openInNewTab: false,
- disabled: false,
- activeClass: 'router-link-active',
- exactActiveClass: 'router-link-exact-active',
- size: 'large',
- })
- const emit = defineEmits<{
- click: [event: MouseEvent]
- }>()
- const target = computed(() => {
- if (props.target) return props.target
- if (props.openInNewTab) return '_blank'
- return undefined
- })
- const linkClass = computed(() => {
- const { base } = getLinkClasses()
- if (props.disabled) return `${base} pointer-events-none`
- return base
- })
- const fontSizeClassMap = {
- xs: 'text-[10px] leading-[10px]',
- small: 'text-xs leading-snug',
- medium: 'text-sm leading-snug',
- large: 'text-base leading-snug',
- xl: 'text-xl leading-snug',
- }
- const { href, route, navigate, isActive, isExactActive } = useLink({
- to: toRef(props, 'link'),
- replace: toRef(props, 'replace'),
- })
- const isInternalLink = computed(() => {
- if (props.external || props.restApi) return false
- if (props.internal) return true
- // zammad desktop urls
- if (route.value.fullPath.startsWith('/#')) return false
- return route.value.matched.length > 0 && route.value.name !== 'Error'
- })
- const app = useApplicationStore()
- const path = computed(() => {
- if (isInternalLink.value) {
- return href.value
- }
- if (props.restApi) {
- return `${app.config.api_path}${props.link}`
- }
- return props.link as string
- })
- const onClick = (event: MouseEvent) => {
- if (props.disabled) {
- stopEvent(event, { immediatePropagation: true })
- return
- }
- emit('click', event)
- if (isInternalLink.value) {
- navigate(event)
- }
- // Stop the scroll-to-top behavior or navigation on regular links when href is just '#'.
- if (!isInternalLink.value && props.link === '#') {
- stopEvent(event, { propagation: false })
- }
- }
- defineExpose({
- isActive,
- isExactActive,
- })
- </script>
- <template>
- <a
- data-test-id="common-link"
- :href="path"
- :target="target"
- :rel="rel"
- :class="[
- linkClass,
- fontSizeClassMap[props.size],
- {
- [activeClass]: isActive,
- [exactActiveClass]: isExactActive,
- },
- ]"
- @click="onClick"
- >
- <slot></slot>
- </a>
- </template>
|