statistic graph and table

main
Dilmurod 11 months ago
parent a6fd4e318a
commit 7e02e9efdd

@ -31,7 +31,7 @@
"react-redux": "^8.1.3", "react-redux": "^8.1.3",
"react-router-dom": "^6.15.0", "react-router-dom": "^6.15.0",
"react-scripts": "5.0.1", "react-scripts": "5.0.1",
"recharts": "^2.10.4", "recharts": "^2.13.3",
"redux": "^4.2.1", "redux": "^4.2.1",
"redux-thunk": "^2.4.2", "redux-thunk": "^2.4.2",
"typescript": "^4.9.5", "typescript": "^4.9.5",

@ -43,6 +43,11 @@
color: #fff; color: #fff;
} }
.tt-icon-collapsed {
width: 31.05px;
height: 32px;
}
.isnot { .isnot {
display: none; display: none;
} }
@ -241,15 +246,25 @@
transition: 0.5s; transition: 0.5s;
} }
.btn-refresh-false:active {
border: 1px solid rgba(249, 158, 44, 1);
color: white;
transition: 0.5s;
}
.btn-refresh-dark:active {
color: white;
transition: 0.5s;
}
.title { .title {
font-family: Inter; font-family: Inter;
font-size: 24px; font-size: 24px;
font-weight: 700; font-weight: 700;
line-height: 28px; line-height: 28px;
letter-spacing: -0.04em; letter-spacing: -4%;
text-align: left; text-align: left;
margin-right: 12px; margin-right: 12px;
color: #bbb;
} }
.filter { .filter {
@ -262,9 +277,9 @@
align-items: end; align-items: end;
justify-content: space-between; justify-content: space-between;
flex-direction: column; flex-direction: column;
gap: 10px;
} }
.search-input-false { .search-input-false {
border: none; border: none;
outline: none; outline: none;
@ -541,11 +556,11 @@
border-radius: 8px; border-radius: 8px;
border: 1px solid #777777; border: 1px solid #777777;
box-shadow: 0px 1px 3px 0px rgba(20, 22, 41, 0.1); box-shadow: 0px 1px 3px 0px rgba(20, 22, 41, 0.1);
background: #333; background: #fff;
display: flex; display: flex;
align-items: center; align-items: center;
cursor: pointer; cursor: pointer;
color: #bbb; color: #0f111c;
} }
.btn-modal-action-false:hover { .btn-modal-action-false:hover {
background: #e6e6e6; background: #e6e6e6;
@ -735,32 +750,39 @@
} }
.card_stat { .card_stat {
font-weight: 500; /* font-weight: 500; */
font-size: 20px; /* font-size: 20px; */
padding: 20px; /* padding: 20px; */
background: #deeeff; background-color: #deeeff;
border-radius: 2px; /* border-radius: 2px; */
width: 200px; color: #fff;
height: 180px; width: 156px;
text-align: center; height: 159px;
-webkit-box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08); border-radius: 8px;
display: flex;
align-items: center;
justify-content: center;
flex-direction: column;
gap: 10px;
/* text-align: center; */
/* -webkit-box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08);
-moz-box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08); -moz-box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08);
box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08); box-shadow: 0px 0px 12px 1px rgba(34, 60, 80, 0.08);
cursor: pointer; cursor: pointer;
transition: 0.4s; transition: 0.4s; */
}
.card_stat:hover {
background: #cfe4fc;
} }
.card_stat span { .card_stat span {
font-weight: 700; font-weight: 700;
font-size: 44px; font-size: 28px;
display: block; line-height: 40px;
text-align: center; letter-spacing: -2%;
margin-top: 15px; }
color: #464646;
.card_stat p {
font-weight: 500;
font-size: 14px;
line-height: 20px;
} }
.ant-modal-close { .ant-modal-close {
@ -774,6 +796,10 @@
justify-content: space-between; justify-content: space-between;
} }
.request-radio-group {
margin-left: 20px;
}
.call-requests { .call-requests {
background: rgba(246, 137, 0, 1); background: rgba(246, 137, 0, 1);
padding: 3px 5px; padding: 3px 5px;
@ -787,11 +813,118 @@
display: none; display: none;
} }
.ant-upload-list-text{ .ant-upload-list-text {
overflow: scroll; overflow: scroll;
height: 50px; height: 50px;
} }
.ant-modal-mask { .ant-modal-mask {
background-color: rgba(0, 0, 0, 0.5) !important; /* Используйте нужный вам цвет фона */ background-color: rgba(
0,
0,
0,
0.5
) !important; /* Используйте нужный вам цвет фона */
}
.ant-pagination {
position: fixed;
bottom: 90px;
right: 20px;
color: #f99e2c;
}
.ant-pagination-item-active {
background-color: #f99e2c;
color: white;
}
.ant-pagination .ant-pagination-item-active a {
color: #0f111c;
}
.ant-pagination .ant-pagination-item-active a:hover {
color: #f99e3c;
}
.ant-pagination .ant-pagination-item-active {
width: 32px;
height: 32px;
border-color: #d7d8e0;
border-radius: 8px;
}
.ant-pagination .ant-pagination-item-active:hover {
border-color: #f99e3c;
}
.ant-pagination-prev,
.ant-pagination-next {
width: 40px;
height: 32px;
background-color: #fff;
border: 1px solid #d7d8e0;
border-radius: 8px;
background-color: transparent;
}
.btn-refresh-dark img {
animation: none;
transition: transform 0.2s linear;
}
.btn-refresh-dark:active img {
animation: spin 1s linear infinite;
}
.btn-refresh-dark img.spin {
animation: spin 1s linear;
}
.btn-refresh-false img {
animation: none;
transition: transform 0.2s linear;
}
.btn-refresh-false:active img {
animation: spin 1s linear infinite;
}
.btn-refresh-false img.spin {
animation: spin 1s linear;
}
.ant-btn-default {
color: #0f111c;
}
@keyframes spin {
0% {
transform: rotate(0deg);
}
100% {
transform: rotate(1440deg);
}
}
@media (max-width: 768px) {
.btn-add {
padding: 7px 10px 7px 10px;
}
.btn-refresh-false {
padding: 7px 10px 7px 10px;
}
.statistics-header {
flex-direction: column;
justify-content: start;
align-items: start;
gap: 16px;
}
.requests-filter {
flex-direction: column;
justify-content: start;
align-items: start;
gap: 16px;
}
.request-radio-group {
margin-left: 0;
}
.profile-statistics {
flex-wrap: wrap;
gap: 10px;
}
} }

