1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
|
/*
* SonarQube
* Copyright (C) 2009-2024 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 { IconFile, IconFileCode } from '@sonarsource/echoes-react';
import { without } from 'lodash';
import { FacetBox, FacetItem } from '~design-system';
import { SOURCE_SCOPES } from '../../../helpers/constants';
import { translate, translateWithParameters } from '../../../helpers/l10n';
import { Dict } from '../../../types/types';
import { Query, formatFacetStat } from '../utils';
import { FacetItemsList } from './FacetItemsList';
import { MultipleSelectionHint } from './MultipleSelectionHint';
export interface ScopeFacetProps {
fetching: boolean;
onChange: (changes: Partial<Query>) => void;
onToggle: (property: string) => void;
open: boolean;
scopes: string[];
stats: Dict<number> | undefined;
}
export function ScopeFacet(props: ScopeFacetProps) {
const { fetching, open, scopes = [], stats = {} } = props;
const nbSelectableItems = SOURCE_SCOPES.filter(({ scope }) => stats[scope]).length;
const nbSelectedItems = scopes.length;
const property = 'scopes';
const headerId = `facet_${property}`;
return (
<FacetBox
className="it__search-navigator-facet-box it__search-navigator-facet-header"
clearIconLabel={translate('clear')}
count={nbSelectedItems}
countLabel={translateWithParameters('x_selected', nbSelectedItems)}
data-property={property}
id={headerId}
loading={fetching}
name={translate('issues.facet.scopes')}
onClear={() => props.onChange({ scopes: [] })}
onClick={() => props.onToggle('scopes')}
open={open}
>
<>
<FacetItemsList labelledby={headerId}>
{SOURCE_SCOPES.map(({ scope }) => {
const active = scopes.includes(scope);
const stat = stats[scope];
return (
<FacetItem
active={active}
className="it__search-navigator-facet"
icon={
{
MAIN: <IconFile className="sw-mr-1" />,
TEST: <IconFileCode className="sw-mr-1" />,
}[scope]
}
key={scope}
name={translate('issue.scope', scope)}
onClick={(itemValue: string, multiple: boolean) => {
if (multiple) {
props.onChange({
scopes: active ? without(scopes, itemValue) : [...scopes, itemValue],
});
} else {
props.onChange({
scopes: active && scopes.length === 1 ? [] : [itemValue],
});
}
}}
stat={formatFacetStat(stat) ?? 0}
value={scope}
/>
);
})}
</FacetItemsList>
<MultipleSelectionHint
nbSelectableItems={nbSelectableItems}
nbSelectedItems={nbSelectedItems}
/>
</>
</FacetBox>
);
}
|