Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add React Hooks Linting #174

Merged
merged 39 commits into from
Aug 26, 2024
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
39 commits
Select commit Hold shift + click to select a range
ffef748
migrated previous code
SanjeevLakhwani Jul 3, 2024
354e897
removed redundant state variables
SanjeevLakhwani Jul 3, 2024
7549508
reorganized redux state
SanjeevLakhwani Jul 15, 2024
b15000f
corrected app load sequence
SanjeevLakhwani Jul 17, 2024
3eb72db
added routing for projects and datasets
SanjeevLakhwani Jul 22, 2024
df495da
improved index.tsx
SanjeevLakhwani Jul 25, 2024
0561f84
provided correct navigation for chart clicks
SanjeevLakhwani Jul 25, 2024
829df07
corrected allow clear button
SanjeevLakhwani Jul 25, 2024
9854b2e
removed redundant log
SanjeevLakhwani Jul 25, 2024
be24708
modified UI
SanjeevLakhwani Jul 29, 2024
b7b039a
Added dataset title
SanjeevLakhwani Jul 29, 2024
a55cd51
corrected token refresh import url
SanjeevLakhwani Aug 5, 2024
4600d38
removed React.FC
SanjeevLakhwani Aug 5, 2024
6e035c4
updated modal UI
SanjeevLakhwani Aug 5, 2024
922b637
changed db icon to svg
SanjeevLakhwani Aug 5, 2024
5e40e99
fixed compile issues with rebase
SanjeevLakhwani Aug 8, 2024
7233115
fixed index url redirect
SanjeevLakhwani Aug 8, 2024
772afed
Added transitions to on hover effects to match antd design
SanjeevLakhwani Aug 8, 2024
73adcdb
PR css changes
SanjeevLakhwani Aug 8, 2024
9200f5d
added translations
SanjeevLakhwani Aug 8, 2024
a58619c
removed stray )
SanjeevLakhwani Aug 19, 2024
1194e93
added handling for null edge case
SanjeevLakhwani Aug 19, 2024
d7a7c6e
Added type for selectedProject useState
SanjeevLakhwani Aug 19, 2024
ab86cfa
Typed Discovery
SanjeevLakhwani Aug 22, 2024
ed3101e
Removed React.FC
SanjeevLakhwani Aug 22, 2024
cee5e06
provide null option for type
SanjeevLakhwani Aug 22, 2024
86a634d
Removed React.useState
SanjeevLakhwani Aug 22, 2024
923d2e0
organized imports
SanjeevLakhwani Aug 22, 2024
fbbae9c
Added react hook linter
SanjeevLakhwani Aug 26, 2024
16edddb
resolved some hook linter errors
SanjeevLakhwani Aug 26, 2024
514fae4
enabled same page routing for ChooseProjectModal.tsx
SanjeevLakhwani Aug 26, 2024
0c879fa
resolved all react-hook-linter issues
SanjeevLakhwani Aug 26, 2024
289ee8d
Merge pull request #171 from bento-platform/features/new-discovery
SanjeevLakhwani Aug 26, 2024
ef7a0d3
Added react hook linter
SanjeevLakhwani Aug 26, 2024
4f8cc43
resolved some hook linter errors
SanjeevLakhwani Aug 26, 2024
8fa9607
resolved all react-hook-linter issues
SanjeevLakhwani Aug 26, 2024
3c6b10f
resolved all react-hook-linter issues after rebase
SanjeevLakhwani Aug 26, 2024
d3504a7
Merge remote-tracking branch 'origin/lint-react-hooks' into lint-reac…
SanjeevLakhwani Aug 26, 2024
52754d5
resolved issues with index.tsx
SanjeevLakhwani Aug 26, 2024
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
25 changes: 17 additions & 8 deletions .eslintrc.js
Original file line number Diff line number Diff line change
Expand Up @@ -13,21 +13,30 @@ module.exports = {
node: true,
},
plugins: ['react', '@typescript-eslint'],
extends: ['eslint:recommended', 'plugin:react/recommended', 'plugin:@typescript-eslint/recommended', 'plugin:prettier/recommended'],
extends: [
'eslint:recommended',
'plugin:react/recommended',
'plugin:@typescript-eslint/recommended',
'plugin:prettier/recommended',
'plugin:react-hooks/recommended',
],
rules: {
'react/prop-types': 'off', // disable prop-types since we're using TypeScript
'@typescript-eslint/explicit-module-boundary-types': 'off', // allow implicit return types
'@typescript-eslint/no-empty-interface': 'off', // allow empty interfaces
'no-unused-vars': 'off', // disable no-unused-vars since @typescript-eslint/no-unused-vars does the same
'@typescript-eslint/no-unused-vars': ['error', { argsIgnorePattern: '^_' }], // enable @typescript-eslint/no-unused-vars
'@typescript-eslint/no-non-null-assertion': 'off',
'comma-dangle': ['error', {
'arrays': 'always-multiline',
'objects': 'always-multiline',
'imports': 'always-multiline',
'exports': 'always-multiline',
}],
'semi': ['error', 'always'],
'comma-dangle': [
'error',
{
arrays: 'always-multiline',
objects: 'always-multiline',
imports: 'always-multiline',
exports: 'always-multiline',
},
],
semi: ['error', 'always'],
},
settings: {
react: {
Expand Down
98 changes: 13 additions & 85 deletions package-lock.json

Large diffs are not rendered by default.

1 change: 1 addition & 0 deletions package.json
Original file line number Diff line number Diff line change
Expand Up @@ -24,6 +24,7 @@
"bento-auth-js": "^5.1.1",
"bento-charts": "^2.6.8",
"dotenv": "^16.3.1",
"eslint-plugin-react-hooks": "^4.6.2",
"i18next": "^23.7.7",
"i18next-browser-languagedetector": "^8.0.0",
"i18next-http-backend": "^2.4.2",
Expand Down
13 changes: 9 additions & 4 deletions src/js/components/Beacon/BeaconQueryUi.tsx
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
import React, { useEffect, useState, ReactNode } from 'react';
import React, { useEffect, useState, ReactNode, useCallback, useMemo } from 'react';
import { useAppSelector, useAppDispatch, useTranslationDefault, useQueryWithAuthIfAllowed } from '@/hooks';
import { Button, Card, Col, Form, Row, Space, Tooltip, Typography } from 'antd';
import { InfoCircleOutlined } from '@ant-design/icons';
Expand Down Expand Up @@ -64,8 +64,13 @@ const BeaconQueryUi = () => {
const isAuthenticated = useIsAuthenticated();

const dispatch = useAppDispatch();
const launchEmptyQuery = () => dispatch(makeBeaconQuery(requestPayload({}, [])));
const formInitialValues = { 'Assembly ID': beaconAssemblyIds.length === 1 && beaconAssemblyIds[0] };
const launchEmptyQuery = useCallback(() => dispatch(makeBeaconQuery(requestPayload({}, []))), [dispatch]);
const formInitialValues = useMemo(
() => ({
'Assembly ID': beaconAssemblyIds.length === 1 && beaconAssemblyIds[0],
}),
[beaconAssemblyIds]
);
const uiInstructions = hasVariants ? 'Search by genomic variants, clinical metadata or both.' : '';

const hasError = hasFormError || hasApiError;
Expand All @@ -88,7 +93,7 @@ const BeaconQueryUi = () => {

// set assembly id options matching what's in gohan
form.setFieldsValue(formInitialValues);
}, [isFetchingBeaconConfig, isAuthenticated]);
}, [beaconAssemblyIds.length, form, formInitialValues, isFetchingBeaconConfig, isAuthenticated, launchEmptyQuery]);

