新增:并入shop、cms、oa三大模块代码

This commit is contained in:
2024-09-13 00:10:29 +08:00
parent 0068553e35
commit ea5a48fa29
424 changed files with 64497 additions and 67 deletions

View File

@@ -0,0 +1,676 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑应用' : '添加应用'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="应用名称" name="appName">
<a-input
allow-clear
placeholder="请输入应用名称"
v-model:value="form.appName"
/>
</a-form-item>
<a-form-item label="应用标识" name="appCode">
<a-input
allow-clear
placeholder="请输入应用标识"
v-model:value="form.appCode"
/>
</a-form-item>
<a-form-item label="应用秘钥" name="appSecret">
<a-input
allow-clear
placeholder="请输入应用秘钥"
v-model:value="form.appSecret"
/>
</a-form-item>
<a-form-item label="上级id, 0是顶级" name="parentId">
<a-input
allow-clear
placeholder="请输入上级id, 0是顶级"
v-model:value="form.parentId"
/>
</a-form-item>
<a-form-item label="应用类型" name="appType">
<a-input
allow-clear
placeholder="请输入应用类型"
v-model:value="form.appType"
/>
</a-form-item>
<a-form-item label="应用类型" name="appTypeMultiple">
<a-input
allow-clear
placeholder="请输入应用类型"
v-model:value="form.appTypeMultiple"
/>
</a-form-item>
<a-form-item label="类型, 0菜单, 1按钮" name="menuType">
<a-input
allow-clear
placeholder="请输入类型, 0菜单, 1按钮"
v-model:value="form.menuType"
/>
</a-form-item>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="企业名称" name="companyName">
<a-input
allow-clear
placeholder="请输入企业名称"
v-model:value="form.companyName"
/>
</a-form-item>
<a-form-item label="应用图标" name="appIcon">
<a-input
allow-clear
placeholder="请输入应用图标"
v-model:value="form.appIcon"
/>
</a-form-item>
<a-form-item label="二维码" name="appQrcode">
<a-input
allow-clear
placeholder="请输入二维码"
v-model:value="form.appQrcode"
/>
</a-form-item>
<a-form-item label="链接地址" name="appUrl">
<a-input
allow-clear
placeholder="请输入链接地址"
v-model:value="form.appUrl"
/>
</a-form-item>
<a-form-item label="后台管理地址" name="adminUrl">
<a-input
allow-clear
placeholder="请输入后台管理地址"
v-model:value="form.adminUrl"
/>
</a-form-item>
<a-form-item label="下载地址" name="downUrl">
<a-input
allow-clear
placeholder="请输入下载地址"
v-model:value="form.downUrl"
/>
</a-form-item>
<a-form-item label="链接地址" name="serverUrl">
<a-input
allow-clear
placeholder="请输入链接地址"
v-model:value="form.serverUrl"
/>
</a-form-item>
<a-form-item label="文件服务器" name="fileUrl">
<a-input
allow-clear
placeholder="请输入文件服务器"
v-model:value="form.fileUrl"
/>
</a-form-item>
<a-form-item label="回调地址" name="callbackUrl">
<a-input
allow-clear
placeholder="请输入回调地址"
v-model:value="form.callbackUrl"
/>
</a-form-item>
<a-form-item label="腾讯文档地址" name="docsUrl">
<a-input
allow-clear
placeholder="请输入腾讯文档地址"
v-model:value="form.docsUrl"
/>
</a-form-item>
<a-form-item label="代码仓库地址" name="gitUrl">
<a-input
allow-clear
placeholder="请输入代码仓库地址"
v-model:value="form.gitUrl"
/>
</a-form-item>
<a-form-item label="原型图地址" name="prototypeUrl">
<a-input
allow-clear
placeholder="请输入原型图地址"
v-model:value="form.prototypeUrl"
/>
</a-form-item>
<a-form-item label="IP白名单" name="ipAddress">
<a-input
allow-clear
placeholder="请输入IP白名单"
v-model:value="form.ipAddress"
/>
</a-form-item>
<a-form-item label="应用截图" name="images">
<a-input
allow-clear
placeholder="请输入应用截图"
v-model:value="form.images"
/>
</a-form-item>
<a-form-item label="应用包名" name="packageName">
<a-input
allow-clear
placeholder="请输入应用包名"
v-model:value="form.packageName"
/>
</a-form-item>
<a-form-item label="下载次数" name="clicks">
<a-input
allow-clear
placeholder="请输入下载次数"
v-model:value="form.clicks"
/>
</a-form-item>
<a-form-item label="安装次数" name="installs">
<a-input
allow-clear
placeholder="请输入安装次数"
v-model:value="form.installs"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="应用介绍" name="content">
<a-input
allow-clear
placeholder="请输入应用介绍"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="项目需求" name="requirement">
<a-input
allow-clear
placeholder="请输入项目需求"
v-model:value="form.requirement"
/>
</a-form-item>
<a-form-item label="开发者(个人或公司)" name="developer">
<a-input
allow-clear
placeholder="请输入开发者(个人或公司)"
v-model:value="form.developer"
/>
</a-form-item>
<a-form-item label="项目负责人" name="director">
<a-input
allow-clear
placeholder="请输入项目负责人"
v-model:value="form.director"
/>
</a-form-item>
<a-form-item label="项目经理" name="projectDirector">
<a-input
allow-clear
placeholder="请输入项目经理"
v-model:value="form.projectDirector"
/>
</a-form-item>
<a-form-item label="业务员" name="salesman">
<a-input
allow-clear
placeholder="请输入业务员"
v-model:value="form.salesman"
/>
</a-form-item>
<a-form-item label="软件定价" name="price">
<a-input
allow-clear
placeholder="请输入软件定价"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="划线价格" name="linePrice">
<a-input
allow-clear
placeholder="请输入划线价格"
v-model:value="form.linePrice"
/>
</a-form-item>
<a-form-item label="评分" name="score">
<a-input
allow-clear
placeholder="请输入评分"
v-model:value="form.score"
/>
</a-form-item>
<a-form-item label="星级" name="star">
<a-input
allow-clear
placeholder="请输入星级"
v-model:value="form.star"
/>
</a-form-item>
<a-form-item label="菜单路由地址" name="path">
<a-input
allow-clear
placeholder="请输入菜单路由地址"
v-model:value="form.path"
/>
</a-form-item>
<a-form-item label="菜单组件地址, 目录可为空" name="component">
<a-input
allow-clear
placeholder="请输入菜单组件地址, 目录可为空"
v-model:value="form.component"
/>
</a-form-item>
<a-form-item label="权限标识" name="authority">
<a-input
allow-clear
placeholder="请输入权限标识"
v-model:value="form.authority"
/>
</a-form-item>
<a-form-item label="打开位置" name="target">
<a-input
allow-clear
placeholder="请输入打开位置"
v-model:value="form.target"
/>
</a-form-item>
<a-form-item label="是否隐藏, 0否, 1是(仅注册路由不显示在左侧菜单)" name="hide">
<a-input
allow-clear
placeholder="请输入是否隐藏, 0否, 1是(仅注册路由不显示在左侧菜单)"
v-model:value="form.hide"
/>
</a-form-item>
<a-form-item label="禁止搜索1禁止 0 允许" name="search">
<a-input
allow-clear
placeholder="请输入禁止搜索1禁止 0 允许"
v-model:value="form.search"
/>
</a-form-item>
<a-form-item label="菜单侧栏选中的path" name="active">
<a-input
allow-clear
placeholder="请输入菜单侧栏选中的path"
v-model:value="form.active"
/>
</a-form-item>
<a-form-item label="其它路由元信息" name="meta">
<a-input
allow-clear
placeholder="请输入其它路由元信息"
v-model:value="form.meta"
/>
</a-form-item>
<a-form-item label="版本0正式版 1体验版 2开发版" name="edition">
<a-input
allow-clear
placeholder="请输入版本0正式版 1体验版 2开发版"
v-model:value="form.edition"
/>
</a-form-item>
<a-form-item label="版本号" name="version">
<a-input
allow-clear
placeholder="请输入版本号"
v-model:value="form.version"
/>
</a-form-item>
<a-form-item label="是否已安装" name="isUse">
<a-input
allow-clear
placeholder="请输入是否已安装"
v-model:value="form.isUse"
/>
</a-form-item>
<a-form-item label="附近1" name="file1">
<a-input
allow-clear
placeholder="请输入附近1"
v-model:value="form.file1"
/>
</a-form-item>
<a-form-item label="附件2" name="file2">
<a-input
allow-clear
placeholder="请输入附件2"
v-model:value="form.file2"
/>
</a-form-item>
<a-form-item label="附件3" name="file3">
<a-input
allow-clear
placeholder="请输入附件3"
v-model:value="form.file3"
/>
</a-form-item>
<a-form-item label="是否显示续费提醒" name="showExpiration">
<a-input
allow-clear
placeholder="请输入是否显示续费提醒"
v-model:value="form.showExpiration"
/>
</a-form-item>
<a-form-item label="是否作为案例展示" name="showCase">
<a-input
allow-clear
placeholder="请输入是否作为案例展示"
v-model:value="form.showCase"
/>
</a-form-item>
<a-form-item label="是否显示在首页" name="showIndex">
<a-input
allow-clear
placeholder="请输入是否显示在首页"
v-model:value="form.showIndex"
/>
</a-form-item>
<a-form-item label="是否推荐" name="recommend">
<a-input
allow-clear
placeholder="请输入是否推荐"
v-model:value="form.recommend"
/>
</a-form-item>
<a-form-item label="到期时间" name="expirationTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.expirationTime"
/>
</a-form-item>
<a-form-item label="续费金额" name="renewMoney">
<a-input
allow-clear
placeholder="请输入续费金额"
v-model:value="form.renewMoney"
/>
</a-form-item>
<a-form-item label="应用状态" name="appStatus">
<a-input
allow-clear
placeholder="请输入应用状态"
v-model:value="form.appStatus"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="机构id" name="organizationId">
<a-input
allow-clear
placeholder="请输入机构id"
v-model:value="form.organizationId"
/>
</a-form-item>
<a-form-item label="租户编号" name="tenantCode">
<a-input
allow-clear
placeholder="请输入租户编号"
v-model:value="form.tenantCode"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaApp, updateOaApp } from '@/api/oa/oaApp';
import { OaApp } from '@/api/oa/oaApp/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaApp | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaApp>({
appId: undefined,
appName: undefined,
appCode: undefined,
appSecret: undefined,
parentId: undefined,
appType: undefined,
appTypeMultiple: undefined,
menuType: undefined,
companyId: undefined,
companyName: undefined,
appIcon: undefined,
appQrcode: undefined,
appUrl: undefined,
adminUrl: undefined,
downUrl: undefined,
serverUrl: undefined,
fileUrl: undefined,
callbackUrl: undefined,
docsUrl: undefined,
gitUrl: undefined,
prototypeUrl: undefined,
ipAddress: undefined,
images: undefined,
packageName: undefined,
clicks: undefined,
installs: undefined,
comments: undefined,
content: undefined,
requirement: undefined,
developer: undefined,
director: undefined,
projectDirector: undefined,
salesman: undefined,
price: undefined,
linePrice: undefined,
score: undefined,
star: undefined,
path: undefined,
component: undefined,
authority: undefined,
target: undefined,
hide: undefined,
search: undefined,
active: undefined,
meta: undefined,
edition: undefined,
version: undefined,
isUse: undefined,
file1: undefined,
file2: undefined,
file3: undefined,
showExpiration: undefined,
showCase: undefined,
showIndex: undefined,
recommend: undefined,
expirationTime: undefined,
renewMoney: undefined,
appStatus: undefined,
sortNumber: undefined,
status: undefined,
deleted: undefined,
userId: undefined,
organizationId: undefined,
tenantCode: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAppId: undefined,
oaAppName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAppName: [
{
required: true,
type: 'string',
message: '请填写应用名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaApp : addOaApp;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,599 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAppId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAppEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAppEdit from './components/oaAppEdit.vue';
import { pageOaApp, removeOaApp, removeBatchOaApp } from '@/api/oa/oaApp';
import type { OaApp, OaAppParam } from '@/api/oa/oaApp/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaApp[]>([]);
// 当前编辑数据
const current = ref<OaApp | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaApp({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
width: 90,
},
{
title: '应用名称',
dataIndex: 'appName',
key: 'appName',
align: 'center',
},
{
title: '应用标识',
dataIndex: 'appCode',
key: 'appCode',
align: 'center',
},
{
title: '应用秘钥',
dataIndex: 'appSecret',
key: 'appSecret',
align: 'center',
},
{
title: '上级id, 0是顶级',
dataIndex: 'parentId',
key: 'parentId',
align: 'center',
},
{
title: '应用类型',
dataIndex: 'appType',
key: 'appType',
align: 'center',
},
{
title: '应用类型',
dataIndex: 'appTypeMultiple',
key: 'appTypeMultiple',
align: 'center',
},
{
title: '类型, 0菜单, 1按钮',
dataIndex: 'menuType',
key: 'menuType',
align: 'center',
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '企业名称',
dataIndex: 'companyName',
key: 'companyName',
align: 'center',
},
{
title: '应用图标',
dataIndex: 'appIcon',
key: 'appIcon',
align: 'center',
},
{
title: '二维码',
dataIndex: 'appQrcode',
key: 'appQrcode',
align: 'center',
},
{
title: '链接地址',
dataIndex: 'appUrl',
key: 'appUrl',
align: 'center',
},
{
title: '后台管理地址',
dataIndex: 'adminUrl',
key: 'adminUrl',
align: 'center',
},
{
title: '下载地址',
dataIndex: 'downUrl',
key: 'downUrl',
align: 'center',
},
{
title: '链接地址',
dataIndex: 'serverUrl',
key: 'serverUrl',
align: 'center',
},
{
title: '文件服务器',
dataIndex: 'fileUrl',
key: 'fileUrl',
align: 'center',
},
{
title: '回调地址',
dataIndex: 'callbackUrl',
key: 'callbackUrl',
align: 'center',
},
{
title: '腾讯文档地址',
dataIndex: 'docsUrl',
key: 'docsUrl',
align: 'center',
},
{
title: '代码仓库地址',
dataIndex: 'gitUrl',
key: 'gitUrl',
align: 'center',
},
{
title: '原型图地址',
dataIndex: 'prototypeUrl',
key: 'prototypeUrl',
align: 'center',
},
{
title: 'IP白名单',
dataIndex: 'ipAddress',
key: 'ipAddress',
align: 'center',
},
{
title: '应用截图',
dataIndex: 'images',
key: 'images',
align: 'center',
},
{
title: '应用包名',
dataIndex: 'packageName',
key: 'packageName',
align: 'center',
},
{
title: '下载次数',
dataIndex: 'clicks',
key: 'clicks',
align: 'center',
},
{
title: '安装次数',
dataIndex: 'installs',
key: 'installs',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '应用介绍',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '项目需求',
dataIndex: 'requirement',
key: 'requirement',
align: 'center',
},
{
title: '开发者(个人或公司)',
dataIndex: 'developer',
key: 'developer',
align: 'center',
},
{
title: '项目负责人',
dataIndex: 'director',
key: 'director',
align: 'center',
},
{
title: '项目经理',
dataIndex: 'projectDirector',
key: 'projectDirector',
align: 'center',
},
{
title: '业务员',
dataIndex: 'salesman',
key: 'salesman',
align: 'center',
},
{
title: '软件定价',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '划线价格',
dataIndex: 'linePrice',
key: 'linePrice',
align: 'center',
},
{
title: '评分',
dataIndex: 'score',
key: 'score',
align: 'center',
},
{
title: '星级',
dataIndex: 'star',
key: 'star',
align: 'center',
},
{
title: '菜单路由地址',
dataIndex: 'path',
key: 'path',
align: 'center',
},
{
title: '菜单组件地址, 目录可为空',
dataIndex: 'component',
key: 'component',
align: 'center',
},
{
title: '权限标识',
dataIndex: 'authority',
key: 'authority',
align: 'center',
},
{
title: '打开位置',
dataIndex: 'target',
key: 'target',
align: 'center',
},
{
title: '是否隐藏, 0否, 1是(仅注册路由不显示在左侧菜单)',
dataIndex: 'hide',
key: 'hide',
align: 'center',
},
{
title: '禁止搜索1禁止 0 允许',
dataIndex: 'search',
key: 'search',
align: 'center',
},
{
title: '菜单侧栏选中的path',
dataIndex: 'active',
key: 'active',
align: 'center',
},
{
title: '其它路由元信息',
dataIndex: 'meta',
key: 'meta',
align: 'center',
},
{
title: '版本0正式版 1体验版 2开发版',
dataIndex: 'edition',
key: 'edition',
align: 'center',
},
{
title: '版本号',
dataIndex: 'version',
key: 'version',
align: 'center',
},
{
title: '是否已安装',
dataIndex: 'isUse',
key: 'isUse',
align: 'center',
},
{
title: '附近1',
dataIndex: 'file1',
key: 'file1',
align: 'center',
},
{
title: '附件2',
dataIndex: 'file2',
key: 'file2',
align: 'center',
},
{
title: '附件3',
dataIndex: 'file3',
key: 'file3',
align: 'center',
},
{
title: '是否显示续费提醒',
dataIndex: 'showExpiration',
key: 'showExpiration',
align: 'center',
},
{
title: '是否作为案例展示',
dataIndex: 'showCase',
key: 'showCase',
align: 'center',
},
{
title: '是否显示在首页',
dataIndex: 'showIndex',
key: 'showIndex',
align: 'center',
},
{
title: '是否推荐',
dataIndex: 'recommend',
key: 'recommend',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'expirationTime',
key: 'expirationTime',
align: 'center',
},
{
title: '续费金额',
dataIndex: 'renewMoney',
key: 'renewMoney',
align: 'center',
},
{
title: '应用状态',
dataIndex: 'appStatus',
key: 'appStatus',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '机构id',
dataIndex: 'organizationId',
key: 'organizationId',
align: 'center',
},
{
title: '租户编号',
dataIndex: 'tenantCode',
key: 'tenantCode',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAppParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaApp) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaApp) => {
const hide = message.loading('请求中..', 0);
removeOaApp(row.oaAppId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaApp(selection.value.map((d) => d.oaAppId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaApp) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaApp'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,212 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑应用参数' : '添加应用参数'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="应用ID" name="appId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="名称" name="name">
<a-input
allow-clear
placeholder="请输入名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1删除" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAppField, updateOaAppField } from '@/api/oa/oaAppField';
import { OaAppField } from '@/api/oa/oaAppField/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAppField | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAppField>({
id: undefined,
appId: undefined,
name: undefined,
comments: undefined,
userId: undefined,
status: undefined,
sortNumber: undefined,
tenantId: undefined,
createTime: undefined,
oaAppFieldId: undefined,
oaAppFieldName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAppFieldName: [
{
required: true,
type: 'string',
message: '请填写应用参数名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAppField : addOaAppField;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,251 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAppFieldId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAppFieldEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAppFieldEdit from './components/oaAppFieldEdit.vue';
import { pageOaAppField, removeOaAppField, removeBatchOaAppField } from '@/api/oa/oaAppField';
import type { OaAppField, OaAppFieldParam } from '@/api/oa/oaAppField/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAppField[]>([]);
// 当前编辑数据
const current = ref<OaAppField | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAppField({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'id',
key: 'id',
align: 'center',
width: 90,
},
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1删除',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAppFieldParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAppField) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAppField) => {
const hide = message.loading('请求中..', 0);
removeOaAppField(row.oaAppFieldId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAppField(selection.value.map((d) => d.oaAppFieldId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAppField) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAppField'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,242 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑续费管理' : '添加续费管理'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="应用ID" name="appId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="续费金额" name="money">
<a-input
allow-clear
placeholder="请输入续费金额"
v-model:value="form.money"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="开始时间" name="startTime">
<a-input
allow-clear
placeholder="请输入开始时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="付款凭证" name="images">
<a-input
allow-clear
placeholder="请输入付款凭证"
v-model:value="form.images"
/>
</a-form-item>
<a-form-item label="用户姓名" name="nickname">
<a-input
allow-clear
placeholder="请输入用户姓名"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAppRenew, updateOaAppRenew } from '@/api/oa/oaAppRenew';
import { OaAppRenew } from '@/api/oa/oaAppRenew/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAppRenew | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAppRenew>({
appRenewId: undefined,
appId: undefined,
money: undefined,
comments: undefined,
startTime: undefined,
endTime: undefined,
companyId: undefined,
userId: undefined,
images: undefined,
nickname: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaAppRenewId: undefined,
oaAppRenewName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAppRenewName: [
{
required: true,
type: 'string',
message: '请填写续费管理名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAppRenew : addOaAppRenew;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,275 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAppRenewId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAppRenewEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAppRenewEdit from './components/oaAppRenewEdit.vue';
import { pageOaAppRenew, removeOaAppRenew, removeBatchOaAppRenew } from '@/api/oa/oaAppRenew';
import type { OaAppRenew, OaAppRenewParam } from '@/api/oa/oaAppRenew/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAppRenew[]>([]);
// 当前编辑数据
const current = ref<OaAppRenew | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAppRenew({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'appRenewId',
key: 'appRenewId',
align: 'center',
width: 90,
},
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '续费金额',
dataIndex: 'money',
key: 'money',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '开始时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '付款凭证',
dataIndex: 'images',
key: 'images',
align: 'center',
},
{
title: '用户姓名',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAppRenewParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAppRenew) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAppRenew) => {
const hide = message.loading('请求中..', 0);
removeOaAppRenew(row.oaAppRenewId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAppRenew(selection.value.map((d) => d.oaAppRenewId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAppRenew) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAppRenew'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,228 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑项目域名' : '添加项目域名'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="应用ID" name="appId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="域名类型" name="name">
<a-input
allow-clear
placeholder="请输入域名类型"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="域名" name="domain">
<a-input
allow-clear
placeholder="请输入域名"
v-model:value="form.domain"
/>
</a-form-item>
<a-form-item label="账号" name="account">
<a-input
allow-clear
placeholder="请输入账号"
v-model:value="form.account"
/>
</a-form-item>
<a-form-item label="密码" name="password">
<a-input
allow-clear
placeholder="请输入密码"
v-model:value="form.password"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAppUrl, updateOaAppUrl } from '@/api/oa/oaAppUrl';
import { OaAppUrl } from '@/api/oa/oaAppUrl/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAppUrl | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAppUrl>({
appUrlId: undefined,
appId: undefined,
name: undefined,
domain: undefined,
account: undefined,
password: undefined,
comments: undefined,
sortNumber: undefined,
status: undefined,
createTime: undefined,
tenantId: undefined,
oaAppUrlId: undefined,
oaAppUrlName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAppUrlName: [
{
required: true,
type: 'string',
message: '请填写项目域名名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAppUrl : addOaAppUrl;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,263 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAppUrlId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAppUrlEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAppUrlEdit from './components/oaAppUrlEdit.vue';
import { pageOaAppUrl, removeOaAppUrl, removeBatchOaAppUrl } from '@/api/oa/oaAppUrl';
import type { OaAppUrl, OaAppUrlParam } from '@/api/oa/oaAppUrl/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAppUrl[]>([]);
// 当前编辑数据
const current = ref<OaAppUrl | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAppUrl({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'appUrlId',
key: 'appUrlId',
align: 'center',
width: 90,
},
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '域名类型',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '域名',
dataIndex: 'domain',
key: 'domain',
align: 'center',
},
{
title: '账号',
dataIndex: 'account',
key: 'account',
align: 'center',
},
{
title: '密码',
dataIndex: 'password',
key: 'password',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAppUrlParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAppUrl) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAppUrl) => {
const hide = message.loading('请求中..', 0);
removeOaAppUrl(row.oaAppUrlId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAppUrl(selection.value.map((d) => d.oaAppUrlId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAppUrl) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAppUrl'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,201 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑应用成员' : '添加应用成员'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="角色10体验成员 20开发者成员 30管理员 " name="role">
<a-input
allow-clear
placeholder="请输入角色10体验成员 20开发者成员 30管理员 "
v-model:value="form.role"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="应用ID" name="appId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="昵称" name="nickname">
<a-input
allow-clear
placeholder="请输入昵称"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAppUser, updateOaAppUser } from '@/api/oa/oaAppUser';
import { OaAppUser } from '@/api/oa/oaAppUser/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAppUser | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAppUser>({
appUserId: undefined,
role: undefined,
userId: undefined,
appId: undefined,
nickname: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaAppUserId: undefined,
oaAppUserName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAppUserName: [
{
required: true,
type: 'string',
message: '请填写应用成员名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAppUser : addOaAppUser;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,245 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAppUserId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAppUserEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAppUserEdit from './components/oaAppUserEdit.vue';
import { pageOaAppUser, removeOaAppUser, removeBatchOaAppUser } from '@/api/oa/oaAppUser';
import type { OaAppUser, OaAppUserParam } from '@/api/oa/oaAppUser/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAppUser[]>([]);
// 当前编辑数据
const current = ref<OaAppUser | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAppUser({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'appUserId',
key: 'appUserId',
align: 'center',
width: 90,
},
{
title: '角色10体验成员 20开发者成员 30管理员 ',
dataIndex: 'role',
key: 'role',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '昵称',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAppUserParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAppUser) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAppUser) => {
const hide = message.loading('请求中..', 0);
removeOaAppUser(row.oaAppUserId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAppUser(selection.value.map((d) => d.oaAppUserId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAppUser) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAppUser'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,444 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑服务器资产记录表' : '添加服务器资产记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="资产名称" name="name">
<a-input
allow-clear
placeholder="请输入资产名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="资产标识" name="code">
<a-input
allow-clear
placeholder="请输入资产标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="资产类型" name="type">
<a-input
allow-clear
placeholder="请输入资产类型"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="服务器厂商" name="brand">
<a-input
allow-clear
placeholder="请输入服务器厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="服务器配置" name="configuration">
<a-input
allow-clear
placeholder="请输入服务器配置"
v-model:value="form.configuration"
/>
</a-form-item>
<a-form-item label="初始账号" name="account">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.account"
/>
</a-form-item>
<a-form-item label="初始密码" name="password">
<a-input
allow-clear
placeholder="请输入初始密码"
v-model:value="form.password"
/>
</a-form-item>
<a-form-item label="(阿里云/腾讯云)登录账号" name="brandAccount">
<a-input
allow-clear
placeholder="请输入(阿里云/腾讯云)登录账号"
v-model:value="form.brandAccount"
/>
</a-form-item>
<a-form-item label="(阿里云/腾讯云)登录密码" name="brandPassword">
<a-input
allow-clear
placeholder="请输入(阿里云/腾讯云)登录密码"
v-model:value="form.brandPassword"
/>
</a-form-item>
<a-form-item label="宝塔面板" name="panel">
<a-input
allow-clear
placeholder="请输入宝塔面板"
v-model:value="form.panel"
/>
</a-form-item>
<a-form-item label="宝塔面板账号" name="panelAccount">
<a-input
allow-clear
placeholder="请输入宝塔面板账号"
v-model:value="form.panelAccount"
/>
</a-form-item>
<a-form-item label="宝塔面板密码" name="panelPassword">
<a-input
allow-clear
placeholder="请输入宝塔面板密码"
v-model:value="form.panelPassword"
/>
</a-form-item>
<a-form-item label="财务信息-合同金额" name="financeAmount">
<a-input
allow-clear
placeholder="请输入财务信息-合同金额"
v-model:value="form.financeAmount"
/>
</a-form-item>
<a-form-item label="购买年限" name="financeYears">
<a-input
allow-clear
placeholder="请输入购买年限"
v-model:value="form.financeYears"
/>
</a-form-item>
<a-form-item label="续费金额" name="financeRenew">
<a-input
allow-clear
placeholder="请输入续费金额"
v-model:value="form.financeRenew"
/>
</a-form-item>
<a-form-item label="客户名称" name="financeCustomerName">
<a-input
allow-clear
placeholder="请输入客户名称"
v-model:value="form.financeCustomerName"
/>
</a-form-item>
<a-form-item label="客户联系人" name="financeCustomerContact">
<a-input
allow-clear
placeholder="请输入客户联系人"
v-model:value="form.financeCustomerContact"
/>
</a-form-item>
<a-form-item label="客户联系电话" name="financeCustomerPhone">
<a-input
allow-clear
placeholder="请输入客户联系电话"
v-model:value="form.financeCustomerPhone"
/>
</a-form-item>
<a-form-item label="客户ID" name="customerId">
<a-input
allow-clear
placeholder="请输入客户ID"
v-model:value="form.customerId"
/>
</a-form-item>
<a-form-item label="客户名称" name="customerName">
<a-input
allow-clear
placeholder="请输入客户名称"
v-model:value="form.customerName"
/>
</a-form-item>
<a-form-item label="开放端口" name="openPort">
<a-input
allow-clear
placeholder="请输入开放端口"
v-model:value="form.openPort"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="可见性(public,private,protected)" name="visibility">
<a-input
allow-clear
placeholder="请输入可见性(public,private,protected)"
v-model:value="form.visibility"
/>
</a-form-item>
<a-form-item label="宝塔接口秘钥" name="btSign">
<a-input
allow-clear
placeholder="请输入宝塔接口秘钥"
v-model:value="form.btSign"
/>
</a-form-item>
<a-form-item label="文章排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="客户ID" name="companyId">
<a-input
allow-clear
placeholder="请输入客户ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="机构id" name="organizationId">
<a-input
allow-clear
placeholder="请输入机构id"
v-model:value="form.organizationId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssets, updateOaAssets } from '@/api/oa/oaAssets';
import { OaAssets } from '@/api/oa/oaAssets/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssets | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssets>({
assetsId: undefined,
name: undefined,
code: undefined,
type: undefined,
brand: undefined,
configuration: undefined,
account: undefined,
password: undefined,
brandAccount: undefined,
brandPassword: undefined,
panel: undefined,
panelAccount: undefined,
panelPassword: undefined,
financeAmount: undefined,
financeYears: undefined,
financeRenew: undefined,
financeCustomerName: undefined,
financeCustomerContact: undefined,
financeCustomerPhone: undefined,
customerId: undefined,
customerName: undefined,
openPort: undefined,
content: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
visibility: undefined,
btSign: undefined,
sortNumber: undefined,
comments: undefined,
companyId: undefined,
userId: undefined,
organizationId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsId: undefined,
oaAssetsName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsName: [
{
required: true,
type: 'string',
message: '请填写服务器资产记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssets : addOaAssets;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,425 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsEdit from './components/oaAssetsEdit.vue';
import { pageOaAssets, removeOaAssets, removeBatchOaAssets } from '@/api/oa/oaAssets';
import type { OaAssets, OaAssetsParam } from '@/api/oa/oaAssets/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssets[]>([]);
// 当前编辑数据
const current = ref<OaAssets | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssets({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '资产ID',
dataIndex: 'assetsId',
key: 'assetsId',
align: 'center',
width: 90,
},
{
title: '资产名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '资产标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '资产类型',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '服务器厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '服务器配置',
dataIndex: 'configuration',
key: 'configuration',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'account',
key: 'account',
align: 'center',
},
{
title: '初始密码',
dataIndex: 'password',
key: 'password',
align: 'center',
},
{
title: '(阿里云/腾讯云)登录账号',
dataIndex: 'brandAccount',
key: 'brandAccount',
align: 'center',
},
{
title: '(阿里云/腾讯云)登录密码',
dataIndex: 'brandPassword',
key: 'brandPassword',
align: 'center',
},
{
title: '宝塔面板',
dataIndex: 'panel',
key: 'panel',
align: 'center',
},
{
title: '宝塔面板账号',
dataIndex: 'panelAccount',
key: 'panelAccount',
align: 'center',
},
{
title: '宝塔面板密码',
dataIndex: 'panelPassword',
key: 'panelPassword',
align: 'center',
},
{
title: '财务信息-合同金额',
dataIndex: 'financeAmount',
key: 'financeAmount',
align: 'center',
},
{
title: '购买年限',
dataIndex: 'financeYears',
key: 'financeYears',
align: 'center',
},
{
title: '续费金额',
dataIndex: 'financeRenew',
key: 'financeRenew',
align: 'center',
},
{
title: '客户名称',
dataIndex: 'financeCustomerName',
key: 'financeCustomerName',
align: 'center',
},
{
title: '客户联系人',
dataIndex: 'financeCustomerContact',
key: 'financeCustomerContact',
align: 'center',
},
{
title: '客户联系电话',
dataIndex: 'financeCustomerPhone',
key: 'financeCustomerPhone',
align: 'center',
},
{
title: '客户ID',
dataIndex: 'customerId',
key: 'customerId',
align: 'center',
},
{
title: '客户名称',
dataIndex: 'customerName',
key: 'customerName',
align: 'center',
},
{
title: '开放端口',
dataIndex: 'openPort',
key: 'openPort',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '可见性(public,private,protected)',
dataIndex: 'visibility',
key: 'visibility',
align: 'center',
},
{
title: '宝塔接口秘钥',
dataIndex: 'btSign',
key: 'btSign',
align: 'center',
},
{
title: '文章排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '客户ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '机构id',
dataIndex: 'organizationId',
key: 'organizationId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssets) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssets) => {
const hide = message.loading('请求中..', 0);
removeOaAssets(row.oaAssetsId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssets(selection.value.map((d) => d.oaAssetsId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssets) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssets'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,284 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑代码仓库' : '添加代码仓库'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="名称" name="name">
<a-input
allow-clear
placeholder="请输入名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="英文标识" name="code">
<a-input
allow-clear
placeholder="请输入英文标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="仓库地址" name="gitUrl">
<a-input
allow-clear
placeholder="请输入仓库地址"
v-model:value="form.gitUrl"
/>
</a-form-item>
<a-form-item label="仓库品牌" name="brand">
<a-input
allow-clear
placeholder="请输入仓库品牌"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="价格" name="price">
<a-input
allow-clear
placeholder="请输入价格"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsCode, updateOaAssetsCode } from '@/api/oa/oaAssetsCode';
import { OaAssetsCode } from '@/api/oa/oaAssetsCode/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsCode | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsCode>({
id: undefined,
name: undefined,
code: undefined,
gitUrl: undefined,
brand: undefined,
price: undefined,
content: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
sortNumber: undefined,
comments: undefined,
userId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsCodeId: undefined,
oaAssetsCodeName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsCodeName: [
{
required: true,
type: 'string',
message: '请填写代码仓库名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsCode : addOaAssetsCode;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,305 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsCodeId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsCodeEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsCodeEdit from './components/oaAssetsCodeEdit.vue';
import { pageOaAssetsCode, removeOaAssetsCode, removeBatchOaAssetsCode } from '@/api/oa/oaAssetsCode';
import type { OaAssetsCode, OaAssetsCodeParam } from '@/api/oa/oaAssetsCode/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsCode[]>([]);
// 当前编辑数据
const current = ref<OaAssetsCode | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsCode({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: 'ID',
dataIndex: 'id',
key: 'id',
align: 'center',
width: 90,
},
{
title: '名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '英文标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '仓库地址',
dataIndex: 'gitUrl',
key: 'gitUrl',
align: 'center',
},
{
title: '仓库品牌',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '价格',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsCodeParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsCode) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsCode) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsCode(row.oaAssetsCodeId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsCode(selection.value.map((d) => d.oaAssetsCodeId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsCode) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsCode'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,300 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑域名管理记录表' : '添加域名管理记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="域名" name="name">
<a-input
allow-clear
placeholder="请输入域名"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="域名标识" name="code">
<a-input
allow-clear
placeholder="请输入域名标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="注册厂商" name="brand">
<a-input
allow-clear
placeholder="请输入注册厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="初始账号" name="account">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.account"
/>
</a-form-item>
<a-form-item label="初始密码" name="password">
<a-input
allow-clear
placeholder="请输入初始密码"
v-model:value="form.password"
/>
</a-form-item>
<a-form-item label="价格" name="price">
<a-input
allow-clear
placeholder="请输入价格"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="ssl证书" name="ssl">
<a-input
allow-clear
placeholder="请输入ssl证书"
v-model:value="form.ssl"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsDomain, updateOaAssetsDomain } from '@/api/oa/oaAssetsDomain';
import { OaAssetsDomain } from '@/api/oa/oaAssetsDomain/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsDomain | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsDomain>({
domainId: undefined,
name: undefined,
code: undefined,
brand: undefined,
account: undefined,
password: undefined,
price: undefined,
content: undefined,
ssl: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
sortNumber: undefined,
comments: undefined,
userId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsDomainId: undefined,
oaAssetsDomainName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsDomainName: [
{
required: true,
type: 'string',
message: '请填写域名管理记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsDomain : addOaAssetsDomain;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,317 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsDomainId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsDomainEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsDomainEdit from './components/oaAssetsDomainEdit.vue';
import { pageOaAssetsDomain, removeOaAssetsDomain, removeBatchOaAssetsDomain } from '@/api/oa/oaAssetsDomain';
import type { OaAssetsDomain, OaAssetsDomainParam } from '@/api/oa/oaAssetsDomain/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsDomain[]>([]);
// 当前编辑数据
const current = ref<OaAssetsDomain | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsDomain({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: 'ID',
dataIndex: 'domainId',
key: 'domainId',
align: 'center',
width: 90,
},
{
title: '域名',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '域名标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '注册厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'account',
key: 'account',
align: 'center',
},
{
title: '初始密码',
dataIndex: 'password',
key: 'password',
align: 'center',
},
{
title: '价格',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: 'ssl证书',
dataIndex: 'ssl',
key: 'ssl',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsDomainParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsDomain) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsDomain) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsDomain(row.oaAssetsDomainId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsDomain(selection.value.map((d) => d.oaAssetsDomainId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsDomain) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsDomain'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,300 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑企业邮箱记录表' : '添加企业邮箱记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="域名" name="name">
<a-input
allow-clear
placeholder="请输入域名"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="域名标识" name="code">
<a-input
allow-clear
placeholder="请输入域名标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="主机型号" name="type">
<a-input
allow-clear
placeholder="请输入主机型号"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="品牌厂商" name="brand">
<a-input
allow-clear
placeholder="请输入品牌厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="初始账号" name="system">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.system"
/>
</a-form-item>
<a-form-item label="价格" name="price">
<a-input
allow-clear
placeholder="请输入价格"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="ssl证书" name="ssl">
<a-input
allow-clear
placeholder="请输入ssl证书"
v-model:value="form.ssl"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsEmail, updateOaAssetsEmail } from '@/api/oa/oaAssetsEmail';
import { OaAssetsEmail } from '@/api/oa/oaAssetsEmail/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsEmail | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsEmail>({
emailId: undefined,
name: undefined,
code: undefined,
type: undefined,
brand: undefined,
system: undefined,
price: undefined,
content: undefined,
ssl: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
sortNumber: undefined,
comments: undefined,
userId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsEmailId: undefined,
oaAssetsEmailName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsEmailName: [
{
required: true,
type: 'string',
message: '请填写企业邮箱记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsEmail : addOaAssetsEmail;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,317 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsEmailId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsEmailEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsEmailEdit from './components/oaAssetsEmailEdit.vue';
import { pageOaAssetsEmail, removeOaAssetsEmail, removeBatchOaAssetsEmail } from '@/api/oa/oaAssetsEmail';
import type { OaAssetsEmail, OaAssetsEmailParam } from '@/api/oa/oaAssetsEmail/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsEmail[]>([]);
// 当前编辑数据
const current = ref<OaAssetsEmail | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsEmail({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: 'ID',
dataIndex: 'emailId',
key: 'emailId',
align: 'center',
width: 90,
},
{
title: '域名',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '域名标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '主机型号',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '品牌厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'system',
key: 'system',
align: 'center',
},
{
title: '价格',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: 'ssl证书',
dataIndex: 'ssl',
key: 'ssl',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsEmailParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsEmail) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsEmail) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsEmail(row.oaAssetsEmailId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsEmail(selection.value.map((d) => d.oaAssetsEmailId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsEmail) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsEmail'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,444 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑服务器资产记录表' : '添加服务器资产记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="资产名称" name="name">
<a-input
allow-clear
placeholder="请输入资产名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="资产标识" name="code">
<a-input
allow-clear
placeholder="请输入资产标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="资产类型" name="type">
<a-input
allow-clear
placeholder="请输入资产类型"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="服务器厂商" name="brand">
<a-input
allow-clear
placeholder="请输入服务器厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="服务器配置" name="configuration">
<a-input
allow-clear
placeholder="请输入服务器配置"
v-model:value="form.configuration"
/>
</a-form-item>
<a-form-item label="初始账号" name="account">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.account"
/>
</a-form-item>
<a-form-item label="初始密码" name="password">
<a-input
allow-clear
placeholder="请输入初始密码"
v-model:value="form.password"
/>
</a-form-item>
<a-form-item label="(阿里云/腾讯云)登录账号" name="brandAccount">
<a-input
allow-clear
placeholder="请输入(阿里云/腾讯云)登录账号"
v-model:value="form.brandAccount"
/>
</a-form-item>
<a-form-item label="(阿里云/腾讯云)登录密码" name="brandPassword">
<a-input
allow-clear
placeholder="请输入(阿里云/腾讯云)登录密码"
v-model:value="form.brandPassword"
/>
</a-form-item>
<a-form-item label="宝塔面板" name="panel">
<a-input
allow-clear
placeholder="请输入宝塔面板"
v-model:value="form.panel"
/>
</a-form-item>
<a-form-item label="宝塔面板账号" name="panelAccount">
<a-input
allow-clear
placeholder="请输入宝塔面板账号"
v-model:value="form.panelAccount"
/>
</a-form-item>
<a-form-item label="宝塔面板密码" name="panelPassword">
<a-input
allow-clear
placeholder="请输入宝塔面板密码"
v-model:value="form.panelPassword"
/>
</a-form-item>
<a-form-item label="财务信息-合同金额" name="financeAmount">
<a-input
allow-clear
placeholder="请输入财务信息-合同金额"
v-model:value="form.financeAmount"
/>
</a-form-item>
<a-form-item label="购买年限" name="financeYears">
<a-input
allow-clear
placeholder="请输入购买年限"
v-model:value="form.financeYears"
/>
</a-form-item>
<a-form-item label="续费金额" name="financeRenew">
<a-input
allow-clear
placeholder="请输入续费金额"
v-model:value="form.financeRenew"
/>
</a-form-item>
<a-form-item label="客户名称" name="financeCustomerName">
<a-input
allow-clear
placeholder="请输入客户名称"
v-model:value="form.financeCustomerName"
/>
</a-form-item>
<a-form-item label="客户联系人" name="financeCustomerContact">
<a-input
allow-clear
placeholder="请输入客户联系人"
v-model:value="form.financeCustomerContact"
/>
</a-form-item>
<a-form-item label="客户联系电话" name="financeCustomerPhone">
<a-input
allow-clear
placeholder="请输入客户联系电话"
v-model:value="form.financeCustomerPhone"
/>
</a-form-item>
<a-form-item label="客户ID" name="customerId">
<a-input
allow-clear
placeholder="请输入客户ID"
v-model:value="form.customerId"
/>
</a-form-item>
<a-form-item label="客户名称" name="customerName">
<a-input
allow-clear
placeholder="请输入客户名称"
v-model:value="form.customerName"
/>
</a-form-item>
<a-form-item label="开放端口" name="openPort">
<a-input
allow-clear
placeholder="请输入开放端口"
v-model:value="form.openPort"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="可见性(public,private,protected)" name="visibility">
<a-input
allow-clear
placeholder="请输入可见性(public,private,protected)"
v-model:value="form.visibility"
/>
</a-form-item>
<a-form-item label="宝塔接口秘钥" name="btSign">
<a-input
allow-clear
placeholder="请输入宝塔接口秘钥"
v-model:value="form.btSign"
/>
</a-form-item>
<a-form-item label="文章排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="客户ID" name="companyId">
<a-input
allow-clear
placeholder="请输入客户ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="机构id" name="organizationId">
<a-input
allow-clear
placeholder="请输入机构id"
v-model:value="form.organizationId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsServer, updateOaAssetsServer } from '@/api/oa/oaAssetsServer';
import { OaAssetsServer } from '@/api/oa/oaAssetsServer/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsServer | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsServer>({
serverId: undefined,
name: undefined,
code: undefined,
type: undefined,
brand: undefined,
configuration: undefined,
account: undefined,
password: undefined,
brandAccount: undefined,
brandPassword: undefined,
panel: undefined,
panelAccount: undefined,
panelPassword: undefined,
financeAmount: undefined,
financeYears: undefined,
financeRenew: undefined,
financeCustomerName: undefined,
financeCustomerContact: undefined,
financeCustomerPhone: undefined,
customerId: undefined,
customerName: undefined,
openPort: undefined,
content: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
visibility: undefined,
btSign: undefined,
sortNumber: undefined,
comments: undefined,
companyId: undefined,
userId: undefined,
organizationId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsServerId: undefined,
oaAssetsServerName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsServerName: [
{
required: true,
type: 'string',
message: '请填写服务器资产记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsServer : addOaAssetsServer;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,425 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsServerId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsServerEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsServerEdit from './components/oaAssetsServerEdit.vue';
import { pageOaAssetsServer, removeOaAssetsServer, removeBatchOaAssetsServer } from '@/api/oa/oaAssetsServer';
import type { OaAssetsServer, OaAssetsServerParam } from '@/api/oa/oaAssetsServer/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsServer[]>([]);
// 当前编辑数据
const current = ref<OaAssetsServer | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsServer({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '资产ID',
dataIndex: 'serverId',
key: 'serverId',
align: 'center',
width: 90,
},
{
title: '资产名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '资产标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '资产类型',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '服务器厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '服务器配置',
dataIndex: 'configuration',
key: 'configuration',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'account',
key: 'account',
align: 'center',
},
{
title: '初始密码',
dataIndex: 'password',
key: 'password',
align: 'center',
},
{
title: '(阿里云/腾讯云)登录账号',
dataIndex: 'brandAccount',
key: 'brandAccount',
align: 'center',
},
{
title: '(阿里云/腾讯云)登录密码',
dataIndex: 'brandPassword',
key: 'brandPassword',
align: 'center',
},
{
title: '宝塔面板',
dataIndex: 'panel',
key: 'panel',
align: 'center',
},
{
title: '宝塔面板账号',
dataIndex: 'panelAccount',
key: 'panelAccount',
align: 'center',
},
{
title: '宝塔面板密码',
dataIndex: 'panelPassword',
key: 'panelPassword',
align: 'center',
},
{
title: '财务信息-合同金额',
dataIndex: 'financeAmount',
key: 'financeAmount',
align: 'center',
},
{
title: '购买年限',
dataIndex: 'financeYears',
key: 'financeYears',
align: 'center',
},
{
title: '续费金额',
dataIndex: 'financeRenew',
key: 'financeRenew',
align: 'center',
},
{
title: '客户名称',
dataIndex: 'financeCustomerName',
key: 'financeCustomerName',
align: 'center',
},
{
title: '客户联系人',
dataIndex: 'financeCustomerContact',
key: 'financeCustomerContact',
align: 'center',
},
{
title: '客户联系电话',
dataIndex: 'financeCustomerPhone',
key: 'financeCustomerPhone',
align: 'center',
},
{
title: '客户ID',
dataIndex: 'customerId',
key: 'customerId',
align: 'center',
},
{
title: '客户名称',
dataIndex: 'customerName',
key: 'customerName',
align: 'center',
},
{
title: '开放端口',
dataIndex: 'openPort',
key: 'openPort',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '可见性(public,private,protected)',
dataIndex: 'visibility',
key: 'visibility',
align: 'center',
},
{
title: '宝塔接口秘钥',
dataIndex: 'btSign',
key: 'btSign',
align: 'center',
},
{
title: '文章排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '客户ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '机构id',
dataIndex: 'organizationId',
key: 'organizationId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsServerParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsServer) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsServer) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsServer(row.oaAssetsServerId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsServer(selection.value.map((d) => d.oaAssetsServerId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsServer) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsServer'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,300 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑网站管理记录表' : '添加网站管理记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="网站名称" name="name">
<a-input
allow-clear
placeholder="请输入网站名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="域名标识" name="code">
<a-input
allow-clear
placeholder="请输入域名标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="主机型号" name="type">
<a-input
allow-clear
placeholder="请输入主机型号"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="品牌厂商" name="brand">
<a-input
allow-clear
placeholder="请输入品牌厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="初始账号" name="system">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.system"
/>
</a-form-item>
<a-form-item label="价格" name="price">
<a-input
allow-clear
placeholder="请输入价格"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="ssl证书" name="ssl">
<a-input
allow-clear
placeholder="请输入ssl证书"
v-model:value="form.ssl"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsSite, updateOaAssetsSite } from '@/api/oa/oaAssetsSite';
import { OaAssetsSite } from '@/api/oa/oaAssetsSite/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsSite | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsSite>({
siteId: undefined,
name: undefined,
code: undefined,
type: undefined,
brand: undefined,
system: undefined,
price: undefined,
content: undefined,
ssl: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
sortNumber: undefined,
comments: undefined,
userId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsSiteId: undefined,
oaAssetsSiteName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsSiteName: [
{
required: true,
type: 'string',
message: '请填写网站管理记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsSite : addOaAssetsSite;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,317 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsSiteId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsSiteEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsSiteEdit from './components/oaAssetsSiteEdit.vue';
import { pageOaAssetsSite, removeOaAssetsSite, removeBatchOaAssetsSite } from '@/api/oa/oaAssetsSite';
import type { OaAssetsSite, OaAssetsSiteParam } from '@/api/oa/oaAssetsSite/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsSite[]>([]);
// 当前编辑数据
const current = ref<OaAssetsSite | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsSite({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: 'ID',
dataIndex: 'siteId',
key: 'siteId',
align: 'center',
width: 90,
},
{
title: '网站名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '域名标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '主机型号',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '品牌厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'system',
key: 'system',
align: 'center',
},
{
title: '价格',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: 'ssl证书',
dataIndex: 'ssl',
key: 'ssl',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsSiteParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsSite) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsSite) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsSite(row.oaAssetsSiteId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsSite(selection.value.map((d) => d.oaAssetsSiteId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsSite) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsSite'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,201 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑服务器成员管理' : '添加服务器成员管理'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="角色10体验成员 20开发者成员 30管理员 " name="role">
<a-input
allow-clear
placeholder="请输入角色10体验成员 20开发者成员 30管理员 "
v-model:value="form.role"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="应用ID" name="assetsId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.assetsId"
/>
</a-form-item>
<a-form-item label="昵称" name="nickname">
<a-input
allow-clear
placeholder="请输入昵称"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsUser, updateOaAssetsUser } from '@/api/oa/oaAssetsUser';
import { OaAssetsUser } from '@/api/oa/oaAssetsUser/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsUser | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsUser>({
id: undefined,
role: undefined,
userId: undefined,
assetsId: undefined,
nickname: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaAssetsUserId: undefined,
oaAssetsUserName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsUserName: [
{
required: true,
type: 'string',
message: '请填写服务器成员管理名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsUser : addOaAssetsUser;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,245 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsUserId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsUserEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsUserEdit from './components/oaAssetsUserEdit.vue';
import { pageOaAssetsUser, removeOaAssetsUser, removeBatchOaAssetsUser } from '@/api/oa/oaAssetsUser';
import type { OaAssetsUser, OaAssetsUserParam } from '@/api/oa/oaAssetsUser/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsUser[]>([]);
// 当前编辑数据
const current = ref<OaAssetsUser | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsUser({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'id',
key: 'id',
align: 'center',
width: 90,
},
{
title: '角色10体验成员 20开发者成员 30管理员 ',
dataIndex: 'role',
key: 'role',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '应用ID',
dataIndex: 'assetsId',
key: 'assetsId',
align: 'center',
},
{
title: '昵称',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsUserParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsUser) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsUser) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsUser(row.oaAssetsUserId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsUser(selection.value.map((d) => d.oaAssetsUserId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsUser) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsUser'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,300 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑虚拟主机记录表' : '添加虚拟主机记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="域名" name="name">
<a-input
allow-clear
placeholder="请输入域名"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="域名标识" name="code">
<a-input
allow-clear
placeholder="请输入域名标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="主机型号" name="type">
<a-input
allow-clear
placeholder="请输入主机型号"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="品牌厂商" name="brand">
<a-input
allow-clear
placeholder="请输入品牌厂商"
v-model:value="form.brand"
/>
</a-form-item>
<a-form-item label="初始账号" name="system">
<a-input
allow-clear
placeholder="请输入初始账号"
v-model:value="form.system"
/>
</a-form-item>
<a-form-item label="价格" name="price">
<a-input
allow-clear
placeholder="请输入价格"
v-model:value="form.price"
/>
</a-form-item>
<a-form-item label="详情内容" name="content">
<a-input
allow-clear
placeholder="请输入详情内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="ssl证书" name="ssl">
<a-input
allow-clear
placeholder="请输入ssl证书"
v-model:value="form.ssl"
/>
</a-form-item>
<a-form-item label="购买时间" name="startTime">
<a-input
allow-clear
placeholder="请输入购买时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="到期时间" name="endTime">
<a-input
allow-clear
placeholder="请输入到期时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="置顶状态" name="isTop">
<a-input
allow-clear
placeholder="请输入置顶状态"
v-model:value="form.isTop"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="描述" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1冻结" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaAssetsVhost, updateOaAssetsVhost } from '@/api/oa/oaAssetsVhost';
import { OaAssetsVhost } from '@/api/oa/oaAssetsVhost/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaAssetsVhost | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaAssetsVhost>({
vhostId: undefined,
name: undefined,
code: undefined,
type: undefined,
brand: undefined,
system: undefined,
price: undefined,
content: undefined,
ssl: undefined,
startTime: undefined,
endTime: undefined,
isTop: undefined,
sortNumber: undefined,
comments: undefined,
userId: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaAssetsVhostId: undefined,
oaAssetsVhostName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaAssetsVhostName: [
{
required: true,
type: 'string',
message: '请填写虚拟主机记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaAssetsVhost : addOaAssetsVhost;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,317 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaAssetsVhostId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaAssetsVhostEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaAssetsVhostEdit from './components/oaAssetsVhostEdit.vue';
import { pageOaAssetsVhost, removeOaAssetsVhost, removeBatchOaAssetsVhost } from '@/api/oa/oaAssetsVhost';
import type { OaAssetsVhost, OaAssetsVhostParam } from '@/api/oa/oaAssetsVhost/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaAssetsVhost[]>([]);
// 当前编辑数据
const current = ref<OaAssetsVhost | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaAssetsVhost({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: 'ID',
dataIndex: 'vhostId',
key: 'vhostId',
align: 'center',
width: 90,
},
{
title: '域名',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '域名标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '主机型号',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '品牌厂商',
dataIndex: 'brand',
key: 'brand',
align: 'center',
},
{
title: '初始账号',
dataIndex: 'system',
key: 'system',
align: 'center',
},
{
title: '价格',
dataIndex: 'price',
key: 'price',
align: 'center',
},
{
title: '详情内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: 'ssl证书',
dataIndex: 'ssl',
key: 'ssl',
align: 'center',
},
{
title: '购买时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '到期时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '置顶状态',
dataIndex: 'isTop',
key: 'isTop',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '描述',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1冻结',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaAssetsVhostParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaAssetsVhost) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaAssetsVhost) => {
const hide = message.loading('请求中..', 0);
removeOaAssetsVhost(row.oaAssetsVhostId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaAssetsVhost(selection.value.map((d) => d.oaAssetsVhostId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaAssetsVhost) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaAssetsVhost'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,548 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑企业信息' : '添加企业信息'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="企业简称" name="shortName">
<a-input
allow-clear
placeholder="请输入企业简称"
v-model:value="form.shortName"
/>
</a-form-item>
<a-form-item label="企业全称" name="companyName">
<a-input
allow-clear
placeholder="请输入企业全称"
v-model:value="form.companyName"
/>
</a-form-item>
<a-form-item label="企业标识" name="companyCode">
<a-input
allow-clear
placeholder="请输入企业标识"
v-model:value="form.companyCode"
/>
</a-form-item>
<a-form-item label="类型 10企业 20政府单位" name="companyType">
<a-input
allow-clear
placeholder="请输入类型 10企业 20政府单位"
v-model:value="form.companyType"
/>
</a-form-item>
<a-form-item label="企业类型多选" name="companyTypeMultiple">
<a-input
allow-clear
placeholder="请输入企业类型多选"
v-model:value="form.companyTypeMultiple"
/>
</a-form-item>
<a-form-item label="应用标识" name="companyLogo">
<a-input
allow-clear
placeholder="请输入应用标识"
v-model:value="form.companyLogo"
/>
</a-form-item>
<a-form-item label="应用类型" name="appType">
<a-input
allow-clear
placeholder="请输入应用类型"
v-model:value="form.appType"
/>
</a-form-item>
<a-form-item label="绑定域名" name="domain">
<a-input
allow-clear
placeholder="请输入绑定域名"
v-model:value="form.domain"
/>
</a-form-item>
<a-form-item label="联系电话" name="phone">
<a-input
allow-clear
placeholder="请输入联系电话"
v-model:value="form.phone"
/>
</a-form-item>
<a-form-item label="座机电话" name="tel">
<a-input
allow-clear
placeholder="请输入座机电话"
v-model:value="form.tel"
/>
</a-form-item>
<a-form-item label="邮箱" name="email">
<a-input
allow-clear
placeholder="请输入邮箱"
v-model:value="form.email"
/>
</a-form-item>
<a-form-item label="发票抬头" name="invoiceHeader">
<a-input
allow-clear
placeholder="请输入发票抬头"
v-model:value="form.invoiceHeader"
/>
</a-form-item>
<a-form-item label="企业法人" name="businessEntity">
<a-input
allow-clear
placeholder="请输入企业法人"
v-model:value="form.businessEntity"
/>
</a-form-item>
<a-form-item label="服务开始时间" name="startTime">
<a-input
allow-clear
placeholder="请输入服务开始时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="服务到期时间" name="expirationTime">
<a-input
allow-clear
placeholder="请输入服务到期时间"
v-model:value="form.expirationTime"
/>
</a-form-item>
<a-form-item label="应用版本 10体验版 20授权版 30旗舰版" name="version">
<a-input
allow-clear
placeholder="请输入应用版本 10体验版 20授权版 30旗舰版"
v-model:value="form.version"
/>
</a-form-item>
<a-form-item label="成员数量(人数上限)" name="members">
<a-input
allow-clear
placeholder="请输入成员数量(人数上限)"
v-model:value="form.members"
/>
</a-form-item>
<a-form-item label="成员数量(当前)" name="users">
<a-input
allow-clear
placeholder="请输入成员数量(当前)"
v-model:value="form.users"
/>
</a-form-item>
<a-form-item label="行业类型(父级)" name="industryParent">
<a-input
allow-clear
placeholder="请输入行业类型(父级)"
v-model:value="form.industryParent"
/>
</a-form-item>
<a-form-item label="行业类型(子级)" name="industryChild">
<a-input
allow-clear
placeholder="请输入行业类型(子级)"
v-model:value="form.industryChild"
/>
</a-form-item>
<a-form-item label="部门数量" name="departments">
<a-input
allow-clear
placeholder="请输入部门数量"
v-model:value="form.departments"
/>
</a-form-item>
<a-form-item label="存储空间" name="storage">
<a-input
allow-clear
placeholder="请输入存储空间"
v-model:value="form.storage"
/>
</a-form-item>
<a-form-item label="存储空间(上限)" name="storageMax">
<a-input
allow-clear
placeholder="请输入存储空间(上限)"
v-model:value="form.storageMax"
/>
</a-form-item>
<a-form-item label="所在国家" name="country">
<a-input
allow-clear
placeholder="请输入所在国家"
v-model:value="form.country"
/>
</a-form-item>
<a-form-item label="所在省份" name="province">
<a-input
allow-clear
placeholder="请输入所在省份"
v-model:value="form.province"
/>
</a-form-item>
<a-form-item label="所在城市" name="city">
<a-input
allow-clear
placeholder="请输入所在城市"
v-model:value="form.city"
/>
</a-form-item>
<a-form-item label="所在辖区" name="region">
<a-input
allow-clear
placeholder="请输入所在辖区"
v-model:value="form.region"
/>
</a-form-item>
<a-form-item label="街道地址" name="address">
<a-input
allow-clear
placeholder="请输入街道地址"
v-model:value="form.address"
/>
</a-form-item>
<a-form-item label="经度" name="longitude">
<a-input
allow-clear
placeholder="请输入经度"
v-model:value="form.longitude"
/>
</a-form-item>
<a-form-item label="纬度" name="latitude">
<a-input
allow-clear
placeholder="请输入纬度"
v-model:value="form.latitude"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="是否实名认证" name="authentication">
<a-input
allow-clear
placeholder="请输入是否实名认证"
v-model:value="form.authentication"
/>
</a-form-item>
<a-form-item label="企业默认主体" name="authoritative">
<a-input
allow-clear
placeholder="请输入企业默认主体"
v-model:value="form.authoritative"
/>
</a-form-item>
<a-form-item label="request合法域名" name="requestUrl">
<a-input
allow-clear
placeholder="请输入request合法域名"
v-model:value="form.requestUrl"
/>
</a-form-item>
<a-form-item label="socket合法域名" name="socketUrl">
<a-input
allow-clear
placeholder="请输入socket合法域名"
v-model:value="form.socketUrl"
/>
</a-form-item>
<a-form-item label="主控端域名" name="serverUrl">
<a-input
allow-clear
placeholder="请输入主控端域名"
v-model:value="form.serverUrl"
/>
</a-form-item>
<a-form-item label="业务域名" name="modulesUrl">
<a-input
allow-clear
placeholder="请输入业务域名"
v-model:value="form.modulesUrl"
/>
</a-form-item>
<a-form-item label="是否推荐" name="recommend">
<a-input
allow-clear
placeholder="请输入是否推荐"
v-model:value="form.recommend"
/>
</a-form-item>
<a-form-item label="点赞数量" name="likes">
<a-input
allow-clear
placeholder="请输入点赞数量"
v-model:value="form.likes"
/>
</a-form-item>
<a-form-item label="点击数量" name="clicks">
<a-input
allow-clear
placeholder="请输入点击数量"
v-model:value="form.clicks"
/>
</a-form-item>
<a-form-item label="购买数量" name="buys">
<a-input
allow-clear
placeholder="请输入购买数量"
v-model:value="form.buys"
/>
</a-form-item>
<a-form-item label="是否含税, 0不含, 1含" name="isTax">
<a-input
allow-clear
placeholder="请输入是否含税, 0不含, 1含"
v-model:value="form.isTax"
/>
</a-form-item>
<a-form-item label="当前克隆的租户ID" name="planId">
<a-input
allow-clear
placeholder="请输入当前克隆的租户ID"
v-model:value="form.planId"
/>
</a-form-item>
<a-form-item label="状态" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="排序号" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaCompany, updateOaCompany } from '@/api/oa/oaCompany';
import { OaCompany } from '@/api/oa/oaCompany/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaCompany | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaCompany>({
companyId: undefined,
shortName: undefined,
companyName: undefined,
companyCode: undefined,
companyType: undefined,
companyTypeMultiple: undefined,
companyLogo: undefined,
appType: undefined,
domain: undefined,
phone: undefined,
tel: undefined,
email: undefined,
invoiceHeader: undefined,
businessEntity: undefined,
startTime: undefined,
expirationTime: undefined,
version: undefined,
members: undefined,
users: undefined,
industryParent: undefined,
industryChild: undefined,
departments: undefined,
storage: undefined,
storageMax: undefined,
country: undefined,
province: undefined,
city: undefined,
region: undefined,
address: undefined,
longitude: undefined,
latitude: undefined,
comments: undefined,
authentication: undefined,
authoritative: undefined,
requestUrl: undefined,
socketUrl: undefined,
serverUrl: undefined,
modulesUrl: undefined,
recommend: undefined,
likes: undefined,
clicks: undefined,
buys: undefined,
isTax: undefined,
planId: undefined,
status: undefined,
sortNumber: undefined,
userId: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaCompanyId: undefined,
oaCompanyName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaCompanyName: [
{
required: true,
type: 'string',
message: '请填写企业信息名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaCompany : addOaCompany;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,503 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaCompanyId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaCompanyEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaCompanyEdit from './components/oaCompanyEdit.vue';
import { pageOaCompany, removeOaCompany, removeBatchOaCompany } from '@/api/oa/oaCompany';
import type { OaCompany, OaCompanyParam } from '@/api/oa/oaCompany/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaCompany[]>([]);
// 当前编辑数据
const current = ref<OaCompany | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaCompany({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '企业id',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
width: 90,
},
{
title: '企业简称',
dataIndex: 'shortName',
key: 'shortName',
align: 'center',
},
{
title: '企业全称',
dataIndex: 'companyName',
key: 'companyName',
align: 'center',
},
{
title: '企业标识',
dataIndex: 'companyCode',
key: 'companyCode',
align: 'center',
},
{
title: '类型 10企业 20政府单位',
dataIndex: 'companyType',
key: 'companyType',
align: 'center',
},
{
title: '企业类型多选',
dataIndex: 'companyTypeMultiple',
key: 'companyTypeMultiple',
align: 'center',
},
{
title: '应用标识',
dataIndex: 'companyLogo',
key: 'companyLogo',
align: 'center',
},
{
title: '应用类型',
dataIndex: 'appType',
key: 'appType',
align: 'center',
},
{
title: '绑定域名',
dataIndex: 'domain',
key: 'domain',
align: 'center',
},
{
title: '联系电话',
dataIndex: 'phone',
key: 'phone',
align: 'center',
},
{
title: '座机电话',
dataIndex: 'tel',
key: 'tel',
align: 'center',
},
{
title: '邮箱',
dataIndex: 'email',
key: 'email',
align: 'center',
},
{
title: '发票抬头',
dataIndex: 'invoiceHeader',
key: 'invoiceHeader',
align: 'center',
},
{
title: '企业法人',
dataIndex: 'businessEntity',
key: 'businessEntity',
align: 'center',
},
{
title: '服务开始时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '服务到期时间',
dataIndex: 'expirationTime',
key: 'expirationTime',
align: 'center',
},
{
title: '应用版本 10体验版 20授权版 30旗舰版',
dataIndex: 'version',
key: 'version',
align: 'center',
},
{
title: '成员数量(人数上限)',
dataIndex: 'members',
key: 'members',
align: 'center',
},
{
title: '成员数量(当前)',
dataIndex: 'users',
key: 'users',
align: 'center',
},
{
title: '行业类型(父级)',
dataIndex: 'industryParent',
key: 'industryParent',
align: 'center',
},
{
title: '行业类型(子级)',
dataIndex: 'industryChild',
key: 'industryChild',
align: 'center',
},
{
title: '部门数量',
dataIndex: 'departments',
key: 'departments',
align: 'center',
},
{
title: '存储空间',
dataIndex: 'storage',
key: 'storage',
align: 'center',
},
{
title: '存储空间(上限)',
dataIndex: 'storageMax',
key: 'storageMax',
align: 'center',
},
{
title: '所在国家',
dataIndex: 'country',
key: 'country',
align: 'center',
},
{
title: '所在省份',
dataIndex: 'province',
key: 'province',
align: 'center',
},
{
title: '所在城市',
dataIndex: 'city',
key: 'city',
align: 'center',
},
{
title: '所在辖区',
dataIndex: 'region',
key: 'region',
align: 'center',
},
{
title: '街道地址',
dataIndex: 'address',
key: 'address',
align: 'center',
},
{
title: '经度',
dataIndex: 'longitude',
key: 'longitude',
align: 'center',
},
{
title: '纬度',
dataIndex: 'latitude',
key: 'latitude',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '是否实名认证',
dataIndex: 'authentication',
key: 'authentication',
align: 'center',
},
{
title: '企业默认主体',
dataIndex: 'authoritative',
key: 'authoritative',
align: 'center',
},
{
title: 'request合法域名',
dataIndex: 'requestUrl',
key: 'requestUrl',
align: 'center',
},
{
title: 'socket合法域名',
dataIndex: 'socketUrl',
key: 'socketUrl',
align: 'center',
},
{
title: '主控端域名',
dataIndex: 'serverUrl',
key: 'serverUrl',
align: 'center',
},
{
title: '业务域名',
dataIndex: 'modulesUrl',
key: 'modulesUrl',
align: 'center',
},
{
title: '是否推荐',
dataIndex: 'recommend',
key: 'recommend',
align: 'center',
},
{
title: '点赞数量',
dataIndex: 'likes',
key: 'likes',
align: 'center',
},
{
title: '点击数量',
dataIndex: 'clicks',
key: 'clicks',
align: 'center',
},
{
title: '购买数量',
dataIndex: 'buys',
key: 'buys',
align: 'center',
},
{
title: '是否含税, 0不含, 1含',
dataIndex: 'isTax',
key: 'isTax',
align: 'center',
},
{
title: '当前克隆的租户ID',
dataIndex: 'planId',
key: 'planId',
align: 'center',
},
{
title: '状态',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '排序号',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaCompanyParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaCompany) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaCompany) => {
const hide = message.loading('请求中..', 0);
removeOaCompany(row.oaCompanyId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaCompany(selection.value.map((d) => d.oaCompanyId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaCompany) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaCompany'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,212 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑企业参数' : '添加企业参数'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="名称" name="name">
<a-input
allow-clear
placeholder="请输入名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1删除" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaCompanyField, updateOaCompanyField } from '@/api/oa/oaCompanyField';
import { OaCompanyField } from '@/api/oa/oaCompanyField/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaCompanyField | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaCompanyField>({
id: undefined,
companyId: undefined,
name: undefined,
comments: undefined,
userId: undefined,
status: undefined,
sortNumber: undefined,
tenantId: undefined,
createTime: undefined,
oaCompanyFieldId: undefined,
oaCompanyFieldName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaCompanyFieldName: [
{
required: true,
type: 'string',
message: '请填写企业参数名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaCompanyField : addOaCompanyField;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,251 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaCompanyFieldId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaCompanyFieldEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaCompanyFieldEdit from './components/oaCompanyFieldEdit.vue';
import { pageOaCompanyField, removeOaCompanyField, removeBatchOaCompanyField } from '@/api/oa/oaCompanyField';
import type { OaCompanyField, OaCompanyFieldParam } from '@/api/oa/oaCompanyField/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaCompanyField[]>([]);
// 当前编辑数据
const current = ref<OaCompanyField | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaCompanyField({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'id',
key: 'id',
align: 'center',
width: 90,
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '状态, 0正常, 1删除',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaCompanyFieldParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaCompanyField) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaCompanyField) => {
const hide = message.loading('请求中..', 0);
removeOaCompanyField(row.oaCompanyFieldId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaCompanyField(selection.value.map((d) => d.oaCompanyFieldId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaCompanyField) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaCompanyField'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,201 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑成员管理' : '添加成员管理'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="角色10体验成员 20开发者成员 30管理员 " name="role">
<a-input
allow-clear
placeholder="请输入角色10体验成员 20开发者成员 30管理员 "
v-model:value="form.role"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="昵称" name="nickname">
<a-input
allow-clear
placeholder="请输入昵称"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaCompanyUser, updateOaCompanyUser } from '@/api/oa/oaCompanyUser';
import { OaCompanyUser } from '@/api/oa/oaCompanyUser/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaCompanyUser | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaCompanyUser>({
companyUserId: undefined,
role: undefined,
userId: undefined,
companyId: undefined,
nickname: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaCompanyUserId: undefined,
oaCompanyUserName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaCompanyUserName: [
{
required: true,
type: 'string',
message: '请填写成员管理名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaCompanyUser : addOaCompanyUser;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,245 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaCompanyUserId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaCompanyUserEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaCompanyUserEdit from './components/oaCompanyUserEdit.vue';
import { pageOaCompanyUser, removeOaCompanyUser, removeBatchOaCompanyUser } from '@/api/oa/oaCompanyUser';
import type { OaCompanyUser, OaCompanyUserParam } from '@/api/oa/oaCompanyUser/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaCompanyUser[]>([]);
// 当前编辑数据
const current = ref<OaCompanyUser | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaCompanyUser({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'companyUserId',
key: 'companyUserId',
align: 'center',
width: 90,
},
{
title: '角色10体验成员 20开发者成员 30管理员 ',
dataIndex: 'role',
key: 'role',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '昵称',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaCompanyUserParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaCompanyUser) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaCompanyUser) => {
const hide = message.loading('请求中..', 0);
removeOaCompanyUser(row.oaCompanyUserId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaCompanyUser(selection.value.map((d) => d.oaCompanyUserId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaCompanyUser) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaCompanyUser'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,260 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑常用链接' : '添加常用链接'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="链接名称" name="name">
<a-input
allow-clear
placeholder="请输入链接名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="图标" name="icon">
<a-input
allow-clear
placeholder="请输入图标"
v-model:value="form.icon"
/>
</a-form-item>
<a-form-item label="链接地址" name="url">
<a-input
allow-clear
placeholder="请输入链接地址"
v-model:value="form.url"
/>
</a-form-item>
<a-form-item label="链接分类" name="linkType">
<a-input
allow-clear
placeholder="请输入链接分类"
v-model:value="form.linkType"
/>
</a-form-item>
<a-form-item label="应用ID" name="appId">
<a-input
allow-clear
placeholder="请输入应用ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="所属栏目" name="categoryId">
<a-input
allow-clear
placeholder="请输入所属栏目"
v-model:value="form.categoryId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="是否推荐" name="recommend">
<a-input
allow-clear
placeholder="请输入是否推荐"
v-model:value="form.recommend"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="状态, 0正常, 1待确认" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaLink, updateOaLink } from '@/api/oa/oaLink';
import { OaLink } from '@/api/oa/oaLink/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaLink | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaLink>({
id: undefined,
name: undefined,
icon: undefined,
url: undefined,
linkType: undefined,
appId: undefined,
categoryId: undefined,
userId: undefined,
recommend: undefined,
comments: undefined,
sortNumber: undefined,
deleted: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaLinkId: undefined,
oaLinkName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaLinkName: [
{
required: true,
type: 'string',
message: '请填写常用链接名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaLink : addOaLink;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,287 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaLinkId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaLinkEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaLinkEdit from './components/oaLinkEdit.vue';
import { pageOaLink, removeOaLink, removeBatchOaLink } from '@/api/oa/oaLink';
import type { OaLink, OaLinkParam } from '@/api/oa/oaLink/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaLink[]>([]);
// 当前编辑数据
const current = ref<OaLink | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaLink({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'id',
key: 'id',
align: 'center',
width: 90,
},
{
title: '链接名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '图标',
dataIndex: 'icon',
key: 'icon',
align: 'center',
},
{
title: '链接地址',
dataIndex: 'url',
key: 'url',
align: 'center',
},
{
title: '链接分类',
dataIndex: 'linkType',
key: 'linkType',
align: 'center',
},
{
title: '应用ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '所属栏目',
dataIndex: 'categoryId',
key: 'categoryId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '是否推荐',
dataIndex: 'recommend',
key: 'recommend',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '状态, 0正常, 1待确认',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaLinkParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaLink) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaLink) => {
const hide = message.loading('请求中..', 0);
removeOaLink(row.oaLinkId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaLink(selection.value.map((d) => d.oaLinkId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaLink) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaLink'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,340 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑产品记录表' : '添加产品记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="产品名称" name="name">
<a-input
allow-clear
placeholder="请输入产品名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="产品标识" name="code">
<a-input
allow-clear
placeholder="请输入产品标识"
v-model:value="form.code"
/>
</a-form-item>
<a-form-item label="产品详情" name="content">
<a-input
allow-clear
placeholder="请输入产品详情"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="产品类型" name="type">
<a-input
allow-clear
placeholder="请输入产品类型"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="产品图标" name="logo">
<a-input
allow-clear
placeholder="请输入产品图标"
v-model:value="form.logo"
/>
</a-form-item>
<a-form-item label="产品金额" name="money">
<a-input
allow-clear
placeholder="请输入产品金额"
v-model:value="form.money"
/>
</a-form-item>
<a-form-item label="初始销量" name="salesInitial">
<a-input
allow-clear
placeholder="请输入初始销量"
v-model:value="form.salesInitial"
/>
</a-form-item>
<a-form-item label="实际销量" name="salesActual">
<a-input
allow-clear
placeholder="请输入实际销量"
v-model:value="form.salesActual"
/>
</a-form-item>
<a-form-item label="库存总量(包含所有sku)" name="stockTotal">
<a-input
allow-clear
placeholder="请输入库存总量(包含所有sku)"
v-model:value="form.stockTotal"
/>
</a-form-item>
<a-form-item label="背景颜色" name="backgroundColor">
<a-input
allow-clear
placeholder="请输入背景颜色"
v-model:value="form.backgroundColor"
/>
</a-form-item>
<a-form-item label="背景图片" name="backgroundImage">
<a-input
allow-clear
placeholder="请输入背景图片"
v-model:value="form.backgroundImage"
/>
</a-form-item>
<a-form-item label="背景图片(gif)" name="backgroundGif">
<a-input
allow-clear
placeholder="请输入背景图片(gif)"
v-model:value="form.backgroundGif"
/>
</a-form-item>
<a-form-item label="购买链接" name="buyUrl">
<a-input
allow-clear
placeholder="请输入购买链接"
v-model:value="form.buyUrl"
/>
</a-form-item>
<a-form-item label="控制台链接" name="adminUrl">
<a-input
allow-clear
placeholder="请输入控制台链接"
v-model:value="form.adminUrl"
/>
</a-form-item>
<a-form-item label="附件" name="files">
<a-input
allow-clear
placeholder="请输入附件"
v-model:value="form.files"
/>
</a-form-item>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="状态, 0已上架, 1已下架" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaProduct, updateOaProduct } from '@/api/oa/oaProduct';
import { OaProduct } from '@/api/oa/oaProduct/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaProduct | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaProduct>({
productId: undefined,
name: undefined,
code: undefined,
content: undefined,
type: undefined,
logo: undefined,
money: undefined,
salesInitial: undefined,
salesActual: undefined,
stockTotal: undefined,
backgroundColor: undefined,
backgroundImage: undefined,
backgroundGif: undefined,
buyUrl: undefined,
adminUrl: undefined,
files: undefined,
companyId: undefined,
userId: undefined,
sortNumber: undefined,
comments: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaProductId: undefined,
oaProductName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaProductName: [
{
required: true,
type: 'string',
message: '请填写产品记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaProduct : addOaProduct;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,347 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaProductId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaProductEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaProductEdit from './components/oaProductEdit.vue';
import { pageOaProduct, removeOaProduct, removeBatchOaProduct } from '@/api/oa/oaProduct';
import type { OaProduct, OaProductParam } from '@/api/oa/oaProduct/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaProduct[]>([]);
// 当前编辑数据
const current = ref<OaProduct | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaProduct({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '产品ID',
dataIndex: 'productId',
key: 'productId',
align: 'center',
width: 90,
},
{
title: '产品名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '产品标识',
dataIndex: 'code',
key: 'code',
align: 'center',
},
{
title: '产品详情',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '产品类型',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '产品图标',
dataIndex: 'logo',
key: 'logo',
align: 'center',
},
{
title: '产品金额',
dataIndex: 'money',
key: 'money',
align: 'center',
},
{
title: '初始销量',
dataIndex: 'salesInitial',
key: 'salesInitial',
align: 'center',
},
{
title: '实际销量',
dataIndex: 'salesActual',
key: 'salesActual',
align: 'center',
},
{
title: '库存总量(包含所有sku)',
dataIndex: 'stockTotal',
key: 'stockTotal',
align: 'center',
},
{
title: '背景颜色',
dataIndex: 'backgroundColor',
key: 'backgroundColor',
align: 'center',
},
{
title: '背景图片',
dataIndex: 'backgroundImage',
key: 'backgroundImage',
align: 'center',
},
{
title: '背景图片(gif)',
dataIndex: 'backgroundGif',
key: 'backgroundGif',
align: 'center',
},
{
title: '购买链接',
dataIndex: 'buyUrl',
key: 'buyUrl',
align: 'center',
},
{
title: '控制台链接',
dataIndex: 'adminUrl',
key: 'adminUrl',
align: 'center',
},
{
title: '附件',
dataIndex: 'files',
key: 'files',
align: 'center',
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '状态, 0已上架, 1已下架',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaProductParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaProduct) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaProduct) => {
const hide = message.loading('请求中..', 0);
removeOaProduct(row.oaProductId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaProduct(selection.value.map((d) => d.oaProductId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaProduct) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaProduct'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,276 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑产品标签记录表' : '添加产品标签记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="产品ID" name="productId">
<a-input
allow-clear
placeholder="请输入产品ID"
v-model:value="form.productId"
/>
</a-form-item>
<a-form-item label="标签名称" name="name">
<a-input
allow-clear
placeholder="请输入标签名称"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="标签类型" name="type">
<a-input
allow-clear
placeholder="请输入标签类型"
v-model:value="form.type"
/>
</a-form-item>
<a-form-item label="产品标签详情" name="content">
<a-input
allow-clear
placeholder="请输入产品标签详情"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="背景颜色" name="backgroundColor">
<a-input
allow-clear
placeholder="请输入背景颜色"
v-model:value="form.backgroundColor"
/>
</a-form-item>
<a-form-item label="背景图片" name="backgroundImage">
<a-input
allow-clear
placeholder="请输入背景图片"
v-model:value="form.backgroundImage"
/>
</a-form-item>
<a-form-item label="附件" name="files">
<a-input
allow-clear
placeholder="请输入附件"
v-model:value="form.files"
/>
</a-form-item>
<a-form-item label="企业ID" name="companyId">
<a-input
allow-clear
placeholder="请输入企业ID"
v-model:value="form.companyId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="状态, 0已上架, 1已下架" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaProductTabs, updateOaProductTabs } from '@/api/oa/oaProductTabs';
import { OaProductTabs } from '@/api/oa/oaProductTabs/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaProductTabs | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaProductTabs>({
tabId: undefined,
productId: undefined,
name: undefined,
type: undefined,
content: undefined,
backgroundColor: undefined,
backgroundImage: undefined,
files: undefined,
companyId: undefined,
userId: undefined,
sortNumber: undefined,
comments: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaProductTabsId: undefined,
oaProductTabsName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaProductTabsName: [
{
required: true,
type: 'string',
message: '请填写产品标签记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaProductTabs : addOaProductTabs;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,299 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaProductTabsId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaProductTabsEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaProductTabsEdit from './components/oaProductTabsEdit.vue';
import { pageOaProductTabs, removeOaProductTabs, removeBatchOaProductTabs } from '@/api/oa/oaProductTabs';
import type { OaProductTabs, OaProductTabsParam } from '@/api/oa/oaProductTabs/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaProductTabs[]>([]);
// 当前编辑数据
const current = ref<OaProductTabs | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaProductTabs({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '产品标签ID',
dataIndex: 'tabId',
key: 'tabId',
align: 'center',
width: 90,
},
{
title: '产品ID',
dataIndex: 'productId',
key: 'productId',
align: 'center',
},
{
title: '标签名称',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '标签类型',
dataIndex: 'type',
key: 'type',
align: 'center',
},
{
title: '产品标签详情',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '背景颜色',
dataIndex: 'backgroundColor',
key: 'backgroundColor',
align: 'center',
},
{
title: '背景图片',
dataIndex: 'backgroundImage',
key: 'backgroundImage',
align: 'center',
},
{
title: '附件',
dataIndex: 'files',
key: 'files',
align: 'center',
},
{
title: '企业ID',
dataIndex: 'companyId',
key: 'companyId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '状态, 0已上架, 1已下架',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaProductTabsParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaProductTabs) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaProductTabs) => {
const hide = message.loading('请求中..', 0);
removeOaProductTabs(row.oaProductTabsId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaProductTabs(selection.value.map((d) => d.oaProductTabsId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaProductTabs) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaProductTabs'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,420 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑任务记录表' : '添加任务记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="工单类型" name="taskType">
<a-input
allow-clear
placeholder="请输入工单类型"
v-model:value="form.taskType"
/>
</a-form-item>
<a-form-item label="任务内容" name="name">
<a-input
allow-clear
placeholder="请输入任务内容"
v-model:value="form.name"
/>
</a-form-item>
<a-form-item label="问题描述" name="content">
<a-input
allow-clear
placeholder="请输入问题描述"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="工单附件" name="files">
<a-input
allow-clear
placeholder="请输入工单附件"
v-model:value="form.files"
/>
</a-form-item>
<a-form-item label="工单发起人" name="promoter">
<a-input
allow-clear
placeholder="请输入工单发起人"
v-model:value="form.promoter"
/>
</a-form-item>
<a-form-item label="受理人" name="commander">
<a-input
allow-clear
placeholder="请输入受理人"
v-model:value="form.commander"
/>
</a-form-item>
<a-form-item label="工单状态, 0未开始 1已指派 " name="progress">
<a-input
allow-clear
placeholder="请输入工单状态, 0未开始 1已指派 "
v-model:value="form.progress"
/>
</a-form-item>
<a-form-item label="优先级" name="priority">
<a-input
allow-clear
placeholder="请输入优先级"
v-model:value="form.priority"
/>
</a-form-item>
<a-form-item label="品质要求" name="quality">
<a-input
allow-clear
placeholder="请输入品质要求"
v-model:value="form.quality"
/>
</a-form-item>
<a-form-item label="时限(天)" name="day">
<a-input
allow-clear
placeholder="请输入时限(天)"
v-model:value="form.day"
/>
</a-form-item>
<a-form-item label="手机号" name="phone">
<a-input
allow-clear
placeholder="请输入手机号"
v-model:value="form.phone"
/>
</a-form-item>
<a-form-item label="开始时间" name="startTime">
<a-input
allow-clear
placeholder="请输入开始时间"
v-model:value="form.startTime"
/>
</a-form-item>
<a-form-item label="结束时间" name="endTime">
<a-input
allow-clear
placeholder="请输入结束时间"
v-model:value="form.endTime"
/>
</a-form-item>
<a-form-item label="逾期天数" name="overdueDays">
<a-input
allow-clear
placeholder="请输入逾期天数"
v-model:value="form.overdueDays"
/>
</a-form-item>
<a-form-item label="项目ID" name="appId">
<a-input
allow-clear
placeholder="请输入项目ID"
v-model:value="form.appId"
/>
</a-form-item>
<a-form-item label="机构id" name="organizationId">
<a-input
allow-clear
placeholder="请输入机构id"
v-model:value="form.organizationId"
/>
</a-form-item>
<a-form-item label="项目ID" name="projectId">
<a-input
allow-clear
placeholder="请输入项目ID"
v-model:value="form.projectId"
/>
</a-form-item>
<a-form-item label="客户ID" name="customerId">
<a-input
allow-clear
placeholder="请输入客户ID"
v-model:value="form.customerId"
/>
</a-form-item>
<a-form-item label="资产ID" name="assetsId">
<a-input
allow-clear
placeholder="请输入资产ID"
v-model:value="form.assetsId"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="是否已查阅" name="isRead">
<a-input
allow-clear
placeholder="请输入是否已查阅"
v-model:value="form.isRead"
/>
</a-form-item>
<a-form-item label="最后回复人" name="lastReadUser">
<a-input
allow-clear
placeholder="请输入最后回复人"
v-model:value="form.lastReadUser"
/>
</a-form-item>
<a-form-item label="发起人昵称" name="nickname">
<a-input
allow-clear
placeholder="请输入发起人昵称"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="发起人头像" name="avatar">
<a-input
allow-clear
placeholder="请输入发起人头像"
v-model:value="form.avatar"
/>
</a-form-item>
<a-form-item label="最后回复人头像" name="lastAvatar">
<a-input
allow-clear
placeholder="请输入最后回复人头像"
v-model:value="form.lastAvatar"
/>
</a-form-item>
<a-form-item label="最后回复人昵称" name="lastNickname">
<a-input
allow-clear
placeholder="请输入最后回复人昵称"
v-model:value="form.lastNickname"
/>
</a-form-item>
<a-form-item label="订单是否已结算(0未结算 1已结算)" name="isSettled">
<a-input
allow-clear
placeholder="请输入订单是否已结算(0未结算 1已结算)"
v-model:value="form.isSettled"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="状态, 0待处理, 1已完成" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaTask, updateOaTask } from '@/api/oa/oaTask';
import { OaTask } from '@/api/oa/oaTask/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaTask | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaTask>({
taskId: undefined,
taskType: undefined,
name: undefined,
content: undefined,
files: undefined,
promoter: undefined,
commander: undefined,
progress: undefined,
priority: undefined,
quality: undefined,
day: undefined,
phone: undefined,
startTime: undefined,
endTime: undefined,
overdueDays: undefined,
appId: undefined,
organizationId: undefined,
projectId: undefined,
customerId: undefined,
assetsId: undefined,
userId: undefined,
isRead: undefined,
lastReadUser: undefined,
nickname: undefined,
avatar: undefined,
lastAvatar: undefined,
lastNickname: undefined,
isSettled: undefined,
sortNumber: undefined,
comments: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaTaskId: undefined,
oaTaskName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaTaskName: [
{
required: true,
type: 'string',
message: '请填写任务记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaTask : addOaTask;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,407 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaTaskId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaTaskEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaTaskEdit from './components/oaTaskEdit.vue';
import { pageOaTask, removeOaTask, removeBatchOaTask } from '@/api/oa/oaTask';
import type { OaTask, OaTaskParam } from '@/api/oa/oaTask/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaTask[]>([]);
// 当前编辑数据
const current = ref<OaTask | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaTask({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '工单ID',
dataIndex: 'taskId',
key: 'taskId',
align: 'center',
width: 90,
},
{
title: '工单类型',
dataIndex: 'taskType',
key: 'taskType',
align: 'center',
},
{
title: '任务内容',
dataIndex: 'name',
key: 'name',
align: 'center',
},
{
title: '问题描述',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '工单附件',
dataIndex: 'files',
key: 'files',
align: 'center',
},
{
title: '工单发起人',
dataIndex: 'promoter',
key: 'promoter',
align: 'center',
},
{
title: '受理人',
dataIndex: 'commander',
key: 'commander',
align: 'center',
},
{
title: '工单状态, 0未开始 1已指派 ',
dataIndex: 'progress',
key: 'progress',
align: 'center',
},
{
title: '优先级',
dataIndex: 'priority',
key: 'priority',
align: 'center',
},
{
title: '品质要求',
dataIndex: 'quality',
key: 'quality',
align: 'center',
},
{
title: '时限(天)',
dataIndex: 'day',
key: 'day',
align: 'center',
},
{
title: '手机号',
dataIndex: 'phone',
key: 'phone',
align: 'center',
},
{
title: '开始时间',
dataIndex: 'startTime',
key: 'startTime',
align: 'center',
},
{
title: '结束时间',
dataIndex: 'endTime',
key: 'endTime',
align: 'center',
},
{
title: '逾期天数',
dataIndex: 'overdueDays',
key: 'overdueDays',
align: 'center',
},
{
title: '项目ID',
dataIndex: 'appId',
key: 'appId',
align: 'center',
},
{
title: '机构id',
dataIndex: 'organizationId',
key: 'organizationId',
align: 'center',
},
{
title: '项目ID',
dataIndex: 'projectId',
key: 'projectId',
align: 'center',
},
{
title: '客户ID',
dataIndex: 'customerId',
key: 'customerId',
align: 'center',
},
{
title: '资产ID',
dataIndex: 'assetsId',
key: 'assetsId',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '是否已查阅',
dataIndex: 'isRead',
key: 'isRead',
align: 'center',
},
{
title: '最后回复人',
dataIndex: 'lastReadUser',
key: 'lastReadUser',
align: 'center',
},
{
title: '发起人昵称',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '发起人头像',
dataIndex: 'avatar',
key: 'avatar',
align: 'center',
},
{
title: '最后回复人头像',
dataIndex: 'lastAvatar',
key: 'lastAvatar',
align: 'center',
},
{
title: '最后回复人昵称',
dataIndex: 'lastNickname',
key: 'lastNickname',
align: 'center',
},
{
title: '订单是否已结算(0未结算 1已结算)',
dataIndex: 'isSettled',
key: 'isSettled',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '状态, 0待处理, 1已完成',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaTaskParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaTask) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaTask) => {
const hide = message.loading('请求中..', 0);
removeOaTask(row.oaTaskId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaTask(selection.value.map((d) => d.oaTaskId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaTask) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaTask'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,258 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑数据统计' : '添加数据统计'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="待处理数" name="pending">
<a-input
allow-clear
placeholder="请输入待处理数"
v-model:value="form.pending"
/>
</a-form-item>
<a-form-item label="闲置的工单(废弃)" name="unused">
<a-input
allow-clear
placeholder="请输入闲置的工单(废弃)"
v-model:value="form.unused"
/>
</a-form-item>
<a-form-item label="已完成数(废弃)" name="completed">
<a-input
allow-clear
placeholder="请输入已完成数(废弃)"
v-model:value="form.completed"
/>
</a-form-item>
<a-form-item label="今天处理数" name="today">
<a-input
allow-clear
placeholder="请输入今天处理数"
v-model:value="form.today"
/>
</a-form-item>
<a-form-item label="本月处理数" name="month">
<a-input
allow-clear
placeholder="请输入本月处理数"
v-model:value="form.month"
/>
</a-form-item>
<a-form-item label="今年处理数" name="year">
<a-input
allow-clear
placeholder="请输入今年处理数"
v-model:value="form.year"
/>
</a-form-item>
<a-form-item label="总工单数" name="total">
<a-input
allow-clear
placeholder="请输入总工单数"
v-model:value="form.total"
/>
</a-form-item>
<a-form-item label="部门ID" name="organizationId">
<a-input
allow-clear
placeholder="请输入部门ID"
v-model:value="form.organizationId"
/>
</a-form-item>
<a-form-item label="角色ID" name="roleId">
<a-input
allow-clear
placeholder="请输入角色ID"
v-model:value="form.roleId"
/>
</a-form-item>
<a-form-item label="角色标识" name="roleCode">
<a-input
allow-clear
placeholder="请输入角色标识"
v-model:value="form.roleCode"
/>
</a-form-item>
<a-form-item label="更新时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入更新时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaTaskCount, updateOaTaskCount } from '@/api/oa/oaTaskCount';
import { OaTaskCount } from '@/api/oa/oaTaskCount/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaTaskCount | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaTaskCount>({
taskCountId: undefined,
userId: undefined,
pending: undefined,
unused: undefined,
completed: undefined,
today: undefined,
month: undefined,
year: undefined,
total: undefined,
organizationId: undefined,
roleId: undefined,
roleCode: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaTaskCountId: undefined,
oaTaskCountName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaTaskCountName: [
{
required: true,
type: 'string',
message: '请填写数据统计名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaTaskCount : addOaTaskCount;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,287 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaTaskCountId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaTaskCountEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaTaskCountEdit from './components/oaTaskCountEdit.vue';
import { pageOaTaskCount, removeOaTaskCount, removeBatchOaTaskCount } from '@/api/oa/oaTaskCount';
import type { OaTaskCount, OaTaskCountParam } from '@/api/oa/oaTaskCount/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaTaskCount[]>([]);
// 当前编辑数据
const current = ref<OaTaskCount | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaTaskCount({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'taskCountId',
key: 'taskCountId',
align: 'center',
width: 90,
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '待处理数',
dataIndex: 'pending',
key: 'pending',
align: 'center',
},
{
title: '闲置的工单(废弃)',
dataIndex: 'unused',
key: 'unused',
align: 'center',
},
{
title: '已完成数(废弃)',
dataIndex: 'completed',
key: 'completed',
align: 'center',
},
{
title: '今天处理数',
dataIndex: 'today',
key: 'today',
align: 'center',
},
{
title: '本月处理数',
dataIndex: 'month',
key: 'month',
align: 'center',
},
{
title: '今年处理数',
dataIndex: 'year',
key: 'year',
align: 'center',
},
{
title: '总工单数',
dataIndex: 'total',
key: 'total',
align: 'center',
},
{
title: '部门ID',
dataIndex: 'organizationId',
key: 'organizationId',
align: 'center',
},
{
title: '角色ID',
dataIndex: 'roleId',
key: 'roleId',
align: 'center',
},
{
title: '角色标识',
dataIndex: 'roleCode',
key: 'roleCode',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '更新时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaTaskCountParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaTaskCount) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaTaskCount) => {
const hide = message.loading('请求中..', 0);
removeOaTaskCount(row.oaTaskCountId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaTaskCount(selection.value.map((d) => d.oaTaskCountId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaTaskCount) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaTaskCount'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,260 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑工单回复记录表' : '添加工单回复记录表'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="上级id, 0是顶级" name="parentId">
<a-input
allow-clear
placeholder="请输入上级id, 0是顶级"
v-model:value="form.parentId"
/>
</a-form-item>
<a-form-item label="工单ID" name="taskId">
<a-input
allow-clear
placeholder="请输入工单ID"
v-model:value="form.taskId"
/>
</a-form-item>
<a-form-item label="内容" name="content">
<a-input
allow-clear
placeholder="请输入内容"
v-model:value="form.content"
/>
</a-form-item>
<a-form-item label="机密信息" name="confidential">
<a-input
allow-clear
placeholder="请输入机密信息"
v-model:value="form.confidential"
/>
</a-form-item>
<a-form-item label="联系电话" name="phone">
<a-input
allow-clear
placeholder="请输入联系电话"
v-model:value="form.phone"
/>
</a-form-item>
<a-form-item label="工单附件" name="files">
<a-input
allow-clear
placeholder="请输入工单附件"
v-model:value="form.files"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="排序(数字越小越靠前)" name="sortNumber">
<a-input-number
:min="0"
:max="9999"
class="ele-fluid"
placeholder="请输入排序号"
v-model:value="form.sortNumber"
/>
</a-form-item>
<a-form-item label="备注" name="comments">
<a-textarea
:rows="4"
:maxlength="200"
placeholder="请输入描述"
v-model:value="form.comments"
/>
</a-form-item>
<a-form-item label="状态, 0待处理, 1已完成" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
<a-form-item label="是否删除, 0否, 1是" name="deleted">
<a-input
allow-clear
placeholder="请输入是否删除, 0否, 1是"
v-model:value="form.deleted"
/>
</a-form-item>
<a-form-item label="修改时间" name="updateTime">
<a-input
allow-clear
placeholder="请输入修改时间"
v-model:value="form.updateTime"
/>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaTaskRecord, updateOaTaskRecord } from '@/api/oa/oaTaskRecord';
import { OaTaskRecord } from '@/api/oa/oaTaskRecord/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaTaskRecord | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaTaskRecord>({
taskRecordId: undefined,
parentId: undefined,
taskId: undefined,
content: undefined,
confidential: undefined,
phone: undefined,
files: undefined,
userId: undefined,
sortNumber: undefined,
comments: undefined,
status: undefined,
deleted: undefined,
tenantId: undefined,
createTime: undefined,
updateTime: undefined,
oaTaskRecordId: undefined,
oaTaskRecordName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaTaskRecordName: [
{
required: true,
type: 'string',
message: '请填写工单回复记录表名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaTaskRecord : addOaTaskRecord;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,287 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaTaskRecordId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaTaskRecordEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaTaskRecordEdit from './components/oaTaskRecordEdit.vue';
import { pageOaTaskRecord, removeOaTaskRecord, removeBatchOaTaskRecord } from '@/api/oa/oaTaskRecord';
import type { OaTaskRecord, OaTaskRecordParam } from '@/api/oa/oaTaskRecord/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaTaskRecord[]>([]);
// 当前编辑数据
const current = ref<OaTaskRecord | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaTaskRecord({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '回复ID',
dataIndex: 'taskRecordId',
key: 'taskRecordId',
align: 'center',
width: 90,
},
{
title: '上级id, 0是顶级',
dataIndex: 'parentId',
key: 'parentId',
align: 'center',
},
{
title: '工单ID',
dataIndex: 'taskId',
key: 'taskId',
align: 'center',
},
{
title: '内容',
dataIndex: 'content',
key: 'content',
align: 'center',
},
{
title: '机密信息',
dataIndex: 'confidential',
key: 'confidential',
align: 'center',
},
{
title: '联系电话',
dataIndex: 'phone',
key: 'phone',
align: 'center',
},
{
title: '工单附件',
dataIndex: 'files',
key: 'files',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '排序(数字越小越靠前)',
dataIndex: 'sortNumber',
key: 'sortNumber',
align: 'center',
},
{
title: '备注',
dataIndex: 'comments',
key: 'comments',
align: 'center',
},
{
title: '状态, 0待处理, 1已完成',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '是否删除, 0否, 1是',
dataIndex: 'deleted',
key: 'deleted',
align: 'center',
},
{
title: '创建时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '修改时间',
dataIndex: 'updateTime',
key: 'updateTime',
align: 'center',
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaTaskRecordParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaTaskRecord) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaTaskRecord) => {
const hide = message.loading('请求中..', 0);
removeOaTaskRecord(row.oaTaskRecordId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaTaskRecord(selection.value.map((d) => d.oaTaskRecordId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaTaskRecord) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaTaskRecord'
};
</script>
<style lang="less" scoped></style>

View File

@@ -0,0 +1,201 @@
<!-- 编辑弹窗 -->
<template>
<ele-modal
:width="800"
:visible="visible"
:maskClosable="false"
:maxable="maxable"
:title="isUpdate ? '编辑工单成员' : '添加工单成员'"
:body-style="{ paddingBottom: '28px' }"
@update:visible="updateVisible"
@ok="save"
>
<a-form
ref="formRef"
:model="form"
:rules="rules"
:label-col="styleResponsive ? { md: 4, sm: 5, xs: 24 } : { flex: '90px' }"
:wrapper-col="
styleResponsive ? { md: 19, sm: 19, xs: 24 } : { flex: '1' }
"
>
<a-form-item label="角色10体验成员 20开发者成员 30管理员 " name="role">
<a-input
allow-clear
placeholder="请输入角色10体验成员 20开发者成员 30管理员 "
v-model:value="form.role"
/>
</a-form-item>
<a-form-item label="用户ID" name="userId">
<a-input
allow-clear
placeholder="请输入用户ID"
v-model:value="form.userId"
/>
</a-form-item>
<a-form-item label="工单ID" name="taskId">
<a-input
allow-clear
placeholder="请输入工单ID"
v-model:value="form.taskId"
/>
</a-form-item>
<a-form-item label="昵称" name="nickname">
<a-input
allow-clear
placeholder="请输入昵称"
v-model:value="form.nickname"
/>
</a-form-item>
<a-form-item label="状态, 0待处理, 1已完成" name="status">
<a-radio-group v-model:value="form.status">
<a-radio :value="0">显示</a-radio>
<a-radio :value="1">隐藏</a-radio>
</a-radio-group>
</a-form-item>
</a-form>
</ele-modal>
</template>
<script lang="ts" setup>
import { ref, reactive, watch } from 'vue';
import { Form, message } from 'ant-design-vue';
import { assignObject, uuid } from 'ele-admin-pro';
import { addOaTaskUser, updateOaTaskUser } from '@/api/oa/oaTaskUser';
import { OaTaskUser } from '@/api/oa/oaTaskUser/model';
import { useThemeStore } from '@/store/modules/theme';
import { storeToRefs } from 'pinia';
import { ItemType } from 'ele-admin-pro/es/ele-image-upload/types';
import { FormInstance } from 'ant-design-vue/es/form';
import { FileRecord } from '@/api/system/file/model';
// 是否是修改
const isUpdate = ref(false);
const useForm = Form.useForm;
// 是否开启响应式布局
const themeStore = useThemeStore();
const { styleResponsive } = storeToRefs(themeStore);
const props = defineProps<{
// 弹窗是否打开
visible: boolean;
// 修改回显的数据
data?: OaTaskUser | null;
}>();
const emit = defineEmits<{
(e: 'done'): void;
(e: 'update:visible', visible: boolean): void;
}>();
// 提交状态
const loading = ref(false);
// 是否显示最大化切换按钮
const maxable = ref(true);
// 表格选中数据
const formRef = ref<FormInstance | null>(null);
const images = ref<ItemType[]>([]);
// 用户信息
const form = reactive<OaTaskUser>({
taskUserId: undefined,
role: undefined,
userId: undefined,
taskId: undefined,
nickname: undefined,
status: undefined,
tenantId: undefined,
createTime: undefined,
oaTaskUserId: undefined,
oaTaskUserName: '',
status: 0,
comments: '',
sortNumber: 100
});
/* 更新visible */
const updateVisible = (value: boolean) => {
emit('update:visible', value);
};
// 表单验证规则
const rules = reactive({
oaTaskUserName: [
{
required: true,
type: 'string',
message: '请填写工单成员名称',
trigger: 'blur'
}
]
});
const chooseImage = (data: FileRecord) => {
images.value.push({
uid: data.id,
url: data.path,
status: 'done'
});
form.image = data.path;
};
const onDeleteItem = (index: number) => {
images.value.splice(index, 1);
form.image = '';
};
const { resetFields } = useForm(form, rules);
/* 保存编辑 */
const save = () => {
if (!formRef.value) {
return;
}
formRef.value
.validate()
.then(() => {
loading.value = true;
const formData = {
...form
};
const saveOrUpdate = isUpdate.value ? updateOaTaskUser : addOaTaskUser;
saveOrUpdate(formData)
.then((msg) => {
loading.value = false;
message.success(msg);
updateVisible(false);
emit('done');
})
.catch((e) => {
loading.value = false;
message.error(e.message);
});
})
.catch(() => {});
};
watch(
() => props.visible,
(visible) => {
if (visible) {
images.value = [];
if (props.data) {
assignObject(form, props.data);
if(props.data.image){
images.value.push({
uid: uuid(),
url: props.data.image,
status: 'done'
})
}
isUpdate.value = true;
} else {
isUpdate.value = false;
}
} else {
resetFields();
}
},
{ immediate: true }
);
</script>

View File

@@ -0,0 +1,42 @@
<!-- 搜索表单 -->
<template>
<a-space :size="10" style="flex-wrap: wrap">
<a-button type="primary" class="ele-btn-icon" @click="add">
<template #icon>
<PlusOutlined />
</template>
<span>添加</span>
</a-button>
</a-space>
</template>
<script lang="ts" setup>
import { PlusOutlined } from '@ant-design/icons-vue';
import type { GradeParam } from '@/api/user/grade/model';
import { watch } from 'vue';
const props = withDefaults(
defineProps<{
// 选中的角色
selection?: [];
}>(),
{}
);
const emit = defineEmits<{
(e: 'search', where?: GradeParam): void;
(e: 'add'): void;
(e: 'remove'): void;
(e: 'batchMove'): void;
}>();
// 新增
const add = () => {
emit('add');
};
watch(
() => props.selection,
() => {}
);
</script>

View File

@@ -0,0 +1,245 @@
<template>
<div class="page">
<div class="ele-body">
<a-card :bordered="false" :body-style="{ padding: '16px' }">
<ele-pro-table
ref="tableRef"
row-key="oaTaskUserId"
:columns="columns"
:datasource="datasource"
:customRow="customRow"
tool-class="ele-toolbar-form"
class="sys-org-table"
>
<template #toolbar>
<search
@search="reload"
:selection="selection"
@add="openEdit"
@remove="removeBatch"
@batchMove="openMove"
/>
</template>
<template #bodyCell="{ column, record }">
<template v-if="column.key === 'image'">
<a-image :src="record.image" :width="50" />
</template>
<template v-if="column.key === 'status'">
<a-tag v-if="record.status === 0" color="green">显示</a-tag>
<a-tag v-if="record.status === 1" color="red">隐藏</a-tag>
</template>
<template v-if="column.key === 'action'">
<a-space>
<a @click="openEdit(record)">修改</a>
<a-divider type="vertical" />
<a-popconfirm
title="确定要删除此记录吗?"
@confirm="remove(record)"
>
<a class="ele-text-danger">删除</a>
</a-popconfirm>
</a-space>
</template>
</template>
</ele-pro-table>
</a-card>
<!-- 编辑弹窗 -->
<OaTaskUserEdit v-model:visible="showEdit" :data="current" @done="reload" />
</div>
</div>
</template>
<script lang="ts" setup>
import { createVNode, ref } from 'vue';
import { message, Modal } from 'ant-design-vue';
import { ExclamationCircleOutlined } from '@ant-design/icons-vue';
import type { EleProTable } from 'ele-admin-pro';
import { toDateString } from 'ele-admin-pro';
import type {
DatasourceFunction,
ColumnItem
} from 'ele-admin-pro/es/ele-pro-table/types';
import Search from './components/search.vue';
import OaTaskUserEdit from './components/oaTaskUserEdit.vue';
import { pageOaTaskUser, removeOaTaskUser, removeBatchOaTaskUser } from '@/api/oa/oaTaskUser';
import type { OaTaskUser, OaTaskUserParam } from '@/api/oa/oaTaskUser/model';
// 表格实例
const tableRef = ref<InstanceType<typeof EleProTable> | null>(null);
// 表格选中数据
const selection = ref<OaTaskUser[]>([]);
// 当前编辑数据
const current = ref<OaTaskUser | null>(null);
// 是否显示编辑弹窗
const showEdit = ref(false);
// 是否显示批量移动弹窗
const showMove = ref(false);
// 加载状态
const loading = ref(true);
// 表格数据源
const datasource: DatasourceFunction = ({
page,
limit,
where,
orders,
filters
}) => {
if (filters) {
where.status = filters.status;
}
return pageOaTaskUser({
...where,
...orders,
page,
limit
});
};
// 表格列配置
const columns = ref<ColumnItem[]>([
{
title: '自增ID',
dataIndex: 'taskUserId',
key: 'taskUserId',
align: 'center',
width: 90,
},
{
title: '角色10体验成员 20开发者成员 30管理员 ',
dataIndex: 'role',
key: 'role',
align: 'center',
},
{
title: '用户ID',
dataIndex: 'userId',
key: 'userId',
align: 'center',
},
{
title: '工单ID',
dataIndex: 'taskId',
key: 'taskId',
align: 'center',
},
{
title: '昵称',
dataIndex: 'nickname',
key: 'nickname',
align: 'center',
},
{
title: '状态, 0待处理, 1已完成',
dataIndex: 'status',
key: 'status',
align: 'center',
},
{
title: '加入时间',
dataIndex: 'createTime',
key: 'createTime',
align: 'center',
sorter: true,
ellipsis: true,
customRender: ({ text }) => toDateString(text, 'yyyy-MM-dd')
},
{
title: '操作',
key: 'action',
width: 180,
fixed: 'right',
align: 'center',
hideInSetting: true
}
]);
/* 搜索 */
const reload = (where?: OaTaskUserParam) => {
selection.value = [];
tableRef?.value?.reload({ where: where });
};
/* 打开编辑弹窗 */
const openEdit = (row?: OaTaskUser) => {
current.value = row ?? null;
showEdit.value = true;
};
/* 打开批量移动弹窗 */
const openMove = () => {
showMove.value = true;
};
/* 删除单个 */
const remove = (row: OaTaskUser) => {
const hide = message.loading('请求中..', 0);
removeOaTaskUser(row.oaTaskUserId)
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
};
/* 批量删除 */
const removeBatch = () => {
if (!selection.value.length) {
message.error('请至少选择一条数据');
return;
}
Modal.confirm({
title: '提示',
content: '确定要删除选中的记录吗?',
icon: createVNode(ExclamationCircleOutlined),
maskClosable: true,
onOk: () => {
const hide = message.loading('请求中..', 0);
removeBatchOaTaskUser(selection.value.map((d) => d.oaTaskUserId))
.then((msg) => {
hide();
message.success(msg);
reload();
})
.catch((e) => {
hide();
message.error(e.message);
});
}
});
};
/* 查询 */
const query = () => {
loading.value = true;
};
/* 自定义行属性 */
const customRow = (record: OaTaskUser) => {
return {
// 行点击事件
onClick: () => {
// console.log(record);
},
// 行双击事件
onDblclick: () => {
openEdit(record);
}
};
};
query();
</script>
<script lang="ts">
export default {
name: 'OaTaskUser'
};
</script>
<style lang="less" scoped></style>