跳转到主要内容

标签(标签)

资源精选(342) Go开发(108) Go语言(103) Go(99) angular(83) LLM(79) 大语言模型(63) 人工智能(53) 前端开发(50) LangChain(43) golang(43) 机器学习(39) Go工程师(38) Go程序员(38) Go开发者(36) React(34) Go基础(29) Python(24) Vue(23) Web开发(20) Web技术(19) 精选资源(19) 深度学习(19) Java(18) ChatGTP(17) Cookie(16) android(16) 前端框架(13) JavaScript(13) Next.js(12) 安卓(11) 聊天机器人(10) typescript(10) 资料精选(10) NLP(10) 第三方Cookie(9) Redwoodjs(9) ChatGPT(9) LLMOps(9) Go语言中级开发(9) 自然语言处理(9) PostgreSQL(9) 区块链(9) mlops(9) 安全(9) 全栈开发(8) OpenAI(8) Linux(8) AI(8) GraphQL(8) iOS(8) 软件架构(7) RAG(7) Go语言高级开发(7) AWS(7) C++(7) 数据科学(7) 智能体(6) whisper(6) Prisma(6) 隐私保护(6) JSON(6) DevOps(6) 数据可视化(6) wasm(6) 计算机视觉(6) 算法(6) Rust(6) 微服务(6) 隐私沙盒(5) FedCM(5) 语音识别(5) Angular开发(5) 快速应用开发(5) 提示工程(5) Agent(5) LLaMA(5) 低代码开发(5) Go测试(5) gorm(5) REST API(5) kafka(5) 推荐系统(5) WebAssembly(5) GameDev(5) CMS(5) CSS(5) machine-learning(5) 机器人(5) 游戏开发(5) Blockchain(5) Web安全(5) nextjs(5) Kotlin(5) 低代码平台(5) 机器学习资源(5) Go资源(5) Nodejs(5) PHP(5) Swift(5) RAG架构(4) devin(4) Blitz(4) javascript框架(4) Redwood(4) GDPR(4) 生成式人工智能(4) Angular16(4) Alpaca(4) 编程语言(4) SAML(4) JWT(4) JSON处理(4) Go并发(4) 移动开发(4) 移动应用(4) security(4) 隐私(4) spring-boot(4) 物联网(4) 网络安全(4) API(4) Ruby(4) 信息安全(4) flutter(4) 专家智能体(3) Chrome(3) CHIPS(3) 3PC(3) SSE(3) 人工智能软件工程师(3) LLM Agent(3) Remix(3) Ubuntu(3) GPT4All(3) 软件开发(3) 问答系统(3) 开发工具(3) 最佳实践(3) RxJS(3) SSR(3) Node.js(3) Dolly(3) 移动应用开发(3) 低代码(3) IAM(3) Web框架(3) CORS(3) 基准测试(3) Go语言数据库开发(3) Oauth2(3) 并发(3) 主题(3) Theme(3) earth(3) nginx(3) 软件工程(3) azure(3) keycloak(3) 生产力工具(3) gpt3(3) 工作流(3) C(3) jupyter(3) 认证(3) prometheus(3) GAN(3) Spring(3) 逆向工程(3) 应用安全(3) Docker(3) Django(3) R(3) .NET(3) 大数据(3) Hacking(3) 渗透测试(3) C++资源(3) Mac(3) 微信小程序(3) Python资源(3) JHipster(3) 语言模型(2) 可穿戴设备(2) JDK(2) SQL(2) Apache(2) Hashicorp Vault(2) Spring Cloud Vault(2) Go语言Web开发(2) Go测试工程师(2) WebSocket(2) 容器化(2) AES(2) 加密(2) 输入验证(2) ORM(2) Fiber(2) Postgres(2) Gorilla Mux(2) Go数据库开发(2) 模块(2) 泛型(2) 指针(2) HTTP(2) PostgreSQL开发(2) Vault(2) K8s(2) Spring boot(2) R语言(2) 深度学习资源(2) 半监督学习(2) semi-supervised-learning(2) architecture(2) 普罗米修斯(2) 嵌入模型(2) productivity(2) 编码(2) Qt(2) 前端(2) Rust语言(2) NeRF(2) 神经辐射场(2) 元宇宙(2) CPP(2) 数据分析(2) spark(2) 流处理(2) Ionic(2) 人体姿势估计(2) human-pose-estimation(2) 视频处理(2) deep-learning(2) kotlin语言(2) kotlin开发(2) burp(2) Chatbot(2) npm(2) quantum(2) OCR(2) 游戏(2) game(2) 内容管理系统(2) MySQL(2) python-books(2) pentest(2) opengl(2) IDE(2) 漏洞赏金(2) Web(2) 知识图谱(2) PyTorch(2) 数据库(2) reverse-engineering(2) 数据工程(2) swift开发(2) rest(2) robotics(2) ios-animation(2) 知识蒸馏(2) 安卓开发(2) nestjs(2) solidity(2) 爬虫(2) 面试(2) 容器(2) C++精选(2) 人工智能资源(2) Machine Learning(2) 备忘单(2) 编程书籍(2) angular资源(2) 速查表(2) cheatsheets(2) SecOps(2) mlops资源(2) R资源(2) DDD(2) 架构设计模式(2) 量化(2) Hacking资源(2) 强化学习(2) flask(2) 设计(2) 性能(2) Sysadmin(2) 系统管理员(2) Java资源(2) 机器学习精选(2) android资源(2) android-UI(2) Mac资源(2) iOS资源(2) Vue资源(2) flutter资源(2) JavaScript精选(2) JavaScript资源(2) Rust开发(2) deeplearning(2) RAD(2)

