/* * SonarQube * Copyright (C) 2009-2022 SonarSource SA * mailto:info AT sonarsource DOT com * * This program is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 3 of the License, or (at your option) any later version. * * This program is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public License * along with this program; if not, write to the Free Software Foundation, * Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ import classNames from 'classnames'; import { orderBy, sortBy, without } from 'lodash'; import * as React from 'react'; import Tooltip from '../../../components/controls/Tooltip'; import FacetBox from '../../../components/facet/FacetBox'; import FacetHeader from '../../../components/facet/FacetHeader'; import FacetItem from '../../../components/facet/FacetItem'; import FacetItemsList from '../../../components/facet/FacetItemsList'; import { translate } from '../../../helpers/l10n'; import { formatMeasure } from '../../../helpers/measures'; import { Dict } from '../../../types/types'; import { FacetKey } from '../query'; export interface BasicProps { onChange: (changes: Dict) => void; onToggle: (facet: FacetKey) => void; open: boolean; stats?: Dict; values: string[]; } interface Props extends BasicProps { children?: React.ReactNode; disabled?: boolean; disabledHelper?: string; halfWidth?: boolean; options?: string[]; property: FacetKey; renderFooter?: () => React.ReactNode; renderName?: (value: string) => React.ReactNode; renderTextName?: (value: string) => string; singleSelection?: boolean; } export default class Facet extends React.PureComponent { handleItemClick = (itemValue: string, multiple: boolean) => { const { values } = this.props; let newValue; if (this.props.singleSelection) { const value = values.length ? values[0] : undefined; newValue = itemValue === value ? undefined : itemValue; } else if (multiple) { newValue = orderBy( values.includes(itemValue) ? without(values, itemValue) : [...values, itemValue] ); } else { newValue = values.includes(itemValue) && values.length < 2 ? [] : [itemValue]; } this.props.onChange({ [this.props.property]: newValue }); }; handleHeaderClick = () => this.props.onToggle(this.props.property); handleClear = () => this.props.onChange({ [this.props.property]: [] }); getStat = (value: string) => this.props.stats && this.props.stats[value]; renderItem = (value: string) => { const active = this.props.values.includes(value); const stat = this.getStat(value); const { renderName = defaultRenderName, renderTextName = defaultRenderName } = this.props; return ( ); }; render() { const { disabled, renderTextName = defaultRenderName, stats } = this.props; const values = this.props.values.map(renderTextName); const items = this.props.options || (stats && sortBy( Object.keys(stats), key => -stats[key], key => renderTextName(key).toLowerCase() )); return ( {translate('coding_rules.facet', this.props.property)} } onClear={this.handleClear} onClick={disabled ? undefined : this.handleHeaderClick} open={this.props.open && !disabled} values={values}> {this.props.children} {this.props.open && items !== undefined && ( {items.map(this.renderItem)} )} {this.props.open && this.props.renderFooter !== undefined && this.props.renderFooter()} ); } } function defaultRenderName(value: string) { return value; }