primevue-mirror/components/lib/chips/Chips.vue

270 lines
9.0 KiB
Vue
Raw Normal View History

2022-09-06 12:03:37 +00:00
<template>
2024-02-11 08:10:29 +00:00
<div :class="cx('root')" v-bind="ptmi('root')">
2022-09-14 11:26:01 +00:00
<ul
ref="container"
2023-05-24 09:27:50 +00:00
:class="cx('container')"
2022-09-14 11:26:01 +00:00
tabindex="-1"
role="listbox"
aria-orientation="horizontal"
:aria-labelledby="ariaLabelledby"
:aria-label="ariaLabel"
:aria-activedescendant="focused ? focusedOptionId : undefined"
@click="onWrapperClick()"
@focus="onContainerFocus"
@blur="onContainerBlur"
@keydown="onContainerKeyDown"
2023-05-05 09:36:30 +00:00
v-bind="ptm('container')"
2022-09-14 11:26:01 +00:00
>
<li
v-for="(val, i) of modelValue"
:key="`${i}_${val}`"
:id="id + '_chips_item_' + i"
role="option"
2023-05-24 09:27:50 +00:00
:class="cx('token', { index: i })"
2022-09-14 11:26:01 +00:00
:aria-label="val"
:aria-selected="true"
:aria-setsize="modelValue.length"
:aria-posinset="i + 1"
2023-05-05 09:36:30 +00:00
v-bind="ptm('token')"
2023-05-24 09:27:50 +00:00
:data-p-focused="focusedIndex === i"
2022-09-14 11:26:01 +00:00
>
2024-04-08 12:20:51 +00:00
<slot name="chip" :class="cx('label')" :index="i" :value="val" :removeCallback="(event) => removeOption(event, i)">
2024-04-09 06:41:29 +00:00
<Chip :class="cx('label')" :label="val" :removeIcon="removeTokenIcon" removable :unstyled="unstyled" @remove="removeItem($event, i)" :pt="ptm('label')">
2024-04-08 12:20:51 +00:00
<template #removeicon>
<slot name="removetokenicon" :class="cx('removeTokenIcon')" :index="i" :removeCallback="(event) => removeItem(event, i)" />
</template>
</Chip>
</slot>
2022-09-06 12:03:37 +00:00
</li>
2023-05-24 09:27:50 +00:00
<li :class="cx('inputToken')" role="option" v-bind="ptm('inputToken')">
2022-09-14 11:26:01 +00:00
<input
ref="input"
:id="inputId"
type="text"
:class="inputClass"
:style="inputStyle"
:disabled="disabled || maxedOut"
:placeholder="placeholder"
:aria-invalid="invalid || undefined"
2022-09-14 11:26:01 +00:00
@focus="onFocus($event)"
@blur="onBlur($event)"
@input="onInput"
@keydown="onKeyDown($event)"
@paste="onPaste($event)"
2023-05-05 09:36:30 +00:00
v-bind="{ ...inputProps, ...ptm('input') }"
2022-09-14 11:26:01 +00:00
/>
2022-09-06 12:03:37 +00:00
</li>
</ul>
</div>
</template>
<script>
2024-04-08 12:20:51 +00:00
import Chip from 'primevue/chip';
2022-09-14 11:26:01 +00:00
import { UniqueComponentId } from 'primevue/utils';
2023-05-24 09:27:50 +00:00
import BaseChips from './BaseChips.vue';
2022-09-06 12:03:37 +00:00
export default {
name: 'Chips',
2023-05-24 09:27:50 +00:00
extends: BaseChips,
2024-02-11 08:10:29 +00:00
inheritAttrs: false,
2022-09-06 12:03:37 +00:00
emits: ['update:modelValue', 'add', 'remove', 'focus', 'blur'],
data() {
return {
2024-02-02 15:46:45 +00:00
id: this.$attrs.id,
2022-09-06 12:03:37 +00:00
inputValue: null,
focused: false,
focusedIndex: null
};
},
2024-02-02 15:46:45 +00:00
watch: {
'$attrs.id': {
immediate: true,
handler: function (newValue) {
this.id = newValue || UniqueComponentId();
2024-03-27 23:27:46 +00:00
}
}
2024-02-02 15:46:45 +00:00
},
2022-09-06 12:03:37 +00:00
methods: {
onWrapperClick() {
this.$refs.input.focus();
},
onInput(event) {
this.inputValue = event.target.value;
this.focusedIndex = null;
},
onFocus(event) {
this.focused = true;
this.focusedIndex = null;
this.$emit('focus', event);
},
onBlur(event) {
this.focused = false;
this.focusedIndex = null;
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
if (this.addOnBlur) {
this.addItem(event, event.target.value, false);
}
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
this.$emit('blur', event);
},
onKeyDown(event) {
const inputValue = event.target.value;
2022-09-14 11:26:01 +00:00
switch (event.code) {
2022-09-06 12:03:37 +00:00
case 'Backspace':
if (inputValue.length === 0 && this.modelValue && this.modelValue.length > 0) {
if (this.focusedIndex !== null) {
this.removeItem(event, this.focusedIndex);
2022-09-14 11:26:01 +00:00
} else this.removeItem(event, this.modelValue.length - 1);
2022-09-06 12:03:37 +00:00
}
2022-09-14 11:26:01 +00:00
break;
2022-09-06 12:03:37 +00:00
case 'Enter':
case 'NumpadEnter':
2022-09-06 12:03:37 +00:00
if (inputValue && inputValue.trim().length && !this.maxedOut) {
this.addItem(event, inputValue, true);
}
2022-09-14 11:26:01 +00:00
break;
2022-09-06 12:03:37 +00:00
case 'ArrowLeft':
if (inputValue.length === 0 && this.modelValue && this.modelValue.length > 0) {
this.$refs.container.focus();
}
2022-09-14 11:26:01 +00:00
break;
2022-09-06 12:03:37 +00:00
case 'ArrowRight':
event.stopPropagation();
2022-09-14 11:26:01 +00:00
break;
2022-09-06 12:03:37 +00:00
default:
if (this.separator) {
2022-12-08 11:04:25 +00:00
if (this.separator === event.key || event.key.match(this.separator)) {
2022-09-06 12:03:37 +00:00
this.addItem(event, inputValue, true);
}
}
2022-09-14 11:26:01 +00:00
break;
2022-09-06 12:03:37 +00:00
}
},
onPaste(event) {
if (this.separator) {
let separator = this.separator.replace('\\n', '\n').replace('\\r', '\r').replace('\\t', '\t');
2022-09-06 12:03:37 +00:00
let pastedData = (event.clipboardData || window['clipboardData']).getData('Text');
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
if (pastedData) {
let value = this.modelValue || [];
let pastedValues = pastedData.split(separator);
2022-09-14 11:26:01 +00:00
pastedValues = pastedValues.filter((val) => this.allowDuplicate || value.indexOf(val) === -1);
2022-09-06 12:03:37 +00:00
value = [...value, ...pastedValues];
this.updateModel(event, value, true);
}
}
},
onContainerFocus() {
this.focused = true;
},
onContainerBlur() {
this.focusedIndex = -1;
this.focused = false;
},
onContainerKeyDown(event) {
switch (event.code) {
case 'ArrowLeft':
this.onArrowLeftKeyOn(event);
break;
case 'ArrowRight':
this.onArrowRightKeyOn(event);
break;
case 'Backspace':
this.onBackspaceKeyOn(event);
break;
default:
break;
}
},
onArrowLeftKeyOn() {
if (this.inputValue.length === 0 && this.modelValue && this.modelValue.length > 0) {
this.focusedIndex = this.focusedIndex === null ? this.modelValue.length - 1 : this.focusedIndex - 1;
if (this.focusedIndex < 0) this.focusedIndex = 0;
}
},
onArrowRightKeyOn() {
if (this.inputValue.length === 0 && this.modelValue && this.modelValue.length > 0) {
if (this.focusedIndex === this.modelValue.length - 1) {
this.focusedIndex = null;
this.$refs.input.focus();
2022-09-14 11:26:01 +00:00
} else {
2022-09-06 12:03:37 +00:00
this.focusedIndex++;
}
}
},
onBackspaceKeyOn(event) {
if (this.focusedIndex !== null) {
this.removeItem(event, this.focusedIndex);
}
},
updateModel(event, value, preventDefault) {
this.$emit('update:modelValue', value);
this.$emit('add', {
originalEvent: event,
value: value
});
this.$refs.input.value = '';
this.inputValue = '';
2022-12-08 11:04:25 +00:00
setTimeout(() => {
this.maxedOut && (this.focused = false);
}, 0);
2022-09-06 12:03:37 +00:00
if (preventDefault) {
event.preventDefault();
}
},
addItem(event, item, preventDefault) {
if (item && item.trim().length) {
2022-09-14 11:26:01 +00:00
let value = this.modelValue ? [...this.modelValue] : [];
2022-09-06 12:03:37 +00:00
if (this.allowDuplicate || value.indexOf(item) === -1) {
value.push(item);
this.updateModel(event, value, preventDefault);
}
}
},
removeItem(event, index) {
if (this.disabled) {
return;
}
let values = [...this.modelValue];
const removedItem = values.splice(index, 1);
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
this.focusedIndex = null;
this.$refs.input.focus();
this.$emit('update:modelValue', values);
this.$emit('remove', {
originalEvent: event,
value: removedItem
});
}
},
computed: {
maxedOut() {
return this.max && this.modelValue && this.max === this.modelValue.length;
},
focusedOptionId() {
return this.focusedIndex !== null ? `${this.id}_chips_item_${this.focusedIndex}` : null;
}
},
components: {
2024-04-08 12:20:51 +00:00
Chip
2022-09-06 12:03:37 +00:00
}
2022-09-14 11:26:01 +00:00
};
2022-09-06 12:03:37 +00:00
</script>