From 8a69ca2c2890839bc2c8fc8f0148d4f788a4e0dc Mon Sep 17 00:00:00 2001 From: Victor Lin <13424970+victorlin@users.noreply.github.com> Date: Wed, 18 Oct 2023 15:59:30 -0700 Subject: [PATCH] Add ability to show/hide panel options --- src/components/controls/panelChevron.tsx | 25 ++++++++++++ src/components/controls/panelHeader.tsx | 51 ++++++++++++++++++------ src/components/controls/panelSection.tsx | 13 +++++- src/components/controls/styles.js | 2 +- 4 files changed, 77 insertions(+), 14 deletions(-) create mode 100644 src/components/controls/panelChevron.tsx diff --git a/src/components/controls/panelChevron.tsx b/src/components/controls/panelChevron.tsx new file mode 100644 index 000000000..520a99fce --- /dev/null +++ b/src/components/controls/panelChevron.tsx @@ -0,0 +1,25 @@ +import React from "react"; +import styled from 'styled-components'; +import { FaChevronRight, FaChevronDown } from "react-icons/fa"; + +const Container = styled.span` + padding-right: 6px; + color: ${(props) => props.theme.color}; +` + +type Props = { + show: boolean +} + +/** + * An interactive chevron to show/hide a panel's options. + */ +export const PanelChevron = ({ show }: Props) => { + const icon = show ? : + + return ( + + {icon} + + ) +} diff --git a/src/components/controls/panelHeader.tsx b/src/components/controls/panelHeader.tsx index 20437f5fe..84e24dedf 100644 --- a/src/components/controls/panelHeader.tsx +++ b/src/components/controls/panelHeader.tsx @@ -4,6 +4,7 @@ import { togglePanelDisplay } from "../../actions/panelDisplay"; import { HeaderContainer } from "./styles"; import Toggle from "./toggle"; import { AnnotatedTitle, Title, Tooltip } from "./annotatedTitle"; +import { PanelChevron } from "./panelChevron"; /** Panel identifier used internally. */ export type PanelId = string; @@ -15,30 +16,56 @@ type Props = { /** Indicates panel visibility. */ panelIsVisible: boolean + + /** Indicates whether there are options for the panel. */ + hasOptions: boolean + + /** Indicates options visibility. */ + optionsAreVisible: boolean + + /** Update options visibility. */ + setOptionsAreVisible: React.Dispatch> } /** * A header used by all panel controls, containing an interactive title. */ -export const PanelHeader = ({ panel, title, tooltip, panelIsVisible }: Props) => { +export const PanelHeader = ({ panel, title, tooltip, panelIsVisible, hasOptions, optionsAreVisible, setOptionsAreVisible }: Props) => { const dispatch = useAppDispatch(); function togglePanelVisibility() { dispatch(togglePanelDisplay(panel)) } + function toggleOptionsVisibility() { + setOptionsAreVisible(!optionsAreVisible); + } + return ( - - - + + + {hasOptions && + } + + + event.stopPropagation()}> + + ); }; diff --git a/src/components/controls/panelSection.tsx b/src/components/controls/panelSection.tsx index 7dc6594b8..de22a67a9 100644 --- a/src/components/controls/panelSection.tsx +++ b/src/components/controls/panelSection.tsx @@ -23,6 +23,14 @@ export const PanelSection = ({ panel, title, tooltip, options=undefined }: Props const panelIsVisible = panelsToDisplay.includes(panel) + // Initially, panel visibility determines options visibility. + const [optionsAreVisible, setOptionsAreVisible] = React.useState(panelIsVisible); + + // Subsequent panel visibility updates also determines options visibility. + React.useEffect(() => { + setOptionsAreVisible(panelIsVisible) + }, [panelIsVisible]) + return ( - {panelIsVisible && options} + {optionsAreVisible && options} ); }; diff --git a/src/components/controls/styles.js b/src/components/controls/styles.js index f4bc723b4..30b57b5db 100644 --- a/src/components/controls/styles.js +++ b/src/components/controls/styles.js @@ -49,7 +49,7 @@ export const PanelSectionContainer = styled.div` `; export const TitleAndIconContainer = styled.span` - display: flex; + display: inline-flex; align-items: center; `;