n20-common-lib 1.2.22 → 1.2.25

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,563 +1,563 @@
1
- <template>
2
- <!-- eslint-disable no-irregular-whitespace -->
3
- <div v-loading="loadingV" class="login-form-wrap p-a">
4
- <slot name="header"></slot>
5
- <svg class="cut-qrcode-h">
6
- <image
7
- v-if="theType === 'qrcode'"
8
- width="52"
9
- height="52"
10
- xlink:href="./qrt.png"
11
- />
12
- <image v-else width="52" height="52" xlink:href="./qr.png" />
13
- <path d="M0,0L52,0L52,52Z" @click="cutQrcode" />
14
- </svg>
15
- <!-- 账号/密码登录 -->
16
- <div v-if="theType === 'account'">
17
- <el-form
18
- ref="login-form"
19
- :model="form"
20
- :rules="rules"
21
- :hide-required-asterisk="true"
22
- @submit.native.prevent="submitLogin"
23
- >
24
- <el-form-item class="m-b" label="账号" prop="username">
25
- <el-input v-model="form.username" placeholder="请输入账号" />
26
- </el-form-item>
27
- <el-form-item class="m-b" label="密码" prop="password">
28
- <el-input
29
- v-model="form.password"
30
- placeholder="请输入密码"
31
- show-password
32
- />
33
- <!-- 不自动填充密码
34
- <input
35
- v-model="fakePwd"
36
- type="password"
37
- name="password"
38
- hidden="hidden"
39
- @change="fakePwd = 'a1b2c3d4'"
40
- />
41
- <el-input
42
- v-model="form.password"
43
- placeholder="请输入密码"
44
- :class="{ 'login-password': !showPwd }"
45
- >
46
- <i
47
- slot="suffix"
48
- class="el-input__icon el-icon-view"
49
- style="pointer-events: all"
50
- @click.stop="showPwd = !showPwd"
51
- ></i>
52
- </el-input>
53
- 不自动填充密码End -->
54
- </el-form-item>
55
- <el-button
56
- class="m-t"
57
- type="primary"
58
- native-type="submit"
59
- style="width: 100%"
60
- >登 录</el-button
61
- >
62
- </el-form>
63
- </div>
64
- <!-- 二维码扫码登录 -->
65
- <qrCode v-else-if="theType === 'qrcode'" @qrcode-login="QrcodeLogin" />
66
- <!-- 手机验证码登录 -->
67
- <div v-if="theType === 'pthon'">
68
- <el-form
69
- ref="login-form"
70
- :model="form"
71
- :rules="rules"
72
- :hide-required-asterisk="true"
73
- @submit.native.prevent="submitLogin"
74
- >
75
- <el-form-item class="m-b" label="手机号" prop="phone">
76
- <el-input v-model="form.phone" placeholder="请输入手机号" />
77
- </el-form-item>
78
- <el-form-item class="m-b" label="验证码" prop="code">
79
- <br />
80
- <div class="flex-box">
81
- <el-input
82
- v-model="form.code"
83
- class="flex-item m-r-s"
84
- placeholder="请输入验证码"
85
- />
86
- <el-button
87
- plain
88
- style="4em"
89
- :disabled="!form.phone || codeNum !== '验证码'"
90
- @click="sendMsg('phone', '手机号')"
91
- >{{ codeNum }}</el-button
92
- >
93
- </div>
94
- </el-form-item>
95
- <el-button
96
- class="m-t"
97
- type="primary"
98
- native-type="submit"
99
- style="width: 100%"
100
- >登 录</el-button
101
- >
102
- </el-form>
103
- </div>
104
- <!-- 账户密码验证码登录 -->
105
- <div v-if="theType === 'accountPthon'">
106
- <el-form
107
- ref="login-form"
108
- :model="form"
109
- :rules="rules"
110
- :hide-required-asterisk="true"
111
- @submit.native.prevent="submitLogin"
112
- >
113
- <el-form-item class="m-b" label="账号" prop="username">
114
- <el-input v-model="form.username" placeholder="请输入账号" />
115
- </el-form-item>
116
- <el-form-item class="m-b" label="密码" prop="password">
117
- <el-input
118
- v-model="form.password"
119
- placeholder="请输入密码"
120
- show-password
121
- />
122
- </el-form-item>
123
- <el-form-item class="m-b" label="验证码" prop="code">
124
- <br />
125
- <div class="flex-box">
126
- <el-input
127
- v-model="form.code"
128
- class="flex-item m-r-s"
129
- placeholder="请输入验证码"
130
- />
131
- <el-button
132
- plain
133
- style="4em"
134
- :disabled="!form.username || codeNum !== '验证码'"
135
- @click="sendMsg('username', '账户')"
136
- >{{ codeNum }}</el-button
137
- >
138
- </div>
139
- </el-form-item>
140
- <el-button
141
- class="m-t"
142
- type="primary"
143
- native-type="submit"
144
- style="width: 100%"
145
- >登 录</el-button
146
- >
147
- </el-form>
148
- </div>
149
- <!-- -->
150
- <div v-if="theType !== 'qrcode'" class="flex-box flex-lr m-t-s m-b-s">
151
- <el-dropdown
152
- size="mini"
153
- placement="bottom-start"
154
- @command="changeLoginType"
155
- >
156
- <span>其他登录方式</span>
157
- <el-dropdown-menu slot="dropdown">
158
- <template v-for="item in loginTypes">
159
- <el-dropdown-item
160
- v-if="item !== theType"
161
- :key="item"
162
- :command="item"
163
- >{{ loginTypeMap[item] }}</el-dropdown-item
164
- >
165
- </template>
166
- </el-dropdown-menu>
167
- </el-dropdown>
168
-
169
- <el-link :underline="false" @click="forgetPw">忘记密码</el-link>
170
- </div>
171
- <slot name="footer"></slot>
172
- </div>
173
- </template>
174
-
175
- <script>
176
- import decode from 'jsonwebtoken/decode'
177
- import axios from '../../utils/axios'
178
- import auth from '../../utils/auth'
179
- import qrCode from './qrcode.vue'
180
- import { siteTree2menus, siteTree2RelaNos } from './utils'
181
- import retrievePw from './retrievePw.vue'
182
-
183
- function encode(pwd = '', key = 0) {
184
- if (pwd === '') return ''
185
-
186
- let pwdArr = pwd.trim().split('')
187
- let resultArr = pwdArr.map((oPwd) =>
188
- String.fromCharCode(oPwd.charCodeAt() ^ key)
189
- )
190
- let result = resultArr.join('')
191
-
192
- return window.btoa(window.unescape(window.encodeURIComponent(result)))
193
- }
194
-
195
- export default {
196
- name: 'LoginForm',
197
- components: {
198
- qrCode,
199
- retrievePw
200
- },
201
- props: {
202
- loginTypes: {
203
- type: Array,
204
- default: () => ['account']
205
- },
206
- loginThen: {
207
- type: Function,
208
- default: undefined
209
- }
210
- },
211
- data() {
212
- return {
213
- showPwd: false,
214
- fakePwd: 'a1b2c3d4',
215
- loadingV: false,
216
- loginTypeMap: {
217
- account: '账号密码登录',
218
- pthon: '手机验证码登录',
219
- qrcode: '二维码扫码登录',
220
- accountPthon: '账户密码验证码登录'
221
- },
222
- theType: this.loginTypes[0],
223
- userNo: undefined,
224
- form: {
225
- username: '',
226
- password: '',
227
- code: ''
228
- },
229
- pwd: '',
230
- rules: {
231
- username: [{ required: true, message: '请输入账号!', trigger: 'blur' }],
232
- password: [{ required: true, message: '请输入密码!', trigger: 'blur' }],
233
- code: [{ required: true, message: '请输入验证码!', trigger: 'blur' }],
234
- phone: [
235
- { required: true, message: '请输入手机号!', trigger: 'blur' },
236
- {
237
- pattern: /^1\d{10}$/,
238
- message: '请输入正确手机号!',
239
- trigger: 'blur'
240
- }
241
- ]
242
- },
243
- fPwV: false,
244
- codeNum: '验证码',
245
- getAzCount: 0
246
- }
247
- },
248
- methods: {
249
- // 忘记密码
250
- forgetPw() {
251
- this.$emit('changType', false)
252
- },
253
- // 切换登录方式
254
- changeLoginType(val) {
255
- this.theType = val
256
- },
257
- // 切换二维码小图标
258
- cutQrcode() {
259
- if (this.theType !== 'qrcode') {
260
- this.theType = 'qrcode'
261
- } else {
262
- this.theType = this.loginTypes[0]
263
- }
264
- },
265
- // 发送短信
266
- outTime(num) {
267
- clearTimeout(this.setTimeNum)
268
- if (num > 0) {
269
- num--
270
- this.codeNum = num + 'S'
271
-
272
- this.setTimeNum = setTimeout(() => {
273
- this.outTime(num)
274
- }, 1000)
275
- num--
276
- } else {
277
- clearTimeout(this.setTimeNum)
278
- this.codeNum = '验证码'
279
- }
280
- },
281
- sendMsg(type = 'phone', typeName = '手机号') {
282
- this.$refs['login-form'].validateField(type, (valid) => {
283
- if (!valid) {
284
- this.outTime(60)
285
-
286
- axios
287
- .get(
288
- `/bems/prod_1.0/uas/api/authorization/captcha/${this.form[type]}/1`,
289
- null,
290
- { loading: false, noMsg: true }
291
- )
292
- .then(() => {
293
- this.$message.success(
294
- `${typeName}: ${this.form[type]} 验证码发送成功`
295
- )
296
- })
297
- .catch(() => {
298
- clearTimeout(this.setTimeNum)
299
- this.codeNum = '验证码'
300
- this.$message.warning(
301
- `${typeName}: ${this.form[type]} 验证码发送失败`
302
- )
303
- })
304
- }
305
- })
306
- },
307
- loadingFn() {
308
- this.loadingV = false
309
- },
310
- // 登录
311
- submitLogin() {
312
- this.$refs['login-form'].validate((valid) => {
313
- if (!valid) return false
314
-
315
- auth.removeToken()
316
- this.theType === 'account' && this.passwordLogin().catch(this.loadingFn)
317
- this.theType === 'pthon' && this.phoneLogin().catch(this.loadingFn)
318
- this.theType === 'accountPthon' &&
319
- this.accountPthonLogin().catch(this.loadingFn)
320
- })
321
- },
322
- // 获取加密参数
323
- authorizationCode(username, password, code) {
324
- this.getAzCount = this.getAzCount + 1
325
-
326
- let pwdArr = code ? [username, password, code] : [username, password]
327
- let pwd = pwdArr.join('@@')
328
-
329
- return new Promise((resolve, reject) => {
330
- if (this.getAzCount <= 3) {
331
- axios
332
- .get(
333
- '/bems/prod_1.0/uas/api/authorization/authorizationCode',
334
- {},
335
- { loading: false }
336
- )
337
- .then(({ data = {} }) => {
338
- try {
339
- pwd = encode(pwd, data.authorizationValue)
340
- this.pwd = pwd + '@@' + data.authorizationKey
341
-
342
- this.getAzCount = 0
343
- resolve()
344
- } catch (error) {
345
- console.error('密码加密错误:' + JSON.stringify(data))
346
- this.authorizationCode(username, password, code) // 再次请求
347
- .then(resolve)
348
- .catch(reject)
349
- }
350
- })
351
- } else {
352
- // 当重新请求大于3次,抛出错误
353
- this.$message.warning(`登录失败,请重新登录!`)
354
- this.getAzCount = 0
355
- reject()
356
- }
357
- })
358
- },
359
- // 账号/密码登录
360
- async passwordLogin() {
361
- this.loadingV = true
362
-
363
- let { username, password, code } = this.form
364
- await this.authorizationCode(username, password, code)
365
- // 请求该账号要不要强制验证码登录
366
- let cL = await axios.post(
367
- '/bems/prod_1.0/uas/api/authorization/checkLogin',
368
- { grantType: 'password', username, password: this.pwd, code },
369
- {
370
- loading: false
371
- }
372
- )
373
-
374
- // 输入验证码
375
- if (cL.data === true) {
376
- this.loadingV = false
377
- this.theType = 'accountPthon'
378
-
379
- this.sendMsg('username', '账户')
380
- this.outTime(60)
381
- return
382
- }
383
- // 签名验签
384
- else if (cL.data === false) {
385
- // 获取线上签名uDn
386
- let uDn = await axios.get(
387
- `/bems/prod_1.0/dssc/sign/getUserDn/${username}/password`,
388
- null,
389
- { loading: false }
390
- )
391
- let { checkCert, userDn } = uDn.data
392
- // 走本地签名
393
- let sign = undefined
394
- let certDn = undefined
395
- if (checkCert) {
396
- let { getSign } = await import('../../plugins/Sign/index')
397
- sign = getSign(username, userDn)
398
- certDn = userDn
399
- }
400
-
401
- await this.authorizationCode(username, password, code) // authorizationValue 使用一次后就失效了
402
- axios
403
- .post(
404
- '/bems/prod_1.0/uas/api/authorization/login',
405
- {
406
- grantType: 'password',
407
- username,
408
- password: this.pwd,
409
- code,
410
- sign,
411
- certDn
412
- },
413
- {
414
- loading: this.loadingFn
415
- }
416
- )
417
- .then((res) => {
418
- this.loginAfter(res.data)
419
- })
420
- return
421
- }
422
-
423
- // 正常登录
424
- await this.authorizationCode(username, password, code) // authorizationValue 使用一次后就失效了
425
- axios
426
- .post(
427
- '/bems/prod_1.0/uas/api/authorization/login',
428
- { grantType: 'password', username, password: this.pwd, code },
429
- { loading: this.loadingFn }
430
- )
431
- .then((res) => {
432
- this.loginAfter(res.data)
433
- })
434
- },
435
- // 二维码扫码登录
436
- QrcodeLogin(userNo, qrcode) {
437
- axios
438
- .post(
439
- '/bems/prod_1.0/uas/api/authorization/login',
440
- {
441
- grantType: 'qr_code',
442
- type: 'qr_code',
443
- username: userNo,
444
- password: qrcode
445
- },
446
- { loading: false }
447
- )
448
- .then((res) => {
449
- this.loginAfter(res.data)
450
- })
451
- },
452
- // 手机号/验证码登录
453
- async phoneLogin() {
454
- this.loadingV = true
455
- let { phone, code } = this.form
456
- await this.authorizationCode(phone, code)
457
- let pwd = this.pwd
458
-
459
- axios
460
- .post(
461
- '/bems/prod_1.0/uas/api/authorization/login',
462
- { grantType: 'mobile', username: phone, password: pwd },
463
- { loading: this.loadingFn }
464
- )
465
- .then((res) => {
466
- this.loginAfter(res.data)
467
- })
468
- },
469
- async accountPthonLogin() {
470
- this.loadingV = true
471
- let { username, password, code } = this.form
472
- await this.authorizationCode(username, password, code)
473
- let pwd = this.pwd
474
-
475
- axios
476
- .post(
477
- '/bems/prod_1.0/uas/api/authorization/login',
478
- { grantType: 'user_password_code', username, password: pwd, code },
479
- { loading: this.loadingFn }
480
- )
481
- .then((res) => {
482
- this.loginAfter(res.data)
483
- })
484
- },
485
- // 登录成功后
486
- loginAfter(data) {
487
- auth.setToken(data.accessToken)
488
- try {
489
- let tokenObj = decode(data.accessToken)
490
- let userInfo = JSON.parse(tokenObj.user_name)
491
- this.userNo = userInfo.userNo
492
- } catch (error) {
493
- console.error('accessToken解析错误:' + data.accessToken)
494
- }
495
-
496
- Promise.all([
497
- this.getUserInfo(),
498
- this.getMicroAppsInfo(),
499
- this.getCltRelaNos()
500
- ]).then(() => {
501
- if (this.loginThen) {
502
- this.loginThen().then(() => {
503
- this.$router.replace('/')
504
- })
505
- } else {
506
- this.$router.replace('/')
507
- }
508
- })
509
- },
510
- // 获取用户基本信息
511
- async getUserInfo() {
512
- let username = this.userNo
513
- let { data } = await axios.get(
514
- // '/bems/prod_1.0/user/api/userController/userCltAdDep/' + username
515
- '/bems/prod_1.0/user/api/userController/userCltAdRole/' + username
516
- )
517
- let userInfo = data || { uname: username }
518
-
519
- sessionStorage.setItem('userNo', username)
520
- sessionStorage.setItem('userInfo', JSON.stringify(userInfo))
521
- sessionStorage.setItem('loginTime', Date.now())
522
- },
523
- // 获取子应用相关列表
524
- async getMicroAppsInfo() {
525
- this.loadingV = true
526
- let { data } = await axios.get('/bems/1.0/menuTree/tree', null, {
527
- loading: this.loadingFn
528
- })
529
- data || (data = [])
530
-
531
- let layoutMenus = siteTree2menus(data)
532
- sessionStorage.setItem('menuTree', JSON.stringify(layoutMenus))
533
-
534
- let relaNos = siteTree2RelaNos(data)
535
- sessionStorage.setItem('relaNos', JSON.stringify(relaNos))
536
-
537
- // XXX: 兼容旧版resources
538
- sessionStorage.setItem('resources', JSON.stringify(data))
539
- },
540
- // 新增不同单位对应不同角色,具有不同权限
541
- getCltRelaNos() {
542
- let pro = new Promise((resolve) => {
543
- axios
544
- .post(
545
- '/bems/prod_1.0/user/api/userController/qryBusyInfoNo',
546
- { uno: this.userNo },
547
- { loading: false, noMsg: true }
548
- )
549
- .then(({ data: cltrelaMap = [] }) => {
550
- let cltrelaNos = {}
551
- cltrelaMap.forEach((item) => {
552
- cltrelaNos[item.cltNo] = item.relaNos
553
- })
554
- sessionStorage.setItem('cltrelaNos', JSON.stringify(cltrelaNos))
555
- })
556
- .finally(resolve)
557
- })
558
-
559
- return pro
560
- }
561
- }
562
- }
563
- </script>
1
+ <template>
2
+ <!-- eslint-disable no-irregular-whitespace -->
3
+ <div v-loading="loadingV" class="login-form-wrap p-a">
4
+ <slot name="header"></slot>
5
+ <svg class="cut-qrcode-h">
6
+ <image
7
+ v-if="theType === 'qrcode'"
8
+ width="52"
9
+ height="52"
10
+ xlink:href="./qrt.png"
11
+ />
12
+ <image v-else width="52" height="52" xlink:href="./qr.png" />
13
+ <path d="M0,0L52,0L52,52Z" @click="cutQrcode" />
14
+ </svg>
15
+ <!-- 账号/密码登录 -->
16
+ <div v-if="theType === 'account'">
17
+ <el-form
18
+ ref="login-form"
19
+ :model="form"
20
+ :rules="rules"
21
+ :hide-required-asterisk="true"
22
+ @submit.native.prevent="submitLogin"
23
+ >
24
+ <el-form-item class="m-b" label="账号" prop="username">
25
+ <el-input v-model="form.username" placeholder="请输入账号" />
26
+ </el-form-item>
27
+ <el-form-item class="m-b" label="密码" prop="password">
28
+ <el-input
29
+ v-model="form.password"
30
+ placeholder="请输入密码"
31
+ show-password
32
+ />
33
+ <!-- 不自动填充密码
34
+ <input
35
+ v-model="fakePwd"
36
+ type="password"
37
+ name="password"
38
+ hidden="hidden"
39
+ @change="fakePwd = 'a1b2c3d4'"
40
+ />
41
+ <el-input
42
+ v-model="form.password"
43
+ placeholder="请输入密码"
44
+ :class="{ 'login-password': !showPwd }"
45
+ >
46
+ <i
47
+ slot="suffix"
48
+ class="el-input__icon el-icon-view"
49
+ style="pointer-events: all"
50
+ @click.stop="showPwd = !showPwd"
51
+ ></i>
52
+ </el-input>
53
+ 不自动填充密码End -->
54
+ </el-form-item>
55
+ <el-button
56
+ class="m-t"
57
+ type="primary"
58
+ native-type="submit"
59
+ style="width: 100%"
60
+ >登 录</el-button
61
+ >
62
+ </el-form>
63
+ </div>
64
+ <!-- 二维码扫码登录 -->
65
+ <qrCode v-else-if="theType === 'qrcode'" @qrcode-login="QrcodeLogin" />
66
+ <!-- 手机验证码登录 -->
67
+ <div v-if="theType === 'pthon'">
68
+ <el-form
69
+ ref="login-form"
70
+ :model="form"
71
+ :rules="rules"
72
+ :hide-required-asterisk="true"
73
+ @submit.native.prevent="submitLogin"
74
+ >
75
+ <el-form-item class="m-b" label="手机号" prop="phone">
76
+ <el-input v-model="form.phone" placeholder="请输入手机号" />
77
+ </el-form-item>
78
+ <el-form-item class="m-b" label="验证码" prop="code">
79
+ <br />
80
+ <div class="flex-box">
81
+ <el-input
82
+ v-model="form.code"
83
+ class="flex-item m-r-s"
84
+ placeholder="请输入验证码"
85
+ />
86
+ <el-button
87
+ plain
88
+ style="4em"
89
+ :disabled="!form.phone || codeNum !== '验证码'"
90
+ @click="sendMsg('phone', '手机号')"
91
+ >{{ codeNum }}</el-button
92
+ >
93
+ </div>
94
+ </el-form-item>
95
+ <el-button
96
+ class="m-t"
97
+ type="primary"
98
+ native-type="submit"
99
+ style="width: 100%"
100
+ >登 录</el-button
101
+ >
102
+ </el-form>
103
+ </div>
104
+ <!-- 账户密码验证码登录 -->
105
+ <div v-if="theType === 'accountPthon'">
106
+ <el-form
107
+ ref="login-form"
108
+ :model="form"
109
+ :rules="rules"
110
+ :hide-required-asterisk="true"
111
+ @submit.native.prevent="submitLogin"
112
+ >
113
+ <el-form-item class="m-b" label="账号" prop="username">
114
+ <el-input v-model="form.username" placeholder="请输入账号" />
115
+ </el-form-item>
116
+ <el-form-item class="m-b" label="密码" prop="password">
117
+ <el-input
118
+ v-model="form.password"
119
+ placeholder="请输入密码"
120
+ show-password
121
+ />
122
+ </el-form-item>
123
+ <el-form-item class="m-b" label="验证码" prop="code">
124
+ <br />
125
+ <div class="flex-box">
126
+ <el-input
127
+ v-model="form.code"
128
+ class="flex-item m-r-s"
129
+ placeholder="请输入验证码"
130
+ />
131
+ <el-button
132
+ plain
133
+ style="4em"
134
+ :disabled="!form.username || codeNum !== '验证码'"
135
+ @click="sendMsg('username', '账户')"
136
+ >{{ codeNum }}</el-button
137
+ >
138
+ </div>
139
+ </el-form-item>
140
+ <el-button
141
+ class="m-t"
142
+ type="primary"
143
+ native-type="submit"
144
+ style="width: 100%"
145
+ >登 录</el-button
146
+ >
147
+ </el-form>
148
+ </div>
149
+ <!-- -->
150
+ <div v-if="theType !== 'qrcode'" class="flex-box flex-lr m-t-s m-b-s">
151
+ <el-dropdown
152
+ size="mini"
153
+ placement="bottom-start"
154
+ @command="changeLoginType"
155
+ >
156
+ <span>其他登录方式</span>
157
+ <el-dropdown-menu slot="dropdown">
158
+ <template v-for="item in loginTypes">
159
+ <el-dropdown-item
160
+ v-if="item !== theType"
161
+ :key="item"
162
+ :command="item"
163
+ >{{ loginTypeMap[item] }}</el-dropdown-item
164
+ >
165
+ </template>
166
+ </el-dropdown-menu>
167
+ </el-dropdown>
168
+
169
+ <el-link :underline="false" @click="forgetPw">忘记密码</el-link>
170
+ </div>
171
+ <slot name="footer"></slot>
172
+ </div>
173
+ </template>
174
+
175
+ <script>
176
+ import decode from 'jsonwebtoken/decode'
177
+ import axios from '../../utils/axios'
178
+ import auth from '../../utils/auth'
179
+ import qrCode from './qrcode.vue'
180
+ import { siteTree2menus, siteTree2RelaNos } from './utils'
181
+ import retrievePw from './retrievePw.vue'
182
+
183
+ function encode(pwd = '', key = 0) {
184
+ if (pwd === '') return ''
185
+
186
+ let pwdArr = pwd.trim().split('')
187
+ let resultArr = pwdArr.map((oPwd) =>
188
+ String.fromCharCode(oPwd.charCodeAt() ^ key)
189
+ )
190
+ let result = resultArr.join('')
191
+
192
+ return window.btoa(window.unescape(window.encodeURIComponent(result)))
193
+ }
194
+
195
+ export default {
196
+ name: 'LoginForm',
197
+ components: {
198
+ qrCode,
199
+ retrievePw
200
+ },
201
+ props: {
202
+ loginTypes: {
203
+ type: Array,
204
+ default: () => ['account']
205
+ },
206
+ loginThen: {
207
+ type: Function,
208
+ default: undefined
209
+ }
210
+ },
211
+ data() {
212
+ return {
213
+ showPwd: false,
214
+ fakePwd: 'a1b2c3d4',
215
+ loadingV: false,
216
+ loginTypeMap: {
217
+ account: '账号密码登录',
218
+ pthon: '手机验证码登录',
219
+ qrcode: '二维码扫码登录',
220
+ accountPthon: '账户密码验证码登录'
221
+ },
222
+ theType: this.loginTypes[0],
223
+ userNo: undefined,
224
+ form: {
225
+ username: '',
226
+ password: '',
227
+ code: ''
228
+ },
229
+ pwd: '',
230
+ rules: {
231
+ username: [{ required: true, message: '请输入账号!', trigger: 'blur' }],
232
+ password: [{ required: true, message: '请输入密码!', trigger: 'blur' }],
233
+ code: [{ required: true, message: '请输入验证码!', trigger: 'blur' }],
234
+ phone: [
235
+ { required: true, message: '请输入手机号!', trigger: 'blur' },
236
+ {
237
+ pattern: /^1\d{10}$/,
238
+ message: '请输入正确手机号!',
239
+ trigger: 'blur'
240
+ }
241
+ ]
242
+ },
243
+ fPwV: false,
244
+ codeNum: '验证码',
245
+ getAzCount: 0
246
+ }
247
+ },
248
+ methods: {
249
+ // 忘记密码
250
+ forgetPw() {
251
+ this.$emit('changType', false)
252
+ },
253
+ // 切换登录方式
254
+ changeLoginType(val) {
255
+ this.theType = val
256
+ },
257
+ // 切换二维码小图标
258
+ cutQrcode() {
259
+ if (this.theType !== 'qrcode') {
260
+ this.theType = 'qrcode'
261
+ } else {
262
+ this.theType = this.loginTypes[0]
263
+ }
264
+ },
265
+ // 发送短信
266
+ outTime(num) {
267
+ clearTimeout(this.setTimeNum)
268
+ if (num > 0) {
269
+ num--
270
+ this.codeNum = num + 'S'
271
+
272
+ this.setTimeNum = setTimeout(() => {
273
+ this.outTime(num)
274
+ }, 1000)
275
+ num--
276
+ } else {
277
+ clearTimeout(this.setTimeNum)
278
+ this.codeNum = '验证码'
279
+ }
280
+ },
281
+ sendMsg(type = 'phone', typeName = '手机号') {
282
+ this.$refs['login-form'].validateField(type, (valid) => {
283
+ if (!valid) {
284
+ this.outTime(60)
285
+
286
+ axios
287
+ .get(
288
+ `/bems/prod_1.0/uas/api/authorization/captcha/${this.form[type]}/1`,
289
+ null,
290
+ { loading: false, noMsg: true }
291
+ )
292
+ .then(() => {
293
+ this.$message.success(
294
+ `${typeName}: ${this.form[type]} 验证码发送成功`
295
+ )
296
+ })
297
+ .catch(() => {
298
+ clearTimeout(this.setTimeNum)
299
+ this.codeNum = '验证码'
300
+ this.$message.warning(
301
+ `${typeName}: ${this.form[type]} 验证码发送失败`
302
+ )
303
+ })
304
+ }
305
+ })
306
+ },
307
+ loadingFn() {
308
+ this.loadingV = false
309
+ },
310
+ // 登录
311
+ submitLogin() {
312
+ this.$refs['login-form'].validate((valid) => {
313
+ if (!valid) return false
314
+
315
+ auth.removeToken()
316
+ this.theType === 'account' && this.passwordLogin().catch(this.loadingFn)
317
+ this.theType === 'pthon' && this.phoneLogin().catch(this.loadingFn)
318
+ this.theType === 'accountPthon' &&
319
+ this.accountPthonLogin().catch(this.loadingFn)
320
+ })
321
+ },
322
+ // 获取加密参数
323
+ authorizationCode(username, password, code) {
324
+ this.getAzCount = this.getAzCount + 1
325
+
326
+ let pwdArr = code ? [username, password, code] : [username, password]
327
+ let pwd = pwdArr.join('@@')
328
+
329
+ return new Promise((resolve, reject) => {
330
+ if (this.getAzCount <= 3) {
331
+ axios
332
+ .get(
333
+ '/bems/prod_1.0/uas/api/authorization/authorizationCode',
334
+ {},
335
+ { loading: false }
336
+ )
337
+ .then(({ data = {} }) => {
338
+ try {
339
+ pwd = encode(pwd, data.authorizationValue)
340
+ this.pwd = pwd + '@@' + data.authorizationKey
341
+
342
+ this.getAzCount = 0
343
+ resolve()
344
+ } catch (error) {
345
+ console.error('密码加密错误:' + JSON.stringify(data))
346
+ this.authorizationCode(username, password, code) // 再次请求
347
+ .then(resolve)
348
+ .catch(reject)
349
+ }
350
+ })
351
+ } else {
352
+ // 当重新请求大于3次,抛出错误
353
+ this.$message.warning(`登录失败,请重新登录!`)
354
+ this.getAzCount = 0
355
+ reject()
356
+ }
357
+ })
358
+ },
359
+ // 账号/密码登录
360
+ async passwordLogin() {
361
+ this.loadingV = true
362
+
363
+ let { username, password, code } = this.form
364
+ await this.authorizationCode(username, password, code)
365
+ // 请求该账号要不要强制验证码登录
366
+ let cL = await axios.post(
367
+ '/bems/prod_1.0/uas/api/authorization/checkLogin',
368
+ { grantType: 'password', username, password: this.pwd, code },
369
+ {
370
+ loading: false
371
+ }
372
+ )
373
+
374
+ // 输入验证码
375
+ if (cL.data === true) {
376
+ this.loadingV = false
377
+ this.theType = 'accountPthon'
378
+
379
+ this.sendMsg('username', '账户')
380
+ this.outTime(60)
381
+ return
382
+ }
383
+ // 签名验签
384
+ else if (cL.data === false) {
385
+ // 获取线上签名uDn
386
+ let uDn = await axios.get(
387
+ `/bems/prod_1.0/dssc/sign/getUserDn/${username}/password`,
388
+ null,
389
+ { loading: false }
390
+ )
391
+ let { checkCert, userDn } = uDn.data
392
+ // 走本地签名
393
+ let sign = undefined
394
+ let certDn = undefined
395
+ if (checkCert) {
396
+ let { getSign } = await import('../../plugins/Sign/index')
397
+ sign = getSign(username, userDn)
398
+ certDn = userDn
399
+ }
400
+
401
+ await this.authorizationCode(username, password, code) // authorizationValue 使用一次后就失效了
402
+ axios
403
+ .post(
404
+ '/bems/prod_1.0/uas/api/authorization/login',
405
+ {
406
+ grantType: 'password',
407
+ username,
408
+ password: this.pwd,
409
+ code,
410
+ sign,
411
+ certDn
412
+ },
413
+ {
414
+ loading: this.loadingFn
415
+ }
416
+ )
417
+ .then((res) => {
418
+ this.loginAfter(res.data)
419
+ })
420
+ return
421
+ }
422
+
423
+ // 正常登录
424
+ await this.authorizationCode(username, password, code) // authorizationValue 使用一次后就失效了
425
+ axios
426
+ .post(
427
+ '/bems/prod_1.0/uas/api/authorization/login',
428
+ { grantType: 'password', username, password: this.pwd, code },
429
+ { loading: this.loadingFn }
430
+ )
431
+ .then((res) => {
432
+ this.loginAfter(res.data)
433
+ })
434
+ },
435
+ // 二维码扫码登录
436
+ QrcodeLogin(userNo, qrcode) {
437
+ axios
438
+ .post(
439
+ '/bems/prod_1.0/uas/api/authorization/login',
440
+ {
441
+ grantType: 'qr_code',
442
+ type: 'qr_code',
443
+ username: userNo,
444
+ password: qrcode
445
+ },
446
+ { loading: false }
447
+ )
448
+ .then((res) => {
449
+ this.loginAfter(res.data)
450
+ })
451
+ },
452
+ // 手机号/验证码登录
453
+ async phoneLogin() {
454
+ this.loadingV = true
455
+ let { phone, code } = this.form
456
+ await this.authorizationCode(phone, code)
457
+ let pwd = this.pwd
458
+
459
+ axios
460
+ .post(
461
+ '/bems/prod_1.0/uas/api/authorization/login',
462
+ { grantType: 'mobile', username: phone, password: pwd },
463
+ { loading: this.loadingFn }
464
+ )
465
+ .then((res) => {
466
+ this.loginAfter(res.data)
467
+ })
468
+ },
469
+ async accountPthonLogin() {
470
+ this.loadingV = true
471
+ let { username, password, code } = this.form
472
+ await this.authorizationCode(username, password, code)
473
+ let pwd = this.pwd
474
+
475
+ axios
476
+ .post(
477
+ '/bems/prod_1.0/uas/api/authorization/login',
478
+ { grantType: 'user_password_code', username, password: pwd, code },
479
+ { loading: this.loadingFn }
480
+ )
481
+ .then((res) => {
482
+ this.loginAfter(res.data)
483
+ })
484
+ },
485
+ // 登录成功后
486
+ loginAfter(data) {
487
+ auth.setToken(data.accessToken)
488
+ try {
489
+ let tokenObj = decode(data.accessToken)
490
+ let userInfo = JSON.parse(tokenObj.user_name)
491
+ this.userNo = userInfo.userNo
492
+ } catch (error) {
493
+ console.error('accessToken解析错误:' + data.accessToken)
494
+ }
495
+
496
+ Promise.all([
497
+ this.getUserInfo(),
498
+ this.getMicroAppsInfo(),
499
+ this.getCltRelaNos()
500
+ ]).then(() => {
501
+ if (this.loginThen) {
502
+ this.loginThen().then(() => {
503
+ this.$router.replace('/')
504
+ })
505
+ } else {
506
+ this.$router.replace('/')
507
+ }
508
+ })
509
+ },
510
+ // 获取用户基本信息
511
+ async getUserInfo() {
512
+ let username = this.userNo
513
+ let { data } = await axios.get(
514
+ // '/bems/prod_1.0/user/api/userController/userCltAdDep/' + username
515
+ '/bems/prod_1.0/user/api/userController/userCltAdRole/' + username
516
+ )
517
+ let userInfo = data || { uname: username }
518
+
519
+ sessionStorage.setItem('userNo', username)
520
+ sessionStorage.setItem('userInfo', JSON.stringify(userInfo))
521
+ sessionStorage.setItem('loginTime', Date.now())
522
+ },
523
+ // 获取子应用相关列表
524
+ async getMicroAppsInfo() {
525
+ this.loadingV = true
526
+ let { data } = await axios.get('/bems/1.0/menuTree/tree', null, {
527
+ loading: this.loadingFn
528
+ })
529
+ data || (data = [])
530
+
531
+ let layoutMenus = siteTree2menus(data)
532
+ sessionStorage.setItem('menuTree', JSON.stringify(layoutMenus))
533
+
534
+ let relaNos = siteTree2RelaNos(data)
535
+ sessionStorage.setItem('relaNos', JSON.stringify(relaNos))
536
+
537
+ // XXX: 兼容旧版resources
538
+ sessionStorage.setItem('resources', JSON.stringify(data))
539
+ },
540
+ // 新增不同单位对应不同角色,具有不同权限
541
+ getCltRelaNos() {
542
+ let pro = new Promise((resolve) => {
543
+ axios
544
+ .post(
545
+ '/bems/prod_1.0/user/api/userController/qryBusyInfoNo',
546
+ { uno: this.userNo },
547
+ { loading: false, noMsg: true }
548
+ )
549
+ .then(({ data: cltrelaMap = [] }) => {
550
+ let cltrelaNos = {}
551
+ cltrelaMap.forEach((item) => {
552
+ cltrelaNos[item.cltNo] = item.relaNos
553
+ })
554
+ sessionStorage.setItem('cltrelaNos', JSON.stringify(cltrelaNos))
555
+ })
556
+ .finally(resolve)
557
+ })
558
+
559
+ return pro
560
+ }
561
+ }
562
+ }
563
+ </script>