@ -1,6 +1,13 @@
import React, { useCallback, useEffect, useState } from "react"; import React, { useCallback, useEffect, useState } from "react";
import "./App.css"; import "./App.css";
import { Layout, Menu, ConfigProvider, Dropdown, notification } from "antd"; import {
Layout,
Menu,
ConfigProvider,
Dropdown,
notification,
Button,
} from "antd";
import { Routes, Route, Navigate, useLocation } from "react-router-dom"; import { Routes, Route, Navigate, useLocation } from "react-router-dom";
import { mainItems, superItems } from "./Utils/sidebar"; import { mainItems, superItems } from "./Utils/sidebar";
import Login from "./Auth/Login"; import Login from "./Auth/Login";
@ -9,6 +16,8 @@ import { LogoutApi } from "./API/auth/Logout";
import { Link } from "react-router-dom"; import { Link } from "react-router-dom";
import { MenuProps } from "antd"; import { MenuProps } from "antd";
// @ts-ignore // @ts-ignore
import TT_ELD from "./assets/tticon.svg";
import collapsedIcon from "./assets/collapsed.png";
import themeBtn from "./assets/theme-btn.svg"; import themeBtn from "./assets/theme-btn.svg";
// @ts-ignore // @ts-ignore
import avatar from "./assets/avatar-img.svg"; import avatar from "./assets/avatar-img.svg";
@ -44,6 +53,7 @@ import Requests from "./Components/Requests/Requests";
import { callController } from "./API/LayoutApi/callrequests"; import { callController } from "./API/LayoutApi/callrequests";
import Call from "./Components/CallRequests/Call"; import Call from "./Components/CallRequests/Call";
import { dark, light } from "./Utils/styles"; import { dark, light } from "./Utils/styles";
// import Input from "antd/es/input/Input";
const { Header, Sider, Content } = Layout; const { Header, Sider, Content } = Layout;
const userJSON: any = localStorage.getItem("user"); const userJSON: any = localStorage.getItem("user");
const userObject = JSON.parse(userJSON); const userObject = JSON.parse(userJSON);
@ -265,8 +275,9 @@ const App: React.FC = () => {
// taskSocket = new WebSocket( // taskSocket = new WebSocket(
// `ws://10.10.10.64:8080/global/?user_id=${admin_id}` // `ws://10.10.10.64:8080/global/?user_id=${admin_id}`
// ); // );
// taskSocket = new WebSocket(`wss://api.tteld.co/global/?user_id=${admin_id}`);
taskSocket = new WebSocket( taskSocket = new WebSocket(
`wss://api.tteld.co/global/?user_id=${admin_id}` `wss://ontime-socket.tteld.co/global/?user_id=${admin_id}`
); );
taskSocket.addEventListener("open", (event) => { taskSocket.addEventListener("open", (event) => {
@ -291,6 +302,28 @@ const App: React.FC = () => {
connect(); connect();
}, []); }, []);
useEffect(() => {
const handleResize = () => {
if (window.innerWidth <= 768) {
setCollapsed(true);
} else {
setCollapsed(false);
}
};
handleResize();
window.addEventListener("resize", handleResize);
return () => {
window.removeEventListener("resize", handleResize);
};
}, []);
const toggleCollapsed = () => {
setCollapsed(!collapsed);
};
// function checkConnection() { // function checkConnection() {
// if (!isLive) { // if (!isLive) {
// connect(); // connect();
@ -354,22 +387,41 @@ const App: React.FC = () => {
) : ( ) : (
<Sider <Sider
theme={"dark"} theme={"dark"}
collapsible
collapsed={collapsed} collapsed={collapsed}
onCollapse={(value) => setCollapsed(value)} onCollapse={toggleCollapsed}
style={{ style={{
height: "100vh", height: "100vh",
background: background:
theme === true ? "#202020" : "rgba(20, 22, 41, 1)", theme === true ? "#202020" : "rgba(20, 22, 41, 1)",
}} }}
> >
<p <div
onClick={rep} style={{
style={{ cursor: "pointer" }} display: "flex",
className={collapsed ? "logo-collapsed" : "logo"} alignItems: "center",
justifyContent: "space-around",
}}
> >
TT ELD <p
</p> onClick={rep}
style={{ cursor: "pointer" }}
className={collapsed ? "logo-collapsed" : "logo"}
>
{collapsed ? (
<img
className="tt-icon-collapsed"
src={TT_ELD}
alt="Icon"
/>
) : (
"TT ELD"
)}
</p>
<button onClick={toggleCollapsed} style={{ all: "unset" }}>
<img src={collapsedIcon} />
</button>
</div>
<Menu <Menu
theme={"dark"} theme={"dark"}
mode="inline" mode="inline"
@ -404,8 +456,7 @@ const App: React.FC = () => {
className="site-layout-background" className="site-layout-background"
style={{ style={{
padding: 0, padding: 0,
background: background: theme === true ? "#202020" : "#F5F5F8",
theme === true ? "#202020" : "rgba(215, 216, 224, 1)",
display: "flex", display: "flex",
justifyContent: "end", justifyContent: "end",
alignItems: "center", alignItems: "center",
@ -482,7 +533,7 @@ const App: React.FC = () => {
padding: 24, padding: 24,
minHeight: "92vh", minHeight: "92vh",
maxHeight: "calc(90vh - 10px)", maxHeight: "calc(90vh - 10px)",
overflowY: "scroll", overflowY: "auto",
background: theme === true ? "#202020" : "#fff", background: theme === true ? "#202020" : "#fff",
}} }}
> >

@ -108,10 +108,10 @@ const Login: React.FC = () => {
Log In Log In
</Button> </Button>
<h5> <h5>
<Link to='/auth/reset_password'>Forgot password?</Link> <Link to="/auth/reset_password">Forgot password?</Link>
<br /> <br />
Don't have an account? Don't have an account?
<Link to='/auth/register'> Create one now</Link> <Link to="/auth/register"> Create one now</Link>
</h5> </h5>
</Space> </Space>
</form> </form>

@ -1,8 +1,14 @@
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import CallTable from "./CallTable"; import CallTable from "./CallTable";
import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons";
import { useCallData } from "../../Hooks/CallRequests"; import { useCallData } from "../../Hooks/CallRequests";
import { Button, Input, Radio, RadioChangeEvent, Space } from "antd"; import {
Button,
Input,
Radio,
RadioChangeEvent,
Space,
Typography,
} from "antd";
import { TCall } from "../../types/CallRequests/TCall"; import { TCall } from "../../types/CallRequests/TCall";
import { TSocket } from "../../types/common/TSocket"; import { TSocket } from "../../types/common/TSocket";
@ -16,6 +22,8 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
page_size: 10, page_size: 10,
}); });
const theme = localStorage.getItem("theme") === "true" ? true : false;
useEffect(() => { useEffect(() => {
setTableData(data?.data); setTableData(data?.data);
}, [data]); }, [data]);
@ -74,7 +82,7 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
<div> <div>
<div> <div>
<div className="header d-flex"> <div className="header d-flex">
<p className="title">Call Requests</p> <Typography className="title">Call Requests</Typography>
</div> </div>
<div className="filter d-flex"> <div className="filter d-flex">
<Radio.Group <Radio.Group
@ -89,16 +97,14 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
</div> </div>
</div> </div>
<CallTable data={tableData} isLoading={isLoading} refetch={refetch} /> <CallTable data={tableData} isLoading={isLoading} refetch={refetch} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
{/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", justifyContent: "flex-end" }} wrap> <Space style={{ width: "100%", justifyContent: "flex-end" }} wrap>
<Button <Button onClick={Previos} disabled={data?.previous ? false : true}>
type="primary" <img src={leftPagination} />
icon={<StepBackwardOutlined />} </Button>
onClick={Previos}
disabled={data?.previous ? false : true}
></Button>
<Input <Input
style={{ width: 50, textAlign: "right" }} style={{ width: 30, textAlign: "center" }}
value={page} value={page}
onChange={(e) => { onChange={(e) => {
let num = e.target.value; let num = e.target.value;
@ -107,14 +113,11 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
} }
}} }}
/> />
<Button <Button onClick={Next} disabled={data?.next ? false : true}>
type="primary" <img src={rightPagination} />
icon={<StepForwardOutlined />} </Button>
onClick={Next}
disabled={data?.next ? false : true}
></Button>
</Space> </Space>
</Space> </Space> */}
</div> </div>
); );
}; };

