Fix command menu keyboard input & refactor group (#1706)
* - fix command menu keyboard shortcuts - refactor command groups * - refactor the MenuItemCommand to use cmdk * - fixed matching commands multiple displays * - fixed array count problems react with boolean
This commit is contained in:
31
front/src/modules/command-menu/components/CommandGroup.tsx
Normal file
31
front/src/modules/command-menu/components/CommandGroup.tsx
Normal file
@ -0,0 +1,31 @@
|
|||||||
|
import React from 'react';
|
||||||
|
import styled from '@emotion/styled';
|
||||||
|
import { Command } from 'cmdk';
|
||||||
|
|
||||||
|
const StyledGroup = styled(Command.Group)`
|
||||||
|
[cmdk-group-heading] {
|
||||||
|
align-items: center;
|
||||||
|
color: ${({ theme }) => theme.font.color.light};
|
||||||
|
display: flex;
|
||||||
|
font-size: ${({ theme }) => theme.font.size.xs};
|
||||||
|
font-weight: ${({ theme }) => theme.font.weight.semiBold};
|
||||||
|
padding-bottom: ${({ theme }) => theme.spacing(2)};
|
||||||
|
padding-left: ${({ theme }) => theme.spacing(2)};
|
||||||
|
padding-right: ${({ theme }) => theme.spacing(1)};
|
||||||
|
padding-top: ${({ theme }) => theme.spacing(2)};
|
||||||
|
text-transform: uppercase;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
type OwnProps = {
|
||||||
|
heading: string;
|
||||||
|
children: React.ReactNode | React.ReactNode[];
|
||||||
|
};
|
||||||
|
|
||||||
|
export const CommandGroup = ({ heading, children }: OwnProps) => {
|
||||||
|
if (!children || !React.Children.count(children)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
return <StyledGroup heading={heading}>{children}</StyledGroup>;
|
||||||
|
};
|
||||||
@ -19,11 +19,11 @@ import { commandMenuCommandsState } from '../states/commandMenuCommandsState';
|
|||||||
import { isCommandMenuOpenedState } from '../states/isCommandMenuOpenedState';
|
import { isCommandMenuOpenedState } from '../states/isCommandMenuOpenedState';
|
||||||
import { Command, CommandType } from '../types/Command';
|
import { Command, CommandType } from '../types/Command';
|
||||||
|
|
||||||
|
import { CommandGroup } from './CommandGroup';
|
||||||
import { CommandMenuItem } from './CommandMenuItem';
|
import { CommandMenuItem } from './CommandMenuItem';
|
||||||
import {
|
import {
|
||||||
StyledDialog,
|
StyledDialog,
|
||||||
StyledEmpty,
|
StyledEmpty,
|
||||||
StyledGroup,
|
|
||||||
StyledInput,
|
StyledInput,
|
||||||
StyledList,
|
StyledList,
|
||||||
} from './CommandMenuStyles';
|
} from './CommandMenuStyles';
|
||||||
@ -130,14 +130,10 @@ export const CommandMenu = () => {
|
|||||||
onValueChange={setSearch}
|
onValueChange={setSearch}
|
||||||
/>
|
/>
|
||||||
<StyledList>
|
<StyledList>
|
||||||
{matchingCreateCommand.length < 1 &&
|
<StyledEmpty>No results found.</StyledEmpty>
|
||||||
matchingNavigateCommand.length < 1 &&
|
<CommandGroup heading="Create">
|
||||||
people.length < 1 &&
|
{matchingCreateCommand.length === 1 &&
|
||||||
companies.length < 1 &&
|
matchingCreateCommand.map((cmd) => (
|
||||||
activities.length < 1 && <StyledEmpty>No results found.</StyledEmpty>}
|
|
||||||
{matchingCreateCommand.length > 0 && (
|
|
||||||
<StyledGroup heading="Create">
|
|
||||||
{matchingCreateCommand.map((cmd) => (
|
|
||||||
<CommandMenuItem
|
<CommandMenuItem
|
||||||
to={cmd.to}
|
to={cmd.to}
|
||||||
key={cmd.label}
|
key={cmd.label}
|
||||||
@ -147,11 +143,10 @@ export const CommandMenu = () => {
|
|||||||
shortcuts={cmd.shortcuts || []}
|
shortcuts={cmd.shortcuts || []}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</StyledGroup>
|
</CommandGroup>
|
||||||
)}
|
<CommandGroup heading="Navigate">
|
||||||
{matchingNavigateCommand.length > 0 && (
|
{matchingNavigateCommand.length === 1 &&
|
||||||
<StyledGroup heading="Navigate">
|
matchingNavigateCommand.map((cmd) => (
|
||||||
{matchingNavigateCommand.map((cmd) => (
|
|
||||||
<CommandMenuItem
|
<CommandMenuItem
|
||||||
to={cmd.to}
|
to={cmd.to}
|
||||||
key={cmd.label}
|
key={cmd.label}
|
||||||
@ -160,57 +155,50 @@ export const CommandMenu = () => {
|
|||||||
shortcuts={cmd.shortcuts || []}
|
shortcuts={cmd.shortcuts || []}
|
||||||
/>
|
/>
|
||||||
))}
|
))}
|
||||||
</StyledGroup>
|
</CommandGroup>
|
||||||
)}
|
<CommandGroup heading="People">
|
||||||
{people.length > 0 && (
|
{people.map((person) => (
|
||||||
<StyledGroup heading="People">
|
<CommandMenuItem
|
||||||
{people.map((person) => (
|
key={person.id}
|
||||||
<CommandMenuItem
|
to={`person/${person.id}`}
|
||||||
key={person.id}
|
label={person.displayName}
|
||||||
to={`person/${person.id}`}
|
Icon={() => (
|
||||||
label={person.displayName}
|
<Avatar
|
||||||
Icon={() => (
|
type="rounded"
|
||||||
<Avatar
|
avatarUrl={null}
|
||||||
type="rounded"
|
colorId={person.id}
|
||||||
avatarUrl={null}
|
placeholder={person.displayName}
|
||||||
colorId={person.id}
|
/>
|
||||||
placeholder={person.displayName}
|
)}
|
||||||
/>
|
/>
|
||||||
)}
|
))}
|
||||||
/>
|
</CommandGroup>
|
||||||
))}
|
<CommandGroup heading="Companies">
|
||||||
</StyledGroup>
|
{companies.map((company) => (
|
||||||
)}
|
<CommandMenuItem
|
||||||
{companies.length > 0 && (
|
key={company.id}
|
||||||
<StyledGroup heading="Companies">
|
label={company.name}
|
||||||
{companies.map((company) => (
|
to={`companies/${company.id}`}
|
||||||
<CommandMenuItem
|
Icon={() => (
|
||||||
key={company.id}
|
<Avatar
|
||||||
label={company.name}
|
colorId={company.id}
|
||||||
to={`companies/${company.id}`}
|
placeholder={company.name}
|
||||||
Icon={() => (
|
avatarUrl={getLogoUrlFromDomainName(company.domainName)}
|
||||||
<Avatar
|
/>
|
||||||
colorId={company.id}
|
)}
|
||||||
placeholder={company.name}
|
/>
|
||||||
avatarUrl={getLogoUrlFromDomainName(company.domainName)}
|
))}
|
||||||
/>
|
</CommandGroup>
|
||||||
)}
|
<CommandGroup heading="Notes">
|
||||||
/>
|
{activities.map((activity) => (
|
||||||
))}
|
<CommandMenuItem
|
||||||
</StyledGroup>
|
Icon={IconNotes}
|
||||||
)}
|
key={activity.id}
|
||||||
{activities.length > 0 && (
|
label={activity.title ?? ''}
|
||||||
<StyledGroup heading="Notes">
|
onClick={() => openActivityRightDrawer(activity.id)}
|
||||||
{activities.map((activity) => (
|
/>
|
||||||
<CommandMenuItem
|
))}
|
||||||
Icon={IconNotes}
|
</CommandGroup>
|
||||||
key={activity.id}
|
|
||||||
label={activity.title ?? ''}
|
|
||||||
onClick={() => openActivityRightDrawer(activity.id)}
|
|
||||||
/>
|
|
||||||
))}
|
|
||||||
</StyledGroup>
|
|
||||||
)}
|
|
||||||
</StyledList>
|
</StyledList>
|
||||||
</StyledDialog>
|
</StyledDialog>
|
||||||
);
|
);
|
||||||
|
|||||||
@ -1,4 +1,3 @@
|
|||||||
import React from 'react';
|
|
||||||
import { useNavigate } from 'react-router-dom';
|
import { useNavigate } from 'react-router-dom';
|
||||||
|
|
||||||
import { IconArrowUpRight } from '@/ui/icon';
|
import { IconArrowUpRight } from '@/ui/icon';
|
||||||
|
|||||||
@ -37,48 +37,6 @@ export const StyledInput = styled(Command.Input)`
|
|||||||
}
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
export const StyledMenuItem = styled(Command.Item)`
|
|
||||||
align-items: center;
|
|
||||||
border-radius: ${({ theme }) => theme.border.radius.sm};
|
|
||||||
color: ${({ theme }) => theme.font.color.primary};
|
|
||||||
cursor: pointer;
|
|
||||||
display: flex;
|
|
||||||
font-size: ${({ theme }) => theme.font.size.md};
|
|
||||||
gap: ${({ theme }) => theme.spacing(3)};
|
|
||||||
height: 40px;
|
|
||||||
justify-content: space-between;
|
|
||||||
padding: 0 ${({ theme }) => theme.spacing(1)};
|
|
||||||
position: relative;
|
|
||||||
transition: all 150ms ease;
|
|
||||||
transition-property: none;
|
|
||||||
user-select: none;
|
|
||||||
&:hover {
|
|
||||||
background: ${({ theme }) => theme.background.transparent.light};
|
|
||||||
}
|
|
||||||
&[data-selected='true'] {
|
|
||||||
background: ${({ theme }) => theme.background.tertiary};
|
|
||||||
/* Could be nice to add a caret like this for better accessibility in the future
|
|
||||||
But it needs to be consistend with other picker dropdown (e.g. company)
|
|
||||||
&:after {
|
|
||||||
background: ${({ theme }) => theme.background.quaternary};
|
|
||||||
content: '';
|
|
||||||
height: 100%;
|
|
||||||
left: 0;
|
|
||||||
position: absolute;
|
|
||||||
width: 3px;
|
|
||||||
z-index: ${({ theme }) => theme.lastLayerZIndex};
|
|
||||||
} */
|
|
||||||
}
|
|
||||||
&[data-disabled='true'] {
|
|
||||||
color: ${({ theme }) => theme.font.color.light};
|
|
||||||
cursor: not-allowed;
|
|
||||||
}
|
|
||||||
svg {
|
|
||||||
height: 16px;
|
|
||||||
width: 16px;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const StyledList = styled(Command.List)`
|
export const StyledList = styled(Command.List)`
|
||||||
background: ${({ theme }) => theme.background.secondary};
|
background: ${({ theme }) => theme.background.secondary};
|
||||||
height: min(300px, var(--cmdk-list-height));
|
height: min(300px, var(--cmdk-list-height));
|
||||||
@ -89,22 +47,6 @@ export const StyledList = styled(Command.List)`
|
|||||||
transition-property: height;
|
transition-property: height;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
export const StyledGroup = styled(Command.Group)`
|
|
||||||
[cmdk-group-heading] {
|
|
||||||
align-items: center;
|
|
||||||
color: ${({ theme }) => theme.font.color.light};
|
|
||||||
display: flex;
|
|
||||||
font-size: ${({ theme }) => theme.font.size.xs};
|
|
||||||
font-weight: ${({ theme }) => theme.font.weight.semiBold};
|
|
||||||
padding-bottom: ${({ theme }) => theme.spacing(2)};
|
|
||||||
padding-left: ${({ theme }) => theme.spacing(2)};
|
|
||||||
padding-right: ${({ theme }) => theme.spacing(1)};
|
|
||||||
padding-top: ${({ theme }) => theme.spacing(2)};
|
|
||||||
text-transform: uppercase;
|
|
||||||
user-select: none;
|
|
||||||
}
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const StyledEmpty = styled(Command.Empty)`
|
export const StyledEmpty = styled(Command.Empty)`
|
||||||
align-items: center;
|
align-items: center;
|
||||||
color: ${({ theme }) => theme.font.color.light};
|
color: ${({ theme }) => theme.font.color.light};
|
||||||
@ -114,34 +56,3 @@ export const StyledEmpty = styled(Command.Empty)`
|
|||||||
justify-content: center;
|
justify-content: center;
|
||||||
white-space: pre-wrap;
|
white-space: pre-wrap;
|
||||||
`;
|
`;
|
||||||
|
|
||||||
export const StyledSeparator = styled(Command.Separator)``;
|
|
||||||
|
|
||||||
export const StyledIconAndLabelContainer = styled.div`
|
|
||||||
align-items: center;
|
|
||||||
display: flex;
|
|
||||||
gap: ${({ theme }) => theme.spacing(2)};
|
|
||||||
`;
|
|
||||||
export const StyledIconContainer = styled.div`
|
|
||||||
align-items: center;
|
|
||||||
background: ${({ theme }) => theme.background.transparent.light};
|
|
||||||
border-radius: ${({ theme }) => theme.border.radius.sm};
|
|
||||||
color: ${({ theme }) => theme.font.color.secondary};
|
|
||||||
display: flex;
|
|
||||||
padding: ${({ theme }) => theme.spacing(1)};
|
|
||||||
`;
|
|
||||||
export const StyledShortCut = styled.div`
|
|
||||||
background-color: ${({ theme }) => theme.background.transparent.light};
|
|
||||||
border-radius: ${({ theme }) => theme.border.radius.sm};
|
|
||||||
color: ${({ theme }) => theme.font.color.light};
|
|
||||||
margin-left: ${({ theme }) => theme.spacing(1)};
|
|
||||||
margin-right: ${({ theme }) => theme.spacing(1)};
|
|
||||||
padding: ${({ theme }) => theme.spacing(1)};
|
|
||||||
`;
|
|
||||||
|
|
||||||
export const StyledShortcutsContainer = styled.div`
|
|
||||||
align-items: center;
|
|
||||||
color: ${({ theme }) => theme.font.color.light};
|
|
||||||
display: flex;
|
|
||||||
font-size: ${({ theme }) => theme.font.size.sm};
|
|
||||||
`;
|
|
||||||
|
|||||||
@ -1,10 +1,10 @@
|
|||||||
import { useTheme } from '@emotion/react';
|
import { useTheme } from '@emotion/react';
|
||||||
import styled from '@emotion/styled';
|
import styled from '@emotion/styled';
|
||||||
|
import { Command } from 'cmdk';
|
||||||
|
|
||||||
import { IconComponent } from '@/ui/icon/types/IconComponent';
|
import { IconComponent } from '@/ui/icon/types/IconComponent';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
StyledMenuItemBase,
|
|
||||||
StyledMenuItemLabel,
|
StyledMenuItemLabel,
|
||||||
StyledMenuItemLeftContent,
|
StyledMenuItemLeftContent,
|
||||||
} from '../internals/components/StyledMenuItemBase';
|
} from '../internals/components/StyledMenuItemBase';
|
||||||
@ -32,8 +32,51 @@ const StyledCommandText = styled.div`
|
|||||||
padding-right: ${({ theme }) => theme.spacing(2)};
|
padding-right: ${({ theme }) => theme.spacing(2)};
|
||||||
`;
|
`;
|
||||||
|
|
||||||
const StyledMenuItemCommandContainer = styled(StyledMenuItemBase)`
|
const StyledMenuItemCommandContainer = styled(Command.Item)`
|
||||||
|
--horizontal-padding: ${({ theme }) => theme.spacing(1)};
|
||||||
|
--vertical-padding: ${({ theme }) => theme.spacing(2)};
|
||||||
|
align-items: center;
|
||||||
|
border-radius: ${({ theme }) => theme.border.radius.sm};
|
||||||
|
color: ${({ theme }) => theme.font.color.secondary};
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: row;
|
||||||
|
font-size: ${({ theme }) => theme.font.size.sm};
|
||||||
|
gap: ${({ theme }) => theme.spacing(2)};
|
||||||
|
height: calc(32px - 2 * var(--vertical-padding));
|
||||||
height: 24px;
|
height: 24px;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: var(--vertical-padding) var(--horizontal-padding);
|
||||||
|
position: relative;
|
||||||
|
transition: all 150ms ease;
|
||||||
|
transition-property: none;
|
||||||
|
user-select: none;
|
||||||
|
width: calc(100% - 2 * var(--horizontal-padding));
|
||||||
|
&:hover {
|
||||||
|
background: ${({ theme }) => theme.background.transparent.light};
|
||||||
|
}
|
||||||
|
&[data-selected='true'] {
|
||||||
|
background: ${({ theme }) => theme.background.tertiary};
|
||||||
|
/* Could be nice to add a caret like this for better accessibility in the future
|
||||||
|
But it needs to be consistend with other picker dropdown (e.g. company)
|
||||||
|
&:after {
|
||||||
|
background: ${({ theme }) => theme.background.quaternary};
|
||||||
|
content: '';
|
||||||
|
height: 100%;
|
||||||
|
left: 0;
|
||||||
|
position: absolute;
|
||||||
|
width: 3px;
|
||||||
|
z-index: ${({ theme }) => theme.lastLayerZIndex};
|
||||||
|
} */
|
||||||
|
}
|
||||||
|
&[data-disabled='true'] {
|
||||||
|
color: ${({ theme }) => theme.font.color.light};
|
||||||
|
cursor: not-allowed;
|
||||||
|
}
|
||||||
|
svg {
|
||||||
|
height: 16px;
|
||||||
|
width: 16px;
|
||||||
|
}
|
||||||
`;
|
`;
|
||||||
|
|
||||||
export type MenuItemProps = {
|
export type MenuItemProps = {
|
||||||
@ -54,7 +97,7 @@ export const MenuItemCommand = ({
|
|||||||
const theme = useTheme();
|
const theme = useTheme();
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<StyledMenuItemCommandContainer onClick={onClick} className={className}>
|
<StyledMenuItemCommandContainer onSelect={onClick} className={className}>
|
||||||
<StyledMenuItemLeftContent>
|
<StyledMenuItemLeftContent>
|
||||||
{LeftIcon && (
|
{LeftIcon && (
|
||||||
<StyledBigIconContainer>
|
<StyledBigIconContainer>
|
||||||
|
|||||||
Reference in New Issue
Block a user