As a frontend developer, coding forms is a habit. We put forms everywhere to add user data through an application, and some parts of them are often repeated or reused within same organisation’s apps and libs.

Goal

The goal of this article is to show a clean way to code a reusable form section, as it could be shared by the design system of your organisation for example.

How to split a form into reusable parts?

  1. Isolate the form section to be reusable
  2. Transform it into component
  3. Make this form section robust using Validation and Unit tests
  4. Make Angular consider this component as a standard form input using ControlValueAccessor interface

In order to keep this article synthetic, we will only focus on the 4th part here!

Why this article ?

The Angular documentation about ControlValueAccessor is pretty weak and the most of dedicated articles only explain the basic usage of it : creating a custom input recognized by Angular forms.
Here we want to split a part of a form composed of multiple inputs to make it standalone and reusable as if it were a single input.

This approach is quite well explained at the end of this article but does not show a running full code example.

Prerequisites

  1. Angular basics
  2. Reactive forms
  3. Basic usage of ControlValueAccessor:

Angular Custom Form Controls - Complete Guide

The Angular Forms and ReactiveForms modules come with a series of built-in directives that make it very simple to bind…

blog.angular-university.io

Practice

See the result

If you already cannot wait for the running example, you can have a look here. Or study only the code.

Context

In the below example we will simulate a CV creation platform with:

  • a CV form to add a new CV
  • a list of already added CVs

A CV will be constituted of some applicant’s personal data and a list of work experiences. Each experience will have its own related data.

export interface Cv {
  firstName: string;
  lastName: string;
  birthDate: Date;
  experiences: Experience[];
}

export interface Experience {
  company: string;
  role: string;
  startDate: Date;
  endDate: Date | null;
}

Initial CV form

Here is the code of the above form:

<form [formGroup]="cvGroup" (submit)="addCv($event)">
  <fieldset>
    <legend>Personal data</legend>
    <p>
      <label for="firstName">First name</label>
      <input type="text" id="firstName" formControlName="firstName" />
    </p>
    <p>
      <label for="lastName">Last name</label>
      <input type="text" id="lastName" formControlName="lastName" />
    </p>
    <p>
      <label for="birthDate">Date of birth</label>
      <input type="date" id="birthDate" formControlName="birthDate" />
    </p>
  </fieldset>
  <ul>
    <!-- TODO: Add a dynamic list of work experience forms -->
  </ul>
  <div class="cv-form-actions">
    <button type="button" (click)="addExperience()">ADD EXPERIENCE</button>
    <button type="submit">ADD CV</button>
  </div>
</form>

We can observe a first fieldset to collect applicant personal data with standard inputs firstNamelastName and birthDate, and then we would like to implement a dynamic list of forms, where each form would collect the details of a job experienced by the applicant.

Experience form

To achieve this goal, we create a form dedicated to a work experience:

