forked from tomalaforge/angular-challenges
-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
feat: challenge 40 - forms control value accessor
- Loading branch information
Stanislav
committed
Dec 7, 2023
1 parent
b164792
commit 29abc23
Showing
29 changed files
with
509 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
{ | ||
"extends": ["../../../.eslintrc.json"], | ||
"ignorePatterns": ["!**/*"], | ||
"overrides": [ | ||
{ | ||
"files": ["*.ts"], | ||
"extends": [ | ||
"plugin:@nx/angular", | ||
"plugin:@angular-eslint/template/process-inline-templates" | ||
], | ||
"rules": { | ||
"@angular-eslint/directive-selector": [ | ||
"error", | ||
{ | ||
"type": "attribute", | ||
"prefix": "app", | ||
"style": "camelCase" | ||
} | ||
], | ||
"@angular-eslint/component-selector": [ | ||
"error", | ||
{ | ||
"type": "element", | ||
"prefix": "app", | ||
"style": "kebab-case" | ||
} | ||
] | ||
} | ||
}, | ||
{ | ||
"files": ["*.html"], | ||
"extends": ["plugin:@nx/angular-template"], | ||
"rules": {} | ||
} | ||
] | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,13 @@ | ||
# Control Value Accessor | ||
|
||
> author: thomas-laforge | ||
### Run Application | ||
|
||
```bash | ||
npx nx serve performance-control-value-accessor | ||
``` | ||
|
||
### Documentation and Instruction | ||
|
||
Challenge documentation is [here](https://angular-challenges.vercel.app/challenges/forms/40-control-value-accessor/). |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
/* eslint-disable */ | ||
export default { | ||
displayName: 'forms-control-value-accessor', | ||
preset: '../../../jest.preset.js', | ||
setupFilesAfterEnv: ['<rootDir>/src/test-setup.ts'], | ||
coverageDirectory: '../../../coverage/apps/forms/control-value-accessor', | ||
transform: { | ||
'^.+\\.(ts|mjs|js|html)$': [ | ||
'jest-preset-angular', | ||
{ | ||
tsconfig: '<rootDir>/tsconfig.spec.json', | ||
stringifyContentPathRegex: '\\.(html|svg)$', | ||
}, | ||
], | ||
}, | ||
transformIgnorePatterns: ['node_modules/(?!.*\\.mjs$)'], | ||
snapshotSerializers: [ | ||
'jest-preset-angular/build/serializers/no-ng-attributes', | ||
'jest-preset-angular/build/serializers/ng-snapshot', | ||
'jest-preset-angular/build/serializers/html-comment', | ||
], | ||
}; |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,86 @@ | ||
{ | ||
"name": "forms-control-value-accessor", | ||
"$schema": "../../../node_modules/nx/schemas/project-schema.json", | ||
"projectType": "application", | ||
"prefix": "app", | ||
"sourceRoot": "apps/forms/control-value-accessor/src", | ||
"tags": [], | ||
"targets": { | ||
"build": { | ||
"executor": "@angular-devkit/build-angular:application", | ||
"outputs": ["{options.outputPath}"], | ||
"options": { | ||
"outputPath": "dist/apps/forms/control-value-accessor", | ||
"index": "apps/forms/control-value-accessor/src/index.html", | ||
"browser": "apps/forms/control-value-accessor/src/main.ts", | ||
"polyfills": ["zone.js"], | ||
"tsConfig": "apps/forms/control-value-accessor/tsconfig.app.json", | ||
"inlineStyleLanguage": "scss", | ||
"assets": [ | ||
"apps/forms/control-value-accessor/src/favicon.ico", | ||
"apps/forms/control-value-accessor/src/assets" | ||
], | ||
"styles": ["apps/forms/control-value-accessor/src/styles.scss"], | ||
"scripts": [] | ||
}, | ||
"configurations": { | ||
"production": { | ||
"budgets": [ | ||
{ | ||
"type": "initial", | ||
"maximumWarning": "500kb", | ||
"maximumError": "1mb" | ||
}, | ||
{ | ||
"type": "anyComponentStyle", | ||
"maximumWarning": "2kb", | ||
"maximumError": "4kb" | ||
} | ||
], | ||
"outputHashing": "all" | ||
}, | ||
"development": { | ||
"optimization": false, | ||
"extractLicenses": false, | ||
"sourceMap": true | ||
} | ||
}, | ||
"defaultConfiguration": "production" | ||
}, | ||
"serve": { | ||
"executor": "@angular-devkit/build-angular:dev-server", | ||
"configurations": { | ||
"production": { | ||
"buildTarget": "forms-control-value-accessor:build:production" | ||
}, | ||
"development": { | ||
"buildTarget": "forms-control-value-accessor:build:development" | ||
} | ||
}, | ||
"defaultConfiguration": "development" | ||
}, | ||
"extract-i18n": { | ||
"executor": "@angular-devkit/build-angular:extract-i18n", | ||
"options": { | ||
"buildTarget": "forms-control-value-accessor:build" | ||
} | ||
}, | ||
"lint": { | ||
"executor": "@nx/eslint:lint", | ||
"outputs": ["{options.outputFile}"], | ||
"options": { | ||
"lintFilePatterns": [ | ||
"apps/forms/control-value-accessor/**/*.ts", | ||
"apps/forms/control-value-accessor/**/*.html" | ||
] | ||
} | ||
}, | ||
"test": { | ||
"executor": "@nx/jest:jest", | ||
"outputs": ["{workspaceRoot}/coverage/{projectRoot}"], | ||
"options": { | ||
"jestConfig": "apps/forms/control-value-accessor/jest.config.ts" | ||
} | ||
} | ||
} | ||
} |
8 changes: 8 additions & 0 deletions
8
apps/forms/control-value-accessor/src/app/app.component.spec.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,8 @@ | ||
import { render } from '@testing-library/angular'; | ||
import { AppComponent } from './app.component'; | ||
|
||
describe('AppComponent', () => { | ||
test('...', async () => { | ||
await render(AppComponent); | ||
}); | ||
}); |
15 changes: 15 additions & 0 deletions
15
apps/forms/control-value-accessor/src/app/app.component.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,15 @@ | ||
import { Component } from '@angular/core'; | ||
import { FeedbackFormComponent } from './feedback-form/feedback-form.component'; | ||
|
||
@Component({ | ||
standalone: true, | ||
imports: [FeedbackFormComponent], | ||
selector: 'app-root', | ||
template: `<app-feedback-form | ||
(feedBackSubmit)="apiCall($event)"></app-feedback-form>`, | ||
}) | ||
export class AppComponent { | ||
apiCall(event: Record<string, string | null>): void { | ||
console.log(event); | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,5 @@ | ||
import { ApplicationConfig } from '@angular/core'; | ||
|
||
export const appConfig: ApplicationConfig = { | ||
providers: [], | ||
}; |
16 changes: 16 additions & 0 deletions
16
apps/forms/control-value-accessor/src/app/feedback-control/feedback-control.component.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,16 @@ | ||
<svg width="0" height="0" xmlns="http://www.w3.org/2000/svg"> | ||
<symbol id="star" width="50" height="50" xmlns="http://www.w3.org/2000/svg"> | ||
<path | ||
d="M31.77 11.857H19.74L15.99.5l-3.782 11.357H0l9.885 6.903-3.692 11.21 9.736-7.05 9.796 6.962-3.722-11.18 9.766-6.845z" /> | ||
</symbol> | ||
</svg> | ||
<div class="rating"> | ||
@for (item of [].constructor(5); track item) { | ||
<svg | ||
class="star" | ||
[class.star-active]="isStarActive($index, value)" | ||
(click)="setRating($index)"> | ||
<use xlink:href="#star"></use> | ||
</svg> | ||
} | ||
</div> |
26 changes: 26 additions & 0 deletions
26
apps/forms/control-value-accessor/src/app/feedback-control/feedback-control.component.scss
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,26 @@ | ||
.rating { | ||
display: flex; | ||
justify-content: center; | ||
padding: 0 10px; | ||
|
||
&:hover { | ||
.star { | ||
fill: #ffd055; | ||
} | ||
} | ||
} | ||
|
||
.star { | ||
width: 50px; | ||
height: 50px; | ||
fill: #cccccc; | ||
cursor: pointer; | ||
|
||
&:hover ~ .star { | ||
fill: #d8d8d8; | ||
} | ||
} | ||
|
||
.star-active { | ||
fill: #ffd055 !important; | ||
} |
25 changes: 25 additions & 0 deletions
25
apps/forms/control-value-accessor/src/app/feedback-control/feedback-control.component.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,25 @@ | ||
import { EventEmitter } from '@angular/core'; | ||
import { Output } from '@angular/core'; | ||
import { Component } from '@angular/core'; | ||
|
||
@Component({ | ||
standalone: true, | ||
selector: 'app-feedback-control', | ||
templateUrl: 'feedback-control.component.html', | ||
styleUrls: ['feedback-control.component.scss'], | ||
}) | ||
export class FeedbackControlComponent { | ||
@Output() | ||
readonly ratingUpdated: EventEmitter<string> = new EventEmitter<string>(); | ||
|
||
value: number | null = null; | ||
|
||
setRating(index: number): void { | ||
this.value = index + 1; | ||
this.ratingUpdated.emit(`${this.value}`); | ||
} | ||
|
||
isStarActive(index: number, value: number | null): boolean { | ||
return value ? index < value : false; | ||
} | ||
} |
28 changes: 28 additions & 0 deletions
28
apps/forms/control-value-accessor/src/app/feedback-form/feedback-form.component.html
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,28 @@ | ||
<form [formGroup]="feedbackForm" class="feedback-form"> | ||
<legend class="feedback-form-title">Tell us what you think</legend> | ||
<input | ||
class="feedback-form-control" | ||
[formControl]="feedbackForm.controls.name" | ||
placeholder="Name" | ||
type="text" /> | ||
<input | ||
class="feedback-form-control" | ||
[formControl]="feedbackForm.controls.email" | ||
placeholder="Email" | ||
type="text" /> | ||
<app-feedback-control (ratingUpdated)="rating = $event"> | ||
</app-feedback-control> | ||
<textarea | ||
class="feedback-form-control" | ||
[formControl]="feedbackForm.controls.comment" | ||
placeholder="Сomment text"></textarea> | ||
<button | ||
class="feedback-form-submit" | ||
type="submit" | ||
[disabled]=" | ||
feedbackForm.touched && (rating === null || feedbackForm.invalid) | ||
" | ||
(click)="submitForm()"> | ||
Submit | ||
</button> | ||
</form> |
50 changes: 50 additions & 0 deletions
50
apps/forms/control-value-accessor/src/app/feedback-form/feedback-form.component.scss
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,50 @@ | ||
* { | ||
box-sizing: border-box; | ||
} | ||
|
||
:host { | ||
display: block; | ||
padding: 20px; | ||
} | ||
|
||
.feedback-form { | ||
display: flex; | ||
flex-direction: column; | ||
width: 500px; | ||
padding: 20px; | ||
border: 1px solid #000000; | ||
} | ||
|
||
.feedback-form-title { | ||
margin-bottom: 20px; | ||
font-size: 24px; | ||
} | ||
|
||
.feedback-form-control { | ||
max-height: 200px; | ||
margin-bottom: 20px; | ||
padding: 12px 12px 12px 20px; | ||
border-radius: 0; | ||
background-color: #fbfbfb; | ||
color: #3c3c3c; | ||
font-size: 18px; | ||
|
||
&:focus { | ||
padding: 10px 10px 10px 18px; | ||
border: 2px solid #054ada; | ||
outline: none; | ||
background: #fff; | ||
} | ||
} | ||
|
||
.feedback-form-submit { | ||
padding: 10px; | ||
background-color: #054ada; | ||
color: #ffffff; | ||
font-size: 18px; | ||
|
||
&[disabled] { | ||
background-color: #cccccc; | ||
color: #ffffff; | ||
} | ||
} |
42 changes: 42 additions & 0 deletions
42
apps/forms/control-value-accessor/src/app/feedback-form/feedback-form.component.ts
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,42 @@ | ||
import { EventEmitter } from '@angular/core'; | ||
import { Output } from '@angular/core'; | ||
import { Component } from '@angular/core'; | ||
import { Validators } from '@angular/forms'; | ||
import { FormControl } from '@angular/forms'; | ||
import { FormGroup } from '@angular/forms'; | ||
import { ReactiveFormsModule } from '@angular/forms'; | ||
import { FeedbackControlComponent } from '../feedback-control/feedback-control.component'; | ||
|
||
@Component({ | ||
standalone: true, | ||
imports: [FeedbackControlComponent, ReactiveFormsModule], | ||
selector: 'app-feedback-form', | ||
templateUrl: 'feedback-form.component.html', | ||
styleUrls: ['feedback-form.component.scss'], | ||
}) | ||
export class FeedbackFormComponent { | ||
@Output() | ||
readonly feedBackSubmit: EventEmitter<Record<string, string | null>> = | ||
new EventEmitter<Record<string, string | null>>(); | ||
|
||
readonly feedbackForm = new FormGroup({ | ||
name: new FormControl('', { | ||
validators: Validators.required, | ||
}), | ||
email: new FormControl('', { | ||
validators: Validators.required, | ||
}), | ||
comment: new FormControl(), | ||
}); | ||
|
||
rating: string | null = null; | ||
|
||
submitForm(): void { | ||
this.feedBackSubmit.emit({ | ||
...this.feedbackForm.value, | ||
rating: this.rating, | ||
}); | ||
|
||
this.feedbackForm.reset(); | ||
} | ||
} |
Empty file.
Binary file not shown.
Oops, something went wrong.