!125 ♥️发布 vue 版本 5.2.0 与 cloud 版本 2.2.0

Merge pull request !125 from 疯狂的狮子Li/dev
This commit is contained in:
疯狂的狮子Li 2024-06-20 02:15:41 +00:00 committed by Gitee
commit abb4f543b9
No known key found for this signature in database
GPG Key ID: 173E9B9CA92EEF8F
18 changed files with 93 additions and 85 deletions

View File

@ -11,7 +11,7 @@ VITE_APP_BASE_API = '/dev-api'
VITE_APP_CONTEXT_PATH = '/' VITE_APP_CONTEXT_PATH = '/'
# 监控地址 # 监控地址
VITE_APP_MONITRO_ADMIN = 'http://localhost:9090/admin/applications' VITE_APP_MONITOR_ADMIN = 'http://localhost:9090/admin/applications'
# SnailJob 控制台地址 # SnailJob 控制台地址
VITE_APP_SNAILJOB_ADMIN = 'http://localhost:8800/snail-job' VITE_APP_SNAILJOB_ADMIN = 'http://localhost:8800/snail-job'

View File

@ -8,7 +8,7 @@ VITE_APP_ENV = 'production'
VITE_APP_CONTEXT_PATH = '/' VITE_APP_CONTEXT_PATH = '/'
# 监控地址 # 监控地址
VITE_APP_MONITRO_ADMIN = '/admin/applications' VITE_APP_MONITOR_ADMIN = '/admin/applications'
# SnailJob 控制台地址 # SnailJob 控制台地址
VITE_APP_SNAILJOB_ADMIN = '/snail-job' VITE_APP_SNAILJOB_ADMIN = '/snail-job'

View File

