|
14 | 14 | * limitations under the License.
|
15 | 15 | */
|
16 | 16 |
|
17 |
| -/* eslint-disable eqeqeq */ |
18 |
| -import { ALL_RESOURCE_KIND_FILTER } from '@Shared/constants' |
| 17 | +import { ChangeEvent, useEffect, useMemo, useState } from 'react' |
| 18 | + |
19 | 19 | import { ReactComponent as ICCaretDown } from '@Icons/ic-caret-down.svg'
|
20 | 20 | import { PopupMenu, StyledRadioGroup as RadioGroup } from '../../../Common'
|
21 |
| -import { NodeFilters, NodeStatus, StatusFilterButtonType } from './types' |
| 21 | +import { StatusFilterButtonType } from './types' |
| 22 | + |
22 | 23 | import './StatusFilterButtonComponent.scss'
|
| 24 | +import { getNodesCount, getStatusFilters } from './utils' |
23 | 25 |
|
24 |
| -export const StatusFilterButtonComponent = ({ nodes, selectedTab, handleFilterClick }: StatusFilterButtonType) => { |
25 |
| - const maxInlineFilterCount = 4 |
26 |
| - let allNodeCount: number = 0 |
27 |
| - let healthyNodeCount: number = 0 |
28 |
| - let progressingNodeCount: number = 0 |
29 |
| - let failedNodeCount: number = 0 |
30 |
| - let missingNodeCount: number = 0 |
31 |
| - let driftedNodeCount: number = 0 |
| 26 | +export const StatusFilterButtonComponent = ({ |
| 27 | + nodes, |
| 28 | + selectedTab, |
| 29 | + handleFilterClick, |
| 30 | + maxInlineFiltersCount = 0, |
| 31 | +}: StatusFilterButtonType) => { |
| 32 | + // STATES |
| 33 | + const [overflowFilterIndex, setOverflowFilterIndex] = useState(0) |
32 | 34 |
|
33 |
| - nodes?.forEach((_node) => { |
34 |
| - const _nodeHealth = _node.health?.status |
| 35 | + // STATUS FILTERS |
| 36 | + const { allResourceKindFilter, statusFilters } = useMemo(() => getStatusFilters(getNodesCount(nodes)), [nodes]) |
35 | 37 |
|
36 |
| - if (_node.hasDrift) { |
37 |
| - driftedNodeCount += 1 |
38 |
| - } |
| 38 | + useEffect(() => { |
| 39 | + const filterIndex = statusFilters.findIndex(({ status }) => status === selectedTab) |
| 40 | + setOverflowFilterIndex(Math.max(filterIndex, 0)) |
| 41 | + }, [statusFilters]) |
39 | 42 |
|
40 |
| - if (_nodeHealth?.toLowerCase() === NodeStatus.Healthy) { |
41 |
| - healthyNodeCount += 1 |
42 |
| - } else if (_nodeHealth?.toLowerCase() === NodeStatus.Degraded) { |
43 |
| - failedNodeCount += 1 |
44 |
| - } else if (_nodeHealth?.toLowerCase() === NodeStatus.Progressing) { |
45 |
| - progressingNodeCount += 1 |
46 |
| - } else if (_nodeHealth?.toLowerCase() === NodeStatus.Missing) { |
47 |
| - missingNodeCount += 1 |
| 43 | + const showOverflowFilters = maxInlineFiltersCount > 0 && statusFilters.length > maxInlineFiltersCount |
| 44 | + |
| 45 | + const inlineFilters = useMemo(() => { |
| 46 | + if (showOverflowFilters) { |
| 47 | + const min = Math.max(0, Math.min(overflowFilterIndex - 1, statusFilters.length - maxInlineFiltersCount)) |
| 48 | + const max = Math.min(min + maxInlineFiltersCount, statusFilters.length) |
| 49 | + |
| 50 | + return statusFilters.slice(min, max) |
48 | 51 | }
|
49 |
| - allNodeCount += 1 |
50 |
| - }) |
51 | 52 |
|
52 |
| - const handleInlineFilterClick = (e) => { |
53 |
| - handleFilterClick(e.target.value) |
54 |
| - } |
| 53 | + return statusFilters |
| 54 | + }, [statusFilters.length, overflowFilterIndex, maxInlineFiltersCount]) |
55 | 55 |
|
56 |
| - const filterOptions = [ |
57 |
| - { status: ALL_RESOURCE_KIND_FILTER, count: allNodeCount, isSelected: selectedTab == ALL_RESOURCE_KIND_FILTER }, |
58 |
| - { status: NodeStatus.Missing, count: missingNodeCount, isSelected: NodeStatus.Missing == selectedTab }, |
59 |
| - { status: NodeStatus.Degraded, count: failedNodeCount, isSelected: NodeStatus.Degraded == selectedTab }, |
60 |
| - { |
61 |
| - status: NodeStatus.Progressing, |
62 |
| - count: progressingNodeCount, |
63 |
| - isSelected: NodeStatus.Progressing == selectedTab, |
64 |
| - }, |
65 |
| - { status: NodeStatus.Healthy, count: healthyNodeCount, isSelected: NodeStatus.Healthy == selectedTab }, |
66 |
| - window._env_.FEATURE_CONFIG_DRIFT_ENABLE && { |
67 |
| - status: NodeFilters.drifted, |
68 |
| - count: driftedNodeCount, |
69 |
| - isSelected: selectedTab === NodeFilters.drifted, |
70 |
| - }, |
71 |
| - ] |
72 |
| - const validFilterOptions = filterOptions.filter(({ count }) => count > 0) |
73 |
| - const displayedInlineFilters = validFilterOptions.slice( |
74 |
| - 0, |
75 |
| - Math.min(maxInlineFilterCount, validFilterOptions.length), |
76 |
| - ) |
77 |
| - const overflowFilters = |
78 |
| - validFilterOptions.length > maxInlineFilterCount ? validFilterOptions.slice(maxInlineFilterCount) : null |
| 56 | + const handleInlineFilterClick = (e: ChangeEvent<HTMLInputElement>) => { |
| 57 | + const { value } = e.target |
| 58 | + if (value === allResourceKindFilter.status) { |
| 59 | + setOverflowFilterIndex(0) |
| 60 | + } |
| 61 | + if (selectedTab !== value) { |
| 62 | + handleFilterClick(value) |
| 63 | + } |
| 64 | + } |
79 | 65 |
|
80 |
| - const renderOverflowFilters = () => |
81 |
| - overflowFilters ? ( |
82 |
| - <PopupMenu autoClose> |
83 |
| - <PopupMenu.Button |
84 |
| - isKebab |
85 |
| - rootClassName="flex p-4 dc__border dc__no-left-radius dc__right-radius-4 bg__primary dc__hover-n50" |
86 |
| - > |
87 |
| - <ICCaretDown className="icon-dim-14 scn-6" /> |
88 |
| - </PopupMenu.Button> |
89 |
| - <PopupMenu.Body rootClassName="w-150 py-4 mt-4" style={{ left: '136px' }}> |
90 |
| - {overflowFilters.map((filter) => ( |
91 |
| - <button |
92 |
| - key={filter.status} |
93 |
| - type="button" |
94 |
| - className={`dc__transparent w-100 py-6 px-8 flex left dc__gap-8 fs-13 lh-20 fw-4 cn-9 ${filter.isSelected ? 'bcb-1' : 'bg__primary dc__hover-n50'}`} |
95 |
| - onClick={() => handleFilterClick(filter.status)} |
96 |
| - > |
97 |
| - <span |
98 |
| - className={`dc__app-summary__icon icon-dim-16 ${filter.status} ${filter.status}--node`} |
99 |
| - style={{ zIndex: 'unset' }} |
100 |
| - /> |
101 |
| - <span className="dc__first-letter-capitalize flex-grow-1 text-left">{filter.status}</span> |
102 |
| - <span>{filter.count}</span> |
103 |
| - </button> |
104 |
| - ))} |
105 |
| - </PopupMenu.Body> |
106 |
| - </PopupMenu> |
107 |
| - ) : null |
| 66 | + const handleOverflowFilterClick = (status: string, index: number) => () => { |
| 67 | + if (selectedTab !== status) { |
| 68 | + setOverflowFilterIndex(index) |
| 69 | + handleFilterClick(status) |
| 70 | + } |
| 71 | + } |
108 | 72 |
|
109 | 73 | return (
|
110 |
| - <> |
| 74 | + <div className="flexbox"> |
111 | 75 | <RadioGroup
|
112 |
| - className={`gui-yaml-switch status-filter-button ${overflowFilters ? 'with-menu-button' : ''}`} |
| 76 | + className={`gui-yaml-switch status-filter-button ${showOverflowFilters ? 'with-menu-button' : ''}`} |
113 | 77 | name="status-filter-button"
|
114 | 78 | initialTab={selectedTab}
|
115 | 79 | disabled={false}
|
116 | 80 | onChange={handleInlineFilterClick}
|
117 | 81 | >
|
118 |
| - {displayedInlineFilters.map((filter, index) => ( |
| 82 | + <RadioGroup.Radio |
| 83 | + key={allResourceKindFilter.status} |
| 84 | + value={allResourceKindFilter.status} |
| 85 | + tippyPlacement="top" |
| 86 | + tippyContent={allResourceKindFilter.status} |
| 87 | + tippyClass="w-100 dc__first-letter-capitalize" |
| 88 | + > |
| 89 | + <span className="dc__first-letter-capitalize">{`${allResourceKindFilter.status} (${allResourceKindFilter.count})`}</span> |
| 90 | + </RadioGroup.Radio> |
| 91 | + {inlineFilters.map(({ status, count }) => ( |
119 | 92 | <RadioGroup.Radio
|
120 |
| - key={filter.status} |
121 |
| - value={filter.status} |
122 |
| - showTippy={index !== 0} |
| 93 | + key={status} |
| 94 | + value={status} |
| 95 | + showTippy |
123 | 96 | tippyPlacement="top"
|
124 |
| - tippyContent={filter.status} |
| 97 | + tippyContent={status} |
125 | 98 | tippyClass="w-100 dc__first-letter-capitalize"
|
126 | 99 | >
|
127 |
| - {index !== 0 ? ( |
128 |
| - <> |
129 |
| - <span |
130 |
| - className={`dc__app-summary__icon icon-dim-16 ${filter.status} ${filter.status}--node`} |
131 |
| - style={{ zIndex: 'unset' }} |
132 |
| - /> |
133 |
| - <span>{filter.count}</span> |
134 |
| - </> |
135 |
| - ) : ( |
136 |
| - <span className="dc__first-letter-capitalize">{`${filter.status} (${filter.count})`}</span> |
137 |
| - )} |
| 100 | + <span |
| 101 | + className={`dc__app-summary__icon icon-dim-16 ${status} ${status}--node`} |
| 102 | + style={{ zIndex: 'unset' }} |
| 103 | + /> |
| 104 | + <span>{count}</span> |
138 | 105 | </RadioGroup.Radio>
|
139 | 106 | ))}
|
140 | 107 | </RadioGroup>
|
141 |
| - {renderOverflowFilters()} |
142 |
| - </> |
| 108 | + {showOverflowFilters && ( |
| 109 | + <PopupMenu autoClose> |
| 110 | + <PopupMenu.Button |
| 111 | + isKebab |
| 112 | + rootClassName="flex p-4 dc__border dc__no-left-radius dc__right-radius-4 bg__primary dc__hover-n50" |
| 113 | + > |
| 114 | + <ICCaretDown className="icon-dim-14 scn-6" /> |
| 115 | + </PopupMenu.Button> |
| 116 | + <PopupMenu.Body rootClassName="w-150 py-4 mt-4" style={{ left: '136px' }}> |
| 117 | + {statusFilters.map(({ status, count }, index) => ( |
| 118 | + <button |
| 119 | + key={status} |
| 120 | + type="button" |
| 121 | + className={`dc__transparent w-100 py-6 px-8 flex left dc__gap-8 fs-13 lh-20 fw-4 cn-9 dc__hover-n50 ${selectedTab === status ? 'bcb-1' : ''}`} |
| 122 | + onClick={handleOverflowFilterClick(status, index)} |
| 123 | + > |
| 124 | + <span |
| 125 | + className={`dc__app-summary__icon icon-dim-16 ${status} ${status}--node`} |
| 126 | + style={{ zIndex: 'unset' }} |
| 127 | + /> |
| 128 | + <span className="dc__first-letter-capitalize flex-grow-1 text-left">{status}</span> |
| 129 | + <span>{count}</span> |
| 130 | + </button> |
| 131 | + ))} |
| 132 | + </PopupMenu.Body> |
| 133 | + </PopupMenu> |
| 134 | + )} |
| 135 | + </div> |
143 | 136 | )
|
144 | 137 | }
|
0 commit comments