user.vue 11.3 KB
Newer Older
1 2
<template>
  <div>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
3
    <warning-bar title="注:右上角头像下拉可切换角色" />
Mr.奇淼('s avatar
Mr.奇淼( 已提交
4 5
    <div class="gva-table-box">
      <div class="gva-btn-list">
P
piexlmax 已提交
6
        <el-button size="mini" type="primary" icon="plus" @click="addUser">新增用户</el-button>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
7 8
      </div>
      <el-table :data="tableData">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
9
        <el-table-column align="left" label="头像" min-width="50">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
10
          <template #default="scope">
11
            <CustomPic style="margin-top:8px" :pic-src="scope.row.headerImg" />
Mr.奇淼('s avatar
Mr.奇淼( 已提交
12 13
          </template>
        </el-table-column>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
14 15
        <el-table-column align="left" label="UUID" min-width="250" prop="uuid" />
        <el-table-column align="left" label="用户名" min-width="150" prop="userName" />
Mr.奇淼('s avatar
Mr.奇淼( 已提交
16 17
        <el-table-column align="left" label="昵称" min-width="100" prop="nickName">
          <template #default="scope">
P
piexlmax 已提交
18 19 20 21 22
            <p v-if="!scope.row.editFlag" class="nickName">{{ scope.row.nickName }}
              <el-icon class="pointer" color="#66b1ff" @click="openEidt(scope.row)">
                <edit />
              </el-icon>
            </p>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
23 24
            <p v-if="scope.row.editFlag" class="nickName">
              <el-input v-model="scope.row.nickName" />
P
piexlmax 已提交
25 26 27 28 29 30
              <el-icon class="pointer" color="#67c23a" @click="enterEdit(scope.row)">
                <check />
              </el-icon>
              <el-icon class="pointer" color="#f23c3c" @click="closeEdit(scope.row)">
                <close />
              </el-icon>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
31 32 33
            </p>
          </template>
        </el-table-column>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
34
        <el-table-column align="left" label="用户角色" min-width="150">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
35 36 37 38 39 40 41 42 43 44 45 46 47
          <template #default="scope">
            <el-cascader
              v-model="scope.row.authorityIds"
              :options="authOptions"
              :show-all-levels="false"
              collapse-tags
              :props="{ multiple:true,checkStrictly: true,label:'authorityName',value:'authorityId',disabled:'disabled',emitPath:false}"
              :clearable="false"
              @visible-change="(flag)=>{changeAuthority(scope.row,flag)}"
              @remove-tag="()=>{changeAuthority(scope.row,false)}"
            />
          </template>
        </el-table-column>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
48
        <el-table-column align="left" label="操作" min-width="150">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
49
          <template #default="scope">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
50
            <el-popover :visible="scope.row.visible" placement="top" width="160">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
51
              <p>确定要删除此用户吗</p>
52
              <div style="text-align: right; margin-top: 8px;">
Mr.奇淼('s avatar
Mr.奇淼( 已提交
53 54 55 56
                <el-button size="mini" type="text" @click="scope.row.visible = false">取消</el-button>
                <el-button type="primary" size="mini" @click="deleteUser(scope.row)">确定</el-button>
              </div>
              <template #reference>
P
piexlmax 已提交
57
                <el-button type="text" icon="delete" size="mini">删除</el-button>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
58 59
              </template>
            </el-popover>
P
piexlmax 已提交
60
            <el-button type="text" icon="magic-stick" size="mini" @click="resetPassword(scope.row)">重置密码</el-button>
Mr.奇淼('s avatar
Mr.奇淼( 已提交
61 62 63 64 65 66 67 68 69 70 71 72 73 74 75
          </template>
        </el-table-column>
      </el-table>
      <div class="gva-pagination">
        <el-pagination
          :current-page="page"
          :page-size="pageSize"
          :page-sizes="[10, 30, 50, 100]"
          :total="total"
          layout="total, sizes, prev, pager, next, jumper"
          @current-change="handleCurrentChange"
          @size-change="handleSizeChange"
        />
      </div>
    </div>
76
    <el-dialog v-model="addUserDialog" custom-class="user-dialog" title="新增用户">
P
piexlmax 已提交
77 78
      <el-form ref="userForm" :rules="rules" :model="userInfo" label-width="80px">
        <el-form-item label="用户名" prop="username">
何秀钢 已提交
79
          <el-input v-model="userInfo.username" />
80
        </el-form-item>
P
piexlmax 已提交
81
        <el-form-item label="密码" prop="password">
何秀钢 已提交
82
          <el-input v-model="userInfo.password" />
83
        </el-form-item>
P
piexlmax 已提交
84
        <el-form-item label="别名" prop="nickName">
何秀钢 已提交
85
          <el-input v-model="userInfo.nickName" />
86
        </el-form-item>
P
piexlmax 已提交
87
        <el-form-item label="用户角色" prop="authorityId">
88
          <el-cascader
Mr.奇淼('s avatar
Mr.奇淼( 已提交
89
            v-model="userInfo.authorityIds"
P
piexlmax 已提交
90
            style="width:100%"
Mr.奇淼('s avatar
Mr.奇淼( 已提交
91 92
            :options="authOptions"
            :show-all-levels="false"
93 94
            :props="{ multiple:true,checkStrictly: true,label:'authorityName',value:'authorityId',disabled:'disabled',emitPath:false}"
            :clearable="false"
何秀钢 已提交
95
          />
96
        </el-form-item>
P
piexlmax 已提交
97 98 99 100 101 102 103
        <el-form-item label="头像" label-width="80px">
          <div style="display:inline-block" @click="openHeaderChange">
            <img v-if="userInfo.headerImg" class="header-img-box" :src="(userInfo.headerImg && userInfo.headerImg.slice(0, 4) !== 'http')?path+userInfo.headerImg:userInfo.headerImg">
            <div v-else class="header-img-box">从媒体库选择</div>
          </div>
        </el-form-item>

104
      </el-form>
105 106
      <template #footer>
        <div class="dialog-footer">
P
piexlmax 已提交
107 108
          <el-button size="small" @click="closeAddUserDialog">取 消</el-button>
          <el-button size="small" type="primary" @click="enterAddUserDialog">确 定</el-button>
109 110
        </div>
      </template>
111
    </el-dialog>
何秀钢 已提交
112
    <ChooseImg ref="chooseImg" :target="userInfo" :target-key="`headerImg`" />
113 114 115 116 117
  </div>
</template>

<script>
// 获取列表内容封装在mixins内部  getTableData方法 初始化已封装完成
118
const path = import.meta.env.VITE_BASE_API
Mr.奇淼('s avatar
Mr.奇淼( 已提交
119 120
import {
  getUserList,
121
  setUserAuthorities,
Mr.奇淼('s avatar
Mr.奇淼( 已提交
122 123
  register,
  deleteUser
何秀钢 已提交
124 125 126 127
} from '@/api/user'
import { getAuthorityList } from '@/api/authority'
import infoList from '@/mixins/infoList'
import { mapGetters } from 'vuex'
Mr.奇淼('s avatar
Mr.奇淼( 已提交
128 129
import CustomPic from '@/components/customPic/index.vue'
import ChooseImg from '@/components/chooseImg/index.vue'
P
piexlmax 已提交
130
import warningBar from '@/components/warningBar/warningBar.vue'
Mr.奇淼('s avatar
Mr.奇淼( 已提交
131
import { setUserInfo, resetPassword } from '@/api/user.js'
132
export default {
何秀钢 已提交
133
  name: 'Api',
P
piexlmax 已提交
134
  components: { CustomPic, ChooseImg, warningBar },
135 136 137 138
  mixins: [infoList],
  data() {
    return {
      listApi: getUserList,
Mr.奇淼('s avatar
Mr.奇淼( 已提交
139
      path: path,
140 141
      authOptions: [],
      addUserDialog: false,
Mr.奇淼('s avatar
Mr.奇淼( 已提交
142
      backNickName: '',
143
      userInfo: {
何秀钢 已提交
144 145 146 147
        username: '',
        password: '',
        nickName: '',
        headerImg: '',
148 149
        authorityId: '',
        authorityIds: []
Mr.奇淼('s avatar
Mr.奇淼( 已提交
150 151
      },
      rules: {
Mr.奇淼('s avatar
Mr.奇淼( 已提交
152
        username: [
何秀钢 已提交
153 154
          { required: true, message: '请输入用户名', trigger: 'blur' },
          { min: 5, message: '最低5位字符', trigger: 'blur' }
Mr.奇淼('s avatar
Mr.奇淼( 已提交
155
        ],
Mr.奇淼('s avatar
Mr.奇淼( 已提交
156
        password: [
何秀钢 已提交
157 158
          { required: true, message: '请输入用户密码', trigger: 'blur' },
          { min: 6, message: '最低6位字符', trigger: 'blur' }
Mr.奇淼('s avatar
Mr.奇淼( 已提交
159 160
        ],
        nickName: [
何秀钢 已提交
161
          { required: true, message: '请输入用户昵称', trigger: 'blur' }
Mr.奇淼('s avatar
Mr.奇淼( 已提交
162 163
        ],
        authorityId: [
何秀钢 已提交
164
          { required: true, message: '请选择用户角色', trigger: 'blur' }
Mr.奇淼('s avatar
Mr.奇淼( 已提交
165
        ]
Mr.奇淼('s avatar
Mr.奇淼( 已提交
166
      }
何秀钢 已提交
167
    }
168
  },
169
  computed: {
何秀钢 已提交
170 171
    ...mapGetters('user', ['token'])
  },
172 173 174 175 176
  watch: {
    tableData() {
      this.setAuthorityIds()
    }
  },
何秀钢 已提交
177
  async created() {
178
    await this.getTableData()
何秀钢 已提交
179 180
    const res = await getAuthorityList({ page: 1, pageSize: 999 })
    this.setOptions(res.data.list)
181
  },
182
  methods: {
Mr.奇淼('s avatar
Mr.奇淼( 已提交
183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208
    resetPassword(row) {
      this.$confirm(
        '是否将此用户密码重置为123456?',
        '警告',
        {
          confirmButtonText: '确定',
          cancelButtonText: '取消',
          type: 'warning',
        }
      ).then(async() => {
        const res = await resetPassword({
          ID: row.ID,
        })
        if (res.code === 0) {
          this.$message({
            type: 'success',
            message: res.msg,
          })
        } else {
          this.$message({
            type: 'error',
            message: res.msg,
          })
        }
      })
    },
209 210
    setAuthorityIds() {
      this.tableData && this.tableData.forEach((user) => {
Mr.奇淼('s avatar
Mr.奇淼( 已提交
211
        const authorityIds = user.authorities && user.authorities.map(i => {
212 213
          return i.authorityId
        })
214
        user.authorityIds = authorityIds
215 216
      })
    },
何秀钢 已提交
217
    openHeaderChange() {
218 219
      this.$refs.chooseImg.open()
    },
Mr.奇淼('s avatar
Mr.奇淼( 已提交
220
    setOptions(authData) {
何秀钢 已提交
221 222
      this.authOptions = []
      this.setAuthorityOptions(authData, this.authOptions)
223
    },
Mr.奇淼('s avatar
Mr.奇淼( 已提交
224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247
    openEidt(row) {
      if (this.tableData.some(item => item.editFlag)) {
        this.$message('当前存在正在编辑的用户')
        return
      }
      this.backNickName = row.nickName
      row.editFlag = true
    },
    async enterEdit(row) {
      const res = await setUserInfo({ nickName: row.nickName, ID: row.ID })
      if (res.code === 0) {
        this.$message({
          type: 'success',
          message: '设置成功'
        })
      }
      this.backNickName = ''
      row.editFlag = false
    },
    closeEdit(row) {
      row.nickName = this.backNickName
      this.backNickName = ''
      row.editFlag = false
    },
Mr.奇淼('s avatar
Mr.奇淼( 已提交
248 249
    setAuthorityOptions(AuthorityData, optionsData) {
      AuthorityData &&
Mr.奇淼('s avatar
Mr.奇淼( 已提交
250
        AuthorityData.forEach(item => {
251
          if (item.children && item.children.length) {
Mr.奇淼('s avatar
Mr.奇淼( 已提交
252 253 254 255
            const option = {
              authorityId: item.authorityId,
              authorityName: item.authorityName,
              children: []
何秀钢 已提交
256 257 258
            }
            this.setAuthorityOptions(item.children, option.children)
            optionsData.push(option)
Mr.奇淼('s avatar
Mr.奇淼( 已提交
259 260 261 262
          } else {
            const option = {
              authorityId: item.authorityId,
              authorityName: item.authorityName
何秀钢 已提交
263 264
            }
            optionsData.push(option)
265
          }
何秀钢 已提交
266
        })
267
    },
Mr.奇淼('s avatar
Mr.奇淼( 已提交
268
    async deleteUser(row) {
何秀钢 已提交
269 270
      const res = await deleteUser({ id: row.ID })
      if (res.code === 0) {
271 272
        this.$message.success('删除成功')
        await this.getTableData()
何秀钢 已提交
273
        row.visible = false
274
      }
275
    },
276
    async enterAddUserDialog() {
277
      this.userInfo.authorityId = this.userInfo.authorityIds[0]
Mr.奇淼('s avatar
Mr.奇淼( 已提交
278 279
      this.$refs.userForm.validate(async valid => {
        if (valid) {
何秀钢 已提交
280 281 282
          const res = await register(this.userInfo)
          if (res.code === 0) {
            this.$message({ type: 'success', message: '创建成功' })
Mr.奇淼('s avatar
Mr.奇淼( 已提交
283
          }
何秀钢 已提交
284 285
          await this.getTableData()
          this.closeAddUserDialog()
Mr.奇淼('s avatar
Mr.奇淼( 已提交
286
        }
何秀钢 已提交
287
      })
288 289
    },
    closeAddUserDialog() {
何秀钢 已提交
290
      this.$refs.userForm.resetFields()
291 292
      this.userInfo.headerImg = ''
      this.userInfo.authorityIds = []
何秀钢 已提交
293
      this.addUserDialog = false
294 295
    },
    addUser() {
何秀钢 已提交
296
      this.addUserDialog = true
297
    },
298 299 300
    async changeAuthority(row, flag) {
      if (flag) {
        return
301
      }
302 303 304 305 306 307 308 309 310
      this.$nextTick(async() => {
        const res = await setUserAuthorities({
          ID: row.ID,
          authorityIds: row.authorityIds
        })
        if (res.code === 0) {
          this.$message({ type: 'success', message: '角色设置成功' })
        }
      })
311
    },
312
  }
何秀钢 已提交
313
}
314
</script>
315

何秀钢 已提交
316
<style lang="scss">
317
.user-dialog {
318 319 320 321 322 323 324 325 326
  .header-img-box {
  width: 200px;
  height: 200px;
  border: 1px dashed #ccc;
  border-radius: 20px;
  text-align: center;
  line-height: 200px;
  cursor: pointer;
}
327 328 329 330 331 332 333 334 335 336 337 338 339 340 341 342 343 344 345
  .avatar-uploader .el-upload:hover {
    border-color: #409eff;
  }
  .avatar-uploader-icon {
    border: 1px dashed #d9d9d9 !important;
    border-radius: 6px;
    font-size: 28px;
    color: #8c939d;
    width: 178px;
    height: 178px;
    line-height: 178px;
    text-align: center;
  }
  .avatar {
    width: 178px;
    height: 178px;
    display: block;
  }
}
Mr.奇淼('s avatar
Mr.奇淼( 已提交
346 347 348 349 350 351 352 353 354 355
.nickName{
  display: flex;
  justify-content: flex-start;
  align-items: center;
}
.pointer{
  cursor: pointer;
  font-size: 16px;
  margin-left: 2px;
}
何秀钢 已提交
356
</style>