Compare commits

...

37 Commits

Author SHA1 Message Date
65a880db47 略改一点user模块 2024-07-06 07:48:50 +08:00
4edc0d242c Merge remote-tracking branch 'origin/personal/pjq/organization' into merge/1
# Conflicts:
#	src/components/HelloWorld.vue
#	src/router/authentication.js
2024-07-06 07:43:02 +08:00
4dc09d2983 Merge remote-tracking branch 'origin/Chester' into merge/1
# Conflicts:
#	package-lock.json
#	package.json
#	src/assets/avatar.jpg
#	src/router/authentication.js
#	src/services/authenticationService.js
#	src/store/authentication.js
#	src/views/authentication/Login.vue
#	src/views/authentication/Profile.vue
#	src/views/authentication/Register.vue
2024-07-06 07:42:24 +08:00
d77a0f458d Merge remote-tracking branch 'origin/MeetingJerry' into merge/1
# Conflicts:
#	public/background.jpg
#	src/assets/avatar.jpg
#	src/router/authentication.js
#	src/views/authentication/Login.vue
#	src/views/authentication/Register.vue
2024-07-06 07:41:11 +08:00
cfbb897cf0 删去登录版 2024-07-06 06:39:15 +08:00
0598216ecc 删去登录版 2024-07-06 06:38:35 +08:00
55e468ce79 删去登录版 2024-07-06 06:37:37 +08:00
541b092691 演示版? 2024-07-06 06:27:11 +08:00
ef704a17f5 注释掉了,现在为初始为更改 2024-07-06 01:05:03 +08:00
d814ee2aef 尝试添加个人查看部门信息 2024-07-06 01:00:13 +08:00
2179458f24 修改api 2024-07-05 22:05:40 +08:00
351806d05f 完全体,不改了 2024-07-05 19:18:25 +08:00
ca5ca2bb95 美观和汉化 2024-07-05 18:24:26 +08:00
d087dc171f 基本完成 2024-07-05 14:40:30 +08:00
cc404e0ec4 完善了搜索功能 2024-07-05 14:02:48 +08:00
d92732dfe6 完善了删除功能 2024-07-05 01:52:04 +08:00
6d9c25f3ce 汉化 2024-07-04 23:50:48 +08:00
0aa703b529 添加了课程列表页面 2024-07-04 23:14:57 +08:00
e0c0a10a43 改了很多 2024-07-04 23:06:05 +08:00
4762408c99 可以返回搜索到的值,但无法显示在列表里 2024-07-04 22:11:39 +08:00
0293c09ebd 1 2024-07-04 21:21:59 +08:00
37bc08761d 可以返回搜索到的值,但无法显示在列表里 2024-07-04 20:49:48 +08:00
2a21170ba3 Merge branch 'refs/heads/styleGet' into personal/pjq/test
# Conflicts:
#	package.json
2024-07-04 17:30:31 +08:00
c9ab7e221a 基本完成,再完善其他的功能与界面就行 2024-07-04 17:25:49 +08:00
34efcf19ee 基本完成,再完善其他的功能与界面就行 2024-07-04 16:47:53 +08:00
4f33e9c8ca 登录按钮有问题,导出功能未实现 2024-07-04 16:45:59 +08:00
e4a21e1c6c 回滚备份 2024-07-04 00:12:46 +08:00
4b0c7ec534 回滚备份 2024-07-03 20:18:30 +08:00
b00e35c31c 写了点
还没debug 诶嘿
2024-07-03 15:58:37 +08:00
e9c59574c5 不错 2024-07-03 01:30:40 +08:00
46237d6283 不错 2024-07-03 01:19:02 +08:00
1c2ded6dcd Merge branch 'refs/heads/main' into styleGet
# Conflicts:
#	src/App.vue
#	src/router/index.js
2024-06-29 17:17:37 +08:00
f369e99292 基本完成,再完善其他的功能与界面就行 2024-06-29 17:10:39 +08:00
b613331464 登录没问题了,正在写注册,但似乎要用ref之类的东西,这个版本是没有的,之后尝试修改 2024-06-25 16:22:01 +08:00
1452208822 还是有问题,不过现在后端可以正常收到username和password 2024-06-25 14:54:39 +08:00
d5611562e6 修改No2.尝试将routes中的authenticationRoutes修改了,没有问题,将authentication.vue从components移动到新建的文件夹views中,测试完成了authentication.js中的login方法,不知为何删除#{apiUrl}'/checkLogin'中的${apiUrl}就能用,不删就不行,留待下次学习 2024-06-24 19:20:30 +08:00
b9c8842a85 第一次尝试将各个url分开访问 2024-06-24 17:15:21 +08:00
21 changed files with 1699 additions and 56 deletions

View File

@ -4,7 +4,7 @@
<meta charset="UTF-8" />
<link rel="icon" type="image/svg+xml" href="/vite.svg" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Vite + Vue</title>
<title>测盟汇管理系统</title>
</head>
<body>
<div id="app"></div>

2
package-lock.json generated
View File

