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

chore: hide table rows while results are loading #6089

Closed
Closed
Show file tree
Hide file tree
Changes from 4 commits
Commits
Show all changes
32 commits
Select commit Hold shift + click to select a range
6711204
chore: move loading indicator into table, hide table contents while l…
jordanvn Nov 14, 2024
662eaea
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 14, 2024
9c6034d
place loading indicator outside of table, resize table based on contents
jordanvn Nov 15, 2024
e352bd6
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
1491905
chore: update styling to allow full vertical scrolling for table
jordanvn Nov 15, 2024
0fac109
chore(tests): update tests to fit new approach to loading
jordanvn Nov 15, 2024
4538b9c
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
04bca01
chore: move loading label to prop, return null instead of false, retu…
jordanvn Nov 15, 2024
4b9e70e
chore: removing duplicate data-testid
jordanvn Nov 15, 2024
56f4097
chore: moving data-table view out of table component and into data ta…
jordanvn Nov 15, 2024
fdd7a20
chore: move loading indicator element to data, process label at view …
jordanvn Nov 15, 2024
29b3617
fix(tests): update data in tests to include new changes
jordanvn Nov 15, 2024
bf3030e
chore: change loading indicator to react node, make isLoading optional
jordanvn Nov 15, 2024
b7e4da4
chore: pass in isLoading at view level, remove isLoading from tableData
jordanvn Nov 15, 2024
9cb48ef
chore: remove unnecessary usage of isLoading, destructure data
jordanvn Nov 15, 2024
82d996a
chore: use object shorthand
jordanvn Nov 15, 2024
24bd121
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
ac34eda
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
60887c5
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
a30db2f
chore: removing loading indicator component in favor of composable
jordanvn Nov 15, 2024
d4adb68
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
015a70a
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 15, 2024
3077998
chore: set isLoading to true on loading indicator composables
jordanvn Nov 16, 2024
d3cd630
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 16, 2024
93a796c
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 16, 2024
e79f718
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 16, 2024
013b42f
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 18, 2024
b878f2d
Merge branch 'feat-storage-browser/main' into feat-storage-browser/ta…
jordanvn Nov 19, 2024
e376108
chore: removing usage of loading indicator inside table
jordanvn Nov 19, 2024
dcd69aa
chore: removing unnecessary usage of loading indicator and loading in…
jordanvn Nov 19, 2024
2ebf0e6
chore: reverting now-unnecessary style changes
jordanvn Nov 19, 2024
95fcac5
chore: remove unnecessary inclusion of optional isLoading prop, fix u…
jordanvn Nov 19, 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
jordanvn marked this conversation as resolved.
Show resolved Hide resolved
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import React from 'react';

import { IconElement, SpanElement, ViewElement } from '../context/elements';
import { STORAGE_BROWSER_BLOCK } from '../constants';

export const LoadingIndicator = (): React.JSX.Element => (
<ViewElement className={`${STORAGE_BROWSER_BLOCK}__loading-indicator`}>
<IconElement
className={`${STORAGE_BROWSER_BLOCK}__loading-indicator-icon`}
variant="loading"
/>
<SpanElement
className={`${STORAGE_BROWSER_BLOCK}__loading-indicator-text`}
aria-live="polite"
>
Loading
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This needs to be provided as a prop so it can handled by display text

Copy link
Member Author

@jordanvn jordanvn Nov 15, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated to be provided as a prop, passing in existing DEFAULT_LIST_VIEW_DISPLAY_TEXT.loadingIndicatorLabel

</SpanElement>
</ViewElement>
);
Original file line number Diff line number Diff line change
@@ -0,0 +1,13 @@
import React from 'react';
import { render, screen } from '@testing-library/react';
import { LoadingIndicator } from '../LoadingIndicator';

describe('LoadingIndicator', () => {
it('renders', () => {
render(<LoadingIndicator />);

const indicator = screen.getByText('Loading');

expect(indicator).toBeInTheDocument();
});
});
Original file line number Diff line number Diff line change
Expand Up @@ -11,17 +11,22 @@ import { TextDataCell } from './dataCells/TextDataCell';
import { DataTableDataCell, DataTableHeader } from './types';
import { WithKey } from '../../components/types';
import { CheckboxHeader } from './headers/CheckboxHeader';
import { LoadingIndicator } from '../../components/LoadingIndicator';
import { STORAGE_BROWSER_BLOCK } from '../../constants';
import { ViewElement } from '../../context/elements';

