feat: statuses colors

This commit is contained in:
2025-10-11 12:15:03 +04:00
parent 5e56daa765
commit a899177623
15 changed files with 222 additions and 130 deletions

View File

@ -1,6 +1,7 @@
import React, { FC } from "react"; import React, { FC } from "react";
import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react"; import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react";
import { Box, Group, Menu, Text } from "@mantine/core"; import { Box, Menu } from "@mantine/core";
import DropdownMenuItem from "@/components/ui/DropdownMenuItem/DropdownMenuItem";
import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon"; import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon";
import { BoardSchema } from "@/lib/client"; import { BoardSchema } from "@/lib/client";
@ -32,26 +33,16 @@ const BoardMenu: FC<Props> = ({
</Box> </Box>
</Menu.Target> </Menu.Target>
<Menu.Dropdown> <Menu.Dropdown>
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={startEditing}
e.stopPropagation(); icon={<IconEdit />}
startEditing(); label={"Переименовать"}
}}> />
<Group wrap={"nowrap"}> <DropdownMenuItem
<IconEdit /> onClick={() => onDeleteBoard(board)}
<Text>Переименовать</Text> icon={<IconTrash />}
</Group> label={"Удалить"}
</Menu.Item> />
<Menu.Item
onClick={e => {
e.stopPropagation();
onDeleteBoard(board);
}}>
<Group wrap={"nowrap"}>
<IconTrash />
<Text>Удалить</Text>
</Group>
</Menu.Item>
</Menu.Dropdown> </Menu.Dropdown>
</Menu> </Menu>
); );

View File

@ -1,4 +0,0 @@
.header {
border-bottom: solid dodgerblue 3px;
}

View File

