datahub/datahub-web-react/src/app/search/EntitySearchResults.tsx

147 lines
5.5 KiB
TypeScript
Raw Normal View History

import React, { useState, useEffect } from 'react';
import { FilterOutlined } from '@ant-design/icons';
import { Alert, Button, Card, Divider, List, Modal, Pagination, Row, Typography } from 'antd';
import { SearchCfg } from '../../conf';
import { useGetSearchResultsQuery } from '../../graphql/search.generated';
import { EntityType, FacetFilterInput } from '../../types.generated';
import { IconStyleType } from '../entity/Entity';
import { Message } from '../shared/Message';
import { useEntityRegistry } from '../useEntityRegistry';
import { SearchFilters } from './SearchFilters';
import { filtersToGraphqlParams } from './utils/filtersToGraphqlParams';
const styles = {
loading: { marginTop: '10%' },
addFilters: { backgroundColor: '#F5F5F5' },
resultSummary: { color: 'gray', marginTop: '36px' },
resultHeaderCardBody: { padding: '16px 24px' },
resultHeaderCard: { right: '52px', top: '-40px', position: 'absolute' },
resultList: { width: '100%', borderColor: '#f0f0f0', marginTop: '12px', padding: '16px 32px' },
paginationRow: { padding: 40 },
resultsContainer: { width: '100%', padding: '20px 132px' },
};
interface Props {
type: EntityType;
query: string;
page: number;
filters: Array<FacetFilterInput>;
onChangeFilters: (filters: Array<FacetFilterInput>) => void;
onChangePage: (page: number) => void;
}
export const EntitySearchResults = ({ type, query, page, filters, onChangeFilters, onChangePage }: Props) => {
const [isEditingFilters, setIsEditingFilters] = useState(false);
const [selectedFilters, setSelectedFilters] = useState(filters);
useEffect(() => {
setSelectedFilters(filters);
}, [filters]);
const entityRegistry = useEntityRegistry();
const { loading, error, data } = useGetSearchResultsQuery({
variables: {
input: {
type,
query,
start: (page - 1) * SearchCfg.RESULTS_PER_PAGE,
count: SearchCfg.RESULTS_PER_PAGE,
filters: filtersToGraphqlParams(filters),
},
},
});
const results = data?.search?.entities || [];
const pageStart = data?.search?.start || 0;
const pageSize = data?.search?.count || 0;
const totalResults = data?.search?.total || 0;
const lastResultIndex =
pageStart * pageSize + pageSize > totalResults ? totalResults : pageStart * pageSize + pageSize;
const onFilterSelect = (selected: boolean, field: string, value: string) => {
const newFilters = selected
? [...selectedFilters, { field, value }]
: selectedFilters.filter((filter) => filter.field !== field || filter.value !== value);
setSelectedFilters(newFilters);
};
const onEditFilters = () => {
setIsEditingFilters(true);
};
const onApplyFilters = () => {
onChangeFilters(selectedFilters);
setIsEditingFilters(false);
};
const onCloseEditFilters = () => {
setIsEditingFilters(false);
setSelectedFilters(filters);
};
if (error || (!loading && !error && !data)) {
return <Alert type="error" message={error?.message || 'Entity failed to load'} />;
}
return (
<div style={styles.resultsContainer}>
{loading && <Message type="loading" content="Loading..." style={styles.loading} />}
<Button style={styles.addFilters} onClick={onEditFilters} data-testid="filters-button">
<FilterOutlined />
Filters{' '}
{filters.length > 0 && (
<>
{' '}
(<b>{filters.length}</b>)
</>
)}
</Button>
<Modal
title="Filters"
footer={<Button onClick={onApplyFilters}>Apply</Button>}
visible={isEditingFilters}
destroyOnClose
onCancel={onCloseEditFilters}
>
<SearchFilters
facets={data?.search?.facets || []}
selectedFilters={selectedFilters}
onFilterSelect={onFilterSelect}
/>
</Modal>
<Typography.Paragraph style={styles.resultSummary}>
Showing{' '}
<b>
{(page - 1) * pageSize} - {lastResultIndex}
</b>{' '}
of <b>{totalResults}</b> results
</Typography.Paragraph>
<List
header={
<Card bodyStyle={styles.resultHeaderCardBody} style={styles.resultHeaderCard as any}>
{entityRegistry.getIcon(type, 36, IconStyleType.ACCENT)}
</Card>
}
style={styles.resultList}
dataSource={results}
split={false}
renderItem={(item, index) => (
<>
<List.Item>{entityRegistry.renderSearchResult(type, item)}</List.Item>
{index < results.length - 1 && <Divider />}
</>
)}
bordered
/>
<Row justify="center" style={styles.paginationRow}>
<Pagination
current={page}
pageSize={pageSize}
total={totalResults}
showLessItems
onChange={onChangePage}
/>
</Row>
</div>
);
};