opnform-host-nginx/client/components/forms/FlatSelectInput.vue

189 lines
5.2 KiB
Vue

<template>
<input-wrapper v-bind="inputWrapperProps">
<template #label>
<slot name="label" />
</template>
<Loader
v-if="loading"
key="loader"
class="h-6 w-6 text-nt-blue mx-auto"
/>
<div
v-else
class="relative overflow-hidden"
:class="[
theme.default.input,
theme.default.borderRadius,
{
'!ring-red-500 !ring-2 !border-transparent': hasError,
'!cursor-not-allowed !bg-gray-200 dark:!bg-gray-800': disabled,
},
]"
>
<template
v-if="options && options.length"
>
<div
v-for="(option) in options"
:key="option[optionKey]"
:role="multiple?'checkbox':'radio'"
:aria-checked="isSelected(option[optionKey])"
:class="[
theme.FlatSelectInput.spacing.vertical,
theme.FlatSelectInput.fontSize,
theme.FlatSelectInput.option,
{
'!cursor-not-allowed !bg-gray-200 dark:!bg-gray-800': disableOptions.includes(option[optionKey]),
},
]"
@click="onSelect(option[optionKey])"
>
<template v-if="multiple">
<CheckboxIcon
:is-checked="isSelected(option[optionKey])"
:color="color"
:theme="theme"
/>
</template>
<template v-else>
<RadioButtonIcon
:is-checked="isSelected(option[optionKey])"
:color="color"
:theme="theme"
/>
</template>
<UTooltip
:text="disableOptionsTooltip"
:prevent="!disableOptions.includes(option[optionKey])"
class="w-full"
>
<slot
name="option"
:option="option"
:selected="isSelected(option[optionKey])"
>
<p class="flex-grow">
{{ option[displayKey] }}
</p>
</slot>
</UTooltip>
</div>
</template>
<div
v-else
:class="[
theme.FlatSelectInput.spacing.horizontal,
theme.FlatSelectInput.spacing.vertical,
theme.FlatSelectInput.fontSize,
theme.FlatSelectInput.option,
'!text-gray-500 !cursor-not-allowed'
]"
>
{{ $t('forms.select.noOptionAvailable') }}
</div>
</div>
<template #help>
<slot name="help" />
</template>
<template #error>
<slot name="error" />
</template>
</input-wrapper>
</template>
<script>
import {inputProps, useFormInput} from "./useFormInput.js"
import InputWrapper from "./components/InputWrapper.vue"
import RadioButtonIcon from "./components/RadioButtonIcon.vue"
import CheckboxIcon from "./components/CheckboxIcon.vue"
/**
* Options: {name,value} objects
*/
export default {
name: "FlatSelectInput",
components: {InputWrapper, RadioButtonIcon, CheckboxIcon},
props: {
...inputProps,
options: {type: Array, required: true},
optionKey: {type: String, default: "value"},
emitKey: {type: String, default: "value"},
displayKey: {type: String, default: "name"},
loading: {type: Boolean, default: false},
multiple: { type: Boolean, default: false },
disableOptions: { type: Array, default: () => [] },
disableOptionsTooltip: { type: String, default: "Not allowed" },
clearable: { type: Boolean, default: false },
},
setup(props, context) {
return {
...useFormInput(props, context),
}
},
data() {
return {}
},
computed: {
selectedOptions() {
if (!this.compVal) return []
if (this.multiple) {
return this.options.filter(option => this.compVal.includes(option[this.optionKey]))
}
return this.options.find(option => option[this.optionKey] === this.compVal) || null
},
},
methods: {
onSelect(value) {
if (this.disabled || this.disableOptions.includes(value)) {
return
}
if (this.multiple) {
const emitValue = Array.isArray(this.compVal) ? [...this.compVal] : []
// Already in value, remove it only if clearable or not the last item
if (this.isSelected(value)) {
if (this.clearable || emitValue.length > 1) {
this.compVal = emitValue.filter((item) => item !== value)
}
return
}
// Otherwise add value
emitValue.push(value)
this.compVal = emitValue
} else {
// For single select, only change value if it's different or clearable
if (this.compVal !== value || this.clearable) {
this.compVal = this.compVal === value && this.clearable ? null : value
}
}
},
isSelected(value) {
if (!this.compVal) return false
if (this.multiple) {
return this.compVal.includes(value)
}
return this.compVal === value
},
getOptionName(option) {
return option ? option[this.displayKey] : ''
},
getSelectedOptionsNames() {
if (!this.compVal) return []
if (this.multiple) {
return this.selectedOptions.map(option => option[this.displayKey])
}
return [this.getOptionName(this.selectedOptions)]
},
},
}
</script>