Преглед на файлове

1. 首页加loading
2. tree-column demo

daxiong.yang преди 7 години
родител
ревизия
4195300a37
променени са 5 файла, в които са добавени 479 реда и са изтрити 30 реда
  1. 167 0
      src/components/table-tree-column/index.vue
  2. 44 0
      src/components/table-tree-column/util.js
  3. 17 6
      src/views/layout/index.vue
  4. 138 0
      src/views/tree-table/data.js
  5. 113 24
      src/views/tree-table/index.vue

+ 167 - 0
src/components/table-tree-column/index.vue

@ -0,0 +1,167 @@
1
<template>
2
  <el-table-column :prop="prop" :label="label" :width="width">
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)}">
6
          <i :class="icon(scope.row)"></i>
7
          <i :class="floderIcon(scope.row)" style="padding-right: 7px;"></i>
8
        </span>
9
        <span>{{scope.row[prop]}}</span>
10
      </span>
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>
14
        </span>
15
        <span>{{scope.row[prop]}}</span>
16
      </span>
17
    </template>
18
  </el-table-column>
19
</template>
20
<script>
21
  import util from './util'
22
  export default {
23
    name: 'el-table-tree-column',
24
    props: {
25
      prop: {
26
        type: String
27
      },
28
      label: {
29
        type: String
30
      },
31
      width: {
32
        type: String
33
      },
34
      treeKey: {
35
        type: String,
36
        default: 'id'
37
      },
38
      childNumKey: {
39
        type: String,
40
        default: 'child_num'
41
      },
42
      parentKey: {
43
        type: String,
44
        default: 'parentId'
45
      },
46
      levelKey: {
47
        type: String,
48
        default: 'depth'
49
      },
50
      childKey: {
51
        type: String,
52
        default: 'children'
53
      },
54
      fileIcon: {
55
        type: String,
56
        default: 'el-icon-file'
57
      },
58
      folderIcon: {
59
        type: String,
60
        default: 'el-icon-folder'
61
      },
62
      remote: {
63
        type: Function,
64
        default: null
65
      }
66
    },
67
    computed: {
68
      owner () {
69
        let parent = this.$parent
70
        while (parent && !parent.tableId) {
71
          parent = parent.$parent
72
        }
73
        return parent
74
      }
75
    },
76
    data () {
77
      return { loading: false }
78
    },
79
    methods: {
80
      floderIcon (row) {
81
        var expanded = row.$extra && row.$extra.expanded
82
        return expanded ? this.folderIcon + '-open' : this.folderIcon
83
      },
84
      hasChild (row) {
85
        if (row[this.childNumKey] !== undefined) {
86
          return row[this.childNumKey] > 0 || false
87
        } else if (row[this.childKey] !== undefined) {
88
          return row[this.childKey].length > 0 || false
89
        } else {
90
          return false
91
        }
92
      },
93
      paddingLeft (row) {
94
        return parseInt(row[this.levelKey]) * 14 + 'px'
95
      },
96
      icon (row) {
97
        if (row.$extra && row.$extra.loading === true) return 'el-icon-loading'
98
        return row.$extra && row.$extra.expanded ? 'el-icon-caret-bottom' : 'el-icon-caret-right'
99
      },
100
      doexpanded (index, row) {
101
        var vm = this
102
        // var data = JSON.parse(JSON.stringify(this.owner.store.states._data))
103
        var data = this.owner.store.states._data
104
        debugger
105
        if (data[index].$extra === undefined) {
106
          data[index].$extra = { expanded: true }
107
        } else {
108
          data[index].$extra.expanded = !data[index].$extra.expanded
109
        }
110
        if (data[index].$extra.expanded) {
111
          if (this.remote != null) {
112
            var hash = util.hash()
113
            data[index].$extra.expanded = false
114
            data[index].$extra.hash = hash
115
            data[index].$extra.loading = true
116
            vm.owner.store.commit('setData', data)
117
            this.remote(row, function (result) {
118
              let list = vm.owner.store.states._data
119
              let _index = util.index(hash, list)
120
              list[_index].$extra = {
121
                loading: false,
122
                expanded: (result && result.length > 0) || false
123
              }
124
              if (result && result.length > 0) {
125
                var prefix = list.slice(0, _index + 1)
126
                var i = 0
127
                while (i < _index + 1) {
128
                  list.shift()
129
                  i++
130
                }
131
                list = prefix.concat(result).concat(list)
132
              } else {
133
                list[_index][vm.childNumKey] = 0
134
              }
135
              vm.owner.store.commit('setData', list)
136
            })
137
          } else {
138
            var prefix = data.slice(0, index + 1)
139
            var i = 0
140
            while (i < index + 1) {
141
              data.shift()
142
              i++
143
            }
144
            data = prefix.concat(row[vm.childKey]).concat(data)
145
            this.owner.store.commit('setData', data)
146
          }
147
        } else {
148
          var id = row[vm.treeKey]
149
          var result = []
150
          var removeIds = util.descendantsIds(
151
            id,
152
            data,
153
            this.parentKey,
154
            this.treeKey
155
          )
156
          data.forEach(function (item) {
157
            if (util.indexOf(item[vm.treeKey], removeIds) === -1) {
158
              result.push(item)
159
            }
160
          })
161
          data = result
162
          this.owner.store.commit('setData', data)
163
        }
164
      }
165
    }
166
  }