@ -1,6 +1,6 @@
{ {
"name": "ruoyi-vue-plus", "name": "ruoyi-vue-plus",
"version": "5.2.0-BETA", "version": "5.2.0",
"description": "RuoYi-Vue-Plus多租户管理系统", "description": "RuoYi-Vue-Plus多租户管理系统",
"author": "LionLi", "author": "LionLi",
"license": "MIT", "license": "MIT",
@ -30,7 +30,7 @@
"diagram-js": "12.3.0", "diagram-js": "12.3.0",
"didi": "9.0.2", "didi": "9.0.2",
"echarts": "5.5.0", "echarts": "5.5.0",
"element-plus": "2.7.2", "element-plus": "2.7.5",
"file-saver": "2.0.5", "file-saver": "2.0.5",
"fuse.js": "7.0.0", "fuse.js": "7.0.0",
"highlight.js": "11.9.0", "highlight.js": "11.9.0",

View File

@ -1,10 +1,14 @@
import request from '@/utils/request'; import request from '@/utils/request';
// 绑定账号 // 绑定账号
export function authBinding(source: string) { export function authBinding(source: string, tenantId: string) {
return request({ return request({
url: '/auth/binding/' + source, url: '/auth/binding/' + source,
method: 'get' method: 'get',
params: {
tenantId: tenantId,
domain: window.location.host
}
}); });
} }

View File

@ -26,6 +26,7 @@ export interface UserQuery extends PageQuery {
*/ */
export interface UserVO extends BaseEntity { export interface UserVO extends BaseEntity {
userId: string | number; userId: string | number;
tenantId: string;
deptId: number; deptId: number;
userName: string; userName: string;
nickName: string; nickName: string;

View File

@ -53,7 +53,7 @@ export const getHistoryList = (businessKey: string): AxiosPromise<Record<string,
* @param businessKey id * @param businessKey id
* @returns * @returns
*/ */
export const getHistoryRecord = (businessKey: string) => { export const getHistoryRecord = (businessKey: string | number) => {
return request({ return request({
url: `/workflow/processInstance/getHistoryRecord/${businessKey}`, url: `/workflow/processInstance/getHistoryRecord/${businessKey}`,
method: 'get' method: 'get'

View File

@ -281,12 +281,11 @@ const singleUserSelectRef = ref<InstanceType<typeof UserSelect>>();
const roleSelectRef = ref<InstanceType<typeof RoleSelect>>(); const roleSelectRef = ref<InstanceType<typeof RoleSelect>>();
const dueDateRef = ref<InstanceType<typeof DueDate>>(); const dueDateRef = ref<InstanceType<typeof DueDate>>();
const isMultiple = ref(true);
const openUserSelect = () => { const openUserSelect = () => {
userSelectRef.value.open(); userSelectRef.value.open();
}; };
const openSingleUserSelect = () => { const openSingleUserSelect = () => {
if (formData.value.assignee.includes('$')) { if (formData.value.assignee?.includes('$')) {
formData.value.assignee = ''; formData.value.assignee = '';
} }
singleUserSelectRef.value.open(); singleUserSelectRef.value.open();

View File

@ -67,7 +67,7 @@ const tabActiveName = ref('bpmn');
const bpmnViewRef = ref<BpmnView>(); const bpmnViewRef = ref<BpmnView>();
// //
const init = async (businessKey: string) => { const init = async (businessKey: string | number) => {
visible.value = true; visible.value = true;
loading.value = true; loading.value = true;
tabActiveName.value = 'bpmn'; tabActiveName.value = 'bpmn';

View File

@ -1,22 +1,28 @@
<template> <template>
<el-dialog v-model="visible" draggable :title="title" :width="width" :height="height" append-to-body <el-dialog v-model="visible" draggable :title="title" :width="width" :height="height" append-to-body :close-on-click-modal="false">
:close-on-click-modal="false"> <div v-if="multiInstance === 'add'" class="p-2">
<div class="p-2" v-if="multiInstance === 'add'">
<el-row :gutter="20"> <el-row :gutter="20">
<!-- 部门树 --> <!-- 部门树 -->
<el-col :lg="4" :xs="24" style=""> <el-col :lg="4" :xs="24" style="">
<el-card shadow="hover"> <el-card shadow="hover">
<el-input v-model="deptName" placeholder="请输入部门名称" prefix-icon="Search" clearable /> <el-input v-model="deptName" placeholder="请输入部门名称" prefix-icon="Search" clearable />
<el-tree class="mt-2" ref="deptTreeRef" node-key="id" :data="deptOptions" <el-tree
:props="{ label: 'label', children: 'children' }" :expand-on-click-node="false" ref="deptTreeRef"
:filter-node-method="filterNode" highlight-current default-expand-all class="mt-2"
@node-click="handleNodeClick"></el-tree> node-key="id"
:data="deptOptions"
:props="{ label: 'label', children: 'children' }"
:expand-on-click-node="false"
:filter-node-method="filterNode"
highlight-current
default-expand-all
@node-click="handleNodeClick"
></el-tree>
</el-card> </el-card>
</el-col> </el-col>
<el-col :lg="20" :xs="24"> <el-col :lg="20" :xs="24">
<transition :enter-active-class="proxy?.animate.searchAnimate.enter" <transition :enter-active-class="proxy?.animate.searchAnimate.enter" :leave-active-class="proxy?.animate.searchAnimate.leave">
:leave-active-class="proxy?.animate.searchAnimate.leave"> <div v-show="showSearch" class="search">
<div class="search" v-show="showSearch">
<el-form ref="queryFormRef" :model="queryParams" :inline="true"> <el-form ref="queryFormRef" :model="queryParams" :inline="true">
<el-form-item label="用户名称" prop="userName"> <el-form-item label="用户名称" prop="userName">
<el-input v-model="queryParams.userName" placeholder="请输入用户名称" clearable @keyup.enter="handleQuery" /> <el-input v-model="queryParams.userName" placeholder="请输入用户名称" clearable @keyup.enter="handleQuery" />
@ -25,8 +31,8 @@
<el-input v-model="queryParams.phonenumber" placeholder="请输入手机号码" clearable @keyup.enter="handleQuery" /> <el-input v-model="queryParams.phonenumber" placeholder="请输入手机号码" clearable @keyup.enter="handleQuery" />
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<el-button type="primary" @click="handleQuery" icon="Search">搜索</el-button> <el-button type="primary" icon="Search" @click="handleQuery">搜索</el-button>
<el-button @click="resetQuery" icon="Refresh">重置</el-button> <el-button icon="Refresh" @click="resetQuery">重置</el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
</div> </div>
@ -35,17 +41,16 @@
<el-card shadow="hover"> <el-card shadow="hover">
<template #header> <template #header>
<el-row :gutter="10"> <el-row :gutter="10">
<right-toolbar v-model:showSearch="showSearch" @queryTable="handleQuery" :search="true"></right-toolbar> <right-toolbar v-model:showSearch="showSearch" :search="true" @query-table="handleQuery"></right-toolbar>
</el-row> </el-row>
</template> </template>
<el-table v-loading="loading" :data="userList" ref="multipleTableRef" row-key="userId" <el-table ref="multipleTableRef" v-loading="loading" :data="userList" row-key="userId" @selection-change="handleSelectionChange">
@selection-change="handleSelectionChange">
<el-table-column type="selection" width="50" align="center" /> <el-table-column type="selection" width="50" align="center" />
<el-table-column label="用户编号" align="center" key="userId" prop="userId" /> <el-table-column key="userId" label="用户编号" align="center" prop="userId" />
<el-table-column label="用户名称" align="center" key="userName" prop="userName" :show-overflow-tooltip="true" /> <el-table-column key="userName" label="用户名称" align="center" prop="userName" :show-overflow-tooltip="true" />
<el-table-column label="用户昵称" align="center" key="nickName" prop="nickName" :show-overflow-tooltip="true" /> <el-table-column key="nickName" label="用户昵称" align="center" prop="nickName" :show-overflow-tooltip="true" />
<el-table-column label="手机号码" align="center" key="phonenumber" prop="phonenumber" width="120" /> <el-table-column key="phonenumber" label="手机号码" align="center" prop="phonenumber" width="120" />
<el-table-column label="创建时间" align="center" prop="createTime" width="160"> <el-table-column label="创建时间" align="center" prop="createTime" width="160">
<template #default="scope"> <template #default="scope">
<span>{{ scope.row.createTime }}</span> <span>{{ scope.row.createTime }}</span>
@ -53,18 +58,23 @@
</el-table-column> </el-table-column>
</el-table> </el-table>
<pagination v-show="total > 0" :total="total" v-model:page="queryParams.pageNum" <pagination
v-model:limit="queryParams.pageSize" @pagination="handleQuery" /> v-show="total > 0"
v-model:page="queryParams.pageNum"
v-model:limit="queryParams.pageSize"
:total="total"
@pagination="handleQuery"
/>
</el-card> </el-card>
<el-card shadow="hover"> <el-card shadow="hover">
<el-tag v-for="(user, index) in chooseUserList" :key="user.userId" style="margin:2px" closable <el-tag v-for="(user, index) in chooseUserList" :key="user.userId" style="margin: 2px" closable @close="handleCloseTag(user, index)"
@close="handleCloseTag(user, index)">{{ user.userName }} >{{ user.userName }}
</el-tag> </el-tag>
</el-card> </el-card>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
<div class="p-2" v-if="multiInstance === 'delete'"> <div v-if="multiInstance === 'delete'" class="p-2">
<el-table v-loading="loading" :data="taskList" @selection-change="handleTaskSelection"> <el-table v-loading="loading" :data="taskList" @selection-change="handleTaskSelection">
<el-table-column type="selection" width="55" /> <el-table-column type="selection" width="55" />
<el-table-column prop="name" label="任务名称" /> <el-table-column prop="name" label="任务名称" />
@ -285,7 +295,7 @@ const handleNodeClick = (data: DeptVO) => {
//tag //tag
const handleCloseTag = (user: UserVO, index: any) => { const handleCloseTag = (user: UserVO, index: any) => {
if (multipleTableRef.value.selection && multipleTableRef.value.selection.length > 0) { if (multipleTableRef.value.selection && multipleTableRef.value.selection.length > 0) {
multipleTableRef.value.selection.forEach((u: UserVO, i: Number) => { multipleTableRef.value.selection.forEach((u: UserVO, i: number) => {
if (user.userId === u.userId) { if (user.userId === u.userId) {
multipleTableRef.value.selection.splice(i, 1); multipleTableRef.value.selection.splice(i, 1);
} }

View File

@ -17,7 +17,9 @@ const loading = ref(true);
const code = route.query.code as string; const code = route.query.code as string;
const state = route.query.state as string; const state = route.query.state as string;
const source = route.query.source as string; const source = route.query.source as string;
const tenantId = localStorage.getItem('tenantId') ? (localStorage.getItem('tenantId') as string) : '000000'; const stateJson = JSON.parse(atob(state));
const tenantId = stateJson.tenantId as string ? stateJson.tenantId as string : '000000';
const domain = stateJson.domain as string;
const processResponse = async (res: any) => { const processResponse = async (res: any) => {
if (res.code !== 200) { if (res.code !== 200) {
@ -60,12 +62,21 @@ const loginByCode = async (data: LoginData) => {
}; };
const init = async () => { const init = async () => {
//
let host = window.location.host;
if (domain !== host) {
let urlFull = new URL(window.location.href);
urlFull.host = domain;
window.location.href = urlFull.toString();
return;
}
const data: LoginData = { const data: LoginData = {
socialCode: code, socialCode: code,
socialState: state, socialState: state,
tenantId: tenantId, tenantId: tenantId,
source: source, source: source,
clientId: 'e5cd7e4891bf95d1d19206ce24a7b32e', clientId: import.meta.env.VITE_APP_CLIENT_ID,
grantType: 'social' grantType: 'social'
}; };

View File

@ -10,6 +10,7 @@ export const useUserStore = defineStore('user', () => {
const name = ref(''); const name = ref('');
const nickname = ref(''); const nickname = ref('');
const userId = ref<string | number>(''); const userId = ref<string | number>('');
const tenantId = ref<string>('');
const avatar = ref(''); const avatar = ref('');
const roles = ref<Array<string>>([]); // 用户角色编码集合 → 判断路由权限 const roles = ref<Array<string>>([]); // 用户角色编码集合 → 判断路由权限
const permissions = ref<Array<string>>([]); // 用户权限编码集合 → 判断按钮权限 const permissions = ref<Array<string>>([]); // 用户权限编码集合 → 判断按钮权限
@ -49,6 +50,7 @@ export const useUserStore = defineStore('user', () => {
nickname.value = user.nickName; nickname.value = user.nickName;
avatar.value = profile; avatar.value = profile;
userId.value = user.userId; userId.value = user.userId;
tenantId.value = user.tenantId;
return Promise.resolve(); return Promise.resolve();
} }
return Promise.reject(err); return Promise.reject(err);
@ -69,6 +71,7 @@ export const useUserStore = defineStore('user', () => {
return { return {
userId, userId,
tenantId,
token, token,
nickname, nickname,
avatar, avatar,

5
src/types/env.d.ts vendored
View File

@ -11,9 +11,8 @@ interface ImportMetaEnv {
VITE_APP_BASE_API: string; VITE_APP_BASE_API: string;
VITE_APP_BASE_URL: string; VITE_APP_BASE_URL: string;
VITE_APP_CONTEXT_PATH: string; VITE_APP_CONTEXT_PATH: string;
VITE_APP_MONITRO_ADMIN: string; VITE_APP_MONITOR_ADMIN: string;
VITE_APP_POWERJOB_ADMIN: string; VITE_APP_SNAILJOB_ADMIN: string;
VITE_APP_EASYRETRY_ADMIN: string;
VITE_APP_ENV: string; VITE_APP_ENV: string;
VITE_APP_ENCRYPT: string; VITE_APP_ENCRYPT: string;
VITE_APP_RSA_PUBLIC_KEY: string; VITE_APP_RSA_PUBLIC_KEY: string;

View File

@ -33,7 +33,7 @@
* 部署方式 Docker 容器编排 一键部署业务集群<br /> * 部署方式 Docker 容器编排 一键部署业务集群<br />
* 国际化 SpringMessage Spring标准国际化方案<br /> * 国际化 SpringMessage Spring标准国际化方案<br />
</p> </p>
<p><b>当前版本:</b> <span>v5.2.0-BETA</span></p> <p><b>当前版本:</b> <span>v5.2.0</span></p>
<p> <p>
<el-tag type="danger">&yen;免费开源</el-tag> <el-tag type="danger">&yen;免费开源</el-tag>
</p> </p>
@ -77,7 +77,7 @@
* 分布式监控 PrometheusGrafana 全方位性能监控<br /> * 分布式监控 PrometheusGrafana 全方位性能监控<br />
* 其余与 Vue 版本一致<br /> * 其余与 Vue 版本一致<br />
</p> </p>
<p><b>当前版本:</b> <span>v2.2.0-BETA</span></p> <p><b>当前版本:</b> <span>v2.2.0</span></p>
<p> <p>
<el-tag type="danger">&yen;免费开源</el-tag> <el-tag type="danger">&yen;免费开源</el-tag>
</p> </p>

View File

@ -186,20 +186,12 @@ const initTenantList = async () => {
} }
}; };
//
watch(
() => loginForm.value.tenantId,
() => {
localStorage.setItem('tenantId', String(loginForm.value.tenantId));
}
);
/** /**
* 第三方登录 * 第三方登录
* @param type * @param type
*/ */
const doSocialLogin = (type: string) => { const doSocialLogin = (type: string) => {
authBinding(type).then((res: any) => { authBinding(type, loginForm.value.tenantId).then((res: any) => {
if (res.code === HttpStatus.SUCCESS) { if (res.code === HttpStatus.SUCCESS) {
// //
window.location.href = res.data; window.location.href = res.data;

View File

@ -5,5 +5,5 @@
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
const url = ref(import.meta.env.VITE_APP_MONITRO_ADMIN); const url = ref(import.meta.env.VITE_APP_MONITOR_ADMIN);
</script> </script>

View File

@ -58,6 +58,7 @@
<script lang="ts" setup> <script lang="ts" setup>
import { authUnlock, authBinding } from '@/api/system/social/auth'; import { authUnlock, authBinding } from '@/api/system/social/auth';
import { propTypes } from '@/utils/propTypes'; import { propTypes } from '@/utils/propTypes';
import useUserStore from "@/store/modules/user";
const { proxy } = getCurrentInstance() as ComponentInternalInstance; const { proxy } = getCurrentInstance() as ComponentInternalInstance;
@ -83,7 +84,7 @@ const unlockAuth = (row: any) => {
}; };
const authUrl = (source: string) => { const authUrl = (source: string) => {
authBinding(source).then((res: any) => { authBinding(source, useUserStore().tenantId).then((res: any) => {
if (res.code === 200) { if (res.code === 200) {
window.location.href = res.data; window.location.href = res.data;
} else { } else {

View File

@ -3,34 +3,10 @@
<el-card shadow="never"> <el-card shadow="never">
<div style="display: flex; justify-content: space-between"> <div style="display: flex; justify-content: space-between">
<div> <div>
<el-button <el-button v-if="submitButtonShow" :loading="buttonLoading" type="info" @click="submitForm('draft')">暂存</el-button>
v-if=" <el-button v-if="submitButtonShow" :loading="buttonLoading" type="primary" @click="submitForm('submit')"> </el-button>
routeParams.type === 'add' || <el-button v-if="approvalButtonShow" :loading="buttonLoading" type="primary" @click="approvalVerifyOpen">审批</el-button>
(routeParams.type === 'update' && form.status && (form.status === 'draft' || form.status === 'cancel' || form.status === 'back')) <el-button v-if="form && form.id && form.status !== 'draft'" type="primary" @click="handleApprovalRecord">流程进度</el-button>
"
:loading="buttonLoading"
type="info"
@click="submitForm('draft')"
>暂存</el-button
>
<el-button
v-if="
routeParams.type === 'add' ||
(routeParams.type === 'update' && form.status && (form.status === 'draft' || form.status === 'cancel' || form.status === 'back'))
"
:loading="buttonLoading"
type="primary"
@click="submitForm('submit')"
> </el-button
>
<el-button
v-if="routeParams.type === 'approval' && form.status && form.status === 'waiting'"
:loading="buttonLoading"
type="primary"
@click="approvalVerifyOpen"
>审批</el-button
>
<el-button v-if="form.status !== 'draft'" type="primary" @click="handleApprovalRecord">流程进度</el-button>
</div> </div>
<div> <div>
<el-button style="float: right" @click="goBack()">返回</el-button> <el-button style="float: right" @click="goBack()">返回</el-button>
@ -154,7 +130,7 @@ const changeLeaveTime = () => {
const startDate = new Date(leaveTime.value[0]).getTime(); const startDate = new Date(leaveTime.value[0]).getTime();
const endDate = new Date(leaveTime.value[1]).getTime(); const endDate = new Date(leaveTime.value[1]).getTime();
const diffInMilliseconds = endDate - startDate; const diffInMilliseconds = endDate - startDate;
form.value.leaveDays = Math.floor(diffInMilliseconds / (1000 * 60 * 60 * 24)); form.value.leaveDays = Math.floor(diffInMilliseconds / (1000 * 60 * 60 * 24)) + 1;
}; };
/** 获取详情 */ /** 获取详情 */
const getInfo = () => { const getInfo = () => {
@ -246,6 +222,21 @@ const goBack = () => {
const approvalVerifyOpen = async () => { const approvalVerifyOpen = async () => {
submitVerifyRef.value.openDialog(routeParams.value.taskId); submitVerifyRef.value.openDialog(routeParams.value.taskId);
}; };
//
const submitButtonShow = computed(() => {
return (
routeParams.value.type === 'add' ||
(routeParams.value.type === 'update' &&
form.value.status &&
(form.value.status === 'draft' || form.value.status === 'cancel' || form.value.status === 'back'))
);
});
//
const approvalButtonShow = computed(() => {
return routeParams.value.type === 'approval' && form.value.status && form.value.status === 'waiting';
});
onMounted(() => { onMounted(() => {
nextTick(async () => { nextTick(async () => {
routeParams.value = proxy.$route.query; routeParams.value = proxy.$route.query;

View File

@ -19,9 +19,6 @@
"paths": { "paths": {
"@/*": ["src/*"] "@/*": ["src/*"]
}, },
"compilerOptions": {
"types": ["element-plus/global"]
},
"types": ["vite/client"], "types": ["vite/client"],
"skipLibCheck": true, "skipLibCheck": true,
"removeComments": true, "removeComments": true,