From d4c2c2f32f153bd13c4b97d921bf69b9d382f8a0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Adrian=20Or=C5=82=C3=B3w?= <adrian.orlow@fishbrain.com> Date: Wed, 21 Feb 2024 20:23:45 +0100 Subject: [PATCH] feat: add drugs chemicals search for target --- .../BioEntityDrawer.component.test.tsx | 189 +++++++++++++++++- .../BioEntityDrawer.component.tsx | 35 +++- .../ChemicalsList.component.test.tsx | 141 +++++++++++++ .../ChemicalsList/ChemicalsList.component.tsx | 23 +++ .../BioEntityDrawer/ChemicalsList/index.ts | 1 + .../DrugsList/DrugsList.component.test.tsx | 141 +++++++++++++ .../DrugsList/DrugsList.component.tsx | 23 +++ .../Drawer/BioEntityDrawer/DrugsList/index.ts | 1 + .../CollapsibleSection.component.tsx | 34 +++- .../BioEntitiesPinsList.component.test.tsx | 13 +- ...BioEntitiesPinsListItem.component.test.tsx | 10 +- .../BioEntitiesPinsListItem.component.tsx | 37 ++-- .../BioEntitiesPinsListItem.types.ts | 7 + src/constants/fetchData.ts | 8 + src/models/targetSearchNameResult.ts | 5 + src/redux/apiPath.ts | 4 + src/redux/drawer/drawer.constants.ts | 5 +- src/redux/drawer/drawer.reducers.ts | 62 +++++- src/redux/drawer/drawer.selectors.ts | 19 ++ src/redux/drawer/drawer.slice.ts | 12 +- src/redux/drawer/drawer.thunks.ts | 77 +++++++ src/redux/drawer/drawer.types.ts | 3 + src/types/fetchDataState.ts | 2 + src/types/models.ts | 2 + 24 files changed, 801 insertions(+), 53 deletions(-) create mode 100644 src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.test.tsx create mode 100644 src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.tsx create mode 100644 src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/index.ts create mode 100644 src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.test.tsx create mode 100644 src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.tsx create mode 100644 src/components/Map/Drawer/BioEntityDrawer/DrugsList/index.ts create mode 100644 src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.types.ts create mode 100644 src/constants/fetchData.ts create mode 100644 src/models/targetSearchNameResult.ts create mode 100644 src/redux/drawer/drawer.thunks.ts diff --git a/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.test.tsx b/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.test.tsx index dbff8087..b2fad199 100644 --- a/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.test.tsx +++ b/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.test.tsx @@ -1,26 +1,31 @@ /* eslint-disable no-magic-numbers */ -import { - InitialStoreState, - getReduxWrapperWithStore, -} from '@/utils/testing/getReduxWrapperWithStore'; -import { render, screen } from '@testing-library/react'; -import { DRAWER_INITIAL_STATE } from '@/redux/drawer/drawer.constants'; -import { StoreType } from '@/redux/store'; +import { FIRST_ARRAY_ELEMENT } from '@/constants/common'; import { bioEntitiesContentFixture, bioEntityContentFixture, } from '@/models/fixtures/bioEntityContentsFixture'; -import { FIRST_ARRAY_ELEMENT } from '@/constants/common'; +import { MODELS_MOCK_SHORT } from '@/models/mocks/modelsMock'; import { BIOENTITY_INITIAL_STATE_MOCK, BIO_ENTITY_LINKING_TO_SUBMAP_DATA_MOCK, } from '@/redux/bioEntity/bioEntity.mock'; -import { MODELS_MOCK_SHORT } from '@/models/mocks/modelsMock'; +import { DRAWER_INITIAL_STATE } from '@/redux/drawer/drawer.constants'; import { MODELS_INITIAL_STATE_MOCK } from '@/redux/models/models.mock'; +import { INITIAL_STORE_STATE_MOCK } from '@/redux/root/root.fixtures'; +import { AppDispatch, RootState } from '@/redux/store'; +import { getReduxStoreWithActionsListener } from '@/utils/testing/getReduxStoreActionsListener'; +import { InitialStoreState } from '@/utils/testing/getReduxWrapperWithStore'; +import { act, render, screen } from '@testing-library/react'; +import { MockStoreEnhanced } from 'redux-mock-store'; import { BioEntityDrawer } from './BioEntityDrawer.component'; -const renderComponent = (initialStoreState: InitialStoreState = {}): { store: StoreType } => { - const { Wrapper, store } = getReduxWrapperWithStore(initialStoreState); +const renderComponent = ( + initialStoreState: InitialStoreState = {}, +): { store: MockStoreEnhanced<Partial<RootState>, AppDispatch> } => { + const { Wrapper, store } = getReduxStoreWithActionsListener({ + ...INITIAL_STORE_STATE_MOCK, + ...initialStoreState, + }); return ( render( @@ -80,6 +85,8 @@ describe('BioEntityDrawer - component', () => { ...DRAWER_INITIAL_STATE, bioEntityDrawerState: { bioentityId: bioEntity.id, + drugs: {}, + chemicals: {}, }, }, }); @@ -114,6 +121,8 @@ describe('BioEntityDrawer - component', () => { ...DRAWER_INITIAL_STATE, bioEntityDrawerState: { bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, }, }, }); @@ -148,6 +157,8 @@ describe('BioEntityDrawer - component', () => { ...DRAWER_INITIAL_STATE, bioEntityDrawerState: { bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, }, }, }); @@ -173,6 +184,8 @@ describe('BioEntityDrawer - component', () => { ...DRAWER_INITIAL_STATE, bioEntityDrawerState: { bioentityId: bioEntity.id, + drugs: {}, + chemicals: {}, }, }, }); @@ -194,6 +207,8 @@ describe('BioEntityDrawer - component', () => { ...DRAWER_INITIAL_STATE, bioEntityDrawerState: { bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, }, }, models: { @@ -204,5 +219,157 @@ describe('BioEntityDrawer - component', () => { expect(screen.getByTestId('associated-submap')).toBeInTheDocument(); }); + + it('should display chemicals list header', () => { + renderComponent({ + bioEntity: { + data: [ + { + searchQueryElement: '', + loading: 'succeeded', + error: { name: '', message: '' }, + data: [ + { + ...bioEntityContentFixture, + bioEntity: { + ...bioEntityContentFixture.bioEntity, + fullName: null, + }, + }, + ], + }, + ], + loading: 'succeeded', + error: { message: '', name: '' }, + }, + drawer: { + ...DRAWER_INITIAL_STATE, + bioEntityDrawerState: { + bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, + }, + }, + }); + + expect(screen.getByText('Drugs for target')).toBeInTheDocument(); + }); + + it('should display drugs list header', () => { + renderComponent({ + bioEntity: { + data: [ + { + searchQueryElement: '', + loading: 'succeeded', + error: { name: '', message: '' }, + data: [ + { + ...bioEntityContentFixture, + bioEntity: { + ...bioEntityContentFixture.bioEntity, + fullName: null, + }, + }, + ], + }, + ], + loading: 'succeeded', + error: { message: '', name: '' }, + }, + drawer: { + ...DRAWER_INITIAL_STATE, + bioEntityDrawerState: { + bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, + }, + }, + }); + + expect(screen.getByText('Chemicals for target', { exact: false })).toBeInTheDocument(); + }); + + it('should fetch drugs on drugs for target click', () => { + const { store } = renderComponent({ + bioEntity: { + data: [ + { + searchQueryElement: '', + loading: 'succeeded', + error: { name: '', message: '' }, + data: [ + { + ...bioEntityContentFixture, + bioEntity: { + ...bioEntityContentFixture.bioEntity, + fullName: null, + }, + }, + ], + }, + ], + loading: 'succeeded', + error: { message: '', name: '' }, + }, + drawer: { + ...DRAWER_INITIAL_STATE, + bioEntityDrawerState: { + bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, + }, + }, + }); + + const button = screen.getByText('Drugs for target', { exact: false }); + act(() => { + button.click(); + }); + + expect(store.getActions()[0].type).toBe('drawer/getDrugsForBioEntityDrawerTarget/pending'); + }); + + it('should fetch chemicals on chemicals for target click', () => { + const { store } = renderComponent({ + bioEntity: { + data: [ + { + searchQueryElement: '', + loading: 'succeeded', + error: { name: '', message: '' }, + data: [ + { + ...bioEntityContentFixture, + bioEntity: { + ...bioEntityContentFixture.bioEntity, + fullName: null, + }, + }, + ], + }, + ], + loading: 'succeeded', + error: { message: '', name: '' }, + }, + drawer: { + ...DRAWER_INITIAL_STATE, + bioEntityDrawerState: { + bioentityId: bioEntityContentFixture.bioEntity.id, + drugs: {}, + chemicals: {}, + }, + }, + }); + + const button = screen.getByText('Chemicals for target', { exact: false }); + act(() => { + button.click(); + }); + + expect(store.getActions()[0].type).toBe( + 'drawer/getChemicalsForBioEntityDrawerTarget/pending', + ); + }); }); }); diff --git a/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.tsx b/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.tsx index 6f3af872..1c37d3bd 100644 --- a/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.tsx +++ b/src/components/Map/Drawer/BioEntityDrawer/BioEntityDrawer.component.tsx @@ -1,13 +1,36 @@ import { ZERO } from '@/constants/common'; -import { searchedFromMapBioEntityElement } from '@/redux/bioEntity/bioEntity.selectors'; +import { + searchedFromMapBioEntityElement, + searchedFromMapBioEntityElementRelatedSubmapSelector, +} from '@/redux/bioEntity/bioEntity.selectors'; +import { + getChemicalsForBioEntityDrawerTarget, + getDrugsForBioEntityDrawerTarget, +} from '@/redux/drawer/drawer.thunks'; +import { useAppDispatch } from '@/redux/hooks/useAppDispatch'; import { useAppSelector } from '@/redux/hooks/useAppSelector'; import { DrawerHeading } from '@/shared/DrawerHeading'; +import { CollapsibleSection } from '../ExportDrawer/CollapsibleSection'; import { AnnotationItem } from './AnnotationItem'; import { AssociatedSubmap } from './AssociatedSubmap'; +import { ChemicalsList } from './ChemicalsList'; +import { DrugsList } from './DrugsList'; import { OverlayData } from './OverlayData'; +const TARGET_PREFIX: ElementSearchResultType = `ALIAS`; + export const BioEntityDrawer = (): React.ReactNode => { + const dispatch = useAppDispatch(); const bioEntityData = useAppSelector(searchedFromMapBioEntityElement); + const relatedSubmap = useAppSelector(searchedFromMapBioEntityElementRelatedSubmapSelector); + const currentTargetId = bioEntityData?.id ? `${TARGET_PREFIX}:${bioEntityData.id}` : ''; + + const fetchChemicalsForTarget = (): void => { + dispatch(getChemicalsForBioEntityDrawerTarget(currentTargetId)); + }; + const fetchDrugsForTarget = (): void => { + dispatch(getDrugsForBioEntityDrawerTarget(currentTargetId)); + }; if (!bioEntityData) { return null; @@ -48,6 +71,16 @@ export const BioEntityDrawer = (): React.ReactNode => { /> ))} <AssociatedSubmap /> + {!relatedSubmap && ( + <> + <CollapsibleSection title="Drugs for target" onOpened={fetchDrugsForTarget}> + <DrugsList /> + </CollapsibleSection> + <CollapsibleSection title="Chemicals for target" onOpened={fetchChemicalsForTarget}> + <ChemicalsList /> + </CollapsibleSection> + </> + )} <OverlayData /> </div> </div> diff --git a/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.test.tsx b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.test.tsx new file mode 100644 index 00000000..03adc8fd --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.test.tsx @@ -0,0 +1,141 @@ +import { DEFAULT_FETCH_DATA } from '@/constants/fetchData'; +import { chemicalsFixture } from '@/models/fixtures/chemicalsFixture'; +import { INITIAL_STORE_STATE_MOCK } from '@/redux/root/root.fixtures'; +import { StoreType } from '@/redux/store'; +import { InitialStoreState } from '@/utils/testing/getReduxStoreActionsListener'; +import { getReduxWrapperWithStore } from '@/utils/testing/getReduxWrapperWithStore'; +import { render, screen } from '@testing-library/react'; +import { ChemicalsList } from './ChemicalsList.component'; + +const renderComponent = (initialStoreState: InitialStoreState = {}): { store: StoreType } => { + const { Wrapper, store } = getReduxWrapperWithStore(initialStoreState); + + return ( + render( + <Wrapper> + <ChemicalsList /> + </Wrapper>, + ), + { + store, + } + ); +}; + +describe('ChemicalsList - component', () => { + describe('when chemicals data is loading', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: {}, + chemicals: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'pending', + data: [], + }, + }, + }, + }, + }); + }); + + it('should show loading indicator', () => { + expect(screen.getByAltText('spinner icon')).toBeInTheDocument(); + }); + }); + + describe('when chemicals data is empty', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: {}, + chemicals: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: [], + }, + }, + }, + }, + }); + }); + + it('should show text with info', () => { + expect(screen.getByText('List is empty')).toBeInTheDocument(); + }); + }); + + describe('when chemicals data is present and valid', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: {}, + chemicals: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: chemicalsFixture, + }, + }, + }, + }, + }); + }); + + it.each(chemicalsFixture)('should show bio entitity card', chemical => { + expect(screen.getByText(chemical.name)).toBeInTheDocument(); + }); + }); + + describe('when chemicals data is present but for different bio entity id', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: {}, + chemicals: { + '2137': { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: chemicalsFixture, + }, + }, + }, + }, + }); + }); + + it.each(chemicalsFixture)('should not show bio entitity card', chemical => { + expect(screen.queryByText(chemical.name)).not.toBeInTheDocument(); + }); + + it('should show text with info', () => { + expect(screen.getByText('List is empty')).toBeInTheDocument(); + }); + }); +}); diff --git a/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.tsx b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.tsx new file mode 100644 index 00000000..ffbacbd9 --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/ChemicalsList.component.tsx @@ -0,0 +1,23 @@ +import { currentSearchedBioEntityChemicalsSelector } from '@/redux/drawer/drawer.selectors'; +import { useAppSelector } from '@/redux/hooks/useAppSelector'; +import { LoadingIndicator } from '@/shared/LoadingIndicator'; +import { ZERO } from '@/constants/common'; +import { BioEntitiesPinsListItem } from '../../SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem'; + +export const ChemicalsList = (): JSX.Element => { + const chemicals = useAppSelector(currentSearchedBioEntityChemicalsSelector); + const chemicalsData = chemicals.data || []; + + if (chemicals.loading === 'pending') { + return <LoadingIndicator />; + } + + return ( + <div> + {chemicalsData.map(drug => ( + <BioEntitiesPinsListItem key={`${drug.id}`} pin={drug} name={drug.name} /> + ))} + {chemicalsData.length === ZERO && 'List is empty'} + </div> + ); +}; diff --git a/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/index.ts b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/index.ts new file mode 100644 index 00000000..ee38f9ad --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/ChemicalsList/index.ts @@ -0,0 +1 @@ +export { ChemicalsList } from './ChemicalsList.component'; diff --git a/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.test.tsx b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.test.tsx new file mode 100644 index 00000000..0ec503c4 --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.test.tsx @@ -0,0 +1,141 @@ +import { DEFAULT_FETCH_DATA } from '@/constants/fetchData'; +import { drugsFixture } from '@/models/fixtures/drugFixtures'; +import { INITIAL_STORE_STATE_MOCK } from '@/redux/root/root.fixtures'; +import { StoreType } from '@/redux/store'; +import { InitialStoreState } from '@/utils/testing/getReduxStoreActionsListener'; +import { getReduxWrapperWithStore } from '@/utils/testing/getReduxWrapperWithStore'; +import { render, screen } from '@testing-library/react'; +import { DrugsList } from './DrugsList.component'; + +const renderComponent = (initialStoreState: InitialStoreState = {}): { store: StoreType } => { + const { Wrapper, store } = getReduxWrapperWithStore(initialStoreState); + + return ( + render( + <Wrapper> + <DrugsList /> + </Wrapper>, + ), + { + store, + } + ); +}; + +describe('DrugsList - component', () => { + describe('when drugs data is loading', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'pending', + data: [], + }, + }, + chemicals: {}, + }, + }, + }); + }); + + it('should show loading indicator', () => { + expect(screen.getByAltText('spinner icon')).toBeInTheDocument(); + }); + }); + + describe('when drugs data is empty', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: [], + }, + }, + chemicals: {}, + }, + }, + }); + }); + + it('should show text with info', () => { + expect(screen.getByText('List is empty')).toBeInTheDocument(); + }); + }); + + describe('when drugs data is present and valid', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: { + [`${bioEntityId}`]: { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: drugsFixture, + }, + }, + chemicals: {}, + }, + }, + }); + }); + + it.each(drugsFixture)('should show bio entitity card', drug => { + expect(screen.getByText(drug.name)).toBeInTheDocument(); + }); + }); + + describe('when drugs data is present but for different bio entity id', () => { + beforeEach(() => { + const bioEntityId = 5000; + + renderComponent({ + drawer: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioEntityDrawerState: { + ...INITIAL_STORE_STATE_MOCK.drawer, + bioentityId: bioEntityId, + drugs: { + '2137': { + ...DEFAULT_FETCH_DATA, + loading: 'succeeded', + data: drugsFixture, + }, + }, + chemicals: {}, + }, + }, + }); + }); + + it.each(drugsFixture)('should not show bio entitity card', drug => { + expect(screen.queryByText(drug.name)).not.toBeInTheDocument(); + }); + + it('should show text with info', () => { + expect(screen.getByText('List is empty')).toBeInTheDocument(); + }); + }); +}); diff --git a/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.tsx b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.tsx new file mode 100644 index 00000000..998a0a77 --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/DrugsList.component.tsx @@ -0,0 +1,23 @@ +import { ZERO } from '@/constants/common'; +import { currentSearchedBioEntityDrugsSelector } from '@/redux/drawer/drawer.selectors'; +import { useAppSelector } from '@/redux/hooks/useAppSelector'; +import { LoadingIndicator } from '@/shared/LoadingIndicator'; +import { BioEntitiesPinsListItem } from '../../SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem'; + +export const DrugsList = (): JSX.Element => { + const drugs = useAppSelector(currentSearchedBioEntityDrugsSelector); + const drugsData = drugs.data || []; + + if (drugs.loading === 'pending') { + return <LoadingIndicator />; + } + + return ( + <div> + {drugsData.map(drug => ( + <BioEntitiesPinsListItem key={`${drug.id}`} pin={drug} name={drug.name} /> + ))} + {drugsData.length === ZERO && 'List is empty'} + </div> + ); +}; diff --git a/src/components/Map/Drawer/BioEntityDrawer/DrugsList/index.ts b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/index.ts new file mode 100644 index 00000000..1389bcc4 --- /dev/null +++ b/src/components/Map/Drawer/BioEntityDrawer/DrugsList/index.ts @@ -0,0 +1 @@ +export { DrugsList } from './DrugsList.component'; diff --git a/src/components/Map/Drawer/ExportDrawer/CollapsibleSection/CollapsibleSection.component.tsx b/src/components/Map/Drawer/ExportDrawer/CollapsibleSection/CollapsibleSection.component.tsx index b0d478bb..2dc750a9 100644 --- a/src/components/Map/Drawer/ExportDrawer/CollapsibleSection/CollapsibleSection.component.tsx +++ b/src/components/Map/Drawer/ExportDrawer/CollapsibleSection/CollapsibleSection.component.tsx @@ -1,3 +1,4 @@ +import { ZERO } from '@/constants/common'; import { Accordion, AccordionItem, @@ -5,22 +6,35 @@ import { AccordionItemHeading, AccordionItemPanel, } from '@/shared/Accordion'; +import { ID } from 'react-accessible-accordion/dist/types/components/ItemContext'; type CollapsibleSectionProps = { title: string; children: React.ReactNode; + onOpened?(): void; }; export const CollapsibleSection = ({ title, children, -}: CollapsibleSectionProps): React.ReactNode => ( - <Accordion allowZeroExpanded> - <AccordionItem> - <AccordionItemHeading> - <AccordionItemButton>{title}</AccordionItemButton> - </AccordionItemHeading> - <AccordionItemPanel>{children}</AccordionItemPanel> - </AccordionItem> - </Accordion> -); + onOpened, +}: CollapsibleSectionProps): React.ReactNode => { + const handleOnChange = (ids: ID[]): void => { + const hasBeenOpened = ids.length > ZERO; + + if (hasBeenOpened && onOpened) { + onOpened(); + } + }; + + return ( + <Accordion allowZeroExpanded onChange={handleOnChange}> + <AccordionItem> + <AccordionItemHeading> + <AccordionItemButton>{title}</AccordionItemButton> + </AccordionItemHeading> + <AccordionItemPanel>{children}</AccordionItemPanel> + </AccordionItem> + </Accordion> + ); +}; diff --git a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsList.component.test.tsx b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsList.component.test.tsx index a03b5b4c..32311422 100644 --- a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsList.component.test.tsx +++ b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsList.component.test.tsx @@ -1,12 +1,12 @@ /* eslint-disable no-magic-numbers */ -import { render, screen } from '@testing-library/react'; +import { bioEntitiesContentFixture } from '@/models/fixtures/bioEntityContentsFixture'; +import { StoreType } from '@/redux/store'; +import { BioEntityContent } from '@/types/models'; import { InitialStoreState, getReduxWrapperWithStore, } from '@/utils/testing/getReduxWrapperWithStore'; -import { bioEntitiesContentFixture } from '@/models/fixtures/bioEntityContentsFixture'; -import { StoreType } from '@/redux/store'; -import { BioEntityContent } from '@/types/models'; +import { render, screen } from '@testing-library/react'; import { BioEntitiesPinsList } from './BioEntitiesPinsList.component'; const renderComponent = ( @@ -30,7 +30,10 @@ const renderComponent = ( describe('BioEntitiesPinsList - component ', () => { it('should display list of bio entites elements', () => { renderComponent(bioEntitiesContentFixture); + const bioEntitiesWithFullName = bioEntitiesContentFixture.filter(({ bioEntity }) => + Boolean(bioEntity.fullName), + ); - expect(screen.getAllByTestId('bio-entity-name')).toHaveLength(10); + expect(screen.getAllByTestId('bio-entity-name')).toHaveLength(bioEntitiesWithFullName.length); }); }); diff --git a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.test.tsx b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.test.tsx index 9c245a24..6049a6e2 100644 --- a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.test.tsx +++ b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.test.tsx @@ -1,12 +1,12 @@ /* eslint-disable no-magic-numbers */ -import { render, screen } from '@testing-library/react'; +import { bioEntitiesContentFixture } from '@/models/fixtures/bioEntityContentsFixture'; +import { StoreType } from '@/redux/store'; +import { BioEntity } from '@/types/models'; import { InitialStoreState, getReduxWrapperWithStore, } from '@/utils/testing/getReduxWrapperWithStore'; -import { bioEntitiesContentFixture } from '@/models/fixtures/bioEntityContentsFixture'; -import { StoreType } from '@/redux/store'; -import { BioEntity } from '@/types/models'; +import { render, screen } from '@testing-library/react'; import { BioEntitiesPinsListItem } from './BioEntitiesPinsListItem.component'; const BIO_ENTITY = bioEntitiesContentFixture[0].bioEntity; @@ -53,7 +53,7 @@ describe('BioEntitiesPinsListItem - component ', () => { renderComponent(bioEntity.name, bioEntity); - expect(screen.getAllByTestId('bio-entity-symbol')[0].textContent).toHaveLength(0); + expect(screen.queryAllByTestId('bio-entity-symbol')).toHaveLength(0); }); it('should display string type of bio entity element', () => { renderComponent(BIO_ENTITY.name, BIO_ENTITY); diff --git a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.tsx b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.tsx index c4322112..8dc30490 100644 --- a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.tsx +++ b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.component.tsx @@ -1,11 +1,11 @@ -import { twMerge } from 'tailwind-merge'; import { Icon } from '@/shared/Icon'; -import { BioEntity } from '@/types/models'; +import { twMerge } from 'tailwind-merge'; import { getPinColor } from '../../../ResultsList/PinsList/PinsListItem/PinsListItem.component.utils'; +import { PinListBioEntity } from './BioEntitiesPinsListItem.types'; interface BioEntitiesPinsListItemProps { name: string; - pin: BioEntity; + pin: PinListBioEntity; } export const BioEntitiesPinsListItem = ({ @@ -17,21 +17,26 @@ export const BioEntitiesPinsListItem = ({ <div className="flex w-full flex-row items-center gap-2"> <Icon name="pin" className={twMerge('mr-2 shrink-0', getPinColor('bioEntity'))} /> <p> - {pin.stringType}: <span className="w-full font-bold">{name}</span> + {pin.stringType ? `${pin.stringType}: ` : ''} + <span className="w-full font-bold">{name}</span> </p> </div> - <p className="font-bold leading-6"> - Full name:{' '} - <span className="w-full font-normal" data-testid="bio-entity-name"> - {pin.fullName || ``} - </span> - </p> - <p className="font-bold leading-6"> - Symbol:{' '} - <span className="w-full font-normal" data-testid="bio-entity-symbol"> - {pin.symbol || ``} - </span> - </p> + {pin.fullName && ( + <p className="font-bold leading-6"> + Full name:{' '} + <span className="w-full font-normal" data-testid="bio-entity-name"> + {pin.fullName} + </span> + </p> + )} + {pin.symbol && ( + <p className="font-bold leading-6"> + Symbol:{' '} + <span className="w-full font-normal" data-testid="bio-entity-symbol"> + {pin.symbol} + </span> + </p> + )} <p className="font-bold leading-6"> Synonyms: <span className="w-full font-normal">{pin.synonyms.join(', ')}</span> </p> diff --git a/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.types.ts b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.types.ts new file mode 100644 index 00000000..03cfcae8 --- /dev/null +++ b/src/components/Map/Drawer/SearchDrawerWrapper/BioEntitiesResultsList/BioEntitiesPinsList/BioEntitiesPinsListItem/BioEntitiesPinsListItem.types.ts @@ -0,0 +1,7 @@ +import { BioEntity } from '@/types/models'; + +export type PinListBioEntity = Pick<BioEntity, 'synonyms' | 'references'> & { + symbol?: BioEntity['symbol']; + stringType?: BioEntity['stringType']; + fullName?: BioEntity['fullName']; +}; diff --git a/src/constants/fetchData.ts b/src/constants/fetchData.ts new file mode 100644 index 00000000..692b2f95 --- /dev/null +++ b/src/constants/fetchData.ts @@ -0,0 +1,8 @@ +import { FetchDataState } from '@/types/fetchDataState'; +import { DEFAULT_ERROR } from './errors'; + +export const DEFAULT_FETCH_DATA: FetchDataState<[]> = { + data: [], + loading: 'idle', + error: DEFAULT_ERROR, +}; diff --git a/src/models/targetSearchNameResult.ts b/src/models/targetSearchNameResult.ts new file mode 100644 index 00000000..721a63eb --- /dev/null +++ b/src/models/targetSearchNameResult.ts @@ -0,0 +1,5 @@ +import { z } from 'zod'; + +export const targetSearchNameResult = z.object({ + name: z.string(), +}); diff --git a/src/redux/apiPath.ts b/src/redux/apiPath.ts index ac815f25..5051aa2a 100644 --- a/src/redux/apiPath.ts +++ b/src/redux/apiPath.ts @@ -36,6 +36,10 @@ export const apiPath = { `projects/${PROJECT_ID}/models/*/bioEntities/reactions/?id=${ids.join(',')}&size=1000`, getDrugsStringWithQuery: (searchQuery: string): string => `projects/${PROJECT_ID}/drugs:search?query=${searchQuery}`, + getDrugsStringWithColumnsTarget: (columns: string, target: string): string => + `projects/${PROJECT_ID}/drugs:search?columns=${columns}&target=${target}`, + getChemicalsStringWithColumnsTarget: (columns: string, target: string): string => + `projects/${PROJECT_ID}/chemicals:search?columns=${columns}&target=${target}`, getModelsString: (): string => `projects/${PROJECT_ID}/models/`, getChemicalsStringWithQuery: (searchQuery: string): string => `projects/${PROJECT_ID}/chemicals:search?query=${searchQuery}`, diff --git a/src/redux/drawer/drawer.constants.ts b/src/redux/drawer/drawer.constants.ts index f1035f97..33890a23 100644 --- a/src/redux/drawer/drawer.constants.ts +++ b/src/redux/drawer/drawer.constants.ts @@ -11,7 +11,10 @@ export const DRAWER_INITIAL_STATE: DrawerState = { selectedSearchElement: '', }, reactionDrawerState: {}, - bioEntityDrawerState: {}, + bioEntityDrawerState: { + drugs: {}, + chemicals: {}, + }, overlayDrawerState: { currentStep: 0, }, diff --git a/src/redux/drawer/drawer.reducers.ts b/src/redux/drawer/drawer.reducers.ts index 8a60b60f..3a72aa53 100644 --- a/src/redux/drawer/drawer.reducers.ts +++ b/src/redux/drawer/drawer.reducers.ts @@ -6,7 +6,11 @@ import type { OpenSearchDrawerWithSelectedTabReducerAction, } from '@/redux/drawer/drawer.types'; import type { DrawerName } from '@/types/drawerName'; -import type { PayloadAction } from '@reduxjs/toolkit'; +import type { ActionReducerMapBuilder, PayloadAction } from '@reduxjs/toolkit'; +import { + getChemicalsForBioEntityDrawerTarget, + getDrugsForBioEntityDrawerTarget, +} from './drawer.thunks'; export const openDrawerReducer = (state: DrawerState, action: PayloadAction<DrawerName>): void => { state.isOpen = true; @@ -109,3 +113,59 @@ export const openBioEntityDrawerByIdReducer = ( state.bioEntityDrawerState.bioentityId = action.payload; state.searchDrawerState.selectedSearchElement = action.payload.toString(); }; + +export const getBioEntityDrugsForTargetReducers = ( + builder: ActionReducerMapBuilder<DrawerState>, +): void => { + builder.addCase(getDrugsForBioEntityDrawerTarget.pending, state => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.drugs[bioEntityId] = { + ...state.bioEntityDrawerState.drugs[bioEntityId], + loading: 'pending', + }; + }); + builder.addCase(getDrugsForBioEntityDrawerTarget.fulfilled, (state, action) => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.drugs[bioEntityId] = { + ...state.bioEntityDrawerState.drugs[bioEntityId], + data: action.payload || [], + loading: 'succeeded', + }; + }); + builder.addCase(getDrugsForBioEntityDrawerTarget.rejected, state => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.drugs[bioEntityId] = { + ...state.bioEntityDrawerState.drugs[bioEntityId], + loading: 'failed', + }; + // TODO to discuss manage state of failure + }); +}; + +export const getBioEntityChemicalsForTargetReducers = ( + builder: ActionReducerMapBuilder<DrawerState>, +): void => { + builder.addCase(getChemicalsForBioEntityDrawerTarget.pending, state => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.chemicals[bioEntityId] = { + ...state.bioEntityDrawerState.chemicals[bioEntityId], + loading: 'pending', + }; + }); + builder.addCase(getChemicalsForBioEntityDrawerTarget.fulfilled, (state, action) => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.chemicals[bioEntityId] = { + ...state.bioEntityDrawerState.chemicals[bioEntityId], + data: action.payload || [], + loading: 'succeeded', + }; + }); + builder.addCase(getChemicalsForBioEntityDrawerTarget.rejected, state => { + const bioEntityId = state.bioEntityDrawerState.bioentityId || ''; + state.bioEntityDrawerState.chemicals[bioEntityId] = { + ...state.bioEntityDrawerState.chemicals[bioEntityId], + loading: 'failed', + }; + // TODO to discuss manage state of failure + }); +}; diff --git a/src/redux/drawer/drawer.selectors.ts b/src/redux/drawer/drawer.selectors.ts index 26e9a90b..30d9831c 100644 --- a/src/redux/drawer/drawer.selectors.ts +++ b/src/redux/drawer/drawer.selectors.ts @@ -1,3 +1,4 @@ +import { DEFAULT_FETCH_DATA } from '@/constants/fetchData'; import { rootSelector } from '@/redux/root/root.selectors'; import { assertNever } from '@/utils/assertNever'; import { createSelector } from '@reduxjs/toolkit'; @@ -46,6 +47,24 @@ export const currentSearchedBioEntityId = createSelector( state => state.bioentityId, ); +export const currentSearchedBioEntityDrugsSelector = createSelector( + bioEntityDrawerStateSelector, + currentSearchedBioEntityId, + (state, currentBioEntityId) => + currentBioEntityId && state.drugs[currentBioEntityId] + ? state.drugs[currentBioEntityId] + : DEFAULT_FETCH_DATA, +); + +export const currentSearchedBioEntityChemicalsSelector = createSelector( + bioEntityDrawerStateSelector, + currentSearchedBioEntityId, + (state, currentBioEntityId) => + currentBioEntityId && state.chemicals[currentBioEntityId] + ? state.chemicals[currentBioEntityId] + : DEFAULT_FETCH_DATA, +); + export const resultListSelector = createSelector( rootSelector, currentStepTypeSelector, diff --git a/src/redux/drawer/drawer.slice.ts b/src/redux/drawer/drawer.slice.ts index 98e073ae..99d928db 100644 --- a/src/redux/drawer/drawer.slice.ts +++ b/src/redux/drawer/drawer.slice.ts @@ -1,21 +1,23 @@ import { createSlice } from '@reduxjs/toolkit'; +import { DRAWER_INITIAL_STATE } from './drawer.constants'; import { closeDrawerReducer, + displayAddOverlaysDrawerReducer, displayBioEntitiesListReducer, displayChemicalsListReducer, displayDrugsListReducer, displayEntityDetailsReducer, displayGroupedSearchResultsReducer, + getBioEntityChemicalsForTargetReducers, + getBioEntityDrugsForTargetReducers, + openBioEntityDrawerByIdReducer, openDrawerReducer, openOverlaysDrawerReducer, - openBioEntityDrawerByIdReducer, openReactionDrawerByIdReducer, openSearchDrawerWithSelectedTabReducer, openSubmapsDrawerReducer, selectTabReducer, - displayAddOverlaysDrawerReducer, } from './drawer.reducers'; -import { DRAWER_INITIAL_STATE } from './drawer.constants'; const drawerSlice = createSlice({ name: 'drawer', @@ -36,6 +38,10 @@ const drawerSlice = createSlice({ openReactionDrawerById: openReactionDrawerByIdReducer, openBioEntityDrawerById: openBioEntityDrawerByIdReducer, }, + extraReducers: builder => { + getBioEntityDrugsForTargetReducers(builder); + getBioEntityChemicalsForTargetReducers(builder); + }, }); export const { diff --git a/src/redux/drawer/drawer.thunks.ts b/src/redux/drawer/drawer.thunks.ts new file mode 100644 index 00000000..7e60536f --- /dev/null +++ b/src/redux/drawer/drawer.thunks.ts @@ -0,0 +1,77 @@ +import { chemicalSchema } from '@/models/chemicalSchema'; +import { drugSchema } from '@/models/drugSchema'; +import { targetSearchNameResult } from '@/models/targetSearchNameResult'; +import { axiosInstanceNewAPI } from '@/services/api/utils/axiosInstance'; +import { Chemical, Drug, TargetSearchNameResult } from '@/types/models'; +import { validateDataUsingZodSchema } from '@/utils/validateDataUsingZodSchema'; +import { createAsyncThunk } from '@reduxjs/toolkit'; +import { apiPath } from '../apiPath'; + +const QUERY_COLUMN_NAME = 'name'; + +const getDrugsNamesForTarget = async (target: string): Promise<TargetSearchNameResult[]> => { + const response = await axiosInstanceNewAPI.get<TargetSearchNameResult[]>( + apiPath.getDrugsStringWithColumnsTarget(QUERY_COLUMN_NAME, target), + ); + + const isDataValid = (obj: unknown): obj is TargetSearchNameResult => + validateDataUsingZodSchema(obj, targetSearchNameResult); + + return response.data.filter(isDataValid); +}; + +const getDrugsByName = async (drugName: string): Promise<Drug[]> => { + const response = await axiosInstanceNewAPI.get<Drug[]>(apiPath.getDrugsStringWithQuery(drugName)); + + const isDataValid = (obj: unknown): obj is Drug => validateDataUsingZodSchema(obj, drugSchema); + + return response.data.filter(isDataValid); +}; + +export const getDrugsForBioEntityDrawerTarget = createAsyncThunk( + 'drawer/getDrugsForBioEntityDrawerTarget', + async (target: string): Promise<Drug[]> => { + const drugsNames = await getDrugsNamesForTarget(target); + const drugsArrays = await Promise.all( + drugsNames.map(({ name }) => getDrugsByName(encodeURIComponent(name))), + ); + const drugs = drugsArrays.flat(); + + return drugs; + }, +); + +const getChemicalsNamesForTarget = async (target: string): Promise<TargetSearchNameResult[]> => { + const response = await axiosInstanceNewAPI.get<TargetSearchNameResult[]>( + apiPath.getChemicalsStringWithColumnsTarget(QUERY_COLUMN_NAME, target), + ); + + const isDataValid = (obj: unknown): obj is TargetSearchNameResult => + validateDataUsingZodSchema(obj, targetSearchNameResult); + + return response.data.filter(isDataValid); +}; + +const getChemicalsByName = async (chemicalName: string): Promise<Chemical[]> => { + const response = await axiosInstanceNewAPI.get<Chemical[]>( + apiPath.getChemicalsStringWithQuery(chemicalName), + ); + + const isDataValid = (obj: unknown): obj is Chemical => + validateDataUsingZodSchema(obj, chemicalSchema); + + return response.data.filter(isDataValid); +}; + +export const getChemicalsForBioEntityDrawerTarget = createAsyncThunk( + 'drawer/getChemicalsForBioEntityDrawerTarget', + async (target: string): Promise<Chemical[]> => { + const chemicalsNames = await getChemicalsNamesForTarget(target); + const chemicalsArrays = await Promise.all( + chemicalsNames.map(({ name }) => getChemicalsByName(encodeURIComponent(name))), + ); + const chemicals = chemicalsArrays.flat(); + + return chemicals; + }, +); diff --git a/src/redux/drawer/drawer.types.ts b/src/redux/drawer/drawer.types.ts index 075fcd19..f4104f25 100644 --- a/src/redux/drawer/drawer.types.ts +++ b/src/redux/drawer/drawer.types.ts @@ -1,4 +1,5 @@ import type { DrawerName } from '@/types/drawerName'; +import { KeyedFetchDataState } from '@/types/fetchDataState'; import { BioEntityContent, Chemical, Drug } from '@/types/models'; import { PayloadAction } from '@reduxjs/toolkit'; @@ -20,6 +21,8 @@ export type ReactionDrawerState = { export type BioEntityDrawerState = { bioentityId?: number; + drugs: KeyedFetchDataState<Drug[], []>; + chemicals: KeyedFetchDataState<Chemical[], []>; }; export type DrawerState = { diff --git a/src/types/fetchDataState.ts b/src/types/fetchDataState.ts index 0ee6719c..95131583 100644 --- a/src/types/fetchDataState.ts +++ b/src/types/fetchDataState.ts @@ -15,3 +15,5 @@ export type MultiFetchDataState<T> = { loading: Loading; error: Error; }; + +export type KeyedFetchDataState<T, T2 = undefined> = Record<string, FetchDataState<T, T2>>; diff --git a/src/types/models.ts b/src/types/models.ts index fbc40b49..df6237ad 100644 --- a/src/types/models.ts +++ b/src/types/models.ts @@ -48,6 +48,7 @@ import { referenceSchema } from '@/models/referenceSchema'; import { sessionSchemaValid } from '@/models/sessionValidSchema'; import { statisticsSchema } from '@/models/statisticsSchema'; import { targetSchema } from '@/models/targetSchema'; +import { targetSearchNameResult } from '@/models/targetSearchNameResult'; import { z } from 'zod'; export type Project = z.infer<typeof projectSchema>; @@ -96,3 +97,4 @@ export type ExportNetwork = z.infer<typeof exportNetworkchema>; export type ExportElements = z.infer<typeof exportElementsSchema>; export type MinervaPlugin = z.infer<typeof pluginSchema>; // Plugin type interfers with global Plugin type export type GeneVariant = z.infer<typeof geneVariant>; +export type TargetSearchNameResult = z.infer<typeof targetSearchNameResult>; -- GitLab