feat(个人中心): 个人中心 -apikey&本地执行&构建报错 fix
This commit is contained in:
parent
2ee166f00e
commit
9acba4f68e
|
@ -1,7 +1,34 @@
|
||||||
import MSR from '@/api/http/index';
|
import MSR from '@/api/http/index';
|
||||||
import { GetMenuListUrl, getPublicKeyUrl, isLoginUrl, LoginUrl, LogoutUrl } from '@/api/requrls/user';
|
import {
|
||||||
|
AddAPIKEYUrl,
|
||||||
|
AddLocalConfigUrl,
|
||||||
|
DeleteAPIKEYUrl,
|
||||||
|
DisableAPIKEYUrl,
|
||||||
|
DisableLocalConfigUrl,
|
||||||
|
EnableAPIKEYUrl,
|
||||||
|
EnableLocalConfigUrl,
|
||||||
|
GetAPIKEYListUrl,
|
||||||
|
GetLocalConfigUrl,
|
||||||
|
GetMenuListUrl,
|
||||||
|
GetPublicKeyUrl,
|
||||||
|
isLoginUrl,
|
||||||
|
LoginUrl,
|
||||||
|
LogoutUrl,
|
||||||
|
UpdateAPIKEYUrl,
|
||||||
|
UpdateLocalConfigUrl,
|
||||||
|
ValidAPIKEYUrl,
|
||||||
|
ValidLocalConfigUrl,
|
||||||
|
} from '@/api/requrls/user';
|
||||||
|
|
||||||
import type { LoginData, LoginRes } from '@/models/user';
|
import type {
|
||||||
|
AddLocalConfigParams,
|
||||||
|
APIKEY,
|
||||||
|
LocalConfig,
|
||||||
|
LoginData,
|
||||||
|
LoginRes,
|
||||||
|
UpdateAPIKEYParams,
|
||||||
|
UpdateLocalConfigParams,
|
||||||
|
} from '@/models/user';
|
||||||
|
|
||||||
import type { RouteRecordNormalized } from 'vue-router';
|
import type { RouteRecordNormalized } from 'vue-router';
|
||||||
|
|
||||||
|
@ -22,5 +49,70 @@ export function getMenuList() {
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getPublicKeyRequest() {
|
export function getPublicKeyRequest() {
|
||||||
return MSR.get<string>({ url: getPublicKeyUrl }, { ignoreCancelToken: true });
|
return MSR.get<string>({ url: GetPublicKeyUrl }, { ignoreCancelToken: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-更新本地执行
|
||||||
|
export function updateLocalConfig(data: UpdateLocalConfigParams) {
|
||||||
|
return MSR.post({ url: UpdateLocalConfigUrl, data });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-添加本地执行
|
||||||
|
export function addLocalConfig(data: AddLocalConfigParams) {
|
||||||
|
return MSR.post({ url: AddLocalConfigUrl, data });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-验证本地执行配置
|
||||||
|
export function validLocalConfig(id: string) {
|
||||||
|
return MSR.get<boolean>({ url: ValidLocalConfigUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-获取本地执行配置
|
||||||
|
export function getLocalConfig() {
|
||||||
|
return MSR.get<LocalConfig[]>({ url: GetLocalConfigUrl });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-启用本地执行配置
|
||||||
|
export function enableLocalConfig(id: string) {
|
||||||
|
return MSR.get({ url: EnableLocalConfigUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-禁用本地执行配置
|
||||||
|
export function disableLocalConfig(id: string) {
|
||||||
|
return MSR.get({ url: DisableLocalConfigUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-修改 APIKEY
|
||||||
|
export function updateAPIKEY(data: UpdateAPIKEYParams) {
|
||||||
|
return MSR.post({ url: UpdateAPIKEYUrl, data });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-验证 APIKEY
|
||||||
|
export function validAPIKEY() {
|
||||||
|
return MSR.get({ url: ValidAPIKEYUrl });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-获取 APIKEY 列表
|
||||||
|
export function getAPIKEYList() {
|
||||||
|
return MSR.get<APIKEY[]>({ url: GetAPIKEYListUrl });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-开启 APIKEY
|
||||||
|
export function enableAPIKEY(id: string) {
|
||||||
|
return MSR.get({ url: EnableAPIKEYUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-关闭 APIKEY
|
||||||
|
export function disableAPIKEY(id: string) {
|
||||||
|
return MSR.get({ url: DisableAPIKEYUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-删除 APIKEY
|
||||||
|
export function deleteAPIKEY(id: string) {
|
||||||
|
return MSR.get({ url: DeleteAPIKEYUrl, params: id });
|
||||||
|
}
|
||||||
|
|
||||||
|
// 个人设置-生成 APIKEY
|
||||||
|
export function addAPIKEY() {
|
||||||
|
return MSR.get({ url: AddAPIKEYUrl });
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,4 +2,17 @@ export const LoginUrl = '/login';
|
||||||
export const isLoginUrl = '/is-login';
|
export const isLoginUrl = '/is-login';
|
||||||
export const LogoutUrl = '/signout';
|
export const LogoutUrl = '/signout';
|
||||||
export const GetMenuListUrl = '/api/user/menu';
|
export const GetMenuListUrl = '/api/user/menu';
|
||||||
export const getPublicKeyUrl = '/get-key';
|
export const GetPublicKeyUrl = '/get-key';
|
||||||
|
export const UpdateLocalConfigUrl = '/user/local/config/update'; // 个人设置-更新本地执行配置
|
||||||
|
export const AddLocalConfigUrl = '/user/local/config/add'; // 个人设置-添加本地执行
|
||||||
|
export const ValidLocalConfigUrl = '/user/local/config/validate'; // 个人设置-验证本地执行
|
||||||
|
export const GetLocalConfigUrl = '/user/local/config/get'; // 个人设置-获取本地执行配置
|
||||||
|
export const EnableLocalConfigUrl = '/user/local/config/enable'; // 个人设置-启用本地执行配置
|
||||||
|
export const DisableLocalConfigUrl = '/user/local/config/disable'; // 个人设置-禁用本地执行配置
|
||||||
|
export const UpdateAPIKEYUrl = '/user/api/key/update'; // 个人设置-更新 APIKEY
|
||||||
|
export const ValidAPIKEYUrl = '/user/api/key/validate'; // 个人设置-验证 APIKEY
|
||||||
|
export const GetAPIKEYListUrl = '/user/api/key/list'; // 个人设置-获取 APIKEY 列表
|
||||||
|
export const EnableAPIKEYUrl = '/user/api/key/enable'; // 个人设置-开启 APIKEY
|
||||||
|
export const DisableAPIKEYUrl = '/user/api/key/disable'; // 个人设置-关闭 APIKEY
|
||||||
|
export const DeleteAPIKEYUrl = '/user/api/key/delete'; // 个人设置-删除 APIKEY
|
||||||
|
export const AddAPIKEYUrl = '/user/api/key/add'; // 个人设置-生成 APIKEY
|
||||||
|
|
|
@ -62,4 +62,5 @@ declare global {
|
||||||
declare global {
|
declare global {
|
||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
export type { Component, ComponentPublicInstance, ComputedRef, ExtractDefaultPropTypes, ExtractPropTypes, ExtractPublicPropTypes, InjectionKey, PropType, Ref, VNode, WritableComputedRef } from 'vue'
|
export type { Component, ComponentPublicInstance, ComputedRef, ExtractDefaultPropTypes, ExtractPropTypes, ExtractPublicPropTypes, InjectionKey, PropType, Ref, VNode, WritableComputedRef } from 'vue'
|
||||||
|
import('vue')
|
||||||
}
|
}
|
||||||
|
|
|
@ -10,7 +10,15 @@
|
||||||
</a-tooltip>
|
</a-tooltip>
|
||||||
</div>
|
</div>
|
||||||
<a-tooltip :content="t('ms.personal.maxTip')" position="right" :disabled="apiKeyList.length < 5">
|
<a-tooltip :content="t('ms.personal.maxTip')" position="right" :disabled="apiKeyList.length < 5">
|
||||||
<a-button type="outline" class="w-[60px]" :disabled="apiKeyList.length >= 5">{{ t('common.new') }}</a-button>
|
<a-button
|
||||||
|
type="outline"
|
||||||
|
class="w-[60px]"
|
||||||
|
:disabled="apiKeyList.length >= 5"
|
||||||
|
:loading="newLoading"
|
||||||
|
@click="newApiKey"
|
||||||
|
>
|
||||||
|
{{ t('common.new') }}
|
||||||
|
</a-button>
|
||||||
</a-tooltip>
|
</a-tooltip>
|
||||||
<a-spin class="api-list-content" :loading="loading">
|
<a-spin class="api-list-content" :loading="loading">
|
||||||
<div v-for="item of apiKeyList" :key="item.id" class="api-item">
|
<div v-for="item of apiKeyList" :key="item.id" class="api-item">
|
||||||
|
@ -38,14 +46,16 @@
|
||||||
</div>
|
</div>
|
||||||
<div class="px-[16px]">
|
<div class="px-[16px]">
|
||||||
<div class="api-item-label">{{ t('ms.personal.desc') }}</div>
|
<div class="api-item-label">{{ t('ms.personal.desc') }}</div>
|
||||||
<a-tooltip :content="item.desc">
|
<a-tooltip :content="item.description" :disabled="!item.description">
|
||||||
<div class="api-item-value one-line-text">{{ item.desc }}</div>
|
<div class="api-item-value one-line-text">{{ item.description || '-' }}</div>
|
||||||
</a-tooltip>
|
</a-tooltip>
|
||||||
<div class="api-item-label">{{ t('ms.personal.createTime') }}</div>
|
<div class="api-item-label">{{ t('ms.personal.createTime') }}</div>
|
||||||
<div class="api-item-value">{{ dayjs(item.createTime).format('YYYY-MM-DD HH:mm:ss') }}</div>
|
<div class="api-item-value">
|
||||||
|
{{ dayjs(item.createTime).format('YYYY-MM-DD HH:mm:ss') }}
|
||||||
|
</div>
|
||||||
<div class="api-item-label">{{ t('ms.personal.expireTime') }}</div>
|
<div class="api-item-label">{{ t('ms.personal.expireTime') }}</div>
|
||||||
<div class="api-item-value">
|
<div class="api-item-value">
|
||||||
{{ dayjs(item.expireTime).format('YYYY-MM-DD HH:mm:ss') }}
|
{{ item.forever ? t('ms.personal.forever') : dayjs(item.expireTime).format('YYYY-MM-DD HH:mm:ss') }}
|
||||||
<a-tooltip v-if="item.isExpire" :content="t('ms.personal.expiredTip')">
|
<a-tooltip v-if="item.isExpire" :content="t('ms.personal.expiredTip')">
|
||||||
<MsIcon type="icon-icon_warning_filled" class="ml-[4px] text-[rgb(var(--warning-6))]" />
|
<MsIcon type="icon-icon_warning_filled" class="ml-[4px] text-[rgb(var(--warning-6))]" />
|
||||||
</a-tooltip>
|
</a-tooltip>
|
||||||
|
@ -61,7 +71,7 @@
|
||||||
v-model:model-value="item.enable"
|
v-model:model-value="item.enable"
|
||||||
size="small"
|
size="small"
|
||||||
:before-change="() => handleBeforeEnableChange(item)"
|
:before-change="() => handleBeforeEnableChange(item)"
|
||||||
></a-switch>
|
/>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</a-spin>
|
</a-spin>
|
||||||
|
@ -112,6 +122,7 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { onBeforeMount } from 'vue';
|
||||||
import { useClipboard } from '@vueuse/core';
|
import { useClipboard } from '@vueuse/core';
|
||||||
import { FormInstance, Message } from '@arco-design/web-vue';
|
import { FormInstance, Message } from '@arco-design/web-vue';
|
||||||
import dayjs from 'dayjs';
|
import dayjs from 'dayjs';
|
||||||
|
@ -121,83 +132,66 @@
|
||||||
import { ActionsItem } from '@/components/pure/ms-table-more-action/types';
|
import { ActionsItem } from '@/components/pure/ms-table-more-action/types';
|
||||||
import MsTag from '@/components/pure/ms-tag/ms-tag.vue';
|
import MsTag from '@/components/pure/ms-tag/ms-tag.vue';
|
||||||
|
|
||||||
|
import {
|
||||||
|
addAPIKEY,
|
||||||
|
deleteAPIKEY,
|
||||||
|
disableAPIKEY,
|
||||||
|
enableAPIKEY,
|
||||||
|
getAPIKEYList,
|
||||||
|
updateAPIKEY,
|
||||||
|
} from '@/api/modules/user/index';
|
||||||
import { useI18n } from '@/hooks/useI18n';
|
import { useI18n } from '@/hooks/useI18n';
|
||||||
import useModal from '@/hooks/useModal';
|
import useModal from '@/hooks/useModal';
|
||||||
|
|
||||||
|
import { APIKEY } from '@/models/user';
|
||||||
|
|
||||||
const { copy } = useClipboard();
|
const { copy } = useClipboard();
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const { openModal } = useModal();
|
const { openModal } = useModal();
|
||||||
|
|
||||||
interface ApiKeyItem {
|
const loading = ref(false);
|
||||||
id: string;
|
interface APIKEYItem extends APIKEY {
|
||||||
accessKey: string;
|
|
||||||
secretKey: string;
|
|
||||||
desc: string;
|
|
||||||
createTime: number;
|
|
||||||
expireTime: number;
|
|
||||||
enable: boolean;
|
|
||||||
desensitization: boolean;
|
|
||||||
isExpire: boolean;
|
isExpire: boolean;
|
||||||
|
desensitization: boolean;
|
||||||
|
}
|
||||||
|
const apiKeyList = ref<APIKEYItem[]>([]);
|
||||||
|
|
||||||
|
async function initApiKeys() {
|
||||||
|
try {
|
||||||
|
loading.value = true;
|
||||||
|
const res = await getAPIKEYList();
|
||||||
|
apiKeyList.value = res.map((item) => ({
|
||||||
|
...item,
|
||||||
|
isExpire: item.forever ? false : item.expireTime < Date.now(),
|
||||||
|
desensitization: true,
|
||||||
|
}));
|
||||||
|
} catch (error) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(error);
|
||||||
|
} finally {
|
||||||
|
loading.value = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onBeforeMount(() => {
|
||||||
|
initApiKeys();
|
||||||
|
});
|
||||||
|
|
||||||
|
const newLoading = ref(false);
|
||||||
|
async function newApiKey() {
|
||||||
|
try {
|
||||||
|
newLoading.value = true;
|
||||||
|
await addAPIKEY();
|
||||||
|
Message.success(t('common.newSuccess'));
|
||||||
|
initApiKeys();
|
||||||
|
} catch (error) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(error);
|
||||||
|
} finally {
|
||||||
|
newLoading.value = false;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
const loading = ref(false);
|
|
||||||
const apiKeyList = ref<ApiKeyItem[]>([
|
|
||||||
{
|
|
||||||
id: '238dy23d2',
|
|
||||||
accessKey: 'dueiouhwded',
|
|
||||||
secretKey: 'asdasdas',
|
|
||||||
desc: '92387yd9283d2',
|
|
||||||
createTime: 1629782400000,
|
|
||||||
expireTime: 1629982400000,
|
|
||||||
enable: true,
|
|
||||||
desensitization: true,
|
|
||||||
isExpire: false,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'ih02i3d23',
|
|
||||||
accessKey: 'sdshsd',
|
|
||||||
secretKey: 'poj4f',
|
|
||||||
desc: '92387yd9283d2',
|
|
||||||
createTime: 1629782400000,
|
|
||||||
expireTime: 1629982400000,
|
|
||||||
enable: false,
|
|
||||||
desensitization: true,
|
|
||||||
isExpire: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: '34hy34h3',
|
|
||||||
accessKey: 'sdshsd',
|
|
||||||
secretKey: 'poj4f',
|
|
||||||
desc: '92387yd9283d2',
|
|
||||||
createTime: 1629782400000,
|
|
||||||
expireTime: 1629982400000,
|
|
||||||
enable: false,
|
|
||||||
desensitization: true,
|
|
||||||
isExpire: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'f23',
|
|
||||||
accessKey: 'sdshsd',
|
|
||||||
secretKey: 'poj4f',
|
|
||||||
desc: '92387yd9283d2',
|
|
||||||
createTime: 1629782400000,
|
|
||||||
expireTime: 1629982400000,
|
|
||||||
enable: false,
|
|
||||||
desensitization: true,
|
|
||||||
isExpire: true,
|
|
||||||
},
|
|
||||||
{
|
|
||||||
id: 'ih02i3ed23',
|
|
||||||
accessKey: 'sdshsd',
|
|
||||||
secretKey: 'poj4f',
|
|
||||||
desc: '92387yd9283d2',
|
|
||||||
createTime: 1629782400000,
|
|
||||||
expireTime: 1629982400000,
|
|
||||||
enable: false,
|
|
||||||
desensitization: true,
|
|
||||||
isExpire: true,
|
|
||||||
},
|
|
||||||
]);
|
|
||||||
const actions: ActionsItem[] = [
|
const actions: ActionsItem[] = [
|
||||||
{
|
{
|
||||||
label: t('ms.personal.validTime'),
|
label: t('ms.personal.validTime'),
|
||||||
|
@ -218,11 +212,11 @@
|
||||||
Message.success(t('ms.personal.copySuccess'));
|
Message.success(t('ms.personal.copySuccess'));
|
||||||
}
|
}
|
||||||
|
|
||||||
function desensitization(item: ApiKeyItem) {
|
function desensitization(item: APIKEYItem) {
|
||||||
item.desensitization = !item.desensitization;
|
item.desensitization = !item.desensitization;
|
||||||
}
|
}
|
||||||
|
|
||||||
async function handleBeforeEnableChange(item: ApiKeyItem) {
|
async function handleBeforeEnableChange(item: APIKEYItem) {
|
||||||
if (item.enable) {
|
if (item.enable) {
|
||||||
openModal({
|
openModal({
|
||||||
type: 'error',
|
type: 'error',
|
||||||
|
@ -235,7 +229,9 @@
|
||||||
},
|
},
|
||||||
onBeforeOk: async () => {
|
onBeforeOk: async () => {
|
||||||
try {
|
try {
|
||||||
|
await disableAPIKEY(item.id);
|
||||||
Message.success(t('ms.personal.closeSuccess'));
|
Message.success(t('ms.personal.closeSuccess'));
|
||||||
|
item.enable = false;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
@ -246,15 +242,17 @@
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
try {
|
try {
|
||||||
|
await enableAPIKEY(item.id);
|
||||||
Message.success(t('ms.personal.openSuccess'));
|
Message.success(t('ms.personal.openSuccess'));
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function deleteApiKey(item: ApiKeyItem) {
|
function deleteApiKey(item: APIKEYItem) {
|
||||||
openModal({
|
openModal({
|
||||||
type: 'error',
|
type: 'error',
|
||||||
title: t('ms.personal.confirmDelete'),
|
title: t('ms.personal.confirmDelete'),
|
||||||
|
@ -266,7 +264,9 @@
|
||||||
},
|
},
|
||||||
onBeforeOk: async () => {
|
onBeforeOk: async () => {
|
||||||
try {
|
try {
|
||||||
|
await deleteAPIKEY(item.id);
|
||||||
Message.success(t('common.deleteSuccess'));
|
Message.success(t('common.deleteSuccess'));
|
||||||
|
initApiKeys();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
@ -290,8 +290,15 @@
|
||||||
timeFormRef.value?.validate(async (errors) => {
|
timeFormRef.value?.validate(async (errors) => {
|
||||||
if (!errors) {
|
if (!errors) {
|
||||||
try {
|
try {
|
||||||
|
await updateAPIKEY({
|
||||||
|
id: apiKeyList.value[0].id,
|
||||||
|
description: timeForm.value.desc,
|
||||||
|
expireTime: timeForm.value.activeTimeType === 'forever' ? 0 : dayjs(timeForm.value.time).valueOf(),
|
||||||
|
forever: timeForm.value.activeTimeType === 'forever',
|
||||||
|
});
|
||||||
Message.success(t('common.updateSuccess'));
|
Message.success(t('common.updateSuccess'));
|
||||||
done(true);
|
done(true);
|
||||||
|
initApiKeys();
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
@ -308,12 +315,12 @@
|
||||||
timeForm.value = { ...defaultTimeForm };
|
timeForm.value = { ...defaultTimeForm };
|
||||||
}
|
}
|
||||||
|
|
||||||
function handleMoreActionSelect(item: ActionsItem, apiKey: ApiKeyItem) {
|
function handleMoreActionSelect(item: ActionsItem, apiKey: APIKEYItem) {
|
||||||
if (item.eventTag === 'time') {
|
if (item.eventTag === 'time') {
|
||||||
timeForm.value = {
|
timeForm.value = {
|
||||||
activeTimeType: 'forever',
|
activeTimeType: apiKey.forever ? 'forever' : 'custom',
|
||||||
time: apiKey.expireTime ? dayjs(apiKey.expireTime).format('YYYY-MM-DD HH:mm:ss') : '',
|
time: apiKey.expireTime ? dayjs(apiKey.expireTime).format('YYYY-MM-DD HH:mm:ss') : '',
|
||||||
desc: apiKey.desc,
|
desc: apiKey.description,
|
||||||
};
|
};
|
||||||
timeModalVisible.value = true;
|
timeModalVisible.value = true;
|
||||||
} else if (item.eventTag === 'delete') {
|
} else if (item.eventTag === 'delete') {
|
||||||
|
|
|
@ -2,96 +2,126 @@
|
||||||
<div class="mb-[16px] flex items-center justify-between">
|
<div class="mb-[16px] flex items-center justify-between">
|
||||||
<div class="font-medium text-[var(--color-text-1)]">{{ t('ms.personal.localExecution') }}</div>
|
<div class="font-medium text-[var(--color-text-1)]">{{ t('ms.personal.localExecution') }}</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="grid grid-cols-2 gap-[16px]">
|
<a-spin class="w-full" :loading="loading">
|
||||||
<div class="config-card">
|
<div class="grid grid-cols-2 gap-[16px]">
|
||||||
<div class="config-card-title">
|
<div class="config-card">
|
||||||
<div class="config-card-title-text">{{ t('ms.personal.apiLocalExecution') }}</div>
|
<div class="config-card-title">
|
||||||
<MsTag theme="outline" :type="tagMap[apiConfig.status].type" size="small" class="px-[4px]">
|
<div class="config-card-title-text">{{ t('ms.personal.apiLocalExecution') }}</div>
|
||||||
{{ tagMap[apiConfig.status].text }}
|
<MsTag
|
||||||
</MsTag>
|
v-if="apiConfig.status !== 'none'"
|
||||||
</div>
|
theme="outline"
|
||||||
<a-input
|
:type="tagMap[apiConfig.status].type"
|
||||||
v-model:model-value="apiConfig.url"
|
|
||||||
:placeholder="t('ms.personal.apiLocalExecutionPlaceholder')"
|
|
||||||
class="mb-[16px]"
|
|
||||||
></a-input>
|
|
||||||
<div class="config-card-footer">
|
|
||||||
<a-button
|
|
||||||
type="outline"
|
|
||||||
class="px-[8px]"
|
|
||||||
size="mini"
|
|
||||||
:disabled="apiConfig.url.trim() === ''"
|
|
||||||
:loading="testApiLoading"
|
|
||||||
@click="testApi"
|
|
||||||
>
|
|
||||||
{{ t('ms.personal.test') }}
|
|
||||||
</a-button>
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div class="mr-[4px] text-[12px] leading-[16px] text-[var(--color-text-4)]">
|
|
||||||
{{ t('ms.personal.priorityLocalExec') }}
|
|
||||||
</div>
|
|
||||||
<a-switch
|
|
||||||
v-model:model-value="apiConfig.isPriorityLocalExec"
|
|
||||||
size="small"
|
size="small"
|
||||||
:disabled="apiConfig.status !== 1 || testApiLoading"
|
class="px-[4px]"
|
||||||
:before-change="(val) => handleApiPriorityBeforeChange(val)"
|
>
|
||||||
/>
|
{{ tagMap[apiConfig.status].text }}
|
||||||
|
</MsTag>
|
||||||
|
</div>
|
||||||
|
<a-input
|
||||||
|
v-model:model-value="apiConfig.userUrl"
|
||||||
|
:placeholder="t('ms.personal.apiLocalExecutionPlaceholder')"
|
||||||
|
class="mb-[16px]"
|
||||||
|
@press-enter="testApi"
|
||||||
|
></a-input>
|
||||||
|
<div class="config-card-footer">
|
||||||
|
<a-button
|
||||||
|
type="outline"
|
||||||
|
class="px-[8px]"
|
||||||
|
size="mini"
|
||||||
|
:disabled="apiConfig.userUrl.trim() === ''"
|
||||||
|
:loading="testApiLoading"
|
||||||
|
@click="testApi"
|
||||||
|
>
|
||||||
|
{{ t('ms.personal.test') }}
|
||||||
|
</a-button>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<div class="mr-[4px] text-[12px] leading-[16px] text-[var(--color-text-4)]">
|
||||||
|
{{ t('ms.personal.priorityLocalExec') }}
|
||||||
|
</div>
|
||||||
|
<a-switch
|
||||||
|
v-model:model-value="apiConfig.enable"
|
||||||
|
size="small"
|
||||||
|
:disabled="apiConfig.id === '' || testApiLoading"
|
||||||
|
:before-change="(val) => handleApiPriorityBeforeChange(val)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="config-card">
|
||||||
|
<div class="config-card-title">
|
||||||
|
<div class="config-card-title-text">{{ t('ms.personal.uiLocalExecution') }}</div>
|
||||||
|
<MsTag
|
||||||
|
v-if="uiConfig.status !== 'none'"
|
||||||
|
theme="outline"
|
||||||
|
:type="tagMap[uiConfig.status].type"
|
||||||
|
size="small"
|
||||||
|
class="px-[4px]"
|
||||||
|
>
|
||||||
|
{{ tagMap[uiConfig.status].text }}
|
||||||
|
</MsTag>
|
||||||
|
</div>
|
||||||
|
<a-input
|
||||||
|
v-model:model-value="uiConfig.userUrl"
|
||||||
|
:placeholder="t('ms.personal.uiLocalExecutionPlaceholder')"
|
||||||
|
class="mb-[16px]"
|
||||||
|
@press-enter="testUi"
|
||||||
|
></a-input>
|
||||||
|
<div class="config-card-footer">
|
||||||
|
<a-button
|
||||||
|
type="outline"
|
||||||
|
class="px-[8px]"
|
||||||
|
size="mini"
|
||||||
|
:disabled="uiConfig.userUrl.trim() === ''"
|
||||||
|
:loading="testUiLoading"
|
||||||
|
@click="testUi"
|
||||||
|
>
|
||||||
|
{{ t('ms.personal.test') }}
|
||||||
|
</a-button>
|
||||||
|
<div class="flex items-center">
|
||||||
|
<div class="mr-[4px] text-[12px] leading-[16px] text-[var(--color-text-4)]">
|
||||||
|
{{ t('ms.personal.priorityLocalExec') }}
|
||||||
|
</div>
|
||||||
|
<a-switch
|
||||||
|
v-model:model-value="uiConfig.enable"
|
||||||
|
size="small"
|
||||||
|
:disabled="uiConfig.id === '' || testUiLoading"
|
||||||
|
:before-change="(val) => handleUiPriorityBeforeChange(val)"
|
||||||
|
/>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="config-card">
|
</a-spin>
|
||||||
<div class="config-card-title">
|
|
||||||
<div class="config-card-title-text">{{ t('ms.personal.uiLocalExecution') }}</div>
|
|
||||||
<MsTag theme="outline" :type="tagMap[uiConfig.status].type" size="small" class="px-[4px]">
|
|
||||||
{{ tagMap[uiConfig.status].text }}
|
|
||||||
</MsTag>
|
|
||||||
</div>
|
|
||||||
<a-input
|
|
||||||
v-model:model-value="uiConfig.url"
|
|
||||||
:placeholder="t('ms.personal.uiLocalExecutionPlaceholder')"
|
|
||||||
class="mb-[16px]"
|
|
||||||
></a-input>
|
|
||||||
<div class="config-card-footer">
|
|
||||||
<a-button
|
|
||||||
type="outline"
|
|
||||||
class="px-[8px]"
|
|
||||||
size="mini"
|
|
||||||
:disabled="uiConfig.url.trim() === ''"
|
|
||||||
:loading="testUiLoading"
|
|
||||||
@click="testUi"
|
|
||||||
>
|
|
||||||
{{ t('ms.personal.test') }}
|
|
||||||
</a-button>
|
|
||||||
<div class="flex items-center">
|
|
||||||
<div class="mr-[4px] text-[12px] leading-[16px] text-[var(--color-text-4)]">
|
|
||||||
{{ t('ms.personal.priorityLocalExec') }}
|
|
||||||
</div>
|
|
||||||
<a-switch
|
|
||||||
v-model:model-value="uiConfig.isPriorityLocalExec"
|
|
||||||
size="small"
|
|
||||||
:disabled="uiConfig.status !== 1 || testUiLoading"
|
|
||||||
:before-change="(val) => handleUiPriorityBeforeChange(val)"
|
|
||||||
/>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script setup lang="ts">
|
<script setup lang="ts">
|
||||||
|
import { onBeforeMount } from 'vue';
|
||||||
import { Message } from '@arco-design/web-vue';
|
import { Message } from '@arco-design/web-vue';
|
||||||
|
|
||||||
import MsTag, { TagType } from '@/components/pure/ms-tag/ms-tag.vue';
|
import MsTag, { TagType } from '@/components/pure/ms-tag/ms-tag.vue';
|
||||||
|
|
||||||
|
import {
|
||||||
|
addLocalConfig,
|
||||||
|
disableLocalConfig,
|
||||||
|
enableLocalConfig,
|
||||||
|
getLocalConfig,
|
||||||
|
updateLocalConfig,
|
||||||
|
validLocalConfig,
|
||||||
|
} from '@/api/modules/user/index';
|
||||||
import { useI18n } from '@/hooks/useI18n';
|
import { useI18n } from '@/hooks/useI18n';
|
||||||
|
|
||||||
|
import { LocalConfig } from '@/models/user';
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
|
|
||||||
type Status = 0 | 1 | 2;
|
type Status = 0 | 1 | 2 | 'none'; // 0 未配置 1 测试通过 2 测试失败
|
||||||
interface TagMapItem {
|
interface TagMapItem {
|
||||||
type: TagType;
|
type: TagType;
|
||||||
text: string;
|
text: string;
|
||||||
}
|
}
|
||||||
|
interface Config {
|
||||||
|
status: Status;
|
||||||
|
}
|
||||||
const tagMap: Record<Status, TagMapItem> = {
|
const tagMap: Record<Status, TagMapItem> = {
|
||||||
0: {
|
0: {
|
||||||
type: 'default',
|
type: 'default',
|
||||||
|
@ -105,18 +135,44 @@
|
||||||
type: 'danger',
|
type: 'danger',
|
||||||
text: t('ms.personal.testFail'),
|
text: t('ms.personal.testFail'),
|
||||||
},
|
},
|
||||||
|
none: {
|
||||||
|
type: 'default',
|
||||||
|
text: '',
|
||||||
|
},
|
||||||
};
|
};
|
||||||
|
const loading = ref(false);
|
||||||
const testApiLoading = ref(false);
|
const testApiLoading = ref(false);
|
||||||
const apiConfig = ref({
|
const apiConfig = ref<LocalConfig & Config>({
|
||||||
url: '',
|
id: '',
|
||||||
status: 1 as Status,
|
userUrl: '',
|
||||||
isPriorityLocalExec: false,
|
enable: false,
|
||||||
|
type: 'API',
|
||||||
|
status: 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
async function testApi() {
|
async function testApi() {
|
||||||
try {
|
try {
|
||||||
testApiLoading.value = true;
|
testApiLoading.value = true;
|
||||||
Message.success(t('ms.personal.testPass'));
|
if (apiConfig.value.id) {
|
||||||
|
// 已经存在配置
|
||||||
|
await updateLocalConfig({
|
||||||
|
id: apiConfig.value.id,
|
||||||
|
userUrl: apiConfig.value.userUrl.trim(),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const result = await addLocalConfig({
|
||||||
|
type: 'API',
|
||||||
|
userUrl: apiConfig.value.userUrl.trim(),
|
||||||
|
});
|
||||||
|
apiConfig.value.id = result.id;
|
||||||
|
}
|
||||||
|
const res = await validLocalConfig(apiConfig.value.id);
|
||||||
|
apiConfig.value.status = res ? 1 : 2;
|
||||||
|
if (res) {
|
||||||
|
Message.success(t('ms.personal.testPass'));
|
||||||
|
} else {
|
||||||
|
Message.error(t('ms.personal.testFail'));
|
||||||
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
|
@ -127,6 +183,11 @@
|
||||||
|
|
||||||
async function handleApiPriorityBeforeChange(val: string | number | boolean) {
|
async function handleApiPriorityBeforeChange(val: string | number | boolean) {
|
||||||
try {
|
try {
|
||||||
|
if (val) {
|
||||||
|
await enableLocalConfig(apiConfig.value.id);
|
||||||
|
} else {
|
||||||
|
await disableLocalConfig(apiConfig.value.id);
|
||||||
|
}
|
||||||
Message.success(val ? t('ms.personal.apiLocalExecutionOpen') : t('ms.personal.apiLocalExecutionClose'));
|
Message.success(val ? t('ms.personal.apiLocalExecutionOpen') : t('ms.personal.apiLocalExecutionClose'));
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
@ -137,26 +198,52 @@
|
||||||
}
|
}
|
||||||
|
|
||||||
const testUiLoading = ref(false);
|
const testUiLoading = ref(false);
|
||||||
const uiConfig = ref({
|
const uiConfig = ref<LocalConfig & Config>({
|
||||||
url: '',
|
id: '',
|
||||||
status: 1 as Status,
|
userUrl: '',
|
||||||
isPriorityLocalExec: false,
|
enable: false,
|
||||||
|
type: 'UI',
|
||||||
|
status: 0,
|
||||||
});
|
});
|
||||||
|
|
||||||
async function testUi() {
|
async function testUi() {
|
||||||
try {
|
try {
|
||||||
testApiLoading.value = true;
|
testUiLoading.value = true;
|
||||||
Message.success(t('ms.personal.testPass'));
|
if (uiConfig.value.id) {
|
||||||
|
// 已经存在配置
|
||||||
|
await updateLocalConfig({
|
||||||
|
id: uiConfig.value.id,
|
||||||
|
userUrl: uiConfig.value.userUrl.trim(),
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
const result = await addLocalConfig({
|
||||||
|
type: 'UI',
|
||||||
|
userUrl: uiConfig.value.userUrl.trim(),
|
||||||
|
});
|
||||||
|
uiConfig.value.id = result.id;
|
||||||
|
}
|
||||||
|
const res = await validLocalConfig(uiConfig.value.id);
|
||||||
|
uiConfig.value.status = res ? 1 : 2;
|
||||||
|
if (res) {
|
||||||
|
Message.success(t('ms.personal.testPass'));
|
||||||
|
} else {
|
||||||
|
Message.error(t('ms.personal.testFail'));
|
||||||
|
}
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
// eslint-disable-next-line no-console
|
// eslint-disable-next-line no-console
|
||||||
console.log(error);
|
console.log(error);
|
||||||
} finally {
|
} finally {
|
||||||
testApiLoading.value = false;
|
testUiLoading.value = false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
async function handleUiPriorityBeforeChange(val: string | number | boolean) {
|
async function handleUiPriorityBeforeChange(val: string | number | boolean) {
|
||||||
try {
|
try {
|
||||||
|
if (val) {
|
||||||
|
await enableLocalConfig(uiConfig.value.id);
|
||||||
|
} else {
|
||||||
|
await disableLocalConfig(uiConfig.value.id);
|
||||||
|
}
|
||||||
Message.success(val ? t('ms.personal.uiLocalExecutionOpen') : t('ms.personal.uiLocalExecutionClose'));
|
Message.success(val ? t('ms.personal.uiLocalExecutionOpen') : t('ms.personal.uiLocalExecutionClose'));
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
|
@ -165,6 +252,33 @@
|
||||||
return false;
|
return false;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
onBeforeMount(async () => {
|
||||||
|
try {
|
||||||
|
loading.value = true;
|
||||||
|
const res = await getLocalConfig();
|
||||||
|
if (Array.isArray(res)) {
|
||||||
|
res.forEach((config) => {
|
||||||
|
if (config.type === 'API') {
|
||||||
|
apiConfig.value = {
|
||||||
|
...config,
|
||||||
|
status: 'none',
|
||||||
|
};
|
||||||
|
} else if (config.type === 'UI') {
|
||||||
|
uiConfig.value = {
|
||||||
|
...config,
|
||||||
|
status: 'none',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
// eslint-disable-next-line no-console
|
||||||
|
console.log(error);
|
||||||
|
} finally {
|
||||||
|
loading.value = false;
|
||||||
|
}
|
||||||
|
});
|
||||||
</script>
|
</script>
|
||||||
|
|
||||||
<style lang="less" scoped>
|
<style lang="less" scoped>
|
||||||
|
|
|
@ -50,14 +50,18 @@
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const activeMenu = ref('baseInfo');
|
||||||
|
|
||||||
watch(
|
watch(
|
||||||
() => innerVisible.value,
|
() => innerVisible.value,
|
||||||
(val) => {
|
(val) => {
|
||||||
emit('update:visible', val);
|
emit('update:visible', val);
|
||||||
|
if (!val) {
|
||||||
|
activeMenu.value = 'baseInfo';
|
||||||
|
}
|
||||||
}
|
}
|
||||||
);
|
);
|
||||||
|
|
||||||
const activeMenu = ref('baseInfo');
|
|
||||||
const menuList = ref([
|
const menuList = ref([
|
||||||
{
|
{
|
||||||
name: 'personal',
|
name: 'personal',
|
||||||
|
|
|
@ -59,11 +59,11 @@
|
||||||
// 获取当前列表里边所有包含cascade的item
|
// 获取当前列表里边所有包含cascade的item
|
||||||
if (currentFormCreateRules) {
|
if (currentFormCreateRules) {
|
||||||
const cascade = currentFormCreateRules
|
const cascade = currentFormCreateRules
|
||||||
.map((item: Record<string, any>) => item.link)
|
.map((item) => item.link)
|
||||||
.filter((item: Record<string, any>) => item)
|
.filter((item) => item)
|
||||||
.flatMap((flatItem: any) => flatItem);
|
.flatMap((flatItem: any) => flatItem);
|
||||||
// 给所有的link上边关联的某个item 进行绑定监视
|
// 给所有的link上边关联的某个item 进行绑定监视
|
||||||
return currentFormCreateRules.filter((item: Record<string, any>) => {
|
return currentFormCreateRules.filter((item) => {
|
||||||
return cascade.indexOf(item.field) > -1;
|
return cascade.indexOf(item.field) > -1;
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -88,9 +88,9 @@
|
||||||
(val) => {
|
(val) => {
|
||||||
// 监视当前改变请求获取当前方法下边的options 和获取多有的字段值
|
// 监视当前改变请求获取当前方法下边的options 和获取多有的字段值
|
||||||
if (val) {
|
if (val) {
|
||||||
val.forEach(async (item: any) => {
|
val.forEach((item) => {
|
||||||
if (item.value) {
|
if (item.value) {
|
||||||
await getOptionsRequest(item);
|
getOptionsRequest(item as FormRuleItem);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
|
@ -42,6 +42,13 @@
|
||||||
|
|
||||||
const currentKey = ref(props.defaultKey);
|
const currentKey = ref(props.defaultKey);
|
||||||
|
|
||||||
|
watch(
|
||||||
|
() => props.defaultKey,
|
||||||
|
(val) => {
|
||||||
|
currentKey.value = val;
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
const toggleMenu = (itemName: string) => {
|
const toggleMenu = (itemName: string) => {
|
||||||
if (itemName) {
|
if (itemName) {
|
||||||
currentKey.value = itemName;
|
currentKey.value = itemName;
|
||||||
|
|
|
@ -75,4 +75,5 @@ export default {
|
||||||
'common.more': 'More',
|
'common.more': 'More',
|
||||||
'common.recycle': 'Recycle Bin',
|
'common.recycle': 'Recycle Bin',
|
||||||
'common.new': 'New',
|
'common.new': 'New',
|
||||||
|
'common.newSuccess': 'Added successfully',
|
||||||
};
|
};
|
||||||
|
|
|
@ -75,4 +75,5 @@ export default {
|
||||||
'common.more': '更多',
|
'common.more': '更多',
|
||||||
'common.recycle': '回收站',
|
'common.recycle': '回收站',
|
||||||
'common.new': '新增',
|
'common.new': '新增',
|
||||||
|
'common.newSuccess': '新增成功',
|
||||||
};
|
};
|
||||||
|
|
|
@ -37,3 +37,42 @@ export interface LoginRes {
|
||||||
userRoleRelations: UserRole[];
|
userRoleRelations: UserRole[];
|
||||||
userRoles: UserRole[];
|
userRoles: UserRole[];
|
||||||
}
|
}
|
||||||
|
// 更新本地执行配置
|
||||||
|
export interface UpdateLocalConfigParams {
|
||||||
|
id: string;
|
||||||
|
userUrl: string;
|
||||||
|
}
|
||||||
|
// 本地执行配置类型
|
||||||
|
export type LocalConfigType = 'API' | 'UI';
|
||||||
|
// 添加本地执行
|
||||||
|
export interface AddLocalConfigParams {
|
||||||
|
userUrl: string;
|
||||||
|
type: LocalConfigType;
|
||||||
|
}
|
||||||
|
// 本地执行配置
|
||||||
|
export interface LocalConfig {
|
||||||
|
id: string;
|
||||||
|
userUrl: string;
|
||||||
|
enable: boolean;
|
||||||
|
type: LocalConfigType;
|
||||||
|
createUser?: string;
|
||||||
|
}
|
||||||
|
// 更新 APIKEY
|
||||||
|
export interface UpdateAPIKEYParams {
|
||||||
|
id: string;
|
||||||
|
forever: boolean;
|
||||||
|
expireTime: number;
|
||||||
|
description: string;
|
||||||
|
}
|
||||||
|
// APIKEY
|
||||||
|
export interface APIKEY {
|
||||||
|
id: string;
|
||||||
|
createUser: string;
|
||||||
|
accessKey: string;
|
||||||
|
secretKey: string;
|
||||||
|
createTime: number;
|
||||||
|
enable: boolean;
|
||||||
|
forever: boolean;
|
||||||
|
expireTime: number;
|
||||||
|
description: string;
|
||||||
|
}
|
||||||
|
|
|
@ -7,8 +7,6 @@ import { useI18n } from '@/hooks/useI18n';
|
||||||
|
|
||||||
import { FormCreateKeyEnum } from '@/enums/formCreateEnum';
|
import { FormCreateKeyEnum } from '@/enums/formCreateEnum';
|
||||||
|
|
||||||
import type { Rule } from '@form-create/arco-design';
|
|
||||||
|
|
||||||
const { t } = useI18n();
|
const { t } = useI18n();
|
||||||
const useFormCreateStore = defineStore('form-create', {
|
const useFormCreateStore = defineStore('form-create', {
|
||||||
persist: false,
|
persist: false,
|
||||||
|
@ -111,12 +109,10 @@ const useFormCreateStore = defineStore('form-create', {
|
||||||
const formValue = formValueApi.formData();
|
const formValue = formValueApi.formData();
|
||||||
// 设置自定义属性给到searchSelect
|
// 设置自定义属性给到searchSelect
|
||||||
const formCreateRuleArr = this.formCreateRuleMap.get(key);
|
const formCreateRuleArr = this.formCreateRuleMap.get(key);
|
||||||
if (formCreateRuleArr) {
|
const formCreateItem = formCreateRuleArr?.find((items) => cascadeItem.field === items.field);
|
||||||
const formCreateItem = formCreateRuleArr.find((item: FormRuleItem) => cascadeItem.field === item.field);
|
if (formCreateItem) {
|
||||||
if (formCreateItem && formCreateItem.props) {
|
formCreateItem.props.keyword = val.value;
|
||||||
formCreateItem.props.keyword = val.value;
|
formCreateItem.props.formValue = formValue;
|
||||||
formCreateItem.props.formValue = formValue;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
},
|
},
|
||||||
|
|
Loading…
Reference in New Issue