Skip to content

Commit

Permalink
create new bad practice components and add to activity dashboard(#199)
Browse files Browse the repository at this point in the history
  • Loading branch information
iam-flo committed Dec 30, 2024
1 parent 5b4b560 commit 0f2385d
Show file tree
Hide file tree
Showing 7 changed files with 215 additions and 58 deletions.
1 change: 1 addition & 0 deletions webapp/src/app/core/header/header.component.html
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@
</a>
@if (user()?.roles?.includes('admin')) {
<a hlmBtn variant="link" routerLink="/workspace">Workspace</a>
<a hlmBtn variant="link" [routerLink]="'/activity/' + user()!.username">Activity</a>
}
</div>
<app-request-feature class="hidden sm:inline-block" />
Expand Down
88 changes: 36 additions & 52 deletions webapp/src/app/home/activity/activity-dashboard.component.html
Original file line number Diff line number Diff line change
@@ -1,59 +1,43 @@
<div class="flex flex-col items-center">
<div class="">
<div class="grid grid-cols-1 xl:grid-cols-5 gap-y-4 xl:gap-8">
<div class="space-y-2 col-span-1">
<div class="flex flex-col gap-2 mb-4">
<h1 class="text-xl font-semibold">Activities</h1>
<p>
You currently have <span class="font-semibold">{{ numberOfPullRequests() }}</span> open pull requests and <span class="font-semibold">{{ numberOfBadPractices() }}</span> detected bad
practices.
</p>
</div>
<div class="grid grid-cols-1 xl:grid-cols-4 gap-y-4 xl:gap-8">
<div class="space-y-2 col-span-1">
<div class="flex flex-col gap-2 mb-4">
<h1 class="text-xl font-semibold">Activities</h1>
<p>
You currently have <span class="font-semibold">{{ numberOfPullRequests() }}</span> open pull requests and
<span class="font-semibold">{{ numberOfBadPractices() }}</span> detected bad practices.
</p>
</div>
<div class="col-span-2">
<div class="flex flex-col justify-between">
</div>
<div class="col-span-2">
<div class="flex flex-col justify-between gap-2">
<span class="flex flex-row justify-between items-center">
<h1 class="text-xl font-semibold">Your open pull requests</h1>
<div class="flex flex-col gap-2 m-1 mr-3">
@if (query.data()?.pullRequests) {
@for (pullRequest of query.data()?.pullRequests; track pullRequest.id) {
<app-issue-card
class="w-full"
[title]="pullRequest.title"
[number]="pullRequest.number"
[additions]="pullRequest.additions"
[deletions]="pullRequest.deletions"
[htmlUrl]="pullRequest.htmlUrl"
[repositoryName]="pullRequest.repository?.name"
[createdAt]="pullRequest.createdAt"
[state]="pullRequest.state"
[isDraft]="pullRequest.isDraft"
[isMerged]="pullRequest.isMerged"
[pullRequestLabels]="pullRequest.labels"
>
</app-issue-card>
}
}
</div>
</div>
</div>
<div class="col-span-2">
<div class="flex flex-col justify-between">
<h1 class="text-xl font-semibold">Detected bad practices</h1>
<div class="flex flex-col gap-2 m-1 mr-3">
@if (query.data()?.pullRequests) {
@for (pullRequest of query.data()?.pullRequests; track pullRequest.id) {
@for (badPractice of pullRequest.badPractices; track badPractice.title) {
<app-bad-practice-card
[title]="badPractice.title"
[description]="badPractice.description"
[repositoryName]="pullRequest.repository?.name"
[number]="pullRequest.number"
>
</app-bad-practice-card>
}
}
<button hlmBtn aria-describedby=">Detect bad practices" class="gap-2" (click)="this.detectBadPractices()">
<lucide-angular [img]="RefreshCcw" class="size-4" />
<span>Detect bad practices</span>
</button>
</span>
<div class="flex flex-col gap-2 m-1 mr-3">
@if (query.data()?.pullRequests) {
@for (pullRequest of query.data()?.pullRequests; track pullRequest.id) {
<app-pull-request-bad-practice-card
[title]="pullRequest.title"
[number]="pullRequest.number"
[additions]="pullRequest.additions"
[deletions]="pullRequest.deletions"
[htmlUrl]="pullRequest.htmlUrl"
[repositoryName]="pullRequest.repository?.name"
[createdAt]="pullRequest.createdAt"
[state]="pullRequest.state"
[isDraft]="pullRequest.isDraft"
[isMerged]="pullRequest.isMerged"
[pullRequestLabels]="pullRequest.labels"
[badPractices]="pullRequest.badPractices"
>
</app-pull-request-bad-practice-card>
}
</div>
}
</div>
</div>
</div>
Expand Down
13 changes: 10 additions & 3 deletions webapp/src/app/home/activity/activity-dashboard.component.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,13 +3,14 @@ import { ActivityService } from '@app/core/modules/openapi';
import { injectQuery } from '@tanstack/angular-query-experimental';
import { combineLatest, lastValueFrom, map, timer } from 'rxjs';
import { ActivatedRoute } from '@angular/router';
import { IssueCardComponent } from '@app/user/issue-card/issue-card.component';
import { BadPracticeCardComponent } from '@app/user/bad-practice-card/bad-practice-card.component';
import { PullRequestBadPracticeCardComponent } from '@app/user/pull-request-bad-practice-card/pull-request-bad-practice-card.component';
import { LucideAngularModule, RefreshCcw } from 'lucide-angular';
import { HlmButtonDirective } from '@spartan-ng/ui-button-helm';

@Component({
selector: 'app-activity-dashboard',
standalone: true,
imports: [IssueCardComponent, BadPracticeCardComponent],
imports: [PullRequestBadPracticeCardComponent, LucideAngularModule, HlmButtonDirective],
templateUrl: './activity-dashboard.component.html',
styles: ``
})
Expand All @@ -29,4 +30,10 @@ export class ActivityDashboardComponent {
enabled: !!this.userLogin,
queryFn: async () => lastValueFrom(combineLatest([this.activityService.getActivityByUser(this.userLogin!), timer(400)]).pipe(map(([activity]) => activity)))
}));

