From 97b7fa20bf7198883c22f44c4803d5227b0d6e8e Mon Sep 17 00:00:00 2001 From: Jan-Erik S <6697396+Bhavatu@users.noreply.github.com> Date: Tue, 15 Oct 2024 19:47:24 +0300 Subject: [PATCH] WIP: Use relatedPlanIndicators in umbrella plan instead of planindicators. Filter by relatedPlanIndicators common categories instead of planIndicator categories --- common/__generated__/graphql.ts | 66 +++++- components/actions/ActionListFilters.tsx | 1 + components/indicators/IndicatorList.tsx | 255 +++++++++++++++++++---- 3 files changed, 269 insertions(+), 53 deletions(-) diff --git a/common/__generated__/graphql.ts b/common/__generated__/graphql.ts index 2b666023..87f2eba8 100644 --- a/common/__generated__/graphql.ts +++ b/common/__generated__/graphql.ts @@ -190,7 +190,7 @@ export type Action = { indicatorsCount?: Maybe; leadParagraph: Scalars['String']; links: Array; - /** Describe the reason why this action has has this status */ + /** Describe the reason why this action has this status */ manualStatusReason?: Maybe; /** Set if this action is merged with another action */ mergedActions: Array; @@ -1865,7 +1865,7 @@ export type DataPoint = { export type Dataset = { __typename?: 'Dataset'; dataPoints: Array; - schema: DatasetSchema; + schema?: Maybe; scope?: Maybe; uuid: Scalars['UUID']; }; @@ -3669,6 +3669,7 @@ export type Query = { planPage?: Maybe; plansForHostname?: Maybe>>; relatedPlanActions?: Maybe>; + relatedPlanIndicators?: Maybe>; search?: Maybe; workflowStates?: Maybe>>; }; @@ -3764,6 +3765,14 @@ export type QueryRelatedPlanActionsArgs = { }; +export type QueryRelatedPlanIndicatorsArgs = { + category?: InputMaybe; + first?: InputMaybe; + orderBy?: InputMaybe; + plan: Scalars['ID']; +}; + + export type QuerySearchArgs = { autocomplete?: InputMaybe; includeRelatedPlans?: InputMaybe; @@ -5604,6 +5613,7 @@ export type IndicatorHightlightListQuery = ( export type IndicatorListQueryVariables = Exact<{ plan: Scalars['ID']; + relatedPlanIndicators: Scalars['Boolean']; }>; @@ -5689,7 +5699,49 @@ export type IndicatorListQuery = ( & { __typename?: 'CommonIndicator' } ) | null } & { __typename?: 'Indicator' } - ) | null> | null } + ) | null> | null, relatedPlanIndicators?: Array<( + { id: string, name: string, level?: string | null, timeResolution: IndicatorTimeResolution, organization: ( + { id: string, name: string } + & { __typename?: 'Organization' } + ), common?: ( + { id: string, name: string, normalizations?: Array<( + { unit?: ( + { shortName?: string | null } + & { __typename?: 'Unit' } + ) | null, normalizer?: ( + { name: string, id: string, identifier?: string | null } + & { __typename?: 'CommonIndicator' } + ) | null } + & { __typename?: 'CommonIndicatorNormalization' } + ) | null> | null } + & { __typename?: 'CommonIndicator' } + ) | null, latestGraph?: ( + { id: string } + & { __typename?: 'IndicatorGraph' } + ) | null, latestValue?: ( + { id: string, date?: string | null, value: number, normalizedValues?: Array<( + { normalizerId?: string | null, value?: number | null } + & { __typename?: 'NormalizedValue' } + ) | null> | null } + & { __typename?: 'IndicatorValue' } + ) | null, unit: ( + { shortName?: string | null } + & { __typename?: 'Unit' } + ), categories: Array<( + { id: string, name: string, parent?: ( + { id: string } + & { __typename?: 'Category' } + ) | null, type: ( + { id: string, identifier: string, name: string } + & { __typename?: 'CategoryType' } + ), common?: ( + { id: string, identifier: string, name: string, order: number } + & { __typename?: 'CommonCategory' } + ) | null } + & { __typename?: 'Category' } + )> } + & { __typename?: 'Indicator' } + )> | null } & { __typename?: 'Query' } ); @@ -7805,7 +7857,7 @@ export type GetActionDetailsQuery = ( ) } & { __typename: 'AttributeRichText' | 'AttributeText' } )>, datasets?: Array<( - { uuid: any, schema: ( + { uuid: any, schema?: ( { uuid: any, timeResolution: DatasetSchemaTimeResolution, unit: string, dimensionCategories: Array<( { order: number, category: ( { uuid: any, label: string, dimension: ( @@ -7817,7 +7869,7 @@ export type GetActionDetailsQuery = ( & { __typename?: 'DatasetSchemaDimensionCategory' } )> } & { __typename?: 'DatasetSchema' } - ), dataPoints: Array<( + ) | null, dataPoints: Array<( { uuid: any, value?: number | null, date: any, dimensionCategories: Array<( { uuid: any, label: string, dimension: ( { uuid: any } @@ -10864,7 +10916,7 @@ export type GetContentPageQuery = ( ) } & { __typename: 'AttributeRichText' | 'AttributeText' } )> | null, datasets?: Array<( - { uuid: any, schema: ( + { uuid: any, schema?: ( { uuid: any, timeResolution: DatasetSchemaTimeResolution, unit: string, dimensionCategories: Array<( { order: number, category: ( { uuid: any, label: string, dimension: ( @@ -10876,7 +10928,7 @@ export type GetContentPageQuery = ( & { __typename?: 'DatasetSchemaDimensionCategory' } )> } & { __typename?: 'DatasetSchema' } - ), dataPoints: Array<( + ) | null, dataPoints: Array<( { uuid: any, value?: number | null, date: any, dimensionCategories: Array<( { uuid: any, label: string, dimension: ( { uuid: any } diff --git a/components/actions/ActionListFilters.tsx b/components/actions/ActionListFilters.tsx index 9e9a2ad6..c5819cb5 100644 --- a/components/actions/ActionListFilters.tsx +++ b/components/actions/ActionListFilters.tsx @@ -668,6 +668,7 @@ class CategoryFilter extends DefaultFilter { (cat) => cat.parent, (cat) => cat.children ); + console.log('sortedcats', sortedCats); this.catById = this.filterByCommonCategory ? new Map(sortedCats.map((c) => [c.common.id, c])) : new Map(sortedCats.map((c) => [c.id, c])); diff --git a/components/indicators/IndicatorList.tsx b/components/indicators/IndicatorList.tsx index d2192f47..7bb082a7 100644 --- a/components/indicators/IndicatorList.tsx +++ b/components/indicators/IndicatorList.tsx @@ -1,14 +1,11 @@ 'use client'; import React, { useState } from 'react'; - import { gql } from '@apollo/client'; import { Container } from 'reactstrap'; - import ContentLoader from 'components/common/ContentLoader'; import { usePlan } from '../../context/plan'; import ErrorMessage from 'components/common/ErrorMessage'; - import IndicatorsHero from './IndicatorsHero'; import IndicatorListFiltered from './IndicatorListFiltered'; import ActionListFilters, { @@ -36,31 +33,85 @@ const createFilterUnusedCategories = ) ); -const getFilterConfig = (categoryType, indicators) => ({ +const createFilterUnusedCommonCategories = + (indicators: Indicator[]) => (commonCategory: any) => + indicators.find(({ categories }) => + categories.find( + (category) => + category.common && category.common.id === commonCategory.id + ) + ); + +const collectCommonCategories = (indicators) => { + const commonCategories = {}; + indicators.forEach((indicator) => { + indicator.categories.forEach((category) => { + if (category.common) { + const typeIdentifier = category.common.type.identifier; + if (!commonCategories[typeIdentifier]) { + commonCategories[typeIdentifier] = { + categories: new Set(), + type: { ...category.common.type }, + }; + } + commonCategories[typeIdentifier].categories.add(category.common); + } + }); + }); + return Object.entries(commonCategories).map(([typeIdentifier, data]) => ({ + typeIdentifier, + categories: Array.from(data?.categories), + type: data?.type, + })); +}; + +const includeRelatedPlanIndicators = true; + +const getFilterConfig = (categoryType, indicators, commonCategories) => ({ mainFilters: [ - { - __typename: 'CategoryTypeFilterBlock', - field: 'category', - id: '817256d7-a6fb-4af1-bbba-096171eb0d36', - style: 'dropdown', - showAllLabel: '', - depth: null, - categoryType: { - ...categoryType, - categories: categoryType.categories.filter( - createFilterUnusedCategories(indicators) - ), - hideCategoryIdentifiers: true, - selectionType: 'SINGLE', - }, - }, + ...(includeRelatedPlanIndicators + ? commonCategories.map(({ typeIdentifier, categories, type }) => ({ + __typename: 'CategoryTypeFilterBlock', + field: 'category', + id: `common_${typeIdentifier}`, + style: 'dropdown', + showAllLabel: '', + depth: null, + categoryType: { + ...type, + name: type.name || typeIdentifier, + categories: categories.filter( + createFilterUnusedCommonCategories(indicators) + ), + hideCategoryIdentifiers: true, + selectionType: 'SINGLE', + }, + })) + : [ + { + __typename: 'CategoryTypeFilterBlock', + field: 'category', + id: '817256d7-a6fb-4af1-bbba-096171eb0d36', + style: 'dropdown', + showAllLabel: '', + depth: null, + categoryType: { + ...categoryType, + categories: categoryType.categories.filter( + createFilterUnusedCategories(indicators) + ), + hideCategoryIdentifiers: true, + selectionType: 'SINGLE', + }, + }, + ]), ], primaryFilters: [], advancedFilters: [], }); const GET_INDICATOR_LIST = gql` - query IndicatorList($plan: ID!) { + query IndicatorList($plan: ID!, $relatedPlanIndicators: Boolean!) { plan(id: $plan) { id features { @@ -130,11 +181,17 @@ const GET_INDICATOR_LIST = gql` parent { id } + common @include(if: $relatedPlanIndicators) { + type { + identifier + name + } + } } } hasIndicatorRelationships } - planIndicators(plan: $plan) { + planIndicators(plan: $plan) @skip(if: $relatedPlanIndicators) { id common { id @@ -162,6 +219,67 @@ const GET_INDICATOR_LIST = gql` } } } + relatedPlanIndicators(plan: $plan) @include(if: $relatedPlanIndicators) { + id + name + level(plan: $plan) + timeResolution + organization { + id + name + } + common { + id + name + normalizations { + unit { + shortName + } + normalizer { + name + id + identifier + } + } + } + latestGraph { + id + } + latestValue { + id + date + value + normalizedValues { + normalizerId + value + } + } + unit { + shortName + } + categories { + id + name + parent { + id + } + type { + id + identifier + name + } + common { + id + identifier + name + order + type { + identifier + name + } + } + } + } } `; @@ -175,26 +293,55 @@ const filterIndicators = ( filters: Filters, categoryIdentifier?: string ) => { - const filterByCategory = (indicator) => - !categoryIdentifier || - !filters[getCategoryString(categoryIdentifier)] || - !!indicator.categories.find( + const filterByCategory = (indicator) => { + if ( + !categoryIdentifier || + !filters[getCategoryString(categoryIdentifier)] + ) { + return true; + } + return indicator.categories.some( ({ type, id }) => filters[getCategoryString(type.identifier)] === id ); + }; + + const filterByCommonCategory = (indicator) => { + const activeFilters = Object.entries(filters).filter( + ([key, value]) => value !== undefined && value !== null + ); - const filterBySearch = (indicator) => - !filters['name'] || - indicator.name.toLowerCase().includes(filters['name'].toLowerCase()); + if (activeFilters.length === 0) { + return true; + } - return indicators.filter( - (indicator) => filterByCategory(indicator) && filterBySearch(indicator) - ); + return activeFilters.every(([filterKey, filterValue]) => { + return indicator.categories.some((category) => { + if (category.common) { + return category.common.id === filterValue; + } + return false; + }); + }); + }; + + const filterBySearch = (indicator) => { + if (!filters['name']) { + return true; + } + return indicator.name.toLowerCase().includes(filters['name'].toLowerCase()); + }; + + return indicators.filter((indicator) => { + const categoryResult = filterByCategory(indicator); + const commonCategoryResult = filterByCommonCategory(indicator); + const searchResult = filterBySearch(indicator); + return ( + (!includeRelatedPlanIndicators ? categoryResult : commonCategoryResult) && + searchResult + ); + }); }; -/** - * IndicatorListFiltered currently only accepts and displays a single category type, - * so we use the first usableForIndicators category type which has associated indicators. - */ const getFirstUsableCategoryType = (categoryTypes, indicators) => categoryTypes.find((categoryType) => indicators.find((indicator) => @@ -215,7 +362,10 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { const { loading, error, data } = useQuery( GET_INDICATOR_LIST, { - variables: { plan: plan.identifier }, + variables: { + plan: plan.identifier, + relatedPlanIndicators: includeRelatedPlanIndicators, + }, } ); @@ -230,21 +380,18 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { const handleFilterChange = (id: string, val: FilterValue) => { setFilters((state) => { const newFilters = { ...state, [id]: val }; - updateSearchParams(newFilters); - return newFilters; }); }; const hasInsights = (data) => { const { plan } = data; - // Check if any of the indicators has causality link return plan.hasIndicatorRelationships === true; }; const getIndicatorListProps = (data) => { - const { plan } = data; + const { plan, relatedPlanIndicators } = data; const displayMunicipality = plan.features.hasActionPrimaryOrgs === true; const displayNormalizedValues = undefined !== @@ -258,7 +405,6 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { const indicators = indicatorLevels.map((il) => { const { indicator, level } = il; - return { ...indicator, level: level.toLowerCase() }; }); @@ -267,6 +413,9 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { leadContent: generalContent.indicatorListLeadContent, displayMunicipality, displayNormalizedValues, + relatedPlanIndicators: includeRelatedPlanIndicators + ? relatedPlanIndicators + : [], }; }; @@ -274,16 +423,30 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { if (error) return ; const indicatorListProps = getIndicatorListProps(data); - const hierarchy = processCommonIndicatorHierarchy(data?.planIndicators); + + const indicators = includeRelatedPlanIndicators + ? indicatorListProps.relatedPlanIndicators + : indicatorListProps.indicators; + + const commonCategories = collectCommonCategories(indicators); + const hierarchy = processCommonIndicatorHierarchy( + includeRelatedPlanIndicators + ? indicatorListProps.relatedPlanIndicators + : data?.planIndicators + ); const showInsights = (displayInsights ?? true) && hasInsights(data); const categoryType = getFirstUsableCategoryType( data?.plan?.categoryTypes, - indicatorListProps.indicators + indicators ); const filterConfig = categoryType - ? getFilterConfig(categoryType, indicatorListProps.indicators) + ? getFilterConfig( + categoryType, + indicators, + includeRelatedPlanIndicators ? commonCategories : [] + ) : {}; const filterSections: ActionListFilterSection[] = @@ -297,7 +460,7 @@ const IndicatorList = ({ leadContent, displayInsights }: Props) => { }); const filteredIndicators = filterIndicators( - indicatorListProps.indicators, + indicators, filters, categoryType?.identifier );