// Disables max query param if user is authenticated and authorized
useQueryWithAuthIfAllowed();
Expand Down
2 changes: 1 addition & 1 deletion src/js/components/Beacon/Filter.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ const Filter = ({ filter, form, querySections, removeFilter, isRequired }: Filte
form.setFieldsValue({
[`filterValue${filter.index}`]: valueOptions[0].value,
});
}, [valueOptions]);
}, [filter.index, form, valueOptions]);

const renderLabel = (searchField: Field) => {
const units = searchField.config?.units;
Expand Down
81 changes: 71 additions & 10 deletions src/js/components/BentoAppRouter.tsx
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
import React, { useEffect } from 'react';
import { Routes, Route } from 'react-router-dom';
import { Routes, Route, useNavigate, useParams, Outlet } from 'react-router-dom';
import { useAutoAuthenticate, useIsAuthenticated } from 'bento-auth-js';
import { useAppDispatch } from '@/hooks';
import { useAppDispatch, useAppSelector } from '@/hooks';

import { makeGetConfigRequest, makeGetServiceInfoRequest } from '@/features/config/config.store';
import { makeGetAboutRequest } from '@/features/content/content.store';
Expand All @@ -11,19 +11,56 @@ import { makeGetProvenanceRequest } from '@/features/provenance/provenance.store
import { getBeaconConfig } from '@/features/beacon/beaconConfig.store';
import { fetchGohanData, fetchKatsuData } from '@/features/ingestion/lastIngestion.store';
import { makeGetDataTypes } from '@/features/dataTypes/dataTypes.store';
import { getProjects, selectScope } from '@/features/metadata/metadata.store';

import PublicOverview from './Overview/PublicOverview';
import Search from './Search/Search';
import ProvenanceTab from './Provenance/ProvenanceTab';
import BeaconQueryUi from './Beacon/BeaconQueryUi';
import { BentoRoute } from '@/types/routes';
import Loader from '@/components/Loader';
import { validProjectDataset } from '@/utils/router';
import DefaultLayout from '@/components/Util/DefaultLayout';

const ScopedRoute = () => {
const { projectId, datasetId } = useParams();
const dispatch = useAppDispatch();
const navigate = useNavigate();
const { projects } = useAppSelector((state) => state.metadata);

useEffect(() => {
// Update selectedScope based on URL parameters
const valid = validProjectDataset(projects, projectId, datasetId);
if (datasetId === valid.dataset && projectId === valid.project) {
dispatch(selectScope(valid));
} else {
const oldpath = location.pathname.split('/').filter(Boolean);
const newPath = [oldpath[0]];

if (valid.dataset) {
newPath.push('p', valid.project as string, 'd', valid.dataset);
} else if (valid.project) {
newPath.push('p', valid.project);
}

const oldPathLength = oldpath.length;
if (oldpath[oldPathLength - 3] === 'p' || oldpath[oldPathLength - 3] === 'd') {
newPath.push(oldpath[oldPathLength - 1]);
}
const newPathString = '/' + newPath.join('/');
navigate(newPathString, { replace: true });
}
}, [projects, projectId, datasetId, dispatch, navigate]);

return <Outlet />;
};

const BentoAppRouter = () => {
const dispatch = useAppDispatch();

const { isAutoAuthenticating } = useAutoAuthenticate();
const isAuthenticated = useIsAuthenticated();
const { selectedScope, isFetching: isFetchingProjects } = useAppSelector((state) => state.metadata);

useEffect(() => {
dispatch(makeGetConfigRequest()).then(() => dispatch(getBeaconConfig()));
Expand All @@ -33,25 +70,49 @@ const BentoAppRouter = () => {
dispatch(makeGetProvenanceRequest());
dispatch(makeGetKatsuPublic());
dispatch(fetchKatsuData());
}, [dispatch, selectedScope]);

useEffect(() => {
dispatch(getProjects());
dispatch(makeGetAboutRequest());
dispatch(fetchGohanData());
dispatch(makeGetServiceInfoRequest());

if (isAuthenticated) {
dispatch(makeGetDataTypes());
}
}, [isAuthenticated]);
}, [dispatch, isAuthenticated]);

if (isAutoAuthenticating) {
if (isAutoAuthenticating || isFetchingProjects) {
return <Loader />;
}

return (
<Routes>
<Route path={`/${BentoRoute.Overview}`} element={<PublicOverview />} />
<Route path={`/${BentoRoute.Search}/*`} element={<Search />} />
<Route path={`/${BentoRoute.Beacon}/*`} element={<BeaconQueryUi />} />
<Route path={`/${BentoRoute.Provenance}/*`} element={<ProvenanceTab />} />
<Route path="/*" element={<PublicOverview />} />
<Route element={<DefaultLayout />}>
<Route path="/" element={<ScopedRoute />}>
<Route index element={<PublicOverview />} />
<Route path={BentoRoute.Overview} element={<PublicOverview />} />
<Route path={BentoRoute.Search} element={<Search />} />
<Route path={BentoRoute.Beacon} element={<BeaconQueryUi />} />
<Route path={BentoRoute.Provenance} element={<ProvenanceTab />} />
</Route>

<Route path="/p/:projectId" element={<ScopedRoute />}>
<Route index element={<PublicOverview />} />
<Route path={BentoRoute.Overview} element={<PublicOverview />} />
<Route path={BentoRoute.Search} element={<Search />} />
<Route path={BentoRoute.Beacon} element={<BeaconQueryUi />} />
<Route path={BentoRoute.Provenance} element={<ProvenanceTab />} />
</Route>

<Route path="/p/:projectId/d/:datasetId" element={<ScopedRoute />}>
<Route index element={<PublicOverview />} />
<Route path={BentoRoute.Overview} element={<PublicOverview />} />
<Route path={BentoRoute.Search} element={<Search />} />
<Route path={BentoRoute.Beacon} element={<BeaconQueryUi />} />
<Route path={BentoRoute.Provenance} element={<ProvenanceTab />} />
</Route>
</Route>
</Routes>
);
};
Expand Down
82 changes: 82 additions & 0 deletions src/js/components/ChooseProjectModal.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,82 @@
import React, { useState } from 'react';
import { Tabs, List, Avatar, Modal, Button, Space, Typography } from 'antd';
import { useAppSelector, useTranslationCustom, useTranslationDefault } from '@/hooks';
import { Link, useLocation } from 'react-router-dom';
import { FaDatabase } from 'react-icons/fa';
import { getCurrentPage } from '@/utils/router';

const ChooseProjectModal = ({ isModalOpen, setIsModalOpen }: ChooseProjectModalProps) => {
const td = useTranslationDefault();
const t = useTranslationCustom();
const location = useLocation();
const { projects, selectedScope } = useAppSelector((state) => state.metadata);
const [selectedProject, setSelectedProject] = useState<string | undefined>(
selectedScope.project ?? projects[0]?.identifier ?? undefined
);

const baseURL = '/' + location.pathname.split('/')[1];
const page = getCurrentPage();

const closeModal = () => setIsModalOpen(false);

return (
<Modal title={td('Select Scope')} open={isModalOpen} onCancel={closeModal} footer={null} width={800}>
<Tabs
tabPosition="left"
activeKey={selectedProject}
onChange={(key) => setSelectedProject(key)}
tabBarExtraContent={
<Link to={baseURL}>
<Button>{td('Clear')}</Button>
</Link>
}
items={projects.map(({ identifier, title, datasets, description }) => {
return {
key: identifier,
label: t(title),
children: (
<Space direction="vertical">
<Space align="baseline" size="large">
<Typography.Title level={4} className="no-margin-top">
{td('About')} {t(title)}
</Typography.Title>
<Link to={`${baseURL}/p/${selectedProject}/${page}`} key="3">
<Typography.Link>{td('Select')}</Typography.Link>
</Link>
</Space>
<Typography.Text>{t(description)}</Typography.Text>
<Typography.Title level={5} className="no-margin-top">
{td('Datasets')}
</Typography.Title>
<List
dataSource={datasets}
bordered
renderItem={(item) => (
<Link to={`${baseURL}/p/${identifier}/d/${item.identifier}/${page}`}>
<List.Item className="select-dataset-hover" key={item.identifier}>
<List.Item.Meta
avatar={<Avatar style={{ backgroundColor: '#33ccff' }} icon={<FaDatabase />} />}
title={
<Link to={`${baseURL}/p/${identifier}/d/${item.identifier}/${page}`}>{t(item.title)}</Link>
}
description={t(item.description)}
/>
</List.Item>
</Link>
)}
/>
</Space>
),
};
})}
/>
</Modal>
);
};

interface ChooseProjectModalProps {
isModalOpen: boolean;
setIsModalOpen: (isOpen: boolean) => void;
}

export default ChooseProjectModal;
9 changes: 6 additions & 3 deletions src/js/components/Overview/Chart.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -13,17 +13,20 @@ import {
CHART_TYPE_PIE,
ChartConfig,
} from '@/types/chartConfig';
import { useAppSelector } from '@/hooks';
import { scopeToUrl } from '@/utils/router';

const Chart = memo(({ chartConfig, data, units, id, isClickable }: ChartProps) => {
const { t, i18n } = useTranslation();
const navigate = useNavigate();
const { selectedScope } = useAppSelector((state) => state.metadata);
const translateMap = ({ x, y }: { x: string; y: number }) => ({ x: t(x), y });
const removeMissing = ({ x }: { x: string }) => x !== 'missing';
const barChartOnClickHandler = (d: { payload: { x: string } }) => {
navigate(`/${i18n.language}/search?${id}=${d.payload.x}`);
navigate(`/${i18n.language}${scopeToUrl(selectedScope)}/search?${id}=${d.payload.x}`);
};
const pieChartOnClickHandler = (d: { name: string }) => {
navigate(`/${i18n.language}/search?${id}=${d.name}`);
navigate(`/${i18n.language}${scopeToUrl(selectedScope)}/search?${id}=${d.name}`);
};

const { chart_type: type } = chartConfig;
Expand Down Expand Up @@ -58,7 +61,7 @@ const Chart = memo(({ chartConfig, data, units, id, isClickable }: ChartProps) =
height={PIE_CHART_HEIGHT}
preFilter={removeMissing}
dataMap={translateMap}
{...(isClickable ? { onClick: pieChartOnClickHandler } : {})}
onClick={pieChartOnClickHandler}
/>
);
case CHART_TYPE_CHOROPLETH: {
Expand Down
2 changes: 1 addition & 1 deletion src/js/components/Overview/Drawer/ChartTree.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -43,7 +43,7 @@ const ChartTree = ({ charts, section }: ChartTreeProps) => {
),
key: id,
})),
[charts, t, td]
[charts, dispatch, section, t, td]
);

