From ce64ae9d6bfb127c9e74ca3932a4554c87cb2ece Mon Sep 17 00:00:00 2001 From: Pianist038801 Date: Mon, 25 Oct 2021 11:39:34 -0400 Subject: [PATCH 1/3] feat: add comparison tool to execution list V1 Signed-off-by: Pianist038801 --- src/components/Entities/EntityDetails.tsx | 16 +++++++- src/components/Entities/EntityExecutions.tsx | 18 ++++++++- .../Entities/EntityExecutionsBarChart.tsx | 8 +++- .../Executions/ExecutionFilters.tsx | 24 ++++++++++- .../filters/useSingleFilterState.ts | 2 +- src/components/common/BarChart.tsx | 40 +++++++++++++++---- src/components/hooks/useChartState.ts | 27 +++++++++++++ 7 files changed, 121 insertions(+), 14 deletions(-) create mode 100644 src/components/hooks/useChartState.ts diff --git a/src/components/Entities/EntityDetails.tsx b/src/components/Entities/EntityDetails.tsx index d464827c0..d8cb199e2 100644 --- a/src/components/Entities/EntityDetails.tsx +++ b/src/components/Entities/EntityDetails.tsx @@ -4,6 +4,7 @@ import { contentMarginGridUnits } from 'common/layout'; import { WaitForData } from 'components/common/WaitForData'; import { EntityDescription } from 'components/Entities/EntityDescription'; import { useProject } from 'components/hooks/useProjects'; +import { useChartState } from 'components/hooks/useChartState'; import { LaunchForm } from 'components/Launch/LaunchForm/LaunchForm'; import { ResourceIdentifier, ResourceType } from 'models/Common/types'; import * as React from 'react'; @@ -82,6 +83,7 @@ export const EntityDetails: React.FC = ({ const [showLaunchForm, setShowLaunchForm] = React.useState(false); const onLaunch = () => setShowLaunchForm(true); const onCancelLaunch = () => setShowLaunchForm(false); + const { chartIds, onToggle, clearCharts } = useChartState(); return ( @@ -120,10 +122,20 @@ export const EntityDetails: React.FC = ({ ) : null} - {!versionView && } + {!versionView && ( + + )} {sections.executions && !versionView ? (
- +
) : null} {sections.launch ? ( diff --git a/src/components/Entities/EntityExecutions.tsx b/src/components/Entities/EntityExecutions.tsx index 77a52a048..d4012388b 100644 --- a/src/components/Entities/EntityExecutions.tsx +++ b/src/components/Entities/EntityExecutions.tsx @@ -25,10 +25,16 @@ const useStyles = makeStyles((theme: Theme) => ({ export interface EntityExecutionsProps { id: ResourceIdentifier; + chartIds: string[]; + clearCharts: () => void; } /** The tab/page content for viewing a workflow's executions */ -export const EntityExecutions: React.FC = ({ id }) => { +export const EntityExecutions: React.FC = ({ + id, + chartIds, + clearCharts +}) => { const { domain, project, resourceType } = id; const styles = useStyles(); const filtersState = useWorkflowExecutionFiltersState(); @@ -50,6 +56,10 @@ export const EntityExecutions: React.FC = ({ id }) => { } ); + if (chartIds.length > 0) + executions.value = executions.value.filter(item => + chartIds.includes(item.id.name) + ); /** Don't render component until finish fetching user profile */ if (filtersState.filters[4].status !== 'LOADED') { return null; @@ -61,7 +71,11 @@ export const EntityExecutions: React.FC = ({ id }) => { All Executions in the Workflow
- +
({ export interface EntityExecutionsBarChartProps { id: ResourceIdentifier; + onToggle: (id: string) => void; + chartIds: string[]; } const getExecutionTimeData = (executions: Execution[], fillSize = 100) => { @@ -88,7 +90,9 @@ const getStartExecutionTime = (executions: Execution[]) => { * @constructor */ export const EntityExecutionsBarChart: React.FC = ({ - id + id, + onToggle, + chartIds }) => { const styles = useStyles(); const { domain, project, resourceType } = id; @@ -116,6 +120,7 @@ export const EntityExecutionsBarChart: React.FC = const handleClickItem = React.useCallback(item => { if (item.metadata) { + onToggle(item.metadata.name); // const executionId = item.metadata as WorkflowExecutionIdentifier; // history.push(Routes.ExecutionDetails.makeUrl(executionId)); } @@ -133,6 +138,7 @@ export const EntityExecutionsBarChart: React.FC =
= ({ filter }) => { */ export const ExecutionFilters: React.FC<{ filters: (FilterState | BooleanFilterState)[]; -}> = ({ filters }) => { + chartIds: string[]; + clearCharts: () => void; +}> = ({ filters, chartIds, clearCharts }) => { const styles = useStyles(); + filters = filters.map(filter => { + const onChangeFunc = filter.onChange; + filter.onChange = value => { + clearCharts(); + onChangeFunc(value); + }; + return filter; + }); + return (
{filters.map((filter: any) => { @@ -102,6 +113,17 @@ export const ExecutionFilters: React.FC<{ /> ); })} + {chartIds.length > 0 && ( + <>} + className={styles.filterButton} + buttonText="Clear Manually Selected Executions" + onReset={clearCharts} + key="charts" + /> + )}
); }; diff --git a/src/components/Executions/filters/useSingleFilterState.ts b/src/components/Executions/filters/useSingleFilterState.ts index be1ab1d33..ca2255839 100644 --- a/src/components/Executions/filters/useSingleFilterState.ts +++ b/src/components/Executions/filters/useSingleFilterState.ts @@ -61,7 +61,7 @@ export function useSingleFilterState({ useEffect(() => { const { value } = selectedOption; - const queryValue = value === defaultValue.value ? undefined : value; + const queryValue = value; setQueryStateValue(queryStateKey, queryValue); }, [selectedOption, queryStateKey]); diff --git a/src/components/common/BarChart.tsx b/src/components/common/BarChart.tsx index e1cfc3c87..b4dde32e7 100644 --- a/src/components/common/BarChart.tsx +++ b/src/components/common/BarChart.tsx @@ -27,7 +27,8 @@ const useStyles = makeStyles((theme: Theme) => ({ item: { flex: 1, display: 'flex', - alignItems: 'flex-end', + flexDirection: 'column', + alignItems: 'center', '&:last-child': { marginRight: 0 } @@ -37,7 +38,16 @@ const useStyles = makeStyles((theme: Theme) => ({ flex: 1, marginRight: theme.spacing(0.25), minHeight: theme.spacing(0.75), - cursor: 'pointer' + cursor: 'pointer', + width: '80%', + marginLeft: '10%' + }, + circle: { + borderRadius: '50%', + backgroundColor: 'purple', + height: 6, + marginTop: 3, + width: 6 } })); @@ -50,12 +60,14 @@ interface BarChartData { interface BarChartItemProps extends BarChartData { onClick?: () => void; + isSelected: boolean; } interface BarChartProps { data: BarChartData[]; startDate?: string; onClickItem?: (item: any) => void; + chartIds: string[]; } /** @@ -67,6 +79,7 @@ interface BarChartProps { const BarChartItem: React.FC = ({ value, color, + isSelected, tooltip, onClick }) => { @@ -74,11 +87,19 @@ const BarChartItem: React.FC = ({ const [position, setPosition] = React.useState({ x: 0, y: 0 }); const content = ( -
+ <> +
+
+ ); return ( @@ -121,6 +142,7 @@ const BarChartItem: React.FC = ({ * @constructor */ export const BarChart: React.FC = ({ + chartIds, data, startDate, onClickItem @@ -154,6 +176,10 @@ export const BarChart: React.FC = ({ tooltip={item.tooltip} onClick={handleClickItem(item)} key={`bar-chart-item-${index}`} + isSelected={ + item.metadata && + chartIds.includes(item.metadata.name) + } /> ))}
diff --git a/src/components/hooks/useChartState.ts b/src/components/hooks/useChartState.ts new file mode 100644 index 000000000..c59b29366 --- /dev/null +++ b/src/components/hooks/useChartState.ts @@ -0,0 +1,27 @@ +import { useState } from 'react'; + +export function useChartState() { + const [chartIds, setChartIds] = useState([]); + const onToggle = (id: string) => { + setChartIds(curIds => { + const newChartIds = [...curIds]; + if (newChartIds.includes(id)) { + const index = newChartIds.indexOf(id); + newChartIds.splice(index, 1); + } else { + newChartIds.push(id); + } + return newChartIds; + }); + }; + + const clearCharts = () => { + setChartIds([]); + }; + + return { + onToggle, + clearCharts, + chartIds + }; +} From be5ee7417f3789ff4550f55c94fc1762c904d14d Mon Sep 17 00:00:00 2001 From: Pianist038801 Date: Tue, 26 Oct 2021 12:07:12 -0400 Subject: [PATCH 2/3] fix: fix failing tests Signed-off-by: Pianist038801 --- src/components/Executions/ExecutionFilters.tsx | 10 +++++----- src/components/Executions/filters/types.ts | 1 + 2 files changed, 6 insertions(+), 5 deletions(-) diff --git a/src/components/Executions/ExecutionFilters.tsx b/src/components/Executions/ExecutionFilters.tsx index caf1e4d48..76cedd177 100644 --- a/src/components/Executions/ExecutionFilters.tsx +++ b/src/components/Executions/ExecutionFilters.tsx @@ -60,16 +60,16 @@ const RenderFilter: React.FC<{ filter: FilterState }> = ({ filter }) => { */ export const ExecutionFilters: React.FC<{ filters: (FilterState | BooleanFilterState)[]; - chartIds: string[]; - clearCharts: () => void; + chartIds?: string[]; + clearCharts?: () => void; }> = ({ filters, chartIds, clearCharts }) => { const styles = useStyles(); filters = filters.map(filter => { const onChangeFunc = filter.onChange; filter.onChange = value => { - clearCharts(); - onChangeFunc(value); + if (clearCharts) clearCharts(); + if (onChangeFunc) onChangeFunc(value); }; return filter; }); @@ -113,7 +113,7 @@ export const ExecutionFilters: React.FC<{ /> ); })} - {chartIds.length > 0 && ( + {chartIds && chartIds.length > 0 && ( FilterOperation[]; onReset: () => void; + onChange?: (value) => void; } export interface SingleFilterState From 85a0fcfbb23db56cc12fe637b16c208fadc6d5d1 Mon Sep 17 00:00:00 2001 From: Pianist038801 Date: Tue, 26 Oct 2021 12:58:32 -0400 Subject: [PATCH 3/3] feat: use opacity instead of dot Signed-off-by: Pianist038801 --- src/components/common/BarChart.tsx | 23 ++++++++--------------- 1 file changed, 8 insertions(+), 15 deletions(-) diff --git a/src/components/common/BarChart.tsx b/src/components/common/BarChart.tsx index b4dde32e7..e7cc956c4 100644 --- a/src/components/common/BarChart.tsx +++ b/src/components/common/BarChart.tsx @@ -41,13 +41,6 @@ const useStyles = makeStyles((theme: Theme) => ({ cursor: 'pointer', width: '80%', marginLeft: '10%' - }, - circle: { - borderRadius: '50%', - backgroundColor: 'purple', - height: 6, - marginTop: 3, - width: 6 } })); @@ -90,14 +83,12 @@ const BarChartItem: React.FC = ({ <>
-
); @@ -177,8 +168,10 @@ export const BarChart: React.FC = ({ onClick={handleClickItem(item)} key={`bar-chart-item-${index}`} isSelected={ - item.metadata && - chartIds.includes(item.metadata.name) + chartIds.length === 0 + ? true + : item.metadata && + chartIds.includes(item.metadata.name) } /> ))}