<fieldset [formGroup]="experienceGroup">
  <legend>Experience</legend>
  <p>
    <label for="company">Company</label>
    <input
      type="text"
      id="company"
      formControlName="company"
      (blur)="onTouched()"
    />
  </p>
  <p>
    <label for="role">Role</label>
    <input
      type="text"
      id="role"
      formControlName="role"
      (blur)="onTouched()"
    />
  </p>
  <p>
    <label for="startDate">Start date</label>
    <input
      type="date"
      id="startDate"
      formControlName="startDate"
      (blur)="onTouched()"
    />
  </p>
  <p>
    <label for="endDate">End date</label>
    <input
      type="date"
      id="endDate"
      formControlName="endDate"
      (blur)="onTouched()"
    />
  </p>
</fieldset>

The experience form is composed of afieldset with some related standard inputs as companyrolestartDate and endDate.

Now we want this component to generate an Experience instance as we declared it before.

1. First we need a FormGroup representing the work experience form:

experienceGroup = new FormGroup({
  company: new FormControl('', {
    nonNullable: true,
    validators: Validators.required,
  }),
  role: new FormControl('', {
    nonNullable: true,
    validators: Validators.required,
  }),
  startDate: new FormControl(new Date(), {
    nonNullable: true,
    validators: Validators.required,
  }),
  endDate: new FormControl(null),
});

2. Then we want to trigger a form value change when the experience form is valid:

private _detroyed = new Subject<void>();

ngOnInit(): void {
  this.experienceGroup.valueChanges
    .pipe(
      filter(
        (experience: Partial<Experience>) =>
          this.experienceGroup.valid &&
          !!experience.company?.trim() &&
          !!experience.role?.trim() &&
          !!experience.startDate
      ),
      takeUntil(this._detroyed)
    )
    .subscribe((experience: Experience) => this._onChange(experience));
}

ngOnDestroy(): void {
  this._detroyed.next();
  this._detroyed.complete();
}

private _onChange = (experience: Experience) => {};

Note that we trigger experience value change only when companyrole and startDate are correctly filled.

3. And finally we add the ControlValueAccessor implementation to make Angular being able to recognize our component as a FormControl:

onTouched = () => {};

registerOnChange(fn: (experience: Experience) => void): void {
  this._onChange = fn;
}

registerOnTouched(fn: () => void): void {
  this.onTouched = fn;
}

writeValue(experience: Experience | undefined | null): void {
  this.experienceGroup.setValue(experience ?? EXPERIENCE_DEFAULT, {
    emitEvent: false,
  });
}

setDisabledState(isDisabled: boolean): void {
  isDisabled ? this.experienceGroup.disable() : this.experienceGroup.enable();
}

private _onChange = (experience: Experience) => {};

All gathered results in:

export const EXPERIENCE_DEFAULT: Experience = {
  company: '',
  role: '',
  startDate: new Date(),
  endDate: null,
};

@Component({
  selector: 'experience-form',
  standalone: true,
  imports: [CommonModule, ReactiveFormsModule],
  templateUrl: './experience-form.component.html',
  styleUrls: ['./experience-form.component.scss'],
  providers: [
    {
      provide: NG_VALUE_ACCESSOR,
      multi: true,
      useExisting: forwardRef(() => ExperienceFormComponent),
    },
  ],
})
export class ExperienceFormComponent
  implements ControlValueAccessor, OnInit, OnDestroy, Validator
{
  experienceGroup = new FormGroup({
    company: new FormControl(EXPERIENCE_DEFAULT.company, {
      nonNullable: true,
      validators: Validators.required,
    }),
    role: new FormControl(EXPERIENCE_DEFAULT.role, {
      nonNullable: true,
      validators: Validators.required,
    }),
    startDate: new FormControl(EXPERIENCE_DEFAULT.startDate, {
      nonNullable: true,
      validators: Validators.required,
    }),
    endDate: new FormControl(EXPERIENCE_DEFAULT.endDate),
  });
  disabled = false;

  private _detroyed = new Subject<void>();

  ngOnInit(): void {
    this.experienceGroup.valueChanges
      .pipe(
        filter(
          (experience: Partial<Experience>) =>
            this.experienceGroup.valid &&
            !!experience.company?.trim() &&
            !!experience.role?.trim() &&
            !!experience.startDate
        ),
        takeUntil(this._detroyed)
      )
      .subscribe((experience: Experience) => this._onChange(experience));
  }

  ngOnDestroy(): void {
    this._detroyed.next();
    this._detroyed.complete();
  }

  onTouched = () => {};

  registerOnChange(fn: (experience: Experience) => void): void {
    this._onChange = fn;
  }

  registerOnTouched(fn: () => void): void {
    this.onTouched = fn;
  }

  writeValue(experience: Experience | undefined | null): void {
    this.experienceGroup.setValue(experience ?? EXPERIENCE_DEFAULT, {
      emitEvent: false,
    });
  }

  setDisabledState(isDisabled: boolean): void {
    isDisabled ? this.experienceGroup.disable() : this.experienceGroup.enable();
  }

  private _onChange = (experience: Experience) => {};
}

