From 489dc7e991a8b626c05f3839564bcffc8e37d030 Mon Sep 17 00:00:00 2001
From: =?UTF-8?q?Tadeusz=20Miesi=C4=85c?= <tadeusz.miesiac@gmail.com>
Date: Tue, 19 Dec 2023 12:02:22 +0100
Subject: [PATCH] feat(overlays): cover rendering cases for additional
 properties for bioEntities

---
 .../OverlayListItem.component.test.tsx        | 46 ++++++++++++-
 .../OverlayListItem.component.tsx             |  3 +
 .../hooks/useEmptyBackground.test.ts          | 43 +++++++++++++
 .../hooks/useEmptyBackground.ts               | 22 +++++++
 .../overlaysLayer/getOverlayFeatures.ts       | 64 +++++++++++++------
 .../overlaysLayer/useOlMapOverlaysLayer.ts    | 12 +++-
 src/constants/hexColors.ts                    |  1 +
 src/hooks/useTriColorLerp.ts                  | 14 +++-
 .../fixtures/overlayBioEntityFixture.ts       | 10 +++
 src/models/mocks/modelsMock.ts                | 18 ++++++
 src/redux/backgrounds/background.selectors.ts |  9 +++
 .../configuration/configuration.selectors.ts  |  6 ++
 .../overlayBioEntity.utils.ts                 |  1 +
 src/types/OLrendering.ts                      |  3 +
 src/types/models.ts                           |  2 +
 .../getHexTricolorGradientColorWithAlpha.ts   |  3 +-
 src/utils/convert/hexToRgb.test.ts            |  6 +-
 src/utils/convert/hexToRgb.ts                 |  5 +-
 todo.txt                                      | 14 ----
 19 files changed, 234 insertions(+), 48 deletions(-)
 create mode 100644 src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.test.ts
 create mode 100644 src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.ts
 create mode 100644 src/constants/hexColors.ts
 create mode 100644 src/models/fixtures/overlayBioEntityFixture.ts
 delete mode 100644 todo.txt

diff --git a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.test.tsx b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.test.tsx
index 21e54427..dd3cc6ea 100644
--- a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.test.tsx
+++ b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.test.tsx
@@ -1,11 +1,28 @@
 import { StoreType } from '@/redux/store';
-import { render, screen } from '@testing-library/react';
+import { act, render, screen } from '@testing-library/react';
 import {
   InitialStoreState,
   getReduxWrapperWithStore,
 } from '@/utils/testing/getReduxWrapperWithStore';
+import {
+  BACKGROUNDS_MOCK,
+  BACKGROUND_INITIAL_STATE_MOCK,
+} from '@/redux/backgrounds/background.mock';
+import { initialMapStateFixture } from '@/redux/map/map.fixtures';
+import { mockNetworkNewAPIResponse } from '@/utils/mockNetworkResponse';
+import { OVERLAY_BIO_ENTITY_INITIAL_STATE_MOCK } from '@/redux/overlayBioEntity/overlayBioEntity.mock';
+import { HttpStatusCode } from 'axios';
+import { overlayBioEntityFixture } from '@/models/fixtures/overlayBioEntityFixture';
+import { apiPath } from '@/redux/apiPath';
+import { CORE_PD_MODEL_MOCK } from '@/models/mocks/modelsMock';
+import { MODELS_INITIAL_STATE_MOCK } from '@/redux/models/models.mock';
+import { parseOverlayBioEntityToOlRenderingFormat } from '@/redux/overlayBioEntity/overlayBioEntity.utils';
 import { OverlayListItem } from './OverlayListItem.component';
 
