feat: 🎸 封装了一个滚动加载组件
This commit is contained in:
20
src/components/InfiniteScroll/index.css
Normal file
20
src/components/InfiniteScroll/index.css
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
.infinite-scroll-container {
|
||||||
|
height: 100%;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.loading-indicator {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 16px;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-more-data {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
align-items: center;
|
||||||
|
padding: 16px;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
57
src/components/InfiniteScroll/index.jsx
Normal file
57
src/components/InfiniteScroll/index.jsx
Normal file
@@ -0,0 +1,57 @@
|
|||||||
|
import { useEffect, useRef, useState } from "react";
|
||||||
|
import { Empty } from "@arco-design/web-react";
|
||||||
|
import { useSelector } from "react-redux";
|
||||||
|
import { useDispatch } from "react-redux";
|
||||||
|
import { setLoadingTrue, setLoadingFalse } from "@/store/slices/loadingSlice";
|
||||||
|
|
||||||
|
const InfiniteScroll = ({
|
||||||
|
loadMore,
|
||||||
|
hasMore,
|
||||||
|
children,
|
||||||
|
threshold = 50,
|
||||||
|
className = "",
|
||||||
|
}) => {
|
||||||
|
const dispatch = useDispatch();
|
||||||
|
const containerRef = useRef(null);
|
||||||
|
const loading = useSelector((state) => state.loading.value);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleScroll = () => {
|
||||||
|
if (!containerRef.current || loading || !hasMore) return;
|
||||||
|
|
||||||
|
const { scrollTop, scrollHeight, clientHeight } = containerRef.current;
|
||||||
|
|
||||||
|
if (scrollTop + clientHeight >= scrollHeight - threshold) {
|
||||||
|
dispatch(setLoadingTrue());
|
||||||
|
loadMore().finally(() => {
|
||||||
|
dispatch(setLoadingFalse());
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const container = containerRef.current;
|
||||||
|
if (container) {
|
||||||
|
container.addEventListener("scroll", handleScroll);
|
||||||
|
// 初始加载时检查是否需要加载更多
|
||||||
|
handleScroll();
|
||||||
|
}
|
||||||
|
|
||||||
|
return () => {
|
||||||
|
if (container) {
|
||||||
|
container.removeEventListener("scroll", handleScroll);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
}, [loadMore, hasMore, threshold, loading]);
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div
|
||||||
|
ref={containerRef}
|
||||||
|
className={`infinite-scroll-container ${className}`}
|
||||||
|
>
|
||||||
|
{children}
|
||||||
|
{!hasMore && !loading && <Empty description="没有更多了" />}
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
};
|
||||||
|
|
||||||
|
export default InfiniteScroll;
|
||||||
@@ -11,7 +11,7 @@ const Layout = ({ children }) => {
|
|||||||
return (
|
return (
|
||||||
<div className="app-layout">
|
<div className="app-layout">
|
||||||
<Sidebar isCollapsed={isCollapsed} setIsCollapsed={setIsCollapsed} />
|
<Sidebar isCollapsed={isCollapsed} setIsCollapsed={setIsCollapsed} />
|
||||||
<Spin loading={loading} size={40} className="app-layout-spin">
|
<Spin block loading={loading} size={40} className="app-layout-spin">
|
||||||
<main className="main-content">{children}</main>
|
<main className="main-content">{children}</main>
|
||||||
</Spin>
|
</Spin>
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@@ -1,3 +1,4 @@
|
|||||||
|
import { useState, useEffect } from "react";
|
||||||
import StartClass from "./components/StartClass";
|
import StartClass from "./components/StartClass";
|
||||||
import QuickAccess from "./components/QuickAccess";
|
import QuickAccess from "./components/QuickAccess";
|
||||||
import CalendarTaskModule from "./components/CalendarTaskModule";
|
import CalendarTaskModule from "./components/CalendarTaskModule";
|
||||||
@@ -8,6 +9,8 @@ import TaskList from "./components/TaskList";
|
|||||||
import "./index.css";
|
import "./index.css";
|
||||||
|
|
||||||
const Dashboard = () => {
|
const Dashboard = () => {
|
||||||
|
const [data, setData] = useState({});
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="dashboard">
|
<div className="dashboard">
|
||||||
<StageProgress showBlockageAlert={true} />
|
<StageProgress showBlockageAlert={true} />
|
||||||
|
|||||||
@@ -13,6 +13,8 @@
|
|||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
padding: 20px;
|
padding: 20px;
|
||||||
overflow: hidden;
|
overflow: hidden;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
|
||||||
.user-portfolio-search-area {
|
.user-portfolio-search-area {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
@@ -31,9 +33,7 @@
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
.user-portfolio-list {
|
.user-portfolio-list {
|
||||||
width: 100%;
|
|
||||||
overflow-y: auto;
|
overflow-y: auto;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
display: flex;
|
display: flex;
|
||||||
@@ -63,7 +63,8 @@
|
|||||||
align-items: flex-start;
|
align-items: flex-start;
|
||||||
flex-direction: column;
|
flex-direction: column;
|
||||||
|
|
||||||
> span {
|
.user-portfolio-item-title {
|
||||||
|
width: 100%;
|
||||||
border: 1px solid #2c7aff;
|
border: 1px solid #2c7aff;
|
||||||
background-color: #e8f3ff;
|
background-color: #e8f3ff;
|
||||||
height: 20px;
|
height: 20px;
|
||||||
@@ -73,7 +74,11 @@
|
|||||||
color: #2c7aff;
|
color: #2c7aff;
|
||||||
font-size: 12px;
|
font-size: 12px;
|
||||||
font-weight: 600;
|
font-weight: 600;
|
||||||
|
overflow: hidden;
|
||||||
|
text-overflow: ellipsis;
|
||||||
|
white-space: nowrap;
|
||||||
}
|
}
|
||||||
|
|
||||||
> div {
|
> div {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
height: 24px;
|
height: 24px;
|
||||||
|
|||||||
@@ -1,21 +1,29 @@
|
|||||||
import { useState } from "react";
|
import { useState, useEffect } from "react";
|
||||||
import { Input } from "@arco-design/web-react";
|
import { Input, Empty } from "@arco-design/web-react";
|
||||||
import { mockData } from "@/data/mockData";
|
import InfiniteScroll from "@/components/InfiniteScroll";
|
||||||
import ProjectCasesModal from "./components/ProjectCasesModal";
|
import ProjectCasesModal from "./components/ProjectCasesModal";
|
||||||
|
import { getProjectsList } from "@/services/projectLibrary";
|
||||||
|
|
||||||
import "./index.css";
|
import "./index.css";
|
||||||
|
|
||||||
const InputSearch = Input.Search;
|
const InputSearch = Input.Search;
|
||||||
const { projectLibrary } = mockData;
|
const PAGE_SIZE = 10;
|
||||||
|
|
||||||
const ProjectLibrary = () => {
|
const ProjectLibrary = () => {
|
||||||
const [modalData, setModalData] = useState(undefined);
|
const [modalData, setModalData] = useState(undefined);
|
||||||
|
const [projectList, setProjectList] = useState([]);
|
||||||
const [projectCasesModalVisible, setProjectCasesModalVisible] =
|
const [projectCasesModalVisible, setProjectCasesModalVisible] =
|
||||||
useState(false);
|
useState(false);
|
||||||
|
const [page, setPage] = useState(1);
|
||||||
|
const [hasMore, setHasMore] = useState(true);
|
||||||
|
|
||||||
const onSearch = (value) => {
|
const onSearch = (value) => {
|
||||||
console.log(value);
|
setProjectList([]);
|
||||||
|
setHasMore(true);
|
||||||
|
setPage(1);
|
||||||
|
fetchProjects(value, 1);
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleProjectClick = (item) => {
|
const handleProjectClick = (item) => {
|
||||||
setModalData(item);
|
setModalData(item);
|
||||||
setProjectCasesModalVisible(true);
|
setProjectCasesModalVisible(true);
|
||||||
@@ -26,6 +34,24 @@ const ProjectLibrary = () => {
|
|||||||
setModalData(undefined);
|
setModalData(undefined);
|
||||||
};
|
};
|
||||||
|
|
||||||
|
const fetchProjects = async (searchValue = "", pageNum) => {
|
||||||
|
try {
|
||||||
|
// 这里使用真实API替换模拟数据
|
||||||
|
const response = await getProjectsList({
|
||||||
|
search: searchValue,
|
||||||
|
page: pageNum ?? page,
|
||||||
|
pageSize: PAGE_SIZE,
|
||||||
|
});
|
||||||
|
if (response.success) {
|
||||||
|
setProjectList((prevList) => [...prevList, ...response.data]);
|
||||||
|
setHasMore(response?.data.length === PAGE_SIZE);
|
||||||
|
setPage((prevPage) => prevPage + 1);
|
||||||
|
}
|
||||||
|
} catch (error) {
|
||||||
|
console.error("Failed to fetch projects:", error);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="user-portfolio-page">
|
<div className="user-portfolio-page">
|
||||||
<div className="user-portfolio-wrapper">
|
<div className="user-portfolio-wrapper">
|
||||||
@@ -36,17 +62,22 @@ const ProjectLibrary = () => {
|
|||||||
searchButton="搜索"
|
searchButton="搜索"
|
||||||
/>
|
/>
|
||||||
</div>
|
</div>
|
||||||
<ul className="user-portfolio-list">
|
<InfiniteScroll
|
||||||
{projectLibrary?.projects?.map((item) => (
|
loadMore={fetchProjects}
|
||||||
|
hasMore={hasMore}
|
||||||
|
threshold={100}
|
||||||
|
className="user-portfolio-list"
|
||||||
|
>
|
||||||
|
{projectList.map((item) => (
|
||||||
<li className="user-portfolio-item" key={item.id}>
|
<li className="user-portfolio-item" key={item.id}>
|
||||||
<span>{item.subtitle}</span>
|
<p className="user-portfolio-item-title">{item.description}</p>
|
||||||
<div>
|
<div>
|
||||||
<p>{item.title}</p>
|
<p>{item.name}</p>
|
||||||
<span onClick={() => handleProjectClick(item)}>详情 > </span>
|
<span onClick={() => handleProjectClick(item)}>详情 ></span>
|
||||||
</div>
|
</div>
|
||||||
</li>
|
</li>
|
||||||
))}
|
))}
|
||||||
</ul>
|
</InfiniteScroll>
|
||||||
</div>
|
</div>
|
||||||
<ProjectCasesModal
|
<ProjectCasesModal
|
||||||
data={modalData}
|
data={modalData}
|
||||||
|
|||||||
9
src/services/dashboard.js
Normal file
9
src/services/dashboard.js
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
import request from "@/utils/request";
|
||||||
|
// 获取主页信息
|
||||||
|
export async function getDashboardStatistics(studentId) {
|
||||||
|
return request.get(`/api/dashboard/stats/${studentId}`);
|
||||||
|
}
|
||||||
|
// 获取学习进度
|
||||||
|
export async function getLearningProgressSummary(studentId) {
|
||||||
|
return request.get(`/api/dashboard/learning-summary/${studentId}`);
|
||||||
|
}
|
||||||
7
src/services/index.js
Normal file
7
src/services/index.js
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
import {
|
||||||
|
getDashboardStatistics,
|
||||||
|
getLearningProgressSummary,
|
||||||
|
} from "./dashboard";
|
||||||
|
import { getProjectsList } from "./projectLibrary";
|
||||||
|
|
||||||
|
export { getDashboardStatistics, getLearningProgressSummary, getProjectsList };
|
||||||
5
src/services/projectLibrary.js
Normal file
5
src/services/projectLibrary.js
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import request from "@/utils/request";
|
||||||
|
// 获取项目列表
|
||||||
|
export async function getProjectsList(params) {
|
||||||
|
return request.get(`/api/projects`, { params });
|
||||||
|
}
|
||||||
Reference in New Issue
Block a user