@ -1960,7 +1960,7 @@
},
"node_modules/vue-router": {
"version": "4.4.0",
"resolved": "https://registry.npmmirror.com/vue-router/-/vue-router-4.4.0.tgz",
"resolved": "https://registry.npmjs.org/vue-router/-/vue-router-4.4.0.tgz",
"integrity": "sha512-HB+t2p611aIZraV2aPSRNXf0Z/oLZFrlygJm+sZbdJaW6lcFqEDQwnzUBXn+DApw+/QzDU/I9TeWx9izEjTmsA==",
"dependencies": {
"@vue/devtools-api": "^6.5.1"

BIN
public/background1.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 300 KiB

BIN
public/background2.jpg Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.9 MiB

View File

@ -1,7 +1,7 @@
<script setup>
import { ref } from 'vue'
import { ref } from 'vue';
const count = ref(0)
const count = ref(0);
</script>
<template>
@ -13,7 +13,7 @@ const count = ref(0)
<img src="../assets/vue.svg" class="logo vue" alt="Vue logo" />
</a>
</div>
<h1>{{ this.$route.meta.msg }}</h1>
<h1>{{ $route.meta.msg }}</h1>
<div class="card">
<el-button type="success" @click="count++">count is {{ count }}</el-button>
@ -25,19 +25,24 @@ const count = ref(0)
<p>
Check out
<a href="https://vuejs.org/guide/quick-start.html#local" target="_blank"
>create-vue</a
>, the official Vue + Vite starter
<a href="https://vuejs.org/guide/quick-start.html#local" target="_blank">create-vue</a>, the official Vue + Vite starter
</p>
<p>
Learn more about IDE Support for Vue in the
<a
href="https://vuejs.org/guide/scaling-up/tooling.html#ide-support"
target="_blank"
>Vue Docs Scaling up Guide</a
>.
<a href="https://vuejs.org/guide/scaling-up/tooling.html#ide-support" target="_blank">Vue Docs Scaling up Guide</a>.
</p>
<p class="read-the-docs">Click on the Vite and Vue logos to learn more</p>
<!-- 添加导航链接 -->
<ul>
<li><router-link to="/login">Login</router-link></li>
<li><router-link to="/register">Register</router-link></li>
<li><router-link to="/courseList">Courses</router-link></li>
<li><router-link to="/meetings">Meetings</router-link></li>
<li><router-link to="/news">News</router-link></li>
<li><router-link to="/organizations">Organizations</router-link></li>
<li><router-link to="/users">Users</router-link></li>
</ul>
</template>
<style scoped>
@ -56,4 +61,4 @@ const count = ref(0)
.read-the-docs {
color: #888;
}
</style>
</style>

View File

@ -1,9 +1,23 @@
// import CourseList from '../views/course-management/CourseList.vue'
// import CourseDetail from '../views/course-management/CourseDetail.vue'
// import CourseEdit from '../views/course-management/CourseEdit.vue'
import Course from '../views/course-management/Course.vue';
import CourseList from '../views/course-management/CourseList.vue';
import BackgroundWrapper from "@views/course-management/BackgroundWrapper.vue";
export default [
// { path: '/courses', component: CourseList },
// { path: '/courses/:id', component: CourseDetail },
// { path: '/courses/:id/edit', component: CourseEdit }
]
{
path: '/course',
name: 'Course',
component: Course
},
{
path: '/courseList',
name: 'CourseList',
component: CourseList
},
{
path: '/backgroundWrapper',
name: 'BackgroundWrapper',
component: BackgroundWrapper
}
];

View File

@ -1,9 +1,24 @@
// import MeetingList from '../views/meeting-management/MeetingList.vue'
// import MeetingDetail from '../views/meeting-management/MeetingDetail.vue'
// import MeetingEdit from '../views/meeting-management/MeetingEdit.vue'
// meetingManagementRoutes.js
import MeetingManagement from "/src/views/meeting-management/MeetingManagement.vue";
import AddMeeting from "/src/views/meeting-management/AddMeeting.vue";
import EditMeeting from "/src/views/meeting-management/EditMeeting.vue";
export default [
// { path: '/meetings', component: MeetingList },
// { path: '/meetings/:id', component: MeetingDetail },
// { path: '/meetings/:id/edit', component: MeetingEdit }
]
const routes = [
{
path: '/Meeting',
name: 'MeetingManagement',
component: MeetingManagement
},
{
path: '/Meeting/add',
name: 'AddMeeting',
component: AddMeeting
},
{
path: '/Meeting/edit/:id',
name: 'EditMeeting',
component: EditMeeting
}
];
export default routes;

View File

@ -1,9 +1,7 @@
// import OrganizationList from '../views/organization-management/OrganizationList.vue'
// import OrganizationDetail from '../views/organization-management/OrganizationDetail.vue'
// import OrganizationEdit from '../views/organization-management/OrganizationEdit.vue'
import Organization_main from '../views/organization-management/Organization_main.vue'
export default [
// { path: '/organizations', component: OrganizationList },
{ path: '/organizations', component: Organization_main},
// { path: '/organizations/:id', component: OrganizationDetail },
// { path: '/organizations/:id/edit', component: OrganizationEdit }
]

View File

@ -0,0 +1,42 @@
import axios from 'axios';
class MeetingService {
getAllMeetings() {
return axios.get(`/api/meetings/listAll`);
}
getMeetingById(id) {
// 使用 POST 方法并传递请求体
return axios.post(`/api/meetings/getMeetingById`, { id });
}
createMeeting(meeting) {
return axios.post(`/api/meetings/addMeeting`, meeting);
}
updateMeeting(id, meeting) {
// Convert meeting object to a map
const meetingMap = {
id: id,
name: meeting.name,
organizer: meeting.organizer,
startTime: meeting.startTime,
endTime: meeting.endTime,
content: meeting.content,
status: meeting.status
};
return axios.post(`/api/meetings/updateMeeting`, meetingMap);
}
deleteMeeting(id) {
// 使用 POST 方法并传递请求体
return axios.post(`/api/meetings/deleteMeeting`, { id });
}
searchMeetings(params) {
// 使用 POST 方法并传递请求体
return axios.post(`/api/meetings/searchMeetings`, params);
}
}
export default new MeetingService();

View File

@ -0,0 +1,24 @@
import axios from "axios";
export default {
getAllOrganizations() {
return axios.get('api/organizations/listAll').then(response => {
return response.data;
});
},
addOrganization(organization) {
return axios.post('api/organizations/addOrganization', organization).then(response => {
return response.data;
});
},
deleteOrganization(organizationId) {
return axios.post('api/organizations/deleteOrganization' , {organizationId:organizationId},{headers: {
'Content-Type': 'application/json' // 设置 Content-Type 为 application/json
}}).then(response => {
return response.data;
});
},
getOrganizationByName(organizationName){
return axios.get('api/organizations/getByOrganizationName?')
}
}

View File

@ -1,8 +1,10 @@
import axios from "axios";
import store from "@store/index.js"
export default {
addTenant(tenant) {
const url = '/api/tenant/addTenant'
const token = store.getters['authentication/token']
const url = `/api/tenant/addTenant?token=${token}`
const data = {
symbol: tenant.symbol,
contact: tenant.contact,
@ -23,20 +25,23 @@ export default {
async getAll() {
const url = '/api/tenant/getAllTenant'
const token = store.getters['authentication/token']
const url = `/api/tenant/getAllTenant?token=${token}`
const response = await axios.get(url)
return response.data;
},
update(tenant) {
const url = '/api/tenant/updateTenant';
const token = store.getters['authentication/token']
const url = `/api/tenant/updateTenant?token=${token}`;
return axios.post(url, tenant)
.then(response => {
});
},
delete(tenant) {
const url = '/api/tenant/deleteTenant';
const token = store.getters['authentication/token']
const url = `/api/tenant/deleteTenant?token=${token}`;
return axios.post(url, tenant)
.then(response => {
});

View File

@ -1,8 +1,11 @@
import axios from "axios";
import {useStore} from 'vuex';
import store from '@store/index.js'
export default {
addUser(user){
const url='/api/user/addUser'
const token = store.getters['authentication/token']
const url=`/api/user/addUser?token=${token}`
const data={
account: user.account,
name: user.name,
@ -25,13 +28,15 @@ export default {
async getAll() {
const url = '/api/user/getAll'
const token = store.getters['authentication/token']
const url = `/api/user/getAll?token=${token}`
const response = await axios.get(url)
return response.data;
},
update(user){
const url='/api/user/update';
const token = store.getters['authentication/token']
const url=`/api/user/update?token=${token}`
return axios.post(url,user)
.then(() => {
location.href="/userManagement";
@ -39,7 +44,8 @@ export default {
},
async delete(user){
const url='/api/user/delete';
const token = store.getters['authentication/token']
const url=`/api/user/delete?token=${token}`
return await axios.post(url, user)
}

View File

@ -1,27 +1,96 @@
import MeetingService from '../services/meetingService';
const state = {
meetings: []
}
meetings: [],
currentMeeting: null
};
const mutations = {
setMeetings(state, meetings) {
state.meetings = meetings
state.meetings = meetings;
},
setCurrentMeeting(state, meeting) {
state.currentMeeting = meeting;
}
}
};
const actions = {
fetchMeetings({ commit }) {
// 模拟API调用
const meetings = [
{ id: 1, title: 'Meeting 1' },
{ id: 2, title: 'Meeting 2' }
]
commit('setMeetings', meetings)
async fetchMeetings({ commit }) {
try {
const response = await MeetingService.getAllMeetings();
commit('setMeetings', response.data);
} catch (error) {
console.error('Failed to fetch meetings:', error);
}
},
async searchMeetings({ commit }, params) {
try {
const formattedparams = {
name: params.name,
organizer: params.organizer,
startTime: params.startTime,
};
const response = await MeetingService.searchMeetings(formattedparams);
commit('setMeetings', response.data);
} catch (error) {
console.error('Failed to search meetings:', error);
}
},
async fetchMeetingById({ commit }, id) {
try {
const response = await MeetingService.getMeetingById(id);
commit('setCurrentMeeting', response.data);
} catch (error) {
console.error('Failed to fetch meeting:', error);
}
},
async createMeeting({ dispatch }, meeting) {
try {
const formattedMeeting = {
name: meeting.name,
organizer: meeting.organizer,
startTime: meeting.startTime,
endTime: meeting.endTime,
content: meeting.content,
status: 'Scheduled' // 确保状态为 'Scheduled'
};
await MeetingService.createMeeting(formattedMeeting);
dispatch('fetchMeetings');
} catch (error) {
console.error('Failed to create meeting:', error);
}
},
async editMeeting({ dispatch }, meeting) {
try {
const formattedMeeting = {
name: meeting.name,
organizer: meeting.organizer,
startTime: meeting.startTime,
endTime: meeting.endTime,
content: meeting.content,
status: 'Scheduled' // 确保状态为 'Scheduled'
};
await MeetingService.updateMeeting(meeting.id, formattedMeeting);
dispatch('fetchMeetings');
} catch (error) {
console.error('Failed to update meeting:', error);
}
},
async deleteMeeting({ dispatch }, id) {
try {
await MeetingService.deleteMeeting(id);
dispatch('fetchMeetings');
} catch (error) {
console.error('Failed to delete meeting:', error);
}
}
}
};
const getters = {
allMeetings: state => state.meetings
}
allMeetings: state => state.meetings,
currentMeeting: state => state.currentMeeting
};
export default {
namespaced: true,
@ -29,4 +98,4 @@ export default {
mutations,
actions,
getters
}
};

View File

@ -0,0 +1,17 @@
<template>
<div class="background-wrapper">
<slot></slot>
</div>
</template>
<style scoped>
.background-wrapper {
background: url('@public/background2.jpg') no-repeat center center fixed;
background-size: cover;
width: 100%;
height: 100%;
display: flex;
justify-content: center;
align-items: center;
}
</style>

View File

@ -0,0 +1,425 @@
<script setup>
import {onMounted, ref, computed} from 'vue';
import {
ElButton,
ElForm,
ElFormItem,
ElInput,
ElMessage,
ElMessageBox,
ElUpload
} from 'element-plus';
import {Delete, Plus, Refresh, ZoomIn} from "@element-plus/icons-vue";
import {useRoute, useRouter} from "vue-router";
import {useStore} from "vuex";
import axios from "axios";
import BackgroundWrapper from './BackgroundWrapper.vue'; // BackgroundWrapper
const route = useRoute();
const router = useRouter();
const store = useStore();
const token = ref('')
const form = ref({
token: '',
title: '',
description: '',
orderNo: '',
author: '',
videoPath: '',
imagePath: ''
});
const createMode = ref(false);
const modeTitle = ref('');
const videoFileList = ref([]); //
const imageFileList = ref([]); //
const basePath = '/api/courses'
const videoUploadUrl = basePath + '/upload'; //
const imageUploadUrl = videoUploadUrl; //
const dialogImageUrl = ref('');
const dialogVisible = ref(false); //
//
const id = ref('');
const beforeVideoUpload = (file) => {
const isMp4 = file.type === 'video/mp4';
if (!isMp4) {
ElMessage.error('只能上传 mp4 格式的视频文件!');
return false;
}
const isLt500M = file.size / 1024 / 1024 < 500;
if (!isLt500M) {
ElMessage.error('上传视频大小不能超过 500MB');
return false;
}
return true;
};
const beforeImageUpload = (file) => {
const isImage = file.type === 'image/png' || file.type === 'image/jpg' || file.type === 'image/jpeg';
if (!isImage) {
ElMessage.error('只能上传 png/jpg/jpeg 格式的图片文件!');
return false;
}
const isLt5M = file.size / 1024 / 1024 < 5;
if (!isLt5M) {
ElMessage.error('上传文件大小不能超过 5MB');
return false;
}
return true;
};
const handleVideoSuccess = (response, file, fileList) => {
file.url = response.url;
form.value.videoPath = response.url; //
};
const handleImageSuccess = (response, file, fileList) => {
file.url = response.url;
form.value.imagePath = response.url; //
};
const handleVideoError = (error, file, fileList) => {
ElMessage.error('视频上传失败,请重试!');
};
const handleImageError = (error, file, fileList) => {
ElMessage.error('图片上传失败,请重试!');
};
const handleVideoChange = (file, fileList) => {
if (fileList.length > 1) {
fileList.splice(0, fileList.length - 1)
}
};
const handleImageChange = (file, fileList) => {
if (fileList.length > 1) {
fileList.splice(0, fileList.length - 1)
}
};
const handlePictureCardPreview = (file) => {
dialogImageUrl.value = file.url
dialogVisible.value = true
}
const handleRemoveImage = (file) => {
imageFileList.value = imageFileList.value.filter((item) => item.uid !== file.uid);
}
const handleRemoveVideo = (file) => {
videoFileList.value = videoFileList.value.filter((item) => item.uid !== file.uid);
}
const handleCommit = async () => {
if (form.value.title === '' || form.value.description === '' || form.value.orderNo === '' || form.value.author === '') {
await ElMessageBox.alert('请填写完整信息!');
return;
}
if (isNaN(form.value.orderNo)) {
await ElMessageBox.alert('排序必须是一个数字!');
return;
}
if (imageFileList.value.length === 0 || videoFileList.value.length === 0) {
await ElMessageBox.alert('请选择封面图片和课程视频!');
return;
}
form.value.imagePath = imageFileList.value[0].url;
form.value.videoPath = videoFileList.value[0].url;
if (createMode.value) {
try {
const response = await axios.post(basePath, form.value);
ElMessage.success('添加成功!');
router.push('/courseList');
} catch (e) {
await ElMessageBox.alert(e.response.data.message);
}
} else {
try {
const response = await axios.put(`${basePath}/${id.value}`, form.value);
ElMessage.success('修改成功!');
router.push('/courseList');
} catch (e) {
await ElMessageBox.alert(e.response.data.message);
}
}
};
const isLayoutReady = ref(false);
const fetchCourseDetail = async () => {
let params = {
token: token.value,
}
const res = await axios.get(basePath + '/' + id.value, {params});
form.value.title = res.data.title;
form.value.description = res.data.description;
form.value.orderNo = res.data.orderNo;
form.value.author = res.data.author;
form.value.videoPath = res.data.videoPath;
form.value.imagePath = res.data.imagePath;
imageFileList.value.push({url: res.data.imagePath});
videoFileList.value.push({url: res.data.videoPath});
}
const orderNo = computed({
get() {
return form.value.orderNo;
},
set(value) {
form.value.orderNo = value.replace(/\D/g, ''); //
}
});
onMounted(() => {
token.value = store.getters['authentication/token'];
form.value.token = token.value;
if (route.query.mode === 'create' || route.params.mode === 'create') {
createMode.value = true;
modeTitle.value = '添加课程';
}
if (route.query.mode === 'edit' || route.params.mode === 'edit') {
id.value = route.params.id === undefined ? route.query.id : route.params.id;
createMode.value = false;
modeTitle.value = '修改课程';
fetchCourseDetail();
}
});
</script>
<template>
<BackgroundWrapper class="background-wrapper">
<div class="form-container">
<ElForm :model="form" label-width="120px" @submit.prevent="handleCommit">
<h2>{{ modeTitle }}</h2>
<ElFormItem label="课程名称" required>
<ElInput v-model="form.title" placeholder="请输入课程名称" required></ElInput>
</ElFormItem>
<ElFormItem label="课程封面" required>
<el-upload
:action="imageUploadUrl"
:limit="2"
:before-upload="beforeImageUpload"
:on-success="handleImageSuccess"
:on-error="handleImageError"
:on-change="handleImageChange"
:file-list="imageFileList"
list-type="picture-card"
auto-upload
v-model:file-list="imageFileList"
>
<template #file="{ file }">
<div>
<img class="el-upload-list__item-thumbnail" :src="file.url" alt=""/>
<span class="el-upload-list__item-actions">
<span
class="el-upload-list__item-preview"
@click="handlePictureCardPreview(file)"
>
<el-icon><zoom-in/></el-icon>
</span>
<span
class="el-upload-list__item-delete"
@click="handleRemoveImage(file)"
>
<el-icon><Delete/></el-icon>
</span>
</span>
</div>
</template>
<el-icon v-if="imageFileList.length === 0">
<Plus/>
</el-icon>
<el-icon v-else>
<Refresh/>
</el-icon>
</el-upload>
<div class="tip">
请上传大小不超过 <span style="color: red;">5MB</span> 格式为 <span style="color: red;">png/jpg/jpeg</span> 的文件
</div>
</ElFormItem>
<ElFormItem label="课程视频" required>
<el-upload
:action="videoUploadUrl"
:limit="2"
:before-upload="beforeVideoUpload"
:on-success="handleVideoSuccess"
:on-error="handleVideoError"
:on-change="handleVideoChange"
:file-list="videoFileList"
list-type="picture-card"
auto-upload
v-model:file-list="videoFileList"
>
<template #file="{ file }">
<div>
<video class="el-upload-list__item-thumbnail" :src="file.url" controls></video>
<span class="el-upload-list__item-actions">
<span
class="el-upload-list__item-delete"
@click="handleRemoveVideo(file)"
>
<el-icon><Delete/></el-icon>
</span>
</span>
</div>
</template>
<el-icon v-if="videoFileList.length === 0">
<Plus/>
</el-icon>
<el-icon v-else>
<Refresh/>
</el-icon>
</el-upload>
<div class="tip">
请上传大小不超过 <span style="color: red;">500MB</span> 格式为 <span style="color: red;">mp4</span> 的文件
</div>
</ElFormItem>
<ElFormItem label="课程简介" required>
<ElInput v-model="form.description" placeholder="请输入课程简介" required></ElInput>
</ElFormItem>
<ElFormItem label="课程排序" required>
<ElInput v-model="orderNo" placeholder="请输入课程排序(数字)" required></ElInput>
</ElFormItem>
<ElFormItem label="作者" required>
<ElInput v-model="form.author" placeholder="请输入作者" required></ElInput>
</ElFormItem>
<ElFormItem>
<ElButton type="primary" native-type="submit">确定</ElButton>
<ElButton @click="router.push('/courseList')">取消</ElButton>
</ElFormItem>
</ElForm>
</div>
<el-dialog v-model="dialogVisible" class="image-preview">
<img w-full :src="dialogImageUrl" alt="Preview Image"/>
</el-dialog>
</BackgroundWrapper>
</template>
<style scoped>
html, body {
height: 100%;
margin: 0;
}
.background-wrapper {
position: fixed; /* 确保背景覆盖整个视口 */
top: 0;
left: 0;
width: 100%;
height: 100%;
overflow: auto; /* 只在内容溢出时允许滚动 */
/* 自定义滚动条样式 */
scrollbar-width: thin;
scrollbar-color: #007bff #f0f0f0;
}
/* 针对 Webkit 浏览器的自定义滚动条 */
.background-wrapper::-webkit-scrollbar {
width: 8px;
}
.background-wrapper::-webkit-scrollbar-track {
background: #f0f0f0;
border-radius: 10px;
}
.background-wrapper::-webkit-scrollbar-thumb {
background-color: #007bff;
border-radius: 10px;
border: 3px solid #f0f0f0;
}
.form-container {
display: flex;
justify-content: center;
align-items: center;
height: 100%;
overflow-y: auto; /* 使容器在内容溢出时出现滚动条 */
}
.image-preview {
display: flex;
justify-content: center;
align-items: center;
}
.image-preview img {
max-width: 100%;
height: auto;
}
.tip {
margin-top: 8px;
color: #727272;
font-size: 12px;
}
.el-form {
background: white;
padding: 2rem;
border-radius: 8px;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
width: 65%; /* 增加宽度 */
max-height: 90vh; /* 确保表单不会超过视口高度 */
overflow-y: auto; /* 使表单在内容溢出时出现滚动条 */
/* 自定义滚动条样式 */
scrollbar-width: thin;
scrollbar-color: #007bff #f0f0f0;
}
/* 针对 Webkit 浏览器的自定义滚动条 */
.el-form::-webkit-scrollbar {
width: 8px;
}
.el-form::-webkit-scrollbar-track {
background: #f0f0f0;
border-radius: 10px;
}
.el-form::-webkit-scrollbar-thumb {
background-color: #007bff;
border-radius: 10px;
border: 3px solid #f0f0f0;
}
.el-form-item {
margin-bottom: 1rem;
text-align: left;
}
.el-form-item label {
display: block;
margin-bottom: 0.5rem;
}
.el-input,
.el-select {
width: calc(100% - 12px); /* 设置文本框宽度,使其与滚动条保持一定距离 */
}
.el-button--primary {
background-color: #007bff;
color: white;
}
.el-button--primary:hover {
background-color: #0056b3;
}
.dynamic-width-select {
min-width: 30%;
max-width: 100%;
width: auto; /* 使宽度根据内容调整 */
}
</style>

View File

@ -0,0 +1,333 @@
<script setup>
import { onMounted, ref } from 'vue';
import { ElButton, ElForm, ElFormItem, ElInput, ElMessage, ElMessageBox, ElPagination, ElTable, ElTableColumn } from 'element-plus';
import axios from "axios";
import { useStore } from "vuex";
import { useRouter } from "vue-router";
import Course from "@views/course-management/Course.vue";
import BackgroundWrapper from './BackgroundWrapper.vue';
import _ from 'lodash';
import { write, utils } from 'xlsx';
import { saveAs } from 'file-saver';
const router = useRouter();
const store = useStore();
const token = store.getters['authentication/token'];
const searchTitle = ref('');
const searchAuthor = ref('');
const searchDescription = ref('');
const sortOrder = ref('');
const allCoursesData = ref([]);
const coursesData = ref([]);
const currentPage = ref(1);
const pageSize = ref(10);
const coursesCount = ref(0);
const firstTimeLoad = ref(true);
const selections = ref([]);
const loadCourses = async (forceReload = false) => {
if (forceReload) {
firstTimeLoad.value = true;
allCoursesData.value = [];
}
if (firstTimeLoad.value || allCoursesData.value.length < (currentPage.value * pageSize.value) && (currentPage.value * pageSize.value) <= coursesCount.value) {
let params = {
token: token,
start: allCoursesData.value.length,
end: allCoursesData.value.length + pageSize.value * 2
};
try {
const response = await axios.get('/api/courses', { params });
const data = response.data;
coursesCount.value = data.courseCount;
allCoursesData.value.push(...data.courseList);
} catch (e) {
console.log(e);
}
firstTimeLoad.value = false;
}
coursesData.value = allCoursesData.value.slice((currentPage.value - 1) * pageSize.value, currentPage.value * pageSize.value);
};
onMounted(() => {
loadCourses();
});
const handleSearch = async () => {
try {
const response = await axios.get('/api/courses/search', {
params: {
token: token,
title: searchTitle.value || '',
author: searchAuthor.value || '',
description: searchDescription.value || '',
sortOrder: sortOrder.value || '',
start: 0,
end: pageSize.value
}
});
const data = response.data;
coursesCount.value = data.courseCount;
allCoursesData.value = data.courseList;
coursesData.value = allCoursesData.value.slice(0, pageSize.value);
currentPage.value = 1;
firstTimeLoad.value = false;
} catch (e) {
console.error(e);
}
};
const handleReset = async () => {
searchTitle.value = '';
searchAuthor.value = '';
searchDescription.value = '';
sortOrder.value = '';
await loadCourses(true); //
};
const handleSort = async () => {
//
const sortedCourses = _.cloneDeep(allCoursesData.value).map(course => ({
...course,
orderNo: Number(course.orderNo)
})).sort((a, b) => a.orderNo - b.orderNo);
//
allCoursesData.value = sortedCourses;
//
const start = (currentPage.value - 1) * pageSize.value;
const end = start + pageSize.value;
coursesData.value = allCoursesData.value.slice(start, end);
};
const handleExport = () => {
// 簿
const workbook = utils.book_new();
// allCoursesData
const sheetData = allCoursesData.value.map(course => ({
'课程名称': course.title,
'作者': course.author,
'课程简介': course.description,
'排序': course.orderNo
}));
//
const worksheet = utils.json_to_sheet(sheetData);
// 簿
utils.book_append_sheet(workbook, worksheet, 'Courses');
// Excel
const wbout = write(workbook, { bookType: 'xlsx', type: 'array' });
// 使 file-saver
saveAs(new Blob([wbout], { type: 'application/octet-stream' }), '课程信息.xlsx');
};
const handleEditButton = () => {
if (selections.value.length === 1) {
//
const selectedCourse = selections.value[0];
router.push({ name: 'Course', query: { mode: 'edit', id: selectedCourse.id } });
} else if (selections.value.length > 1) {
//
ElMessage.warning('无法同时修改多个目标');
} else {
//
ElMessage.warning('请先选择要修改的课程');
}
};
const handleDeleteButton = async () => {
if (selections.value.length === 0) {
ElMessage.warning('请先选择要删除的课程');
return;
}
ElMessageBox.confirm(
'是否确认删除选中的课程?此操作不可撤销。',
'确认删除',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
}
).then(async () => {
try {
const deletePromises = selections.value.map(selection =>
axios.delete(`/api/courses/${selection.id}`, { params: { token: token } })
);
await Promise.all(deletePromises);
ElMessage.success('删除成功');
selections.value = []; //
await loadCourses(true); //
} catch (e) {
ElMessage.error('删除失败');
console.error(e);
}
}).catch(() => {
ElMessage.info('已取消删除操作');
});
};
const handleSelectionChange = (newSelections) => {
selections.value = newSelections;
};
const handleEditInTable = (index) => {
router.push({ name: 'Course', query: { mode: 'edit', id: coursesData.value[index].id } });
};
const handleDeleteInTable = async (index) => {
ElMessageBox.confirm(
'是否确认删除该课程?此操作不可撤销。',
'确认删除',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
}
).then(async () => {
try {
await axios.delete(`/api/courses/${coursesData.value[index].id}`, { params: { token: token } });
coursesData.value.splice(index, 1);
allCoursesData.value.splice(index, 1);
coursesCount.value--;
await loadCourses(true); //
} catch (e) {
ElMessage.error('删除失败');
console.error(e);
}
}).catch(() => {
ElMessage.info('已取消删除操作');
});
};
</script>
<template>
<BackgroundWrapper class="background-wrapper">
<div class="center-wrapper">
<div class="container">
<div class="search-container">
<el-form inline>
<el-form-item label="课程名称">
<el-input v-model="searchTitle" placeholder="请输入课程名称" />
</el-form-item>
<el-form-item label="排序">
<el-input v-model="sortOrder" placeholder="请输入排序" />
</el-form-item>
<el-form-item label="作者">
<el-input v-model="searchAuthor" placeholder="请输入作者" />
</el-form-item>
<el-form-item label="课程简介">
<el-input v-model="searchDescription" placeholder="请输入简介" />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="handleSearch">搜索</el-button>
</el-form-item>
<el-form-item>
<el-button @click="handleReset">重置</el-button>
</el-form-item>
</el-form>
</div>
<div class="button-container">
<el-button type="success" @click="router.push('/course?mode=create')">新增</el-button>
<el-button type="warning" @click="handleEditButton">修改</el-button>
<el-button type="danger" @click="handleDeleteButton">删除</el-button>
<el-button type="info" @click="handleSort">排序</el-button>
<el-button type="primary" @click="handleExport">导出</el-button>
</div>
<el-table :data="coursesData" style="width: 100%;" @selection-change="handleSelectionChange">
<el-table-column type="selection" width="55"></el-table-column>
<el-table-column prop="orderNo" label="排序" align="center"></el-table-column>
<el-table-column prop="title" label="课程名称" align="center"></el-table-column>
<el-table-column prop="author" label="作者" align="center"></el-table-column>
<el-table-column prop="description" label="课程简介" align="center"></el-table-column>
<el-table-column label="操作" align="center">
<template #default="scope">
<el-button @click="handleEditInTable(scope.$index)" type="text">修改</el-button>
<el-button @click="handleDeleteInTable(scope.$index)" type="text">删除</el-button>
</template>
</el-table-column>
</el-table>
<div class="pagination-container">
<el-pagination
@size-change="pageSize = $event; loadCourses(true)"
@current-change="currentPage = $event; loadCourses(true)"
:current-page="currentPage"
:page-size="pageSize"
layout="total, sizes, prev, pager, next, jumper"
:total="coursesCount"
:page-sizes="[10, 20, 50, 100]"
>
</el-pagination>
</div>
</div>
</div>
</BackgroundWrapper>
</template>
<style scoped>
html, body {
height: 100%;
margin: 0;
}
.background-wrapper {
position: fixed; /* 确保背景覆盖整个视口 */
top: 0;
left: 0;
width: 100%;
height: 100%;
overflow: auto; /* 只在内容溢出时允许滚动 */
}
.center-wrapper {
display: flex;
justify-content: center;
align-items: center;
width: 100%;
height: 100%;
}
.container {
padding: 20px;
background: rgba(255, 255, 255, 0.8); /* 为了让内容清晰,可加上一层半透明白色背景 */
border-radius: 15px;
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.1);
max-width: 800px; /* 控制页面内容宽度 */
width: 100%;
}
.search-container {
display: flex;
flex-wrap: wrap;
gap: 10px;
}
.button-container {
display: flex;
gap: 10px;
margin-bottom: 10px;
}
.el-table {
margin-bottom: 10px;
}
.pagination-container {
display: flex;
justify-content: flex-end;
margin-bottom: -10px;
}
</style>

View File

@ -0,0 +1,57 @@
<template>
<el-form :model="meeting" label-width="120px" @submit.native.prevent="submitForm">
<el-form-item label="会议名称">
<el-input v-model="meeting.name"></el-input>
</el-form-item>
<el-form-item label="创建人">
<el-input v-model="meeting.organizer"></el-input>
</el-form-item>
<el-form-item label="开始时间">
<el-date-picker v-model="meeting.startTime" type="datetime" placeholder="选择时间"></el-date-picker>
</el-form-item>
<el-form-item label="结束时间">
<el-date-picker v-model="meeting.endTime" type="datetime" placeholder="选择时间"></el-date-picker>
</el-form-item>
<el-form-item label="会议内容">
<el-input type="textarea" v-model="meeting.content"></el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="submitForm">确定</el-button>
<el-button @click="cancel">取消</el-button>
</el-form-item>
</el-form>
</template>
<script>
import { mapActions } from 'vuex';
export default {
name: 'AddMeeting',
data() {
return {
meeting: {
name: '',
organizer: '',
startTime: '',
endTime: '',
content: '',
status: 'Scheduled' // 'Scheduled'
}
};
},
methods: {
...mapActions('meetingManagement', ['createMeeting']),
async submitForm() {
try {
await this.createMeeting(this.meeting);
this.$router.push({ name: 'MeetingManagement' });
} catch (error) {
console.error('Failed to create meeting:', error);
}
},
cancel() {
this.$router.push({ name: 'MeetingManagement' });
}
}
};
</script>

View File

@ -0,0 +1,61 @@
<template>
<el-form :model="meeting" label-width="120px" @submit.native.prevent="submitForm">
<el-form-item label="会议名称">
<el-input v-model="meeting.name"></el-input>
</el-form-item>
<el-form-item label="创建人">
<el-input v-model="meeting.organizer"></el-input>
</el-form-item>
<el-form-item label="开始时间">
<el-date-picker v-model="meeting.startTime" type="datetime" placeholder="选择时间"></el-date-picker>
</el-form-item>
<el-form-item label="结束时间">
<el-date-picker v-model="meeting.endTime" type="datetime" placeholder="选择时间"></el-date-picker>
</el-form-item>
<el-form-item label="会议内容">
<el-input type="textarea" v-model="meeting.content"></el-input>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="submitForm">确定</el-button>
<el-button @click="cancel">取消</el-button>
</el-form-item>
</el-form>
</template>
<script>
import { mapActions, mapGetters } from 'vuex';
export default {
name: 'EditMeeting',
data() {
return {
meeting: {
id: '',
name: '',
organizer: '',
startTime: '',
endTime: '',
content: '',
status: 'Active'
}
};
},
computed: {
...mapGetters('meetingManagement', ['currentMeeting'])
},
methods: {
...mapActions('meetingManagement', ['fetchMeetingById', 'editMeeting']),
async submitForm() {
await this.editMeeting(this.meeting);
this.$router.push({ name: 'MeetingManagement' });
},
cancel() {
this.$router.push({ name: 'MeetingManagement' });
}
},
async mounted() {
await this.fetchMeetingById(this.$route.params.id);
Object.assign(this.meeting, this.currentMeeting);
}
};
</script>

View File

@ -0,0 +1,162 @@
<template>
<div class="meeting-management">
<div class="header">
<el-input v-model="searchName" placeholder="会议名称" class="search-input"></el-input>
<el-input v-model="searchOrganizer" placeholder="创建人" class="search-input"></el-input>
<el-date-picker v-model="searchStartTime" type="datetime" placeholder="开始时间" class="search-input"></el-date-picker>
<el-button type="primary" @click="search">搜索</el-button>
<el-button type="primary" @click="goToAddMeeting">添加会议</el-button>
</div>
<el-table :data="allMeetings" style="width: 100%">
<el-table-column prop="id" label="会议ID" width="80"></el-table-column>
<el-table-column prop="name" label="会议名称"></el-table-column>
<el-table-column prop="organizer" label="组织者"></el-table-column>
<el-table-column prop="startTime" label="开始时间"></el-table-column>
<el-table-column prop="endTime" label="结束时间"></el-table-column>
<el-table-column prop="status" label="状态"></el-table-column>
<el-table-column label="操作" width="180">
<template #default="scope">
<el-button @click="editMeeting(scope.row.id)" type="text">修改</el-button>
<el-button @click="confirmDelete(scope.row.id)" type="text" class="delete-button">删除</el-button>
</template>
</el-table-column>
</el-table>
<el-pagination
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
:current-page="currentPage"
:page-sizes="[10, 20, 30, 40]"
:page-size="pageSize"
layout="total, sizes, prev, pager, next, jumper"
:total="total"
></el-pagination>
</div>
</template>
<script>
import { mapActions, mapGetters } from 'vuex';
import axios from 'axios';
import MeetingService from "@services/meetingService.js";
import MeetingManagement from "@store/meetingManagement.js";
import meetingManagement from "@store/meetingManagement.js";
export default {
name: 'MeetingManagement',
data() {
return {
currentPage: 1,
pageSize: 10,
total: 0,
searchName: '',
searchOrganizer: '',
searchStartTime: null,
allMeeting:''
};
},
computed: {
...mapGetters('meetingManagement', ['allMeetings'])
},
methods: {
...mapActions('meetingManagement', ['fetchMeetings', 'searchMeetings', 'deleteMeeting']),
goToAddMeeting() {
this.$router.push({ name: 'AddMeeting' });
},
editMeeting(id) {
this.$router.push({ name: 'EditMeeting', params: { id } });
},
confirmDelete(id) {
this.$confirm(
'是否确认删除会议管理编号为 "' + id + '" 的数据项?',
'系统提示',
{
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
}
)
.then(() => {
this.deleteMeeting(id).then(() => {
this.fetchMeetings();
});
})
.catch(() => {});
},
handleSizeChange(val) {
this.pageSize = val;
this.fetchMeetings();
},
handleCurrentChange(val) {
this.currentPage = val;
this.fetchMeetings();
},
search() {
const params = {
name: this.searchName,
organizer: this.searchOrganizer,
startTime: this.searchStartTime ? this.searchStartTime.toISOString() : null
};
this.allMeeting = this.searchMeetings(params)
},
exportMeetings() {
axios
.post(
'/api/meetings/export',
{
name: this.searchName,
organizer: this.searchOrganizer,
startTime: this.searchStartTime ? this.searchStartTime.toISOString() : null
},
{ responseType: 'blob' }
)
.then(response => {
const url = window.URL.createObjectURL(new Blob([response.data]));
const link = document.createElement('a');
link.href = url;
link.setAttribute('download', 'meetings.xlsx');
document.body.appendChild(link);
link.click();
})
.catch(error => {
console.error('导出失败', error);
});
}
},
mounted() {
this.fetchMeetings();
}
};
</script>
<style scoped>
.meeting-management {
padding: 20px;
background-color: #f5f5f5;
border-radius: 8px;
}
.header {
display: flex;
justify-content: flex-end;
margin-bottom: 20px;
}
.search-input {
margin-right: 10px;
}
.el-button {
margin-right: 10px;
}
.el-table {
background-color: #fff;
border-radius: 8px;
overflow: hidden;
}
.el-pagination {
margin-top: 20px;
text-align: center;
}
.delete-button {
color: #f56c6c;
}
.delete-button:hover {
color: #ff7875;
}
</style>

View File

@ -0,0 +1,354 @@
<template>
<div class="department-management" v-if="showTable">
<el-row :gutter="20" class="toolbar">
<el-col :span="4">
<el-input v-model="search.name" placeholder="请输入部门名称"></el-input>
</el-col>
<el-col :span="4">
<el-select v-model="search.status" placeholder="部门状态">
<el-option label="正常" value=true></el-option>
<el-option label="停用" value=false></el-option>
<el-option label="全部" value=""></el-option>
</el-select>
</el-col>
<el-col :span="4">
<el-button type="primary" @click="handleSearch">搜索</el-button>
<el-button @click="handleReset">重置</el-button>
</el-col>
</el-row>
<el-button type="primary" @click="showDialogAdd = true">添加部门</el-button>
<el-table
:data="tableData"
style="width: 100%; margin-bottom: 20px"
row-key="organizationId"
:tree-props="{ children: 'children', hasChildren: 'hasChildren' }"
border
default-expand-all
>
<el-table-column prop="organizationId" label="部门ID" width="180" />
<el-table-column prop="parentOrganization.organizationName" label="上级部门" width="180">
<template #default="{ row }">
{{ row.parentOrganization ? row.parentOrganization.organizationName : '无' }}
</template>
</el-table-column>
<el-table-column prop="organizationName" label="部门名称" width="180" />
<el-table-column prop="leader" label="负责人" width="180" />
<el-table-column prop="contactPhone" label="负责人联系电话" width="180" />
<el-table-column prop="email" label="负责人邮箱" width="180" />
<el-table-column prop="organizationStatus" label="部门状态" width="180">
<template #default="{ row }">
<span v-if="row.organizationStatus">
<el-text type="success">正常</el-text>
</span>
<span v-else>
<el-text type="danger">停用</el-text>
</span>
</template>
</el-table-column>
<el-table-column label="操作" width="180">
<template #default="scope" >
<el-button type="danger" text @click="deleteOrganization(scope.row.organizationId)">删除</el-button>
<el-button type="danger" text @click="handleUpdate(scope.row.organizationId)">修改</el-button>
</template>
</el-table-column>
</el-table>
</div>
<!-- 添加表单-->
<el-dialog v-model="showDialogAdd" title="填写信息" width="80%">
<el-form :model="dialogFormAdd" :rules="rules">
<el-form-item label="上级部门" :label-width="formLabelWidth">
<el-select v-model="dialogFormAdd.parentOrganization">
<el-option
v-for="item in originTableData"
:key="item.organizationId"
:label="item.organizationName"
:value="item.organizationId"
>
{{item.organizationName}}
</el-option>
<el-option label="无" value=0></el-option>
</el-select>
</el-form-item>
<el-form-item prop="organizationName" label="部门名称" :label-width="formLabelWidth">
<el-input v-model="dialogFormAdd.organizationName"></el-input>
</el-form-item>
<el-form-item prop="displayOrder" label="显示排列" :label-width="formLabelWidth">
<el-input-number v-model="dialogFormAdd.displayOrder" :min=1 :max="100"></el-input-number>
</el-form-item>
<el-form-item prop="leader" label="负责人" :label-width="formLabelWidth">
<el-input v-model="dialogFormAdd.leader"></el-input>
</el-form-item>
<el-form-item prop="contactPhone" label="联系电话" :label-width="formLabelWidth">
<el-input v-model="dialogFormAdd.contactPhone"></el-input>
</el-form-item>
<el-form-item prop="email" label="邮箱" :label-width="formLabelWidth">
<el-input v-model="dialogFormAdd.email"></el-input>
</el-form-item>
<el-form-item label="部门状态" :label-width="formLabelWidth">
<el-switch v-model="dialogFormAdd.organizationStatus" active-value="true" active-text="启用" inactive-value="false" inactive-text="禁用"></el-switch>
</el-form-item>
</el-form>
<div class="dialog-footer">
<span slot="footer">
<el-button @click="showDialogAdd = false">取消</el-button>
<el-button type="primary" @click="submitForm">提交</el-button>
</span>
</div>
</el-dialog>
</template>
<script lang="ts">
import {ref, onMounted, defineComponent} from 'vue';
import organizationService from '../../services/organizationService';
import {useStore} from 'vuex';
import {ElMessage, ElMessageBox} from "element-plus";
interface Organization {
organizationId: number;
parentOrganization: Organization | null;
organizationName: string;
leader: string;
contactPhone: string;
email: string;
displayOrder: number;
organizationStatus: boolean;
children?: Organization[]; //
hasChildren: boolean; //
}
export default defineComponent({
setup() {
const store = useStore();
const showTable = ref(true);
const show = ref(false);
const showDialogAdd = ref(false);
const formLabelWidth = ref('120px');
const dialogFormAdd = ref(
{
organizationId:0,
parentOrganization: 0,
organizationName: '',
displayOrder: 0,
leader: '',
contactPhone: '',
email: '',
organizationStatus: true,
}
);
const dialogFormPerson = ref({
});
const originTableData = ref<Organization[]>([]);
const tableData = ref<Organization[]>([]);
const treeData = ref();
const noChildrenData = ref();
const fetchData = async () => {
try {
tableData.value=[];
const data = await organizationService.getAllOrganizations();
if(data.lenth===1){
}
console.log('获取到的部门信息', data);
originTableData.value = data;
console.log(originTableData.value);
//buildTree
treeData.value = buildTree(originTableData.value);
console.log('树形结构数据', treeData);
noChildrenData.value = originTableData.value.map(item => ({
...item,
children: [] // children
}));
console.log('无子组织数据', noChildrenData);
tableData.value = treeData.value;
} catch (error) {
console.error('Error fetching organization:', error);
}
};
const buildTree = (flatData: Organization[]): Organization[] => {
const map: Record<number, Organization> = {};
const roots: Organization[] = [];
flatData.forEach(org => {
map[org.organizationId] = org;
org.children = [];
});
flatData.forEach(org => {
if (org.parentOrganization?.organizationId && map[org.parentOrganization.organizationId]) {
map[org.parentOrganization.organizationId].children.push(org);
map[org.parentOrganization.organizationId].hasChildren = false;
// map[org.organizationId].hasChildren = true; //
} else {
roots.push(org);
}
});
const sortTree = (nodes: Organization[]) => {
nodes.sort((a, b) => a.displayOrder - b.displayOrder);
nodes.forEach(node => {
if (node.children && node.children.length > 0) {
sortTree(node.children);
}
});
};
sortTree(roots);
console.log('roots:', roots);
return roots;
};
const submitForm = () => {
console.log('提交表单' , dialogFormAdd.value);
organizationService.addOrganization(dialogFormAdd.value);
ElMessage({
type: 'success',
message: `添加成功`,
})
showDialogAdd.value = false;
dialogFormAdd.value = {
organizationId:0,
parentOrganization: 0,
organizationName: '',
displayOrder: 0,
leader: '',
contactPhone: '',
email: '',
organizationStatus: true,
};
}
const deleteOrganization = (organizationId:number) => {
ElMessageBox.confirm(
'将删除该部门,确定?',
'注意!!!',
{
confirmButtonText: '确认',
cancelButtonText: '取消',
type: 'warning',
center: true,
}
)
.then(() => {
organizationService.deleteOrganization(organizationId);
fetchData();
ElMessage({
type: 'success',
message: '删除成功',
})
})
.catch(() => {
ElMessage({
type: 'info',
message: 'Delete canceled',
})
})
}
const handleUpdate = (organizationId:number) => {
console.log('编辑部门', organizationId);
const organization = originTableData.value.find((item) => item.organizationId === organizationId);
console.log('organization', organization);
dialogFormAdd.value = {
organizationId:organization?.organizationId,
parentOrganization: organization?.parentOrganization?.organizationId || 0,
organizationName: organization?.organizationName,
displayOrder: organization?.displayOrder,
leader: organization?.leader,
contactPhone: organization?.contactPhone,
email: organization?.email,
organizationStatus: organization?.organizationStatus,
}
console.log('dialogFormAdd', dialogFormAdd.value);
showDialogAdd.value = true;
}
const search = ref({
name:'',
status:''
})
const handleSearch = () => {
console.log('搜索', search.value);
// if(search.value.name != ''){
tableData.value = noChildrenData.value.filter(
(data) => {
const isNameValid = search.value && search.value.name;
const isStatusValid = search.value && search.value.status;
return (!isNameValid || data.organizationName.toLowerCase().includes(search.value.name.toLowerCase())) &&
(!isStatusValid || data.organizationStatus === JSON.parse(search.value.status));
}
);
}
const handleReset = () => {
search.value.name = '';
search.value.status = '';
fetchData();
}
onMounted(() => {
fetchData();
});
return {
originTableData,
tableData,
showDialogAdd,
showTable,
show,
dialogFormAdd,
submitForm,
formLabelWidth,
deleteOrganization,
handleUpdate,
search,
handleSearch,
handleReset,
};
},
data() {
return {
rules: {
organizationName:[{
required: true, message: '请输入部门名称', trigger: 'blur'
}],
leader:[{
required: true, message: '请输入负责人名称', trigger: 'blur'
}],
contactPhone:[{
type:'text',required:true,message:'请输入联系电话',trigger:'blur'
}],
email: [{
pattern:/^([a-zA-Z0-9]+[-_\.]?)+@[a-zA-Z0-9]+\.[a-z]+$/ ,required: true ,message: '输入邮箱', trigger: 'blur'
}],
}
};
},
});
</script>
<style scoped>
.department-management {
padding: 20px;
}
.toolbar {
margin-bottom: 20px;
}
.dialog-footer{
text-align: right;
}
</style>

View File

@ -0,0 +1,56 @@
<template>
<div>
<h2>Profile</h2>
<form @submit.prevent="updateProfile">
<div>
<label for="nickname">Nickname:</label>
<input type="text" v-model="user.nickname">
</div>
<div>
<label for="phoneNumber">Phone Number:</label>
<input type="text" v-model="user.phoneNumber">
</div>
<div>
<label for="gender">Gender:</label>
<input type="text" v-model="user.gender">
</div>
<button type="submit">Update Profile</button>
</form>
<form @submit.prevent="changePassword">
<div>
<label for="oldPassword">Old Password:</label>
<input type="password" v-model="oldPassword" required>
</div>
<div>
<label for="newPassword">New Password:</label>
<input type="password" v-model="newPassword" required>
</div>
<button type="submit">Change Password</button>
</form>
</div>
</template>
<script>
import { mapActions, mapState } from 'vuex';
export default {
data() {
return {
oldPassword: '',
newPassword: ''
};
},
computed: {
...mapState(['user'])
},
methods: {
...mapActions(['updateProfile', 'changePassword']),
async updateProfile() {
await this.updateProfile(this.user);
},
async changePassword() {
await this.changePassword({ oldPassword: this.oldPassword, newPassword: this.newPassword });
}
}
};
</script>