diff --git a/webapp/eslint.config.js b/webapp/eslint.config.js index 5329136c..3ecbff00 100644 --- a/webapp/eslint.config.js +++ b/webapp/eslint.config.js @@ -8,6 +8,7 @@ module.exports = [ { ignores: [ '.cache/', + '.angular/', '.git/', '.github/', 'build/', @@ -54,10 +55,7 @@ module.exports = [ style: 'kebab-case', }, ], - '@angular-eslint/prefer-standalone': 'error', - '@angular-eslint/template/prefer-ngsrc': 'error', - '@angular-eslint/template/prefer-self-closing-tags': 'error', - '@angular-eslint/template/prefer-control-flow': 'error', + '@typescript-eslint/no-empty-object-type': 'off', }, }, { diff --git a/webapp/src/app/app.component.html b/webapp/src/app/app.component.html index 55ae350d..460bba46 100644 --- a/webapp/src/app/app.component.html +++ b/webapp/src/app/app.component.html @@ -3,9 +3,9 @@

Header

- - + +
- + - \ No newline at end of file + diff --git a/webapp/src/app/app.component.ts b/webapp/src/app/app.component.ts index 3bf1d72a..513811ab 100644 --- a/webapp/src/app/app.component.ts +++ b/webapp/src/app/app.component.ts @@ -8,7 +8,7 @@ import { HelloComponent } from './example/hello/hello.component'; standalone: true, imports: [RouterOutlet, CounterComponent, HelloComponent], templateUrl: './app.component.html', - styles: [], + styles: [] }) export class AppComponent { title = 'Hephaestus'; diff --git a/webapp/src/app/app.config.ts b/webapp/src/app/app.config.ts index fc69a29e..c3b646d2 100644 --- a/webapp/src/app/app.config.ts +++ b/webapp/src/app/app.config.ts @@ -1,22 +1,18 @@ import { ApplicationConfig, importProvidersFrom, provideExperimentalZonelessChangeDetection } from '@angular/core'; import { provideRouter } from '@angular/router'; import { provideHttpClient, withInterceptorsFromDi } from '@angular/common/http'; -import { - provideAngularQuery, - QueryClient, -} from '@tanstack/angular-query-experimental' -import {LucideAngularModule, Home } from "lucide-angular"; +import { provideAngularQuery, QueryClient } from '@tanstack/angular-query-experimental'; +import { LucideAngularModule, Home } from 'lucide-angular'; import { routes } from './app.routes'; import { BASE_PATH } from './core/modules/openapi'; - export const appConfig: ApplicationConfig = { providers: [ provideExperimentalZonelessChangeDetection(), provideRouter(routes), provideAngularQuery(new QueryClient()), - { provide: BASE_PATH, useValue: "http://localhost:8080" }, + { provide: BASE_PATH, useValue: 'http://localhost:8080' }, provideHttpClient(withInterceptorsFromDi()), importProvidersFrom(LucideAngularModule.pick({ Home })) ] -}; \ No newline at end of file +}; diff --git a/webapp/src/app/example/counter/counter.component.html b/webapp/src/app/example/counter/counter.component.html index 87e73a15..9c3d46a6 100644 --- a/webapp/src/app/example/counter/counter.component.html +++ b/webapp/src/app/example/counter/counter.component.html @@ -1,17 +1,21 @@
-

Counter Title: {{title()}}

-

Value: {{counter()}}

-

HexValue: {{hexCounter()}}

- - Increment - @if (byCount() > 1) { - by {{byCount()}} - } - +

Counter Title: {{ title() }}

+

Value: {{ counter() }}

+

+

HexValue: {{ hexCounter() }}

+

+ + Increment + @if (byCount() > 1) { + by {{ byCount() }} + } + +

+

History

