mirror of
https://github.com/ArthurDanjou/ui.git
synced 2026-01-15 20:48:12 +01:00
228 lines
5.1 KiB
Vue
228 lines
5.1 KiB
Vue
<template>
|
|
<component
|
|
:is="buttonIs"
|
|
ref="button"
|
|
:class="buttonClass"
|
|
:aria-label="ariaLabel"
|
|
v-bind="buttonProps"
|
|
>
|
|
<Icon v-if="isLeading && leadingIconName" :name="leadingIconName" :class="leadingIconClass" aria-hidden="true" />
|
|
<slot>
|
|
<span :class="[truncate ? 'text-left break-all line-clamp-1' : '', compact ? 'hidden sm:block' : '']">
|
|
<span :class="[labelCompact && 'hidden sm:block']">{{ label }}</span>
|
|
<span v-if="labelCompact" class="sm:hidden">{{ labelCompact }}</span>
|
|
</span>
|
|
</slot>
|
|
<Icon v-if="isTrailing && trailingIconName" :name="trailingIconName" :class="trailingIconClass" aria-hidden="true" />
|
|
</component>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, computed, useSlots } from 'vue'
|
|
import type { PropType } from 'vue'
|
|
import type { RouteLocationNormalized, RouteLocationRaw } from 'vue-router'
|
|
import NuxtLink from '#app/components/nuxt-link'
|
|
import Icon from '../elements/Icon.vue'
|
|
import { classNames } from '../../utils'
|
|
import $ui from '#build/ui'
|
|
|
|
const props = defineProps({
|
|
type: {
|
|
type: String,
|
|
default: 'button'
|
|
},
|
|
block: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
label: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
labelCompact: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
loading: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
disabled: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
size: {
|
|
type: String,
|
|
default: 'md',
|
|
validator (value: string) {
|
|
return Object.keys($ui.button.size).includes(value)
|
|
}
|
|
},
|
|
variant: {
|
|
type: String,
|
|
default: 'primary',
|
|
validator (value: string) {
|
|
return Object.keys($ui.button.variant).includes(value)
|
|
}
|
|
},
|
|
icon: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
leadingIcon: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
trailingIcon: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
loadingIcon: {
|
|
type: String,
|
|
default: () => $ui.button.icon.loading
|
|
},
|
|
trailing: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
leading: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
to: {
|
|
type: [String, Object] as PropType<string | RouteLocationNormalized | RouteLocationRaw>,
|
|
default: null
|
|
},
|
|
target: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
ariaLabel: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
rounded: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
roundedClass: {
|
|
type: String,
|
|
default: () => $ui.button.rounded
|
|
},
|
|
baseClass: {
|
|
type: String,
|
|
default: () => $ui.button.base
|
|
},
|
|
iconBaseClass: {
|
|
type: String,
|
|
default: () => $ui.button.icon.base
|
|
},
|
|
leadingIconClass: {
|
|
type: String,
|
|
default: ''
|
|
},
|
|
trailingIconClass: {
|
|
type: String,
|
|
default: ''
|
|
},
|
|
customClass: {
|
|
type: String,
|
|
default: null
|
|
},
|
|
square: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
truncate: {
|
|
type: Boolean,
|
|
default: false
|
|
},
|
|
compact: {
|
|
type: Boolean,
|
|
default: false
|
|
}
|
|
})
|
|
|
|
const slots = useSlots()
|
|
|
|
const button = ref(null)
|
|
|
|
const buttonIs = computed(() => {
|
|
if (props.to) {
|
|
return NuxtLink
|
|
}
|
|
|
|
return 'button'
|
|
})
|
|
|
|
const buttonProps = computed(() => {
|
|
if (props.to) {
|
|
return { to: props.to, target: props.target }
|
|
} else {
|
|
return { disabled: props.disabled || props.loading, type: props.type }
|
|
}
|
|
})
|
|
|
|
const isLeading = computed(() => {
|
|
return (props.icon && props.leading) || (props.icon && !props.trailing) || (props.loading && !props.trailing) || props.leadingIcon
|
|
})
|
|
|
|
const isTrailing = computed(() => {
|
|
return (props.icon && props.trailing) || (props.loading && props.trailing) || props.trailingIcon
|
|
})
|
|
|
|
const isSquare = computed(() => props.square || (!slots.default && !props.label))
|
|
|
|
const buttonClass = computed(() => {
|
|
return classNames(
|
|
props.baseClass,
|
|
$ui.button.size[props.size],
|
|
$ui.button[isSquare.value ? 'square' : (props.compact ? 'compact' : 'spacing')][props.size],
|
|
$ui.button.variant[props.variant],
|
|
props.block ? 'w-full flex justify-center items-center' : 'inline-flex items-center',
|
|
props.rounded ? 'rounded-full' : props.roundedClass,
|
|
props.customClass
|
|
)
|
|
})
|
|
|
|
const leadingIconName = computed(() => {
|
|
if (props.loading) {
|
|
return props.loadingIcon
|
|
}
|
|
|
|
return props.leadingIcon || props.icon
|
|
})
|
|
|
|
const trailingIconName = computed(() => {
|
|
if (props.loading && !isLeading.value) {
|
|
return props.loadingIcon
|
|
}
|
|
|
|
return props.trailingIcon || props.icon
|
|
})
|
|
|
|
const leadingIconClass = computed(() => {
|
|
return classNames(
|
|
props.iconBaseClass,
|
|
$ui.button.icon.size[props.size],
|
|
(!!slots.default || !!props.label?.length) && $ui.button.icon.leading[props.compact ? 'compactSpacing' : 'spacing'][props.size],
|
|
props.leadingIconClass,
|
|
props.loading && 'animate-spin'
|
|
)
|
|
})
|
|
|
|
const trailingIconClass = computed(() => {
|
|
return classNames(
|
|
props.iconBaseClass,
|
|
$ui.button.icon.size[props.size],
|
|
(!!slots.default || !!props.label?.length) && $ui.button.icon.trailing[props.compact ? 'compactSpacing' : 'spacing'][props.size],
|
|
props.trailingIconClass,
|
|
props.loading && !isLeading.value && 'animate-spin'
|
|
)
|
|
})
|
|
</script>
|
|
|
|
<script lang="ts">
|
|
export default { name: 'UButton' }
|
|
</script>
|