2021-02-25 15:27:58 -08:00
|
|
|
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';
|
2021-03-02 11:46:55 -08:00
|
|
|
import { filtersToGraphqlParams } from './utils/filtersToGraphqlParams';
|
2021-02-25 15:27:58 -08:00
|
|
|
|
|
|
|
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,
|
2021-03-02 11:46:55 -08:00
|
|
|
filters: filtersToGraphqlParams(filters),
|
2021-02-25 15:27:58 -08:00
|
|
|
},
|
|
|
|
},
|
|
|
|
});
|
|
|
|
|
|
|
|
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}
|
2021-03-04 23:18:50 -08:00
|
|
|
pageSize={SearchCfg.RESULTS_PER_PAGE}
|
2021-02-25 15:27:58 -08:00
|
|
|
total={totalResults}
|
|
|
|
showLessItems
|
|
|
|
onChange={onChangePage}
|
|
|
|
/>
|
|
|
|
</Row>
|
|
|
|
</div>
|
|
|
|
);
|
|
|
|
};
|