@ -81,27 +81,27 @@ const CallTable = ({
{ {
title: "Company", title: "Company",
dataIndex: "company", dataIndex: "company",
width: "20%", // width: "20%",
}, },
{ {
title: "Driver", title: "Driver",
dataIndex: "driver", dataIndex: "driver",
width: "20%", // width: "20%",
}, },
{ {
title: "Note", title: "Note",
dataIndex: "note", dataIndex: "note",
width: "15%", // width: "15%",
}, },
{ {
title: "Requested at", title: "Requested at",
dataIndex: "time", dataIndex: "time",
width: "15%", // width: "15%",
}, },
{ {
title: "Actions", title: "Actions",
dataIndex: "action", dataIndex: "action",
width: "100px", // width: "100px",
render: (text, record) => { render: (text, record) => {
return ( return (
<div> <div>
@ -133,6 +133,9 @@ const CallTable = ({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
scroll={{ x: "768px" }}
pagination={{ pageSize: 10, size: "default" }}
bordered
/> />
</div> </div>
); );

@ -57,7 +57,7 @@ const AddCompany = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >
@ -80,7 +80,7 @@ const AddCompany = ({
}))} }))}
/> />
</FormAnt.Item> </FormAnt.Item>
<FormAnt.Item {/* <FormAnt.Item
label="Is Active" label="Is Active"
name="is_active" name="is_active"
rules={[ rules={[
@ -88,7 +88,7 @@ const AddCompany = ({
]} ]}
> >
<Switch defaultChecked={true} /> <Switch defaultChecked={true} />
</FormAnt.Item> </FormAnt.Item> */}
<FormAnt.Item <FormAnt.Item
label="USDOT" label="USDOT"
name="usdot" name="usdot"
@ -108,6 +108,13 @@ const AddCompany = ({
<Input /> <Input />
</FormAnt.Item> </FormAnt.Item>
</FormAnt> </FormAnt>
<FormAnt.Item
label="Is Active"
name="is_active"
rules={[{ required: false, message: "Please input company status!" }]}
>
<Switch defaultChecked={true} />
</FormAnt.Item>
</Modal> </Modal>
</div> </div>
); );

@ -3,14 +3,13 @@ import AddCompany from "./AddCompanies";
import CompanyTable from "./CompaniesTable"; import CompanyTable from "./CompaniesTable";
import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons"; import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons";
import { useCompanyPaginated } from "../../Hooks/Companies"; import { useCompanyPaginated } from "../../Hooks/Companies";
import { Button, Input, Space } from "antd"; import { Button, Input, Space, Typography } from "antd";
// @ts-ignore // @ts-ignore
import IconSearch from "../../assets/searchIcon.png"; import IconSearch from "../../assets/searchIcon.png";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
import { role } from "../../App";
const theme = localStorage.getItem("theme") === "true" ? true : false; import { role } from "../../App";
const Company = () => { const Company = () => {
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
@ -51,11 +50,13 @@ const Company = () => {
} }
}; };
const theme = localStorage.getItem("theme") === "true" ? true : false;
return ( return (
<div> <div>
{open && <AddCompany open={open} refetch={refetch} setOpen={setOpen} />} {open && <AddCompany open={open} refetch={refetch} setOpen={setOpen} />}
<div className="header d-flex"> <div className="header d-flex">
<h1 className="title">Companies</h1> <Typography className="title">Companies</Typography>
{role !== "Checker" && ( {role !== "Checker" && (
<button <button
style={{ marginRight: 0 }} style={{ marginRight: 0 }}
@ -80,16 +81,13 @@ const Company = () => {
</div> </div>
<CompanyTable data={data?.data} isLoading={isLoading} /> <CompanyTable data={data?.data} isLoading={isLoading} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical"> {/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", justifyContent: "flex-end" }} wrap> <Space style={{ width: "100%", justifyContent: "flex-end" }} wrap>
<Button <Button onClick={Previos} disabled={data?.previous ? false : true}>
type="primary" <img src={leftPagination} />
icon={<StepBackwardOutlined />} </Button>
onClick={Previos}
disabled={data?.previous ? false : true}
></Button>
<Input <Input
style={{ width: 50, textAlign: "right" }} style={{ width: 30, textAlign: "center" }}
value={page} value={page}
onChange={(e) => { onChange={(e) => {
let num = e.target.value; let num = e.target.value;
@ -98,14 +96,11 @@ const Company = () => {
} }
}} }}
/> />
<Button <Button onClick={Next} disabled={data?.next ? false : true}>
type="primary" <img src={rightPagination} />
icon={<StepForwardOutlined />} </Button>
onClick={Next}
disabled={data?.next ? false : true}
></Button>
</Space> </Space>
</Space> </Space> */}
</div> </div>
); );
}; };

@ -137,7 +137,13 @@ function CompanyTable({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
size="middle" size="small"
scroll={{ x: "768px" }}
pagination={{
pageSize: 10,
size: "default",
}}
bordered
/> />
</div> </div>
); );

@ -6,8 +6,11 @@ import { useCustomerData } from "../../Hooks/Customers";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
// @ts-ignore // @ts-ignore
import leftPagination from "../../assets/pagination-left.png";
import rightPagination from "../../assets/right-pagination.png";
import IconSearch from "../../assets/searchIcon.png"; import IconSearch from "../../assets/searchIcon.png";
import { Button, Input, Space } from "antd";
import { Button, Input, Space, Typography } from "antd";
const Customer = () => { const Customer = () => {
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
@ -48,11 +51,12 @@ const Customer = () => {
}, 1000); }, 1000);
}; };
const theme = localStorage.getItem("theme") === "true" ? true : false; const theme = localStorage.getItem("theme") === "true" ? true : false;
return ( return (
<div> <div>
{open && <AddCustomer open={open} setOpen={setOpen} />} {open && <AddCustomer open={open} setOpen={setOpen} />}
<div className="header d-flex"> <div className="header d-flex">
<p className="title">Drivers</p> <Typography className="title">Drivers</Typography>
<button className="btn-add d-flex" onClick={showModal}> <button className="btn-add d-flex" onClick={showModal}>
<img src={addicon} style={{ marginRight: 8 }} alt="" /> <img src={addicon} style={{ marginRight: 8 }} alt="" />
Add Driver Add Driver
@ -70,16 +74,13 @@ const Customer = () => {
</div> </div>
</div> </div>
<CustomerTable data={data?.data} isLoading={isLoading} /> <CustomerTable data={data?.data} isLoading={isLoading} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical"> {/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", justifyContent: "flex-end" }} wrap> <Space style={{ width: "100%", justifyContent: "flex-end" }} wrap>
<Button <Button onClick={Previos} disabled={!data?.previous}>
type="primary" <img src={leftPagination} />
icon={<StepBackwardOutlined />} </Button>
onClick={Previos}
disabled={!data?.previous}
></Button>
<Input <Input
style={{ width: 50, textAlign: "right" }} style={{ width: 30, textAlign: "center" }}
value={page} value={page}
onChange={(e) => { onChange={(e) => {
let num = e.target.value; let num = e.target.value;
@ -88,14 +89,11 @@ const Customer = () => {
} }
}} }}
/> />
<Button <Button onClick={Next} disabled={!data?.next}>
type="primary" <img src={rightPagination} />
icon={<StepForwardOutlined />} </Button>
onClick={Next}
disabled={!data?.next}
></Button>
</Space> </Space>
</Space> </Space> */}
</div> </div>
); );
}; };

@ -93,7 +93,13 @@ function CustomerTable({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
pagination={false} size="middle"
bordered
pagination={{
pageSize: 10,
size: "default",
}}
scroll={{ x: "768px" }}
/> />
</div> </div>
); );

