Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

upcoming: [M3-7580] - Add list view for Linode Clone/Backup #10182

Merged
Merged
Show file tree
Hide file tree
Changes from 25 commits
Commits
Show all changes
43 commits
Select commit Hold shift + click to select a range
7fb84a2
Add Clone Linode power-off notice
hkhalil-akamai Jan 17, 2024
193f986
Add new feature flag for Linode Clone UI updates
hkhalil-akamai Jan 17, 2024
649ae71
Added changeset: Clone Linode power-off notice
hkhalil-akamai Jan 17, 2024
bc2d297
Reduce spacing between notices
hkhalil-akamai Jan 18, 2024
49f5c11
Refactored DebouncedSearchTextField to enable external state management
hkhalil-akamai Jan 19, 2024
aed317c
Add search to SelectLinodePanel and autopopulate when cloning
hkhalil-akamai Jan 19, 2024
4ed06f9
Fix bug causing filter value to update every time a Linode is selected
hkhalil-akamai Jan 19, 2024
3d4a6f2
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Jan 19, 2024
c2bfb1f
Make clear icon blue on hover
hkhalil-akamai Jan 19, 2024
e3fe3ef
Gate changes behind feature flag
hkhalil-akamai Jan 19, 2024
fbd5106
Added changeset: Search filter in Clone Linode and Create Linode from…
hkhalil-akamai Jan 19, 2024
e63b7de
Added -> upcoming
hkhalil-akamai Jan 23, 2024
5acd91a
Add explanatory comment
hkhalil-akamai Jan 23, 2024
58b2cd3
Change InputAdornment to IconButton
hkhalil-akamai Jan 23, 2024
80beada
Add missing useEffect dependencies
hkhalil-akamai Jan 23, 2024
d4e80b9
Fix unused customValue prop
hkhalil-akamai Jan 23, 2024
d176d08
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Jan 24, 2024
c0afbeb
Simplify export
hkhalil-akamai Jan 24, 2024
305c1c5
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Jan 26, 2024
e7786af
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Jan 30, 2024
2284d00
Add table view to Clone and Backups flow
hkhalil-akamai Feb 12, 2024
c0396b4
Add unit test for SelectLinodeRow
hkhalil-akamai Feb 13, 2024
5563ee8
Switch to useOrder hook
hkhalil-akamai Feb 13, 2024
624b774
Gate table view behind feature flag
hkhalil-akamai Feb 13, 2024
1108f51
Added changeset: List view for Linode Clone and Create from Backup
hkhalil-akamai Feb 13, 2024
e2262ab
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Feb 14, 2024
61c8af0
No clear icon when search field empty
hkhalil-akamai Feb 14, 2024
06acfc3
Extract render functions to components
hkhalil-akamai Feb 14, 2024
91e5c03
linodeID -> linodeId
hkhalil-akamai Feb 14, 2024
112218c
Feedback
hkhalil-akamai Feb 14, 2024
581c37f
Remove unnecessary ExtendedLinodes interface and utility
hkhalil-akamai Feb 15, 2024
828fd93
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Feb 16, 2024
6bb9202
Rename feature flag to camelcase
hkhalil-akamai Feb 21, 2024
1b305d4
Add unit tests for SelectLinodePanel
hkhalil-akamai Feb 21, 2024
e01bc6b
Only show power off action for selected linodes
hkhalil-akamai Feb 21, 2024
66ff449
Feedback @bnussman-akamai
hkhalil-akamai Feb 21, 2024
b2730af
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Feb 21, 2024
ea6d1ad
Bulleted notices
hkhalil-akamai Feb 21, 2024
9625654
Don't show power actions for from backup
hkhalil-akamai Feb 21, 2024
a93733e
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Feb 23, 2024
d3787b2
UX feedback: remove "Notice:" heading
hkhalil-akamai Feb 26, 2024
663824a
Fix table column rendering inconsistencies
hkhalil-akamai Feb 26, 2024
f13a059
Merge remote-tracking branch 'upstream/develop' into M3-7580-linode-c…
hkhalil-akamai Feb 26, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,5 @@
---
"@linode/manager": Upcoming Features
---

