Skip to content

Commit

Permalink
Merge pull request #435 from allen-cell-animated/feature/feature-tool…
Browse files Browse the repository at this point in the history
…tips

- Adds the `GlossaryPanel` component, which renders the list of features in an Ant `Drawer`.
- Adds support for the `descriptions` feature metadata added in `colorizer-data v.1.4.2`.
- Bonus: refactors the `LandingPage` to use the `ExternalLink` convenience component.
  • Loading branch information
ShrimpCryptid authored Sep 5, 2024
2 parents 26d1599 + 427e1b7 commit be04b48
Show file tree
Hide file tree
Showing 6 changed files with 155 additions and 24 deletions.
47 changes: 33 additions & 14 deletions src/Viewer.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -34,7 +34,7 @@ import { useConstructor, useDebounce, useRecentCollections } from "./colorizer/u
import * as urlUtils from "./colorizer/utils/url_utils";
import { SCATTERPLOT_TIME_FEATURE } from "./components/Tabs/scatter_plot_data_utils";
import { DEFAULT_PLAYBACK_FPS } from "./constants";
import { FlexRowAlignCenter } from "./styles/utils";
import { FlexRow, FlexRowAlignCenter } from "./styles/utils";
import { LocationState } from "./types";

import Collection from "./colorizer/Collection";
Expand All @@ -51,6 +51,7 @@ import ColorRampDropdown from "./components/Dropdowns/ColorRampDropdown";
import HelpDropdown from "./components/Dropdowns/HelpDropdown";
import SelectionDropdown from "./components/Dropdowns/SelectionDropdown";
import Export from "./components/Export";
import GlossaryPanel from "./components/GlossaryPanel";
import Header from "./components/Header";
import HoverTooltip from "./components/HoverTooltip";
import IconButton from "./components/IconButton";
Expand Down Expand Up @@ -834,19 +835,37 @@ function Viewer(): ReactElement {
items={collection?.getDatasetKeys() || []}
onChange={handleDatasetChange}
/>
<SelectionDropdown
disabled={disableUi}
label="Feature"
selected={featureKey}
items={getFeatureDropdownData()}
onChange={(value) => {
if (value !== featureKey && dataset) {
replaceFeature(dataset, value);
resetColorRampRangeToDefaults(dataset, value);
reportFeatureSelected(dataset, value);
}
}}
/>
<FlexRow $gap={6}>
<SelectionDropdown
disabled={disableUi}
label="Feature"
showTooltip={true}
// TODO: Once dropdowns are refactored, add description into tooltips
// dataset?.getFeatureData(featureKey)?.description ? (
// // Show as larger element with subtitle if description is given
// <FlexColumn>
// <span style={{ fontSize: "14px" }}>
// {featureKey && dataset?.getFeatureNameWithUnits(featureKey)}
// </span>
// <span style={{ fontSize: "13px", opacity: "0.9" }}>
// {dataset?.getFeatureData(featureKey)?.description}
// </span>
// </FlexColumn>
// ) : (
// dataset?.getFeatureNameWithUnits(featureKey)
// )
selected={featureKey}
items={getFeatureDropdownData()}
onChange={(value) => {
if (value !== featureKey && dataset) {
replaceFeature(dataset, value);
resetColorRampRangeToDefaults(dataset, value);
reportFeatureSelected(dataset, value);
}
}}
/>
<GlossaryPanel dataset={dataset} />
</FlexRow>

<ColorRampDropdown
knownColorRamps={KNOWN_COLOR_RAMPS}
Expand Down
4 changes: 3 additions & 1 deletion src/colorizer/Dataset.ts
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ export type FeatureData = {
unit: string;
type: FeatureType;
categories: string[] | null;
description: string | null;
};

type BackdropData = {
Expand Down Expand Up @@ -159,9 +160,10 @@ export default class Dataset {
data: source.getBuffer(),
min: source.getMin(),
max: source.getMax(),
unit: metadata?.unit || "",
unit: metadata.unit || "",
type: featureType,
categories: featureCategories || null,
description: metadata.description || null,
},
];
}
Expand Down
3 changes: 3 additions & 0 deletions src/colorizer/utils/dataset_utils.ts
Original file line number Diff line number Diff line change
Expand Up @@ -64,8 +64,11 @@ type ManifestFileV1_0_0 = Omit<ManifestFileV0_0_0, "features" | "featureMetadata
unit?: string;
type?: string;
categories?: string[];
// Added in v1.3.0
min?: number | null;
max?: number | null;
// Added in v1.4.2
description?: string;
}[];
/** Optional list of backdrop/overlay images. */
backdrops?: { name: string; key: string; frames: string[] }[];
Expand Down
113 changes: 113 additions & 0 deletions src/components/GlossaryPanel.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,113 @@
import { ReadOutlined } from "@ant-design/icons";
import { Divider, Drawer, Radio, RadioChangeEvent, Tooltip } from "antd";
import React, { ReactElement, useContext, useMemo, useState } from "react";
import styled from "styled-components";

import { Dataset } from "../colorizer";
import { FlexColumn, FlexRowAlignCenter } from "../styles/utils";

import { AppThemeContext } from "./AppStyle";
import IconButton from "./IconButton";

type GlossaryPanelProps = {
dataset: Dataset | null;
};