+const mockedAxiosNewClient = mockNetworkNewAPIResponse();
+const DEFAULT_BACKGROUND_ID = 0;
+const EMPTY_BACKGROUND_ID = 15;
+
 const renderComponent = (initialStoreState: InitialStoreState = {}): { store: StoreType } => {
   const { Wrapper, store } = getReduxWrapperWithStore(initialStoreState);
 
@@ -29,8 +46,31 @@ describe('OverlayListItem - component', () => {
     expect(screen.getByRole('button', { name: 'View' })).toBeInTheDocument();
     expect(screen.getByRole('button', { name: 'Download' })).toBeInTheDocument();
   });
-  // TODO implement when connecting logic to component
-  it.skip('should trigger view overlays on view button click', () => {});
+
+  it('should trigger view overlays on view button click and switch background to Empty if available', async () => {
+    const OVERLAY_ID = 21;
+    const { store } = renderComponent({
+      map: initialMapStateFixture,
+      backgrounds: { ...BACKGROUND_INITIAL_STATE_MOCK, data: BACKGROUNDS_MOCK },
+      overlayBioEntity: OVERLAY_BIO_ENTITY_INITIAL_STATE_MOCK,
+      models: { ...MODELS_INITIAL_STATE_MOCK, data: [CORE_PD_MODEL_MOCK] },
+    });
+    mockedAxiosNewClient
+      .onGet(apiPath.getOverlayBioEntity({ overlayId: OVERLAY_ID, modelId: 5053 }))
+      .reply(HttpStatusCode.Ok, overlayBioEntityFixture);
+
+    expect(store.getState().map.data.backgroundId).toBe(DEFAULT_BACKGROUND_ID);
+
+    const ViewButton = screen.getByRole('button', { name: 'View' });
+    await act(() => {
+      ViewButton.click();
+    });
+
+    expect(store.getState().map.data.backgroundId).toBe(EMPTY_BACKGROUND_ID);
+    expect(store.getState().overlayBioEntity.data).toEqual(
+      parseOverlayBioEntityToOlRenderingFormat(overlayBioEntityFixture, OVERLAY_ID),
+    );
+  });
   // TODO implement when connecting logic to component
   it.skip('should trigger download overlay to PC on download button click', () => {});
 });
diff --git a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.tsx b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.tsx
index b7746386..20f173fe 100644
--- a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.tsx
+++ b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/OverlayListItem.component.tsx
@@ -1,6 +1,7 @@
 import { useAppDispatch } from '@/redux/hooks/useAppDispatch';
 import { getOverlayBioEntityForAllModels } from '@/redux/overlayBioEntity/overlayBioEntity.thunk';
 import { Button } from '@/shared/Button';
