Skip to content

Commit

Permalink
feat(plugins): Hides plugin tabs if no layers support them
Browse files Browse the repository at this point in the history
Closes #2369
  • Loading branch information
DamonU2 committed Jul 17, 2024
1 parent af7f1e8 commit 5b26cb9
Show file tree
Hide file tree
Showing 12 changed files with 119 additions and 42 deletions.
3 changes: 2 additions & 1 deletion .github/workflows/type-build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -44,7 +44,8 @@ jobs:
rm -r common
rm -r packages
git config --global user.name $USER
git config --global user.email "[email protected]"
git config --global user.name "Damon Ulmi"
git add .
git status
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ import { GeoChartConfig } from '@/core/utils/config/reader/uuid-config-reader';
import { logger } from '@/core/utils/logger';

import { AbstractEventProcessor, BatchedPropagationLayerDataArrayByMap } from '@/api/event-processors/abstract-event-processor';
import { UIEventProcessor } from './ui-event-processor';

// GV Important: See notes in header of MapEventProcessor file for information on the paradigm to apply when working with UIEventProcessor vs UIState

Expand Down Expand Up @@ -120,9 +121,12 @@ export class GeochartEventProcessor extends AbstractEventProcessor {
});
});

// set store charts config
// Set store charts config
this.getGeochartState(mapId)?.setterActions.setGeochartCharts(chartData);

// If there is chart data, tab should not be hidden
if (Object.keys(chartData).length) UIEventProcessor.removeHiddenTab(mapId, 'geochart');

