feat: statuses colors
This commit is contained in:
@ -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>
|
||||||
);
|
);
|
||||||
|
|||||||
@ -1,4 +0,0 @@
|
|||||||
|
|
||||||
.header {
|
|
||||||
border-bottom: solid dodgerblue 3px;
|
|
||||||
}
|
|
||||||
@ -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}
|
||||||
/>
|
/>
|
||||||
|
|||||||
@ -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>
|
||||||
|
|||||||
@ -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>
|
||||||
);
|
);
|
||||||
|
|||||||
@ -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>
|
||||||
|
|||||||
50
src/app/deals/modals/ColorPickerModal/ColorPickerModal.tsx
Normal file
50
src/app/deals/modals/ColorPickerModal/ColorPickerModal.tsx
Normal 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;
|
||||||
16
src/app/deals/utils/statusColors.ts
Normal file
16
src/app/deals/utils/statusColors.ts
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
const statusColors = [
|
||||||
|
"#228be6",
|
||||||
|
"#15aabf",
|
||||||
|
"#12b886",
|
||||||
|
"#40c057",
|
||||||
|
"#82c91e",
|
||||||
|
"#fab005",
|
||||||
|
"#fd7e14",
|
||||||
|
"#fa5252",
|
||||||
|
"#e64980",
|
||||||
|
"#be4bdb",
|
||||||
|
"#7950f2",
|
||||||
|
"#4c6ef5",
|
||||||
|
];
|
||||||
|
|
||||||
|
export default statusColors;
|
||||||
@ -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>
|
||||||
);
|
);
|
||||||
|
|||||||
26
src/components/ui/DropdownMenuItem/DropdownMenuItem.tsx
Normal file
26
src/components/ui/DropdownMenuItem/DropdownMenuItem.tsx
Normal 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;
|
||||||
@ -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) => ({
|
||||||
|
|||||||
@ -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;
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -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()])),
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|||||||
@ -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,
|
||||||
};
|
};
|
||||||
|
|||||||
@ -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>
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user