export interface DataTableRow {
content: WithKey<DataTableDataCell>[];
}

export interface DataTableProps {
isLoading: boolean;
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I think it's ok to keep this optional and just treat undefined as falsy?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated

headers: WithKey<DataTableHeader>[];
rows: WithKey<DataTableRow>[];
}

export const DataTable = ({
isLoading,
headers,
rows,
}: DataTableProps): React.JSX.Element => {
Expand Down Expand Up @@ -49,44 +54,51 @@ export const DataTable = ({
}
});

const mappedRows = rows.map(({ key, content }) => ({
key,
content: content.map(({ key, content, type }) => {
switch (type) {
case 'button': {
return {
key,
content: <ButtonDataCell content={content} />,
};
}
case 'checkbox': {
return {
key,
content: <CheckboxDataCell content={content} />,
};
}
case 'date': {
return {
key,
content: <DateDataCell content={content} />,
};
}
case 'number': {
return {
key,
content: <NumberDataCell content={content} />,
};
}
case 'text':
default: {
return {
key,
content: <TextDataCell content={content} />,
};
}
}
}),
}));
Comment on lines -52 to -89
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

There is no change to this section, it is just wrapped in conditional logic

const mappedRows = !isLoading
? rows.map(({ key, content }) => ({
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Prefer early return of empty array for readability

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Agreed. Updated

key,
content: content.map(({ key, content, type }) => {
switch (type) {
case 'button': {
return {
key,
content: <ButtonDataCell content={content} />,
};
}
case 'checkbox': {
return {
key,
content: <CheckboxDataCell content={content} />,
};
}
case 'date': {
return {
key,
content: <DateDataCell content={content} />,
};
}
case 'number': {
return {
key,
content: <NumberDataCell content={content} />,
};
}
case 'text':
default: {
return {
key,
content: <TextDataCell content={content} />,
};
}
}
}),
}))
: [];

return <Table headers={mappedHeaders} rows={mappedRows} />;
return (
<ViewElement className={`${STORAGE_BROWSER_BLOCK}__table-wrapper`}>
<Table headers={mappedHeaders} rows={mappedRows} />
{!!isLoading && <LoadingIndicator />}
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Prefer not to return false in JSX statements

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated to return null instead

</ViewElement>
);
};
Original file line number Diff line number Diff line change
Expand Up @@ -154,6 +154,7 @@ export const useDataTable = (): DataTableProps => {

return {
headers: mappedHeaders ?? [],
isLoading: !!data.isDataRefreshDisabled,
Copy link
Member

@cshfang cshfang Nov 15, 2024

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Assuming this is made optional, wouldn't need the !!

But, more importantly, I think this can be driven off of isLoading?

  const { data } = useControlsContext();
  const { isLoading, loadingIndicator, tableData } = data;

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, removed after updating isLoading to be optional

rows: sortedRows ?? [],
};
};
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@ import { getDestinationPickerTableData } from './utils';

export interface FoldersTableProps {
folders?: FolderData[];
isProcessing?: boolean;
onSelect?: (value: string) => void;
}

Expand All @@ -18,10 +19,11 @@ export const { useFoldersTable, FoldersTableProvider } = createContextUtilities(
);