+import { useEmptyBackground } from './hooks/useEmptyBackground';
 
 interface OverlayListItemProps {
   name: string;
@@ -10,8 +11,10 @@ interface OverlayListItemProps {
 export const OverlayListItem = ({ name, overlayId }: OverlayListItemProps): JSX.Element => {
   const onDownloadOverlay = (): void => {};
   const dispatch = useAppDispatch();
+  const { setBackgroundtoEmptyIfAvailable } = useEmptyBackground();
 
   const onViewOverlay = (): void => {
+    setBackgroundtoEmptyIfAvailable();
     dispatch(getOverlayBioEntityForAllModels({ overlayId }));
   };
 
diff --git a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.test.ts b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.test.ts
new file mode 100644
index 00000000..9c58dfc3
--- /dev/null
+++ b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.test.ts
@@ -0,0 +1,43 @@
+import { getReduxWrapperWithStore } from '@/utils/testing/getReduxWrapperWithStore';
+import { initialMapStateFixture } from '@/redux/map/map.fixtures';
+import {
+  BACKGROUNDS_MOCK,
+  BACKGROUND_INITIAL_STATE_MOCK,
+} from '@/redux/backgrounds/background.mock';
+import { renderHook } from '@testing-library/react';
+import { useEmptyBackground } from './useEmptyBackground';
+
+const DEFAULT_BACKGROUND_ID = 0;
+const EMPTY_BACKGROUND_ID = 15;
+
+describe('useEmptyBackground - hook', () => {
+  describe('returns setEmptyBackground function', () => {
+    it('should not set background to "Empty" if its not available', () => {
+      const { Wrapper, store } = getReduxWrapperWithStore({
+        map: initialMapStateFixture,
+        backgrounds: BACKGROUND_INITIAL_STATE_MOCK,
+      });
+      const { result } = renderHook(() => useEmptyBackground(), { wrapper: Wrapper });
+
+      expect(store.getState().map.data.backgroundId).toBe(DEFAULT_BACKGROUND_ID);
+
+      result.current.setBackgroundtoEmptyIfAvailable();
+
+      expect(store.getState().map.data.backgroundId).toBe(DEFAULT_BACKGROUND_ID);
+    });
+
+    it('should set background to "Empty" if its available', () => {
+      const { Wrapper, store } = getReduxWrapperWithStore({
+        map: initialMapStateFixture,
+        backgrounds: { ...BACKGROUND_INITIAL_STATE_MOCK, data: BACKGROUNDS_MOCK },
+      });
+      const { result } = renderHook(() => useEmptyBackground(), { wrapper: Wrapper });
+
+      expect(store.getState().map.data.backgroundId).toBe(DEFAULT_BACKGROUND_ID);
+
+      result.current.setBackgroundtoEmptyIfAvailable();
+
+      expect(store.getState().map.data.backgroundId).toBe(EMPTY_BACKGROUND_ID);
+    });
+  });
+});
diff --git a/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.ts b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.ts
new file mode 100644
index 00000000..2536fa84
--- /dev/null
+++ b/src/components/Map/Drawer/OverlaysDrawer/GeneralOverlays/OverlayListItem/hooks/useEmptyBackground.ts
@@ -0,0 +1,22 @@
+import { useCallback } from 'react';
+import { emptyBackgroundIdSelector } from '@/redux/backgrounds/background.selectors';
+import { useAppDispatch } from '@/redux/hooks/useAppDispatch';
+import { useAppSelector } from '@/redux/hooks/useAppSelector';
+import { setMapBackground } from '@/redux/map/map.slice';
+
+type UseEmptyBackgroundReturn = {
+  setBackgroundtoEmptyIfAvailable: () => void;
+};
+
+export const useEmptyBackground = (): UseEmptyBackgroundReturn => {
+  const dispatch = useAppDispatch();
+  const emptyBackgroundId = useAppSelector(emptyBackgroundIdSelector);
+
+  const setBackgroundtoEmptyIfAvailable = useCallback(() => {
+    if (emptyBackgroundId) {
+      dispatch(setMapBackground(emptyBackgroundId));
+    }
+  }, [dispatch, emptyBackgroundId]);
+
+  return { setBackgroundtoEmptyIfAvailable };
+};
diff --git a/src/components/Map/MapViewer/utils/config/overlaysLayer/getOverlayFeatures.ts b/src/components/Map/MapViewer/utils/config/overlaysLayer/getOverlayFeatures.ts
index 60a3ea8a..1cb81f70 100644
--- a/src/components/Map/MapViewer/utils/config/overlaysLayer/getOverlayFeatures.ts
+++ b/src/components/Map/MapViewer/utils/config/overlaysLayer/getOverlayFeatures.ts
@@ -1,28 +1,54 @@
 import { ZERO } from '@/constants/common';
-import { GetHex3ColorGradientColorWithAlpha } from '@/hooks/useTriColorLerp';
+import type { GetHex3ColorGradientColorWithAlpha } from '@/hooks/useTriColorLerp';
 import { OverlayBioEntityRender } from '@/types/OLrendering';
+import { convertDecimalToHex } from '@/utils/convert/convertDecimalToHex';
 import { UsePointToProjectionResult } from '@/utils/map/usePointToProjection';
 import Feature from 'ol/Feature';
 import Polygon, { fromExtent } from 'ol/geom/Polygon';
 import { Fill, Style } from 'ol/style';
 
-export const getOverlayFeatures = (
-  bioEntities: OverlayBioEntityRender[],
-  pointToProjection: UsePointToProjectionResult,
-  getHex3ColorGradientColorWithAlpha: GetHex3ColorGradientColorWithAlpha,
-): Feature<Polygon>[] =>
-  bioEntities.map(entity => {
-    const feature = new Feature({
-      geometry: fromExtent([
+export const createOverlayGeometryFeature = (
+  [xMin, yMin, xMax, yMax]: number[],
+  color: string,
+): Feature<Polygon> => {
+  const feature = new Feature({ geometry: fromExtent([xMin, yMin, xMax, yMax]) });
+  feature.setStyle(new Style({ fill: new Fill({ color }) }));
+  return feature;
+};
+
+type GetOverlayFeaturesProps = {
+  bioEntities: OverlayBioEntityRender[];
+  pointToProjection: UsePointToProjectionResult;
+  getHex3ColorGradientColorWithAlpha: GetHex3ColorGradientColorWithAlpha;
+  defaultColor: string;
+};
+
+export const getColorByAvailableProperties = (
+  entity: OverlayBioEntityRender,
+  getHexTricolorGradientColorWithAlpha: GetHex3ColorGradientColorWithAlpha,
+  defaultColor: string,
+): string => {
+  if (entity.value) {
+    return getHexTricolorGradientColorWithAlpha(entity.value || ZERO);
+  }
+  if (entity.color) {
+    return convertDecimalToHex(entity.color.rgb);
+  }
+  return defaultColor;
+};
+
+export const getOverlayFeatures = ({
+  bioEntities,
+  pointToProjection,
+  getHex3ColorGradientColorWithAlpha,
+  defaultColor,
+}: GetOverlayFeaturesProps): Feature<Polygon>[] =>
+  bioEntities.map(entity =>
+    createOverlayGeometryFeature(
+      [
         ...pointToProjection({ x: entity.x1, y: entity.y1 }),
         ...pointToProjection({ x: entity.x2, y: entity.y2 }),
-      ]),
-    });
-    feature.setStyle(
-      new Style({
-        fill: new Fill({ color: getHex3ColorGradientColorWithAlpha(entity.value || ZERO) }),
-      }),
-    );
-
-    return feature;
-  });
+      ],
+      getColorByAvailableProperties(entity, getHex3ColorGradientColorWithAlpha, defaultColor),
+    ),
+  );
diff --git a/src/components/Map/MapViewer/utils/config/overlaysLayer/useOlMapOverlaysLayer.ts b/src/components/Map/MapViewer/utils/config/overlaysLayer/useOlMapOverlaysLayer.ts
index 73fddda3..011dac03 100644
--- a/src/components/Map/MapViewer/utils/config/overlaysLayer/useOlMapOverlaysLayer.ts
+++ b/src/components/Map/MapViewer/utils/config/overlaysLayer/useOlMapOverlaysLayer.ts
@@ -10,12 +10,18 @@ import { getOverlayFeatures } from './getOverlayFeatures';
 
 export const useOlMapOverlaysLayer = (): VectorLayer<VectorSource<Geometry>> => {
   const pointToProjection = usePointToProjection();
-  const { getHex3ColorGradientColorWithAlpha } = useTriColorLerp();
+  const { getHex3ColorGradientColorWithAlpha, defaultColorHex } = useTriColorLerp();
   const bioEntities = useAppSelector(overlayBioEntitiesForCurrentModelSelector);
 
   const features = useMemo(
-    () => getOverlayFeatures(bioEntities, pointToProjection, getHex3ColorGradientColorWithAlpha),
-    [bioEntities, getHex3ColorGradientColorWithAlpha, pointToProjection],
+    () =>
+      getOverlayFeatures({
+        bioEntities,
+        pointToProjection,
+        getHex3ColorGradientColorWithAlpha,
+        defaultColor: defaultColorHex,
+      }),
+    [bioEntities, getHex3ColorGradientColorWithAlpha, pointToProjection, defaultColorHex],
   );
 
   const vectorSource = useMemo(() => {
diff --git a/src/constants/hexColors.ts b/src/constants/hexColors.ts
new file mode 100644
index 00000000..1a81cfbc
--- /dev/null
+++ b/src/constants/hexColors.ts
@@ -0,0 +1 @@
+export const WHITE_HEX_OPACITY_0 = '#00000000';
diff --git a/src/hooks/useTriColorLerp.ts b/src/hooks/useTriColorLerp.ts
index aeff150a..f872b09c 100644
--- a/src/hooks/useTriColorLerp.ts
+++ b/src/hooks/useTriColorLerp.ts
@@ -1,24 +1,30 @@
+import { useCallback } from 'react';
+import { WHITE_HEX_OPACITY_0 } from '@/constants/hexColors';
 import {
   maxColorValSelector,
   minColorValSelector,
   neutralColorValSelector,
   overlayOpacitySelector,
+  simpleColorValSelector,
 } from '@/redux/configuration/configuration.selectors';
 import { useAppSelector } from '@/redux/hooks/useAppSelector';
 import { getHexTricolorGradientColorWithAlpha } from '@/utils/convert/getHexTricolorGradientColorWithAlpha';
-import { useCallback } from 'react';
+import { ONE } from '@/constants/common';
+import { addAlphaToHexString } from '../utils/convert/addAlphaToHexString';
 
 export type GetHex3ColorGradientColorWithAlpha = (position: number) => string;
 
 type UseTriColorLerpReturn = {
   getHex3ColorGradientColorWithAlpha: GetHex3ColorGradientColorWithAlpha;
+  defaultColorHex: string;
 };
 
 export const useTriColorLerp = (): UseTriColorLerpReturn => {
   const minColorValHexString = useAppSelector(minColorValSelector) || '';
   const maxColorValHexString = useAppSelector(maxColorValSelector) || '';
   const neutralColorValHexString = useAppSelector(neutralColorValSelector) || '';
-  const overlayOpacityValue = useAppSelector(overlayOpacitySelector) || '';
+  const overlayOpacityValue = useAppSelector(overlayOpacitySelector) || ONE;
+  const simpleColorValue = useAppSelector(simpleColorValSelector) || WHITE_HEX_OPACITY_0;
 
   const getHex3ColorGradientColorWithAlpha = useCallback(
     (position: number) =>
@@ -32,5 +38,7 @@ export const useTriColorLerp = (): UseTriColorLerpReturn => {
     [minColorValHexString, neutralColorValHexString, maxColorValHexString, overlayOpacityValue],
   );
 
-  return { getHex3ColorGradientColorWithAlpha };
+  const defaultColorHex = addAlphaToHexString(simpleColorValue, Number(overlayOpacityValue));
+
+  return { getHex3ColorGradientColorWithAlpha, defaultColorHex };
 };
diff --git a/src/models/fixtures/overlayBioEntityFixture.ts b/src/models/fixtures/overlayBioEntityFixture.ts
new file mode 100644
index 00000000..da0c6da6
--- /dev/null
+++ b/src/models/fixtures/overlayBioEntityFixture.ts
@@ -0,0 +1,10 @@
+import { ZOD_SEED } from '@/constants';
+import { z } from 'zod';
+// eslint-disable-next-line import/no-extraneous-dependencies
+import { createFixture } from 'zod-fixture';
+import { overlayBioEntitySchema } from '../overlayBioEntitySchema';
+
+export const overlayBioEntityFixture = createFixture(z.array(overlayBioEntitySchema), {
+  seed: ZOD_SEED,
+  array: { min: 3, max: 3 },
+});
diff --git a/src/models/mocks/modelsMock.ts b/src/models/mocks/modelsMock.ts
index 96cd8bf9..5254ae65 100644
--- a/src/models/mocks/modelsMock.ts
+++ b/src/models/mocks/modelsMock.ts
@@ -457,3 +457,21 @@ export const MODELS_MOCK_SHORT: MapModel[] = [
     maxZoom: 5,
   },
 ];
+
+export const CORE_PD_MODEL_MOCK: MapModel = {
+  idObject: 5053,
+  width: 26779.25,
+  height: 13503.0,
+  defaultCenterX: null,
+  defaultCenterY: null,
+  description: '',
+  name: 'Core PD map',
+  defaultZoomLevel: null,
+  tileSize: 256,
+  references: [],
+  authors: [],
+  creationDate: null,
+  modificationDates: [],
+  minZoom: 2,
+  maxZoom: 9,
+};
diff --git a/src/redux/backgrounds/background.selectors.ts b/src/redux/backgrounds/background.selectors.ts
index 596301e1..7c3d5924 100644
--- a/src/redux/backgrounds/background.selectors.ts
+++ b/src/redux/backgrounds/background.selectors.ts
@@ -36,3 +36,12 @@ export const currentBackgroundImagePathSelector = createSelector(
   currentBackgroundImageSelector,
   image => (image ? image.path : ''),
 );
+
+const EMPTY_BACKGROUND_NAME = 'Empty';
+
+export const emptyBackgroundIdSelector = createSelector(backgroundsDataSelector, backgrounds => {
+  const emptyBackground = backgrounds?.find(
+    background => background.name === EMPTY_BACKGROUND_NAME,
+  );
+  return emptyBackground?.id;
+});
diff --git a/src/redux/configuration/configuration.selectors.ts b/src/redux/configuration/configuration.selectors.ts
index cb54cee2..7a694a44 100644
--- a/src/redux/configuration/configuration.selectors.ts
+++ b/src/redux/configuration/configuration.selectors.ts
@@ -6,6 +6,7 @@ import {
   MIN_COLOR_VAL_NAME_ID,
   NEUTRAL_COLOR_VAL_NAME_ID,
   OVERLAY_OPACITY_NAME_ID,
+  SIMPLE_COLOR_VAL_NAME_ID,
 } from './configuration.constants';
 
 const configurationSelector = createSelector(rootSelector, state => state.configuration);
@@ -31,3 +32,8 @@ export const overlayOpacitySelector = createSelector(
   configurationSelector,
   state => configurationAdapterSelectors.selectById(state, OVERLAY_OPACITY_NAME_ID)?.value,
 );
+
+export const simpleColorValSelector = createSelector(
+  configurationSelector,
+  state => configurationAdapterSelectors.selectById(state, SIMPLE_COLOR_VAL_NAME_ID)?.value,
+);
diff --git a/src/redux/overlayBioEntity/overlayBioEntity.utils.ts b/src/redux/overlayBioEntity/overlayBioEntity.utils.ts
index b4231f70..b875e1bb 100644
--- a/src/redux/overlayBioEntity/overlayBioEntity.utils.ts
+++ b/src/redux/overlayBioEntity/overlayBioEntity.utils.ts
@@ -18,6 +18,7 @@ export const parseOverlayBioEntityToOlRenderingFormat = (
         height: entity.left.height,
         value: entity.right.value,
         overlayId,
+        color: entity.right.color,
       });
     }
     return acc;
diff --git a/src/types/OLrendering.ts b/src/types/OLrendering.ts
index 71ce6cdb..3a651659 100644
--- a/src/types/OLrendering.ts
+++ b/src/types/OLrendering.ts
@@ -1,3 +1,5 @@
+import { Color } from './models';
+
 export type OverlayBioEntityRender = {
   id: number;
   modelId: number;
@@ -9,4 +11,5 @@ export type OverlayBioEntityRender = {
   height: number;
   value: number | null;
   overlayId: number;
+  color: Color | null;
 };
diff --git a/src/types/models.ts b/src/types/models.ts
index 885c7fa7..017169d8 100644
--- a/src/types/models.ts
+++ b/src/types/models.ts
@@ -2,6 +2,7 @@ import { bioEntityContentSchema } from '@/models/bioEntityContentSchema';
 import { bioEntityResponseSchema } from '@/models/bioEntityResponseSchema';
 import { bioEntitySchema } from '@/models/bioEntitySchema';
 import { chemicalSchema } from '@/models/chemicalSchema';
+import { colorSchema } from '@/models/colorSchema';
 import { configurationOptionSchema } from '@/models/configurationOptionSchema';
 import { disease } from '@/models/disease';
 import { drugSchema } from '@/models/drugSchema';
@@ -39,3 +40,4 @@ export type ElementSearchResult = z.infer<typeof elementSearchResult>;
 export type ElementSearchResultType = z.infer<typeof elementSearchResultType>;
 export type ConfigurationOption = z.infer<typeof configurationOptionSchema>;
 export type OverlayBioEntity = z.infer<typeof overlayBioEntitySchema>;
+export type Color = z.infer<typeof colorSchema>;
diff --git a/src/utils/convert/getHexTricolorGradientColorWithAlpha.ts b/src/utils/convert/getHexTricolorGradientColorWithAlpha.ts
index d142a465..eacefc8d 100644
--- a/src/utils/convert/getHexTricolorGradientColorWithAlpha.ts
+++ b/src/utils/convert/getHexTricolorGradientColorWithAlpha.ts
@@ -1,3 +1,4 @@
+import { WHITE_HEX_OPACITY_0 } from '@/constants/hexColors';
 import { interpolateThreeColors } from '../lerp/interpolateThreeColors';
 import { addAlphaToHexString } from './addAlphaToHexString';
 import { hexToRgb } from './hexToRgb';
@@ -11,8 +12,6 @@ export type GetHexTricolorGradientColorWithAlphaProps = {
   position: number;
 };
 
-const WHITE_HEX_OPACITY_0 = '#00000000';
-
 export const getHexTricolorGradientColorWithAlpha = ({
   leftColor,
   middleColor,
diff --git a/src/utils/convert/hexToRgb.test.ts b/src/utils/convert/hexToRgb.test.ts
index dcd10d20..4801c8cc 100644
--- a/src/utils/convert/hexToRgb.test.ts
+++ b/src/utils/convert/hexToRgb.test.ts
@@ -3,7 +3,7 @@ import { expandHexToFullFormatIfItsShorthanded, hexToRgb } from './hexToRgb';
 describe('expandHexToFullFormatIfItsShorthanded', () => {
   it('should expand short-handed hex string to full format', () => {
     const result = expandHexToFullFormatIfItsShorthanded('#abc');
-    expect(result).toBe('aabbcc');
+    expect(result).toBe('#aabbcc');
   });
 
   it('should not modify full-format hex string', () => {
@@ -13,12 +13,12 @@ describe('expandHexToFullFormatIfItsShorthanded', () => {
 
   it('should handle hex string without leading #', () => {
     const result = expandHexToFullFormatIfItsShorthanded('abc');
-    expect(result).toBe('aabbcc');
+    expect(result).toBe('#aabbcc');
   });
 
   it('should return original string if it does not match short-hand regex', () => {
     const result = expandHexToFullFormatIfItsShorthanded('invalid');
-    expect(result).toBe('invalid');
+    expect(result).toBe('#invalid');
   });
 });
 
diff --git a/src/utils/convert/hexToRgb.ts b/src/utils/convert/hexToRgb.ts
index 316c0c64..e9b4544b 100644
--- a/src/utils/convert/hexToRgb.ts
+++ b/src/utils/convert/hexToRgb.ts
@@ -4,7 +4,10 @@ export const expandHexToFullFormatIfItsShorthanded = (hexString: string): string
   const fullHexString = hexString.replace(SHORT_HAND_REGEX, (m, r, g, b) => {
     return r + r + g + g + b + b;
   });
-  return fullHexString;
+  const fullHexStringWithPrefix = fullHexString.startsWith('#')
+    ? fullHexString
+    : `#${fullHexString}`;
+  return fullHexStringWithPrefix;
 };
 
 const FULL_HEX_REGEX = /^#?([a-f\d]{2})([a-f\d]{2})([a-f\d]{2})$/i;
diff --git a/todo.txt b/todo.txt
deleted file mode 100644
index e5ee10e4..00000000
--- a/todo.txt
+++ /dev/null
@@ -1,14 +0,0 @@
-- poinformować zespół ze w czw się spóźnię 
-- Check adrian PRs 
-- ADD ALPHA to hex color 
-- render many squares 
-- include case with rgb and value fields 
-- create store, add dummy data 
-- on button click display dummy data 
-- exclude rendering reaction for it 
-- push PR 
-- Fetch data, organise structure for many overlays and turning on and off 
-- add few overlays for element
-- push PR
-- add reaction overlays 
-- push PR 
\ No newline at end of file
-- 
GitLab