167
</script>

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

@ -0,0 +1,44 @@
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
}

+ 17 - 6
src/views/layout/index.vue

@ -1,10 +1,15 @@
1 1
<template>
2
  <div class="site-wrapper" :class="siteWarpperClasses">
3
    <topbar></topbar>
4
    <sidebar></sidebar>
5
    <div class="site-content__wrapper" :style="siteContentWarpperStyles">
6
      <router-view></router-view>
7
    </div>
2
  <div class="site-wrapper"
3
    :class="siteWarpperClasses"
4
    v-loading.fullscreen.lock="loading"
5
    element-loading-text="拼命加载中">
6
    <template v-if="!loading">
7
      <topbar></topbar>
8
      <sidebar></sidebar>
9
      <div class="site-content__wrapper" :style="siteContentWarpperStyles">
10
        <router-view></router-view>
11
      </div>
12
    </template>
8 13
  </div>
9 14
</template>
10 15
@ -15,6 +20,11 @@
15 20
  import API from '@/api'
16 21
  import { mapMutations } from 'vuex'
17 22
  export default {
23
    data () {
24
      return {
25
        loading: true
26
      }
27
    },
18 28
    components: {
19 29
      Topbar,
20 30
      Sidebar,
@ -50,6 +60,7 @@
50 60
      getUserInfo () {
51 61
        API.user.info().then(({data}) => {
52 62
          if (data && data.code === 0) {
63
            this.loading = false
53 64
            this.UPDATE_USER_ID({ id: data.user.userId })
54 65
            this.UPDATE_USER_NAME({ name: data.user.username })
55 66
          }

+ 138 - 0
src/views/tree-table/data.js

@ -0,0 +1,138 @@
1
export default [{
2
  'id': 1,
3
  'label': 'System',
4
  'parent_id': null,
5
  'url': null,
6
  'depth': 0,
7
  'child_num': 3,
8
  'description': 'System Manager',
9
  'children': [{
10
    'id': 2,
11
    'label': 'base',
12
    'parent_id': 1,
13
    'url': null,
14
    'depth': 1,
15
    'child_num': 5,
16
    'description': 'Base Manager',
17
    'expanded': true,
18
    'children': [{
19
      'id': 3,
20
      'label': 'Menus',
21
      'parent_id': 2,
22
      'url': '/menus',
23
      'depth': 2,
24
      'child_num': 0,
25
      'description': 'menu manager'
26
    }, {
27
      'id': 4,
28
      'label': 'Roles',
29
      'parent_id': 2,
30
      'url': '/roles',
31
      'depth': 2,
32
      'child_num': 0,
33
      'description': 'Role Manager'
34
    }, {
35
      'id': 5,
36
      'label': 'Users',
37
      'parent_id': 2,
38
      'url': '/users',
39
      'depth': 2,
40
      'child_num': 0,
41
      'description': 'User Manager'
42
    }]
43
  }]
44
}, {
45
  'id': 6,
46
  'label': 'Customs',
47
  'parent_id': null,
48
  'url': null,
49
  'depth': 0,
50
  'child_num': 2,
51
  'description': 'Custom Manager',
52
  'children': [{
53
    'id': 7,
54
    'label': 'CustomList',
55
    'parent_id': 6,
56
    'url': '/customs',
57
    'depth': 1,
58
    'child_num': 0,
59
    'description': 'CustomList'
60
  }]
61
}, {
62
  'id': 8,
63
  'label': 'Templates',
64
  'parent_id': null,
65
  'url': null,
66
  'depth': 0,
67
  'child_num': 1,
68
  'description': 'Template Manager',
69
  'children': [{
70
    'id': 9,
71
    'label': 'TemplateList',
72
    'parent_id': 8,
73
    'url': '/doc_templates',
74
    'depth': 1,
75
    'child_num': 0,
76
    'description': 'Template Manager'
77
  }]
78
}, {
79
  'id': 10,
80
  'label': 'Bussiness',
81
  'parent_id': null,
82
  'url': null,
83
  'depth': 0,
84
  'child_num': 2,
85
  'description': 'Bussiness Manager',
86
  'expanded': true,
87
  'children': [{
88
    'id': 11,
89
    'label': 'BussinessList',
90
    'parent_id': 10,
91
    'url': null,
92
    'depth': 1,
93
    'child_num': 2,
94
    'description': 'BussinessList',
95
    'children': [{
96
      'id': 12,
97
      'label': 'Currencies',
98
      'parent_id': 11,
99
      'url': '/currencies',
100
      'depth': 2,
101
      'child_num': 0,
102
      'description': 'Currencies'
103
    }, {
104
      'id': 13,
105
      'label': 'Dealtypes',
106
      'parent_id': 11,
107
      'url': '/dealtypes',
108
      'depth': 2,
109
      'child_num': 0,
110
      'description': 'Dealtypes'
111
    }]
112
  }, {
113
    'id': 14,
114
    'label': 'Products',
115
    'parent_id': 10,
116
    'url': null,
117
    'depth': 1,
118
    'child_num': 2,
119
    'description': 'Products',
120
    'children': [{
121
      'id': 15,
122
      'label': 'ProductTypes',
123
      'parent_id': 14,
124
      'url': '/productTypes',
125
      'depth': 2,
126
      'child_num': 0,
127
      'description': 'ProductTypes'
128
    }, {
129
      'id': 16,
130
      'label': 'ProductList',
131
      'parent_id': 14,
132
      'url': '/products',
133
      'depth': 2,
134
      'child_num': 0,
135
      'description': 'ProductList'
136
    }]
137
  }]
138
}]

+ 113 - 24
src/views/tree-table/index.vue

@ -2,16 +2,114 @@
2 2
  <div class="mod-dept">
3 3
    <tree-table
4 4
      :columns="treeTableColumns"
5
      :data="datList"
5
      :data="dataList"
6 6
      border
7
      style="margin-top: 10px;">
7
      style="width: 100%;">
8
      <el-table-column
9
        prop="name"
10
        header-align="center"
11
        align="center"
12
        label="菜单名称">
13
      </el-table-column>
14
      <el-table-column
15
        prop="parentName"
16
        header-align="center"
17
        align="center"
18
        label="上级菜单">
19
      </el-table-column>
20
      <el-table-column
21
        prop="icon"
22
        header-align="center"
23
        align="center"
24
        label="图标">
25
        <template slot-scope="scope">
26
          <i :class="['fa-lg', scope.row.icon]"></i>
27
        </template>
28
      </el-table-column>
29
      <el-table-column
30
        prop="type"
31
        header-align="center"
32
        align="center"
33
        label="类型">
34
        <template slot-scope="scope">
35
          <el-tag v-if="scope.row.type === 0" size="small">目录</el-tag>
36
          <el-tag v-else-if="scope.row.type === 1" size="small" type="success">菜单</el-tag>
37
          <el-tag v-else-if="scope.row.type === 2" size="small" type="info">按钮</el-tag>
38
        </template>
39
      </el-table-column>
40
      <el-table-column
41
        prop="orderNum"
42
        header-align="center"
43
        align="center"
44
        label="排序号">
45
      </el-table-column>
46
      <el-table-column
47
        prop="url"
48
        header-align="center"
49
        align="center"
50
        width="150"
51
        :show-overflow-tooltip="true"
52
        label="菜单URL">
53
      </el-table-column>
54
      <el-table-column
55
        prop="perms"
56
        header-align="center"
57
        align="center"
58
        width="150"
59
        :show-overflow-tooltip="true"
60
        label="授权标识">
61
      </el-table-column>
62
      <el-table-column
63
        header-align="center"
64
        align="center"
65
        width="200"
66
        label="操作">
67
        <template slot-scope="scope">
68
          <el-button v-if="isAuth('sys:user:update')" type="text" size="small" @click="addOrUpdateHandle(scope.row.menuId)">修改</el-button>
69
          <el-button v-if="isAuth('sys:user:delete')" type="text" size="small" @click="deleteHandle(scope.row.menuId)">删除</el-button>
70
        </template>
71
      </el-table-column>
8 72
    </tree-table>
73
    <!-- <el-table :data="treeData" border max-height="250">
74
      <el-table-column
75
        prop="parentName"
76
        header-align="center"
77
        align="center"
78
        label="上级菜单">
79
      </el-table-column>
80
      <table-tree-column 
81
        file-icon="icon icon-file" 
82
        folder-icon="icon icon-folder" 
83
        prop="value" label="名称" width="220">
84
      </table-tree-column>
85
      <el-table-column
86
        prop="icon"
87
        header-align="center"
88
        align="center"
89
        label="图标">
90
        <template slot-scope="scope">
91
          <i :class="['fa-lg', scope.row.icon]"></i>
92
        </template>
93
      </el-table-column>
94
  </el-table> -->
95
    <el-table :data="dataList" border max-height="400">
96
      <el-table-column prop="id" label="id" width="180"></el-table-column>
97
      <table-tree-column 
98
        file-icon="icon icon-file" 
99
        folder-icon="icon icon-folder" 
100
        prop="label" label="MenuName" width="320">
101
      </table-tree-column>
102
      <el-table-column prop="description" label="Description" :show-overflow-tooltip="true" width="180"></el-table-column>
103
    </el-table>
9 104
  </div>
10 105
</template>
11 106
12 107
<script>
13
  import API from '@/api'
14 108
  import TreeTable from '@/components/tree-table'
109
  import TableTreeColumn from '@/components/table-tree-column/index.vue'
110
  import Data from './data.js'
111
  import API from '@/api'
112
  import { treeDataTranslate } from '@/utils'
15 113
  export default {
16 114
    data () {
17 115
      return {
@ -22,36 +120,27 @@
22 120
            width: 200
23 121
          }
24 122
        ],
25
        datList: []
123
        dataList: [],
124
        treeData: []
26 125
      }
27 126
    },
28 127
    components: {
29
      TreeTable
128
      TreeTable,
129
      TableTreeColumn
30 130
    },
31 131
    created () {
32
      this.getDeptList()
132
      this.treeData = Data
133
      console.log(this.treeData)
134
      this.getDataList()
33 135
    },
34 136
    methods: {
35
      // 获取部门列表
36
      getDeptList () {
137
      // 获取数据列表
138
      getDataList () {
139
        this.dataListLoading = true
37 140
        API.menu.list().then(({data}) => {
38
          this.datList = this.dataTranslate(data || [], 'menuId', 'parentId')
141
          this.dataList = treeDataTranslate(data, 'menuId')
142
          this.dataListLoading = false
39 143
        })
40
      },
41
      // 数据转换
42
      dataTranslate (data, sign, parentSign) {
43
        for (var i = 0; i < data.length; i++) {
44
          for (var j = 0; j < data.length; j++) {
45
            if (data[i][parentSign] === data[j][sign]) {
46
              data[j]['children'] = data[j]['children'] || []
47
              data[j]['children'].push(data[i])
48
              data.splice(i, 1)
49
              i--
50
              break
51
            }
52
          }
53
        }
54
        return data
55 144
      }
56 145
    }
57 146
  }