feature/IO-3291-Tasks-Notifications: Checkpoint
This commit is contained in:
@@ -8,11 +8,10 @@ import "./task-center.styles.scss";
|
|||||||
|
|
||||||
const { Text, Title } = Typography;
|
const { Text, Title } = Typography;
|
||||||
|
|
||||||
const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick }, ref) => {
|
const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick, onLoadMore, totalTasks }, ref) => {
|
||||||
const { t } = useTranslation();
|
const { t } = useTranslation();
|
||||||
const virtuosoRef = useRef(null);
|
const virtuosoRef = useRef(null);
|
||||||
|
|
||||||
// Organize tasks into sections
|
|
||||||
const sections = useMemo(() => {
|
const sections = useMemo(() => {
|
||||||
const now = day();
|
const now = day();
|
||||||
const today = now.startOf("day");
|
const today = now.startOf("day");
|
||||||
@@ -80,6 +79,11 @@ const TaskCenterComponent = forwardRef(({ visible, tasks, loading, onTaskClick }
|
|||||||
totalCount={sections.length}
|
totalCount={sections.length}
|
||||||
itemContent={(index, section) => renderSection(section, index)}
|
itemContent={(index, section) => renderSection(section, index)}
|
||||||
/>
|
/>
|
||||||
|
{tasks.length < totalTasks && (
|
||||||
|
<button onClick={onLoadMore} disabled={loading}>
|
||||||
|
{t("general.labels.load_more")}
|
||||||
|
</button>
|
||||||
|
)}
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -5,10 +5,10 @@ import { createStructuredSelector } from "reselect";
|
|||||||
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
import { selectBodyshop, selectCurrentUser } from "../../redux/user/user.selectors";
|
||||||
import { useSocket } from "../../contexts/SocketIO/useSocket";
|
import { useSocket } from "../../contexts/SocketIO/useSocket";
|
||||||
import { useIsEmployee } from "../../utils/useIsEmployee";
|
import { useIsEmployee } from "../../utils/useIsEmployee";
|
||||||
import { QUERY_MY_TASKS_PAGINATED } from "../../graphql/tasks.queries";
|
|
||||||
import TaskCenterComponent from "./task-center.component";
|
import TaskCenterComponent from "./task-center.component";
|
||||||
import dayjs from "../../utils/day";
|
import dayjs from "../../utils/day";
|
||||||
import { setModalContext } from "../../redux/modals/modals.actions";
|
import { setModalContext } from "../../redux/modals/modals.actions";
|
||||||
|
import { QUERY_MY_ACTIVE_TASKS_PAGINATED } from "../../graphql/tasks.queries";
|
||||||
|
|
||||||
const POLL_INTERVAL = 60; // seconds
|
const POLL_INTERVAL = 60; // seconds
|
||||||
|
|
||||||
@@ -23,47 +23,29 @@ const mapDispatchToProps = (dispatch) => ({
|
|||||||
|
|
||||||
const TaskCenterContainer = ({ visible, onClose, bodyshop, currentUser, setTaskUpsertContext }) => {
|
const TaskCenterContainer = ({ visible, onClose, bodyshop, currentUser, setTaskUpsertContext }) => {
|
||||||
const [tasks, setTasks] = useState([]);
|
const [tasks, setTasks] = useState([]);
|
||||||
const [loading, setLoading] = useState(false);
|
|
||||||
const { isConnected } = useSocket();
|
const { isConnected } = useSocket();
|
||||||
const isEmployee = useIsEmployee(bodyshop, currentUser);
|
const isEmployee = useIsEmployee(bodyshop, currentUser);
|
||||||
|
|
||||||
// Compute assignedToId with useMemo to ensure stability
|
|
||||||
const assignedToId = useMemo(() => {
|
const assignedToId = useMemo(() => {
|
||||||
const employee = bodyshop?.employees?.find((e) => e.user_email === currentUser?.email);
|
const employee = bodyshop?.employees?.find((e) => e.user_email === currentUser?.email);
|
||||||
if (employee?.id) {
|
return employee?.id || null;
|
||||||
console.log("AssignedToId computed:", employee.id); // Debug log
|
|
||||||
return employee.id;
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}, [bodyshop, currentUser]);
|
}, [bodyshop, currentUser]);
|
||||||
|
|
||||||
const where = useMemo(
|
|
||||||
() => ({
|
|
||||||
assigned_to: { _eq: assignedToId },
|
|
||||||
deleted: { _eq: false },
|
|
||||||
completed: { _eq: false }
|
|
||||||
}),
|
|
||||||
[assignedToId]
|
|
||||||
);
|
|
||||||
|
|
||||||
const {
|
const {
|
||||||
data,
|
data,
|
||||||
loading: queryLoading,
|
loading: queryLoading,
|
||||||
refetch
|
fetchMore
|
||||||
} = useQuery(QUERY_MY_TASKS_PAGINATED, {
|
} = useQuery(QUERY_MY_ACTIVE_TASKS_PAGINATED, {
|
||||||
variables: {
|
variables: {
|
||||||
bodyshop: bodyshop?.id,
|
bodyshop: bodyshop?.id,
|
||||||
assigned_to: assignedToId,
|
assigned_to: assignedToId,
|
||||||
where,
|
|
||||||
offset: 0,
|
offset: 0,
|
||||||
limit: 50,
|
limit: 50,
|
||||||
order: [{ due_date: "asc_nulls_last" }, { created_at: "desc" }]
|
order: [{ due_date: "asc_nulls_last" }, { created_at: "desc" }]
|
||||||
},
|
},
|
||||||
// Skip query if any required data is missing
|
|
||||||
skip: !bodyshop?.id || !assignedToId || !isEmployee || !currentUser?.email,
|
skip: !bodyshop?.id || !assignedToId || !isEmployee || !currentUser?.email,
|
||||||
fetchPolicy: "cache-and-network",
|
fetchPolicy: "cache-and-network",
|
||||||
pollInterval: isConnected ? 0 : dayjs.duration(POLL_INTERVAL, "seconds").asMilliseconds(),
|
pollInterval: isConnected ? 0 : dayjs.duration(POLL_INTERVAL, "seconds").asMilliseconds(),
|
||||||
// Log errors for debugging
|
|
||||||
onError: (error) => {
|
onError: (error) => {
|
||||||
console.error("Query error:", error);
|
console.error("Query error:", error);
|
||||||
}
|
}
|
||||||
@@ -75,14 +57,28 @@ const TaskCenterContainer = ({ visible, onClose, bodyshop, currentUser, setTaskU
|
|||||||
}
|
}
|
||||||
}, [data]);
|
}, [data]);
|
||||||
|
|
||||||
|
const handleLoadMore = () => {
|
||||||
|
fetchMore({
|
||||||
|
variables: {
|
||||||
|
offset: tasks.length
|
||||||
|
},
|
||||||
|
updateQuery: (prev, { fetchMoreResult }) => {
|
||||||
|
if (!fetchMoreResult) return prev;
|
||||||
|
return {
|
||||||
|
...prev,
|
||||||
|
tasks: [...prev.tasks, ...fetchMoreResult.tasks]
|
||||||
|
};
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
const handleTaskClick = useCallback(
|
const handleTaskClick = useCallback(
|
||||||
(id) => {
|
(id) => {
|
||||||
const task = tasks.find((t) => t.id === id);
|
const task = tasks.find((t) => t.id === id);
|
||||||
if (task) {
|
if (task) {
|
||||||
setTaskUpsertContext({
|
setTaskUpsertContext({
|
||||||
context: {
|
context: {
|
||||||
taskId: task.id,
|
existingTask: task
|
||||||
view: true
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
@@ -95,10 +91,11 @@ const TaskCenterContainer = ({ visible, onClose, bodyshop, currentUser, setTaskU
|
|||||||
visible={visible}
|
visible={visible}
|
||||||
onClose={onClose}
|
onClose={onClose}
|
||||||
tasks={tasks}
|
tasks={tasks}
|
||||||
loading={loading || queryLoading}
|
loading={queryLoading}
|
||||||
onTaskClick={handleTaskClick}
|
onTaskClick={handleTaskClick}
|
||||||
|
onLoadMore={handleLoadMore}
|
||||||
|
totalTasks={data?.tasks_aggregate?.aggregate?.count}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
export default connect(mapStateToProps, mapDispatchToProps)(TaskCenterContainer);
|
export default connect(mapStateToProps, mapDispatchToProps)(TaskCenterContainer);
|
||||||
|
|||||||
@@ -7,6 +7,7 @@ import { connect } from "react-redux";
|
|||||||
import LoadingSkeleton from "../loading-skeleton/loading-skeleton.component.jsx";
|
import LoadingSkeleton from "../loading-skeleton/loading-skeleton.component.jsx";
|
||||||
import JobSearchSelectComponent from "../job-search-select/job-search-select.component.jsx";
|
import JobSearchSelectComponent from "../job-search-select/job-search-select.component.jsx";
|
||||||
import DateTimePicker from "../form-date-time-picker/form-date-time-picker.component.jsx";
|
import DateTimePicker from "../form-date-time-picker/form-date-time-picker.component.jsx";
|
||||||
|
import { Link } from "react-router-dom";
|
||||||
|
|
||||||
const mapStateToProps = createStructuredSelector({
|
const mapStateToProps = createStructuredSelector({
|
||||||
bodyshop: selectBodyshop,
|
bodyshop: selectBodyshop,
|
||||||
@@ -41,7 +42,7 @@ export function TaskUpsertModalComponent({
|
|||||||
];
|
];
|
||||||
|
|
||||||
const generatePresets = (job) => {
|
const generatePresets = (job) => {
|
||||||
if (!job || !selectedJobDetails) return datePickerPresets; // return default presets if no job selected
|
if (!job || !selectedJobDetails) return datePickerPresets;
|
||||||
const relativePresets = [];
|
const relativePresets = [];
|
||||||
|
|
||||||
if (selectedJobDetails?.scheduled_completion) {
|
if (selectedJobDetails?.scheduled_completion) {
|
||||||
@@ -96,13 +97,8 @@ export function TaskUpsertModalComponent({
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
|
||||||
* Change the selected job id
|
|
||||||
* @param jobId
|
|
||||||
*/
|
|
||||||
const changeJobId = (jobId) => {
|
const changeJobId = (jobId) => {
|
||||||
setSelectedJobId(jobId || null);
|
setSelectedJobId(jobId || null);
|
||||||
// Reset the form fields when selectedJobId changes
|
|
||||||
clearRelations();
|
clearRelations();
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -162,6 +158,13 @@ export function TaskUpsertModalComponent({
|
|||||||
required: true
|
required: true
|
||||||
}
|
}
|
||||||
]}
|
]}
|
||||||
|
extra={
|
||||||
|
existingTask && selectedJobId ? (
|
||||||
|
<div style={{ textAlign: "right" }}>
|
||||||
|
<Link to={`/manage/jobs/${selectedJobId}`}>{t("tasks.labels.go_to_job")}</Link>
|
||||||
|
</div>
|
||||||
|
) : null
|
||||||
|
}
|
||||||
>
|
>
|
||||||
<JobSearchSelectComponent
|
<JobSearchSelectComponent
|
||||||
placeholder={t("tasks.placeholders.jobid")}
|
placeholder={t("tasks.placeholders.jobid")}
|
||||||
@@ -202,7 +205,18 @@ export function TaskUpsertModalComponent({
|
|||||||
</Form.Item>
|
</Form.Item>
|
||||||
</Col>
|
</Col>
|
||||||
<Col span={8}>
|
<Col span={8}>
|
||||||
<Form.Item label={t("tasks.fields.billid")} name="billid">
|
<Form.Item
|
||||||
|
label={t("tasks.fields.billid")}
|
||||||
|
name="billid"
|
||||||
|
extra={
|
||||||
|
form.getFieldValue("billid") ? (
|
||||||
|
<Link to={`/manage/bills?billid=${form.getFieldValue("billid")}`}>
|
||||||
|
{t("tasks.links.go_to_bill")} (
|
||||||
|
{selectedJobDetails?.bills?.find((bill) => bill.id === form.getFieldValue("billid"))?.invoice_number})
|
||||||
|
</Link>
|
||||||
|
) : null
|
||||||
|
}
|
||||||
|
>
|
||||||
<Select
|
<Select
|
||||||
allowClear
|
allowClear
|
||||||
placeholder={t("tasks.placeholders.billid")}
|
placeholder={t("tasks.placeholders.billid")}
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
import { useMutation, useQuery } from "@apollo/client";
|
import { useMutation, useQuery } from "@apollo/client";
|
||||||
import { Form, Modal } from "antd";
|
import { Form, Modal } from "antd";
|
||||||
import React, { useEffect, useState } from "react";
|
import { useEffect, useState } from "react";
|
||||||
import { useTranslation } from "react-i18next";
|
import { useTranslation } from "react-i18next";
|
||||||
import { connect } from "react-redux";
|
import { connect } from "react-redux";
|
||||||
import { createStructuredSelector } from "reselect";
|
import { createStructuredSelector } from "reselect";
|
||||||
|
|||||||
@@ -308,6 +308,43 @@ export const QUERY_MY_COMPLETE_TASKS_PAGINATED = gql`
|
|||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
|
export const QUERY_MY_ACTIVE_TASKS_PAGINATED = gql`
|
||||||
|
${PARTIAL_TASK_FIELDS}
|
||||||
|
query QUERY_MY_ACTIVE_TASKS_PAGINATED(
|
||||||
|
$assigned_to: uuid!
|
||||||
|
$bodyshop: uuid!
|
||||||
|
$offset: Int
|
||||||
|
$limit: Int
|
||||||
|
$order: [tasks_order_by!]!
|
||||||
|
) {
|
||||||
|
tasks(
|
||||||
|
offset: $offset
|
||||||
|
limit: $limit
|
||||||
|
order_by: $order
|
||||||
|
where: {
|
||||||
|
assigned_to: { _eq: $assigned_to }
|
||||||
|
bodyshopid: { _eq: $bodyshop }
|
||||||
|
deleted: { _eq: false }
|
||||||
|
completed: { _eq: false }
|
||||||
|
}
|
||||||
|
) {
|
||||||
|
...TaskFields
|
||||||
|
}
|
||||||
|
tasks_aggregate(
|
||||||
|
where: {
|
||||||
|
assigned_to: { _eq: $assigned_to }
|
||||||
|
bodyshopid: { _eq: $bodyshop }
|
||||||
|
deleted: { _eq: false }
|
||||||
|
completed: { _eq: false }
|
||||||
|
}
|
||||||
|
) {
|
||||||
|
aggregate {
|
||||||
|
count
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
export const QUERY_MY_TASKS_PAGINATED = gql`
|
export const QUERY_MY_TASKS_PAGINATED = gql`
|
||||||
${PARTIAL_TASK_FIELDS}
|
${PARTIAL_TASK_FIELDS}
|
||||||
query QUERY_MY_TASKS_PAGINATED(
|
query QUERY_MY_TASKS_PAGINATED(
|
||||||
|
|||||||
Reference in New Issue
Block a user