123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350 |
- <!--
-
- -->
- <template>
- <div>
- <div v-show="isShowSelect" class="mask" @click="isShowSelect = !isShowSelect" />
- <el-popover
- v-model="isShowSelect"
- :disabled="disabled"
- placement="bottom-start"
- :width="width"
- trigger="manual"
- @hide="popoverHide"
- >
- <el-tree
- ref="tree"
- class="common-tree"
- :style="style"
- :data="data"
- :props="defaultProps"
- :show-checkbox="multiple"
- :node-key="nodeKey"
- :disabled="disabled"
- :check-strictly="checkStrictly"
- :default-expand-all="expandAll"
- :expand-on-click-node="false"
- :check-on-click-node="multiple"
- :highlight-current="true"
- @node-click="handleNodeClick"
- @check-change="handleCheckChange"
- />
- <el-select
- slot="reference"
- ref="select"
- v-model="selectedData"
- :style="selectStyle"
- :size="size"
- :disabled="disabled"
- :placeholder="placeholder"
- :multiple="multiple"
- :clearable="clearable"
- :collapse-tags="collapseTags"
- class="tree-select"
- @click.native="isShowSelect = !isShowSelect"
- @remove-tag="removeSelectedNodes"
- @clear="removeSelectedNode"
- @change="changeSelectedNodes"
- >
- <el-option v-for="item in options" :key="item.value" :label="item.label" :value="item.value" />
- </el-select>
- </el-popover>
- </div>
- </template>
- <script>
- export default {
- name: 'TreeSelect',
- props: {
-
- data: {
- type: Array,
- default() {
- return []
- }
- },
- defaultProps: {
- type: Object,
- default() {
- return {}
- }
- },
-
- multiple: {
- type: Boolean,
- default() {
- return false
- }
- },
-
- onlyleaf: {
- type: Boolean,
- default() {
- return false
- }
- },
-
- clearable: {
- type: Boolean,
- default() {
- return false
- }
- },
- disabled: {
- type: Boolean,
- default() {
- return true
- }
- },
- placeholder: {
- type: String,
- default() {
- return '请选择'
- }
- },
-
- collapseTags: {
- type: Boolean,
- default() {
- return false
- }
- },
-
- expandAll: {
- type: Boolean,
- default() {
- return false
- }
- },
- nodeKey: {
- type: String,
- default() {
- return 'id'
- }
- },
-
- checkStrictly: {
- type: Boolean,
- default() {
- return false
- }
- },
-
- checkedKeys: {
- type: Array,
- default() {
- return []
- }
- },
- size: {
- type: String,
- default() {
- return 'small'
- }
- },
- width: {
- type: Number,
- default() {
- return 250
- }
- },
- height: {
- type: Number,
- default() {
- return 300
- }
- }
- },
- data() {
- return {
- isShowSelect: false,
- options: [],
- selectedData: [],
- style: 'width:' + this.width + 'px;' + 'height:' + this.height + 'px;',
- selectStyle: 'width:' + (this.width + 24) + 'px;',
- checkedIds: [],
- checkedData: []
- }
- },
- watch: {
- isShowSelect(val) {
-
- this.$refs.select.blur()
- },
- checkedKeys(val) {
- if (!val) return
- this.checkedKeys = val
- this.initCheckedData()
- }
- },
- mounted() {
- this.initCheckedData()
- },
- methods: {
-
- setSelectOption(node) {
- const tmpMap = {}
- tmpMap.value = node.key
- tmpMap.label = node.label
- this.options = []
- this.options.push(tmpMap)
- this.selectedData = node.key
- },
-
- checkSelectedNode(checkedKeys) {
- var item = checkedKeys[0]
- this.$refs.tree.setCurrentKey(item)
- var node = this.$refs.tree.getNode(item)
- this.setSelectOption(node)
- },
-
- checkSelectedNodes(checkedKeys) {
- this.$refs.tree.setCheckedKeys(checkedKeys)
- },
-
- clearSelectedNode() {
- this.selectedData = ''
- this.$refs.tree.setCurrentKey(null)
- },
-
- clearSelectedNodes() {
- var checkedKeys = this.$refs.tree.getCheckedKeys()
- for (let i = 0; i < checkedKeys.length; i++) {
- this.$refs.tree.setChecked(checkedKeys[i], false)
- }
- },
- initCheckedData() {
- if (this.multiple) {
-
- if (this.checkedKeys.length > 0) {
- this.checkSelectedNodes(this.checkedKeys)
- } else {
- this.clearSelectedNodes()
- }
- } else {
-
- if (this.checkedKeys.length > 0) {
- this.checkSelectedNode(this.checkedKeys)
- } else {
- this.clearSelectedNode()
- }
- }
- },
- popoverHide() {
- if (this.multiple) {
- this.checkedIds = this.$refs.tree.getCheckedKeys()
- this.checkedData = this.$refs.tree.getCheckedNodes()
- } else {
- this.checkedIds = this.$refs.tree.getCurrentKey()
- this.checkedData = this.$refs.tree.getCurrentNode()
- }
- this.$emit('popoverHide', this.checkedIds, this.checkedData)
- },
-
- handleNodeClick(data, node) {
- if (!this.onlyleaf || (this.onlyleaf && (data.haschildren === 0 || data.children == undefined))) {
- if (!this.multiple) {
- this.setSelectOption(node)
- this.isShowSelect = !this.isShowSelect
- this.$emit('change', this.selectedData)
- }
- }
- },
-
- handleCheckChange() {
- var checkedKeys = this.$refs.tree.getCheckedKeys()
- this.options = checkedKeys.map((item) => {
- var node = this.$refs.tree.getNode(item)
- const tmpMap = {}
- tmpMap.value = node.key
- tmpMap.label = node.label
- return tmpMap
- })
- this.selectedData = this.options.map((item) => {
- return item.value
- })
- this.$emit('change', this.selectedData)
- },
-
- removeSelectedNodes(val) {
- this.$refs.tree.setChecked(val, false)
- var node = this.$refs.tree.getNode(val)
- if (!this.checkStrictly && node.childNodes.length > 0) {
- this.treeToList(node).map(item => {
- if (item.childNodes.length <= 0) {
- this.$refs.tree.setChecked(item, false)
- }
- })
- this.handleCheckChange()
- }
- this.$emit('change', this.selectedData)
- },
- treeToList(tree) {
- var queen = []
- var out = []
- queen = queen.concat(tree)
- while (queen.length) {
- var first = queen.shift()
- if (first.childNodes) {
- queen = queen.concat(first.childNodes)
- }
- out.push(first)
- }
- return out
- },
-
- removeSelectedNode() {
- this.clearSelectedNode()
- this.$emit('change', this.selectedData)
- this.popoverHide()
- },
-
- changeSelectedNodes(selectedData) {
-
- if (this.multiple && selectedData.length <= 0) {
- this.clearSelectedNodes()
- }
- this.$emit('change', this.selectedData)
- }
- }
- }
- </script>
- <style scoped>
- .mask{
- width: 100%;
- height: 100%;
- position: fixed;
- top: 0;
- left: 0;
- opacity: 0;
- z-index: 11;
- }
- .common-tree{
- overflow: auto;
- }
-
- </style>
|