You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
memos/web/src/components/QueryList.tsx

171 lines
5.1 KiB
TypeScript

import { useContext, useEffect } from "react";
import appContext from "../stores/appContext";
import useToggle from "../hooks/useToggle";
import useLoading from "../hooks/useLoading";
import Only from "./common/OnlyWhen";
import utils from "../helpers/utils";
import toastHelper from "./Toast";
import { locationService, queryService } from "../services";
import showCreateQueryDialog from "./CreateQueryDialog";
import "../less/query-list.less";
interface Props {}
const QueryList: React.FC<Props> = () => {
const {
queryState: { queries },
locationState: {
query: { filter },
},
} = useContext(appContext);
const loadingState = useLoading();
const sortedQueries = queries
.sort((a, b) => utils.getTimeStampByDate(b.createdAt) - utils.getTimeStampByDate(a.createdAt))
.sort((a, b) => utils.getTimeStampByDate(b.pinnedAt ?? 0) - utils.getTimeStampByDate(a.pinnedAt ?? 0));
useEffect(() => {
queryService
.getMyAllQueries()
.catch(() => {
// do nth
})
.finally(() => {
loadingState.setFinish();
});
}, []);
return (
<div className="queries-wrapper">
<p className="title-text">
<span className="normal-text"></span>
<span className="btn" onClick={() => showCreateQueryDialog()}>
+
</span>
</p>
<Only when={loadingState.isSucceed && sortedQueries.length === 0}>
<div className="create-query-btn-container">
<span className="btn" onClick={() => showCreateQueryDialog()}>
</span>
</div>
</Only>
<div className="queries-container">
{sortedQueries.map((q) => {
return <QueryItemContainer key={q.id} query={q} isActive={q.id === filter} />;
})}
</div>
</div>
);
};
interface QueryItemContainerProps {
query: Model.Query;
isActive: boolean;
}
const QueryItemContainer: React.FC<QueryItemContainerProps> = (props: QueryItemContainerProps) => {
const { query, isActive } = props;
const [showActionBtns, toggleShowActionBtns] = useToggle(false);
const [showConfirmDeleteBtn, toggleConfirmDeleteBtn] = useToggle(false);
const handleQueryClick = () => {
if (isActive) {
locationService.setMemoFilter("");
} else {
if (!["/", "/recycle"].includes(locationService.getState().pathname)) {
locationService.setPathname("/");
}
locationService.setMemoFilter(query.id);
}
};
const handleShowActionBtnClick = (event: React.MouseEvent) => {
event.stopPropagation();
toggleShowActionBtns();
};
const handleActionBtnContainerMouseLeave = () => {
toggleShowActionBtns(false);
};
const handleDeleteMemoClick = async (event: React.MouseEvent) => {
event.stopPropagation();
if (showConfirmDeleteBtn) {
try {
await queryService.deleteQuery(query.id);
} catch (error: any) {
toastHelper.error(error.message);
}
} else {
toggleConfirmDeleteBtn();
}
};
const handleEditQueryBtnClick = (event: React.MouseEvent) => {
event.stopPropagation();
showCreateQueryDialog(query.id);
};
const handlePinQueryBtnClick = async (event: React.MouseEvent) => {
event.stopPropagation();
try {
if (query.pinnedAt) {
await queryService.unpinQuery(query.id);
queryService.editQuery({
...query,
pinnedAt: "",
});
} else {
await queryService.pinQuery(query.id);
queryService.editQuery({
...query,
pinnedAt: utils.getDateTimeString(Date.now()),
});
}
} catch (error) {
// do nth
}
};
const handleDeleteBtnMouseLeave = () => {
toggleConfirmDeleteBtn(false);
};
return (
<>
<div className={`query-item-container ${isActive ? "active" : ""}`} onClick={handleQueryClick}>
<div className="query-text-container">
<span className="icon-text">#</span>
<span className="query-text">{query.title}</span>
</div>
<div className="btns-container">
<span className="action-btn toggle-btn" onClick={handleShowActionBtnClick}>
<img className="icon-img" src={`/icons/more${isActive ? "-white" : ""}.svg`} />
</span>
<div className={`action-btns-wrapper ${showActionBtns ? "" : "hidden"}`} onMouseLeave={handleActionBtnContainerMouseLeave}>
<div className="action-btns-container">
<span className="btn" onClick={handlePinQueryBtnClick}>
{query.pinnedAt ? "取消置顶" : "置顶"}
</span>
<span className="btn" onClick={handleEditQueryBtnClick}>
</span>
<span
className={`btn delete-btn ${showConfirmDeleteBtn ? "final-confirm" : ""}`}
onClick={handleDeleteMemoClick}
onMouseLeave={handleDeleteBtnMouseLeave}
>
{showConfirmDeleteBtn ? "确定删除!" : "删除"}
</span>
</div>
</div>
</div>
</div>
</>
);
};
export default QueryList;