const StyledDrawer = styled(Drawer)`
.ant-drawer-body {
padding-top: 16px;
}
.ant-drawer-header-title {
// Move the close button to the right corner of the header
display: flex;
flex-direction: row-reverse;
}
p {
margin: 0;
}
`;

export default function GlossaryPanel(props: GlossaryPanelProps): ReactElement {
const [showPanel, setShowPanel] = useState(false);
const [alphabetizeFeatures, setAlphabetizeFeatures] = useState(false);

const theme = useContext(AppThemeContext);

const drawerContent = useMemo(() => {
const dataset = props.dataset;
if (dataset === null) {
return null;
}
const allFeatureData = dataset.featureKeys.map((featureKey) => dataset.getFeatureData(featureKey)!);
if (alphabetizeFeatures) {
allFeatureData.sort((a, b) => a.name.localeCompare(b.name));
}

return (
<FlexColumn $gap={15}>
{allFeatureData.map((featureData) => {
const hasDescription = featureData.description && featureData.description !== "";
const key = featureData.key;
return (
<p style={{ lineHeight: "1.5", margin: "0" }} key={key}>
<Tooltip title={hasDescription ? null : "No description provided"} placement="right">
<span style={{ fontWeight: "bold" }}>{dataset.getFeatureNameWithUnits(key)}</span>
</Tooltip>
<br />
{featureData.description}
</p>
);
})}
</FlexColumn>
);
}, [props.dataset, alphabetizeFeatures]);

return (
<>
<Tooltip placement="top" title="Open feature glossary" trigger={["hover", "focus"]}>
<IconButton
type="link"
onClick={() => {
setShowPanel(true);
}}
disabled={!props.dataset}
>
<ReadOutlined style={{ marginTop: "1px" }} /* Visually align with other elements*/ />
</IconButton>
</Tooltip>
<StyledDrawer
zIndex={2000}
width={"calc(min(75vw, 1000px))"}
onClose={() => {
setShowPanel(false);
}}
title={<span style={{ fontSize: "20px" }}>Feature glossary</span>}
open={showPanel}
size="large"
style={{ color: theme.color.text.primary }}
>
<FlexColumn $gap={16}>
<FlexRowAlignCenter $gap={8}>
<h3 style={{ margin: 0, fontWeight: "normal" }} id="glossary-sortby-label">
Sort by
</h3>
<Radio.Group
buttonStyle="solid"
// TODO: Why does Ant not support aria-label on Radio.Group?
// "aria-label"="glossary-sortby-label"
value={alphabetizeFeatures}
onChange={(e: RadioChangeEvent) => setAlphabetizeFeatures(e.target.value)}
>
<Radio.Button value={false}>Feature order</Radio.Button>
<Radio.Button value={true}>Alphabetical</Radio.Button>
</Radio.Group>
</FlexRowAlignCenter>
<Divider />
{drawerContent}
</FlexColumn>
</StyledDrawer>
</>
);
}
11 changes: 2 additions & 9 deletions src/routes/LandingPage.tsx
Original file line number Diff line number Diff line change
@@ -1,13 +1,11 @@
import { faUpRightFromSquare } from "@fortawesome/free-solid-svg-icons";
import { FontAwesomeIcon } from "@fortawesome/react-fontawesome";
import { Button, Divider, Tooltip } from "antd";
import React, { lazy, ReactElement, Suspense } from "react";
import { Link, useNavigate } from "react-router-dom";
import styled from "styled-components";

import { Dataset } from "../colorizer";
import { paramsToUrlQueryString } from "../colorizer/utils/url_utils";
import { FlexColumn, FlexColumnAlignCenter, FlexRowAlignCenter, VisuallyHidden } from "../styles/utils";
import { ExternalLink, FlexColumn, FlexColumnAlignCenter, FlexRowAlignCenter, VisuallyHidden } from "../styles/utils";
import { DatasetEntry, LocationState, ProjectEntry } from "../types";
import { PageRoutes } from "./index";

Expand Down Expand Up @@ -253,12 +251,7 @@ export default function LandingPage(): ReactElement {
const publicationElement = project.publicationLink ? (
<p>
Related publication:{" "}
<a href={project.publicationLink.toString()} target="_blank" rel="noopener noreferrer">
{project.publicationName}
{/* Icon offset slightly to align with text */}
<FontAwesomeIcon icon={faUpRightFromSquare} size="sm" style={{ marginBottom: "-1px", marginLeft: "3px" }} />
<VisuallyHidden>(opens in new tab)</VisuallyHidden>
</a>
<ExternalLink href={project.publicationLink.toString()}>{project.publicationName}</ExternalLink>
</p>
) : null;

Expand Down
1 change: 1 addition & 0 deletions src/styles/utils.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -82,6 +82,7 @@ export function ExternalLink(props: { href: string; children: React.ReactNode })
<a href={props.href} style={{ whiteSpace: "nowrap" }} rel="noopener noreferrer" target="_blank">
{props.children}
<FontAwesomeIcon icon={faUpRightFromSquare} size="sm" style={{ marginBottom: "0px", marginLeft: "3px" }} />
<VisuallyHidden>(opens in new tab)</VisuallyHidden>
</a>
);
}

0 comments on commit be04b48

Please sign in to comment.