@for (item of counterHistory(); track item.dec) { {{ item.dec }} }
-
\ No newline at end of file + diff --git a/webapp/src/app/example/counter/counter.component.ts b/webapp/src/app/example/counter/counter.component.ts index fb2eb490..0b3fb872 100644 --- a/webapp/src/app/example/counter/counter.component.ts +++ b/webapp/src/app/example/counter/counter.component.ts @@ -8,10 +8,10 @@ interface CounterHistoryEntry { } @Component({ - selector: 'counter', + selector: 'app-counter', standalone: true, imports: [AppButtonComponent], - templateUrl: './counter.component.html', + templateUrl: './counter.component.html' }) export class CounterComponent { // we put all our application data inside signals! -> most optimal change detection and re-rendering possible @@ -26,9 +26,9 @@ export class CounterComponent { // Pitfall: conditional logic inside computed // When calling computed initially the dependency signal has to be called, otherwise it will not work return this.counter().toString(16); - }) + }); - counterHistory = signal([{ dec: 0, hex: "0" }]) + counterHistory = signal([{ dec: 0, hex: '0' }]); constructor() { console.log(`counter value: ${this.counter()}`); @@ -36,7 +36,8 @@ export class CounterComponent { // `effect`s goes in constructor (?) // `effectRef` is not necessary, only if needed // Runs once when the effect is declared to collect dependencies, and again when they change - const effectRef = effect((onCleanup) => { + // const effectRef = + effect((onCleanup) => { const currentCount = this.counter(); const currentHexCount = this.hexCounter(); @@ -46,8 +47,8 @@ export class CounterComponent { onCleanup(() => { console.log('Perform cleanup action here'); - }) - }) + }); + }); // effectRef.destroy() at any time! Usually not necessary @@ -57,9 +58,9 @@ export class CounterComponent { increment() { console.log('Updating counter...'); - this.counter.update(counter => counter + this.byCount()); + this.counter.update((counter) => counter + this.byCount()); // Update values of a signal only through Signals API, e.g.`set()` and `update()`, not directly (i.e. `push()`) - this.counterHistory.update(history => [...history, { dec: this.counter(), hex: this.hexCounter() }]) + this.counterHistory.update((history) => [...history, { dec: this.counter(), hex: this.hexCounter() }]); } } diff --git a/webapp/src/app/example/counter/counter.stories.ts b/webapp/src/app/example/counter/counter.stories.ts index 8f8504b5..a751c49e 100644 --- a/webapp/src/app/example/counter/counter.stories.ts +++ b/webapp/src/app/example/counter/counter.stories.ts @@ -8,9 +8,9 @@ const meta: Meta = { component: CounterComponent, tags: ['autodocs'], argTypes: toArgs({ - title: "test", - byCount: 2, - }), + title: 'test', + byCount: 2 + }) }; export default meta; @@ -21,5 +21,5 @@ export const Primary: Story = { args: { title: 'Counter', byCount: 2 - }, -}; \ No newline at end of file + } +}; diff --git a/webapp/src/app/example/counter/counter.ts b/webapp/src/app/example/counter/counter.ts index a16ed0a6..93cf9c2b 100644 --- a/webapp/src/app/example/counter/counter.ts +++ b/webapp/src/app/example/counter/counter.ts @@ -1,4 +1,4 @@ -import { signal } from "@angular/core"; +import { signal } from '@angular/core'; export const counter = signal(0); @@ -18,9 +18,9 @@ export const counter = signal(0); // // inject any dependencies you need here // } // -// // anyone needing to modify the signal +// // anyone needing to modify the signal // // needs to do so in a controlled way // incrementCounter() { // this.counterSignal.update((val) => val + 1); // } -// } \ No newline at end of file +// } diff --git a/webapp/src/app/example/hello/hello.component.ts b/webapp/src/app/example/hello/hello.component.ts index 30c42c61..12a35298 100644 --- a/webapp/src/app/example/hello/hello.component.ts +++ b/webapp/src/app/example/hello/hello.component.ts @@ -1,36 +1,34 @@ import { Component, inject } from '@angular/core'; import { injectMutation, injectQuery, injectQueryClient } from '@tanstack/angular-query-experimental'; -import { AngularQueryDevtools } from '@tanstack/angular-query-devtools-experimental' +import { AngularQueryDevtools } from '@tanstack/angular-query-devtools-experimental'; import { lastValueFrom } from 'rxjs'; import { HelloService } from 'app/core/modules/openapi'; import { AppButtonComponent } from 'app/ui/button/button/button.component'; @Component({ - selector: 'hello', + selector: 'app-hello', standalone: true, imports: [AppButtonComponent, AngularQueryDevtools], templateUrl: './hello.component.html' }) export class HelloComponent { - helloService = inject(HelloService) - queryClient = injectQueryClient() + helloService = inject(HelloService); + queryClient = injectQueryClient(); query = injectQuery(() => ({ queryKey: ['hellos'], - queryFn: async () => lastValueFrom(this.helloService.getAllHellos()), - }) - ); + queryFn: async () => lastValueFrom(this.helloService.getAllHellos()) + })); mutation = injectMutation(() => ({ mutationFn: () => lastValueFrom(this.helloService.addHello()), - onSuccess: () => + onSuccess: () => this.queryClient.invalidateQueries({ - queryKey: ['hellos'], - }), - }) - ); + queryKey: ['hellos'] + }) + })); addHello() { this.mutation.mutate(); } -} \ No newline at end of file +} diff --git a/webapp/src/app/storybook.helper.ts b/webapp/src/app/storybook.helper.ts index a269d051..49322ae7 100644 --- a/webapp/src/app/storybook.helper.ts +++ b/webapp/src/app/storybook.helper.ts @@ -1,13 +1,10 @@ import { InputSignalWithTransform, InputSignal, EventEmitter } from '@angular/core'; -import { Args, ArgTypes } from '@storybook/angular'; import { cva as CVA } from 'class-variance-authority'; import { ClassProp, ClassValue, StringToBoolean } from 'class-variance-authority/types'; // Source: // https://stackoverflow.com/questions/78379300/how-do-i-use-angular-input-signals-with-storybook -export function toArgs( - args: Partial>> -): TransformEventType { +export function toArgs(args: Partial>>): TransformEventType { return args as unknown as TransformEventType; } @@ -22,28 +19,23 @@ type TransformSignalInputType = { }; // Type to extract the type from InputSignal or InputSignalWithTransform -type TransformInputType = - T extends InputSignalWithTransform - ? U - : T extends InputSignal - ? U - : T; - - +type TransformInputType = T extends InputSignalWithTransform ? U : T extends InputSignal ? U : T; // CVA Storybook Helper type ConfigSchema = Record>; type ConfigVariants = { - [Variant in keyof T]?: StringToBoolean | null | undefined; + [Variant in keyof T]?: StringToBoolean | null | undefined; }; type ConfigVariantsMulti = { - [Variant in keyof T]?: StringToBoolean | StringToBoolean[] | undefined; + [Variant in keyof T]?: StringToBoolean | StringToBoolean[] | undefined; }; -type Config = T extends ConfigSchema ? { - variants?: T; - defaultVariants?: ConfigVariants; - compoundVariants?: (T extends ConfigSchema ? (ConfigVariants | ConfigVariantsMulti) & ClassProp : ClassProp)[]; -} : never; +type Config = T extends ConfigSchema + ? { + variants?: T; + defaultVariants?: ConfigVariants; + compoundVariants?: (T extends ConfigSchema ? (ConfigVariants | ConfigVariantsMulti) & ClassProp : ClassProp)[]; + } + : never; function createCVAArgTypes(config?: Config) { if (!config?.variants) { @@ -53,12 +45,14 @@ function createCVAArgTypes(config?: Config) { const variants = config?.variants; return Object.fromEntries( Object.entries(variants).map(([variant, options]) => { - const variantKey = variant as keyof T; - const optionsArray = Object.keys(options) as (keyof T[typeof variantKey])[]; - return [variant, { - control: { type: 'select' }, - options: optionsArray - }]; + const optionsArray = Object.keys(options) as (keyof T[typeof variant])[]; + return [ + variant, + { + control: { type: 'select' }, + options: optionsArray + } + ]; }) ); } @@ -66,7 +60,7 @@ function createCVAArgTypes(config?: Config) { function createCVADefaultArgs(config?: Config) { return (config?.defaultVariants || {}) as { [Variant in keyof T]: keyof T[Variant]; - } + }; } export function cva(base?: ClassValue, config?: Config) { diff --git a/webapp/src/app/ui/button/button/button.component.html b/webapp/src/app/ui/button/button/button.component.html index 8677d1db..57c21bac 100644 --- a/webapp/src/app/ui/button/button/button.component.html +++ b/webapp/src/app/ui/button/button/button.component.html @@ -1,3 +1,3 @@ \ No newline at end of file + diff --git a/webapp/src/app/ui/button/button/button.component.ts b/webapp/src/app/ui/button/button/button.component.ts index 8d3a7235..fa921b14 100644 --- a/webapp/src/app/ui/button/button/button.component.ts +++ b/webapp/src/app/ui/button/button/button.component.ts @@ -14,19 +14,19 @@ const [buttonVariants, args, argTypes] = cva( outline: 'border border-input bg-background hover:bg-accent hover:text-accent-foreground', secondary: 'bg-secondary text-secondary-foreground hover:bg-secondary/80', ghost: 'hover:bg-accent hover:text-accent-foreground', - link: 'text-primary underline-offset-4 hover:underline', + link: 'text-primary underline-offset-4 hover:underline' }, size: { default: 'h-10 px-4 py-2', sm: 'h-9 rounded-md px-3', lg: 'h-11 rounded-md px-8', - icon: 'h-10 w-10', - }, + icon: 'h-10 w-10' + } }, defaultVariants: { variant: 'default', - size: 'default', - }, + size: 'default' + } } ); @@ -37,10 +37,10 @@ interface ButtonVariants extends VariantProps {} @Component({ selector: 'app-button', standalone: true, - templateUrl: './button.component.html', + templateUrl: './button.component.html' }) export class AppButtonComponent { - class = input(''); + class = input(''); variant = input('default'); size = input('default'); disabled = input(false); @@ -48,4 +48,4 @@ export class AppButtonComponent { onClick = output(); computedClass = computed(() => cn(buttonVariants({ variant: this.variant(), size: this.size() }), this.class())); -} \ No newline at end of file +} diff --git a/webapp/src/app/ui/button/button/button.stories.ts b/webapp/src/app/ui/button/button/button.stories.ts index 2c7c7158..a3b254cb 100644 --- a/webapp/src/app/ui/button/button/button.stories.ts +++ b/webapp/src/app/ui/button/button/button.stories.ts @@ -11,17 +11,17 @@ const meta: Meta = { args: { ...args, disabled: false, - onClick: fn(), + onClick: fn() }, argTypes: { ...argTypes, disabled: { - control: 'boolean', + control: 'boolean' }, onClick: { - action: 'onClick', - }, - }, + action: 'onClick' + } + } }; export default meta; @@ -36,17 +36,17 @@ export const Primary: Story = { }, render: (args) => ({ props: args, - template: `Primary`, + template: `Primary` }) }; export const Secondary: Story = { args: { - variant: "secondary", - size: "default" + variant: 'secondary', + size: 'default' }, - render: args => ({ + render: (args) => ({ props: args, template: `Secondary` }) @@ -54,11 +54,11 @@ export const Secondary: Story = { export const Destructive: Story = { args: { - variant: "destructive", - size: "default" + variant: 'destructive', + size: 'default' }, - render: args => ({ + render: (args) => ({ props: args, template: `Destructive` }) @@ -66,11 +66,11 @@ export const Destructive: Story = { export const Outline: Story = { args: { - variant: "outline", - size: "default" + variant: 'outline', + size: 'default' }, - render: args => ({ + render: (args) => ({ props: args, template: `Outline` }) @@ -78,11 +78,11 @@ export const Outline: Story = { export const Ghost: Story = { args: { - variant: "ghost", - size: "default" + variant: 'ghost', + size: 'default' }, - render: args => ({ + render: (args) => ({ props: args, template: `Ghost` }) @@ -90,11 +90,11 @@ export const Ghost: Story = { export const Link: Story = { args: { - variant: "link", - size: "default" + variant: 'link', + size: 'default' }, - render: args => ({ + render: (args) => ({ props: args, template: `Link` }) @@ -103,45 +103,45 @@ export const Link: Story = { export const Icon: Story = { decorators: [ moduleMetadata({ - imports: [LucideAngularModule.pick({ ChevronRight })], - }), + imports: [LucideAngularModule.pick({ ChevronRight })] + }) ], render: (args) => ({ props: { variant: 'outline', - size: 'icon', + size: 'icon' }, - template: ``, + template: `` }) }; export const WithIcon: Story = { decorators: [ moduleMetadata({ - imports: [LucideAngularModule.pick({ Mail })], - }), + imports: [LucideAngularModule.pick({ Mail })] + }) ], render: (args) => ({ props: { variant: 'default', - size: 'default', + size: 'default' }, - template: `Login with Email`, + template: `Login with Email` }) }; export const Loading: Story = { decorators: [ moduleMetadata({ - imports: [LucideAngularModule.pick({ Loader2 })], - }), + imports: [LucideAngularModule.pick({ Loader2 })] + }) ], render: (args) => ({ props: { variant: 'default', size: 'default', - disabled: true, + disabled: true }, - template: `Please wait`, + template: `Please wait` }) }; diff --git a/webapp/src/app/ui/input/input.component.html b/webapp/src/app/ui/input/input.component.html index 8694176e..6b64b478 100644 --- a/webapp/src/app/ui/input/input.component.html +++ b/webapp/src/app/ui/input/input.component.html @@ -1,13 +1 @@ - - - - - + diff --git a/webapp/src/app/ui/input/input.component.ts b/webapp/src/app/ui/input/input.component.ts index 57683265..bd46e268 100644 --- a/webapp/src/app/ui/input/input.component.ts +++ b/webapp/src/app/ui/input/input.component.ts @@ -11,13 +11,13 @@ const [inputVariants, args, argTypes] = cva( size: { default: 'h-10 px-4 py-2', sm: 'h-9 px-2 py-1', - lg: 'h-11 px-4 py-3', - }, + lg: 'h-11 px-4 py-3' + } }, defaultVariants: { - size: 'default', - }, - }, + size: 'default' + } + } ); export { args, argTypes }; @@ -27,7 +27,7 @@ interface InputVariants extends VariantProps {} @Component({ selector: 'app-input', standalone: true, - templateUrl: './input.component.html', + templateUrl: './input.component.html' }) export class AppInputComponent { class = input(''); @@ -45,7 +45,5 @@ export class AppInputComponent { this.valueChange.emit(inputValue); } - computedClass = computed(() => - cn(inputVariants({ size: this.size() }), this.class()), - ); + computedClass = computed(() => cn(inputVariants({ size: this.size() }), this.class())); } diff --git a/webapp/src/app/ui/input/input.stories.ts b/webapp/src/app/ui/input/input.stories.ts index 8beb17ce..3ce17104 100644 --- a/webapp/src/app/ui/input/input.stories.ts +++ b/webapp/src/app/ui/input/input.stories.ts @@ -1,9 +1,4 @@ -import { - argsToTemplate, - moduleMetadata, - type Meta, - type StoryObj, -} from '@storybook/angular'; +import { argsToTemplate, moduleMetadata, type Meta, type StoryObj } from '@storybook/angular'; import { AppInputComponent, args, argTypes } from './input.component'; import { action } from '@storybook/addon-actions'; import { AppButtonComponent } from '@app/ui/button/button/button.component'; @@ -17,22 +12,22 @@ const meta: Meta = { ...args, value: '', disabled: false, - size: 'default', + size: 'default' }, argTypes: { ...argTypes, disabled: { - control: 'boolean', + control: 'boolean' }, onInput: { - action: 'onInput', - }, + action: 'onInput' + } }, decorators: [ moduleMetadata({ - imports: [AppButtonComponent, AppLabelComponent], - }), - ], + imports: [AppButtonComponent, AppLabelComponent] + }) + ] }; export default meta; @@ -41,18 +36,18 @@ type Story = StoryObj; export const Default: Story = { render: (args) => ({ props: args, - template: ``, - }), + template: `` + }) }; export const Disabled: Story = { args: { - disabled: true, + disabled: true }, render: (args) => ({ props: args, - template: ``, - }), + template: `` + }) }; export const WithLabel: Story = { @@ -63,8 +58,8 @@ export const WithLabel: Story = { Label - `, - }), + ` + }) }; export const WithButton: Story = { @@ -74,13 +69,13 @@ export const WithButton: Story = { userInput: '', onButtonClick(value: string) { action('Button Clicked')(`Input Value: ${value}`); - }, + } }, template: `
Submit
- `, - }), + ` + }) }; diff --git a/webapp/src/app/ui/label/label.component.html b/webapp/src/app/ui/label/label.component.html index 3eed1b2c..72d84052 100644 --- a/webapp/src/app/ui/label/label.component.html +++ b/webapp/src/app/ui/label/label.component.html @@ -1,6 +1,3 @@ -