✨ feature: Added Button group component
This commit is contained in:
parent
fedc77ead3
commit
cf4eee75b0
103
src/components/button/VButtonGroup.vue
Normal file
103
src/components/button/VButtonGroup.vue
Normal file
|
@ -0,0 +1,103 @@
|
||||||
|
<script setup lang="ts">
|
||||||
|
import VButton from './VButton.vue';
|
||||||
|
import type IVButtonGroup from './IVButton.type.ts';
|
||||||
|
import { computed, ref, onMounted } from 'vue';
|
||||||
|
|
||||||
|
const props = withDefaults(defineProps<IVButtonGroup>(), {
|
||||||
|
buttons: () => [],
|
||||||
|
inlineLayoutWhen: 'never',
|
||||||
|
size: 'md',
|
||||||
|
align: undefined,
|
||||||
|
iconRight: false,
|
||||||
|
})
|
||||||
|
|
||||||
|
// Reference to <ul> element containing buttons
|
||||||
|
const buttonsEl = ref<HTMLUListElement | null>(null)
|
||||||
|
|
||||||
|
// Determines whether the layout should be in line
|
||||||
|
const inlineAlways = computed(() => ['always', '', true].includes(props.inlineLayoutWhen))
|
||||||
|
const inlineSm = computed(() => ['sm', 'small'].includes(props.inlineLayoutWhen as string))
|
||||||
|
const inlineMd = computed(() => ['md', 'medium'].includes(props.inlineLayoutWhen as string))
|
||||||
|
const inlineLg = computed(() => ['lg', 'large'].includes(props.inlineLayoutWhen as string))
|
||||||
|
|
||||||
|
// Horizontal alignment of button group
|
||||||
|
const center = computed(() => props.align === 'center')
|
||||||
|
const right = computed(() => props.align === 'right')
|
||||||
|
|
||||||
|
// Uniform width for all buttons
|
||||||
|
const equisizedWidth = ref('auto')
|
||||||
|
const groupStyle = computed(() => `--equisized-width: ${equisizedWidth.value};`)
|
||||||
|
|
||||||
|
// Function for calculating maximum button width and applying it to all buttons
|
||||||
|
const computeEquisizedWidth = async () => {
|
||||||
|
let maxWidth = 0
|
||||||
|
await new Promise((resolve) => setTimeout(resolve, 100))
|
||||||
|
buttonsEl.value?.querySelectorAll('.p-button').forEach((btn: Element) => {
|
||||||
|
const button = btn as HTMLButtonElement
|
||||||
|
const width = button.offsetWidth
|
||||||
|
const buttonStyle = window.getComputedStyle(button)
|
||||||
|
const marginLeft = +buttonStyle.marginLeft.replace('px', '')
|
||||||
|
const marginRight = +buttonStyle.marginRight.replace('px', '')
|
||||||
|
button.style.width = 'var(--equisized-width)'
|
||||||
|
const newWidth = width + marginLeft + marginRight
|
||||||
|
if (newWidth > maxWidth) {
|
||||||
|
maxWidth = newWidth
|
||||||
|
}
|
||||||
|
})
|
||||||
|
equisizedWidth.value = `${maxWidth}px`
|
||||||
|
}
|
||||||
|
|
||||||
|
// Calculation of uniform installation width if necessary
|
||||||
|
onMounted(async () => {
|
||||||
|
if (!buttonsEl.value || !props.equisized) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
await computeEquisizedWidth()
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<template>
|
||||||
|
<ul
|
||||||
|
ref="buttonsEl"
|
||||||
|
role="list"
|
||||||
|
:style="groupStyle"
|
||||||
|
class="btns-group"
|
||||||
|
:class="{
|
||||||
|
'btns-group--equisized': equisized,
|
||||||
|
'btns-group--inline': inlineAlways || inlineSm || inlineMd || inlineLg,
|
||||||
|
'btns-group--center': center,
|
||||||
|
'btns-group--right': right,
|
||||||
|
'btns-group--inline-reverse': reverse,
|
||||||
|
}"
|
||||||
|
>
|
||||||
|
<li v-for="({onClick, title, ...button}, i) in buttons" :key="i" role="listitem">
|
||||||
|
<VButton
|
||||||
|
v-bind="button"
|
||||||
|
@click="onClick"
|
||||||
|
:size="props.size"
|
||||||
|
:icon-right="props.iconRight"
|
||||||
|
:title="title"
|
||||||
|
/>
|
||||||
|
</li>
|
||||||
|
<slot/>
|
||||||
|
</ul>
|
||||||
|
</template>
|
||||||
|
|
||||||
|
<style lang="css" scoped>
|
||||||
|
.btns-group{
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
flex-direction: column;
|
||||||
|
gap: 0.5rem;
|
||||||
|
list-style: none;
|
||||||
|
padding: 0px;
|
||||||
|
margin: 0px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btns-group--equisized{width: var(--equisized-width, auto);}
|
||||||
|
.btns-group--inline{flex-direction: row;}
|
||||||
|
.btns-group--center{justify-content: center;}
|
||||||
|
.btns-group--right{justify-content: flex-end;}
|
||||||
|
.btns-group--inline-reverse{flex-direction: row-reverse;}
|
||||||
|
|
||||||
|
</style>
|
Loading…
Reference in New Issue
Block a user