// Log
logger.logInfo('Added GeoChart configs for layer paths:', layerPaths);
}
Expand All @@ -145,6 +149,9 @@ export class GeochartEventProcessor extends AbstractEventProcessor {
// Update the layer data array in the store
this.getGeochartState(mapId)!.setterActions.setGeochartCharts({ ...this.getGeochartState(mapId)?.geochartChartsConfig, ...toAdd });

// Make sure tab is not hidden
UIEventProcessor.removeHiddenTab(mapId, 'geochart');

// Log
logger.logInfo('Added GeoChart configs for layer path:', layerPath);
}
Expand All @@ -171,6 +178,9 @@ export class GeochartEventProcessor extends AbstractEventProcessor {
// Update the layer data array in the store
this.getGeochartState(mapId)!.setterActions.setGeochartCharts({ ...chartConfigs });

// If there are no more geochart layers, hide tab
if (!Object.keys(this.getGeochartState(mapId)!.geochartChartsConfig).length) UIEventProcessor.addHiddenTab(mapId, 'geochart');

// Log
logger.logInfo('Removed GeoChart configs for layer path:', layerPath);
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -303,12 +303,14 @@ export class MapEventProcessor extends AbstractEventProcessor {
}

/**
* Gets the ordered layer info.
* Gets map layer paths in order.
* @param {string} mapId - The map id
* @returns {TypeOrderedLayerInfo[]} The ordered layer info
* @returns {string[]} The ordered layer paths
*/
static getMapLayerOrder(mapId: string): TypeOrderedLayerInfo[] {
return this.getMapStateProtected(mapId).orderedLayerInfo;
static getMapLayerOrder(mapId: string): string[] {
return this.getMapStateProtected(mapId).orderedLayerInfo.map((orderedLayerInfo) => {
return orderedLayerInfo.layerPath;
});
}

static getMapState(mapId: string): TypeMapState {
Expand Down Expand Up @@ -453,6 +455,11 @@ export class MapEventProcessor extends AbstractEventProcessor {
// GV No need to save in the store, because this will trigger an event on MapViewer which will take care of updating the store
}

/**
* Gets the ordered layer info.
* @param {string} mapId - The map id
* @returns {TypeOrderedLayerInfo[]} The ordered layer info
*/
static getMapOrderedLayerInfo(mapId: string): TypeOrderedLayerInfo[] {
return this.getMapStateProtected(mapId).orderedLayerInfo;
}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -15,6 +15,7 @@ import { MapEventProcessor } from './map-event-processor';
import { AbstractGVVector } from '@/geo/layer/gv-layers/vector/abstract-gv-vector';
import { GVWMS } from '@/geo/layer/gv-layers/raster/gv-wms';
import { GVEsriImage } from '@/geo/layer/gv-layers/raster/gv-esri-image';
import { UIEventProcessor } from './ui-event-processor';

// GV Important: See notes in header of MapEventProcessor file for information on the paradigm to apply when working with UIEventProcessor vs UIState

Expand Down Expand Up @@ -85,6 +86,9 @@ export class TimeSliderEventProcessor extends AbstractEventProcessor {

const { defaultValue, field, filtering, minAndMax, values } = timeSliderLayer[layerPath];
this.applyFilters(mapId, layerPath, defaultValue, field, filtering, minAndMax, values);

// Make sure tab is visible
UIEventProcessor.removeHiddenTab(mapId, 'time-slider');
}

/**
Expand All @@ -95,6 +99,10 @@ export class TimeSliderEventProcessor extends AbstractEventProcessor {
static removeTimeSliderLayer(mapId: string, layerPath: string): void {
// Redirect
this.getTimesliderState(mapId)?.setterActions.removeTimeSliderLayer(layerPath);

// If there are no layers with time dimension, hide tab
if (!this.getTimesliderState(mapId) || !Object.keys(this.getTimesliderState(mapId)!.timeSliderLayers).length)
UIEventProcessor.addHiddenTab(mapId, 'time-slider');
}

/**
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -34,13 +34,31 @@ export class UIEventProcessor extends AbstractEventProcessor {
static getCorePackageComponents(mapId: string): TypeMapCorePackages {
return this.getUIState(mapId).corePackagesComponents;
}

static getFooterBarIsCollapsed(mapId: string): boolean {
return this.getUIState(mapId).footerBarIsCollapsed;
}
// #endregion

// **********************************************************
// Static functions for Store Map State to action on API
// **********************************************************
// GV NEVER add a store action who does set state AND map action at a same time.
// GV Review the action in store state to make sure
static addHiddenTab(mapId: string, tab: string): void {
if (!this.getUIState(mapId).hiddenTabs.includes(tab))
this.getUIState(mapId).setterActions.setHiddenTabs([...this.getUIState(mapId).hiddenTabs, tab]);
}

static removeHiddenTab(mapId: string, tab: string): void {
const curHiddenTabs = this.getUIState(mapId).hiddenTabs;
const tabIndex = curHiddenTabs.indexOf(tab);
if (tabIndex !== -1) {
curHiddenTabs.splice(tabIndex, 1);
this.getUIState(mapId).setterActions.setHiddenTabs(curHiddenTabs);
}
}

static setActiveFooterBarTab(mapId: string, id: string): void {
this.getUIState(mapId).setterActions.setActiveFooterBarTab(id);
}
Expand All @@ -52,4 +70,8 @@ export class UIEventProcessor extends AbstractEventProcessor {
static getActiveAppBarTab(mapId: string): ActiveAppBarTabType {
return this.getUIState(mapId).activeAppBarTab;
}

static setFooterBarIsCollapsed(mapId: string, collapsed: boolean): void {
this.getUIState(mapId).setterActions.setFooterBarIsCollapsed(collapsed);
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -119,8 +119,8 @@ export function FooterBar(props: FooterBarProps): JSX.Element | null {
label: `${camelCase(tab)}.title`,
icon: allTabs[tab]?.icon ?? '',
content: allTabs[tab]?.content ?? '',
};
}) as unknown as TypeTabs[];
} as TypeTabs;
});
}, [memoTabs, tabsList]);

/**
Expand Down
2 changes: 1 addition & 1 deletion packages/geoview-core/src/core/stores/state-api.ts
Original file line number Diff line number Diff line change
Expand Up @@ -110,7 +110,7 @@ export class StateApi {
// Apply some ordering logic
const direction = move < 0 ? -1 : 1;
let absoluteMoves = Math.abs(move);
const orderedLayers = [...MapEventProcessor.getMapLayerOrder(this.mapId)];
const orderedLayers = [...MapEventProcessor.getMapOrderedLayerInfo(this.mapId)];
let startingIndex = -1;
for (let i = 0; i < orderedLayers.length; i++) if (orderedLayers[i].layerPath === layerPath) startingIndex = i;
const layerInfo = orderedLayers[startingIndex];
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import { TypeMapCorePackages, TypeNavBarProps, TypeValidAppBarCoreProps } from '
import { useGeoViewStore } from '@/core/stores/stores-managers';
import { TypeSetStore, TypeGetStore } from '@/core/stores/geoview-store';
import { TypeMapFeaturesConfig } from '@/core/types/global-types';
import { UIEventProcessor } from '@/api/event-processors/event-processor-children/ui-event-processor';

// GV Important: See notes in header of MapEventProcessor file for information on the paradigm to apply when working with UIEventProcessor vs UIState

Expand All @@ -23,6 +24,7 @@ export interface IUIState {
appBarComponents: TypeValidAppBarCoreProps[];
corePackagesComponents: TypeMapCorePackages;
focusITem: FocusItemProps;
hiddenTabs: string[];
mapInfoExpanded: boolean;
navBarComponents: TypeNavBarProps;
footerPanelResizeValue: number;
Expand All @@ -31,8 +33,10 @@ export interface IUIState {
setDefaultConfigValues: (geoviewConfig: TypeMapFeaturesConfig) => void;

actions: {
addHiddenTab: (tab: string) => void;
closeModal: () => void;
openModal: (uiFocus: FocusItemProps) => void;
removeHiddenTab: (tab: string) => void;
setActiveFooterBarTab: (id: string) => void;
setActiveAppBarTab: (tabId: string, tabGroup: string, isOpen: boolean) => void;
setActiveTrapGeoView: (active: boolean) => void;
Expand All @@ -48,6 +52,7 @@ export interface IUIState {
setActiveAppBarTab: (tabId: string, tabGroup: string, isOpen: boolean) => void;
setActiveTrapGeoView: (active: boolean) => void;
setFooterPanelResizeValue: (value: number) => void;
setHiddenTabs: (hiddenTabs: string[]) => void;
setMapInfoExpanded: (expanded: boolean) => void;
setFooterBarIsCollapsed: (collapsed: boolean) => void;
};
Expand All @@ -69,6 +74,7 @@ export function initializeUIState(set: TypeSetStore, get: TypeGetStore): IUIStat
activeTrapGeoView: false,
corePackagesComponents: [],
focusITem: { activeElementId: false, callbackElementId: false },
hiddenTabs: ['time-slider', 'geochart'],
mapInfoExpanded: false,
navBarComponents: [],
footerPanelResizeValue: 35,
Expand All @@ -90,6 +96,10 @@ export function initializeUIState(set: TypeSetStore, get: TypeGetStore): IUIStat
// #region ACTIONS

actions: {
addHiddenTab: (tab: string): void => {
// Redirect to event processor
UIEventProcessor.addHiddenTab(get().mapId, tab);
},
closeModal: () => {
// Redirect to setter
get().uiState.setterActions.closeModal();
Expand All @@ -98,6 +108,10 @@ export function initializeUIState(set: TypeSetStore, get: TypeGetStore): IUIStat
// Redirect to setter
get().uiState.setterActions.openModal(uiFocus);
},
removeHiddenTab: (tab: string): void => {
// Redirect to event processor
UIEventProcessor.removeHiddenTab(get().mapId, tab);
},
setActiveFooterBarTab: (id: string) => {
// Redirect to setter
get().uiState.setterActions.setActiveFooterBarTab(id);
Expand Down Expand Up @@ -158,6 +172,14 @@ export function initializeUIState(set: TypeSetStore, get: TypeGetStore): IUIStat
},
});
},
setHiddenTabs: (hiddenTabs: string[]) => {
set({
uiState: {
...get().uiState,
hiddenTabs: [...hiddenTabs],
},
});
},
setFooterPanelResizeValue: (value) => {
set({
uiState: {
Expand Down Expand Up @@ -220,6 +242,7 @@ export const useUICorePackagesComponents = (): TypeMapCorePackages =>
useStore(useGeoViewStore(), (state) => state.uiState.corePackagesComponents);
export const useUIFooterPanelResizeValue = (): number => useStore(useGeoViewStore(), (state) => state.uiState.footerPanelResizeValue);
export const useUIFooterPanelResizeValues = (): number[] => useStore(useGeoViewStore(), (state) => state.uiState.footerPanelResizeValues);
export const useUIHiddenTabs = (): string[] => useStore(useGeoViewStore(), (state) => state.uiState.hiddenTabs);
export const useUIMapInfoExpanded = (): boolean => useStore(useGeoViewStore(), (state) => state.uiState.mapInfoExpanded);
export const useUINavbarComponents = (): TypeNavBarProps => useStore(useGeoViewStore(), (state) => state.uiState.navBarComponents);
export const useUIFooterBarIsCollapsed = (): boolean => useStore(useGeoViewStore(), (state) => state.uiState.footerBarIsCollapsed);
Expand Down
2 changes: 1 addition & 1 deletion packages/geoview-core/src/geo/map/map-viewer.ts
Original file line number Diff line number Diff line change
Expand Up @@ -826,7 +826,7 @@ export class MapViewer {
* @returns {TypeOrderedLayerInfo[]} The ordered layer info
*/
getMapLayerOrderInfo(): TypeOrderedLayerInfo[] {
return MapEventProcessor.getMapLayerOrder(this.mapId);
return MapEventProcessor.getMapOrderedLayerInfo(this.mapId);
}

/**
Expand Down
34 changes: 18 additions & 16 deletions packages/geoview-core/src/ui/button/button.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -42,22 +42,24 @@ export function Button(props: ButtonProps): JSX.Element {

return (
<Tooltip title={t(tooltip || '')} placement={tooltipPlacement || 'bottom'} TransitionComponent={Fade}>
<MaterialButton
fullWidth={fullWidth}
id={id}
size={size || 'medium'}
sx={sx}
variant={variant || 'text'}
className={`${className || ''}`}
onClick={onClick}
autoFocus={autoFocus}
disabled={disabled}
disableRipple={disableRipple}
startIcon={startIcon}
endIcon={endIcon}
>
{!(makeResponsive && mobileView) ? children : null}
</MaterialButton>
<span>
<MaterialButton
fullWidth={fullWidth}
id={id}
size={size || 'medium'}
sx={sx}
variant={variant || 'text'}
className={`${className || ''}`}
onClick={onClick}
autoFocus={autoFocus}
disabled={disabled}
disableRipple={disableRipple}
startIcon={startIcon}
endIcon={endIcon}
>
{!(makeResponsive && mobileView) ? children : null}
</MaterialButton>
</span>
</Tooltip>
);
}
32 changes: 17 additions & 15 deletions packages/geoview-core/src/ui/icon-button/icon-button.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -32,21 +32,23 @@ export function IconButton(props: TypeIconButtonProps): JSX.Element {

function getMaterialIconButton(): JSX.Element {
return (
<MaterialIconButton
id={id}
sx={sx}
aria-label={(t(ariaLabel as string) || t(tooltip as string)) as string}
style={style}
className={className}
onClick={onClick}
tabIndex={tabIndex}
size={size}
ref={iconRef}
disabled={disabled}
color={color}
>
{children && children}
</MaterialIconButton>
<span>
<MaterialIconButton
id={id}
sx={sx}
aria-label={(t(ariaLabel as string) || t(tooltip as string)) as string}
style={style}
className={className}
onClick={onClick}
tabIndex={tabIndex}
size={size}
ref={iconRef}
disabled={disabled}
color={color}
>
{children && children}
</MaterialIconButton>
</span>
);
}

Expand Down
4 changes: 3 additions & 1 deletion packages/geoview-core/src/ui/tabs/tabs.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -10,6 +10,7 @@ import { Select, TypeMenuItemProps } from '@/ui/select/select';
import { getSxClasses } from './tabs-style';
import { TabPanel } from './tab-panel';
import { useMapSize } from '@/core/stores/store-interface-and-intial-values/map-state';
import { useUIHiddenTabs } from '@/core/stores/store-interface-and-intial-values/ui-state';

/**
* Type used for properties of each tab
Expand Down Expand Up @@ -84,6 +85,7 @@ export function Tabs(props: TypeTabsProps): JSX.Element {

// get store values and actions
const mapSize = useMapSize();
const hiddenTabs = useUIHiddenTabs();

// show/hide dropdown based on map size
const initMobileDropdown = mapSize[0] !== 0 ? mapSize[0] < theme.breakpoints.values.sm : false;
Expand Down Expand Up @@ -197,7 +199,7 @@ export function Tabs(props: TypeTabsProps): JSX.Element {
iconPosition="start"
id={`tab-${index}`}
onClick={() => handleClick(index)}
sx={sxClasses.tab}
sx={hiddenTabs.includes(tab.id) ? { display: 'none' } : sxClasses.tab}
{...tabProps}
/>
);
Expand Down

0 comments on commit 5b26cb9

Please sign in to comment.