Browse Source

优化table-tree-column

daxiong.yang 7 years ago
parent
commit
3b9595918f

+ 46 - 76
src/components/table-tree-column/index.vue

1
<template>
1
<template>
2
  <el-table-column :prop="prop" v-bind="$attrs">
2
  <el-table-column :prop="prop" v-bind="$attrs">
3
    <template slot-scope="scope">
3
    <template slot-scope="scope">
4
      <span v-if="hasChild(scope.row)" @click.prevent="doexpanded(scope.$index,scope.row)" >
5
        <span :style="{paddingLeft : paddingLeft(scope.row)}">
4
      <span v-if="hasChild(scope.row)" @click.prevent="expandedHandle(scope.$index, scope.row)">
5
        <span :style="{ 'padding-left': paddingLeft(scope.row) }">
6
          <i :class="icon(scope.row)"></i>
6
          <i :class="icon(scope.row)"></i>
7
          <i :class="floderIcon(scope.row)" style="padding-right: 7px;"></i>
7
          <i :class="floderIcon(scope.row)" style="padding-right: 7px;"></i>
8
        </span>
8
        </span>
9
        <span>{{scope.row[prop]}}</span>
9
        <span>{{scope.row[prop]}}</span>
10
      </span>
10
      </span>
11
      <span v-if="!hasChild(scope.row)">
11
      <span v-if="!hasChild(scope.row)">
12
        <span :style="{paddingLeft : paddingLeft(scope.row)}">
13
          <i :class="fileIcon" style="padding-right: 7px;padding-left:18px"></i>
12
        <span :style="{ 'padding-left': paddingLeft(scope.row) }">
13
          <i :class="fileIcon" style="padding-right: 7px; padding-left:18px;"></i>
14
        </span>
14
        </span>
15
        <span>{{scope.row[prop]}}</span>
15
        <span>{{scope.row[prop]}}</span>
16
      </span>
16
      </span>
19
</template>
19
</template>
20
20
21
<script>
21
<script>
22
  import util from './util'
