maxkb/ui/src/views/authentication/component/SCAN.vue
wxg0103 be0b407cf1 refactor: 钉钉扫码认证设置优化
--bug=1048604 --user=王孝刚 【系统管理】钉钉扫码认证设置优化 https://www.tapd.cn/57709429/s/1622777
2024-12-02 18:21:51 +08:00

263 lines
7.2 KiB
Vue

<template>
<div v-loading="loading">
<el-card shadow="hover" class="border-none cursor" style="--el-card-padding: 16px 24px">
<div v-for="item in platforms" :key="item.key" class="mb-16">
<el-card class="mb-16" shadow="hover" style="--el-card-padding: 16px">
<div class="flex-between">
<div class="flex align-center ml-8 mr-8">
<img :src="item.logoSrc" alt="" class="icon" />
<h5 style="margin-left: 8px">{{ item.name }}</h5>
<el-tag v-if="item.isValid" type="success" class="ml-4">有效</el-tag>
</div>
<div>
<el-button type="primary" v-if="!item.isValid" @click="showDialog(item)"
>接入</el-button
>
<span v-if="item.isValid">
<span class="mr-4">{{ item.isActive ? '已开启' : '未开启' }}</span>
<el-switch
size="small"
v-model="item.isActive"
:disabled="!item.isValid"
@change="changeStatus(item)"
/>
</span>
</div>
</div>
<el-collapse-transition>
<div v-if="item.isValid" class="border-t mt-16">
<el-row :gutter="12" class="mt-16">
<el-col v-for="(value, key) in item.config" :key="key" :span="12">
<el-text type="info">{{ formatFieldName(key, item) }}</el-text>
<div class="mt-4 mb-16 flex align-center">
<span
v-if="key !== 'app_secret'"
class="vertical-middle lighter break-all ellipsis-1"
>{{ value }}</span
>
<span
v-if="key === 'app_secret' && !showPassword[item.key]?.[key]"
class="vertical-middle lighter break-all ellipsis-1"
>************</span
>
<span
v-if="key === 'app_secret' && showPassword[item.key]?.[key]"
class="vertical-middle lighter break-all ellipsis-1"
>{{ value }}</span
>
<el-button type="primary" text @click="() => copyClick(value)">
<AppIcon iconName="app-copy" />
</el-button>
<el-button
v-if="key === 'app_secret'"
type="primary"
text
@click="toggleShowPassword(item.key)"
>
<el-icon v-if="key === 'app_secret' && !showPassword[item.key]?.[key]">
<Hide />
</el-icon>
<el-icon v-if="key === 'app_secret' && showPassword[item.key]?.[key]">
<View />
</el-icon>
</el-button>
</div>
</el-col>
</el-row>
<el-button type="primary" @click="showDialog(item)">编辑</el-button>
<el-button @click="validateConnection(item)">校验</el-button>
</div>
</el-collapse-transition>
</el-card>
</div>
<EditModel ref="EditModelRef" @refresh="refresh" />
</el-card>
</div>
</template>
<script setup lang="ts">
import { reactive, ref, onMounted } from 'vue'
import { copyClick } from '@/utils/clipboard'
import EditModel from './EditModal.vue'
import platformApi from '@/api/platform-source'
import { MsgError, MsgSuccess } from '@/utils/message'
interface PlatformConfig {
[key: string]: string
}
interface Platform {
key: string
logoSrc: string
name: string
isActive: boolean
isValid: boolean
config: PlatformConfig
}
const EditModelRef = ref()
const loading = ref(false)
const platforms = reactive<Platform[]>(initializePlatforms())
const showPassword = reactive<{ [platformKey: string]: { [key: string]: boolean } }>({})
onMounted(() => {
getPlatformInfo()
})
function initializePlatforms(): Platform[] {
return [
createPlatform('wecom', '企业微信'),
createPlatform('dingtalk', '钉钉'),
createPlatform('lark', '飞书')
]
}
function createPlatform(key: string, name: string): Platform {
let logo = ''
switch (key) {
case 'wecom':
logo = 'wechat-work'
break
case 'dingtalk':
logo = 'dingtalk'
break
case 'lark':
logo = 'lark'
break
default:
logo = '' // 默认值
break
}
const config = {
...(key === 'wecom' ? { corp_id: '', agent_id: '' } : { app_key: '' }),
app_secret: '',
callback_url: ''
}
return {
key,
logoSrc: new URL(`../../../assets/logo_${logo}.svg`, import.meta.url).href,
name,
isActive: false,
isValid: false,
config
}
}
function formatFieldName(key?: any, item?: Platform): string {
const fieldNames: { [key: string]: string } = {
corp_id: 'Corp ID',
app_key: item?.key != 'lark' ? 'APP Key' : 'App ID',
app_secret: 'APP Secret',
agent_id: 'Agent ID',
callback_url: '回调地址'
}
return (
fieldNames[key as keyof typeof fieldNames] ||
(key ? key.charAt(0).toUpperCase() + key.slice(1) : '')
)
}
function getPlatformInfo() {
loading.value = true
platformApi.getPlatformInfo(loading).then((res: any) => {
if (res) {
platforms.forEach((platform) => {
const data = res.data.find((item: any) => item.platform === platform.key)
if (data) {
Object.assign(platform, {
isValid: data.is_valid,
isActive: data.is_active,
config: data.config
})
if (platform.key === 'dingtalk') {
const { corp_id, app_key, app_secret } = platform.config
platform.config = {
corp_id,
app_key,
app_secret,
callback_url: platform.config.callback_url
}
}
showPassword[platform.key] = {}
showPassword[platform.key]['app_secret'] = false
}
})
}
})
}
function validateConnection(currentPlatform: Platform) {
platformApi.validateConnection(currentPlatform, loading).then((res: any) => {
res.data ? MsgSuccess('校验成功') : MsgError('校验失败')
})
}
function refresh() {
getPlatformInfo()
}
function changeStatus(currentPlatform: Platform) {
platformApi.updateConfig(currentPlatform, loading).then((res: any) => {
MsgSuccess('操作成功')
})
}
function toggleShowPassword(platformKey: string) {
if (!showPassword[platformKey]) {
showPassword[platformKey] = {}
}
showPassword[platformKey]['app_secret'] = !showPassword[platformKey]['app_secret']
}
function showDialog(platform: Platform) {
EditModelRef.value?.open(platform)
}
</script>
<style lang="scss" scoped>
.icon {
width: 24px;
height: 24px;
}
.flex-between {
display: flex;
justify-content: space-between;
align-items: center;
}
.flex {
display: flex;
}
.align-center {
align-items: center;
}
.ml-4 {
margin-left: 4px;
}
.ml-8 {
margin-left: 8px;
}
.mr-8 {
margin-right: 8px;
}
.mb-16 {
margin-bottom: 16px;
}
.border-t {
border-top: 1px solid #ebeef5;
}
.vertical-middle {
vertical-align: middle;
}
</style>