Skip to content

Commit

Permalink
[frontend] Improve PolarArea widget (#6520) (#6521)
Browse files Browse the repository at this point in the history
  • Loading branch information
lndrtrbn authored Apr 9, 2024
1 parent 3e060b3 commit d9694b1
Show file tree
Hide file tree
Showing 12 changed files with 725 additions and 92 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -4,29 +4,37 @@ import { useTheme } from '@mui/styles';
import { ApexOptions } from 'apexcharts';
import { polarAreaChartOptions } from '../../utils/Charts';
import type { Theme } from '../Theme';
import useDistributionGraphData, { DistributionQueryData } from '../../utils/hooks/useDistributionGraphData';

interface WidgetPolarAreaProps {
data: {
label: string,
value: number
}[]
data: DistributionQueryData
groupBy: string
withExport?: boolean
readonly?: boolean
}

const WidgetPolarArea = ({
data,
groupBy,
withExport,
readonly,
}: WidgetPolarAreaProps) => {
const theme = useTheme<Theme>();
const { buildWidgetLabelsOption, buildWidgetColorsOptions } = useDistributionGraphData();

const chartData = data.map((n) => n.value);
const labels = data.map((n) => n.label);
const chartData = data.flatMap((n) => (n ? (n.value ?? 0) : []));
const labels = buildWidgetLabelsOption(data, groupBy);
const colors = buildWidgetColorsOptions(data, groupBy);

return (
<Chart
options={polarAreaChartOptions(theme, labels) as ApexOptions}
options={polarAreaChartOptions(
theme,
labels,
undefined,
'bottom',
colors,
) as ApexOptions}
series={chartData}
type="polarArea"
width="100%"
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,203 @@
/*
Copyright (c) 2021-2024 Filigran SAS
This file is part of the OpenCTI Enterprise Edition ("EE") and is
licensed under the OpenCTI Non-Commercial License (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
https://github.com/OpenCTI-Platform/opencti/blob/master/LICENSE
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
*/

import { graphql, PreloadedQuery, usePreloadedQuery } from 'react-relay';
import React, { CSSProperties } from 'react';
import { AuditsPolarAreaDistributionQuery } from '@components/common/audits/__generated__/AuditsPolarAreaDistributionQuery.graphql';
import { useFormatter } from '../../../../components/i18n';
import WidgetContainer from '../../../../components/dashboard/WidgetContainer';
import WidgetLoader from '../../../../components/dashboard/WidgetLoader';
import useQueryLoading from '../../../../utils/hooks/useQueryLoading';
import WidgetPolarArea from '../../../../components/dashboard/WidgetPolarArea';
import WidgetNoData from '../../../../components/dashboard/WidgetNoData';
import useGranted, { SETTINGS } from '../../../../utils/hooks/useGranted';
import useEnterpriseEdition from '../../../../utils/hooks/useEnterpriseEdition';
import WidgetAccessDenied from '../../../../components/dashboard/WidgetAccessDenied';
import { DashboardWidgetDataSelection, DashboardWidgetParameters } from '../../../../utils/dashboard';

const auditsPolarAreaDistributionQuery = graphql`
query AuditsPolarAreaDistributionQuery(
$field: String!
$startDate: DateTime
$endDate: DateTime
$dateAttribute: String
$operation: StatsOperation!
$limit: Int
$order: String
$types: [String]
$filters: FilterGroup
$search: String
) {
auditsDistribution(
field: $field
startDate: $startDate
endDate: $endDate
dateAttribute: $dateAttribute
operation: $operation
limit: $limit
order: $order
types: $types
filters: $filters
search: $search
) {
label
value
entity {
... on BasicObject {
id
entity_type
}
... on BasicRelationship {
id
entity_type
}
... on StixObject {
representative {
main
}
}
# objects without representative
... on Creator {
name
}
... on Group {
name
}
}
}
}
`;

interface AuditsPolarAreaComponentProps {
dataSelection: DashboardWidgetDataSelection[]
queryRef: PreloadedQuery<AuditsPolarAreaDistributionQuery>
withExportPopover: boolean
isReadOnly: boolean
}

const AuditsPolarAreaComponent = ({
dataSelection,
queryRef,
withExportPopover,
isReadOnly,
}: AuditsPolarAreaComponentProps) => {
const { auditsDistribution } = usePreloadedQuery(
auditsPolarAreaDistributionQuery,
queryRef,
);

if (
auditsDistribution
&& auditsDistribution.length > 0
) {
const attributeField = dataSelection[0].attribute || 'entity_type';
return (
<WidgetPolarArea
data={[...auditsDistribution]}
groupBy={attributeField}
withExport={withExportPopover}
readonly={isReadOnly}
/>
);
}
return <WidgetNoData />;
};

interface AuditsPolarAreaProps {
startDate: string
endDate: string
dataSelection: DashboardWidgetDataSelection[]
parameters: DashboardWidgetParameters
variant: string
height?: CSSProperties['height']
withExportPopover?: boolean
isReadOnly?: boolean
}

const AuditsPolarAreaQueyRef = ({
startDate,
endDate,
dataSelection,
parameters,
height,
variant,
withExportPopover = false,
isReadOnly = false,
}: AuditsPolarAreaProps) => {
const selection = dataSelection[0];
const { t_i18n } = useFormatter();

const queryRef = useQueryLoading<AuditsPolarAreaDistributionQuery>(
auditsPolarAreaDistributionQuery,
{
types: ['History', 'Activity'],
field: selection.attribute || 'entity_type',
operation: 'count',
startDate,
endDate,
dateAttribute:
selection.date_attribute && selection.date_attribute.length > 0
? selection.date_attribute
: 'timestamp',
// eslint-disable-next-line @typescript-eslint/ban-ts-comment
// @ts-ignore Excepts readonly array as variables but have simple array.
filters: selection.filters,
limit: selection.number ?? 10,
},
);

return (
<WidgetContainer
height={height}
title={parameters.title ?? t_i18n('Distribution of history')}
variant={variant}
>
{queryRef ? (
<React.Suspense fallback={<WidgetLoader />}>
<AuditsPolarAreaComponent
queryRef={queryRef}
dataSelection={dataSelection}
withExportPopover={withExportPopover}
isReadOnly={isReadOnly}
/>
</React.Suspense>
) : (
<WidgetLoader />
)}
</WidgetContainer>
);
};

const AuditsPolarArea = (props: AuditsPolarAreaProps) => {
const { t_i18n } = useFormatter();
const isGrantedToSettings = useGranted([SETTINGS]);
const isEnterpriseEdition = useEnterpriseEdition();

if (!isGrantedToSettings || !isEnterpriseEdition) {
const { height, parameters, variant } = props;
return (
<WidgetContainer
height={height}
title={parameters.title ?? t_i18n('Distribution of history')}
variant={variant}
>
<WidgetAccessDenied />
</WidgetContainer>
);
}
return <AuditsPolarAreaQueyRef {...props} />;
};

export default AuditsPolarArea;
Loading

0 comments on commit d9694b1

Please sign in to comment.