With this experience-form component, we now have a standalone form section that generates an Experience object on change and that can be understood as a simple FormControl by Angular machinery.

Let’s embed it in global CV form!

Add experience-form into cv-form

If we wanted to add only one work experience by CV, the integration would be pretty easy:

<form [formGroup]="cvGroup" (submit)="addCv($event)">
  <fieldset>
    <legend>Personal data</legend>
    <p>
      <label for="firstName">First name</label>
      <input type="text" id="firstName" formControlName="firstName" />
    </p>
    <p>
      <label for="lastName">Last name</label>
      <input type="text" id="lastName" formControlName="lastName" />
    </p>
    <p>
      <label for="birthDate">Date of birth</label>
      <input type="date" id="birthDate" formControlName="birthDate" />
    </p>
  </fieldset>
  <experience-form [formControl]="experience"></experience-form>
  <div class="cv-form-actions">
    <button type="button" (click)="addExperience()">ADD EXPERIENCE</button>
    <button type="submit">ADD CV</button>
  </div>
</form>
@Component({
  selector: 'cv-form',
  standalone: true,
  imports: [CommonModule, ReactiveFormsModule, ExperienceFormComponent],
  templateUrl: './cv-form.component.html',
  styleUrls: ['./cv-form.component.scss'],
  changeDetection: ChangeDetectionStrategy.OnPush,
})
export class CvFormComponent {
  @Output() cvAdded = new EventEmitter<Cv>();

  cvGroup = new FormGroup({
    firstName: new FormControl('', {
      nonNullable: true,
      validators: Validators.required,
    }),
    lastName: new FormControl('', {
      nonNullable: true,
      validators: Validators.required,
    }),
    birthDate: new FormControl(new Date('2000-01-01'), {
      nonNullable: true,
      validators: Validators.required,
    }),
    experience: new FormControl(EXPERIENCE_DEFAULT),
  });

  constructor() {}

  addCv(event: Event): void {
    event.preventDefault();
    this.cvAdded.emit(this.cvGroup.value as Cv);
    this.cvGroup.reset();
  }
}

But here we would like a dynamic list of experience-form added or removed by the applicant depending of his own work experiences. We will use here a very powerfull tool provided by Angular that is dedicated to this issue: FormArray.

If you are not familiar with FormArray, I suggest you this great article about its usage:

Angular FormArray - Complete Guide

In this post, you are going to learn everything that you need to know about the Angular FormArray construct, available…

blog.angular-university.io

With FormArray we are able to add a dynamic array of FormControl to our CV form and its value will be the array of every experience-form value. Let’s do it!

cvGroup = new FormGroup({
  ...,
  experiences: new FormArray([new FormControl(EXPERIENCE_DEFAULT)]),
});

addExperience(): void {
  this.cvGroup.controls.experiences.push(new FormControl(EXPERIENCE_DEFAULT));
}

removeExperience(index: number): void {
  this.cvGroup.controls.experiences.removeAt(index);
}
<ul>
  <li
    class="cv-form-exp"
    *ngFor="
      let expForm of cvGroup.controls.experiences.controls;
      let i = index
    "
  >
    <experience-form [formControl]="expForm"></experience-form>
    <button type="button" (click)="removeExperience(i)">X</button>
  </li>
</ul>

Summary

We have just seen the power of ControlValueAccessor to isolate parts of form and make them standalone components.

This practice enhances the single responsibility component principle and make the form sections reusable through an application or a multi applications project.

标签