diff --git a/.detective/config.json b/.detective/config.json index ea5d8b6..4c94d8e 100644 --- a/.detective/config.json +++ b/.detective/config.json @@ -4,9 +4,32 @@ "apps/backend/src/options", "apps/backend/src/services", "apps/backend/src/utils", - "apps/backend/src/infrastructure" + "apps/backend/src/infrastructure", + "apps/frontend/src/app/features/coupling", + "apps/frontend/src/app/features/hotspot", + "apps/frontend/src/app/features/team-alignment", + "apps/frontend/src/app/shell/about", + "apps/frontend/src/app/shell/filter-tree", + "apps/frontend/src/app/shell/nav", + "apps/frontend/src/app/model", + "apps/frontend/src/app/ui/doughnut", + "apps/frontend/src/app/ui/graph", + "apps/frontend/src/app/ui/limits", + "apps/frontend/src/app/ui/loading", + "apps/frontend/src/app/ui/resizer", + "apps/frontend/src/app/ui/treemap" + ], + "groups": [ + "apps/backend/src", + "apps/backend", + "apps/frontend/src/app/features", + "apps/frontend/src/app/shell", + "apps/frontend/src/app/ui", + "apps/frontend/src/app", + "apps/frontend/src", + "apps/frontend", + "apps" ], - "groups": ["apps/backend/src", "apps/backend", "apps"], "entries": [], "filter": { "files": [], diff --git a/.detective/hash b/.detective/hash index 7c5e3ee..10770bb 100644 --- a/.detective/hash +++ b/.detective/hash @@ -1 +1 @@ -979780ac55f4f8c3680055a838a645cdc8e591a0, v1.1.2 \ No newline at end of file +503c63bcf7fab325fc9687a40ad048b2b16ca636, v1.1.6 \ No newline at end of file diff --git a/.detective/log b/.detective/log index 34cb500..17f768c 100644 --- a/.detective/log +++ b/.detective/log @@ -1,9 +1,47 @@ -"Manfred Steyer ,Sat Sep 28 23:22:37 2024 +0200 947a6f7933d93eeae42e53bbffd8d56f69486095,feat: add resizer for tree" +"Manfred Steyer ,Tue Oct 8 17:44:47 2024 +0200 6ed670acc998f83279713bc9c2a16ef7a79f66a1,feat: filter hotspots using slider and percent" +1 1 .detective/hash +33 3 .detective/log +70 11 apps/backend/src/services/hotspot.ts +8 10 apps/frontend/src/app/features/hotspot/hotspot.component.html +2 0 apps/frontend/src/app/features/hotspot/hotspot.component.ts +1 1 apps/frontend/src/app/features/hotspot/hotspot.store.ts + +"Manfred Steyer ,Sun Sep 29 12:41:39 2024 +0200 4ee102065cca1a50063a871b078ec87680116efb,chore(release): publish 1.1.6" +10 0 CHANGELOG.md +1 1 apps/backend/package.json + +"Manfred Steyer ,Sun Sep 29 12:41:19 2024 +0200 0130a5533f0979c406e469c74305d8e214b4e0ee,fix: prevent word wrap in tree" +5 0 apps/frontend/src/app/shell/filter-tree/filter-tree.component.css + +"Manfred Steyer ,Sun Sep 29 12:29:48 2024 +0200 0f519048ce8d0d999ea0a9fc7a1c8953513eccbf,chore(release): publish 1.1.5" +10 0 CHANGELOG.md +1 1 apps/backend/package.json + +"Manfred Steyer ,Sun Sep 29 12:29:29 2024 +0200 527ed589150ac81ee5bc251526644cfdc6eeb11e,fix: don't wrap tree entries" +4 0 apps/frontend/src/app/shell/nav/nav.component.css + +"Manfred Steyer ,Sun Sep 29 12:12:29 2024 +0200 7a18a2d6fb97ea4d0d2bea59bd99b6b376862772,chore(release): publish 1.1.4" +10 0 CHANGELOG.md +1 1 apps/backend/package.json + +"Manfred Steyer ,Sun Sep 29 12:12:15 2024 +0200 be5836b768b8fc6d307496ace2d037cc7063c56d,fix: always show resize cursor during resizing" +2 0 apps/frontend/src/app/ui/resizer/resizer.component.ts +4 0 apps/frontend/src/styles.scss + +"Manfred Steyer ,Sun Sep 29 00:19:03 2024 +0200 567e19278e6204fd4b29ca9490043fe1fc8d7431,chore(release): publish 1.1.3" +14 0 CHANGELOG.md +1 1 apps/backend/package.json + +"Manfred Steyer ,Sat Sep 28 23:22:37 2024 +0200 7ef551cfee65bf6a037923eb6310ddf72a2a1e52,feat: add resizer for tree" 1 1 .detective/hash 16 0 .detective/log -25 0 apps/frontend/src/app/shell/nav/nav.component.css +16 2 apps/frontend/src/app/shell/nav/nav.component.css 10 6 apps/frontend/src/app/shell/nav/nav.component.html -46 8 apps/frontend/src/app/shell/nav/nav.component.ts +15 8 apps/frontend/src/app/shell/nav/nav.component.ts +10 0 apps/frontend/src/app/ui/resizer/resizer.component.css +1 0 apps/frontend/src/app/ui/resizer/resizer.component.html +22 0 apps/frontend/src/app/ui/resizer/resizer.component.spec.ts +42 0 apps/frontend/src/app/ui/resizer/resizer.component.ts "Manfred Steyer ,Sat Sep 28 19:32:42 2024 +0200 06c8dfeb5e988f17bcf3699e8d3c1748981c7231,refactor: streamline dataflow for hotspot analysis" 1 1 .detective/hash diff --git a/apps/backend/project.json b/apps/backend/project.json index 8b1db5d..d339c51 100644 --- a/apps/backend/project.json +++ b/apps/backend/project.json @@ -12,7 +12,12 @@ "options": { "buildTarget": "backend:build", "runBuildTargetDependencies": false, - "args": ["--path", ".", "--open", "false"] + "args": [ + "--path", + "/Users/manfredsteyer/projects/public/standalone-example-cli", + "--open", + "false" + ] }, "configurations": { "development": { diff --git a/apps/backend/src/services/hotspot.ts b/apps/backend/src/services/hotspot.ts index 212517f..51c1698 100644 --- a/apps/backend/src/services/hotspot.ts +++ b/apps/backend/src/services/hotspot.ts @@ -38,12 +38,26 @@ export type HotspotCriteria = { }; export type AggregatedHotspot = { + parent: string; module: string; count: number; + countWarning: number; + countHotspot: number; + countOk: number; }; export type AggregatedHotspotsResult = { aggregated: AggregatedHotspot[]; + minScore: number; + maxScore: number; + warningBoundary: number; + hotspotBoundary: number; +}; + +type Stats = { + maxScore: number; + scores: Map; + minScore: number; }; export async function findHotspotFiles( @@ -80,28 +94,99 @@ export async function aggregateHotspots( limits: Limits, options: Options ): Promise { - const hotspotResult = await findHotspotFiles(criteria, limits, options); + const phase1Criteria = { + ...criteria, + minScore: 0, + }; + + const hotspotResult = await findHotspotFiles(phase1Criteria, limits, options); const hotspots = hotspotResult.hotspots; const config = loadConfig(options); const modules = config.scopes.map((m) => normalizeFolder(m)); + const stats = collectStats(modules, hotspots); + + const warningBoundary = stats.maxScore * (criteria.minScore / 100); + const hotspotBoundary = + warningBoundary + (stats.maxScore - warningBoundary) / 2; + + const result = aggregateStats( + modules, + stats, + warningBoundary, + hotspotBoundary + ); + + result.sort((a, b) => b.count - a.count); + + return { + aggregated: result, + maxScore: stats.maxScore, + minScore: stats.minScore, + hotspotBoundary, + warningBoundary, + }; +} + +function aggregateStats( + modules: string[], + stats: Stats, + warningBoundary: number, + hotspotBoundary: number +): AggregatedHotspot[] { const result: AggregatedHotspot[] = []; for (const module of modules) { - let count = 0; - for (const hotspot of hotspots) { - if ( - hotspot.fileName.startsWith(module) - //&& hotspot.score >= criteria.minScore - ) { - count++; + const moduleStats = stats.scores.get(module); + + let countWarning = 0; + let countHotspot = 0; + let countOk = 0; + + for (const stat of moduleStats) { + if (stat > hotspotBoundary) { + countHotspot++; + } else if (stat < warningBoundary) { + countOk++; + } else { + countWarning++; } } - result.push({ module: toDisplayFolder(module), count }); + + // const countBelow = moduleStats.length - count; + + const displayFolder = toDisplayFolder(module); + const parent = path.dirname(displayFolder); + + result.push({ + parent, + module: displayFolder, + count: countOk, + countOk, + countWarning, + countHotspot, + }); } + return result; +} - result.sort((a, b) => b.count - a.count); - return { aggregated: result }; +function collectStats(modules: string[], hotspots: FlatHotspot[]) { + let minScore = Number.MAX_VALUE; + let maxScore = 0; + const scores = new Map(); + + for (const module of modules) { + const moduleScores = []; + for (const hotspot of hotspots) { + if (hotspot.fileName.startsWith(module)) { + minScore = Math.min(minScore, hotspot.score); + maxScore = Math.max(maxScore, hotspot.score); + moduleScores.push(hotspot.score); + } + } + scores.set(module, moduleScores); + } + return { maxScore, scores, minScore }; } async function analyzeLogs( diff --git a/apps/frontend/src/app/features/hotspot/hotspot-adapter.ts b/apps/frontend/src/app/features/hotspot/hotspot-adapter.ts new file mode 100644 index 0000000..122bc5e --- /dev/null +++ b/apps/frontend/src/app/features/hotspot/hotspot-adapter.ts @@ -0,0 +1,101 @@ +import 'chartjs-chart-treemap'; + +import { ChartEvent, InteractionItem } from 'chart.js'; + +import { AggregatedHotspot } from '../../model/hotspot-result'; +import { TreeMapChartConfig } from '../../ui/treemap/treemap.component'; + +export type ScoreType = 'hotspot' | 'warning' | 'fine'; +export type AggregatedHotspotWithType = AggregatedHotspot & { + type: ScoreType; +}; + +export function toTreeMapConfig( + aggregated: AggregatedHotspot[] +): TreeMapChartConfig { + const values = aggregated.flatMap((v) => [ + { ...v, count: v.countHotspot, type: 'hotspot' }, + { ...v, count: v.countWarning, type: 'warning' }, + { ...v, count: v.countOk, type: 'fine' }, + ]); + + const options = { + onHover: (event: ChartEvent, elements: InteractionItem[]) => { + const chartElement = event.native?.target as HTMLCanvasElement; + if (elements.length >= 2) { + chartElement.style.cursor = 'pointer'; + } else { + chartElement.style.cursor = 'default'; + } + }, + plugins: { + title: { + display: true, + text: 'Hotspots', + }, + legend: { + display: false, + }, + tooltip: { + callbacks: { + title() { + return 'File Count'; + }, + }, + }, + }, + }; + + const config: TreeMapChartConfig = { + type: 'treemap', + data: { + datasets: [ + { + data: values, + key: 'count', + groups: ['parent', 'module', 'type'], + spacing: 1, + borderWidth: 0.5, + borderColor: '#EFEFEF', + backgroundColor: (ctx) => { + if (typeof ctx.raw?.l !== 'undefined' && ctx.raw?.l < 2) { + return '#EFEFEF'; + } + return getScoreTypeColor(ctx.raw?.g as ScoreType); + }, + captions: { + align: 'center', + display: true, + color: 'black', + font: { + size: 14, + }, + hoverFont: { + size: 16, + weight: 'bold', + }, + padding: 5, + }, + labels: { + display: false, + overflow: 'hidden', + }, + }, + ], + }, + options: options, + }; + + return config; +} + +export function getScoreTypeColor(scoreType: ScoreType) { + switch (scoreType) { + case 'hotspot': + return '#E74C3C'; + case 'warning': + return '#F1C40F'; + case 'fine': + return '#2ECC71'; + } +} diff --git a/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.css b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.css new file mode 100644 index 0000000..17bacc4 --- /dev/null +++ b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.css @@ -0,0 +1,4 @@ +.title { + margin-left: 15px; + margin-top: 10px; +} diff --git a/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.html b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.html new file mode 100644 index 0000000..9007618 --- /dev/null +++ b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.html @@ -0,0 +1,53 @@ +

+
+ {{ module() }} +
+

+ +
+
+ @if (loadingHotspots()) { Determining Hotspots ... + + } @else if (hotspotResult().hotspots.length > 0) { +
+ + + + + + + + + + + + + + + + + + + + + + + +
Module{{ element.fileName }}Commits + {{ element.commits }} + + Complexity + + {{ element.complexity }} + Score + {{ element.score }} +
+
+ } +
+
+ + diff --git a/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.ts b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.ts new file mode 100644 index 0000000..1bf1cf0 --- /dev/null +++ b/apps/frontend/src/app/features/hotspot/hotspot-detail/hotspot-detail.component.ts @@ -0,0 +1,87 @@ +import { CommonModule } from '@angular/common'; +import { + Component, + computed, + effect, + inject, + untracked, + viewChild, +} from '@angular/core'; +import { MatDialogModule } from '@angular/material/dialog'; +import { MatPaginator } from '@angular/material/paginator'; +import { MatProgressBar } from '@angular/material/progress-bar'; +import { MatSortModule } from '@angular/material/sort'; +import { MatTableDataSource, MatTableModule } from '@angular/material/table'; + +import { FlatHotspot } from '../../../model/hotspot-result'; +import { getScoreTypeColor } from '../hotspot-adapter'; +import { HotspotStore } from '../hotspot.store'; + +@Component({ + selector: 'app-hotspot-detail', + standalone: true, + imports: [ + CommonModule, + MatTableModule, + MatSortModule, + MatPaginator, + MatProgressBar, + MatDialogModule, + ], + templateUrl: './hotspot-detail.component.html', + styleUrl: './hotspot-detail.component.css', +}) +export class HotspotDetailComponent { + private hotspotStore = inject(HotspotStore); + + module = this.hotspotStore.filter.module; + color = computed(() => getScoreTypeColor(this.hotspotStore.scoreType())); + + paginator = viewChild(MatPaginator); + detailDataSource = new MatTableDataSource(); + detailColumns = ['fileName', 'commits', 'complexity', 'score']; + + loadingAggregated = this.hotspotStore.loadingAggregated; + loadingHotspots = this.hotspotStore.loadingHotspots; + + aggregatedResult = this.hotspotStore.aggregatedResult; + hotspotResult = this.hotspotStore.hotspotResult; + + formattedHotspots = computed(() => + formatHotspots( + this.hotspotResult().hotspots, + untracked(() => this.hotspotStore.filter.module()) + ) + ); + + constructor() { + effect(() => { + const hotspots = this.formattedHotspots(); + this.detailDataSource.data = hotspots; + }); + + effect(() => { + const paginator = this.paginator(); + if (paginator) { + this.detailDataSource.paginator = paginator; + } + }); + } +} + +function formatHotspots( + hotspot: FlatHotspot[], + selectedModule: string +): FlatHotspot[] { + return hotspot.map((hs) => ({ + ...hs, + fileName: trimSegments(hs.fileName, selectedModule), + })); +} + +function trimSegments(fileName: string, prefix: string): string { + if (fileName.startsWith(prefix)) { + return fileName.substring(prefix.length + 1); + } + return fileName; +} diff --git a/apps/frontend/src/app/features/hotspot/hotspot.component.html b/apps/frontend/src/app/features/hotspot/hotspot.component.html index 7f3b147..6d1fd46 100644 --- a/apps/frontend/src/app/features/hotspot/hotspot.component.html +++ b/apps/frontend/src/app/features/hotspot/hotspot.component.html @@ -1,15 +1,13 @@
- - Min. Score - - + Min. Score: (%) + + + + + {{ minScore().value() }} % Complexity Metric @@ -37,83 +35,7 @@
-
-
- @if (loadingAggregated() || (loadingAggregated() && selectedModule())) { - Determining Hotspots ... - - } @else { - - - - - - - - - - - - - -
Module{{ element.module }}Count{{ element.count }}
- } -
-
- -
-
- @if (loadingHotspots()) { Determining Hotspots ... - - } @else if (hotspotResult().hotspots.length > 0) { -
- - - - - - - - - - - - - - - - - - - - - - - -
Module{{ element.fileName }}Commits - {{ element.commits }} - - Complexity - - {{ element.complexity }} - Score - {{ element.score }} -
- -
- } -
-
+ diff --git a/apps/frontend/src/app/features/hotspot/hotspot.component.ts b/apps/frontend/src/app/features/hotspot/hotspot.component.ts index db1e66a..78898ec 100644 --- a/apps/frontend/src/app/features/hotspot/hotspot.component.ts +++ b/apps/frontend/src/app/features/hotspot/hotspot.component.ts @@ -1,21 +1,16 @@ -import { - Component, - computed, - effect, - inject, - untracked, - viewChild, -} from '@angular/core'; +import { Component, computed, inject } from '@angular/core'; import { takeUntilDestroyed, toObservable } from '@angular/core/rxjs-interop'; import { FormsModule } from '@angular/forms'; +import { MatDialog, MatDialogModule } from '@angular/material/dialog'; import { MatFormFieldModule } from '@angular/material/form-field'; import { MatIconModule } from '@angular/material/icon'; import { MatInputModule } from '@angular/material/input'; -import { MatPaginator, MatPaginatorModule } from '@angular/material/paginator'; +import { MatPaginatorModule } from '@angular/material/paginator'; import { MatProgressBarModule } from '@angular/material/progress-bar'; import { MatSelectModule } from '@angular/material/select'; +import { MatSliderModule } from '@angular/material/slider'; import { MatSortModule } from '@angular/material/sort'; -import { MatTableDataSource, MatTableModule } from '@angular/material/table'; +import { MatTableModule } from '@angular/material/table'; import { MatTooltipModule } from '@angular/material/tooltip'; import { combineLatest, startWith } from 'rxjs'; @@ -24,15 +19,24 @@ import { StatusStore } from '../../data/status.store'; import { AggregatedHotspot, ComplexityMetric, - FlatHotspot, } from '../../model/hotspot-result'; import { Limits } from '../../model/limits'; import { LimitsComponent } from '../../ui/limits/limits.component'; +import { + TreeMapComponent, + TreeMapEvent, +} from '../../ui/treemap/treemap.component'; import { debounceTimeSkipFirst } from '../../utils/debounce'; import { EventService } from '../../utils/event.service'; import { lastSegments } from '../../utils/segments'; import { mirror } from '../../utils/signal-helpers'; +import { + AggregatedHotspotWithType, + ScoreType, + toTreeMapConfig, +} from './hotspot-adapter'; +import { HotspotDetailComponent } from './hotspot-detail/hotspot-detail.component'; import { HotspotStore } from './hotspot.store'; interface Option { @@ -49,12 +53,15 @@ interface Option { MatFormFieldModule, MatInputModule, MatSelectModule, + MatSliderModule, MatProgressBarModule, MatPaginatorModule, + MatDialogModule, LimitsComponent, FormsModule, MatIconModule, MatTooltipModule, + TreeMapComponent, ], templateUrl: './hotspot.component.html', styleUrl: './hotspot.component.css', @@ -66,12 +73,9 @@ export class HotspotComponent { private eventService = inject(EventService); - paginator = viewChild(MatPaginator); - - detailDataSource = new MatTableDataSource(); + private dialog = inject(MatDialog); columnsToDisplay = ['module', 'count']; - detailColumns = ['fileName', 'commits', 'complexity', 'score']; metricOptions: Option[] = [ { id: 'Length', label: 'File Length' }, @@ -83,24 +87,15 @@ export class HotspotComponent { minScore = mirror(this.hotspotStore.filter.minScore); metric = mirror(this.hotspotStore.filter.metric); - selectedModule = mirror(this.hotspotStore.filter.module); loadingAggregated = this.hotspotStore.loadingAggregated; - loadingHotspots = this.hotspotStore.loadingHotspots; - aggregatedResult = this.hotspotStore.aggregatedResult; - hotspotResult = this.hotspotStore.hotspotResult; formattedAggregated = computed(() => formatAggregated(this.aggregatedResult().aggregated) ); - formattedHotspots = computed(() => - formatHotspots( - this.hotspotResult().hotspots, - untracked(() => this.selectedModule().value()) - ) - ); + treeMapConfig = computed(() => toTreeMapConfig(this.formattedAggregated())); constructor() { const loadAggregatedEvents = { @@ -112,71 +107,78 @@ export class HotspotComponent { metric: toObservable(this.metric().value), }; - const loadHotspotEvent = { - ...loadAggregatedEvents, - selectedModule: toObservable(this.selectedModule().value), - }; - const loadAggregatedOptions$ = combineLatest(loadAggregatedEvents).pipe( takeUntilDestroyed() ); - const loadHotspotOptions$ = combineLatest(loadHotspotEvent).pipe( - takeUntilDestroyed() - ); - this.hotspotStore.rxLoadAggregated(loadAggregatedOptions$); - this.hotspotStore.rxLoadHotspots(loadHotspotOptions$); + } - effect(() => { - const hotspots = this.formattedHotspots(); - this.detailDataSource.data = hotspots; + updateLimits(limits: Limits): void { + this.limitsStore.updateLimits(limits); + } + + selectModule(event: TreeMapEvent): void { + const selected = event.entry as AggregatedHotspotWithType; + const selectedModule = [selected.parent, selected.module].join('/'); + const scoreRange = this.getScoreRange(selected); + + this.hotspotStore.rxLoadHotspots({ + limits: this.limits(), + metric: this.metric().value(), + selectedModule, + scoreRange, + scoreType: selected.type, }); - effect(() => { - const paginator = this.paginator(); - if (paginator) { - this.detailDataSource.paginator = paginator; - } + this.dialog.open(HotspotDetailComponent, { + width: '95%', + height: '700px', }); } - updateLimits(limits: Limits): void { - this.limitsStore.updateLimits(limits); - } + private getScoreRange(selected: AggregatedHotspotWithType) { + const range = this.getScoreBoundaries(); + const index = getScoreIndex(selected.type); - selectRow(row: AggregatedHotspot, index: number) { - const selectedModule = this.aggregatedResult().aggregated[index].module; - this.selectedModule().value.set(selectedModule); + const scoreRange = { + from: range[index], + to: range[index + 1], + }; + return scoreRange; } - isSelected(index: number) { - const module = this.aggregatedResult().aggregated[index].module; - const result = module === this.selectedModule().value(); - return result; + private getScoreBoundaries() { + const result = this.aggregatedResult(); + const range = [ + 0, + result.warningBoundary, + result.hotspotBoundary, + result.maxScore, + ]; + return range; } } -function formatAggregated(hotspot: AggregatedHotspot[]): AggregatedHotspot[] { - return hotspot.map((hs) => ({ - ...hs, - module: lastSegments(hs.module, 3), - })); +function getScoreIndex(type: ScoreType) { + let index = 0; + switch (type) { + case 'fine': + index = 0; + break; + case 'warning': + index = 1; + break; + case 'hotspot': + index = 2; + break; + } + return index; } -function formatHotspots( - hotspot: FlatHotspot[], - selectedModule: string -): FlatHotspot[] { +function formatAggregated(hotspot: AggregatedHotspot[]): AggregatedHotspot[] { return hotspot.map((hs) => ({ ...hs, - fileName: trimSegments(hs.fileName, selectedModule), + module: lastSegments(hs.module, 1), })); } - -function trimSegments(fileName: string, prefix: string): string { - if (fileName.startsWith(prefix)) { - return fileName.substring(prefix.length + 1); - } - return fileName; -} diff --git a/apps/frontend/src/app/features/hotspot/hotspot.store.ts b/apps/frontend/src/app/features/hotspot/hotspot.store.ts index bc17081..7ec22b8 100644 --- a/apps/frontend/src/app/features/hotspot/hotspot.store.ts +++ b/apps/frontend/src/app/features/hotspot/hotspot.store.ts @@ -15,6 +15,8 @@ import { import { Limits } from '../../model/limits'; import { injectShowError } from '../../utils/error-handler'; +import { ScoreType } from './hotspot-adapter'; + export type HotspotFilter = { minScore: number; metric: ComplexityMetric; @@ -27,18 +29,34 @@ export type LoadAggregateOptions = { limits: Limits; }; -export type LoadHotspotOptions = LoadAggregateOptions & { +export type LoadHotspotOptions = { selectedModule: string; + scoreRange: ScoreRange; + metric: ComplexityMetric; + limits: Limits; + scoreType: ScoreType; +}; + +export type ScoreRange = { + from: number; + to: number; +}; + +const initScoreRange: ScoreRange = { + from: 0, + to: 0, }; export const HotspotStore = signalStore( { providedIn: 'root' }, withState({ filter: { - minScore: 1, + minScore: 33, metric: 'Length', module: '', + scoreRange: initScoreRange, } as HotspotFilter, + scoreType: 'fine' as ScoreType, aggregatedResult: initAggregatedHotspotsResult, hotspotResult: initHotspotResult, loadingAggregated: false, @@ -86,14 +104,20 @@ export const HotspotStore = signalStore( _loadHotspots(options: LoadHotspotOptions): Observable { const criteria: HotspotCriteria = { metric: options.metric, - minScore: options.minScore, + minScore: 0, module: options.selectedModule, }; - patchState(store, { + patchState(store, (state) => ({ loadingHotspots: true, - filter: criteria, - }); + filter: { + ...state.filter, + module: options.selectedModule, + minScore: state.filter.minScore, + scoreRange: options.scoreRange, + }, + scoreType: options.scoreType, + })); return hotspotService.load(criteria, options.limits).pipe( tap(() => { diff --git a/apps/frontend/src/app/model/hotspot-result.ts b/apps/frontend/src/app/model/hotspot-result.ts index 14be48d..a353d01 100644 --- a/apps/frontend/src/app/model/hotspot-result.ts +++ b/apps/frontend/src/app/model/hotspot-result.ts @@ -25,19 +25,31 @@ export type ComplexityMetric = 'McCabe' | 'Length'; export interface HotspotCriteria { module: string; - minScore: number; metric: ComplexityMetric; + minScore: number; } export interface AggregatedHotspot { + parent: string; module: string; count: number; + countWarning: number; + countHotspot: number; + countOk: number; } export interface AggregatedHotspotsResult { aggregated: AggregatedHotspot[]; + maxScore: number; + minScore: number; + warningBoundary: number; + hotspotBoundary: number; } export const initAggregatedHotspotsResult: AggregatedHotspotsResult = { aggregated: [], + maxScore: 0, + minScore: 0, + warningBoundary: 0, + hotspotBoundary: 0, }; diff --git a/apps/frontend/src/app/ui/treemap/treemap.component.css b/apps/frontend/src/app/ui/treemap/treemap.component.css new file mode 100644 index 0000000..e69de29 diff --git a/apps/frontend/src/app/ui/treemap/treemap.component.html b/apps/frontend/src/app/ui/treemap/treemap.component.html new file mode 100644 index 0000000..bcacfdc --- /dev/null +++ b/apps/frontend/src/app/ui/treemap/treemap.component.html @@ -0,0 +1,3 @@ +
+ +
diff --git a/apps/frontend/src/app/ui/treemap/treemap.component.ts b/apps/frontend/src/app/ui/treemap/treemap.component.ts new file mode 100644 index 0000000..3bcb21a --- /dev/null +++ b/apps/frontend/src/app/ui/treemap/treemap.component.ts @@ -0,0 +1,101 @@ +import { CommonModule } from '@angular/common'; +import { + Component, + ElementRef, + input, + OnChanges, + OnDestroy, + output, + SimpleChanges, + viewChild, +} from '@angular/core'; +import { + CategoryScale, + Chart, + ChartConfiguration, + ChartEvent, + InteractionItem, + Legend, + LinearScale, + Title, + Tooltip, +} from 'chart.js'; +import { TreemapController, TreemapElement } from 'chartjs-chart-treemap'; + +export type TreeMapChartConfig = ChartConfiguration< + 'treemap', + object[], + string +>; + +type TreeMapChart = Chart<'treemap', object[], string>; + +type Item = { + _data: { + children: unknown[]; + }; +}; + +export type TreeMapEvent = { + entry: unknown; +}; + +Chart.register( + TreemapElement, + LinearScale, + CategoryScale, + Tooltip, + Legend, + Title, + TreemapController +); + +@Component({ + selector: 'app-treemap', + standalone: true, + imports: [CommonModule], + templateUrl: './treemap.component.html', + styleUrl: './treemap.component.css', +}) +export class TreeMapComponent implements OnChanges, OnDestroy { + canvasRef = viewChild.required>('canvas'); + chartConfig = input.required(); + + elementSelected = output(); + + private chart: TreeMapChart | undefined; + + ngOnChanges(_changes: SimpleChanges): void { + const canvasRef = this.canvasRef(); + const canvas = canvasRef.nativeElement; + const ctx = canvas.getContext('2d'); + const config = this.chartConfig(); + + if (!ctx) { + throw new Error('2d context not found'); + } + + this.chart?.destroy(); + + config.options = config.options ?? {}; + config.options.onClick = ( + _event: ChartEvent, + elements: InteractionItem[] + ) => { + if (elements.length > 0) { + const element = elements[elements.length - 1]; + const dataIndex = element.index; + const dataset = config.data.datasets[0]; + const data = dataset.data; + const item = data[dataIndex] as Item; + const entry = item._data.children[0]; + this.elementSelected.emit({ entry }); + } + }; + this.chart = new Chart(ctx, config); + } + + ngOnDestroy(): void { + this.chart?.destroy(); + } +} diff --git a/package-lock.json b/package-lock.json index db82541..01c4e4c 100644 --- a/package-lock.json +++ b/package-lock.json @@ -24,6 +24,7 @@ "@softarc/sheriff-core": "^0.17.1", "axios": "^1.6.0", "chart.js": "^4.4.4", + "chartjs-chart-treemap": "^3.1.0", "cytoscape": "^3.30.2", "cytoscape-cola": "^2.5.1", "cytoscape-dagre": "^2.5.0", @@ -12221,6 +12222,15 @@ "pnpm": ">=8" } }, + "node_modules/chartjs-chart-treemap": { + "version": "3.1.0", + "resolved": "https://registry.npmjs.org/chartjs-chart-treemap/-/chartjs-chart-treemap-3.1.0.tgz", + "integrity": "sha512-0LJxj4J9sCTHmrXCFlqtoBKMJDcS7VzFeRgNBRZRwU1QSpCXJKTNk5TysPEs5/YW0XYvZoN8u44RqqLf0pAzQw==", + "license": "MIT", + "peerDependencies": { + "chart.js": ">=3.0.0" + } + }, "node_modules/chokidar": { "version": "3.6.0", "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz", diff --git a/package.json b/package.json index e0df8c0..2329ccb 100644 --- a/package.json +++ b/package.json @@ -23,6 +23,7 @@ "@softarc/sheriff-core": "^0.17.1", "axios": "^1.6.0", "chart.js": "^4.4.4", + "chartjs-chart-treemap": "^3.1.0", "cytoscape": "^3.30.2", "cytoscape-cola": "^2.5.1", "cytoscape-dagre": "^2.5.0",