feature/IO-3291-Tasks-Notifications: Checkpoint
This commit is contained in:
@@ -4,6 +4,7 @@ import { useTranslation } from "react-i18next";
|
||||
import { forwardRef, useMemo, useRef } from "react";
|
||||
import day from "../../utils/day.js";
|
||||
import "./task-center.styles.scss";
|
||||
import { ArrowRightOutlined, CalendarOutlined, ClockCircleOutlined, QuestionCircleOutlined } from "@ant-design/icons";
|
||||
|
||||
const { Title } = Typography;
|
||||
|
||||
@@ -11,6 +12,12 @@ const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick,
|
||||
const { t } = useTranslation();
|
||||
const virtuosoRef = useRef(null);
|
||||
|
||||
const sectionIcons = {
|
||||
[t("tasks.labels.overdue")]: <ClockCircleOutlined style={{ marginRight: 8 }} />,
|
||||
[t("tasks.labels.due_today")]: <CalendarOutlined style={{ marginRight: 8 }} />,
|
||||
[t("tasks.labels.upcoming")]: <ArrowRightOutlined style={{ marginRight: 8 }} />,
|
||||
[t("tasks.labels.no_due_date")]: <QuestionCircleOutlined style={{ marginRight: 8 }} />
|
||||
};
|
||||
const groupedItems = useMemo(() => {
|
||||
const now = day();
|
||||
const today = now.startOf("day");
|
||||
@@ -20,8 +27,7 @@ const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick,
|
||||
const upcoming = tasks.filter((t) => t.due_date && day(t.due_date).isAfter(today));
|
||||
const noDueDate = tasks.filter((t) => !t.due_date);
|
||||
|
||||
const makeGroup = (label, data) =>
|
||||
data.length ? [{ type: "header", label }, ...data.map((task) => ({ type: "task", task }))] : [];
|
||||
const makeGroup = (label, data) => (data.length ? [{ type: "section", label, tasks: data }] : []);
|
||||
|
||||
return [
|
||||
...makeGroup(t("tasks.labels.overdue"), overdue),
|
||||
@@ -31,63 +37,51 @@ const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick,
|
||||
];
|
||||
}, [tasks, t]);
|
||||
|
||||
const renderTask = (index, task) => {
|
||||
const handleClick = () => onTaskClick(task.id);
|
||||
|
||||
const getPriorityColor = (priority) => {
|
||||
switch (priority) {
|
||||
case 1:
|
||||
return "red";
|
||||
case 2:
|
||||
return "orange";
|
||||
case 3:
|
||||
return "green";
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
};
|
||||
|
||||
const priorityColor = getPriorityColor(task.priority);
|
||||
|
||||
return (
|
||||
<div key={`${task.id}-${index}`} className="task-item" onClick={handleClick}>
|
||||
<Badge color={priorityColor || undefined} dot={!!priorityColor} offset={[-4, 4]}>
|
||||
<div className="task-content">
|
||||
<div className="task-header-row">
|
||||
<span className="task-title">{task.title}</span>
|
||||
{task.due_date && (
|
||||
<span className="task-due-date">
|
||||
{t("tasks.labels.due")}: {day(task.due_date).fromNow()}
|
||||
</span>
|
||||
)}
|
||||
</div>
|
||||
|
||||
{/*{task.description && <div className="task-description">{task.description}</div>}*/}
|
||||
|
||||
<div className="task-ro-number">
|
||||
{t("notifications.labels.ro-number", {
|
||||
ro_number: task.job?.ro_number || t("general.labels.na")
|
||||
})}
|
||||
</div>
|
||||
</div>
|
||||
</Badge>
|
||||
</div>
|
||||
);
|
||||
};
|
||||
const renderItem = (index, item) => {
|
||||
if (item.type === "header") {
|
||||
return (
|
||||
<div key={`header-${index}`} className="task-section">
|
||||
<Title level={4} className="section-title">
|
||||
{item.label}
|
||||
</Title>
|
||||
</div>
|
||||
);
|
||||
const getPriorityColor = (priority) => {
|
||||
switch (priority) {
|
||||
case 1:
|
||||
return "red";
|
||||
case 2:
|
||||
return "orange";
|
||||
case 3:
|
||||
return "green";
|
||||
default:
|
||||
return null;
|
||||
}
|
||||
|
||||
return renderTask(index, item.task);
|
||||
};
|
||||
|
||||
const renderSection = (section, index) => (
|
||||
<div key={`section-${index}`} className="task-section">
|
||||
<Title level={4} className="section-title">
|
||||
{sectionIcons[section.label]}
|
||||
{section.label}
|
||||
</Title>
|
||||
<table className="task-table">
|
||||
<tbody>
|
||||
{section.tasks.map((task) => {
|
||||
const priorityColor = getPriorityColor(task.priority);
|
||||
return (
|
||||
<tr key={task.id} className="task-row" onClick={() => onTaskClick(task.id)}>
|
||||
<td className="task-title-cell">
|
||||
<div className="task-title">{task.title}</div>
|
||||
<div className="task-ro-number">
|
||||
{t("notifications.labels.ro-number", {
|
||||
ro_number: task.job?.ro_number || t("general.labels.na")
|
||||
})}
|
||||
</div>
|
||||
</td>
|
||||
<td className="task-due-cell">
|
||||
{task.due_date && <span>{day(task.due_date).fromNow()}</span>}
|
||||
{!!priorityColor && <Badge color={priorityColor} dot style={{ marginLeft: 6 }} />}
|
||||
</td>
|
||||
</tr>
|
||||
);
|
||||
})}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
);
|
||||
|
||||
return (
|
||||
<div className={`task-center ${visible ? "visible" : ""}`} ref={ref}>
|
||||
<div className="task-header">
|
||||
@@ -99,7 +93,7 @@ const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick,
|
||||
ref={virtuosoRef}
|
||||
style={{ height: "400px", width: "100%" }}
|
||||
data={groupedItems}
|
||||
itemContent={renderItem}
|
||||
itemContent={(index, section) => renderSection(section, index)}
|
||||
/>
|
||||
|
||||
{tasks.length < totalTasks && (
|
||||
|
||||
@@ -28,20 +28,19 @@
|
||||
h3 {
|
||||
margin: 0;
|
||||
font-size: 13px;
|
||||
color: rgba(0, 0, 0, 0.85);
|
||||
}
|
||||
}
|
||||
|
||||
.task-section {
|
||||
margin-bottom: 2px;
|
||||
margin: 0;
|
||||
padding: 0;
|
||||
}
|
||||
|
||||
.section-title {
|
||||
padding: 3px 10px;
|
||||
background: #f5f5f5;
|
||||
margin: 0;
|
||||
font-size: 12px;
|
||||
color: rgba(0, 0, 0, 0.85);
|
||||
background: #f5f5f5;
|
||||
font-weight: 500;
|
||||
border-bottom: 1px solid #e8e8e8;
|
||||
position: sticky;
|
||||
@@ -49,64 +48,52 @@
|
||||
z-index: 1;
|
||||
}
|
||||
|
||||
.task-item {
|
||||
padding: 6px 10px;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
background: #fff;
|
||||
cursor: pointer;
|
||||
|
||||
&:hover {
|
||||
background: #f5f5f5;
|
||||
}
|
||||
|
||||
.task-content {
|
||||
display: flex;
|
||||
flex-direction: column;
|
||||
gap: 2px;
|
||||
}
|
||||
|
||||
.task-header-row {
|
||||
display: flex;
|
||||
justify-content: space-between;
|
||||
align-items: center;
|
||||
|
||||
.task-title {
|
||||
font-size: 13px;
|
||||
font-weight: 600;
|
||||
color: rgba(0, 0, 0, 0.9);
|
||||
margin-right: 6px;
|
||||
flex: 1;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.task-due-date {
|
||||
font-size: 11px;
|
||||
color: rgba(0, 0, 0, 0.45);
|
||||
white-space: nowrap;
|
||||
}
|
||||
}
|
||||
|
||||
.task-description {
|
||||
font-size: 12px;
|
||||
color: rgba(0, 0, 0, 0.65);
|
||||
}
|
||||
|
||||
.task-ro-number {
|
||||
font-size: 11px;
|
||||
color: rgba(0, 0, 0, 0.45);
|
||||
}
|
||||
}
|
||||
|
||||
.ant-badge {
|
||||
.task-table {
|
||||
width: 100%;
|
||||
|
||||
.ant-badge-dot {
|
||||
width: 6px;
|
||||
height: 6px;
|
||||
top: 2px;
|
||||
right: 2px;
|
||||
border-collapse: collapse;
|
||||
|
||||
.task-row {
|
||||
cursor: pointer;
|
||||
border-bottom: 1px solid #f0f0f0;
|
||||
|
||||
&:hover {
|
||||
background: #f5f5f5;
|
||||
}
|
||||
|
||||
td {
|
||||
padding: 6px 8px;
|
||||
vertical-align: top;
|
||||
font-size: 12px;
|
||||
line-height: 1.2;
|
||||
}
|
||||
|
||||
.task-title-cell {
|
||||
width: 100%;
|
||||
max-width: 300px; // or whatever fits your layout
|
||||
|
||||
.task-title {
|
||||
font-weight: 500;
|
||||
white-space: nowrap;
|
||||
overflow: hidden;
|
||||
text-overflow: ellipsis;
|
||||
max-width: 100%; // Or a specific width if you want more control
|
||||
display: inline-block;
|
||||
vertical-align: middle;
|
||||
}
|
||||
|
||||
.task-ro-number {
|
||||
margin-top: 5px;
|
||||
font-size: 11px;
|
||||
color: rgba(0, 0, 0, 0.45);
|
||||
}
|
||||
}
|
||||
|
||||
.task-due-cell {
|
||||
text-align: right;
|
||||
white-space: nowrap;
|
||||
color: rgba(0, 0, 0, 0.45);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
@@ -129,4 +116,9 @@
|
||||
cursor: not-allowed;
|
||||
}
|
||||
}
|
||||
|
||||
.ReactVirtuoso__item {
|
||||
margin: 0 !important;
|
||||
padding: 0 !important;
|
||||
}
|
||||
}
|
||||
|
||||
Reference in New Issue
Block a user