List view for Linode Clone and Create from Backup ([#10182](https://github.com/linode/manager/pull/10182))
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,8 @@ import { StyledActionButton } from 'src/components/Button/StyledActionButton';
interface InlineMenuActionProps {
/** Required action text */
actionText: string;
/** Optional height when displayed as a button */
buttonHeight?: number;
/** Optional class names */
className?: string;
/** Optional disabled */
Expand All @@ -27,6 +29,7 @@ interface InlineMenuActionProps {
export const InlineMenuAction = (props: InlineMenuActionProps) => {
const {
actionText,
buttonHeight,
className,
disabled,
href,
Expand All @@ -53,6 +56,7 @@ export const InlineMenuAction = (props: InlineMenuActionProps) => {
disabled={disabled}
loading={loading}
onClick={onClick}
sx={buttonHeight != undefined ? { height: `${buttonHeight}px` } : {}}
abailly-akamai marked this conversation as resolved.
Show resolved Hide resolved
tooltipAnalyticsEvent={tooltipAnalyticsEvent}
tooltipText={tooltip}
{...rest}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,19 +1,28 @@
import { Linode } from '@linode/api-v4/lib/linodes';
import { useMediaQuery } from '@mui/material';
import Grid from '@mui/material/Unstable_Grid2';
import { styled, useTheme } from '@mui/material/styles';
import * as React from 'react';

import { Box } from 'src/components/Box';
import { DebouncedSearchTextField } from 'src/components/DebouncedSearchTextField';
import { Notice } from 'src/components/Notice/Notice';
import { OrderByProps, sortData } from 'src/components/OrderBy';
import Paginate from 'src/components/Paginate';
import { PaginationFooter } from 'src/components/PaginationFooter/PaginationFooter';
import { Paper } from 'src/components/Paper';
import { RenderGuard } from 'src/components/RenderGuard';
import { SelectionCard } from 'src/components/SelectionCard/SelectionCard';
import { Stack } from 'src/components/Stack';
import { Table } from 'src/components/Table';
import { TableBody } from 'src/components/TableBody';
import { TableHead } from 'src/components/TableHead';
import { TableRowEmpty } from 'src/components/TableRowEmpty/TableRowEmpty';
import { Typography } from 'src/components/Typography';
import { useFlags } from 'src/hooks/useFlags';
import { useOrder } from 'src/hooks/useOrder';

import { PowerActionsDialog } from '../PowerActionsDialogOrDrawer';
import { SelectLinodeRow, SelectLinodeTableRowHead } from './SelectLinodeRow';

export interface ExtendedLinode extends Linode {
heading: string;
Expand All @@ -35,7 +44,7 @@ interface Props {
selectedLinodeID?: number;
}

const SelectLinodePanel = (props: Props) => {
export const SelectLinodePanel = (props: Props) => {
const {
disabled,
error,
Expand All @@ -46,13 +55,25 @@ const SelectLinodePanel = (props: Props) => {
selectedLinodeID,
} = props;

const flags = useFlags();
const { handleOrderChange, order, orderBy } = useOrder(
{ order: 'asc', orderBy: 'label' },
'create-select-linode'
);

const orderedLinodes = sortData<ExtendedLinode>(orderBy, order)(linodes);

const flags = useFlags();
const theme = useTheme();
const matchesMdUp = useMediaQuery(theme.breakpoints.up('md'));

const [userSearchText, setUserSearchText] = React.useState<
string | undefined
>(undefined);

const [powerOffLinode, setPowerOffLinode] = React.useState<
{ linodeID: number } | false
>(false);

// Capture the selected linode when this component mounts,
// so it doesn't change when the user selects a different one.
const [preselectedLinodeID] = React.useState(
Expand All @@ -70,14 +91,133 @@ const SelectLinodePanel = (props: Props) => {

const filteredLinodes = React.useMemo(
() =>
linodes.filter((linode) =>
orderedLinodes.filter((linode) =>
linode.label.toLowerCase().includes(searchText.toLowerCase())
),
[linodes, searchText]
[orderedLinodes, searchText]
);

return (
<>
<Paginate data={filteredLinodes}>
{({
count,
data: linodesData,
handlePageChange,
handlePageSizeChange,
page,
pageSize,
}) => {
return (
<>
<StyledPaper data-qa-select-linode-panel>
<Stack gap={0.5} mb={2}>
{error && (
<Notice
spacingBottom={0}
spacingTop={0}
text={error}
variant="error"
/>
)}
{notices &&
!disabled &&
notices.map((notice, i) => (
<Notice
key={i}
spacingBottom={0}
spacingTop={0}
text={notice.text}
variant={notice.level}
/>
))}
</Stack>
<Typography
marginBottom={
flags.linodeCloneUIChanges ? theme.spacing(2) : undefined
}
data-qa-select-linode-header
variant="h2"
>
{!!header ? header : 'Select Linode'}
</Typography>
{flags.linodeCloneUIChanges && (
<DebouncedSearchTextField
customValue={{
onChange: setUserSearchText,
value: searchText,
}}
sx={{
marginBottom: theme.spacing(1),
width: '330px',
}}
clearable
debounceTime={0}
abailly-akamai marked this conversation as resolved.
Show resolved Hide resolved
expand={true}
hideLabel
label=""
placeholder="Search"
/>
)}
<StyledBox>
{(matchesMdUp && flags.linodeCloneUIChanges
? renderTable
: renderCards)({
disabled: disabled ?? false,
handlePowerOff: (linodeID) =>
setPowerOffLinode({ linodeID }),
handleSelection,
orderBy: {
data: linodesData,
handleOrderChange,
order,
orderBy,
},
selectedLinodeID,
})}
</StyledBox>
</StyledPaper>
<PaginationFooter
count={count}
eventCategory={'Clone from existing panel'}
handlePageChange={handlePageChange}
handleSizeChange={handlePageSizeChange}
page={page}
pageSize={pageSize}
/>
</>
);
}}
</Paginate>
{powerOffLinode && (
<PowerActionsDialog
action={'Power Off'}
isOpen={!!powerOffLinode}
linodeId={powerOffLinode.linodeID}
manuallyUpdateConfigs={true}
onClose={() => setPowerOffLinode(false)}
/>
)}
</>
);
};

interface RenderLinodeProps {
disabled: boolean;
handlePowerOff: (linodeID: number) => void;
handleSelection: Props['handleSelection'];
orderBy: OrderByProps<ExtendedLinode>;
selectedLinodeID: number | undefined;
}

const renderCard = (linode: ExtendedLinode) => {
return (
const renderCards = ({
disabled,
handleSelection,
orderBy: { data: linodes },
selectedLinodeID,
}: RenderLinodeProps) => (
<Grid container spacing={2}>
{linodes.map((linode) => (
<SelectionCard
onClick={() => {
handleSelection(linode.id, linode.type, linode.specs.disk);
Expand All @@ -88,92 +228,41 @@ const SelectLinodePanel = (props: Props) => {
key={`selection-card-${linode.id}`}
subheadings={linode.subHeadings}
/>
);
};
))}
</Grid>
);

return (
<Paginate data={filteredLinodes}>
{({
count,
data: linodesData,
handlePageChange,
handlePageSizeChange,
page,
pageSize,
}) => {
return (
<>
<StyledPaper data-qa-select-linode-panel>
<Stack gap={0.5} mb={2}>
{error && (
<Notice
spacingBottom={0}
spacingTop={0}
text={error}
variant="error"
/>
)}
{notices &&
!disabled &&
notices.map((notice, i) => (
<Notice
key={i}
spacingBottom={0}
spacingTop={0}
text={notice.text}
variant={notice.level}
/>
))}
</Stack>
<Typography
marginBottom={
flags.linodeCloneUIChanges ? theme.spacing(2) : undefined
}
data-qa-select-linode-header
variant="h2"
>
{!!header ? header : 'Select Linode'}
</Typography>
{flags.linodeCloneUIChanges && (
<DebouncedSearchTextField
customValue={{
onChange: setUserSearchText,
value: searchText,
}}
sx={{
marginBottom: theme.spacing(1),
width: '330px',
}}
clearable
debounceTime={0}
expand={true}
hideLabel
label=""
placeholder="Search"
/>
)}
<StyledBox>
<Grid container spacing={2}>
{linodesData.map((linode) => {
return renderCard(linode);
})}
</Grid>
</StyledBox>
</StyledPaper>
<PaginationFooter
count={count}
eventCategory={'Clone from existing panel'}
handlePageChange={handlePageChange}
handleSizeChange={handlePageSizeChange}
page={page}
pageSize={pageSize}
/>
</>
);
}}
</Paginate>
);
};
const renderTable = ({
disabled,
handlePowerOff,
handleSelection,
orderBy,
selectedLinodeID,
}: RenderLinodeProps) => (
<Table aria-label="Linode" size="small">
<TableHead style={{ fontSize: '.875rem' }}>
<SelectLinodeTableRowHead orderBy={orderBy} />
</TableHead>
<TableBody role="radiogroup">
{orderBy.data.length > 0 ? (
orderBy.data.map((linode) => (
<SelectLinodeRow
handleSelection={() =>
handleSelection(linode.id, linode.type, linode.specs.disk)
}
disabled={disabled}
handlePowerOff={() => handlePowerOff(linode.id)}
key={linode.id}
linodeId={linode.id}
selected={Number(selectedLinodeID) === linode.id}
/>
))
) : (
<TableRowEmpty colSpan={6} message={'No results'} />
)}
</TableBody>
</Table>
);
hkhalil-akamai marked this conversation as resolved.
Show resolved Hide resolved

const StyledBox = styled(Box, {
label: 'StyledBox',
Expand All @@ -186,5 +275,3 @@ const StyledPaper = styled(Paper, { label: 'StyledPaper' })(({ theme }) => ({
flexGrow: 1,
width: '100%',
}));

export default RenderGuard(SelectLinodePanel);
Loading
Loading