From 20c51abfb6481500b6bfa9a2aeade9eb9b75877d Mon Sep 17 00:00:00 2001 From: Brian Love Date: Thu, 19 Oct 2023 14:42:25 -0400 Subject: [PATCH 1/4] Merge chart components Simplify charts by combining two components that are only used with each other. --- web/gui-v2/src/components/DetailViewChart.jsx | 47 --------------- .../src/components/DetailViewPatents.jsx | 24 +++----- .../src/components/DetailViewPublications.jsx | 45 ++++++--------- web/gui-v2/src/components/TrendsChart.jsx | 57 ++++++++++++++++--- web/gui-v2/src/util/plotly-helpers.js | 6 +- 5 files changed, 78 insertions(+), 101 deletions(-) delete mode 100644 web/gui-v2/src/components/DetailViewChart.jsx diff --git a/web/gui-v2/src/components/DetailViewChart.jsx b/web/gui-v2/src/components/DetailViewChart.jsx deleted file mode 100644 index 81cca27d..00000000 --- a/web/gui-v2/src/components/DetailViewChart.jsx +++ /dev/null @@ -1,47 +0,0 @@ -import React, { Suspense, lazy } from 'react'; -import { css } from '@emotion/react'; - -import SectionHeading from './SectionHeading'; -import { fallback } from '../styles/common-styles'; - -const Plot = lazy(() => import('react-plotly.js')); - -const isSSR = typeof window === "undefined"; - -const styles = { - chartContainer: css` - /* aspect-ratio: 4 / 3; */ - aspect-ratio: 16 / 9; - display: flex; - flex-direction: column; - margin: 0.5rem auto 0; - max-width: 1000px; - `, -}; - - -const Chart = ({ - config, - data, - id, - layout, - title, -}) => { - return ( - !isSSR && - Loading graph...}> - - {title} - -
- -
-
- ); -}; - -export default Chart; diff --git a/web/gui-v2/src/components/DetailViewPatents.jsx b/web/gui-v2/src/components/DetailViewPatents.jsx index c3a3c74a..da64d248 100644 --- a/web/gui-v2/src/components/DetailViewPatents.jsx +++ b/web/gui-v2/src/components/DetailViewPatents.jsx @@ -1,7 +1,7 @@ import React, { useState } from 'react'; import { css } from '@emotion/react'; -import { Autocomplete, Dropdown } from '@eto/eto-ui-components'; +import { Autocomplete } from '@eto/eto-ui-components'; import HeaderWithLink from './HeaderWithLink'; import StatGrid from './StatGrid'; @@ -11,7 +11,6 @@ import TrendsChart from './TrendsChart'; import overall from '../static_data/overall_data.json'; import { patentMap } from '../static_data/table_columns'; import { commas } from '../util'; -import { assemblePlotlyParams } from '../util/plotly-helpers'; const styles = { section: css` @@ -123,18 +122,6 @@ const DetailViewPatents = ({ .map(k => ({ text: patentMap[k].replace(/ patents/i, ''), val: k })) .sort((a, b) => a.text.localeCompare(b.text, 'en', { sensitivity: 'base' })); - const aiSubfieldChartData = assemblePlotlyParams( - "Trends in research....", - overall.years, - [ - [ - aiSubfieldOptions.find(e => e.val === aiSubfield)?.text, - data.patents[aiSubfield].counts - ], - ], - chartLayoutChanges, - ); - return ( <> @@ -164,8 +151,14 @@ const DetailViewPatents = ({ e.val === aiSubfield)?.text, + data.patents[aiSubfield].counts + ], + ]} id="ai-subfield-patents" + layoutChanges={chartLayoutChanges} title={ <> Trends in {data.name}'s patenting in @@ -179,6 +172,7 @@ const DetailViewPatents = ({ /> } + years={overall.years} /> ); diff --git a/web/gui-v2/src/components/DetailViewPublications.jsx b/web/gui-v2/src/components/DetailViewPublications.jsx index 1c2ec55b..ddd3032e 100644 --- a/web/gui-v2/src/components/DetailViewPublications.jsx +++ b/web/gui-v2/src/components/DetailViewPublications.jsx @@ -3,7 +3,6 @@ import { css } from '@emotion/react'; import { Dropdown } from '@eto/eto-ui-components'; -import Chart from './DetailViewChart'; import HeaderWithLink from './HeaderWithLink'; import StatGrid from './StatGrid'; import TableSection from './TableSection'; @@ -12,7 +11,6 @@ import TrendsChart from './TrendsChart'; import overall from '../static_data/overall_data.json'; import { articleMap } from '../static_data/table_columns'; import { commas } from '../util'; -import { assemblePlotlyParams } from '../util/plotly-helpers'; const styles = { noTopMargin: css` @@ -119,27 +117,6 @@ const DetailViewPublications = ({ { text: "Robotics", val: "robotics_pubs" }, ]; - const aiSubfieldChartData = assemblePlotlyParams( - "Trends in research....", - overall.years, - [ - [ - aiSubfieldOptions.find(e => e.val === aiSubfield)?.text, - data.articles[aiSubfield].counts - ], - ], - chartLayoutChanges, - ); - - const topConfs = assemblePlotlyParams( - <>{data.name}'s top AI conference publications, - overall.years, - [ - ["AI top conference publications", data.articles.ai_pubs_top_conf.counts], - ], - chartLayoutChanges, - ); - return ( <> @@ -161,8 +138,14 @@ const DetailViewPublications = ({ e.val === aiSubfield)?.text, + data.articles[aiSubfield].counts + ], + ]} id="ai-subfield-research" + layoutChanges={chartLayoutChanges} title={ <> Trends in {data.name}'s research in @@ -176,11 +159,19 @@ const DetailViewPublications = ({ /> } + years={overall.years} /> -
- -
+ {data.name}'s top AI conference publications} + years={overall.years} + /> ); }; diff --git a/web/gui-v2/src/components/TrendsChart.jsx b/web/gui-v2/src/components/TrendsChart.jsx index 201103fa..3d22f517 100644 --- a/web/gui-v2/src/components/TrendsChart.jsx +++ b/web/gui-v2/src/components/TrendsChart.jsx @@ -1,7 +1,13 @@ -import React from 'react'; +import React, { Suspense, lazy } from 'react'; import { css } from '@emotion/react'; -import Chart from './DetailViewChart'; +import SectionHeading from './SectionHeading'; +import { fallback } from '../styles/common-styles'; +import { assemblePlotlyParams } from '../util/plotly-helpers'; + +const Plot = lazy(() => import('react-plotly.js')); + +const isSSR = typeof window === "undefined"; const styles = { chartWrapper: css` @@ -19,26 +25,61 @@ const styles = { } } `, + chartContainer: css` + /* aspect-ratio: 4 / 3; */ + aspect-ratio: 16 / 9; + display: flex; + flex-direction: column; + margin: 0.5rem auto 0; + max-width: 1000px; + `, }; +/** + * Chart and heading showing trends over time. + * + * @param {object} props + * @param {Array<[string, Array]>} props.data + * @param {object} props.layoutChanges + * @param {boolean|undefined} props.partialStartsIndex + * @param {Array} props.years + * // + * @returns {JSX.Element} + */ const TrendsChart = ({ className: appliedClassName, css: appliedCss, + data: dataRaw, id: appliedId, + layoutChanges, + partialStartsIndex=undefined, title, - ...otherProps + years, }) => { + console.info("TrendsChart:", {dataRaw, layoutChanges, years}); // DEBUG + + const { config, data, layout } = assemblePlotlyParams(years, dataRaw, layoutChanges); + return (
- + {!isSSR && + Loading graph...
}> + + {title} + +
+ +
+ + } ); }; diff --git a/web/gui-v2/src/util/plotly-helpers.js b/web/gui-v2/src/util/plotly-helpers.js index b0d9d0cf..20ffb11f 100644 --- a/web/gui-v2/src/util/plotly-helpers.js +++ b/web/gui-v2/src/util/plotly-helpers.js @@ -17,7 +17,6 @@ const assembleChartData = (name, years, vals, otherParams) => { /** * Generate the parameters for a given Plotly chart. * - * @param {string} title Overall title for the chart * @param {Array} years Array of years corresponding to `data` * @param {Array<[string, Array]>} data Array of tuples representing * individual traces in the chart, each consisting of a string for the trace @@ -25,9 +24,9 @@ const assembleChartData = (name, years, vals, otherParams) => { * @param {object} layoutChanges Any changes that should be merged into the * ETO-standard `layout` object provided by `PlotlyDefaults`. * @returns {object} An object containing the parameters for this Plotly chart: - * `{ config, data, layout, title }` + * `{ config, data, layout }` */ -export const assemblePlotlyParams = (title, years, data, layoutChanges) => { +export const assemblePlotlyParams = (years, data, layoutChanges={}) => { const preparedData = data.map(([traceTitle, traceData, otherParams={}]) => { return assembleChartData(traceTitle, years, traceData, otherParams); }); @@ -41,7 +40,6 @@ export const assemblePlotlyParams = (title, years, data, layoutChanges) => { config, data: preparedData, layout, - title, }; }; From 29c91100481bd062a9b000986aebaac8ab9fdaac Mon Sep 17 00:00:00 2001 From: Brian Love Date: Thu, 19 Oct 2023 18:03:59 -0400 Subject: [PATCH 2/4] Add shading to graphs for years with partial data Closes #142 --- .../src/components/DetailViewPatents.jsx | 1 + .../src/components/DetailViewPublications.jsx | 2 + web/gui-v2/src/components/TrendsChart.jsx | 10 ++- web/gui-v2/src/util/plotly-helpers.js | 58 +++++++++++---- web/gui-v2/src/util/plotly-helpers.test.js | 71 +++++++++++++++++++ 5 files changed, 124 insertions(+), 18 deletions(-) create mode 100644 web/gui-v2/src/util/plotly-helpers.test.js diff --git a/web/gui-v2/src/components/DetailViewPatents.jsx b/web/gui-v2/src/components/DetailViewPatents.jsx index da64d248..a8a84409 100644 --- a/web/gui-v2/src/components/DetailViewPatents.jsx +++ b/web/gui-v2/src/components/DetailViewPatents.jsx @@ -159,6 +159,7 @@ const DetailViewPatents = ({ ]} id="ai-subfield-patents" layoutChanges={chartLayoutChanges} + partialStartIndex={endIx} title={ <> Trends in {data.name}'s patenting in diff --git a/web/gui-v2/src/components/DetailViewPublications.jsx b/web/gui-v2/src/components/DetailViewPublications.jsx index ddd3032e..e8ca9e70 100644 --- a/web/gui-v2/src/components/DetailViewPublications.jsx +++ b/web/gui-v2/src/components/DetailViewPublications.jsx @@ -146,6 +146,7 @@ const DetailViewPublications = ({ ]} id="ai-subfield-research" layoutChanges={chartLayoutChanges} + partialStartIndex={endIx} title={ <> Trends in {data.name}'s research in @@ -169,6 +170,7 @@ const DetailViewPublications = ({ ]} id="ai-top-conference-pubs-2" layoutChanges={chartLayoutChanges} + partialStartIndex={endIx} title={<>{data.name}'s top AI conference publications} years={overall.years} /> diff --git a/web/gui-v2/src/components/TrendsChart.jsx b/web/gui-v2/src/components/TrendsChart.jsx index 3d22f517..d8c8aab0 100644 --- a/web/gui-v2/src/components/TrendsChart.jsx +++ b/web/gui-v2/src/components/TrendsChart.jsx @@ -41,9 +41,9 @@ const styles = { * @param {object} props * @param {Array<[string, Array]>} props.data * @param {object} props.layoutChanges - * @param {boolean|undefined} props.partialStartsIndex + * @param {boolean|undefined} props.partialStartIndex + * @param {string} props.title * @param {Array} props.years - * // * @returns {JSX.Element} */ const TrendsChart = ({ @@ -52,13 +52,11 @@ const TrendsChart = ({ data: dataRaw, id: appliedId, layoutChanges, - partialStartsIndex=undefined, + partialStartIndex=undefined, title, years, }) => { - console.info("TrendsChart:", {dataRaw, layoutChanges, years}); // DEBUG - - const { config, data, layout } = assemblePlotlyParams(years, dataRaw, layoutChanges); + const { config, data, layout } = assemblePlotlyParams(years, dataRaw, layoutChanges, { partialStartIndex }); return (
{ - return { +const assembleChartData = (name, years, vals, otherParams, options={}) => { + const { partialStartIndex } = options; + + const common = { hovertemplate: "%{y}", - mode: 'lines+markers', - type: 'scatter', - ...otherParams, + legendgroup: name, + mode: "lines+markers", name, - x: years, - y: vals, - }; + type: "scatter", + } + + // TODO / QUESTION: How do we want to handle the line/shading style for the + // year specified in `year*End`? Think about and adjust if desired. + const endSolidIx = partialStartIndex ? (partialStartIndex + 1) : undefined; + + const result = [ + { + ...common, + ...otherParams, + x: years.slice(0, endSolidIx), + y: vals.slice(0, endSolidIx), + } + ]; + + if ( partialStartIndex !== undefined ) { + result.push({ + ...common, + ...otherParams, + fill: "tozeroy", + line: { dash: "dash" }, + marker: { color: "lightgray" }, + showlegend: false, + x: years.slice(partialStartIndex), + y: vals.slice(partialStartIndex), + }); + } + + return result; }; /** @@ -23,12 +51,19 @@ const assembleChartData = (name, years, vals, otherParams) => { * title and an array of values (which must be the same length as `years`). * @param {object} layoutChanges Any changes that should be merged into the * ETO-standard `layout` object provided by `PlotlyDefaults`. + * @param {object} options + * @param {number} options.partialStartIndex * @returns {object} An object containing the parameters for this Plotly chart: * `{ config, data, layout }` */ -export const assemblePlotlyParams = (years, data, layoutChanges={}) => { - const preparedData = data.map(([traceTitle, traceData, otherParams={}]) => { - return assembleChartData(traceTitle, years, traceData, otherParams); +export const assemblePlotlyParams = ( + years, + data, + layoutChanges={}, + options={}, +) => { + const preparedData = data.flatMap(([traceTitle, traceData, otherParams={}]) => { + return assembleChartData(traceTitle, years, traceData, otherParams, options); }); const maxY = Math.max( ...data.map(e => Math.max(...e[1])) @@ -42,4 +77,3 @@ export const assemblePlotlyParams = (years, data, layoutChanges={}) => { layout, }; }; - diff --git a/web/gui-v2/src/util/plotly-helpers.test.js b/web/gui-v2/src/util/plotly-helpers.test.js new file mode 100644 index 00000000..f892e80d --- /dev/null +++ b/web/gui-v2/src/util/plotly-helpers.test.js @@ -0,0 +1,71 @@ +import { assemblePlotlyParams } from './plotly-helpers'; + + +const TITLE = "AI top conference publications"; +const YEARS = [ 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021, 2022, 2023 ]; +const INPUT_DATA = [ + [ TITLE, [ 850, 928, 845, 843, 838, 1040, 1357, 1551, 1507, 461, 2 ] ], +]; +const LAYOUT_CHANGES = { + legend: { y: 1.15 }, + margin: { b: 50, l: 50, pad: 4, r: 50, t: 0 }, +}; +const PARTIAL_START = YEARS.findIndex(e => e === 2022); + +describe("Plotly helpers", () => { + it("create the parameters as expected", () => { + + const { config, data, layout } = assemblePlotlyParams(YEARS, INPUT_DATA, LAYOUT_CHANGES, { partialStartIndex: PARTIAL_START }); + + expect(config).toEqual({ + displayModeBar: false, + responsive: true, + }); + expect(data).toEqual([ + { + hovertemplate: "%{y}", + legendgroup: TITLE, + mode: "lines+markers", + name: TITLE, + type: "scatter", + x: [ 2013, 2014, 2015, 2016, 2017, 2018, 2019, 2020, 2021, 2022 ], + y: [ 850, 928, 845, 843, 838, 1040, 1357, 1551, 1507, 461 ], + }, + { + fill: "tozeroy", + hovertemplate: "%{y}", + legendgroup: TITLE, + line: { dash: "dash" }, + marker: { color: "lightgray" }, + mode: "lines+markers", + name: TITLE, + showlegend: false, + type: "scatter", + x: [ 2022, 2023 ], + y: [ 461, 2 ], + }, + ]); + expect(layout).toEqual({ + autosize: true, + xaxis: { + fixedrange: true, + }, + yaxis: { + fixedrange: true, + }, + font: { + family: "GTZirkonRegular, Arial" + }, + legend: { + orientation: "h", + yanchor: "top", + xanchor: "center", + y: 1.15, + x: 0.5 + }, + margin: { b: 50, l: 50, pad: 4, r: 50, t: 0 }, + paper_bgcolor: "rgba(0,0,0,0)", + plot_bgcolor: "rgba(0,0,0,0)", + }); + }); +}); From 3ea3423c770f2a7b106fbf75040491545c91a2ae Mon Sep 17 00:00:00 2001 From: Brian Love Date: Fri, 20 Oct 2023 09:25:58 -0400 Subject: [PATCH 3/4] Increase test timeout --- web/gui-v2/src/components/ListView.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/gui-v2/src/components/ListView.test.js b/web/gui-v2/src/components/ListView.test.js index 591e6689..d137fb71 100644 --- a/web/gui-v2/src/components/ListView.test.js +++ b/web/gui-v2/src/components/ListView.test.js @@ -61,7 +61,7 @@ describe("ListView", () => { for ( const column of INITIAL_COLUMNS.filter(e => e !== REMOVED_COLUMN) ) { expect(screen.getByRole('columnheader', { name: new RegExp(column, 'i') })); } - }, 60000); + }, 90000); }); describe('groups', () => { From 9cfb63c6786a48c4dae01bc74f28681380ec83c3 Mon Sep 17 00:00:00 2001 From: Brian Love Date: Fri, 20 Oct 2023 10:29:53 -0400 Subject: [PATCH 4/4] Use callback instead of reusing saved element Re-query for the dialog element instead of reusing the prior element as suggested in https://github.com/testing-library/dom-testing-library/issues/1204#issuecomment-1581335409 --- web/gui-v2/src/components/ListView.test.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/web/gui-v2/src/components/ListView.test.js b/web/gui-v2/src/components/ListView.test.js index d137fb71..499165cb 100644 --- a/web/gui-v2/src/components/ListView.test.js +++ b/web/gui-v2/src/components/ListView.test.js @@ -54,7 +54,7 @@ describe("ListView", () => { await user.click(removedCheckbox); expect(removedCheckbox.checked).toEqual(false); await user.click(getByRole(dialog, 'button', { name: 'Apply' })); - await waitForElementToBeRemoved(dialog); + await waitForElementToBeRemoved(() => screen.getByRole('dialog')); expect(screen.queryByRole('heading', { name: 'Add/remove columns'})).not.toBeInTheDocument(); // Verify that the changes took effect