diff --git a/react/src/components/ImageList.tsx b/react/src/components/ImageList.tsx index da0e27efc9..8d26a3859e 100644 --- a/react/src/components/ImageList.tsx +++ b/react/src/components/ImageList.tsx @@ -11,6 +11,7 @@ import { BaseImageTags, ConstraintTags, LangTags } from './ImageTags'; import ManageAppsModal from './ManageAppsModal'; import ManageImageResourceLimitModal from './ManageImageResourceLimitModal'; import ResourceNumber from './ResourceNumber'; +import TableColumnsSettingModal from './TableColumnsSettingModal'; import TextHighlighter from './TextHighlighter'; import { ImageListQuery, @@ -24,8 +25,9 @@ import { SettingOutlined, VerticalAlignBottomOutlined, } from '@ant-design/icons'; +import { useLocalStorageState } from 'ahooks'; import { App, Button, Input, Table, Tag, theme, Typography } from 'antd'; -import { ColumnsType } from 'antd/es/table'; +import { ColumnsType, ColumnType } from 'antd/es/table'; import graphql from 'babel-plugin-relay/macro'; import _ from 'lodash'; import { Key, useMemo, useState, useTransition } from 'react'; @@ -61,6 +63,7 @@ const ImageList: React.FC<{ style?: React.CSSProperties }> = ({ style }) => { const [installingImages, setInstallingImages] = useState([]); const { message } = App.useApp(); const [imageSearch, setImageSearch] = useState(''); + const [isOpenColumnsSetting, setIsOpenColumnsSetting] = useState(false); const [isPendingRefreshTransition, startRefreshTransition] = useTransition(); const [isPendingSearchTransition, startSearchTransition] = useTransition(); const baiClient = useSuspendedBackendaiClient(); @@ -310,7 +313,9 @@ const ImageList: React.FC<{ style?: React.CSSProperties }> = ({ style }) => { sorter: (a, b) => a?.digest && b?.digest ? a.digest.localeCompare(b.digest) : 0, render: (text, row) => ( - {row.digest} + + {row.digest} + ), }, { @@ -374,6 +379,13 @@ const ImageList: React.FC<{ style?: React.CSSProperties }> = ({ style }) => { }, ]; + const [displayedColumnKeys, setDisplayedColumnKeys] = useLocalStorageState( + 'backendaiwebui.EnvironmentPage.displayedColumnKeys', + { + defaultValue: columns.map((column) => _.toString(column.key)), + }, + ); + const imageFilterValues = useMemo(() => { return defaultSortedImages?.map((image) => { return { @@ -505,7 +517,11 @@ const ImageList: React.FC<{ style?: React.CSSProperties }> = ({ style }) => { style: { marginRight: token.marginXS }, }} dataSource={filterNonNullItems(filteredImageData)} - columns={columns} + columns={ + columns.filter((column) => + displayedColumnKeys?.includes(_.toString(column.key)), + ) as ColumnType[] + } loading={isPendingSearchTransition} rowSelection={{ type: 'checkbox', @@ -531,6 +547,20 @@ const ImageList: React.FC<{ style?: React.CSSProperties }> = ({ style }) => { showSorterTooltip={false} sortDirections={['descend', 'ascend', 'descend']} /> + +