dilmurod
Dilmurod 4 weeks ago
parent 77863b959d
commit a612475609

Binary file not shown.

@ -6,6 +6,12 @@ import instance from "../api";
export type TAccountingGetParams = {
month: string;
search?: string;
team?: string;
};
export type TAccountingHistoryGetParams = {
search?: string;
team?: string;
};
export const AccountingController = {
@ -19,9 +25,16 @@ export const AccountingController = {
});
return data;
},
async history() {
async history(filterObject: TAccountingHistoryGetParams) {
const params = { ...filterObject };
if (!!filterObject.search) params.search = filterObject.search;
if (!!filterObject.team) params.team = filterObject.team;
const { data } = await instance.get<TAccountingHistory[]>(
`/employees-salaries-history/`
`/employees-salaries-history/`,
{
params,
}
);
return data;
},

@ -30,6 +30,7 @@ export const companyController = {
if (!!filterObject.name) params.name = filterObject.name;
if (!!filterObject.is_active) params.is_active = filterObject.is_active;
if (!!filterObject.page) params.page = filterObject.page;
if (!!filterObject.page_size) params.page_size = filterObject.page_size;
const { data } = await instance.get<TCompany[]>(`companies/`, {
params,

@ -24,9 +24,9 @@ export type TUpdatePostParams = {
};
export const updateController = {
async read(status: string) {
const { data } = await instance.get<TUpdate[]>(
`shift-updates/?status=${status}`
async read(status: string, page: number, page_size: number) {
const { data } = await instance.get(
`shift-updates/?status=${status}&page=${page}&page_size=${page_size}`
);
return data;
},

@ -28,11 +28,10 @@ export const userController = {
async read(filterObject: TUsersGetParams) {
const params = { ...filterObject };
if (!!filterObject.page && filterObject.page !== 0)
params.page = filterObject.page;
params.page_size = filterObject.page_size;
if (!!filterObject.name) params.name = filterObject.name;
if (!!filterObject.page) params.page = filterObject.page;
if (!!filterObject.page_size) params.page_size = filterObject.page_size;
if (Array.isArray(filterObject.team)) {
params.team = filterObject.team.join(", ");
}
@ -40,7 +39,7 @@ export const userController = {
params.role = filterObject.role.join(", ");
}
const { data } = await instance.get<TUser[]>(`users/`, { params });
const { data } = await instance.get(`users/`, { params });
return data;
},

@ -188,7 +188,7 @@
padding: 10px 15px 10px 12px;
border: none;
border-radius: 8px;
background: rgba(249, 158, 44, 1);
background: #f99e2c;
color: rgba(255, 255, 255, 1);
font-size: 14px;
font-weight: 500;
@ -215,10 +215,10 @@
letter-spacing: -0.01em;
text-align: left;
font-family: Inter;
border: 1px solid rgba(215, 216, 224, 1);
border: 1px solid #d7d8e0;
box-shadow: 0px 1px 3px 0px rgba(20, 22, 41, 0.1);
background: rgba(255, 255, 255, 1);
color: rgba(15, 17, 28, 1);
background: #ffffff;
color: #0f111c;
cursor: pointer;
}
.btn-refresh-dark {
@ -926,3 +926,10 @@
gap: 10px;
}
}
.ant-picker-disabled .ant-picker-input input {
color: #bbb !important;
}
.ant-picker-disabled .ant-picker-suffix {
color: #bbb !important;
}

@ -196,7 +196,7 @@ const App: React.FC = () => {
<img alt="" src={callIcon} />
),
getItem(
<Link to="accounting/">Accounting</Link>,
<Link to="accounting/">Salaries</Link>,
"accounting/",
<img alt="" src={accountingIcon} />
)

@ -73,7 +73,7 @@ const Login: React.FC = () => {
size={"large"}
{...input}
type="text"
placeholder="username or e-mail"
placeholder="Username or e-mail"
/>
{(meta.error || meta.submitError) && meta.touched && (
<span style={{ color: "red" }}>

@ -68,7 +68,7 @@ const ResetPassword: React.FC = () => {
size={"large"}
{...input}
type="text"
placeholder="username or e-mail"
placeholder="Username or E-mail"
/>
{(meta.error || meta.submitError) && meta.touched && (
<span style={{ color: "red" }}>

@ -1,22 +1,108 @@
import { Tabs, Typography } from "antd";
import { Button, DatePicker, Tabs, Typography } from "antd";
import { ReloadOutlined } from "@ant-design/icons";
import { theme } from "antd";
import React, { useState } from "react";
import currentMonthActive from "../../assets/currentMonthActive.svg";
import currentMonth from "../../assets/currentMonth.svg";
import lastMonth from "../../assets/lastMonth.svg";
import lastMonthActive from "../../assets/lastMonthActive.svg";
import historyActive from "../../assets/historyActive.svg";
import history from "../../assets/history.svg";
import React, { useMemo, useState } from "react";
import TabPane from "antd/es/tabs/TabPane";
import AccountingCurrent from "./AccountingCurrent";
import AccountingLast from "./AccountingLast";
import AccountingHistory from "./AccountingHistory";
import dayjs from "dayjs";
import { useAccountingData } from "../../Hooks/Accounting";
const Accounting: React.FC = () => {
const { token } = theme.useToken();
const [activeTab, setActiveTab] = useState("1");
const month = activeTab === "1" ? "current" : "last";
const { data, isLoading, refetch } = useAccountingData({
month: month,
});
const now = dayjs();
const perviousMonth = now.subtract(1, "month");
return (
<div>
<div
className="header d-flex statistics-header"
style={{ marginBottom: 16 }}
>
<Typography className="title">Accounting</Typography>
<Typography className="title">Salaries</Typography>
{activeTab == "1" ? (
<div className="d-flex">
<DatePicker
picker="month"
format={"MMMM"}
defaultValue={now}
style={{
marginRight: 10,
width: 260,
height: 36,
}}
disabled
/>
<Button
className="d-flex"
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
padding: 18,
}}
icon={<ReloadOutlined />}
onClick={() => {
refetch();
}}
>
Refresh
</Button>
</div>
) : (
""
)}
{activeTab == "2" ? (
<div className="d-flex">
<DatePicker
picker="month"
format={"MMMM"}
defaultValue={perviousMonth}
style={{
marginRight: 10,
width: 260,
height: 36,
}}
disabled
/>
<Button
className="d-flex"
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
padding: 18,
}}
icon={<ReloadOutlined />}
onClick={() => {
refetch();
}}
>
Refresh
</Button>
</div>
) : (
""
)}
</div>
<Tabs
@ -27,6 +113,11 @@ const Accounting: React.FC = () => {
<TabPane
tab={
<span style={{ display: "flex", alignItems: "center" }}>
<img
style={{ marginRight: 5 }}
src={activeTab === "1" ? currentMonthActive : currentMonth}
alt="icon"
/>
Current Month
</span>
}
@ -37,6 +128,11 @@ const Accounting: React.FC = () => {
<TabPane
tab={
<span style={{ display: "flex", alignItems: "center" }}>
<img
style={{ marginRight: 5 }}
src={activeTab === "2" ? lastMonthActive : lastMonth}
alt="icon"
/>
Last Month
</span>
}
@ -47,6 +143,11 @@ const Accounting: React.FC = () => {
<TabPane
tab={
<span style={{ display: "flex", alignItems: "center" }}>
<img
style={{ marginRight: 5 }}
src={activeTab === "3" ? historyActive : history}
alt="icon"
/>
History
</span>
}

File diff suppressed because it is too large Load Diff

@ -3,7 +3,7 @@ import { useParams } from "react-router-dom";
import axios from "axios";
import Typography from "antd/es/typography/Typography";
import { DatePicker, DatePickerProps, Table } from "antd";
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import dayjs from "dayjs";
interface User {

@ -1,17 +1,149 @@
import { Table, Tooltip } from "antd";
import React from "react";
import tagIcon from "../../assets/tagIcon.png";
import { QuestionCircleOutlined } from "@ant-design/icons";
import {
Button,
Drawer,
Input,
Select,
Table,
Tooltip,
Typography,
} from "antd";
import React, { useEffect, useRef, useState } from "react";
import tagIcon from "../../assets/tagIcon.svg";
import {
CloseOutlined,
QuestionCircleOutlined,
SearchOutlined,
} from "@ant-design/icons";
import { theme } from "antd";
import api from "../../API/api";
import { useAccountingHistory } from "../../Hooks/Accounting";
import { useTeamData } from "../../Hooks/Teams";
const { Title } = Typography;
interface Salary {
id: number;
month: string;
year: number;
number_of_tasks: number;
total_points: number;
salary_type: string;
base_salary: string;
performance_salary: string;
total_salary: string;
}
interface Employee {
id: number;
first_name: string;
last_name: string;
username: string;
}
interface SalaryData {
employee: Employee;
salaries: Salary[];
total: number;
}
const AccountingHistory: React.FC = () => {
const { data, refetch, isLoading } = useAccountingHistory();
const [open, setOpen] = useState(false);
const [userData, setUserData] = useState<SalaryData | null>(null);
const [years, setYears] = useState<number[]>([]);
const [search, setSearch] = useState<string>("");
const [team, setTeam] = useState<any>("");
const themes = localStorage.getItem("theme") === "true" ? true : false;
const { data, refetch, isLoading } = useAccountingHistory({
search: search,
team: team,
});
const [selectedUser, setSelectedUser] = useState<any>(null);
const handleRowClick = async (record: any, e: any) => {
setSelectedUser(record);
setOpen(true);
const id = record.id;
try {
const response = await api.get(`/employee-salaries/${id}/`);
setUserData(response.data);
const newYears = Array.from(
new Set(response.data.salaries.map((salary: Salary) => salary.year))
).map((year) => Number(year));
setYears(newYears);
} catch (error) {
console.error(error);
}
};
useEffect(() => {
if (userData && userData.salaries && userData.salaries.length > 0) {
const newYears = Array.from(
new Set(userData.salaries.map((salary: Salary) => Number(salary.year)))
).sort((a, b) => b - a);
setYears(newYears);
}
}, [userData]);
const { token } = theme.useToken();
const timerRef = useRef<NodeJS.Timeout | null>(null);
const handleSearchChange = (e: React.ChangeEvent<HTMLInputElement>) => {
if (timerRef.current) {
clearTimeout(timerRef.current);
}
const searchText = e.target.value;
timerRef.current = setTimeout(() => {
setSearch(searchText);
}, 1000);
};
const teamData = useTeamData({});
const teamOptions: { label: string; value: any }[] | undefined =
teamData?.data?.map((item) => ({
label: item?.name,
value: item?.name,
}));
const additionalOption = {
label: "all",
value: "",
};
if (teamOptions) {
teamOptions.unshift(additionalOption);
}
return (
<div>
<div style={{ paddingBottom: 40 }}>
<span
style={{
display: "flex",
alignItems: "center",
marginBottom: 10,
}}
>
<div style={{ marginRight: 12 }}>
<Input
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
<Select
style={{ width: 260 }}
placeholder="Team"
onChange={(value: any) => setTeam(value)}
options={teamOptions}
/>
</span>
<Table
size="small"
loading={isLoading}
@ -25,11 +157,51 @@ const AccountingHistory: React.FC = () => {
dataIndex: "no",
key: "no",
width: "5%",
align: "center",
},
{
title: "Username",
dataIndex: "username",
key: "username",
render: (text, record) => (
<Tooltip
title={
record?.full_name?.trim()
? record.full_name
: "User does not have a name"
}
>
<span>{text}</span>
</Tooltip>
),
},
{
title: "Role",
dataIndex: "role",
key: "role",
filters: [
{
text: "Tech Support",
value: "Tech Support",
},
{
text: "Checker",
value: "Checker",
},
{
text: "Accountant",
value: "Accountant",
},
],
filterMultiple: false,
onFilter: (value: any, record: any) => {
return record.role === value;
},
},
{
title: "Team",
dataIndex: "team_name",
key: "team_name",
},
{
title: "Total Tasks",
@ -42,9 +214,30 @@ const AccountingHistory: React.FC = () => {
key: "total_earned_points",
},
{
title: "Total worked months",
title: (
<Tooltip title="Total Worked Months">
<span>TWM</span>&nbsp;
<QuestionCircleOutlined />
</Tooltip>
),
dataIndex: "salary_months_count",
key: "salary_months_count",
// render: (salary_months_count) => {
// const years = Math.floor(salary_months_count / 12);
// const remainingMonths = salary_months_count % 12;
// let result = "";
// if (years > 0) {
// result += `${years} year${years > 1 ? "s" : ""}`;
// }
// if (remainingMonths > 0) {
// result += `${years > 0 ? " " : ""}${remainingMonths} month${
// remainingMonths > 1 ? "s" : ""
// }`;
// }
// return result || "0 months";
// },
},
{
title: "Salary Type",
@ -54,8 +247,10 @@ const AccountingHistory: React.FC = () => {
return <p>Task Based</p>;
} else if (record.salary_type === "hybrid") {
return <p>Hybrid</p>;
} else if (record.salary_type === "fixed") {
return <p>Fixed</p>;
} else {
return <p>{record.salary_type}</p>;
return <p>{record.salary_type}</p>; // Agar boshqa qiymat bo'lsa, oddiy qilib chiqariladi
}
},
filters: [
@ -67,8 +262,13 @@ const AccountingHistory: React.FC = () => {
text: "Task Based",
value: "task_based",
},
{
text: "Fixed",
value: "fixed",
},
],
filterMultiple: false,
// defaultFilteredValue: ["hybrid"],
onFilter: (value: any, record: any) => {
return record.salary_type === value;
},
@ -102,21 +302,12 @@ const AccountingHistory: React.FC = () => {
),
},
{
title: (
<div>
<span>Total Salary</span>&nbsp;
<Tooltip title="The calculation of salary begins at the start of the month and continues to the current day. Select a month to review salary details for prior periods.">
<QuestionCircleOutlined />
</Tooltip>
</div>
),
title: "Total Salary",
dataIndex: "total_earned_salary",
key: "total_earned_salary",
render: (text: string, record: any) => (
<span>${record.total_earned_salary}</span>
),
// sorter: (a: any, b: any) => a.salary - b.total_points,
// sortDirections: ["ascend", "descend"],
},
]}
rowClassName={(record, index) =>
@ -140,7 +331,6 @@ const AccountingHistory: React.FC = () => {
},
showLessItems: true,
}}
// onRow={(record) => ({
// onClick: () => {
// if (record.user && record.user.id) {
@ -150,7 +340,170 @@ const AccountingHistory: React.FC = () => {
// }
// },
// })}
onRow={(record) => ({
onClick: (e) => handleRowClick(record, e),
})}
/>
<Drawer
title={
<Typography className="title">
{selectedUser?.full_name?.trim()
? selectedUser.full_name
: selectedUser?.username}
</Typography>
}
placement="right"
width={850}
onClose={() => setOpen(false)}
closable={false}
open={open}
extra={
<Button
type="text"
icon={<CloseOutlined />}
onClick={() => setOpen(false)}
/>
}
>
<div className="info-div">
<Typography
style={{
fontSize: 18,
fontWeight: 700,
lineHeight: "24px",
letterSpacing: "-0.02em",
marginBottom: 16,
}}
>
Information
</Typography>
<div className="info-body">
<div className="d-flex" style={{ justifyContent: "space-between" }}>
<div>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Username</p>
<p className={!themes ? "info" : "info-dark"}>
{selectedUser?.username}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Role</p>
<p className={!themes ? "info" : "info-dark"}>
{selectedUser?.role}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Total Tasks</p>
<p className={!themes ? "info" : "info-dark"}>
{selectedUser?.total_number_of_tasks}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Total Points</p>
<p className={!themes ? "info" : "info-dark"}>
{selectedUser?.total_earned_points}
</p>
</tr>
</div>
<div>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>
Total Worked Months
</p>
<p className={!themes ? "info" : "info-dark"}>
{selectedUser?.salary_months_count}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Total Bonus</p>
<p className={!themes ? "info" : "info-dark"}>
$
{new Intl.NumberFormat("en-US").format(
selectedUser?.total_bonuses || 0
)}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Total Charge</p>
<p className={!themes ? "info" : "info-dark"}>
$
{new Intl.NumberFormat("en-US").format(
selectedUser?.total_charges || 0
)}
</p>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Total Salary</p>
<p className={!themes ? "info" : "info-dark"}>
$
{new Intl.NumberFormat("en-US").format(
selectedUser?.total_earned_salary || 0
)}
</p>
</tr>
</div>
</div>
</div>
</div>
<div style={{ marginTop: 24 }}>
{years.length === 0 ? (
<p>No data available for the years.</p>
) : (
years.map((year) => (
<div key={year} style={{ marginBottom: 32 }}>
<Title level={3}>{year}</Title>
{userData &&
userData.salaries
.filter((salary) => salary.year === year)
.map((filteredSalary) => (
<Table
key={filteredSalary.id}
dataSource={[filteredSalary]}
columns={[
{
title: <img src={tagIcon} alt="" />,
dataIndex: "no",
width: "5%",
align: "center",
render: (text, record, index) => index + 1,
},
{ title: "Year", dataIndex: "year" },
{ title: "Month", dataIndex: "month" },
{
title: "Total Charges",
dataIndex: "total_charges",
render: (text: string, record: any) => (
<span>${record.total_charges}</span>
),
},
{
title: "Total Bonus",
dataIndex: "total_bonuses",
render: (text: string, record: any) => (
<span>${record.total_bonuses}</span>
),
},
{
title: "Total Salary",
dataIndex: "total_salary",
render: (text: string, record: any) => (
<span>${record.total_salary}</span>
),
},
]}
rowKey="id"
pagination={false}
bordered
/>
))}
</div>
))
)}
</div>
</Drawer>
</div>
);
};

File diff suppressed because it is too large Load Diff

@ -21,7 +21,7 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
const { data, isLoading, refetch } = useCallData({
status: status,
page: page,
page_size: 15,
page_size: 10,
});
// const theme = localStorage.getItem("theme") === "true" ? true : false;
@ -101,7 +101,7 @@ const Call = ({ socketData }: { socketData: TSocket | undefined }) => {
</div>
<CallTable data={tableData} isLoading={isLoading} refetch={refetch} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",

@ -2,7 +2,7 @@ import { Button, Input, Modal, Space, Table, theme } from "antd";
import { TCall } from "../../types/CallRequests/TCall";
import { EditOutlined } from "@ant-design/icons";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import moment from "moment";
import { callController } from "../../API/LayoutApi/callrequests";
import {
@ -12,7 +12,6 @@ import {
} from "react-query";
import { useState } from "react";
import { TPagination } from "../../types/common/TPagination";
import useToken from "antd/es/theme/useToken";
const CallTable = ({
data,
isLoading,
@ -138,23 +137,25 @@ const CallTable = ({
index % 2 === 0 ? "odd-row" : "even-row"
}
scroll={{ x: "768px" }}
pagination={{
pageSize: 10,
size: "default",
style: {
margin: 0,
justifyContent: "end",
position: "fixed",
bottom: 0,
left: 0,
width: "100%",
backgroundColor: token.colorBgContainer,
boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
padding: "10px 0",
zIndex: 1000,
},
}}
// pagination={{
// pageSize: 10,
// size: "default",
// style: {
// margin: 0,
// justifyContent: "end",
// position: "fixed",
// bottom: 0,
// left: 0,
// width: "100%",
// backgroundColor: token.colorBgContainer,
// boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
// padding: "10px 0",
// zIndex: 1000,
// },
// }}
pagination={false}
bordered
size="small"
/>
</div>
);

@ -6,16 +6,13 @@ import {
StepBackwardOutlined,
LeftOutlined,
RightOutlined,
PlusOutlined,
SearchOutlined,
} from "@ant-design/icons";
import { useCompanyPaginated } from "../../Hooks/Companies";
import { Button, Input, Space, Typography } from "antd";
import { theme } from "antd";
// @ts-ignore
import IconSearch from "../../assets/searchIcon.png";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
import { role } from "../../App";
const Company = () => {
@ -67,18 +64,31 @@ const Company = () => {
<div className="header d-flex">
<Typography className="title">Companies</Typography>
{role !== "Checker" && (
<button
style={{ marginRight: 0 }}
className="btn-add d-flex"
// <button
// style={{ marginRight: 0 }}
// className="btn-add d-flex"
// onClick={showModal}
// >
// <img src={addicon} style={{ marginRight: 8 }} alt="" />
// Add Company
// </button>
<Button
style={{
marginRight: 0,
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
className="d-flex"
onClick={showModal}
icon={<PlusOutlined />}
>
<img src={addicon} style={{ marginRight: 8 }} alt="" />
Add Company
</button>
</Button>
)}
</div>
<div className="filter d-flex">
<div className="search-div">
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${themes}`}
@ -86,11 +96,19 @@ const Company = () => {
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div>
<Input
// className={`search-input-${themes}`}
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
</div>
<CompanyTable data={data?.data} isLoading={isLoading} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",

@ -41,7 +41,7 @@ import ontime from "../../assets/ontimeicon.svg";
// @ts-ignore
import tt from "../../assets/tticon.svg";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
// @ts-ignore
import infoIcon from "../../assets/infoIcon.png";
// @ts-ignore

@ -15,7 +15,7 @@ import ontime from "../../assets/ontimeicon.svg";
// @ts-ignore
import tt from "../../assets/tticon.svg";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { role } from "../../App";
import { theme } from "antd";
@ -121,7 +121,7 @@ function CompanyTable({
responsive: ["lg"],
},
{
width: "10%",
width: "8%",
title: "Actions",
dataIndex: "action",
render: ({ id }: { id: string }) => {

@ -1,7 +1,12 @@
import { useRef, useState } from "react";
import AddCustomer from "./AddCustomer";
import CustomerTable from "./CustomersTable";
import { LeftOutlined, RightOutlined } from "@ant-design/icons";
import {
LeftOutlined,
PlusOutlined,
RightOutlined,
SearchOutlined,
} from "@ant-design/icons";
import { useCustomerData } from "../../Hooks/Customers";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
@ -25,7 +30,7 @@ const Customer = () => {
const { data, isLoading, refetch } = useCustomerData({
name: search,
is_active: undefined,
page_size: 15,
page_size: 10,
page: page,
});
@ -59,13 +64,25 @@ const Customer = () => {
{open && <AddCustomer open={open} setOpen={setOpen} />}
<div className="header d-flex">
<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="" />
Add Driver
</button>
</button> */}
<Button
className="d-flex"
onClick={showModal}
icon={<PlusOutlined />}
style={{
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
>
Add Driver
</Button>
</div>
<div className="filter d-flex">
<div className="search-div">
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${themes}`}
@ -73,10 +90,18 @@ const Customer = () => {
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div>
<Input
// className={`search-input-${themes}`}
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
</div>
<CustomerTable data={data?.data} isLoading={isLoading} />
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",

@ -79,7 +79,9 @@ const CustomerEdit = () => {
const [companyName, setCompanyName] = useState<string>("");
const [companyVal, setCompanyVal] = useState<any>();
const { data: companyData } = useCompanyData({ name: companyName });
const { data: companyData } = useCompanyData({
name: companyName,
});
return (
<div>
@ -132,6 +134,7 @@ const CustomerEdit = () => {
label="Company"
>
<Select
showSearch
defaultValue={data?.company?.name}
onSearch={(value: any) => setCompanyName(value)}
onChange={(e: any) => {
@ -159,7 +162,6 @@ const CustomerEdit = () => {
filterOption={false}
autoClearSearchValue={false}
allowClear
// value={companyName}
/>
</Form.Item>
</Col>

@ -2,7 +2,7 @@ import { Table, Tooltip } from "antd";
import { useCompanyData } from "../../Hooks/Companies";
import { TCustomer } from "../../types/Customer/TCustomer";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
// @ts-ignore
import zippy from "../../assets/zippyicon.svg";
// @ts-ignore
@ -31,9 +31,6 @@ function CustomerTable({
const { token } = theme.useToken();
const [pageSize, setPageSize] = useState(10); // Default sahifa hajmi
const [currentPage, setCurrentPage] = useState(1);
const Row = (record: RowProps) => {
return {
onClick: () => {
@ -101,7 +98,7 @@ function CustomerTable({
rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row"
}
size="middle"
size="small"
bordered
// pagination={{
// pageSize: 10,

@ -35,7 +35,7 @@ import {
useProfData,
} from "../../Hooks/Profile";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { role } from "../../App";
import ChangePassword from "./ChangePassword";
const { Option } = Select;

@ -5,6 +5,7 @@ import {
StepBackwardOutlined,
LeftOutlined,
RightOutlined,
SearchOutlined,
} from "@ant-design/icons";
import {
Button,
@ -95,10 +96,10 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
/>
)}
<div className="header d-flex">
<Typography className="title">Requests</Typography>
<Typography className="title">Driver Requests</Typography>
</div>
<div className="filter d-flex requests-filter ">
<div className="search-div">
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${themes}`}
@ -106,6 +107,13 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div>
<Input
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
<Radio.Group
onChange={(e: RadioChangeEvent) => setStatus(e.target.value)}
@ -126,7 +134,7 @@ const Requests = ({ socketData }: { socketData: TSocket | undefined }) => {
setRequestData={setRequestData}
/>
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",

@ -1,6 +1,7 @@
import { Modal, Select, Button } from "antd";
import { TRequests } from "../../types/Requests/TRequests";
import { useEffect, useState } from "react";
import { useEffect, useRef, useState } from "react";
import { PlusOutlined } from "@ant-design/icons";
import {
useCustomerByComanyData,
useCustomerData,
@ -15,6 +16,7 @@ import {
} from "react-query";
import { useCompanyData } from "../../Hooks/Companies";
import { TPagination } from "../../types/common/TPagination";
import { TCompany } from "../../types/Company/TCompany";
const RequestsEdit = ({
modalOpen,
setModalOpen,
@ -35,16 +37,24 @@ const RequestsEdit = ({
const [companyName, setCompanyName] = useState<string>();
const [customerName, setCustomerName] = useState<string>();
const [customerNameCompany, setCustomerNameCompany] = useState<string>();
const [driverId, setDriverId] = useState<number>();
const companyData = useCompanyData({ name: companyName });
const [companyId, setCompanyId] = useState<number>();
const [customerOption, setCustomerOption] = useState<any>();
const companyData = useCompanyData({
name: companyName,
page: 1,
page_size: 5,
});
const customerData = useCustomerData({
name: customerName,
page: 1,
page_size: 5,
for_driver_request: true,
});
const customerDataByCompany = useCustomerByComanyData(
{
name: customerName,
@ -53,19 +63,49 @@ const RequestsEdit = ({
companyId
);
// useEffect(() => {
// if (companyId && customerDataByCompany) {
// const newCustomerOption = customerDataByCompany.data?.data?.map(
// (item) => ({
// label: item.name,
// value: item.id,
// })
// );
// console.log(customerDataByCompany.data);
// if (
// JSON.stringify(newCustomerOption) !== JSON.stringify(customerOption)
// ) {
// setCustomerOption(newCustomerOption);
// }
// } else if (!companyId && customerData) {
// const newCustomerOption = customerData.data?.data.map((item) => ({
// label: `${item?.name} - ${item.company?.name}`,
// value: item.id,
// }));
// if (
// JSON.stringify(newCustomerOption) !== JSON.stringify(customerOption)
// ) {
// setCustomerOption(newCustomerOption);
// }
// }
// }, [companyId, customerData, customerDataByCompany, customerOption]);
useEffect(() => {
if (companyId && customerDataByCompany) {
const newCustomerOption = customerDataByCompany.data?.data?.map(
(item) => ({
label: item.name,
value: item.id,
})
);
const customerData = customerDataByCompany.data;
if (Array.isArray(customerData)) {
const newCustomerOption = customerData.map((item) => ({
label: `${item?.name} - ${item.company?.name}`,
value: item?.id,
}));
if (
JSON.stringify(newCustomerOption) !== JSON.stringify(customerOption)
) {
setCustomerOption(newCustomerOption);
if (
JSON.stringify(newCustomerOption) !== JSON.stringify(customerOption)
) {
setCustomerOption(newCustomerOption);
}
} else {
}
} else if (!companyId && customerData) {
const newCustomerOption = customerData.data?.data.map((item) => ({
@ -84,7 +124,6 @@ const RequestsEdit = ({
const assignClick = () => {
const value = {
...requestData,
status: "Assigned",
customer_id: driverId,
};
requestsController.requestPatch(value, requestData?.id).then(() => {
@ -93,6 +132,28 @@ const RequestsEdit = ({
});
};
const timerRef = useRef<NodeJS.Timeout | null>(null);
const handleSearch = (value: string) => {
if (timerRef.current) {
clearTimeout(timerRef.current);
}
timerRef.current = setTimeout(() => {
setCustomerName(value);
}, 1000);
};
const handleSearchCompany = (value: string) => {
if (timerRef.current) {
clearTimeout(timerRef.current);
}
timerRef.current = setTimeout(() => {
setCompanyName(value);
}, 1000);
};
return (
<div>
<Modal
@ -206,12 +267,21 @@ const RequestsEdit = ({
showSearch
style={{ marginRight: 15, width: "40%" }}
placeholder="Search Company"
onSearch={(value) => setCompanyName(value)}
// onSearch={(value) => setCompanyName(value)}
onSearch={handleSearchCompany}
onChange={(value: number) => setCompanyId(value)}
options={companyData?.data?.map((item) => ({
label: item?.name,
value: item?.id,
}))}
// options={companyData?.data?.map((item: any) => ({
// label: item?.name,
// value: item?.id,
// }))}
options={
Array.isArray((companyData as any)?.data?.data)
? (companyData as any).data.data.map((item: any) => ({
label: item?.name || "No Name",
value: item?.id || "No ID",
}))
: []
}
filterOption={false}
autoClearSearchValue={false}
allowClear
@ -220,10 +290,16 @@ const RequestsEdit = ({
style={{ width: "50%" }}
showSearch
placeholder="Search Driver"
onSearch={(value) => setCustomerName(value)}
// onSearch={(value) => setCustomerName(value)}
onSearch={handleSearch}
onChange={(value: number) => setDriverId(value)}
options={customerOption}
filterOption={false}
// options={options}
filterOption={(input: string, option?: { label?: string }) =>
option?.label
?.toLowerCase()
.includes(input.toLowerCase()) ?? false
}
autoClearSearchValue={false}
allowClear
/>
@ -255,8 +331,8 @@ const RequestsEdit = ({
alignItems: "center",
}}
className="btn-add"
icon={<PlusOutlined />}
>
<img src={plus} alt="" style={{ marginRight: 8 }} />
New driver
</Button>
</div>

@ -5,7 +5,7 @@ import {
RefetchQueryFilters,
} from "react-query";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import moment from "moment";
import { TRequests } from "../../types/Requests/TRequests";
import { useEffect, useState } from "react";
@ -86,12 +86,13 @@ const RequestsTable = ({
...u,
}))}
loading={isLoading}
size="middle"
size="small"
columns={[
{
title: <img src={tagIcon} alt="" />,
dataIndex: "no",
width: "5%",
align: "center",
},
{
title: "Full name",

@ -7,7 +7,7 @@ import {
} from "react-query";
import { TService } from "../../types/Service/TService";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { role } from "../../App";
import { theme } from "antd";
@ -94,6 +94,7 @@ const ServiceTable = ({
},
}}
bordered
size="middle"
/>
</div>
);

@ -4,8 +4,9 @@ import AddService from "./AddService";
import ServiceTable from "./ServiceTable";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
import { PlusOutlined } from "@ant-design/icons";
import { role } from "../../App";
import { Pagination, Space, Typography } from "antd";
import { Button, Pagination, Space, Typography } from "antd";
import { theme } from "antd";
const Service = () => {
@ -13,12 +14,6 @@ const Service = () => {
const { token } = theme.useToken();
const page_size = 10;
const handlePageChange = (page: number) => {
setPage(page);
};
const { data, isLoading, refetch } = useServiceData();
const [open, setOpen] = useState(false);
const showModal = () => {
@ -31,10 +26,23 @@ const Service = () => {
<div className="header d-flex" style={{ marginBottom: "10px" }}>
<Typography className="title">Services</Typography>
{role !== "Checker" && (
<button onClick={showModal} className="btn-add d-flex">
<img src={addicon} style={{ marginRight: 8 }} alt="" />
// <button onClick={showModal} className="btn-add d-flex">
// <img src={addicon} style={{ marginRight: 8 }} alt="" />
// Add Service
// </button>
<Button
style={{
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
onClick={showModal}
className="d-flex"
icon={<PlusOutlined />} // Ant-design ikonkasi
>
Add Service
</button>
</Button>
)}
</div>
<ServiceTable data={data} isLoading={isLoading} refetch={refetch} />

@ -26,10 +26,12 @@ import {
Button,
DatePicker,
DatePickerProps,
Input,
Select,
Tabs,
Typography,
} from "antd";
import { SearchOutlined } from "@ant-design/icons";
import TabPane from "antd/es/tabs/TabPane";
// @ts-ignore
import IconSearch from "../../assets/searchIcon.png";
@ -478,7 +480,7 @@ const Stat = () => {
marginBottom: 10,
}}
>
<div className="search-div" style={{ marginRight: 12 }}>
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${theme}`}
@ -486,6 +488,13 @@ const Stat = () => {
placeholder="Search"
onChange={handleTechSupportSearchChange}
/>
</div> */}
<div>
<Input
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleTechSupportSearchChange}
/>
</div>
</span>
<StatisticsSupportTable
@ -515,7 +524,7 @@ const Stat = () => {
marginBottom: 10,
}}
>
<div className="search-div" style={{ marginRight: 12 }}>
{/* <div className="search-div" style={{ marginRight: 12 }}>
<img src={IconSearch} alt="" />
<input
className={`search-input-${theme}`}
@ -523,6 +532,13 @@ const Stat = () => {
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div style={{ marginRight: 12 }}>
<Input
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
<Select
style={{ width: 260 }}
@ -539,7 +555,7 @@ const Stat = () => {
<Button
type="primary"
onClick={(e) => handleSave("team")}
style={{ marginTop: 10 }}
style={{ marginTop: 10, marginBottom: 40 }}
>
Save as file
</Button>

@ -7,7 +7,7 @@ import {
} from "react-query";
import { QuestionCircleOutlined, QuestionOutlined } from "@ant-design/icons";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { theme } from "antd";
@ -119,6 +119,7 @@ const StatTable = ({
padding: "10px 0",
zIndex: 1000,
},
showLessItems: true,
}}
rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row"

@ -8,7 +8,7 @@ import {
import { theme } from "antd";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
const StatTeamTable = ({
data,
isLoading,

@ -7,7 +7,7 @@ import {
RefetchQueryFilters,
} from "react-query";
import { TStatCreators } from "../../types/Statistic/TStat";
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
const StatisticsSupportTable = ({
data,
@ -22,7 +22,7 @@ const StatisticsSupportTable = ({
}) => {
const { token } = theme.useToken();
return (
<div style={{ maxHeight: "400px" }}>
<div style={{ paddingBottom: 40 }}>
<Table
loading={isLoading}
size="small"
@ -66,6 +66,7 @@ const StatisticsSupportTable = ({
padding: "10px 0",
zIndex: 1000,
},
showLessItems: true,
}}
rowClassName={(record, index) =>
index % 2 === 0 ? "odd-row" : "even-row"

@ -10,6 +10,7 @@ import {
Row,
Col,
} from "antd";
import { CloseOutlined } from "@ant-design/icons";
import { taskController } from "../../API/LayoutApi/tasks";
import { useEffect, useState } from "react";
import { useServiceData } from "../../Hooks/Services";
@ -194,9 +195,11 @@ const AddTask = ({
}}
>
<span>Add Task</span>
<Button type="text" onClick={handleCancel}>
<img src={closeIcon} />
</Button>
<Button
type="text"
onClick={handleCancel}
icon={<CloseOutlined />}
/>
</div>
}
okText="Create"

@ -1,5 +1,14 @@
import { TTask } from "../../types/Tasks/TTasks";
import { Dropdown, MenuProps, Modal, Table, Tabs, Tooltip, theme } from "antd";
import {
Button,
Dropdown,
MenuProps,
Modal,
Table,
Tabs,
Tooltip,
theme,
} from "antd";
import TabPane from "antd/es/tabs/TabPane";
import { role, timeZone } from "../../App";
import { useTaskHistory } from "../../Hooks/Tasks";
@ -11,7 +20,11 @@ import { TTeam } from "../../types/Team/TTeam";
import {
ArrowRightOutlined,
CaretRightOutlined,
CloseOutlined,
EditOutlined,
ForwardOutlined,
RotateRightOutlined,
UploadOutlined,
} from "@ant-design/icons";
import { TSocket } from "../../types/common/TSocket";
// @ts-ignore
@ -248,7 +261,7 @@ const TaskModal = ({
placement="bottom"
arrow={{ pointAtCenter: true }}
>
<button
{/* <button
disabled={recordTask?.status !== "New"}
style={{
marginRight: 12,
@ -259,10 +272,21 @@ const TaskModal = ({
>
<img src={forwardIcon} alt="" />
Forward
</button>
</button> */}
<Button
disabled={recordTask?.status !== "New"}
style={{
marginRight: 10,
color: token.colorText,
backgroundColor: token.colorBgContainer,
}}
icon={<ArrowRightOutlined />}
>
Forward
</Button>
</Dropdown>
)}
<button
{/* <button
style={{
marginLeft: 12,
color: token.colorText,
@ -273,8 +297,19 @@ const TaskModal = ({
>
<img src={uploadIcon} alt="" />
Upload file
</button>
<button
</button> */}
<Button
style={{
marginLeft: 10,
color: token.colorText,
backgroundColor: token.colorBgContainer,
}}
onClick={showUploadModal}
icon={<UploadOutlined />}
>
Upload file
</Button>
{/* <button
onClick={handleCancel}
style={{
marginLeft: 20,
@ -284,7 +319,16 @@ const TaskModal = ({
className={`btn-modal-action-${themes && "dark"}`}
>
<img style={{ margin: 2 }} src={closeIcon} alt="" />
</button>
</button> */}
<Button
onClick={handleCancel}
style={{
marginLeft: 20,
color: token.colorText,
backgroundColor: token.colorBgContainer,
}}
icon={<CloseOutlined />}
/>
</div>
</div>
<div className="TaskModal-content">
@ -317,7 +361,7 @@ const TaskModal = ({
</p>
<div className="info-body">
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Comapany</p>
<p className={!themes ? "sub" : "sub-dark"}>Company</p>
<p className={!themes ? "info" : "info-dark"}>
{recordTask?.company?.name}
</p>
@ -349,7 +393,7 @@ const TaskModal = ({
<p className={!themes ? "info" : "info-dark"}>
{pti === false ? "Do" : "No need"}
</p>
<button
{/* <button
style={{
marginLeft: 10,
background: "#cecece",
@ -361,7 +405,23 @@ const TaskModal = ({
onClick={(e) => setPti(!pti)}
>
change
</button>
</button> */}
<Button
type="primary" // Maxsus bir stil bilan
size="small"
onClick={(e) => setPti(!pti)}
style={{
marginLeft: 10,
background: token.colorBgContainer,
color: "#f68900",
outline: "none",
border: "1px solid #f68900",
// padding: 4,
borderRadius: 4,
}}
>
Change
</Button>
</tr>
<tr>
<p className={!themes ? "sub" : "sub-dark"}>Created at</p>
@ -383,7 +443,7 @@ const TaskModal = ({
onChange={(e) => setText(e.target.value)}
/>
</div>
<button
{/* <button
style={{
marginTop: 20,
color: token.colorText,
@ -394,7 +454,18 @@ const TaskModal = ({
>
<img src={editIcon} alt="" />
Save
</button>
</button> */}
<Button
style={{
marginTop: 20,
color: "#f99e2c",
backgroundColor: token.colorBgContainer,
}}
onClick={(e) => patchTask()}
icon={<EditOutlined />}
>
Save
</Button>
</div>
</TabPane>
<TabPane

@ -15,7 +15,7 @@ import ontime from "../../assets/ontimeicon.svg";
// @ts-ignore
import tt from "../../assets/tticon.svg";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
// @ts-ignore
import tgIcon from "../../assets/telegram.png";
//

@ -6,6 +6,11 @@ import { useTeamData } from "../../Hooks/Teams";
import { useTasks } from "../../Hooks/Tasks";
import { TTask } from "../../types/Tasks/TTasks";
import { isMobile, role, team_id } from "../../App";
import {
PlusOutlined,
ReloadOutlined,
SearchOutlined,
} from "@ant-design/icons";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
//@ts-ignore
@ -219,16 +224,48 @@ const Task = ({
</div>
<div className="d-flex">
{role !== "Checker" && (
<button className="btn-add d-flex" onClick={showModal}>
<img
style={{ marginRight: isMobile ? "0px" : "8px" }}
src={addicon}
alt=""
/>
{!isMobile && "Add Task"}
</button>
// <button className="btn-add d-flex" onClick={showModal}>
// <img
// style={{ marginRight: isMobile ? "0px" : "8px" }}
// src={addicon}
// alt="Add Icon"
// />
// {!isMobile && "Add Task"}
// </button>
<Button
size="middle"
className="d-flex"
onClick={showModal}
icon={!isMobile && <PlusOutlined />}
style={{
marginRight: isMobile ? "0px" : "8px",
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
>
Add Task
</Button>
)}
<button
<Button
className="d-flex" // Bu yerda `false` o'rniga shartni to'g'rilashingiz kerak
style={{
backgroundColor: token.colorBgContainer, // Dynamic background color
color: token.colorText,
padding: 18, // Dynamic text color
}}
icon={<ReloadOutlined />} // Ikonka
onClick={() => {
refetch();
if (!isLive) {
connect();
}
}}
>
Refresh
</Button>
{/* <button
className={`btn-refresh-${false && "dark"} d-flex`}
style={{
backgroundColor: token.colorBgContainer,
@ -247,11 +284,11 @@ const Task = ({
alt=""
/>
{!isMobile && "Refetch"}
</button>
</button> */}
</div>
</div>
<div className={`filter ${isMobile ? "mobile-filter" : "d-flex"}`}>
<div className="search-div">
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${themes}`}
@ -259,6 +296,14 @@ const Task = ({
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div>
<Input
// className={`search-input-${themes}`}
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
<Select
style={{
@ -292,7 +337,7 @@ const Task = ({
showErrorModal={showErrorModal}
setErrorModal={setErrorModal}
/>
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",

@ -61,10 +61,17 @@ const AddTeam = ({
>
<Select
mode="multiple"
options={data?.map((items) => ({
label: items.username,
value: items.id,
}))}
showSearch
options={data?.map(
(item: any): { label: string; value: number } => ({
label: item.username,
value: item.id,
})
)}
filterOption={(input: string, option?: { label?: string }) =>
option?.label?.toLowerCase().includes(input.toLowerCase()) ??
false
}
/>
</FormAnt.Item>
</FormAnt>

@ -20,7 +20,7 @@ import { role } from "../../App";
import { useUserData } from "../../Hooks/Users";
import AddUserToTeam from "./AddUserToTeam";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
// @ts-ignore
import infoIcon from "../../assets/infoIcon.png";
// @ts-ignore
@ -38,6 +38,7 @@ type MyObjectType = {
const TeamEdit = () => {
const { id } = useParams<params>();
const { data, refetch, status }: MyObjectType = useTeamOne(id);
let navigate = useNavigate();
const onSubmit = async (value: any) => {
@ -57,7 +58,7 @@ const TeamEdit = () => {
}
};
const userData = useUserData({ name: "", team: id });
const userData = useUserData({ name: "", team: data?.name });
const [open, setOpen] = useState(false);
const showModal = () => {
@ -147,7 +148,7 @@ const TeamEdit = () => {
key="2"
>
<Table
dataSource={userData?.data?.map((item, i) => ({
dataSource={userData?.data?.map((item: any, i: any) => ({
no: i + 1,
...item,
}))}
@ -178,8 +179,8 @@ const TeamEdit = () => {
/>
<Button
type="primary"
style={{ marginLeft: "auto" }}
size={"large"}
style={{ marginLeft: "auto", marginTop: 15 }}
size="middle"
onClick={showModal}
>
Add User

@ -8,7 +8,7 @@ import {
} from "react-query";
import { timeZone } from "../../App";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { theme } from "antd";
@ -62,7 +62,7 @@ const TeamTable = ({
},
]}
pagination={{
pageSize: 15,
pageSize: 10,
size: "default",
style: {
margin: 0,

@ -1,10 +1,11 @@
import { useEffect, useState } from "react";
import { useState } from "react";
import { useTeamData } from "../../Hooks/Teams";
import TeamTable from "./TeamTable";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
import AddTeam from "./AddTeam";
import { Typography } from "antd";
import { Button, Typography } from "antd";
import { PlusOutlined } from "@ant-design/icons";
const Team = () => {
const { data, isLoading, refetch } = useTeamData({});
@ -20,14 +21,27 @@ const Team = () => {
{open && <AddTeam refetch={refetch} open={open} setOpen={setOpen} />}
<div className="header d-flex" style={{ marginBottom: 16 }}>
<Typography className="title">Teams</Typography>
<button
{/* <button
className="btn-add d-flex"
style={{ marginRight: 0 }}
onClick={showModal}
>
<img src={addicon} style={{ marginRight: 8 }} alt="" />
Add Team
</button>
</button> */}
<Button
className="btn-add d-flex"
style={{
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
onClick={showModal}
icon={<PlusOutlined />} // Ant-design ikonkasi
>
Add Team
</Button>
</div>
<TeamTable data={data} isLoading={isLoading} refetch={refetch} />
</div>

@ -1,23 +1,44 @@
import { useState } from "react";
import { useRef, useState } from "react";
import AddUpdate from "./AddUpdate";
import { Select, Typography, theme } from "antd";
import { Button, Input, Select, Space, Typography, theme } from "antd";
import {
LeftOutlined,
PlusOutlined,
ReloadOutlined,
RightOutlined,
} from "@ant-design/icons";
import UpdateTable from "./UpdateTable";
import { useUpdateData } from "../../Hooks/Update";
//@ts-ignore
import addicon from "../../assets/addiconpng.png";
//@ts-ignore
import refreshicon from "../../assets/refreshIcon.png";
const Update = () => {
const [open, setOpen] = useState(false);
const timerRef = useRef<NodeJS.Timeout | null>(null);
const handleSelectChange = (value: any) => {
if (timerRef.current) {
clearTimeout(timerRef.current);
}
timerRef.current = setTimeout(() => {
setStatus(value);
}, 1000);
};
const [status, setStatus] = useState<any>([
"New",
"In Progress",
"Paper",
"Setup",
]);
const { Option } = Select;
const { data, refetch, isLoading } = useUpdateData(status);
const [page, setPage] = useState<number>(1);
const page_size: number = 10;
const { data, refetch, isLoading } = useUpdateData(status, page, page_size);
const showModal = () => {
setOpen(true);
@ -25,17 +46,42 @@ const Update = () => {
const { token } = theme.useToken();
const Next = () => {
const a = Number(page) + 1;
setPage(a);
};
const Previos = () => {
Number(page);
if (page > 1) {
const a = Number(page) - 1;
setPage(a);
}
};
return (
<div>
{open && <AddUpdate refetch={refetch} open={open} setOpen={setOpen} />}
<div className="header d-flex" style={{ marginBottom: 16 }}>
<Typography className="title">Updates</Typography>
<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="" />
Add
</button>
<button
</button> */}
<Button
style={{
marginRight: 10,
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
className="d-flex"
onClick={showModal}
icon={<PlusOutlined />} // Ant-design ikonkasi
>
Add
</Button>
{/* <button
className={`btn-refresh-${false && "dark"} d-flex`}
style={{
backgroundColor: token.colorBgContainer,
@ -47,14 +93,28 @@ const Update = () => {
>
<img style={{ marginRight: 8 }} src={refreshicon} alt="" />
Refresh
</button>
</button> */}
<Button
className="d-flex"
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
padding: 18,
}}
onClick={() => {
refetch();
}}
icon={<ReloadOutlined />}
>
Refresh
</Button>
</div>
</div>
<div className="filter d-flex">
<Select
style={{ width: 260, marginLeft: 10 }}
placeholder="status"
onChange={(value: any) => setStatus(value)}
placeholder="Status"
onChange={handleSelectChange}
mode="multiple"
defaultValue={[]}
>
@ -65,7 +125,65 @@ const Update = () => {
<Option value="Setup">Setup</Option>
</Select>
</div>
<UpdateTable data={data} refetch={refetch} isLoading={isLoading} />
<UpdateTable data={data?.data} refetch={refetch} isLoading={isLoading} />
<Space style={{ width: "100%", marginTop: 40 }} direction="vertical">
<Space
style={{
justifyContent: "end",
position: "fixed",
bottom: 0,
left: 0,
width: "100%",
backgroundColor: token.colorBgContainer,
boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
padding: "10px 0",
zIndex: 1000,
}}
wrap
>
<Button
onClick={Previos}
disabled={data?.previous ? false : true}
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
border: "none",
}}
>
<LeftOutlined />
</Button>
<Input
disabled
style={{
width: 40,
textAlign: "center",
background: token.colorBgContainer,
border: "1px solid",
borderColor: token.colorText,
color: token.colorText,
cursor: "pointer",
}}
value={page}
onChange={(e) => {
let num = e.target.value;
if (Number(num) && num !== "0") {
setPage(Number(num));
}
}}
/>
<Button
onClick={Next}
disabled={data?.next ? false : true}
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
border: "none",
}}
>
<RightOutlined />
</Button>
</Space>
</Space>
</div>
);
};

@ -15,7 +15,11 @@ import {
Upload,
} from "antd";
import { updateController } from "../../API/LayoutApi/update";
import { UploadOutlined } from "@ant-design/icons";
import {
InfoCircleFilled,
InfoCircleOutlined,
UploadOutlined,
} from "@ant-design/icons";
import Notfound from "../../Utils/Notfound";
import { companyController } from "../../API/LayoutApi/companies";
import { customerController } from "../../API/LayoutApi/customers";
@ -148,6 +152,7 @@ const UpdateEdit = () => {
}
}
const [activeTab, setActiveTab] = useState("1");
return (
<div>
{role !== "Checker" || inCharge == admin_id || inCharge == null ? (
@ -194,14 +199,20 @@ const UpdateEdit = () => {
>
<Row gutter={[16, 10]}>
{companyId !== null && (
<Col span={6}>
<Col span={8}>
<Form.Item
wrapperCol={{ span: "100%" }}
label="Company"
>
{companyValue !== undefined && (
{/* {data?.company.id !== undefined && (
// <Input
// defaultValue={data?.company?.id}
// readOnly
// />
)} */}
{data?.company.id !== undefined && (
<Input
defaultValue={companyValue}
defaultValue={data?.company?.name}
readOnly
/>
)}
@ -209,14 +220,14 @@ const UpdateEdit = () => {
</Col>
)}
{customerId !== null && (
<Col span={6}>
<Col span={8}>
<Form.Item
wrapperCol={{ span: "100%" }}
label="Driver"
>
{customerValue !== undefined && (
{data?.customer.id !== undefined && (
<Input
defaultValue={customerValue}
defaultValue={data?.customer.name}
readOnly
/>
)}

@ -1,4 +1,5 @@
import { Space, Table, Tooltip, theme } from "antd";
import { Button, Space, Table, Tooltip, theme } from "antd";
import { PushpinOutlined } from "@ant-design/icons";
import moment from "moment";
import { useCompanyData } from "../../Hooks/Companies";
import { useCustomerData } from "../../Hooks/Customers";
@ -12,7 +13,7 @@ import {
import { TUpdate } from "../../types/Update/TUpdate";
import { useEffect, useState } from "react";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
// @ts-ignore
import pin from "../../assets/pinicon.png";
// @ts-ignore
@ -39,9 +40,9 @@ const UpdateTable = ({
options?: (RefetchOptions & RefetchQueryFilters<TPageData>) | undefined
) => Promise<QueryObserverResult<TUpdate[], unknown>>;
}) => {
const CompanyData = useCompanyData({});
const CustomerData = useCustomerData({});
const AdminData = useUserData({});
// const CompanyData = useCompanyData({});
// const CustomerData = useCustomerData({});
// const AdminData = useUserData({});
const [isTextSelected, setIsTextSelected] = useState(false);
@ -86,7 +87,7 @@ const UpdateTable = ({
const { token } = theme.useToken();
return (
<div>
<div style={{ paddingBottom: 40 }}>
<Table
onRow={(record) => ({
onClick: (event) => Row(record, event),
@ -94,18 +95,18 @@ const UpdateTable = ({
dataSource={data?.map((u, i) => ({
no: i + 1,
...u,
company_name: CompanyData?.data?.find(
(company: any) => company.id === u.company_id
)?.name,
customer_name: CustomerData?.data?.data?.find(
(customer: any) => customer.id === u.customer_id
)?.name,
in_charge_name: AdminData?.data?.find(
(admin: any) => admin.id === u.provider_id
)?.username,
executor_name: AdminData?.data?.find(
(admin: any) => admin.id === u.executor_id
)?.username,
// company_name: CompanyData?.data?.find(
// (company: any) => company.id === u.company_id
// )?.name,
// customer_name: CustomerData?.data?.data?.find(
// (customer: any) => customer.id === u.customer_id
// )?.name,
// in_charge_name: AdminData?.data?.find(
// (admin: any) => admin.id === u.provider_id
// )?.username,
// executor_name: AdminData?.data?.find(
// (admin: any) => admin.id === u.executor_id
// )?.username,
created: moment(u?.created_at, "YYYY-MM-DD HH:mm:ss").format(
"DD.MM.YYYY HH:mm"
),
@ -116,6 +117,7 @@ const UpdateTable = ({
title: <img src={tagIcon} alt="" />,
dataIndex: "no",
width: "4%",
align: "center",
},
{
title: "Company",
@ -154,15 +156,15 @@ const UpdateTable = ({
},
{
title: "Created by",
dataIndex: "provider ",
dataIndex: "provider",
ellipsis: {
showTitle: false,
},
responsive: ["xl"],
width: "10%",
render: (note: string) => (
<Tooltip placement="topLeft" title={note}>
{note}
render: (provider: { username: string }) => (
<Tooltip placement="topLeft" title={provider?.username}>
{provider?.username}
</Tooltip>
),
},
@ -174,9 +176,9 @@ const UpdateTable = ({
},
responsive: ["lg"],
width: "10%",
render: (note: string) => (
<Tooltip placement="topLeft" title={note}>
{note}
render: (executor: { username: string }) => (
<Tooltip placement="topLeft" title={executor?.username}>
{executor?.username}
</Tooltip>
),
},
@ -243,15 +245,31 @@ const UpdateTable = ({
{
title: "Actions",
dataIndex: "action",
width: "8%",
width: "6%",
align: "center",
render: (record: TUpdate) => {
return (
<div className="notedit">
{record.status !== "Done" && (
<Space>
{record.is_pinned ? (
<button
className="btn-unpin"
// <button
// className="btn-unpin"
// onClick={(e) => {
// const updateData = {
// is_pinned: false,
// };
// updateController
// .updatePatch(updateData, record.id)
// .then(() => {
// refetch();
// });
// }}
// >
// <img src={unpin} alt="" />
// </button>
<Button
style={{ background: "#f99e2c", color: "#fff" }}
onClick={(e) => {
const updateData = {
is_pinned: false,
@ -262,13 +280,26 @@ const UpdateTable = ({
refetch();
});
}}
>
<img src={unpin} alt="" />
</button>
icon={<PushpinOutlined />}
></Button>
) : (
<button
className="btn-pin"
style={{ paddingTop: 2 }}
// <button
// className="btn-pin"
// style={{ paddingTop: 2 }}
// onClick={(e) => {
// const updateData = {
// is_pinned: true,
// };
// updateController
// .updatePatch(updateData, record.id)
// .then(() => {
// refetch();
// });
// }}
// >
// <img src={pin} alt="" />
// </button>
<Button
onClick={(e) => {
const updateData = {
is_pinned: true,
@ -279,9 +310,8 @@ const UpdateTable = ({
refetch();
});
}}
>
<img src={pin} alt="" />
</button>
icon={<PushpinOutlined />}
></Button>
)}
</Space>
)}
@ -296,22 +326,24 @@ const UpdateTable = ({
loading={isLoading}
size="small"
scroll={{ x: "768px" }}
pagination={{
pageSize: 15,
size: "default",
style: {
margin: 0,
justifyContent: "end",
position: "fixed",
bottom: 0,
left: 0,
width: "100%",
backgroundColor: token.colorBgContainer,
boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
padding: "10px 0",
zIndex: 1000,
},
}}
// pagination={{
// pageSize: 10,
// size: "default",
// style: {
// margin: 0,
// justifyContent: "end",
// position: "fixed",
// bottom: 0,
// left: 0,
// width: "100%",
// backgroundColor: token.colorBgContainer,
// boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
// padding: "10px 0",
// zIndex: 1000,
// },
// showLessItems: true,
// }}
pagination={false}
bordered
/>
</div>

@ -87,17 +87,20 @@ const UserEdit = () => {
const [showInput, setShowInput] = useState(false);
const handleChange = (value: string) => {
if (data?.role.name === "Checker") {
setShowInput(value === "hybrid");
if (data?.role.name !== "Accountant") {
setShowInput(true);
}
};
const [form] = Form.useForm();
useEffect(() => {
if (data?.role.name === "Checker" && data?.salary_type === "hybrid") {
if (
(data?.role.name !== "Accountant" && data?.salary_type === "hybrid") ||
data?.salary_type === "fixed"
) {
setShowInput(true);
} else if (data?.role.name !== "Checker") {
} else if (data?.role.name === "Accountant") {
setShowInput(false);
}
}, [data]);
@ -204,7 +207,7 @@ const UserEdit = () => {
</Col>
)}
{data?.role?.name === "Checker" &&
{data?.role?.name !== "Accountant" &&
form.getFieldsValue().role_id !== 3 && (
<Col span={4}>
<Form.Item
@ -218,6 +221,7 @@ const UserEdit = () => {
>
<Option value="task_based">Task based</Option>
<Option value="hybrid">Hybrid</Option>
<Option value="fixed">Fixed</Option>
</Select>
</Form.Item>
</Col>

@ -8,7 +8,7 @@ import {
} from "react-query";
import { useNavigate } from "react-router-dom";
// @ts-ignore
import tagIcon from "../../assets/tagIcon.png";
import tagIcon from "../../assets/tagIcon.svg";
import { isMobile, role } from "../../App";
import { userController } from "../../API/LayoutApi/users";
@ -50,7 +50,7 @@ const UserTable = ({
const { token } = theme.useToken();
return (
<div>
<div style={{ paddingBottom: 40 }}>
<Table
onRow={(record) => Row(record)}
dataSource={data?.map((u, i) => ({
@ -61,7 +61,7 @@ const UserTable = ({
...u,
}))}
loading={isLoading}
size="middle"
size="small"
columns={[
{
title: <img src={tagIcon} alt="" />,
@ -106,24 +106,24 @@ const UserTable = ({
index % 2 === 0 ? "odd-row" : "even-row"
}
scroll={{ x: "768px" }}
pagination={{
pageSize: 10,
size: "default",
style: {
margin: 0,
justifyContent: "end",
position: "fixed",
bottom: 0,
left: 0,
width: "100%",
backgroundColor: token.colorBgContainer,
boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
padding: "10px 0",
zIndex: 1000,
},
showLessItems: true,
}}
// pagination={false}
// pagination={{
// pageSize: 10,
// size: "default",
// style: {
// margin: 0,
// justifyContent: "end",
// position: "fixed",
// bottom: 0,
// left: 0,
// width: "100%",
// backgroundColor: token.colorBgContainer,
// boxShadow: "0 4px 8px rgba(0, 0, 0, 0.4)",
// padding: "10px 0",
// zIndex: 1000,
// },
// showLessItems: true,
// }}
pagination={false}
bordered
/>
</div>

@ -2,7 +2,12 @@ import { useRef, useState } from "react";
import { useUserData } from "../../Hooks/Users";
import AddUser from "./AddUser";
import UserTable from "./UserTable";
import { LeftOutlined, RightOutlined } from "@ant-design/icons";
import {
LeftOutlined,
PlusOutlined,
RightOutlined,
SearchOutlined,
} from "@ant-design/icons";
import { theme } from "antd";
// @ts-ignore
import IconSearch from "../../assets/searchIcon.png";
@ -33,6 +38,8 @@ const User = () => {
const { data, refetch, isLoading } = useUserData({
name: search,
team: "",
page: page,
page_size: 10,
});
const timerRef = useRef<NodeJS.Timeout | null>(null);
@ -55,17 +62,29 @@ const User = () => {
{open && <AddUser open={open} setOpen={setOpen} refetch={refetch} />}
<div className="header d-flex">
<Typography className="title">Users</Typography>
<button
{/* <button
className="btn-add d-flex"
style={{ marginRight: 0 }}
onClick={showModal}
>
<img src={addicon} style={{ marginRight: 8 }} alt="" />
Invite User
</button>
</button> */}
<Button
className="d-flex"
style={{
backgroundColor: "#f99e2c",
color: "white",
padding: 18,
}}
onClick={showModal}
icon={<PlusOutlined />}
>
Invite User
</Button>
</div>
<div className="filter d-flex">
<div className="search-div">
{/* <div className="search-div">
<img src={IconSearch} alt="" />
<input
className={`search-input-${themes}`}
@ -73,11 +92,18 @@ const User = () => {
placeholder="Search"
onChange={handleSearchChange}
/>
</div> */}
<div>
<Input
placeholder="Search"
prefix={<SearchOutlined />}
onChange={handleSearchChange}
/>
</div>
</div>
<UserTable data={data} isLoading={isLoading} refetch={refetch} />
<UserTable data={data?.data} isLoading={isLoading} refetch={refetch} />
{/* <Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space style={{ width: "100%", marginTop: 10 }} direction="vertical">
<Space
style={{
justifyContent: "end",
@ -125,7 +151,7 @@ const User = () => {
<Button
onClick={Next}
// disabled={data?.next ? false : true}
disabled={data?.next ? false : true}
style={{
backgroundColor: token.colorBgContainer,
color: token.colorText,
@ -135,7 +161,7 @@ const User = () => {
<RightOutlined />
</Button>
</Space>
</Space> */}
</Space>
</div>
);
};

@ -2,23 +2,33 @@ import { useQuery } from "react-query";
import {
AccountingController,
TAccountingGetParams,
TAccountingHistoryGetParams,
} from "../../API/LayoutApi/accounting";
export const useAccountingData = ({ month }: TAccountingGetParams) => {
export const useAccountingData = ({
month,
search,
team,
}: TAccountingGetParams) => {
return useQuery(
[`stats/all-users/`, month],
[`/employees-salaries`, month, search, team],
() =>
AccountingController.read({
month,
search,
team,
}),
{ refetchOnWindowFocus: false }
);
};
export const useAccountingHistory = () => {
export const useAccountingHistory = ({
search,
team,
}: TAccountingHistoryGetParams) => {
return useQuery(
[`employees-salaries-history/`],
() => AccountingController.history(),
[`/employees-salaries-history/`, search, team],
() => AccountingController.history({ search, team }),
{
refetchOnWindowFocus: false,
}

@ -7,11 +7,12 @@ import {
export const useCompanyData = ({
name,
page,
page_size,
is_active,
}: TCompanyGetParams) => {
return useQuery(
[`companies/`, name, page, is_active],
() => companyController.read({ name, page, is_active }),
[`companies/`, name, page, is_active,page_size],
() => companyController.read({ name, page, is_active,page_size }),
{ refetchOnWindowFocus: false }
);
};

@ -32,7 +32,7 @@ export const useCustomerByComanyData = (
return useQuery(
[`customers-by-company/${id}`, obj],
() => customerController.customerByCompany(obj, id),
{ refetchOnWindowFocus: false }
{ refetchOnWindowFocus: false, enabled: !!id }
);
};

@ -9,10 +9,14 @@ import { updateController } from "../../API/LayoutApi/update";
// );
// };
export const useUpdateData = (status: string) => {
export const useUpdateData = (
status: string,
page: number,
page_size: number
) => {
return useQuery(
[`shift-updates`, status],
() => updateController.read(status),
[`shift-updates`, status, page, page_size],
() => updateController.read(status, page, page_size),
{ refetchOnWindowFocus: false }
);
};

@ -0,0 +1,4 @@
<svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M11.3563 4.59023C10.6312 4.20262 9.8216 3.99984 8.99935 3.99984L8.99935 8.99984L5.46382 12.5354C6.04524 13.1168 6.76108 13.5459 7.54793 13.7845C8.33477 14.0232 9.16835 14.0642 9.9748 13.9038C10.7813 13.7433 11.5357 13.3865 12.1713 12.8649C12.8069 12.3433 13.3041 11.6729 13.6187 10.9133C13.9334 10.1536 14.0559 9.32805 13.9753 8.50975C13.8947 7.69146 13.6135 6.90567 13.1567 6.22199C12.6999 5.53831 12.0815 4.97784 11.3563 4.59023Z" fill="#A1A2AB"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M17.3327 8.99984C17.3327 13.6022 13.6017 17.3332 8.99935 17.3332C4.39698 17.3332 0.666016 13.6022 0.666016 8.99984C0.666016 4.39746 4.39698 0.666504 8.99935 0.666504C13.6017 0.666504 17.3327 4.39746 17.3327 8.99984ZM15.666 8.99984C15.666 12.6817 12.6812 15.6665 8.99935 15.6665C5.31745 15.6665 2.33268 12.6817 2.33268 8.99984C2.33268 5.31794 5.31745 2.33317 8.99935 2.33317C12.6812 2.33317 15.666 5.31794 15.666 8.99984Z" fill="#A1A2AB"/>
</svg>

After

Width:  |  Height:  |  Size: 1.0 KiB

@ -0,0 +1,4 @@
<svg width="18" height="18" viewBox="0 0 18 18" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M11.3563 4.59023C10.6312 4.20262 9.8216 3.99984 8.99935 3.99984L8.99935 8.99984L5.46382 12.5354C6.04524 13.1168 6.76108 13.5459 7.54793 13.7845C8.33477 14.0232 9.16835 14.0642 9.9748 13.9038C10.7813 13.7433 11.5357 13.3865 12.1713 12.8649C12.8069 12.3433 13.3041 11.6729 13.6187 10.9133C13.9334 10.1536 14.0559 9.32805 13.9753 8.50975C13.8947 7.69146 13.6135 6.90567 13.1567 6.22199C12.6999 5.53831 12.0815 4.97784 11.3563 4.59023Z" fill="#F99E2C"/>
<path fill-rule="evenodd" clip-rule="evenodd" d="M17.3327 8.99984C17.3327 13.6022 13.6017 17.3332 8.99935 17.3332C4.39698 17.3332 0.666016 13.6022 0.666016 8.99984C0.666016 4.39746 4.39698 0.666504 8.99935 0.666504C13.6017 0.666504 17.3327 4.39746 17.3327 8.99984ZM15.666 8.99984C15.666 12.6817 12.6812 15.6665 8.99935 15.6665C5.31745 15.6665 2.33268 12.6817 2.33268 8.99984C2.33268 5.31794 5.31745 2.33317 8.99935 2.33317C12.6812 2.33317 15.666 5.31794 15.666 8.99984Z" fill="#F99E2C"/>
</svg>

After

Width:  |  Height:  |  Size: 1.0 KiB

@ -0,0 +1,3 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M8 0.5C12.1421 0.5 15.5 3.85786 15.5 8C15.5 12.1421 12.1421 15.5 8 15.5C6.57532 15.5 5.20787 15.1015 4.02567 14.3615C3.54412 14.0601 3.09808 13.7047 2.69639 13.303C2.29408 12.9006 1.93834 12.4538 1.63667 11.9714C0.897834 10.7898 0.5 9.42348 0.5 8C0.5 7.77154 0.510233 7.54423 0.53058 7.31856C0.561576 6.97478 0.865395 6.72121 1.20918 6.75221C1.55296 6.7832 1.80653 7.08702 1.77553 7.43081C1.75855 7.61917 1.75 7.80902 1.75 8C1.75 9.18739 2.08106 10.3244 2.69652 11.3086C2.94803 11.7109 3.24473 12.0835 3.58032 12.4192C3.9154 12.7543 4.28742 13.0506 4.6889 13.3019C5.6737 13.9184 6.81161 14.25 8 14.25C11.4518 14.25 14.25 11.4518 14.25 8C14.25 4.54822 11.4518 1.75 8 1.75C6.00226 1.75 4.16627 2.69401 2.99812 4.25164L5.29516 4.25174C5.64033 4.25174 5.92016 4.53157 5.92016 4.87674C5.92016 5.19316 5.68503 5.45465 5.37996 5.49604L5.29516 5.50174H1.54167C1.22525 5.50174 0.963758 5.26662 0.922372 4.96155L0.916667 4.87674V1.12674C0.916667 0.781566 1.19649 0.501744 1.54167 0.501744C1.85808 0.501744 2.11958 0.736873 2.16096 1.04194L2.16667 1.12674L2.16583 3.28623C3.57156 1.5467 5.6969 0.5 8 0.5ZM7.375 3.83333C7.69125 3.83333 7.95288 4.06861 7.99429 4.37356L8 4.45833V8H9.875C10.22 8 10.5 8.28 10.5 8.625C10.5 8.94125 10.2647 9.20288 9.95977 9.24429L9.875 9.25H7.375C7.05875 9.25 6.79712 9.01472 6.75571 8.70977L6.75 8.625V4.45833C6.75 4.11333 7.03 3.83333 7.375 3.83333Z" fill="#A1A2AB"/>
</svg>

After

Width:  |  Height:  |  Size: 1.5 KiB

@ -0,0 +1,3 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M8 0.5C12.1421 0.5 15.5 3.85786 15.5 8C15.5 12.1421 12.1421 15.5 8 15.5C6.57532 15.5 5.20787 15.1015 4.02567 14.3615C3.54412 14.0601 3.09808 13.7047 2.69639 13.303C2.29408 12.9006 1.93834 12.4538 1.63667 11.9714C0.897834 10.7898 0.5 9.42348 0.5 8C0.5 7.77154 0.510233 7.54423 0.53058 7.31856C0.561576 6.97478 0.865395 6.72121 1.20918 6.75221C1.55296 6.7832 1.80653 7.08702 1.77553 7.43081C1.75855 7.61917 1.75 7.80902 1.75 8C1.75 9.18739 2.08106 10.3244 2.69652 11.3086C2.94803 11.7109 3.24473 12.0835 3.58032 12.4192C3.9154 12.7543 4.28742 13.0506 4.6889 13.3019C5.6737 13.9184 6.81161 14.25 8 14.25C11.4518 14.25 14.25 11.4518 14.25 8C14.25 4.54822 11.4518 1.75 8 1.75C6.00226 1.75 4.16627 2.69401 2.99812 4.25164L5.29516 4.25174C5.64033 4.25174 5.92016 4.53157 5.92016 4.87674C5.92016 5.19316 5.68503 5.45465 5.37996 5.49604L5.29516 5.50174H1.54167C1.22525 5.50174 0.963758 5.26662 0.922372 4.96155L0.916667 4.87674V1.12674C0.916667 0.781566 1.19649 0.501744 1.54167 0.501744C1.85808 0.501744 2.11958 0.736873 2.16096 1.04194L2.16667 1.12674L2.16583 3.28623C3.57156 1.5467 5.6969 0.5 8 0.5ZM7.375 3.83333C7.69125 3.83333 7.95288 4.06861 7.99429 4.37356L8 4.45833V8H9.875C10.22 8 10.5 8.28 10.5 8.625C10.5 8.94125 10.2647 9.20288 9.95977 9.24429L9.875 9.25H7.375C7.05875 9.25 6.79712 9.01472 6.75571 8.70977L6.75 8.625V4.45833C6.75 4.11333 7.03 3.83333 7.375 3.83333Z" fill="#F99E2C"/>
</svg>

After

Width:  |  Height:  |  Size: 1.5 KiB

@ -0,0 +1,3 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M12.7917 0.5C14.2874 0.5 15.5 1.71256 15.5 3.20833V12.7917C15.5 14.2874 14.2874 15.5 12.7917 15.5H3.20833C1.71256 15.5 0.5 14.2874 0.5 12.7917V3.20833C0.5 1.71256 1.71256 0.5 3.20833 0.5H12.7917ZM14.25 5.08333H1.75V12.7917C1.75 13.5971 2.40292 14.25 3.20833 14.25H12.7917C13.5971 14.25 14.25 13.5971 14.25 12.7917V5.08333ZM4.45833 10.0833C5.03363 10.0833 5.5 10.5497 5.5 11.125C5.5 11.7003 5.03363 12.1667 4.45833 12.1667C3.88304 12.1667 3.41667 11.7003 3.41667 11.125C3.41667 10.5497 3.88304 10.0833 4.45833 10.0833ZM8 10.0833C8.5753 10.0833 9.04167 10.5497 9.04167 11.125C9.04167 11.7003 8.5753 12.1667 8 12.1667C7.4247 12.1667 6.95833 11.7003 6.95833 11.125C6.95833 10.5497 7.4247 10.0833 8 10.0833ZM4.45833 6.75C5.03363 6.75 5.5 7.21637 5.5 7.79167C5.5 8.36696 5.03363 8.83333 4.45833 8.83333C3.88304 8.83333 3.41667 8.36696 3.41667 7.79167C3.41667 7.21637 3.88304 6.75 4.45833 6.75ZM8 6.75C8.5753 6.75 9.04167 7.21637 9.04167 7.79167C9.04167 8.36696 8.5753 8.83333 8 8.83333C7.4247 8.83333 6.95833 8.36696 6.95833 7.79167C6.95833 7.21637 7.4247 6.75 8 6.75ZM11.5417 6.75C12.117 6.75 12.5833 7.21637 12.5833 7.79167C12.5833 8.36696 12.117 8.83333 11.5417 8.83333C10.9664 8.83333 10.5 8.36696 10.5 7.79167C10.5 7.21637 10.9664 6.75 11.5417 6.75ZM12.7917 1.75H3.20833C2.40292 1.75 1.75 2.40292 1.75 3.20833V3.83333H14.25V3.20833C14.25 2.40292 13.5971 1.75 12.7917 1.75Z" fill="#A1A2AB"/>
</svg>

After

Width:  |  Height:  |  Size: 1.5 KiB

@ -0,0 +1,3 @@
<svg width="16" height="16" viewBox="0 0 16 16" fill="none" xmlns="http://www.w3.org/2000/svg">
<path d="M12.7917 0.5C14.2874 0.5 15.5 1.71256 15.5 3.20833V12.7917C15.5 14.2874 14.2874 15.5 12.7917 15.5H3.20833C1.71256 15.5 0.5 14.2874 0.5 12.7917V3.20833C0.5 1.71256 1.71256 0.5 3.20833 0.5H12.7917ZM14.25 5.08333H1.75V12.7917C1.75 13.5971 2.40292 14.25 3.20833 14.25H12.7917C13.5971 14.25 14.25 13.5971 14.25 12.7917V5.08333ZM4.45833 10.0833C5.03363 10.0833 5.5 10.5497 5.5 11.125C5.5 11.7003 5.03363 12.1667 4.45833 12.1667C3.88304 12.1667 3.41667 11.7003 3.41667 11.125C3.41667 10.5497 3.88304 10.0833 4.45833 10.0833ZM8 10.0833C8.5753 10.0833 9.04167 10.5497 9.04167 11.125C9.04167 11.7003 8.5753 12.1667 8 12.1667C7.4247 12.1667 6.95833 11.7003 6.95833 11.125C6.95833 10.5497 7.4247 10.0833 8 10.0833ZM4.45833 6.75C5.03363 6.75 5.5 7.21637 5.5 7.79167C5.5 8.36696 5.03363 8.83333 4.45833 8.83333C3.88304 8.83333 3.41667 8.36696 3.41667 7.79167C3.41667 7.21637 3.88304 6.75 4.45833 6.75ZM8 6.75C8.5753 6.75 9.04167 7.21637 9.04167 7.79167C9.04167 8.36696 8.5753 8.83333 8 8.83333C7.4247 8.83333 6.95833 8.36696 6.95833 7.79167C6.95833 7.21637 7.4247 6.75 8 6.75ZM11.5417 6.75C12.117 6.75 12.5833 7.21637 12.5833 7.79167C12.5833 8.36696 12.117 8.83333 11.5417 8.83333C10.9664 8.83333 10.5 8.36696 10.5 7.79167C10.5 7.21637 10.9664 6.75 11.5417 6.75ZM12.7917 1.75H3.20833C2.40292 1.75 1.75 2.40292 1.75 3.20833V3.83333H14.25V3.20833C14.25 2.40292 13.5971 1.75 12.7917 1.75Z" fill="#F99E2C"/>
</svg>

After

Width:  |  Height:  |  Size: 1.5 KiB

Binary file not shown.

Before

Width:  |  Height:  |  Size: 412 B

@ -0,0 +1,3 @@
<svg width="14" height="14" viewBox="0 0 14 14" fill="none" xmlns="http://www.w3.org/2000/svg">
<path fill-rule="evenodd" clip-rule="evenodd" d="M6.32334 1.10965C6.3839 0.746476 6.13858 0.402969 5.77541 0.342406C5.41223 0.281843 5.06873 0.527159 5.00816 0.890334L4.43428 4.3317L1.66641 4.33333C1.29822 4.33355 0.999918 4.6322 1.00013 5.00039C1.00035 5.36858 1.299 5.66688 1.66719 5.66666L4.21192 5.66516L3.76718 8.3321L0.999866 8.33333C0.631676 8.33349 0.333332 8.6321 0.333496 9.00029C0.33366 9.36848 0.63227 9.66683 1.00046 9.66666L3.54482 9.66553L3.00705 12.8903C2.94649 13.2535 3.19181 13.597 3.55498 13.6576C3.91816 13.7181 4.26166 13.4728 4.32223 13.1097L4.89667 9.66493L8.21369 9.66345L7.67587 12.8904C7.61534 13.2536 7.86068 13.5971 8.22387 13.6576C8.58705 13.7181 8.93053 13.4728 8.99106 13.1096L9.56552 9.66285L12.3338 9.66161C12.702 9.66145 13.0003 9.36284 13.0002 8.99465C13 8.62646 12.7014 8.32811 12.3332 8.32828L9.78776 8.32941L10.2324 5.66162L13.0005 5.65999C13.3687 5.65977 13.667 5.36112 13.6668 4.99293C13.6666 4.62474 13.3679 4.32644 12.9997 4.32666L10.4546 4.32816L10.9911 1.10959C11.0516 0.746411 10.8062 0.402926 10.4431 0.342396C10.0799 0.281866 9.7364 0.527213 9.67587 0.890393L9.10278 4.32895L5.78616 4.3309L6.32334 1.10965ZM5.5638 5.66437L5.11903 8.33149L8.43593 8.33001L8.88053 5.66242L5.5638 5.66437Z" fill="#9B9DAA"/>
</svg>

After

Width:  |  Height:  |  Size: 1.3 KiB

@ -1,18 +1,18 @@
export type TUpdate = {
id: number;
company_id?: number;
customer_id?: number;
provider_id?: number;
executor_id?: number | null;
company?: number;
customer?: number;
provider?: number;
executor?: number | null;
status: string;
note: string;
solution: string;
is_active: boolean;
is_pinned: boolean;
updated_at: Date;
created_at: Date;
}
id: number;
company_id?: number;
customer_id?: number;
provider_id?: number;
executor_id?: number | null;
company?: number;
customer?: number;
provider?: number;
executor?: number | null;
status: string;
note: string;
solution: string;
is_active: boolean;
is_pinned: boolean;
updated_at: Date;
created_at: Date;
};

@ -1,21 +1,12 @@
export type TUser = {
id: number;
username: string;
team: {id: number, name: string}
role: {id: number, name: string}
first_name: string | '';
last_name: string | '';
is_active: boolean;
is_superuser: boolean;
salary_type: string;
salary_base_amount: number;
}
export type TUserResponse = {
page: number;
page_size: number;
next: string | null;
previous: string | null;
current_time: string;
data: TUser[];
};
id: number;
username: string;
team: { id: number; name: string };
role: { id: number; name: string };
first_name: string | "";
last_name: string | "";
is_active: boolean;
is_superuser: boolean;
salary_type: string;
salary_base_amount: number;
};

Loading…
Cancel
Save