primevue-mirror/packages/primevue/src/tree/Tree.vue

260 lines
10 KiB
Vue
Raw Normal View History

2022-09-06 12:03:37 +00:00
<template>
2024-02-11 23:48:46 +00:00
<div :class="cx('root')" v-bind="ptmi('root')">
<template v-if="loading && loadingMode === 'mask'">
<div :class="cx('mask')" v-bind="ptm('mask')">
2023-05-26 11:22:08 +00:00
<slot name="loadingicon" :class="cx('loadingIcon')">
<i v-if="loadingIcon" :class="[cx('loadingIcon'), 'pi-spin', loadingIcon]" v-bind="ptm('loadingIcon')" />
<SpinnerIcon v-else spin :class="cx('loadingIcon')" v-bind="ptm('loadingIcon')" />
</slot>
2022-09-06 12:03:37 +00:00
</div>
</template>
2024-11-21 08:42:50 +00:00
<IconField v-if="filter" :unstyled="unstyled" :pt="{ ...ptm('pcFilter'), ...ptm('pcFilterContainer') }" :class="cx('pcFilterContainer')">
2024-08-29 07:09:13 +00:00
<InputText v-model="filterValue" autocomplete="off" :class="cx('pcFilterInput')" :placeholder="filterPlaceholder" :unstyled="unstyled" @keydown="onFilterKeydown" :pt="ptm('pcFilterInput')" />
<InputIcon :unstyled="unstyled" :pt="ptm('pcFilterIconContainer')">
<!--TODO: searchicon deprecated since v4.0-->
<slot :name="$slots.filtericon ? 'filtericon' : 'searchicon'" :class="cx('filterIcon')">
<SearchIcon :class="cx('filterIcon')" v-bind="ptm('filterIcon')" />
</slot>
</InputIcon>
</IconField>
<div :class="cx('wrapper')" :style="{ maxHeight: scrollHeight }" v-bind="ptm('wrapper')">
<slot name="header" :value="value" :expandedKeys="expandedKeys" :selectionKeys="selectionKeys" />
<ul :class="cx('rootChildren')" role="tree" :aria-labelledby="ariaLabelledby" :aria-label="ariaLabel" v-bind="ptm('rootChildren')">
2022-09-14 11:26:01 +00:00
<TreeNode
v-for="(node, index) of valueToRender"
:key="node.key"
:node="node"
:templates="$slots"
:level="level + 1"
:index="index"
:expandedKeys="d_expandedKeys"
@node-toggle="onNodeToggle"
@node-click="onNodeClick"
:selectionMode="selectionMode"
:selectionKeys="selectionKeys"
@checkbox-change="onCheckboxChange"
:loadingMode="loadingMode"
2023-08-02 09:32:28 +00:00
:unstyled="unstyled"
2024-01-14 14:09:03 +00:00
:pt="pt"
2022-09-14 11:26:01 +00:00
></TreeNode>
2022-09-06 12:03:37 +00:00
</ul>
<slot name="footer" :value="value" :expandedKeys="expandedKeys" :selectionKeys="selectionKeys" />
2022-09-06 12:03:37 +00:00
</div>
</div>
</template>
<script>
2024-09-09 20:56:48 +00:00
import { isFunction, resolveFieldData } from '@primeuix/utils/object';
2024-06-11 12:21:12 +00:00
import SearchIcon from '@primevue/icons/search';
import SpinnerIcon from '@primevue/icons/spinner';
import IconField from 'primevue/iconfield';
import InputIcon from 'primevue/inputicon';
2024-03-22 15:53:01 +00:00
import InputText from 'primevue/inputtext';
2023-05-26 11:22:08 +00:00
import BaseTree from './BaseTree.vue';
2022-12-08 11:04:25 +00:00
import TreeNode from './TreeNode.vue';
2022-09-06 12:03:37 +00:00
export default {
name: 'Tree',
2023-05-26 11:22:08 +00:00
extends: BaseTree,
2024-02-11 23:48:46 +00:00
inheritAttrs: false,
2023-11-23 15:48:14 +00:00
emits: ['node-expand', 'node-collapse', 'update:expandedKeys', 'update:selectionKeys', 'node-select', 'node-unselect', 'filter'],
2022-09-06 12:03:37 +00:00
data() {
return {
d_expandedKeys: this.expandedKeys || {},
filterValue: null
2022-09-14 11:26:01 +00:00
};
2022-09-06 12:03:37 +00:00
},
watch: {
expandedKeys(newValue) {
this.d_expandedKeys = newValue;
}
},
methods: {
onNodeToggle(node) {
const key = node.key;
if (this.d_expandedKeys[key]) {
delete this.d_expandedKeys[key];
this.$emit('node-collapse', node);
2022-09-14 11:26:01 +00:00
} else {
2022-09-06 12:03:37 +00:00
this.d_expandedKeys[key] = true;
this.$emit('node-expand', node);
}
2022-09-14 11:26:01 +00:00
this.d_expandedKeys = { ...this.d_expandedKeys };
2022-09-06 12:03:37 +00:00
this.$emit('update:expandedKeys', this.d_expandedKeys);
},
onNodeClick(event) {
if (this.selectionMode != null && event.node.selectable !== false) {
const metaSelection = event.nodeTouched ? false : this.metaKeySelection;
const _selectionKeys = metaSelection ? this.handleSelectionWithMetaKey(event) : this.handleSelectionWithoutMetaKey(event);
this.$emit('update:selectionKeys', _selectionKeys);
}
},
onCheckboxChange(event) {
this.$emit('update:selectionKeys', event.selectionKeys);
2022-09-14 11:26:01 +00:00
if (event.check) this.$emit('node-select', event.node);
else this.$emit('node-unselect', event.node);
2022-09-06 12:03:37 +00:00
},
handleSelectionWithMetaKey(event) {
const originalEvent = event.originalEvent;
const node = event.node;
2022-09-14 11:26:01 +00:00
const metaKey = originalEvent.metaKey || originalEvent.ctrlKey;
2022-09-06 12:03:37 +00:00
const selected = this.isNodeSelected(node);
let _selectionKeys;
if (selected && metaKey) {
if (this.isSingleSelectionMode()) {
_selectionKeys = {};
2022-09-14 11:26:01 +00:00
} else {
_selectionKeys = { ...this.selectionKeys };
2022-09-06 12:03:37 +00:00
delete _selectionKeys[node.key];
}
this.$emit('node-unselect', node);
2022-09-14 11:26:01 +00:00
} else {
2022-09-06 12:03:37 +00:00
if (this.isSingleSelectionMode()) {
_selectionKeys = {};
2022-09-14 11:26:01 +00:00
} else if (this.isMultipleSelectionMode()) {
_selectionKeys = !metaKey ? {} : this.selectionKeys ? { ...this.selectionKeys } : {};
2022-09-06 12:03:37 +00:00
}
_selectionKeys[node.key] = true;
this.$emit('node-select', node);
}
return _selectionKeys;
},
handleSelectionWithoutMetaKey(event) {
const node = event.node;
const selected = this.isNodeSelected(node);
let _selectionKeys;
if (this.isSingleSelectionMode()) {
if (selected) {
_selectionKeys = {};
this.$emit('node-unselect', node);
2022-09-14 11:26:01 +00:00
} else {
2022-09-06 12:03:37 +00:00
_selectionKeys = {};
_selectionKeys[node.key] = true;
this.$emit('node-select', node);
}
2022-09-14 11:26:01 +00:00
} else {
2022-09-06 12:03:37 +00:00
if (selected) {
2022-09-14 11:26:01 +00:00
_selectionKeys = { ...this.selectionKeys };
2022-09-06 12:03:37 +00:00
delete _selectionKeys[node.key];
this.$emit('node-unselect', node);
2022-09-14 11:26:01 +00:00
} else {
_selectionKeys = this.selectionKeys ? { ...this.selectionKeys } : {};
2022-09-06 12:03:37 +00:00
_selectionKeys[node.key] = true;
this.$emit('node-select', node);
}
}
return _selectionKeys;
},
isSingleSelectionMode() {
return this.selectionMode === 'single';
},
isMultipleSelectionMode() {
return this.selectionMode === 'multiple';
},
isNodeSelected(node) {
2022-09-14 11:26:01 +00:00
return this.selectionMode && this.selectionKeys ? this.selectionKeys[node.key] === true : false;
2022-09-06 12:03:37 +00:00
},
isChecked(node) {
2022-09-14 11:26:01 +00:00
return this.selectionKeys ? this.selectionKeys[node.key] && this.selectionKeys[node.key].checked : false;
2022-09-06 12:03:37 +00:00
},
isNodeLeaf(node) {
return node.leaf === false ? false : !(node.children && node.children.length);
},
onFilterKeydown(event) {
if (event.code === 'Enter' || event.code === 'NumpadEnter') {
2022-09-06 12:03:37 +00:00
event.preventDefault();
}
2023-11-23 15:48:14 +00:00
this.$emit('filter', { originalEvent: event, value: event.target.value });
2022-09-06 12:03:37 +00:00
},
findFilteredNodes(node, paramsWithoutNode) {
if (node) {
let matched = false;
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
if (node.children) {
let childNodes = [...node.children];
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
node.children = [];
2022-09-14 11:26:01 +00:00
2022-09-06 12:03:37 +00:00
for (let childNode of childNodes) {
2022-09-14 11:26:01 +00:00
let copyChildNode = { ...childNode };
2022-09-06 12:03:37 +00:00
if (this.isFilterMatched(copyChildNode, paramsWithoutNode)) {
matched = true;
node.children.push(copyChildNode);
}
}
}
if (matched) {
return true;
}
}
},
2022-09-14 11:26:01 +00:00
isFilterMatched(node, { searchFields, filterText, strict }) {
2022-09-06 12:03:37 +00:00
let matched = false;
2022-09-14 11:26:01 +00:00
for (let field of searchFields) {
let fieldValue = String(resolveFieldData(node, field)).toLocaleLowerCase(this.filterLocale);
2022-09-14 11:26:01 +00:00
if (fieldValue.indexOf(filterText) > -1) {
2022-09-06 12:03:37 +00:00
matched = true;
}
}
if (!matched || (strict && !this.isNodeLeaf(node))) {
2022-09-14 11:26:01 +00:00
matched = this.findFilteredNodes(node, { searchFields, filterText, strict }) || matched;
2022-09-06 12:03:37 +00:00
}
return matched;
}
},
computed: {
filteredValue() {
let filteredNodes = [];
2024-09-09 20:56:48 +00:00
const searchFields = isFunction(this.filterBy) ? [this.filterBy] : this.filterBy.split(',');
2022-09-06 12:03:37 +00:00
const filterText = this.filterValue.trim().toLocaleLowerCase(this.filterLocale);
const strict = this.filterMode === 'strict';
for (let node of this.value) {
2022-09-14 11:26:01 +00:00
let _node = { ...node };
let paramsWithoutNode = { searchFields, filterText, strict };
2022-09-06 12:03:37 +00:00
2022-09-14 11:26:01 +00:00
if (
(strict && (this.findFilteredNodes(_node, paramsWithoutNode) || this.isFilterMatched(_node, paramsWithoutNode))) ||
(!strict && (this.isFilterMatched(_node, paramsWithoutNode) || this.findFilteredNodes(_node, paramsWithoutNode)))
) {
2022-09-06 12:03:37 +00:00
filteredNodes.push(_node);
}
}
return filteredNodes;
},
valueToRender() {
2022-09-14 11:26:01 +00:00
if (this.filterValue && this.filterValue.trim().length > 0) return this.filteredValue;
else return this.value;
2022-09-06 12:03:37 +00:00
}
},
components: {
2024-03-22 15:53:01 +00:00
TreeNode,
InputText,
InputIcon,
IconField,
2024-03-22 15:53:01 +00:00
SearchIcon,
SpinnerIcon
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>