add actions, reordering base interactions

This commit is contained in:
41666 2021-07-06 09:59:04 -05:00
parent b0c8b2378b
commit 2d589b988f
26 changed files with 383 additions and 427 deletions

View file

@ -0,0 +1,42 @@
import { transitions } from '@roleypoly/design-system/atoms/timings';
import { CategoryContainer } from '@roleypoly/design-system/organisms/role-picker/RolePicker.styled';
import styled, { css } from 'styled-components';
export const CategoryActions = styled.div<{ right?: boolean }>`
display: flex;
flex-direction: row;
${(props) =>
props.right &&
css`
justify-content: flex-end;
`}
& > * {
${(props) => (props.right ? 'margin-left' : 'margin-right')}: 5px;
}
`;
export const ReorderButton = styled.div`
height: 40px;
width: 40px;
font-size: 20px;
margin-right: 5px;
display: flex;
align-items: center;
justify-content: center;
border-radius: 50%;
cursor: pointer;
transition: background-color ${transitions.actionable}s ease-in-out;
&:hover {
background-color: rgba(0, 0, 0, 0.15);
}
`;
export const ReorderCategoryContainer = styled(CategoryContainer)`
display: flex;
align-items: center;
justify-content: flex-start;
user-select: none;
`;

View file

@ -0,0 +1,158 @@
import { BreakpointText } from '@roleypoly/design-system/atoms/breakpoints';
import { Button } from '@roleypoly/design-system/atoms/button';
import { FaderOpacity } from '@roleypoly/design-system/atoms/fader';
import { LargeText } from '@roleypoly/design-system/atoms/typography';
import { EditorCategory } from '@roleypoly/design-system/molecules/editor-category';
import { CategoryContainer } from '@roleypoly/design-system/organisms/role-picker/RolePicker.styled';
import { Category, CategoryType, PresentableGuild, Role } from '@roleypoly/types';
import KSUID from 'ksuid';
import { sortBy } from 'lodash';
import React from 'react';
import { CgReorder } from 'react-icons/cg';
import { GoArrowDown, GoArrowUp, GoCheck, GoGrabber, GoPlus } from 'react-icons/go';
import {
CategoryActions,
ReorderButton,
ReorderCategoryContainer,
} from './ServerCategoryEditor.styled';
type Props = {
guild: PresentableGuild;
onChange: (categories: PresentableGuild['data']['categories']) => void;
};
const resetOrder = (categories: Category[]) =>
sortBy(categories, ['position', 'id']).map((c, index) => ({ ...c, position: index }));
const forceOrder = (categories: Category[]) =>
categories.map((c, index) => ({ ...c, position: index }));
export const ServerCategoryEditor = (props: Props) => {
const [reorderMode, setReorderMode] = React.useState(false);
const updateSingleCategory = (category: Category) => {
const newCategories = props.guild.data.categories.map((c) => {
if (c.id === category.id) {
return category;
}
return c;
});
props.onChange(newCategories);
};
const createCategory = () => {
// Reset order now that we're creating a new category
const categories = resetOrder(props.guild.data.categories);
const newCategory: Category = {
id: KSUID.randomSync().toString(),
name: 'New Category',
type: CategoryType.Multi,
position: categories.length,
roles: [],
hidden: false,
};
props.onChange([...categories, newCategory]);
};
const onReorder = (categories: Category[]) => {
setReorderMode(false);
props.onChange(resetOrder(categories));
};
if (reorderMode) {
return <ReorderMode {...props} exitReorderMode={onReorder} />;
}
return (
<div>
<CategoryActions>
<Button color="muted" size="small" onClick={() => createCategory()}>
Create New <GoPlus />
</Button>
<Button color="muted" size="small" onClick={() => setReorderMode(true)}>
Change Order <CgReorder />
</Button>
</CategoryActions>
{sortBy(props.guild.data.categories, ['position', 'id']).map((category, idx) => (
<CategoryContainer key={idx}>
<EditorCategory
key={idx}
category={category}
title={category.name}
selectedRoles={[]}
roles={
category.roles
.map((role) => props.guild.roles.find((r) => r.id === role))
.filter((r) => r !== undefined) as Role[]
}
onChange={updateSingleCategory}
type={category.type === CategoryType.Single ? 'single' : 'multi'}
/>
</CategoryContainer>
))}
</div>
);
};
const ReorderMode = (props: Props & { exitReorderMode: (final: Category[]) => void }) => {
const [categories, setCategories] = React.useState(props.guild.data.categories);
React.useEffect(() => {
setCategories(props.guild.data.categories);
}, [props.guild.data.categories]);
const handleReorder = (category: Category, direction: 'up' | 'down') => () => {
const newCategories = [...categories];
const index = newCategories.findIndex((c) => c.id === category.id);
const newIndex = direction === 'up' ? index - 1 : index + 1;
if (newIndex < 0 || newIndex > newCategories.length - 1) {
return;
}
newCategories.splice(index, 1);
newCategories.splice(newIndex, 0, category);
setCategories(forceOrder(newCategories));
};
return (
<div>
<CategoryActions right>
<Button
color="muted"
size="small"
onClick={() => props.exitReorderMode(props.guild.data.categories)}
>
Reset
</Button>
<Button
color="primary"
size="small"
onClick={() => props.exitReorderMode(categories)}
>
<BreakpointText small="Save" large="Save Order" /> <GoCheck />
</Button>
</CategoryActions>
{sortBy(categories, ['position', 'id']).map((category, idx, array) => (
<ReorderCategoryContainer key={idx}>
<ReorderButton data-tip="Drag to reorder" style={{ cursor: 'grab' }}>
<GoGrabber />
</ReorderButton>
<FaderOpacity isVisible={idx !== 0}>
<ReorderButton onClick={handleReorder(category, 'up')} data-tip="Move up">
<GoArrowUp />
</ReorderButton>
</FaderOpacity>
<FaderOpacity isVisible={categories.length - 1 !== idx}>
<ReorderButton onClick={handleReorder(category, 'down')} data-tip="Move down">
<GoArrowDown />
</ReorderButton>
</FaderOpacity>
<LargeText>{category.name}</LargeText>
</ReorderCategoryContainer>
))}
</div>
);
};

View file

@ -0,0 +1 @@
export * from './ServerCategoryEditor';