@ -12,10 +12,10 @@ const ChangePassword = () => {
}; };
return ( return (
<div style={{ width: 500 }}> <div style={{ width: 350 }}>
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >

@ -1,4 +1,16 @@
import { useState } from "react"; import { useState } from "react";
import {
LineChart,
Line,
XAxis,
YAxis,
CartesianGrid,
Tooltip,
Legend,
ResponsiveContainer,
} from "recharts";
import { TProfilePutParams, prof } from "../../API/LayoutApi/profile"; import { TProfilePutParams, prof } from "../../API/LayoutApi/profile";
import { import {
Button, Button,
@ -32,7 +44,7 @@ const Profile = () => {
const [range, setRange] = useState<any>(1); const [range, setRange] = useState<any>(1);
const onSubmit = async (value: TProfilePutParams) => { const onSubmit = async (value: TProfilePutParams) => {
await prof.profPatch(value) await prof.profPatch(value);
refetch(); refetch();
}; };
@ -62,6 +74,20 @@ const Profile = () => {
start_date: startDate, start_date: startDate,
end_date: endDate, end_date: endDate,
}); });
const formatDate = (dateString: string): string => {
const date = new Date(dateString);
return new Intl.DateTimeFormat("en-EN", {
day: "2-digit",
month: "short",
}).format(date);
};
const chartData = lineData?.daily_stats.map((stat: any) => ({
date: formatDate(stat.date),
tasks: stat.number_of_tasks,
}));
return ( return (
<div> <div>
<Spin size="large" spinning={!data}> <Spin size="large" spinning={!data}>
@ -84,40 +110,40 @@ const Profile = () => {
onFinish={onSubmit} onFinish={onSubmit}
> >
<Row gutter={[16, 10]}> <Row gutter={[16, 10]}>
<Col span={6}> <Col xs={24} sm={12} md={8} lg={6}>
<Form.Item <Form.Item
wrapperCol={{ span: "100%" }} wrapperCol={{ span: "100%" }}
label="First name" label="First name"
name="first_name" name="first_name"
> >
<Input /> <Input placeholder="Enter first name" />
</Form.Item> </Form.Item>
</Col> </Col>
<Col span={6}> <Col xs={24} sm={12} md={8} lg={6}>
<Form.Item <Form.Item
wrapperCol={{ span: "100%" }} wrapperCol={{ span: "100%" }}
label="Last name" label="Last name"
name="last_name" name="last_name"
> >
<Input /> <Input placeholder="Enter last name" />
</Form.Item> </Form.Item>
</Col> </Col>
<Col span={6}> <Col xs={24} sm={12} md={8} lg={6}>
<Form.Item <Form.Item
wrapperCol={{ span: "100%" }} wrapperCol={{ span: "100%" }}
label="Username" label="Username"
name="username" name="username"
> >
<Input /> <Input placeholder="Enter username" />
</Form.Item> </Form.Item>
</Col> </Col>
<Col span={6}> <Col xs={24} sm={12} md={8} lg={6}>
<Form.Item <Form.Item
wrapperCol={{ span: "100%" }} wrapperCol={{ span: "100%" }}
label="E-mail" label="E-mail"
name="email" name="email"
> >
<Input /> <Input placeholder="Enter email" />
</Form.Item> </Form.Item>
</Col> </Col>
</Row> </Row>
@ -138,7 +164,7 @@ const Profile = () => {
> >
<Row gutter={[16, 10]}> <Row gutter={[16, 10]}>
{data && data.team !== "" && ( {data && data.team !== "" && (
<Col span={6}> <Col>
<Form.Item <Form.Item
wrapperCol={{ span: "100%" }} wrapperCol={{ span: "100%" }}
label="Team" label="Team"
@ -161,40 +187,111 @@ const Profile = () => {
> >
<RangePicker onCalendarChange={datePick} /> <RangePicker onCalendarChange={datePick} />
</div> </div>
<div <div
style={{ style={{
display: "flex", display: "flex",
width: "100%", alignItems: "start",
height: "70vh", justifyContent: "space-between",
gap: 15,
marginTop: 35,
}} }}
> >
<div <div
style={{ style={{
marginBottom: 50, width: 156,
marginTop: 20, height: 330,
marginLeft: 30,
width: "80%",
display: "flex", display: "flex",
justifyContent: "space-between", flexDirection: "column",
gap: 15,
}} }}
> >
<p className="card_stat"> <div
Average:{" "} className="card_stat"
style={{ backgroundColor: "#F99E2C" }}
>
<p>Total</p>
<span>{lineData?.total_for_period} </span>
<p>
{role === "Owner" || role === "Tech Support"
? "Tasks"
: "Points"}
</p>
</div>
<div
className="card_stat"
style={{ backgroundColor: "#409CFF" }}
>
<p>Average</p>
<span>{lineData?.avg_stats_for_period} </span> <span>{lineData?.avg_stats_for_period} </span>
{role === "Owner" ? "tasks" : "pts"}/day <p>
</p> {role === "Owner" || role === "Tech Support"
<p className="card_stat"> ? "Tasks a day"
Total: <span>{lineData?.total_for_period} </span> : "Points a day"}{" "}
{role === "Owner" ? "tasks" : "pts"} </p>
</p> </div>
<p className="card_stat"> </div>
Contribution: <span>{lineData?.contribution}</span>%
</p> <ResponsiveContainer
width="100%"
height={370}
style={{ textTransform: "capitalize" }}
>
<LineChart data={chartData}>
<CartesianGrid vertical={false} stroke="#D7D8E080" />
<XAxis
dataKey="date"
style={{
color: "#9B9DAA",
fontSize: 10,
lineHeight: "12.4px",
fontWeight: 400,
}}
/>
<YAxis
style={{
color: "#9B9DAA",
fontSize: 10,
fontWeight: 400,
}}
/>
<Tooltip />
<Legend />
<Line
dataKey="tasks"
stroke="#F99E2C"
activeDot={{ r: 7 }}
/>
</LineChart>
</ResponsiveContainer>
<div
style={{
width: 156,
height: 330,
display: "flex",
flexDirection: "column",
gap: 15,
}}
>
<div
className="card_stat"
style={{ backgroundColor: "#9B51E0" }}
>
<p>Contribution</p>
<span>{lineData?.contribution}%</span>
<p>
{" "}
{role === "Owner" || role === "Tech Support"
? "to Business"
: `to ${data?.team}`}{" "}
</p>
</div>
</div> </div>
</div> </div>
</div> </div>
</Space> </Space>
</TabPane> </TabPane>
<TabPane tab={<span>History</span>} key="2"> <TabPane tab={<span>History</span>} key="2">
<Select <Select
style={{ width: "20%", marginBottom: 10 }} style={{ width: "20%", marginBottom: 10 }}
@ -255,6 +352,7 @@ const Profile = () => {
key: "timestamp", key: "timestamp",
}, },
]} ]}
scroll={{ x: "768px" }}
/> />
</TabPane> </TabPane>
<TabPane tab={<span>Change Password</span>} key="3"> <TabPane tab={<span>Change Password</span>} key="3">

@ -1,7 +1,14 @@
import { useEffect, useRef, useState } from "react"; import { useEffect, useRef, useState } from "react";
import { useRequestsData } from "../../Hooks/Requests"; import { useRequestsData } from "../../Hooks/Requests";
import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons"; import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons";
import { Button, Input, Radio, RadioChangeEvent, Space } from "antd"; import {
Button,
Input,
Radio,
RadioChangeEvent,
Space,
Typography,
} from "antd";
import { TRequests } from "../../types/Requests/TRequests"; import { TRequests } from "../../types/Requests/TRequests";
import { TSocket } from "../../types/common/TSocket"; import { TSocket } from "../../types/common/TSocket";
import RequestsEdit from "./RequestsEdit"; import RequestsEdit from "./RequestsEdit";
@ -79,9 +86,9 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
/> />
)} )}
<div className="header d-flex"> <div className="header d-flex">
<p className="title">Requests</p> <Typography className="title">Requests</Typography>
</div> </div>
<div className="filter d-flex"> <div className="filter d-flex requests-filter ">
<div className="search-div"> <div className="search-div">
<img src={IconSearch} alt="" /> <img src={IconSearch} alt="" />
<input <input
@ -95,7 +102,7 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
onChange={(e: RadioChangeEvent) => setStatus(e.target.value)} onChange={(e: RadioChangeEvent) => setStatus(e.target.value)}
size="middle" size="middle"
value={status} value={status}
style={{ marginLeft: 20 }} className="request-radio-group"
> >
<Radio.Button value={"Pending"}>Pending</Radio.Button> <Radio.Button value={"Pending"}>Pending</Radio.Button>
<Radio.Button value={"Assigned"}>Assigned</Radio.Button> <Radio.Button value={"Assigned"}>Assigned</Radio.Button>
@ -109,16 +116,14 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
setOpenModal={setModalOpen} setOpenModal={setModalOpen}
setRequestData={setRequestData} setRequestData={setRequestData}
/> />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
{/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", justifyContent: "flex-end" }} wrap> <Space style={{ width: "100%", justifyContent: "flex-end" }} wrap>
<Button <Button onClick={Previos} disabled={data?.previous ? false : true}>
type="primary" <img src={leftPagination} />
icon={<StepBackwardOutlined />} </Button>
onClick={Previos}
disabled={data?.previous ? false : true}
></Button>
<Input <Input
style={{ width: 50, textAlign: "right" }} style={{ width: 30, textAlign: "center" }}
value={page} value={page}
onChange={(e) => { onChange={(e) => {
let num = e.target.value; let num = e.target.value;
@ -127,14 +132,11 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
} }
}} }}
/> />
<Button <Button onClick={Next} disabled={data?.next ? false : true}>
type="primary" <img src={rightPagination} />
icon={<StepForwardOutlined />} </Button>
onClick={Next}
disabled={data?.next ? false : true}
></Button>
</Space> </Space>
</Space> </Space> */}
</div> </div>
); );
}; };