export const FoldersTableControl = (): React.JSX.Element => {
const { folders, onSelect } = useFoldersTable();
const { folders, isProcessing, onSelect } = useFoldersTable();

const tableData = getDestinationPickerTableData({
folders,
isLoading: !!isProcessing,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We shouldn't have to feed isLoading to the DataTable via tableData

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Updated

onSelect,
});

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -28,9 +28,11 @@ export const getDestinationListFullPrefix = (

export const getDestinationPickerTableData = ({
folders,
isLoading,
onSelect,
}: {
folders?: { key: string; id: string }[];
isLoading: boolean;
onSelect?: (name: string) => void;
}): DataTableProps => {
const rows: DataTableProps['rows'] = !folders
Expand Down Expand Up @@ -58,6 +60,7 @@ export const getDestinationPickerTableData = ({

const tableData: DataTableProps = {
headers: DESTINATION_PICKER_COLUMNS,
isLoading,
rows,
};
return tableData;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -196,5 +196,5 @@ export const getActionViewTableData = <T extends TaskData = TaskData>({
};
});

return { headers, rows };
return { headers, isLoading: isProcessing, rows };
};
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import { ViewElement } from '../../context/elements';
import { DataTableControl } from '../../controls/DataTableControl';
import { DataRefreshControl } from '../../controls/DataRefreshControl';
import { DropZoneControl } from '../../controls/DropZoneControl';
import { LoadingIndicatorControl } from '../../controls/LoadingIndicatorControl';
import { NavigationControl } from '../../controls/NavigationControl';
import { PaginationControl } from '../../controls/PaginationControl';
import { SearchControl } from '../../controls/SearchControl';
Expand Down Expand Up @@ -128,6 +127,7 @@ export function LocationDetailView({
location,
fileDataItems,
hasFiles,
isLoading,
pageItems,
selectFileLabel,
selectAllFilesLabel,
Expand Down Expand Up @@ -179,7 +179,6 @@ export function LocationDetailView({
<MessageControl />
</ViewElement>
<LocationDetailMessage show={hasError} message={message} />
<LoadingIndicatorControl />
{hasError ? null : (
<DropZoneControl>
<DataTableControl />
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ export const getLocationDetailViewTableData = ({
location,
fileDataItems,
hasFiles,
isLoading,
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should be able to remove this and use context as well here (and anywhere else with the same getXTableData pattern)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yep, this is right. Updated to remove here and like locations

pageItems,
selectFileLabel,
selectAllFilesLabel,
Expand All @@ -29,6 +30,7 @@ export const getLocationDetailViewTableData = ({
location: LocationState;
fileDataItems?: FileData[];
hasFiles: boolean;
isLoading: boolean;
pageItems: LocationItemData[];
selectFileLabel: string;
selectAllFilesLabel: string;
Expand Down Expand Up @@ -102,5 +104,5 @@ export const getLocationDetailViewTableData = ({
}
});

return { headers, rows };
return { headers, isLoading, rows };
};
Original file line number Diff line number Diff line change
Expand Up @@ -3,7 +3,6 @@ import React from 'react';
import { ViewElement } from '../../context/elements';
import { DataRefreshControl } from '../../controls/DataRefreshControl';
import { DataTableControl } from '../../controls/DataTableControl';
import { LoadingIndicatorControl } from '../../controls/LoadingIndicatorControl';
import { PaginationControl } from '../../controls/PaginationControl';
import { SearchControl } from '../../controls/SearchControl';
import { TitleControl } from '../../controls/TitleControl';
Expand Down Expand Up @@ -137,6 +136,7 @@ export function LocationsView({
tableData: getLocationsViewTableData({
getPermissionName,
headers,
isLoading,
pageItems,
onDownload,
onNavigate,
Expand Down Expand Up @@ -180,7 +180,6 @@ export function LocationsView({
/>
</ViewElement>
<LocationsMessage show={hasError} message={message} />
<LoadingIndicatorControl />
{hasError ? null : <DataTableControl />}
<LocationsEmptyMessage show={shouldShowEmptyMessage} />
</div>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,12 +7,14 @@ export const getLocationsViewTableData = ({
onNavigate,
onDownload,
headers,
isLoading,
getDownloadLabel,
getPermissionName,
}: {
pageItems: LocationData[];
onNavigate: (location: LocationData) => void;
headers: LocationViewHeaders;
isLoading: boolean;
onDownload: (location: LocationData) => void;
getDownloadLabel: (fileName: string) => string;
getPermissionName: (permissions: LocationPermissions) => string;
Expand Down Expand Up @@ -79,5 +81,5 @@ export const getLocationsViewTableData = ({
};
});

return { headers, rows };
return { headers, isLoading, rows };
};
Original file line number Diff line number Diff line change
Expand Up @@ -57,7 +57,16 @@
overflow: hidden;
position: relative;
width: 100%;
display: block;
}

&__table-wrapper {
display: grid;
grid-template-rows: auto 1fr;
height: 100%;
}

&__loading-indicator-icon {
animation: var(--storage-browser-loading-animation);
}

&__data-table {
Expand All @@ -67,8 +76,6 @@
border-style: solid;
border-color: var(--amplify-colors-border-secondary);
width: 100%;
height: 100%;
display: block;
}

&__table {
Expand Down
Loading