@ -5,7 +5,6 @@ import { useBoardsContext } from "@/app/deals/contexts/BoardsContext";
import { useStatusesContext } from "@/app/deals/contexts/StatusesContext"; import { useStatusesContext } from "@/app/deals/contexts/StatusesContext";
import InPlaceInput from "@/components/ui/InPlaceInput/InPlaceInput"; import InPlaceInput from "@/components/ui/InPlaceInput/InPlaceInput";
import { StatusSchema } from "@/lib/client"; import { StatusSchema } from "@/lib/client";
import styles from "./StatusColumnHeader.module.css";
type Props = { type Props = {
status: StatusSchema; status: StatusSchema;
@ -29,7 +28,9 @@ const StatusColumnHeader: FC<Props> = ({ status, isDragging }) => {
p={"sm"} p={"sm"}
wrap={"nowrap"} wrap={"nowrap"}
mb={"xs"} mb={"xs"}
className={styles.header}> style={{
borderBottom: `solid ${status.color} 3px`,
}}>
<InPlaceInput <InPlaceInput
value={status.name} value={status.name}
onChange={value => handleSave(value)} onChange={value => handleSave(value)}
@ -53,6 +54,9 @@ const StatusColumnHeader: FC<Props> = ({ status, isDragging }) => {
board={selectedBoard} board={selectedBoard}
status={status} status={status}
handleEdit={startEditing} handleEdit={startEditing}
onStatusColorChange={color =>
statusesCrud.onUpdate(status.id, { color })
}
refetchStatuses={refetchStatuses} refetchStatuses={refetchStatuses}
onDeleteStatus={statusesCrud.onDelete} onDeleteStatus={statusesCrud.onDelete}
/> />

View File

@ -3,17 +3,22 @@ import {
IconDotsVertical, IconDotsVertical,
IconEdit, IconEdit,
IconExchange, IconExchange,
IconPalette,
IconTrash, IconTrash,
} from "@tabler/icons-react"; } from "@tabler/icons-react";
import { Box, Group, Menu, Text } from "@mantine/core"; import { Box, Menu } from "@mantine/core";
import { modals } from "@mantine/modals";
import statusColors from "@/app/deals/utils/statusColors";
import DropdownMenuItem from "@/components/ui/DropdownMenuItem/DropdownMenuItem";
import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon";
import { useDrawersContext } from "@/drawers/DrawersContext"; import { useDrawersContext } from "@/drawers/DrawersContext";
import useIsMobile from "@/hooks/utils/useIsMobile"; import useIsMobile from "@/hooks/utils/useIsMobile";
import { BoardSchema, StatusSchema } from "@/lib/client"; import { BoardSchema, StatusSchema } from "@/lib/client";
import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon";
type Props = { type Props = {
status: StatusSchema; status: StatusSchema;
handleEdit: () => void; handleEdit: () => void;
onStatusColorChange: (color: string) => void;
board: BoardSchema | null; board: BoardSchema | null;
onDeleteStatus: (status: StatusSchema) => void; onDeleteStatus: (status: StatusSchema) => void;
refetchStatuses?: () => void; refetchStatuses?: () => void;
@ -23,6 +28,7 @@ type Props = {
const StatusMenu: FC<Props> = ({ const StatusMenu: FC<Props> = ({
status, status,
handleEdit, handleEdit,
onStatusColorChange,
board, board,
onDeleteStatus, onDeleteStatus,
refetchStatuses, refetchStatuses,
@ -31,6 +37,31 @@ const StatusMenu: FC<Props> = ({
const isMobile = useIsMobile(); const isMobile = useIsMobile();
const { openDrawer } = useDrawersContext(); const { openDrawer } = useDrawersContext();
const openStatusesMobileEditor = () => {
if (!board) return;
openDrawer({
key: "statusesMobileEditorDrawer",
props: {
board,
},
onClose: refetchStatuses,
});
};
const openStatusColorPicker = () => {
if (!board) return;
modals.openContextModal({
modal: "statusColorPickerModal",
title: "Изменение цвета статуса",
withCloseButton: false,
innerProps: {
color: status.color,
onChange: onStatusColorChange,
switches: statusColors,
},
});
};
return ( return (
<Menu> <Menu>
<Menu.Target> <Menu.Target>
@ -43,44 +74,27 @@ const StatusMenu: FC<Props> = ({
</Box> </Box>
</Menu.Target> </Menu.Target>
<Menu.Dropdown> <Menu.Dropdown>
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={handleEdit}
e.stopPropagation(); icon={<IconEdit />}
handleEdit(); label={"Переименовать"}
}}> />
<Group wrap={"nowrap"}> <DropdownMenuItem
<IconEdit /> onClick={openStatusColorPicker}
<Text>Переименовать</Text> icon={<IconPalette />}
</Group> label={"Изменить цвет"}
</Menu.Item> />
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={() => onDeleteStatus(status)}
e.stopPropagation(); icon={<IconTrash />}
onDeleteStatus(status); label={"Удалить"}
}}> />
<Group wrap={"nowrap"}>
<IconTrash />
<Text>Удалить</Text>
</Group>
</Menu.Item>
{isMobile && withChangeOrderButton && ( {isMobile && withChangeOrderButton && (
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={openStatusesMobileEditor}
e.stopPropagation(); icon={<IconExchange />}
if (!board) return; label={"Изменить порядок"}
openDrawer({ />
key: "statusesMobileEditorDrawer",
props: {
board,
},
onClose: refetchStatuses,
});
}}>
<Group wrap={"nowrap"}>
<IconExchange />
<Text>Изменить порядок</Text>
</Group>
</Menu.Item>
)} )}
</Menu.Dropdown> </Menu.Dropdown>
</Menu> </Menu>

View File

@ -1,7 +1,8 @@
import React, { FC } from "react"; import React, { FC } from "react";
import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react"; import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react";
import { Box, Group, Menu, Text } from "@mantine/core"; import { Box, Menu } from "@mantine/core";
import { useProjectsContext } from "@/app/deals/contexts/ProjectsContext"; import { useProjectsContext } from "@/app/deals/contexts/ProjectsContext";
import DropdownMenuItem from "@/components/ui/DropdownMenuItem/DropdownMenuItem";
import { ProjectSchema } from "@/lib/client"; import { ProjectSchema } from "@/lib/client";
import styles from "./../ProjectsEditorDrawer.module.css"; import styles from "./../ProjectsEditorDrawer.module.css";
@ -23,26 +24,16 @@ const ProjectMenu: FC<Props> = ({ project, startEditing }) => {
</Box> </Box>
</Menu.Target> </Menu.Target>
<Menu.Dropdown> <Menu.Dropdown>
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={startEditing}
e.stopPropagation(); icon={<IconEdit />}
startEditing(); label={"Переименовать"}
}}> />
<Group wrap={"nowrap"}> <DropdownMenuItem
<IconEdit /> onClick={() => projectsCrud.onDelete(project)}
<Text>Переименовать</Text> icon={<IconTrash />}
</Group> label={"Удалить"}
</Menu.Item> />
<Menu.Item
onClick={e => {
e.stopPropagation();
projectsCrud.onDelete(project);
}}>
<Group wrap={"nowrap"}>
<IconTrash />
<Text>Удалить</Text>
</Group>
</Menu.Item>
</Menu.Dropdown> </Menu.Dropdown>
</Menu> </Menu>
); );

View File

@ -40,6 +40,7 @@ const StatusMobile: FC<Props> = ({ status, board }) => {
board={board} board={board}
onDeleteStatus={statusesCrud.onDelete} onDeleteStatus={statusesCrud.onDelete}
handleEdit={startEditing} handleEdit={startEditing}
onStatusColorChange={color => statusesCrud.onUpdate(status.id, { color })}
withChangeOrderButton={false} withChangeOrderButton={false}
/> />
</Group> </Group>

View File

@ -0,0 +1,50 @@
"use client";
import { useState } from "react";
import { ColorPicker, Flex, Space, Text } from "@mantine/core";
import { ContextModalProps } from "@mantine/modals";
import statusColors from "@/app/deals/utils/statusColors";
import InlineButton from "@/components/ui/InlineButton/InlineButton";
type Props = {
color: string;
onChange: (color: string) => void;
switches?: string[];
};
const ColorPickerModal = ({
id,
context,
innerProps,
}: ContextModalProps<Props>) => {
const [color, setColor] = useState(innerProps.color);
return (
<Flex
gap={"xs"}
direction={"column"}>
<ColorPicker
format={"hex"}
value={color}
onChange={setColor}
swatches={statusColors}
styles={{
swatch: {
flex: 1,
},
}}
fullWidth
/>
<Space />
<InlineButton
onClick={() => {
innerProps.onChange(color);
context.closeModal(id);
}}>
<Text>Сохранить</Text>
</InlineButton>
</Flex>
);
};
export default ColorPickerModal;

View File

@ -0,0 +1,16 @@
const statusColors = [
"#228be6",
"#15aabf",
"#12b886",
"#40c057",
"#82c91e",
"#fab005",
"#fd7e14",
"#fa5252",
"#e64980",
"#be4bdb",
"#7950f2",
"#4c6ef5",
];
export default statusColors;

View File

@ -1,7 +1,8 @@
import React, { CSSProperties, FC } from "react"; import React, { CSSProperties, FC } from "react";
import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react"; import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react";
import { Box, Flex, Group, Menu, Text } from "@mantine/core"; import { Box, Flex, Menu } from "@mantine/core";
import ActionIconWithTip from "@/components/ui/ActionIconWithTip/ActionIconWithTip"; import ActionIconWithTip from "@/components/ui/ActionIconWithTip/ActionIconWithTip";
import DropdownMenuItem from "@/components/ui/DropdownMenuItem/DropdownMenuItem";
import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon"; import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon";
import useIsMobile from "@/hooks/utils/useIsMobile"; import useIsMobile from "@/hooks/utils/useIsMobile";
@ -31,26 +32,16 @@ const UpdateDeleteTableActions: FC<Props> = ({
</Box> </Box>
</Menu.Target> </Menu.Target>
<Menu.Dropdown> <Menu.Dropdown>
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={onChange}
e.stopPropagation(); icon={<IconEdit />}
onChange(); label={"Редактировать"}
}}> />
<Group wrap={"nowrap"}> <DropdownMenuItem
<IconEdit /> onClick={onDelete}
<Text>Редактировать</Text> icon={<IconTrash />}
</Group> label={"Удалить"}
</Menu.Item> />
<Menu.Item
onClick={e => {
e.stopPropagation();
onDelete();
}}>
<Group wrap={"nowrap"}>
<IconTrash />
<Text>Удалить</Text>
</Group>
</Menu.Item>
</Menu.Dropdown> </Menu.Dropdown>
</Menu> </Menu>
); );

View File

@ -0,0 +1,26 @@
import React, { FC, MouseEventHandler, ReactNode } from "react";
import { Group, Menu, Text } from "@mantine/core";
type Props = {
onClick: MouseEventHandler<HTMLButtonElement>;
icon: ReactNode;
label: string;
};
const DropdownMenuItem: FC<Props> = ({ icon, label, onClick }) => {
const onClickWrapper: MouseEventHandler<HTMLButtonElement> = e => {
e.stopPropagation();
onClick(e);
};
return (
<Menu.Item onClick={onClickWrapper}>
<Group wrap={"nowrap"}>
{icon}
<Text>{label}</Text>
</Group>
</Menu.Item>
);
};
export default DropdownMenuItem;

View File

@ -1,4 +1,5 @@
import { LexoRank } from "lexorank"; import { LexoRank } from "lexorank";
import statusColors from "@/app/deals/utils/statusColors";
import { useCrudOperations } from "@/hooks/cruds/baseCrud"; import { useCrudOperations } from "@/hooks/cruds/baseCrud";
import { import {
CreateStatusSchema, CreateStatusSchema,
@ -10,8 +11,8 @@ import {
deleteStatusMutation, deleteStatusMutation,
updateStatusMutation, updateStatusMutation,
} from "@/lib/client/@tanstack/react-query.gen"; } from "@/lib/client/@tanstack/react-query.gen";
import { getMaxByLexorank } from "@/utils/lexorank/max";
import { getNewLexorank } from "@/utils/lexorank/generation"; import { getNewLexorank } from "@/utils/lexorank/generation";
import { getMaxByLexorank } from "@/utils/lexorank/max";
type Props = { type Props = {
statuses: StatusSchema[]; statuses: StatusSchema[];
@ -48,10 +49,13 @@ export const useStatusesCrud = ({
const newLexorank = getNewLexorank( const newLexorank = getNewLexorank(
lastBoard ? LexoRank.parse(lastBoard.lexorank) : null lastBoard ? LexoRank.parse(lastBoard.lexorank) : null
); );
const nextColorIdx = statuses.length % statusColors.length;
const color = statusColors[nextColorIdx];
return { return {
name: data.name!, name: data.name!,
boardId, boardId,
lexorank: newLexorank.toString(), lexorank: newLexorank.toString(),
color,
}; };
}, },
getUpdateEntity: (old, update) => ({ getUpdateEntity: (old, update) => ({

View File

@ -748,6 +748,10 @@ export type CreateStatusSchema = {
* Lexorank * Lexorank
*/ */
lexorank: string; lexorank: string;
/**
* Color
*/
color: string;
}; };
/** /**
@ -1522,6 +1526,10 @@ export type StatusSchema = {
* Lexorank * Lexorank
*/ */
lexorank: string; lexorank: string;
/**
* Color
*/
color: string;
}; };
/** /**
@ -2013,6 +2021,10 @@ export type UpdateStatusSchema = {
* Lexorank * Lexorank
*/ */
lexorank?: string | null; lexorank?: string | null;
/**
* Color
*/
color?: string | null;
}; };
/** /**

View File

@ -312,6 +312,7 @@ export const zStatusSchema = z.object({
id: z.int(), id: z.int(),
name: z.string(), name: z.string(),
lexorank: z.string(), lexorank: z.string(),
color: z.string(),
}); });
/** /**
@ -556,6 +557,7 @@ export const zCreateStatusSchema = z.object({
name: z.string(), name: z.string(),
boardId: z.int(), boardId: z.int(),
lexorank: z.string(), lexorank: z.string(),
color: z.string(),
}); });
/** /**
@ -1190,6 +1192,7 @@ export const zUpdateServicesKitResponse = z.object({
export const zUpdateStatusSchema = z.object({ export const zUpdateStatusSchema = z.object({
name: z.optional(z.union([z.string(), z.null()])), name: z.optional(z.union([z.string(), z.null()])),
lexorank: z.optional(z.union([z.string(), z.null()])), lexorank: z.optional(z.union([z.string(), z.null()])),
color: z.optional(z.union([z.string(), z.null()])),
}); });
/** /**

View File

@ -1,5 +1,6 @@
import BarcodeTemplateEditorModal from "@/app/barcode-templates/modals/BarcodeTemplateFormModal/BarcodeTemplateEditorModal"; import BarcodeTemplateEditorModal from "@/app/barcode-templates/modals/BarcodeTemplateFormModal/BarcodeTemplateEditorModal";
import ClientEditorModal from "@/app/clients/modals/ClientFormModal/ClientFormModal"; import ClientEditorModal from "@/app/clients/modals/ClientFormModal/ClientFormModal";
import ColorPickerModal from "@/app/deals/modals/ColorPickerModal/ColorPickerModal";
import DealsBoardFiltersModal from "@/app/deals/modals/DealsBoardFiltersModal/DealsBoardFiltersModal"; import DealsBoardFiltersModal from "@/app/deals/modals/DealsBoardFiltersModal/DealsBoardFiltersModal";
import DealsScheduleFiltersModal from "@/app/deals/modals/DealsScheduleFiltersModal/DealsScheduleFiltersModal"; import DealsScheduleFiltersModal from "@/app/deals/modals/DealsScheduleFiltersModal/DealsScheduleFiltersModal";
import DealsTableFiltersModal from "@/app/deals/modals/DealsTableFiltersModal/DealsTableFiltersModal"; import DealsTableFiltersModal from "@/app/deals/modals/DealsTableFiltersModal/DealsTableFiltersModal";
@ -36,4 +37,5 @@ export const modals = {
barcodeTemplateEditorModal: BarcodeTemplateEditorModal, barcodeTemplateEditorModal: BarcodeTemplateEditorModal,
clientEditorModal: ClientEditorModal, clientEditorModal: ClientEditorModal,
printBarcodeModal: PrintBarcodeModal, printBarcodeModal: PrintBarcodeModal,
statusColorPickerModal: ColorPickerModal,
}; };

View File

@ -1,6 +1,7 @@
import React, { FC } from "react"; import React, { FC } from "react";
import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react"; import { IconDotsVertical, IconEdit, IconTrash } from "@tabler/icons-react";
import { Box, Group, Menu, Text } from "@mantine/core"; import { Box, Menu } from "@mantine/core";
import DropdownMenuItem from "@/components/ui/DropdownMenuItem/DropdownMenuItem";
import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon"; import ThemeIcon from "@/components/ui/ThemeIcon/ThemeIcon";
import { DealProductSchema } from "@/lib/client"; import { DealProductSchema } from "@/lib/client";
@ -26,26 +27,16 @@ const ProductMenu: FC<Props> = ({ value, onChange, onDelete }) => {
</Box> </Box>
</Menu.Target> </Menu.Target>
<Menu.Dropdown> <Menu.Dropdown>
<Menu.Item <DropdownMenuItem
onClick={e => { onClick={() => onChange(value)}
e.stopPropagation(); icon={<IconEdit />}
onChange(value); label={"Редактировать"}
}}> />
<Group wrap={"nowrap"}> <DropdownMenuItem
<IconEdit /> onClick={() => onDelete(value)}
<Text>Редактировать</Text> icon={<IconTrash />}
</Group> label={"Удалить"}
</Menu.Item> />
<Menu.Item
onClick={e => {
e.stopPropagation();
onDelete(value);
}}>
<Group wrap={"nowrap"}>
<IconTrash />
<Text>Удалить</Text>
</Group>
</Menu.Item>
</Menu.Dropdown> </Menu.Dropdown>
</Menu> </Menu>
); );