23
  export default {
22
  export default {
24
    name: 'el-table-tree-column',
23
    name: 'table-tree-column',
24
    data () {
25
      return { loading: false }
26
    },
25
    props: {
27
    props: {
26
      prop: {
28
      prop: {
27
        type: String
29
        type: String
53
      folderIcon: {
55
      folderIcon: {
54
        type: String,
56
        type: String,
55
        default: 'el-icon-folder'
57
        default: 'el-icon-folder'
56
      },
57
      remote: {
58
        type: Function,
59
        default: null
60
      }
58
      }
61
    },
59
    },
62
    computed: {
63
      owner () {
64
        let parent = this.$parent
65
        while (parent && !parent.tableId) {
66
          parent = parent.$parent
67
        }
68
        return parent
69
      }
70
    },
71
    data () {
72
      return { loading: false }
73
    },
74
    created () {
60
    created () {
75
      console.log(this.$attrs)
61
      console.log(this.$attrs)
76
    },
62
    },
77
    methods: {
63
    methods: {
78
      floderIcon (row) {
64
      floderIcon (row) {
79
        var expanded = row.$extra && row.$extra.expanded
80
        return expanded ? this.folderIcon + '-open' : this.folderIcon
65
        return row._expanded ? this.folderIcon + '-open' : this.folderIcon
81
      },
66
      },
82
      hasChild (row) {
67
      hasChild (row) {
83
        if (row[this.childNumKey] !== undefined) {
68
        if (row[this.childNumKey] !== undefined) {
92
        return (parseInt(row[this.levelKey]) * 14) + 'px'
77
        return (parseInt(row[this.levelKey]) * 14) + 'px'
93
      },
78
      },
94
      icon (row) {
79
      icon (row) {
95
        if (row.$extra && row.$extra.loading) return 'el-icon-loading'
96
        return row.$extra && row.$extra.expanded ? 'el-icon-caret-bottom' : 'el-icon-caret-right'
80
        return row._expanded ? 'el-icon-caret-bottom' : 'el-icon-caret-right'
97
      },
81
      },
98
      doexpanded (index, row) {
99
        var vm = this
100
        var data = JSON.parse(JSON.stringify(this.owner.store.states._data))
101
        if (data[index].$extra === undefined) {
102
          data[index].$extra = { expanded: true }
103
        } else {
104
          data[index].$extra.expanded = !data[index].$extra.expanded
105
        }
106
        if (data[index].$extra.expanded) {
107
          if (this.remote !== null) {
108
            var hash = util.hash()
109
            data[index].$extra.expanded = false
110
            data[index].$extra.hash = hash
111
            data[index].$extra.loading = true
112
            vm.owner.store.commit('setData', data)
113
            this.remote(row, function (result) {
114
              let list = vm.owner.store.states._data
115
              let _index = util.index(hash, list)
116
              list[_index].$extra = {
117
                loading: false,
118
                expanded: (result && result.length > 0) || false
119
              }
120
              if (result && result.length > 0) {
121
                var prefix = list.slice(0, _index + 1)
122
                var i = 0
123
                while (i < _index + 1) {
124
                  list.shift()
125
                  i++
126
                }
127
                list = prefix.concat(result).concat(list)
128
              } else {
129
                list[_index][vm.childNumKey] = 0
130
              }
131
              vm.owner.store.commit('setData', list)
132
            })
133
          } else {
134
            var prefix = data.slice(0, index + 1)
135
            var i = 0
136
            while (i < index + 1) {
137
              data.shift()
138
              i++
139
            }
140
            data = prefix.concat(row[vm.childKey]).concat(data)
141
            this.owner.store.commit('setData', data)
142
          }
82
      expandedHandle (index, row) {
83
        var data = JSON.parse(JSON.stringify(this.$parent.store.states.data))
84
        data[index]._expanded = !data[index]._expanded
85
        console.log(data)
86
        if (data[index]._expanded) {
87
          data = data.splice(0, index + 1).concat(row[this.childKey]).concat(data)
88
          this.$parent.store.commit('setData', data)
143
        } else {
89
        } else {
144
          var id = row[vm.treeKey]
90
          var id = row[this.treeKey]
145
          var result = []
91
          var result = []
146
          var removeIds = util.descendantsIds(id, data, this.parentKey, this.treeKey)
147
          data.forEach(function (item) {
148
            if (util.indexOf(item[vm.treeKey], removeIds) === -1) {
92
          var removeIds = this.descendantsIds(id, data, this.parentKey, this.treeKey)
93
          data.forEach(item => {
94
            if (removeIds.indexOf(item[this.treeKey]) === -1) {
149
              result.push(item)
95
              result.push(item)
150
            }
96
            }
151
          })
97
          })
152
          data = result
98
          data = result
153
          this.owner.store.commit('setData', data)
99
          this.$parent.store.commit('setData', data)
100
        }
101
      },
102
      descendantsIds (id, data, parentKey, treeKey) {
103
        var result = []
104
        var compare = [id]
105
        var length = -1
106
        // if (compare.length !== -1) {
107
        //   data.forEach(item => {
108
        //     if (compare.indexOf(item[parentKey]) > -1 && compare.indexOf(item[treeKey]) === -1) {
109
        //       result.push(item[treeKey])
110
        //       compare.push(item[treeKey])
111
        //     }
112
        //   })
113
        // }
114
        while (length !== compare.length) {
115
          length = compare.length
116
          console.log(length)
117
          data.forEach(item => {
118
            if (compare.indexOf(item[parentKey]) > -1 && compare.indexOf(item[treeKey]) === -1) {
119
              result.push(item[treeKey])
120
              compare.push(item[treeKey])
121
            }
122
          })
154
        }
123
        }
124
        return result
155
      }
125
      }
156
    }
126
    }
157
  }
127
  }

+ 0 - 44
src/components/table-tree-column/util.js

1
const indexOf = (val, arr) => {
2
  var has = -1
3
  for (var i = 0; i < arr.length; i++) {
4
    if (arr[i] === val) {
5
      has = i
6
      break
7
    }
8
  }
9
  return has
10
}
11
12
const descendantsIds = (id, data, parentKey, treeKey) => {
13
  var result = []
14
  var compare = [id]
15
  var length = -1
16
  while (length !== compare.length) {
17
    length = compare.length
18
    data.forEach(function (item) {
19
      if (indexOf(item[parentKey], compare) > -1 && indexOf(item[treeKey], compare) === -1) {
20
        result.push(item[treeKey])
21
        compare.push(item[treeKey])
22
      }
23
    })
24
  }
25
  return result
26
}
27
const hash = () => Math.floor(Math.random() * Math.random() * Math.random() * Math.random() * 1000)
28
const index = (hash, data) => {
29
  var i = 0
30
  while (data[i]) {
31
    if (data[i].$extra && data[i].$extra.hash === hash) {
32
      break
33
    }
34
    i++
35
  }
36
  return i
37
}
38
39
export default {
40
  indexOf,
41
  descendantsIds,
42
  hash,
43
  index
44
}

+ 2 - 2
src/views/menu/index.vue

20
        prop="name"
20
        prop="name"
21
        header-align="center"
21
        header-align="center"
22
        align="center"
22
        align="center"
23
        file-icon="icon icon-file" 
24
        folder-icon="icon icon-folder" 
25
        treeKey="menuId"
23
        treeKey="menuId"
26
        min-width="120"
24
        min-width="120"
27
        label="名称">
25
        label="名称">
30
        prop="parentName"
28
        prop="parentName"
31
        header-align="center"
29
        header-align="center"
32
        align="center"
30
        align="center"
31
        min-width="120"
33
        label="上级菜单">
32
        label="上级菜单">
34
      </el-table-column>
33
      </el-table-column>
35
      <el-table-column
34
      <el-table-column
75
        label="授权标识">
74
        label="授权标识">
76
      </el-table-column>
75
      </el-table-column>
77
      <el-table-column
76
      <el-table-column
77
        fixed="right"
78
        header-align="center"
78
        header-align="center"
79
        align="center"
79
        align="center"
80
        width="200"
80
        width="200"