detectBadPractices = () => {
console.log('Detecting bad practices');
//this.activityService.detectBadPractices(this.userLogin!).subscribe();
};
protected readonly RefreshCcw = RefreshCcw;
}
Original file line number Diff line number Diff line change
@@ -1,6 +1,5 @@
<a hlmCard variant="profile" class="block p-4 border rounded-lg shadow-sm">
<div hlmCardContent variant="profile">
<p class="text-xs font-medium text-github-muted-foreground">{{ repositoryName() }} #{{ number() }}</p>
<a class="shadow-sm">
<div>
<h1 class="text-lg font-semibold">{{ title() }}</h1>
<div class="text-sm">{{ description() }}</div>
</div>
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,48 @@
<brn-collapsible>
<button brnCollapsibleTrigger>
<a hlmCard variant="profile">
<div hlmCardHeader>
<div class="flex justify-between items-center text-sm text-github-muted-foreground">
<span class="font-medium flex justify-center items-center space-x-1">
@if (isLoading()) {
<hlm-skeleton class="size-5 bg-green-500/30"></hlm-skeleton>
<hlm-skeleton class="h-4 w-16 lg:w-36"></hlm-skeleton>
} @else {
<ng-icon [svg]="issueIconAndColor().icon" size="18" [class]="'mr-1 ' + issueIconAndColor().color"></ng-icon>
{{ repositoryName() }} #{{ number() }} on {{ displayCreated().format('MMM D') }}
}
</span>
<span class="flex items-center space-x-2">
@if (isLoading()) {
<hlm-skeleton class="h-4 w-8 bg-green-500/30"></hlm-skeleton>
<hlm-skeleton class="h-4 w-8 bg-destructive/20"></hlm-skeleton>
} @else {
<span class="text-github-success-foreground font-bold">+{{ additions() }}</span>
<span class="text-github-danger-foreground font-bold">-{{ deletions() }}</span>
}
</span>
</div>
<span class="flex justify-between font-medium">
@if (isLoading()) {
<hlm-skeleton class="h-6 w-3/4"></hlm-skeleton>
} @else {
<div [innerHTML]="displayTitle()" class="truncate"></div>
}
</span>
<span>
<div class="text-left">You have {{ badPractices()?.length }} bad practices in this pull request.</div>
</span>
</div>
@if (!isLoading()) {
<div hlmCardContent class="gap-2 p-0 space-x-0 pl-6 text-left">
<brn-collapsible-content>
@for (badpractice of badPractices(); track badpractice.title) {
<brn-separator hlmSeparator />
<app-bad-practice-card [title]="badpractice.title" [description]="badpractice.description"></app-bad-practice-card>
}
</brn-collapsible-content>
</div>
}
</a>
</button>
</brn-collapsible>
Original file line number Diff line number Diff line change
@@ -0,0 +1,78 @@
import { Component, computed, input } from '@angular/core';
import { PullRequestInfo, LabelInfo, PullRequestBadPractice } from '@app/core/modules/openapi';
import { NgIcon } from '@ng-icons/core';
import { octCheck, octComment, octFileDiff, octGitPullRequest, octGitPullRequestClosed, octGitPullRequestDraft, octGitMerge, octX } from '@ng-icons/octicons';
import { HlmCardModule } from '@spartan-ng/ui-card-helm';
import { HlmSkeletonComponent } from '@spartan-ng/ui-skeleton-helm';

