diff --git a/packages/geoview-core/src/core/components/data-table/data-panel.tsx b/packages/geoview-core/src/core/components/data-table/data-panel.tsx index 51cf3dd9ebe..bf836d85295 100644 --- a/packages/geoview-core/src/core/components/data-table/data-panel.tsx +++ b/packages/geoview-core/src/core/components/data-table/data-panel.tsx @@ -2,6 +2,7 @@ import { useCallback, useEffect, useState } from 'react'; import { useTheme } from '@mui/material/styles'; import { Projection } from 'ol/proj'; import { useTranslation } from 'react-i18next'; +import { IconStack } from '@/app'; import { Box, Typography, @@ -11,7 +12,6 @@ import { ListItemButton, IconButton, ListItemIcon, - SendIcon, ChevronRightIcon, CircularProgress, Button, @@ -124,7 +124,7 @@ export function Datapanel({ layerData, mapId, projectionConfig, language }: Data handleListItemClick(event, index)}> - + {layerName![language]} diff --git a/packages/geoview-core/src/core/components/data-table/data-table-style.ts b/packages/geoview-core/src/core/components/data-table/data-table-style.ts index dafe5d6d8bc..90296abdb36 100644 --- a/packages/geoview-core/src/core/components/data-table/data-table-style.ts +++ b/packages/geoview-core/src/core/components/data-table/data-table-style.ts @@ -36,6 +36,7 @@ export const getSxClasses = (theme: Theme) => ({ listPrimaryText: { minWidth: '0', marginTop: '0.5rem', + marginLeft: '10px', marginBottom: '0.5rem', flex: '1 1 auto', display: 'flex', diff --git a/packages/geoview-core/src/core/components/details/details-style.ts b/packages/geoview-core/src/core/components/details/details-style.ts index b1e3a993ab5..774003aa0c8 100644 --- a/packages/geoview-core/src/core/components/details/details-style.ts +++ b/packages/geoview-core/src/core/components/details/details-style.ts @@ -18,6 +18,7 @@ export const getSxClasses = (theme: Theme) => ({ textOverflow: 'ellipsis', }, listPrimaryText: { + marginLeft: '0.62rem', minWidth: '0', marginTop: '0.5rem', marginBottom: '0.5rem', diff --git a/packages/geoview-core/src/core/components/details/layers-list-footer.tsx b/packages/geoview-core/src/core/components/details/layers-list-footer.tsx index a4365abba3a..bf52c87eb7f 100644 --- a/packages/geoview-core/src/core/components/details/layers-list-footer.tsx +++ b/packages/geoview-core/src/core/components/details/layers-list-footer.tsx @@ -12,7 +12,6 @@ import { ListItemButton, List, ListItemIcon, - SendIcon, Tooltip, IconButton, Grid, @@ -28,7 +27,7 @@ import { ArrowBackIcon, } from '@/ui'; import { FeatureInfo } from './feature-info-new'; -import { PayloadBaseClass, api } from '@/app'; +import { PayloadBaseClass, api, IconStack } from '@/app'; import { EVENT_NAMES } from '@/api/events/event-types'; import { payloadIsAFeatureHighlight, @@ -200,7 +199,7 @@ export function LayersListFooter(props: TypeLayersListProps): JSX.Element { }} > - + {layerData.layerName ? layerData.layerName : t('details.clickOnMap')} diff --git a/packages/geoview-core/src/core/components/icon-stack/helper.ts b/packages/geoview-core/src/core/components/icon-stack/helper.ts new file mode 100644 index 00000000000..76a7dbdb22d --- /dev/null +++ b/packages/geoview-core/src/core/components/icon-stack/helper.ts @@ -0,0 +1,69 @@ +export type TypeLayerInfo = { + layerPath: string; + iconData: string[]; +}; + +export type TypeLayerInfoArray = TypeLayerInfo[]; + +export const layerInfo: TypeLayerInfoArray = [ + { + layerPath: 'esriFeatureLYR4/8', + iconData: [ + '', + '', + '', + ], + }, + { + layerPath: 'geojsonLYR1/polygons.json', + iconData: [ + '', + ], + }, + { + layerPath: 'geojsonLYR1/lines.json', + iconData: [ + '', + '', + ], + }, + { + layerPath: 'uniqueValueId/1', + iconData: [ + '', + '', + '', + ], + }, + { + layerPath: 'esriFeatureLYR5/0', + iconData: [ + '', + '', + '', + ], + }, + { + layerPath: 'historical-flood/0', + iconData: [ + '', + '', + '', + ], + }, + { + layerPath: 'historical-flood/0', + iconData: [ + '', + '', + '', + ], + }, + { + layerPath: 'MSI/msi-94-or-more', + iconData: [ + '', + '', + ], + }, +]; diff --git a/packages/geoview-core/src/core/components/icon-stack/icon-stack-style.ts b/packages/geoview-core/src/core/components/icon-stack/icon-stack-style.ts new file mode 100644 index 00000000000..b2e88f2624b --- /dev/null +++ b/packages/geoview-core/src/core/components/icon-stack/icon-stack-style.ts @@ -0,0 +1,61 @@ +export const getSxClasses = () => ({ + legendIconTransparent: { + display: 'flex', + justifyContent: 'center', + alignItems: 'center', + width: 24, + height: 24, + }, + iconPreviewHoverable: { + position: 'absolute', + left: -3, + top: -2, + padding: 0, + borderRadius: 0, + border: '1px solid', + borderColor: 'grey.600', + boxShadow: 'rgb(0 0 0 / 20%) 0px 3px 1px -2px, rgb(0 0 0 / 14%) 0px 2px 2px 0px, rgb(0 0 0 / 12%) 0px 1px 5px 0px', + transition: 'transform .3s ease-in-out', + '&:hover': { + transform: 'rotate(-18deg) translateX(-8px)', + }, + }, + iconPreviewStacked: { + padding: 0, + borderRadius: 0, + border: '1px solid', + borderColor: 'grey.600', + boxShadow: 'rgb(0 0 0 / 20%) 0px 3px 1px -2px, rgb(0 0 0 / 14%) 0px 2px 2px 0px, rgb(0 0 0 / 12%) 0px 1px 5px 0px', + background: '#fff', + }, + maxIconImg: { + maxWidth: 24, + maxHeight: 24, + }, + legendIcon: { + display: 'flex', + justifyContent: 'center', + alignItems: 'center', + width: 24, + height: 24, + background: '#fff', + }, + stackIconsBox: { + position: 'relative', + marginLeft: 8, + '&:focus': { + outlineColor: 'grey', + }, + }, + iconPreview: { + marginLeft: 8, + padding: 0, + borderRadius: 0, + border: '1px solid', + borderColor: 'palette.grey.600', + boxShadow: 'rgb(0 0 0 / 20%) 0px 3px 1px -2px, rgb(0 0 0 / 14%) 0px 2px 2px 0px, rgb(0 0 0 / 12%) 0px 1px 5px 0px', + '&:focus': { + border: 'revert', + }, + }, +}); diff --git a/packages/geoview-core/src/core/components/icon-stack/icon-stack.tsx b/packages/geoview-core/src/core/components/icon-stack/icon-stack.tsx new file mode 100644 index 00000000000..b47668486f5 --- /dev/null +++ b/packages/geoview-core/src/core/components/icon-stack/icon-stack.tsx @@ -0,0 +1,70 @@ +/* eslint-disable react/require-default-props */ +import React from 'react'; +import { Box, IconButton, BrowserNotSupportedIcon } from '@/ui'; +import { getSxClasses } from './icon-stack-style'; +import { layerInfo, TypeLayerInfo } from './helper'; +// TODO uncomment line below when you have layers and icons in the store +// import { useIconLayerSet } from '@/core/stores/store-interface-and-intial-values/layer-state'; + +export interface TypeIconStackProps { + layerPath: string; + onIconClick?: () => void; + onStackIconClick?: (e: React.KeyboardEvent) => void; +} + +/** + * Icon Stack to represent layer icons + * + * @param {string} layerPath + * @returns {JSX.Element} the icon stack item + */ +export function IconStack({ layerPath, onIconClick, onStackIconClick }: TypeIconStackProps): JSX.Element | null { + const sxClasses = getSxClasses(); + + // TODO use line below when we can get layers and their icons from the store + // const iconData = useIconLayerSet(layerPath); + + // TODO function below is using the fake data, remove it once you have layers and icons in the store + const findIconsData = (layerPathStr: string): string[] | undefined => { + const foundMatchedLayerPath: TypeLayerInfo | undefined = layerInfo.find((layer: TypeLayerInfo) => layer.layerPath === layerPathStr); + return foundMatchedLayerPath?.iconData; + }; + const iconData = findIconsData(layerPath); + + const iconImg: string = iconData && iconData.length > 0 ? iconData[0] : ''; + const iconImgStacked: string = iconData && iconData.length > 1 ? iconData[1] : ''; + const numOfIcons: number | undefined = iconData?.length; + + // TODO for now just use 2 icons + // eslint-disable-next-line no-nested-ternary + return numOfIcons === 1 ? ( + + {iconImg === 'no data' ? ( + + ) : ( + + icon + + )} + + ) : numOfIcons && numOfIcons > 0 ? ( + onStackIconClick?.(e)}> + + + {iconImgStacked && icon} + + + + {iconImg && icon} + + + ) : ( + onStackIconClick?.(e)}> + + + + + + + ); +} diff --git a/packages/geoview-core/src/core/components/index.ts b/packages/geoview-core/src/core/components/index.ts index 119d92d7254..ec12674b8c1 100644 --- a/packages/geoview-core/src/core/components/index.ts +++ b/packages/geoview-core/src/core/components/index.ts @@ -25,6 +25,7 @@ export * from './geolocator/geo-list'; export * from './geolocator/geolocator'; export * from './geolocator/geolocator-style'; export * from './hover-tooltip/hover-tooltip'; +export * from './icon-stack/icon-stack'; export * from './legend-2/legend-api'; export * from './legend/legend-api'; export * from './legend/legend-icon-list'; diff --git a/packages/geoview-core/src/core/components/legend/legend-item.tsx b/packages/geoview-core/src/core/components/legend/legend-item.tsx index bf27d46c483..11556e1731b 100644 --- a/packages/geoview-core/src/core/components/legend/legend-item.tsx +++ b/packages/geoview-core/src/core/components/legend/legend-item.tsx @@ -29,7 +29,7 @@ import { BrowserNotSupportedIcon, Grid, } from '@/ui'; -import { api, payloadIsLegendInfo, NumberPayload, PayloadBaseClass, EsriDynamic } from '@/app'; +import { api, payloadIsLegendInfo, NumberPayload, PayloadBaseClass, EsriDynamic, IconStack } from '@/app'; import { LegendIconList } from './legend-icon-list'; import { AbstractGeoViewLayer, @@ -191,7 +191,7 @@ export function LegendItem(props: TypeLegendItemProps): JSX.Element { const [currentWMSStyle, setCurrentWMSStyle] = useState(); const [iconType, setIconType] = useState(null); const [iconImg, setIconImg] = useState(null); - const [iconImgStacked, setIconImgStacked] = useState(null); + const [, setIconImgStacked] = useState(null); const [iconList, setIconList] = useState(null); const [labelList, setLabelList] = useState(null); const [geometryLayerConfig, setLayerConfig] = useState(null); @@ -541,26 +541,10 @@ export function LegendItem(props: TypeLegendItemProps): JSX.Element { )} )} - {iconType === 'list' && !isLegendOpen && ( - - handleStackIcon(e)} - > - - - {iconImgStacked && icon} - - - - {iconImg && icon} - - - + {!isLegendOpen && ( + handleStackIcon(e)} /> )} + {groupItems.length === 0 && WMSStyles.length < 2 && !iconType && !isLegendOpen && ( diff --git a/packages/geoview-core/src/core/components/time-slider/time-slider-panel.tsx b/packages/geoview-core/src/core/components/time-slider/time-slider-panel.tsx index 6a6b4ff4981..b6d643ff007 100644 --- a/packages/geoview-core/src/core/components/time-slider/time-slider-panel.tsx +++ b/packages/geoview-core/src/core/components/time-slider/time-slider-panel.tsx @@ -12,7 +12,6 @@ import { ListItemButton, ListItemIcon, Paper, - SendIcon, Tooltip, Typography, ArrowForwardIcon, @@ -20,7 +19,7 @@ import { } from '@/ui'; import { getSxClasses } from './time-slider-style'; import { TimeSlider } from './time-slider'; -import { api, getLocalizedValue } from '@/app'; +import { api, getLocalizedValue, IconStack } from '@/app'; import { SliderFilterProps } from './time-slider-api'; import { ResponsiveGrid } from '../responsive-grid/responsive-grid'; @@ -75,7 +74,7 @@ export function TimeSliderPanel(props: TypeTimeSliderProps): JSX.Element { sx={{ height: '67px' }} > - + diff --git a/packages/geoview-core/src/core/components/time-slider/time-slider-style.ts b/packages/geoview-core/src/core/components/time-slider/time-slider-style.ts index 2e98d7c3e61..07361d56721 100644 --- a/packages/geoview-core/src/core/components/time-slider/time-slider-style.ts +++ b/packages/geoview-core/src/core/components/time-slider/time-slider-style.ts @@ -96,6 +96,7 @@ export const getSxClasses = (theme: Theme) => ({ }, gridContainer: { paddingLeft: '1rem', paddingRight: '1rem' }, listPrimaryText: { + marginLeft: '0.62rem', minWidth: '0', padding: '1.3rem 0', flex: '1 1 auto', diff --git a/packages/geoview-core/src/core/utils/utilities.ts b/packages/geoview-core/src/core/utils/utilities.ts index eb6e64f1142..e3b690087ab 100644 --- a/packages/geoview-core/src/core/utils/utilities.ts +++ b/packages/geoview-core/src/core/utils/utilities.ts @@ -57,7 +57,9 @@ function _addNotification(mapId: string, type: NotificationType = 'info', messag }; // ? Need to do lazy import, if not viewer crashes (AppEventProcessor.addAppNotification(mapId, notification)); - import('@/api/event-processors/event-processor-children/app-event-processor').then((mod) => mod.AppEventProcessor.addAppNotification(mapId, notification)); + import('@/api/event-processors/event-processor-children/app-event-processor').then((mod) => + mod.AppEventProcessor.addAppNotification(mapId, notification) + ); } /** diff --git a/packages/geoview-swiper/src/index.tsx b/packages/geoview-swiper/src/index.tsx index 336540dbc67..d74531e9c62 100644 --- a/packages/geoview-swiper/src/index.tsx +++ b/packages/geoview-swiper/src/index.tsx @@ -61,7 +61,8 @@ class SwiperPlugin extends AbstractPlugin { // create the swiper container and insert it after top link const el = document.createElement('div'); el.setAttribute('id', `${mapId}-swiper`); - document.getElementById(`toplink-${mapId}`)?.after(el); + const mapElement = document.getElementById(`map-${mapId}`); + mapElement?.insertBefore(el, mapElement.firstChild); // create the swiper component and render const node = createElement(Swiper, { mapId, config: configObj!, translations: this.translations });