From a01587ee0612ff9bb40b4decc1c651b291dc68e6 Mon Sep 17 00:00:00 2001 From: HendrikThePendric Date: Thu, 12 Dec 2024 17:28:51 +0100 Subject: [PATCH] feat: show all filters in a popover when clicking a button with total filter count --- i18n/en.pot | 7 +- src/components/App.js | 2 +- src/pages/view/SlideshowControlbar.js | 103 +++++++----------- src/pages/view/SlideshowFiltersInfo.js | 80 ++++++++++++++ .../styles/SlideshowControlbar.module.css | 53 +++++---- .../styles/SlideshowFiltersInfo.module.css | 30 +++++ 6 files changed, 185 insertions(+), 90 deletions(-) create mode 100644 src/pages/view/SlideshowFiltersInfo.js create mode 100644 src/pages/view/styles/SlideshowFiltersInfo.module.css diff --git a/i18n/en.pot b/i18n/en.pot index e25c81b88..c70e08f33 100644 --- a/i18n/en.pot +++ b/i18n/en.pot @@ -5,8 +5,8 @@ msgstr "" "Content-Type: text/plain; charset=utf-8\n" "Content-Transfer-Encoding: 8bit\n" "Plural-Forms: nplurals=2; plural=(n != 1)\n" -"POT-Creation-Date: 2024-12-10T13:23:21.815Z\n" -"PO-Revision-Date: 2024-12-10T13:23:21.817Z\n" +"POT-Creation-Date: 2024-12-12T16:07:57.017Z\n" +"PO-Revision-Date: 2024-12-12T16:07:57.018Z\n" msgid "Untitled dashboard" msgstr "Untitled dashboard" @@ -490,6 +490,9 @@ msgstr "No, cancel" msgid "Yes, remove filters" msgstr "Yes, remove filters" +msgid "{{totalFilterCount}} filters active" +msgstr "{{totalFilterCount}} filters active" + msgid "The dashboard couldn't be made available offline. Try again." msgstr "The dashboard couldn't be made available offline. Try again." diff --git a/src/components/App.js b/src/components/App.js index 28820c851..f4e3aa21f 100644 --- a/src/components/App.js +++ b/src/components/App.js @@ -48,7 +48,7 @@ const App = (props) => { return ( systemSettings && ( <> - + { - return `${filter.name}: ${ - filter.values.length > 1 - ? i18n.t('{{count}} selected', { - count: filter.values.length, - }) - : filter.values[0].name - }` -} - const SlideshowControlbar = ({ slideshowItemIndex, exitSlideshow, @@ -29,67 +16,57 @@ const SlideshowControlbar = ({ prevItem, numItems, }) => { - const filters = useSelector(sGetNamedItemFilters) const navigationDisabled = numItems === 1 return (
-
+
-
- - {`${slideshowItemIndex + 1} / ${numItems}`} - -
-
- {filters.map((filter) => ( - +
+ + {`${slideshowItemIndex + 1} / ${numItems}`} + +
+
+
+
) diff --git a/src/pages/view/SlideshowFiltersInfo.js b/src/pages/view/SlideshowFiltersInfo.js new file mode 100644 index 000000000..c89790553 --- /dev/null +++ b/src/pages/view/SlideshowFiltersInfo.js @@ -0,0 +1,80 @@ +import i18n from '@dhis2/d2-i18n' +import { Layer, Popper } from '@dhis2/ui' +import PropTypes from 'prop-types' +import React, { useMemo, useState, useRef } from 'react' +import { useSelector } from 'react-redux' +import { sGetNamedItemFilters } from '../../reducers/itemFilters.js' +import styles from './styles/SlideshowFiltersInfo.module.css' + +const popperModifiers = [ + { + name: 'offset', + options: { + offset: [0, 8], + }, + }, +] +const FilterSection = ({ name, values }) => ( +
+

{name}

+
    + {values.map((value) => ( +
  • + {value.name} +
  • + ))} +
+
+) + +FilterSection.propTypes = { + name: PropTypes.string, + values: PropTypes.arrayOf(PropTypes.shape({ name: PropTypes.string })), +} + +export const SlideshowFiltersInfo = () => { + const [isOpen, setIsOpen] = useState(false) + const ref = useRef(null) + const filters = useSelector(sGetNamedItemFilters) + const totalFilterCount = useMemo( + () => + filters.reduce((total, filter) => total + filter.values.length, 0), + [filters] + ) + + if (filters.length === 0) { + return null + } + + return ( + <> + + {isOpen && ( + setIsOpen(false)}> + + {filters.map((filter) => ( + + ))} + + + )} + + ) +} diff --git a/src/pages/view/styles/SlideshowControlbar.module.css b/src/pages/view/styles/SlideshowControlbar.module.css index bee5956c9..3ad2a293a 100644 --- a/src/pages/view/styles/SlideshowControlbar.module.css +++ b/src/pages/view/styles/SlideshowControlbar.module.css @@ -6,23 +6,38 @@ justify-content: space-between; align-items: center; background-color: black; - block-size: 40px; + padding-inline-start: 4px; + padding-inline-end: 4px; + z-index: 10; +} +.start, +.middle, +.end { + display: flex; + align-items: center; + width: calc(100% / 3); +} + +.start { + justify-content: start; +} + +.middle { + justify-content: center; } -.container > :first-child { - flex-grow: 1; - flex-basis: 0; - margin-inline-start: 4px; +.end { + justify-content: end; } + .container button { color: white; background-color: #171819; border-radius: 3px; border: none; - inline-size: 32px; - block-size: 32px; padding: 4px; cursor: pointer; + flex-grow: 0; } .container button:hover { @@ -32,6 +47,12 @@ .container button:disabled { cursor: not-allowed; } + +.container button.squareButton { + inline-size: 32px; + block-size: 32px; +} + .controls { padding: 4px; border-radius: 5px; @@ -39,6 +60,7 @@ align-items: center; justify-content: center; } + .pageCounter { font-family: 'Roboto', sans-serif; font-weight: 400; @@ -51,20 +73,3 @@ text-align: center; user-select: none; } - -.filters { - flex-grow: 1; - flex-basis: 0; - list-style-type: none; - display: flex; - justify-content: flex-end; - margin-inline-end: 4px; - gap: 4px; -} - -.filter { - padding: 4px; - border-radius: 3px; - color: black; - background-color: #f4f6f8; -} diff --git a/src/pages/view/styles/SlideshowFiltersInfo.module.css b/src/pages/view/styles/SlideshowFiltersInfo.module.css new file mode 100644 index 000000000..23deb876e --- /dev/null +++ b/src/pages/view/styles/SlideshowFiltersInfo.module.css @@ -0,0 +1,30 @@ +.filterButton { + padding: 4px; + inline-size: auto; + border-radius: 3px; + color: black; + background-color: #f4f6f8; +} + +.popover { + border: 1px solid var(--colors-grey400); + border-radius: 4px; + box-shadow: var(--elevations-e400); + padding: var(--spacers-dp8); + background-color: var(--colors-white); +} + +.filterSection { + margin-block-end: var(--spacers-dp4); +} + +.filterSectionHeader { + margin: 0; + padding: 0; +} + +.filterSectionList { + margin: 0; + padding: 0; + padding-inline-start: 18px; +}