import dayjs from 'dayjs';
import { BadPracticeCardComponent } from '@app/user/bad-practice-card/bad-practice-card.component';
import { BrnSeparatorComponent } from '@spartan-ng/ui-separator-brain';
import { HlmSeparatorDirective } from '@spartan-ng/ui-separator-helm';
import { BrnCollapsibleComponent, BrnCollapsibleContentComponent, BrnCollapsibleTriggerDirective } from '@spartan-ng/ui-collapsible-brain';

@Component({
selector: 'app-pull-request-bad-practice-card',
templateUrl: './pull-request-bad-practice-card.component.html',
imports: [
NgIcon,
HlmCardModule,
HlmSkeletonComponent,
BadPracticeCardComponent,
BrnSeparatorComponent,
HlmSeparatorDirective,
BrnCollapsibleComponent,
BrnCollapsibleContentComponent,
BrnCollapsibleTriggerDirective
],
standalone: true
})
export class PullRequestBadPracticeCardComponent {
protected readonly octCheck = octCheck;
protected readonly octX = octX;
protected readonly octComment = octComment;
protected readonly octFileDiff = octFileDiff;

isLoading = input(false);
class = input('');
title = input<string>();
number = input<number>();
additions = input<number>();
deletions = input<number>();
htmlUrl = input<string>();
repositoryName = input<string>();
createdAt = input<string>();
state = input<PullRequestInfo.StateEnum>();
isDraft = input<boolean>();
isMerged = input<boolean>();
pullRequestLabels = input<Array<LabelInfo>>();
badPractices = input<Array<PullRequestBadPractice>>();

displayCreated = computed(() => dayjs(this.createdAt()));
displayTitle = computed(() => (this.title() ?? '').replace(/`([^`]+)`/g, '<code class="textCode">$1</code>'));

issueIconAndColor = computed(() => {
var icon: string;
var color: string;

if (this.state() === PullRequestInfo.StateEnum.Open) {
if (this.isDraft()) {
icon = octGitPullRequestDraft;
color = 'text-github-muted-foreground';
} else {
icon = octGitPullRequest;
color = 'text-github-open-foreground';
}
} else {
if (this.isMerged()) {
icon = octGitMerge;
color = 'text-github-done-foreground';
} else {
icon = octGitPullRequestClosed;
color = 'text-github-closed-foreground';
}
}

return { icon, color };
});
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
import { Meta, StoryObj } from '@storybook/angular';
import { PullRequestBadPracticeCardComponent } from './pull-request-bad-practice-card.component';

const meta: Meta<PullRequestBadPracticeCardComponent> = {
component: PullRequestBadPracticeCardComponent,
tags: ['autodocs'] // Auto-generate docs if enabled
};

export default meta;

type Story = StoryObj<PullRequestBadPracticeCardComponent>;

export const Default: Story = {
args: {
title: 'Add feature X',
number: 12,
additions: 10,
deletions: 5,
htmlUrl: 'http://example.com',
state: 'OPEN',
isDraft: false,
isMerged: false,
repositoryName: 'Artemis',
createdAt: '2024-01-01',
pullRequestLabels: [
{ id: 1, name: 'bug', color: 'f00000' },
{ id: 2, name: 'enhancement', color: '008000' }
],
badPractices: [
{
title: 'Avoid using any type',
description: 'Using the any type defeats the purpose of TypeScript.'
},
{
title: 'Unchecked checkbox in description',
description: 'Unchecked checkboxes in the description are not allowed.'
}
]
}
};

0 comments on commit 0f2385d

Please sign in to comment.