完整的教务系统前端项目 - 包含所有修复和9月份数据

This commit is contained in:
KQL
2025-09-03 13:26:13 +08:00
commit 87b06d3176
270 changed files with 116169 additions and 0 deletions

View File

@@ -0,0 +1,288 @@
import { useState, useCallback, useEffect } from "react";
import { useSelector } from "react-redux";
import { Input } from "@arco-design/web-react";
import Modal from "@/components/Modal";
import InfiniteScroll from "@/components/InfiniteScroll";
import toast from "@/components/Toast";
import FILEICON from "@/assets/images/CompanyJobsPage/file_icon.png";
import ResumeInfoModal from "../ResumeInfoModal";
import { getResumesList, submitResume, getPageData } from "@/services";
import "./index.css";
const InputSearch = Input.Search;
const PAGE_SIZE = 10;
export default ({ visible, onClose, data, directToResume = false }) => {
const studentInfo = useSelector((state) => state.student.studentInfo);
const [resumeModalShow, setResumeModalShow] = useState(directToResume);
const [resumeInfoModalShow, setResumeInfoModalShow] = useState(false);
const [resumeInfoData, setResumeInfoData] = useState(null);
const [resumeList, setResumeList] = useState([]); // 简历列表
const [listPage, setListPage] = useState(1);
const [listHasMore, setListHasMore] = useState(true);
// 处理directToResume参数变化
useEffect(() => {
if (visible && directToResume) {
setResumeModalShow(true);
} else if (visible && !directToResume) {
setResumeModalShow(false);
}
}, [visible, directToResume]);
const handleCloseModal = () => {
setResumeModalShow(false);
setResumeList([]); // 清空简历列表
setListPage(1); // 重置分页
setListHasMore(true); // 重置加载更多状态
onClose();
};
const queryResumeList = useCallback(async () => {
const res = await getResumesList({
page: listPage,
pageSize: PAGE_SIZE,
studentId: studentInfo?.id
});
if (res.success) {
setResumeList((prevList) => {
const newList = [...prevList, ...res.data];
if (res.total === newList?.length) {
setListHasMore(false);
} else {
setListPage((prevPage) => prevPage + 1);
}
return newList;
});
}
}, [listPage, studentInfo?.id]);
// 点击立即投递
const handleClickDeliverBtn = (e) => {
e.stopPropagation();
setResumeModalShow(true);
};
const onSearch = (value) => {
// todo
console.log(value);
};
// 选择简历投递
const userResumesClick = async (item) => {
try {
// 调用投递服务
const result = await submitResume({
resumeId: item.id,
jobId: data?.id,
studentId: studentInfo?.id,
resumeTitle: item.title,
jobPosition: data?.position,
company: data?.company
});
if (result.success) {
// 投递成功,显示成功提示
toast.success(`简历"${item.title}"投递成功!`);
// 关闭模态框
handleCloseModal();
// 输出投递成功信息
console.log('投递成功', {
applicationId: result.data.applicationId,
resumeId: item.id,
jobId: data?.id,
resumeTitle: item.title,
jobPosition: data?.position,
submittedAt: result.data.submittedAt
});
} else {
toast.error(result.message || '投递失败,请重试');
}
} catch (error) {
toast.error('投递失败,请重试');
console.error('投递失败:', error);
}
};
// 点击简历详情
const userResumesBtnClick = async (e, item) => {
e.stopPropagation();
try {
// 获取岗位与面试题页面的数据
const pageDataResponse = await getPageData();
if (pageDataResponse.success) {
const pageData = pageDataResponse.data;
// 直接使用简历列表中的模板数据
const selectedTemplate = item.template;
// 找到对应的行业信息
const matchedIndustry = pageData.industries?.find(industry =>
industry.name === item.industry
);
// 传递数据给 ResumeInfoModal
const resumeData = {
selectedTemplate,
studentResume: pageData.myResume,
industry: matchedIndustry,
jobPosition: item.position
};
console.log('加载简历数据:', {
resumeTitle: item.title,
position: item.position,
industry: item.industry
});
setResumeInfoData(resumeData);
setResumeInfoModalShow(true);
} else {
toast.error('加载简历数据失败');
}
} catch (error) {
console.error('获取简历数据失败:', error);
toast.error('加载简历数据失败');
}
};;;
return (
<>
<Modal visible={visible} onClose={handleCloseModal}>
<div className="job-info-modal-content">
{resumeModalShow ? (
<>
<InputSearch
className="job-info-modal-search"
onSearch={onSearch}
searchButton
placeholder="搜索简历"
/>
{
<InfiniteScroll
loadMore={queryResumeList}
hasMore={listHasMore}
empty={resumeList.length === 0}
className={`${
resumeList.length
? "job-info-modal-user-resumes-list"
: "empty-data-wrapper"
}`}
>
{resumeList.map((item) => (
<li
key={item.id}
className="list-item"
onClick={() => userResumesClick(item)}
>
<div className="list-item-info">
<img src={FILEICON} className="file-icon" />
<div className="file-info">
<p className="file-info-targetPosition">
{item.title}
</p>
{item?.skills?.length > 0 && (
<p className="file-info-skills">
{item?.skills?.join("/")}
</p>
)}
</div>
</div>
<div
className="info-btn"
onClick={(e) => userResumesBtnClick(e, item)}
>
简历详情
</div>
</li>
))}
</InfiniteScroll>
}
</>
) : (
<>
<div className="job-info-modal-content-position-info">
<span className="job-info-modal-content-position-info-position">
{data?.position}
</span>
<span className="job-info-modal-content-position-info-num">
该岗位仅剩{data?.remainingPositions}
</span>
<span className="job-info-modal-content-position-info-salary">
{data?.salary}
</span>
</div>
{data?.tags?.length > 0 && (
<ul className="job-info-modal-info-tags">
{data?.tags?.map((tag, index) => (
<li key={index} className="job-info-modal-info-tag">
{tag}
</li>
))}
</ul>
)}
{data?.details?.description && (
<div className="job-info-modal-content-position-info-description">
<p className="description-title">岗位描述</p>
<p className="description-content">{data?.details?.description}</p>
</div>
)}
{data?.details?.requirements?.length > 0 && (
<div className="job-info-modal-content-position-info-requirements">
<p className="requirements-title">岗位要求</p>
<ul className="requirements-content">
{data?.details?.requirements?.map((item, index) => (
<li key={index} className="requirements-item">
{index + 1}. {item}
</li>
))}
</ul>
</div>
)}
{/* {data?.details?.requirements?.length > 0 && (
<div className="job-info-modal-content-position-info-requirements">
<p className="requirements-title">岗位要求</p>
<ul className="requirements-content">
{data?.details?.requirements?.map((item, index) => (
<li key={index} className="requirements-item">
{index + 1}.{item}
</li>
))}
</ul>
</div>
)} */}
{data?.details?.companyInfo && (
<div className="job-info-modal-content-position-info-companyInfo">
<p className="companyInfo-title">公司介绍</p>
<p className="companyInfo-content">
{data?.details?.companyInfo}
</p>
</div>
)}
<div
className="job-info-modal-btn"
onClick={handleClickDeliverBtn}
>
<i />
<span>立即投递</span>
</div>
</>
)}
</div>
</Modal>
<ResumeInfoModal
visible={resumeInfoModalShow}
data={resumeInfoData}
onClose={() => {
setResumeInfoModalShow(false);
setResumeInfoData(null);
}}
/>
</>
);
};