@ -160,6 +160,9 @@ const RequestsTable = ({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
scroll={{ x: "768px" }}
pagination={{ pageSize: 10, size: "default" }}
bordered
/> />
</div> </div>
); );

@ -40,16 +40,14 @@ const AddService = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >
<FormAnt.Item <FormAnt.Item
label="Title" label="Title"
name="title" name="title"
rules={[ rules={[{ required: true, message: "Please input service title!" }]}
{ required: true, message: "Please input service title!" },
]}
> >
<Input /> <Input />
</FormAnt.Item> </FormAnt.Item>

@ -72,6 +72,11 @@ const ServiceTable = ({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
pagination={{
pageSize: 10,
size: "default",
}}
bordered
/> />
</div> </div>
); );

@ -5,6 +5,7 @@ import ServiceTable from "./ServiceTable";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
import { role } from "../../App"; import { role } from "../../App";
import { Typography } from "antd";
const Service = () => { const Service = () => {
const { data, isLoading, refetch } = useServiceData(); const { data, isLoading, refetch } = useServiceData();
@ -12,11 +13,12 @@ const Service = () => {
const showModal = () => { const showModal = () => {
setOpen(true); setOpen(true);
}; };
return ( return (
<div> <div>
{open && <AddService refetch={refetch} open={open} setOpen={setOpen} />} {open && <AddService refetch={refetch} open={open} setOpen={setOpen} />}
<div className="header d-flex"> <div className="header d-flex" style={{ marginBottom: "10px" }}>
<p className="title">Services</p> <Typography className="title">Services</Typography>
{role !== "Checker" && ( {role !== "Checker" && (
<button onClick={showModal} className="btn-add d-flex"> <button onClick={showModal} className="btn-add d-flex">
<img src={addicon} style={{ marginRight: 8 }} alt="" /> <img src={addicon} style={{ marginRight: 8 }} alt="" />

@ -11,7 +11,14 @@ import { TStatTeam } from "../../types/Statistic/TStat";
import StatTable from "./StatisticTable"; import StatTable from "./StatisticTable";
import StatTeamTable from "./StatisticTeamTable"; import StatTeamTable from "./StatisticTeamTable";
import dayjs from "dayjs"; import dayjs from "dayjs";
import { Button, DatePicker, DatePickerProps, Select, Tabs } from "antd"; import {
Button,
DatePicker,
DatePickerProps,
Select,
Tabs,
Typography,
} from "antd";
import TabPane from "antd/es/tabs/TabPane"; import TabPane from "antd/es/tabs/TabPane";
// @ts-ignore // @ts-ignore
import IconSearch from "../../assets/searchIcon.png"; import IconSearch from "../../assets/searchIcon.png";
@ -113,15 +120,18 @@ const Stat = () => {
const theme = localStorage.getItem("theme") === "true" ? true : false; const theme = localStorage.getItem("theme") === "true" ? true : false;
return ( return (
<div> <div>
<div className="header d-flex" style={{ marginBottom: 16 }}> <div
<p className="title">Statistics</p> className="header d-flex statistics-header "
<div className=""> style={{ marginBottom: 16 }}
>
<Typography className="title">Statistics</Typography>
<div>
<DatePicker <DatePicker
onChange={onChangeDate} onChange={onChangeDate}
picker="month" picker="month"
format={"MMMM"} format={"MMMM"}
defaultValue={now} defaultValue={now}
style={{ marginRight: 10, width: 120 }} style={{ marginRight: 10, width: 120, marginBottom: 10 }}
/> />
<RangePicker style={{ width: 260 }} onCalendarChange={datePick} /> <RangePicker style={{ width: 260 }} onCalendarChange={datePick} />
</div> </div>
@ -156,7 +166,11 @@ const Stat = () => {
isLoading={isLoading} isLoading={isLoading}
refetch={refetch} refetch={refetch}
/> />
<Button type="primary" onClick={(e) => handleSave("team")}> <Button
type="primary"
onClick={(e) => handleSave("team")}
style={{ marginTop: 10 }}
>
Save as file Save as file
</Button> </Button>
</TabPane> </TabPane>

@ -51,11 +51,13 @@ const StatTable = ({
}, },
]} ]}
pagination={{ pagination={{
pageSize: 14, pageSize: 10,
size: "default",
}} }}
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
bordered
/> />
</div> </div>
); );

@ -28,6 +28,9 @@ import ontime from "../../assets/ontimeicon.svg";
import tt from "../../assets/tticon.svg"; import tt from "../../assets/tticon.svg";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
//ts-ignore
import fileUpload from "../../assets/upload-file.png";
import AddCustomer from "../Customers/AddCustomer"; import AddCustomer from "../Customers/AddCustomer";
import AddDriver from "../Companies/AddDriver"; import AddDriver from "../Companies/AddDriver";
import TextArea from "antd/es/input/TextArea"; import TextArea from "antd/es/input/TextArea";
@ -180,7 +183,7 @@ const AddTask = ({
<AddDriver id={companyId} open={driverOpen} setOpen={setDriverOpen} /> <AddDriver id={companyId} open={driverOpen} setOpen={setDriverOpen} />
<Modal <Modal
open={open} open={open}
width={600} width={720}
title="Add task" title="Add task"
okText="Create" okText="Create"
cancelText="Cancel" cancelText="Cancel"
@ -200,11 +203,12 @@ const AddTask = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout={isMobile ? "vertical" : "horizontal"} // layout={isMobile ? "vertical" : "horizontal"}
layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >
<Row gutter={[16, 16]}> <Row gutter={[16, 0]}>
<Col span={24}> <Col span={24}>
<FormAnt.Item <FormAnt.Item
label="Company" label="Company"
@ -244,13 +248,13 @@ const AddTask = ({
style={{ style={{
display: "flex", display: "flex",
justifyContent: "space-around", justifyContent: "space-around",
alignItems: isMobile ? "center" : "none", alignItems: "center",
}} }}
> >
<FormAnt.Item <FormAnt.Item
label="Driver" label="Driver"
name="customer_id" name="customer_id"
style={{ width: "85%" }} style={{ width: "90%" }}
rules={[ rules={[
{ required: true, message: "Please input service points!" }, { required: true, message: "Please input service points!" },
]} ]}
@ -276,7 +280,7 @@ const AddTask = ({
display: "flex", display: "flex",
alignItems: "center", alignItems: "center",
justifyContent: "space-between", justifyContent: "space-between",
marginTop: isMobile ? 5 : 0, marginTop: 5,
}} }}
disabled={!companyId} disabled={!companyId}
> >
@ -286,17 +290,7 @@ const AddTask = ({
</div> </div>
</Col> </Col>
<Col span={isMobile ? 12 : 24}> <Col span={12}>
<FormAnt.Item
label="Service"
name="service_id"
rules={[{ required: true, message: "Please select service!" }]}
>
<Select options={serviceOptions?.sort(sortByLabel)} />
</FormAnt.Item>
</Col>
<Col span={isMobile ? 12 : 24}>
<FormAnt.Item <FormAnt.Item
label="Assigned to" label="Assigned to"
name="assigned_to_id" name="assigned_to_id"
@ -320,12 +314,25 @@ const AddTask = ({
</FormAnt.Item> </FormAnt.Item>
</Col> </Col>
<Col span={12}>
<FormAnt.Item
label="Service"
name="service_id"
rules={[{ required: true, message: "Please select service!" }]}
>
<Select options={serviceOptions?.sort(sortByLabel)} />
</FormAnt.Item>
</Col>
<Col span={isMobile ? 12 : 12}> <Col span={isMobile ? 12 : 12}>
<FormAnt.Item <FormAnt.Item
label="Status" label="Status"
name="status" name="status"
rules={[ rules={[
{ required: false, message: "Please input service points!" }, {
required: false,
message: "Please input service points!",
},
]} ]}
> >
<Select defaultValue="New"> <Select defaultValue="New">
@ -400,7 +407,7 @@ const AddTask = ({
<div> <div>
<Upload.Dragger <Upload.Dragger
name="file" name="file"
height={isMobile ? 100 : 150} height={174}
multiple={true} multiple={true}
customRequest={({ file, onSuccess }: any) => { customRequest={({ file, onSuccess }: any) => {
const formData = new FormData(); const formData = new FormData();
@ -419,9 +426,7 @@ const AddTask = ({
> >
{!isMobile ? ( {!isMobile ? (
<p className={`ant-upload-drag-icon`}> <p className={`ant-upload-drag-icon`}>
<UploadOutlined <img src={fileUpload} />
style={{ color: "rgba(249, 158, 44, 1)" }}
/>
</p> </p>
) : ( ) : (
<UploadOutlined /> <UploadOutlined />
@ -429,10 +434,17 @@ const AddTask = ({
{!isMobile && ( {!isMobile && (
<p <p
className="ant-upload-text" className="ant-upload-text"
style={{ color: "rgba(249, 158, 44, 1)" }} style={{ color: "#9b9daa", fontSize: 14 }}
> >
Click or drag a file here to upload (only .jpeg .jpg <strong> Drag and drop files or </strong>
.png .pdf) <span style={{ color: "#f99e2c" }}>
Click to select
</span>
<br />
<span style={{ fontSize: 13 }}>
Maximum file size is 10 MB <br />
(only .jpeg .jpg .png .pdf)
</span>
</p> </p>
)} )}
</Upload.Dragger> </Upload.Dragger>

@ -548,6 +548,7 @@ const TaskModal = ({
style={{ margin: "0 24px" }} style={{ margin: "0 24px" }}
loading={isLoading} loading={isLoading}
size="small" size="small"
pagination={false}
dataSource={data?.map((u, i) => ({ dataSource={data?.map((u, i) => ({
no: i + 1, no: i + 1,
...u, ...u,

@ -136,28 +136,28 @@ const TaskTable = ({
const columns = useMemo(() => { const columns = useMemo(() => {
const columns = [ const columns = [
{ // {
title: "", // title: "",
dataIndex: "no", // dataIndex: "no",
width: isMobile ? "1%" : "5%", // width: isMobile ? "1%" : "3.5%",
fixed: isMobile ? "left" : false, // fixed: isMobile ? "left" : false,
key: "1", // key: "1",
render: (text: any, record: TTask) => ( // render: (text: any, record: TTask) => (
<div // <div
style={{ // style={{
display: "flex", // display: "flex",
alignItems: "center", // alignItems: "center",
justifyContent: "space-around", // justifyContent: "space-around",
}} // }}
> // >
{record?.via_telegram && ( // {record?.via_telegram && (
<Tooltip placement="topLeft" title={"Created via Telegram"}> // <Tooltip placement="topLeft" title={"Created via Telegram"}>
<img src={tgIcon} alt="" style={{ width: 20, height: 20 }} /> // <img src={tgIcon} alt="" style={{ width: 20, height: 20 }} />
</Tooltip> // </Tooltip>
)} // )}
</div> // </div>
), // ),
}, // },
{ {
title: ( title: (
<div <div
@ -171,7 +171,7 @@ const TaskTable = ({
</div> </div>
), ),
dataIndex: "no", dataIndex: "no",
width: isMobile ? "1%" : "5%", width: isMobile ? "1%" : "3.5%",
fixed: isMobile ? "left" : false, fixed: isMobile ? "left" : false,
key: "2", key: "2",
render: (text?: any, record?: TTask) => ( render: (text?: any, record?: TTask) => (
@ -263,7 +263,7 @@ const TaskTable = ({
{ {
title: "Team", title: "Team",
dataIndex: "assigned_to", dataIndex: "assigned_to",
width: isMobile ? "5%" : "8%", width: isMobile ? "3%" : "7%",
key: "7", key: "7",
ellipsis: { ellipsis: {
showTitle: false, showTitle: false,
@ -277,7 +277,7 @@ const TaskTable = ({
{ {
title: "Assignee", title: "Assignee",
dataIndex: "in_charge", dataIndex: "in_charge",
width: isMobile ? "5%" : "12%", width: isMobile ? "4%" : "9%",
key: "8", key: "8",
ellipsis: { ellipsis: {
showTitle: false, showTitle: false,
@ -291,7 +291,7 @@ const TaskTable = ({
{ {
title: "PTI", title: "PTI",
dataIndex: "pti", dataIndex: "pti",
width: "6%", width: "8%",
key: "8", key: "8",
responsive: ["lg"], responsive: ["lg"],
render: (pti: boolean, record: TTask) => render: (pti: boolean, record: TTask) =>
@ -308,7 +308,7 @@ const TaskTable = ({
{ {
title: "Note", title: "Note",
dataIndex: "note", dataIndex: "note",
width: "12%", width: "10%",
key: "9", key: "9",
responsive: ["lg"], responsive: ["lg"],
ellipsis: { ellipsis: {
@ -338,7 +338,7 @@ const TaskTable = ({
{ {
title: "Actions", title: "Actions",
dataIndex: "action", dataIndex: "action",
width: isMobile ? "3%" : "8%", width: isMobile ? "3%" : "10%",
key: "11", key: "11",
fixed: isMobile ? "right" : false, fixed: isMobile ? "right" : false,
render: (text: string, record: TTask) => { render: (text: string, record: TTask) => {
@ -414,15 +414,16 @@ const TaskTable = ({
), ),
key: u?.id, key: u?.id,
}))} }))}
size="small"
columns={columns as any} columns={columns as any}
pagination={false} size="small"
loading={isLoading} loading={isLoading}
rowClassName={rowClassName} rowClassName={rowClassName}
scroll={ scroll={{ x: "768px" }}
isMobile ? { x: "calc(800px + 40%)" } : { x: "calc(0px + 100%)" }
}
bordered bordered
pagination={{
pageSize: 10,
size: "default",
}}
/> />
</div> </div>
); );

@ -1,9 +1,8 @@
import { useEffect, useRef, useState } from "react"; import { useEffect, useRef, useState } from "react";
import AddTask from "./AddTask"; import AddTask from "./AddTask";
import { Button, Input, Select, Space } from "antd"; import { Button, Input, Select, Space, Typography } from "antd";
import TaskTable from "./TaskTable"; import TaskTable from "./TaskTable";
import { useTeamData } from "../../Hooks/Teams"; import { useTeamData } from "../../Hooks/Teams";
import { StepForwardOutlined, StepBackwardOutlined } from "@ant-design/icons";
import { useTasks } from "../../Hooks/Tasks"; import { useTasks } from "../../Hooks/Tasks";
import { TTask } from "../../types/Tasks/TTasks"; import { TTask } from "../../types/Tasks/TTasks";
import { isMobile, role, team_id } from "../../App"; import { isMobile, role, team_id } from "../../App";
@ -211,7 +210,11 @@ const Task = ({
)} )}
<div className="header d-flex"> <div className="header d-flex">
<div className="header-title d-flex"> <div className="header-title d-flex">
<p className="title">Tasks</p> {/* <p className="title " style={{ color: theme ? "#fff" : "#000" }}>
Tasks
</p> */}
<Typography className="title">Tasks</Typography>
</div> </div>
<div className="d-flex"> <div className="d-flex">
{role !== "Checker" && ( {role !== "Checker" && (
@ -225,7 +228,7 @@ const Task = ({
</button> </button>
)} )}
<button <button
className={`btn-refresh-${theme && "dark"} d-flex`} className={`btn-refresh-${false && "dark"} d-flex`}
onClick={() => { onClick={() => {
refetch(); refetch();
if (!isLive) { if (!isLive) {
@ -259,7 +262,7 @@ const Task = ({
marginTop: isMobile ? 10 : 0, marginTop: isMobile ? 10 : 0,
marginBottom: isMobile ? 10 : 0, marginBottom: isMobile ? 10 : 0,
}} }}
placeholder="status" placeholder="Status"
onChange={(value: any) => setStatus(value)} onChange={(value: any) => setStatus(value)}
mode="multiple" mode="multiple"
> >
@ -271,7 +274,7 @@ const Task = ({
<Select <Select
mode="multiple" mode="multiple"
style={{ width: 260, marginLeft: 12 }} style={{ width: 260, marginLeft: 12 }}
placeholder="team" placeholder="Team"
onChange={(value: any) => setTeam(value)} onChange={(value: any) => setTeam(value)}
options={teamOptions} options={teamOptions}
/> />
@ -284,16 +287,14 @@ const Task = ({
showErrorModal={showErrorModal} showErrorModal={showErrorModal}
setErrorModal={setErrorModal} setErrorModal={setErrorModal}
/> />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical"> {/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", justifyContent: "flex-end" }} wrap> <Space style={{ width: "100%", justifyContent: "flex-end" }} wrap>
<Button <Button onClick={Previos} disabled={data?.previous ? false : true}>
type="primary" <img src={leftPagination} />
icon={<StepBackwardOutlined />} </Button>
onClick={Previos}
disabled={data?.previous ? false : true}
></Button>
<Input <Input
style={{ width: 50, textAlign: "right" }} style={{ width: 30, textAlign: "center" }}
value={page} value={page}
onChange={(e) => { onChange={(e) => {
let num = e.target.value; let num = e.target.value;
@ -302,14 +303,12 @@ const Task = ({
} }
}} }}
/> />
<Button
type="primary" <Button onClick={Next} disabled={data?.next ? false : true}>
icon={<StepForwardOutlined />} <img src={rightPagination} />
onClick={Next} </Button>
disabled={data?.next ? false : true}
></Button>
</Space> </Space>
</Space> </Space> */}
</div> </div>
); );
}; };

@ -41,7 +41,7 @@ const AddTeam = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >

@ -56,6 +56,8 @@ const TeamTable = ({
dataIndex: "created", dataIndex: "created",
}, },
]} ]}
pagination={{ pageSize: 10, size: "default" }}
bordered
/> />
); );
}; };

@ -1,9 +1,10 @@
import { useState } from "react"; import { useEffect, useState } from "react";
import { useTeamData } from "../../Hooks/Teams"; import { useTeamData } from "../../Hooks/Teams";
import TeamTable from "./TeamTable"; import TeamTable from "./TeamTable";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
import AddTeam from "./AddTeam"; import AddTeam from "./AddTeam";
import { Typography } from "antd";
const Team = () => { const Team = () => {
const { data, isLoading, refetch } = useTeamData({}); const { data, isLoading, refetch } = useTeamData({});
@ -11,11 +12,14 @@ const Team = () => {
const showModal = () => { const showModal = () => {
setOpen(true); setOpen(true);
}; };
const theme = localStorage.getItem("theme") === "true" ? true : false;
return ( return (
<div> <div>
{open && <AddTeam refetch={refetch} open={open} setOpen={setOpen} />} {open && <AddTeam refetch={refetch} open={open} setOpen={setOpen} />}
<div className="header d-flex" style={{ marginBottom: 16 }}> <div className="header d-flex" style={{ marginBottom: 16 }}>
<p className="title">Teams</p> <Typography className="title">Teams</Typography>
<button <button
className="btn-add d-flex" className="btn-add d-flex"
style={{ marginRight: 0 }} style={{ marginRight: 0 }}

@ -1,7 +1,7 @@
import { Input, Modal, Form as FormAnt, Select, Upload } from "antd"; import { Input, Modal, Form as FormAnt, Select, Upload } from "antd";
import { updateController } from "../../API/LayoutApi/update"; import { updateController } from "../../API/LayoutApi/update";
import { useState } from "react"; import { useState } from "react";
import { UploadOutlined } from "@ant-design/icons"; import fileUpload from "../../assets/upload-file.png";
import { useCompanyData } from "../../Hooks/Companies"; import { useCompanyData } from "../../Hooks/Companies";
import { useCustomerByComanyData } from "../../Hooks/Customers"; import { useCustomerByComanyData } from "../../Hooks/Customers";
import { import {
@ -10,6 +10,7 @@ import {
RefetchQueryFilters, RefetchQueryFilters,
} from "react-query"; } from "react-query";
import { TUpdate } from "../../types/Update/TUpdate"; import { TUpdate } from "../../types/Update/TUpdate";
import TextArea from "antd/es/input/TextArea";
const { Option } = Select; const { Option } = Select;
const AddUpdate = ({ const AddUpdate = ({
refetch, refetch,
@ -86,7 +87,7 @@ const AddUpdate = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >
@ -110,52 +111,70 @@ const AddUpdate = ({
onChange={(value: any) => setCompanyId(value)} onChange={(value: any) => setCompanyId(value)}
/> />
</FormAnt.Item> </FormAnt.Item>
<FormAnt.Item
label="Driver" <div
name="customer_id" style={{
rules={[ display: "flex",
{ required: false, message: "Please input service points!" }, gap: 5,
]} width: "100%",
justifyContent: "space-between",
}}
> >
<Select <FormAnt.Item
showSearch style={{ width: "50%" }}
placeholder="Search Driver" label="Driver"
onSearch={(value: any) => setCustomerName(value)} name="customer_id"
options={customerData?.data?.data?.map((item) => ({ rules={[
label: item?.name, { required: false, message: "Please input service points!" },
value: item?.id, ]}
}))} >
value={customerName} <Select
filterOption={false} showSearch
autoClearSearchValue={false} placeholder="Search Driver"
allowClear onSearch={(value: any) => setCustomerName(value)}
/> options={customerData?.data?.data?.map((item) => ({
</FormAnt.Item> label: item?.name,
value: item?.id,
}))}
value={customerName}
filterOption={false}
autoClearSearchValue={false}
allowClear
/>
</FormAnt.Item>
<FormAnt.Item
style={{ width: "50%" }}
label="Status"
name="status"
rules={[
{ required: false, message: "Please input service points!" },
]}
>
<Select defaultValue="New">
<Option value="New">New</Option>
<Option value="In Progress">In Progress</Option>
<Option value="Done">Done</Option>
<Option value="Paper">Paper</Option>
<Option value="Setup">Setup</Option>
</Select>
</FormAnt.Item>
</div>
<FormAnt.Item <FormAnt.Item
label="Note" label="Note"
name="note" name="note"
rules={[{ required: true, message: "Make note!" }]} rules={[{ required: true, message: "Make note!" }]}
> >
<Input /> {/* <Input /> */}
</FormAnt.Item> <TextArea
<FormAnt.Item placeholder="Enter notes here"
label="Status" autoSize={{ minRows: 3, maxRows: 5 }}
name="status" style={{ padding: "7px 11px" }}
rules={[ />
{ required: false, message: "Please input service points!" },
]}
>
<Select defaultValue="New" style={{ width: 120 }}>
<Option value="New">New</Option>
<Option value="In Progress">In Progress</Option>
<Option value="Done">Done</Option>
<Option value="Paper">Paper</Option>
<Option value="Setup">Setup</Option>
</Select>
</FormAnt.Item> </FormAnt.Item>
</FormAnt> </FormAnt>
<FormAnt> <FormAnt>
<FormAnt.Item label="File" name="attachment"> <FormAnt.Item name="attachment">
<Upload.Dragger <Upload.Dragger
name="file" name="file"
multiple={true} multiple={true}
@ -181,10 +200,18 @@ const AddUpdate = ({
}} }}
> >
<p className="ant-upload-drag-icon"> <p className="ant-upload-drag-icon">
<UploadOutlined style={{ color: "#36cfc9" }} /> <img src={fileUpload} alt="upload" />
</p> </p>
<p className="ant-upload-text" style={{ color: "#36cfc9" }}> <p
Click or drag file to this area to upload className="ant-upload-text"
style={{ color: "#9b9daa", fontSize: 14 }}
>
Drag and drop files or{" "}
<span style={{ color: "#f99e2c" }}>Click to select</span>
<br />
<span style={{ fontSize: 13, color: "#9b9daa" }}>
Maximum file size is 10 MB
</span>
</p> </p>
</Upload.Dragger> </Upload.Dragger>
</FormAnt.Item> </FormAnt.Item>

@ -1,6 +1,6 @@
import { useState } from "react"; import { useState } from "react";
import AddUpdate from "./AddUpdate"; import AddUpdate from "./AddUpdate";
import { Select } from "antd"; import { Select, Typography } from "antd";
import UpdateTable from "./UpdateTable"; import UpdateTable from "./UpdateTable";
import { useUpdateData } from "../../Hooks/Update"; import { useUpdateData } from "../../Hooks/Update";
//@ts-ignore //@ts-ignore
@ -27,14 +27,14 @@ const Update = () => {
<div> <div>
{open && <AddUpdate refetch={refetch} open={open} setOpen={setOpen} />} {open && <AddUpdate refetch={refetch} open={open} setOpen={setOpen} />}
<div className="header d-flex" style={{ marginBottom: 16 }}> <div className="header d-flex" style={{ marginBottom: 16 }}>
<p className="title">Updates</p> <Typography className="title">Updates</Typography>
<div className="d-flex"> <div className="d-flex">
<button className="btn-add d-flex" onClick={showModal}> <button className="btn-add d-flex" onClick={showModal}>
<img style={{ marginRight: 8 }} src={addicon} alt="" /> <img style={{ marginRight: 8 }} src={addicon} alt="" />
Add Add
</button> </button>
<button <button
className={`btn-refresh-${theme && "dark"} d-flex`} className={`btn-refresh-${false && "dark"} d-flex`}
onClick={() => { onClick={() => {
refetch(); refetch();
}} }}

@ -292,6 +292,9 @@ const UpdateTable = ({
} }
loading={isLoading} loading={isLoading}
size="small" size="small"
scroll={{ x: "768px" }}
pagination={{ pageSize: 10, size: "default" }}
bordered
/> />
</div> </div>
); );

@ -24,7 +24,9 @@ const AddUser = ({
setOpen(!open); setOpen(!open);
}; };
const roleData = useRoleData(); const roleData = useRoleData();
const filteredRoleData = roleData?.data?.filter(role => role.name !== 'Owner'); const filteredRoleData = roleData?.data?.filter(
(role) => role.name !== "Owner"
);
return ( return (
<div> <div>
@ -49,7 +51,7 @@ const AddUser = ({
> >
<FormAnt <FormAnt
form={form} form={form}
layout="horizontal" layout="vertical"
name="form_in_modal" name="form_in_modal"
initialValues={{ modifier: "public" }} initialValues={{ modifier: "public" }}
> >
@ -72,7 +74,7 @@ const AddUser = ({
rules={[{ required: true }]} rules={[{ required: true }]}
> >
<Select <Select
options={filteredRoleData?.map(role => ({ options={filteredRoleData?.map((role) => ({
label: role.name, label: role.name,
value: role.id, value: role.id,
}))} }))}

@ -99,6 +99,9 @@ const UserTable = ({
rowClassName={(record, index) => rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row" index % 2 === 0 ? "odd-row" : "even-row"
} }
scroll={{ x: "768px" }}
pagination={{ pageSize: 10, size: "default" }}
bordered
/> />
</div> </div>
); );

@ -6,6 +6,7 @@ import UserTable from "./UserTable";
import IconSearch from "../../assets/searchIcon.png"; import IconSearch from "../../assets/searchIcon.png";
//@ts-ignore //@ts-ignore
import addicon from "../../assets/addiconpng.png"; import addicon from "../../assets/addiconpng.png";
import { Typography } from "antd";
const User = () => { const User = () => {
const [open, setOpen] = useState(false); const [open, setOpen] = useState(false);
@ -30,11 +31,12 @@ const User = () => {
}, 1000); }, 1000);
}; };
const theme = localStorage.getItem("theme") === "true" ? true : false; const theme = localStorage.getItem("theme") === "true" ? true : false;
return ( return (
<div> <div>
{open && <AddUser open={open} setOpen={setOpen} refetch={refetch} />} {open && <AddUser open={open} setOpen={setOpen} refetch={refetch} />}
<div className="header d-flex"> <div className="header d-flex">
<p className="title">Users</p> <Typography className="title">Users</Typography>
<button <button
className="btn-add d-flex" className="btn-add d-flex"
style={{ marginRight: 0 }} style={{ marginRight: 0 }}

@ -1,4 +1,3 @@
import { Button, Result } from "antd"; import { Button, Result } from "antd";
import { Link } from "react-router-dom"; import { Link } from "react-router-dom";
const Notfound = () => { const Notfound = () => {

Binary file not shown.

After

Width:  |  Height:  |  Size: 268 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 244 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 240 B

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.2 KiB

@ -1,4 +1,3 @@
import React from "react";
import ReactDOM from "react-dom/client"; import ReactDOM from "react-dom/client";
import "./index.css"; import "./index.css";
import App from "./App"; import App from "./App";

@ -0,0 +1,19 @@
declare module "*.png" {
const value: string;
export default value;
}
declare module "*.jpg" {
const value: string;
export default value;
}
declare module "*.jpeg" {
const value: string;
export default value;
}
declare module "*.svg" {
const value: string;
export default value;
}

@ -1,11 +1,7 @@
{ {
"compilerOptions": { "compilerOptions": {
"target": "es5", "target": "es5",
"lib": [ "lib": ["dom", "dom.iterable", "esnext"],
"dom",
"dom.iterable",
"esnext"
],
"allowJs": true, "allowJs": true,
"skipLibCheck": true, "skipLibCheck": true,
"esModuleInterop": true, "esModuleInterop": true,
@ -20,7 +16,5 @@
"noEmit": true, "noEmit": true,
"jsx": "react-jsx" "jsx": "react-jsx"
}, },
"include": [ "include": ["src/**/*", "src/types/**/*.d.ts"]
"src"
]
} }

File diff suppressed because it is too large Load Diff
Loading…
Cancel
Save