import React, { useState, useEffect } from 'react'; import { css } from '@emotion/css'; import { uniqBy } from 'lodash'; // Types import { RichHistoryQuery, ExploreId } from 'app/types/explore'; // Utils import { stylesFactory, useTheme, RangeSlider, MultiSelect, Select, FilterInput } from '@grafana/ui'; import { GrafanaTheme, SelectableValue } from '@grafana/data'; import { SortOrder, mapNumbertoTimeInSlider, mapQueriesToHeadings, createDatasourcesList, filterAndSortQueries, } from 'app/core/utils/richHistory'; // Components import RichHistoryCard from './RichHistoryCard'; import { sortOrderOptions } from './RichHistory'; import { useDebounce } from 'react-use'; export interface Props { queries: RichHistoryQuery[]; sortOrder: SortOrder; activeDatasourceOnly: boolean; datasourceFilters: SelectableValue[]; retentionPeriod: number; exploreId: ExploreId; height: number; onChangeSortOrder: (sortOrder: SortOrder) => void; onSelectDatasourceFilters: (value: SelectableValue[]) => void; } const getStyles = stylesFactory((theme: GrafanaTheme, height: number) => { const bgColor = theme.isLight ? theme.palette.gray5 : theme.palette.dark4; /* 134px is based on the width of the Query history tabs bar, so the content is aligned to right side of the tab */ const cardWidth = '100% - 134px'; const sliderHeight = `${height - 180}px`; return { container: css` display: flex; .label-slider { font-size: ${theme.typography.size.sm}; &:last-of-type { margin-top: ${theme.spacing.lg}; } &:first-of-type { font-weight: ${theme.typography.weight.semibold}; margin-bottom: ${theme.spacing.md}; } } `, containerContent: css` width: calc(${cardWidth}); `, containerSlider: css` width: 129px; margin-right: ${theme.spacing.sm}; .slider { bottom: 10px; height: ${sliderHeight}; width: 129px; padding: ${theme.spacing.sm} 0; } `, slider: css` position: fixed; `, selectors: css` display: flex; justify-content: space-between; flex-wrap: wrap; `, filterInput: css` margin-bottom: ${theme.spacing.sm}; `, multiselect: css` width: 100%; margin-bottom: ${theme.spacing.sm}; .gf-form-select-box__multi-value { background-color: ${bgColor}; padding: ${theme.spacing.xxs} ${theme.spacing.xs} ${theme.spacing.xxs} ${theme.spacing.sm}; border-radius: ${theme.border.radius.sm}; } `, sort: css` width: 170px; `, sessionName: css` display: flex; align-items: flex-start; justify-content: flex-start; margin-top: ${theme.spacing.lg}; h4 { margin: 0 10px 0 0; } `, heading: css` font-size: ${theme.typography.heading.h4}; margin: ${theme.spacing.md} ${theme.spacing.xxs} ${theme.spacing.sm} ${theme.spacing.xxs}; `, footer: css` height: 60px; margin: ${theme.spacing.lg} auto; display: flex; justify-content: center; font-weight: ${theme.typography.weight.light}; font-size: ${theme.typography.size.sm}; a { font-weight: ${theme.typography.weight.semibold}; margin-left: ${theme.spacing.xxs}; } `, queries: css` font-size: ${theme.typography.size.sm}; font-weight: ${theme.typography.weight.regular}; margin-left: ${theme.spacing.xs}; `, }; }); export function RichHistoryQueriesTab(props: Props) { const { datasourceFilters, onSelectDatasourceFilters, queries, onChangeSortOrder, sortOrder, activeDatasourceOnly, retentionPeriod, exploreId, height, } = props; const [timeFilter, setTimeFilter] = useState<[number, number]>([0, retentionPeriod]); const [data, setData] = useState<[RichHistoryQuery[], ReturnType]>([[], []]); const [searchInput, setSearchInput] = useState(''); const [debouncedSearchInput, setDebouncedSearchInput] = useState(''); const theme = useTheme(); const styles = getStyles(theme, height); useDebounce( () => { setDebouncedSearchInput(searchInput); }, 300, [searchInput] ); useEffect(() => { const datasourcesRetrievedFromQueryHistory = uniqBy(queries, 'datasourceName').map((d) => d.datasourceName); const listOfDatasources = createDatasourcesList(datasourcesRetrievedFromQueryHistory); setData([ filterAndSortQueries( queries, sortOrder, datasourceFilters.map((d) => d.value), debouncedSearchInput, timeFilter ), listOfDatasources, ]); }, [timeFilter, queries, sortOrder, datasourceFilters, debouncedSearchInput]); const [filteredQueries, listOfDatasources] = data; /* mappedQueriesToHeadings is an object where query headings (stringified dates/data sources) * are keys and arrays with queries that belong to that headings are values. */ const mappedQueriesToHeadings = mapQueriesToHeadings(filteredQueries, sortOrder); return (
Filter history
{mapNumbertoTimeInSlider(timeFilter[0])}
number[]} />
{mapNumbertoTimeInSlider(timeFilter[1])}
{!activeDatasourceOnly && (
)}
{ setSearchInput(value); }} />