const onChartDrop: TreeProps['onDrop'] = useMemo(() => {
Expand Down
2 changes: 1 addition & 1 deletion src/js/components/Overview/PublicOverview.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -36,7 +36,7 @@ const PublicOverview = () => {
// Save sections to localStorage when they change
if (isFetchingOverviewData) return;
saveToLocalStorage(sections);
}, [sections]);
}, [isFetchingOverviewData, sections]);

useEffect(() => {
const activeLanguage = i18n.language;
Expand Down
2 changes: 1 addition & 1 deletion src/js/components/Provenance/Tables/DistributionsTable.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ const DistributionsTable = ({ distributions }: DistributionsTableProps) => {
],
},
] as ColumnsType<Distribution>,
[td]
[t, td]
);

return (
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ const ExtraPropertiesTable = ({ extraProperties }: ExtraPropertiesTableProps) =>
)),
},
] as ColumnsType<ExtraProperty>,
[td]
[t, td]
);

return <BaseProvenanceTable dataSource={extraProperties} columns={columns} rowKey="category" />;
Expand Down
2 changes: 1 addition & 1 deletion src/js/components/Provenance/Tables/PublicationsTable.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -115,7 +115,7 @@ const PublicationsTable = ({ publications }: PublicationsTableProps) => {
onFilter: (filterValue, p) => p.identifier.identifierSource === filterValue,
},
] as ColumnsType<PrimaryPublication>,
[td, publications]
[t, td, publications]
);

return <BaseProvenanceTable dataSource={publications} columns={columns} rowKey="title" />;
Expand Down
Loading