Формы — неотъемлемая часть большинства веб-приложений: будь то регистрация, ввод данных или опросы. Модуль реактивных форм в Angular отлично подходит для создания статичных форм, но во многих случаях требуется, чтобы форма могла динамически адаптироваться в зависимости от действий пользователя или внешних данных.

В этой статье мы рассмотрим, как создавать динамические формы с использованием автономных компонентов в Angular 19, применяя модульный подход, который избавляет от необходимости использовать традиционные модули Angular. В сопроводительном репозитории на GitHub для оформления форм используется Tailwind CSS, однако в статье внимание сосредоточено исключительно на логике динамических форм. Tailwind и связанные с ним настройки намеренно не включены в примеры, чтобы сохранить акцент на основной теме.

Что такое динамические формы?

Динамические формы позволяют задавать структуру формы — поля, валидаторы, расположение и прочее — во время выполнения, а не на этапе компиляции. Это особенно полезно в следующих ситуациях: — Многошаговые формы с ветками ответов; — Формы, создаваемые на основе ответа от API; — Формы, которые пользователь может настраивать сам — добавлять или удалять поля динамически.

Преимущества использования автономных компонентов

Автономные компоненты упрощают разработку на Angular, поскольку устраняют необходимость в NgModule. Зависимости компонента (например, реактивные формы или маршруты) можно подключать непосредственно внутри самого компонента, что даёт следующие плюсы: — Меньше шаблонного кода; — Лучшая модульность; — Более быстрая разработка.

Для работы с такими формами удобно использовать FormArray и FormGroup из Angular — они дают гибкость при добавлении, удалении и изменении контролов в форме.

Пошаговое руководство по созданию динамических форм

1. Установка Angular и создание нового проекта

Прежде чем перейти к продвинутым примерам, начнём с создания нового приложения.

Для начала установите Angular CLI:

npm install @angular/cli

Теперь создайте новое Angular-приложение. В качестве формата стилей можно выбрать SCSS, а для SSR — указать «Нет»:

ng new dynamic-forms-sample-app

2. Создание динамической формы

Чтобы создать форму динамически, мы используем FormGroup и FormArray из модуля реактивных форм в Angular.

Вот полная реализация:

import { Component } from "@angular/core";
import {
  FormBuilder,
  FormGroup,
  FormArray,
  Validators,
  ReactiveFormsModule,
} from "@angular/forms";

@Component({
  selector: "app-root",
  standalone: true,
  imports: [ReactiveFormsModule],
  templateUrl: "./app.component.html",
  styleUrls: ["./app.component.scss"],
})
export class AppComponent {
  dynamicForm: FormGroup; // Main form group

  constructor(private fb: FormBuilder) {
    this.dynamicForm = this.fb.group({
      name: [""], // Simple input field
      email: [""], // Another input field
      fields: this.fb.array([]), // Dynamic fields will be stored here
    });
  }

  // Getter to access the FormArray for dynamic fields
  get fields(): FormArray {
    return this.dynamicForm.get("fields") as FormArray;
  }

  /**
   * Adds a new field to the dynamic form.
   */
  addField() {
    const fieldGroup = this.fb.group({
      label: [""], // Label for the field
      value: [""], // Value of the field
    });
    this.fields.push(fieldGroup);
  }

  /**
   * Removes a field from the dynamic form at a specific index.
   * @param index Index of the field to be removed.
   */
  removeField(index: number) {
    this.fields.removeAt(index);
  }

  /**
   * Submits the form and logs its current value to the console.
   */
  submitForm() {
    console.log(this.dynamicForm.value);
  }
}

Объяснение методов

addField(). Этот метод создаёт новую группу формы с двумя контролами: label и value. Новая группа добавляется в массив fields. Это позволяет пользователю добавлять поля в форму динамически.

removeField(index: number). Этот метод удаляет группу формы из массива fields по указанному индексу. Полезно, когда пользователь хочет удалить больше не нужное поле.

submitForm(). Метод собирает текущее состояние формы и выводит его в консоль. В реальном приложении данные можно было бы отправить на сервер или использовать для обновления интерфейса.

Шаблон компонента

Откройте файл app.component.html, удалите всё содержимое и добавьте следующий шаблон. В нём динамически отображаются элементы управления формой, а также предусмотрены кнопки для добавления и удаления полей:

<form [formGroup]="dynamicForm" (ngSubmit)="submitForm()">
  <div>
    <label>Name:</label>
    <input formControlName="name" />
  </div>

  <div>
    <label>Email:</label>
    <input formControlName="email" />
  </div>

  <div formArrayName="fields">
    @for(field of fields.controls; let i = $index; track field) {
      <div [formGroupName]="i">
        <label>
          Label:
          <input formControlName="label" />
        </label>
        <label>
          Value:
          <input formControlName="value" />
        </label>
        <button type="button" (click)="removeField(i)">Remove</button>
      </div>
    }
  </div>

  <button type="button" (click)="addField()">Add Field</button>
  <button type="submit">Submit</button>
</form>

Разбор шаблона

Статичные поля (Name и Email). Эти поля отображаются всегда и используют formControlName для привязки к форме.

Динамические поля (fields). Директива *ngFor перебирает FormArray и отображает каждое динамическое поле. У каждого поля есть привязки formControlName для label и value.

Кнопки. Кнопка Add Field вызывает метод addField() для добавления нового динамического поля. Каждое динамическое поле содержит кнопку Remove, которая вызывает removeField().

Создание динамической формы на основе данных из API

Во многих приложениях структура формы поступает из внешнего источника — например, из конфигурации, хранящейся на сервере. Получение конфигурации формы

Предположим, что API возвращает следующий JSON:

{
  "fields": [
    { "label": "Username", "type": "text", "required": true },
    { "label": "Age", "type": "number", "required": false },
    {
      "label": "Gender",
      "type": "select",
      "options": ["Male", "Female"],
      "required": true
    }
  ]
}

Вот как сгенерировать форму динамически на основе этой конфигурации:

import { Component, OnInit } from "@angular/core";
import { FormBuilder, FormGroup, Validators } from "@angular/forms";

@Component({
  selector: "app-dynamic-api-form",
  templateUrl: "./dynamic-api-form.component.html",
})
export class DynamicApiFormComponent implements OnInit {
  dynamicForm!: FormGroup; // The main reactive form instance
  formConfig: any; // The configuration object fetched from the API

  constructor(private fb: FormBuilder) {}

  ngOnInit() {
    this.fetchFormConfig().then((config) => {
      this.formConfig = config;
      this.buildForm(config.fields); // Build the form based on the configuration
    });
  }

  /**
   * Simulates fetching form configuration from an API.
   * In a real application, this would be an HTTP request.
   */
  async fetchFormConfig() {
    // Simulate API call
    return {
      fields: [
        { label: "Username", type: "text", required: true },
        { label: "Age", type: "number", required: false },
        {
          label: "Gender",
          type: "select",
          options: ["Male", "Female"],
          required: true,
        },
      ],
    };
  }

  /**
   * Dynamically creates the form controls based on the fetched configuration.
   */
  buildForm(fields: any[]) {
    const controls: any = {};
    fields.forEach((field) => {
      const validators = field.required ? [Validators.required] : [];
      controls[field.label] = ["", validators];
    });
    this.dynamicForm = this.fb.group(controls);
  }

  /**
   * Handles form submission, logging the form value to the console.
   */
  submitForm() {
    console.log(this.dynamicForm.value);
  }
}

dynamicForm хранит основной экземпляр FormGroup для реактивной формы. Формируется динамически на основе ответа от API.

formConfig содержит объект конфигурации, полученный от API. Определяет поля, их типы, правила валидации и (при необходимости) список опций.

ngOnInit(). Жизненный цикл компонента, который запускается после его инициализации. Вызывает fetchFormConfig() для получения конфигурации формы и её настройки.

fetchFormConfig(). Эмулирует вызов API для получения конфигурации формы. В реальном проекте этот мок следует заменить настоящим HTTP-запросом.

   {
     fields: [
       { label: "Name", type: "text", required: true },
       { label: "Age", type: "number", required: true },
       {
         label: "Gender",
         type: "select",
         required: true,
         options: ["Male", "Female", "Other"],
       },
     ];
   }

buildForm() динамически создаёт FormGroup на основе полученной конфигурации. Для каждого поля в конфигурации добавляет FormControl в FormGroup и назначает валидаторы (например, Validators.required), если поле отмечено как обязательное.

   {
     Name: ['', [Validators.required]],
     Age: ['', [Validators.required]],
     Gender: ['', [Validators.required]]
   }

submitForm() вызывается при отправке формы пользователем. Если форма валидна — выводит значения формы в консоль. Если форма невалидна — выводит сообщение об ошибке.

 {
     Name: 'John Doe',
     Age: 30,
     Gender: 'Male'
   }

Как всё это работает вместе

Инициализация

Когда компонент загружается, ngOnInit() вызывает fetchFormConfig(), чтобы смоделировать получение структуры формы.

Формирование формы

Метод buildForm() использует полученную конфигурацию и динамически создаёт реактивную форму.

Взаимодействие с пользователем

Форма отображается в шаблоне HTML. Пользователь может вводить значения или выбирать опции — в зависимости от типа каждого поля.

Валидация

Элементы формы соблюдают правила валидации (например, обязательные поля). Невалидные поля отображают сообщения об ошибке при взаимодействии с ними.

Отправка формы

При отправке submitForm() проверяет, валидна ли форма, и обрабатывает значения формы соответствующим образом.

Такой разбор помогает чётко понять назначение и работу каждого метода и свойства в TypeScript-коде.

@if(dynamicForm) {
  <form [formGroup]="dynamicForm" (ngSubmit)="submitForm()">
    @for(field of formConfig.fields; track field) {

      <label>{{ field.label }}</label>
      @switch(field.type) { 
        @case('text') {
          <input [formControlName]="field.label" />
        } 
        @case('number') {
          <input
            [formControlName]="field.label"
            type="number"
          />
        } 
        @case('select') {
          <select [formControlName]="field.label">
            @for(option of field.options; track option) {
              <option [value]="option">
                {{ option }}
              </option>
            }
          </select>
        } 
      } 
    }
    <button type="submit">Submit</button>
  </form>
}

Разбор HTML-структуры @if(dynamicForm) обеспечивает отображение формы только после её инициализации и получения конфигурации.

@for(field of formConfig.fields; track field) перебирает массив fields из конфигурации и динамически отображает элементы управления формой.

@switch(field.type) выбирает тип элемента формы для отображения в зависимости от значения type в конфигурации поля (например, text, number или select).

Типы полей ввода: — Текстовые поля (@case ('text')). Отображает элемент для ввода текста. — Числовые поля (@case ('number')). Отображает элемент <input type="number"> для ввода чисел. — Выпадающие списки (@case ('select').Отображает элемент <select>, динамически заполняя его опциями из массива options в конфигурации поля.

Динамическая установка валидаторов

Вы также можете изменять валидаторы на лету, в зависимости от ввода пользователя или определённых условий.

onRoleChange(role: string) {
  const emailControl = this.dynamicForm.get('email');
  if (role === 'admin') {
    emailControl?.setValidators([Validators.required, Validators.email]);
  } else {
    emailControl?.clearValidators();
  }
  emailControl?.updateValueAndValidity();
}

Заключение

Динамические формы в Angular — это гибкий способ создания интерактивных и масштабируемых пользовательских интерфейсов. Используя FormArray, FormGroup и конфигурации, полученные через API, вы можете создавать формы, которые адаптируются под нужды пользователя, сохраняя при этом надёжность и высокую производительность.


«Доктайп» — журнал о фронтенде. Читайте, слушайте и учитесь с нами.

ТелеграмПодкастБесплатные учебники

Читать дальше

AOT против JIT-компилятора: что лучше для разработки на Angular?

AOT против JIT-компилятора: что лучше для разработки на Angular?

Angular — один из самых популярных фреймворков для фронтенда — предлагает два подхода к компиляции: предварительная компиляция и динамическая компиляция во время выполнения. Оба метода играют важную роль в оптимизации приложений на Angular и повышении их производительности. В этом материале мы рассмотрим различия между ними, их преимущества и разберёмся, когда стоит использовать каждый из подходов.

Читать дальше
JS
  • 25 мая 2025
Как обнаружить изменения в Angular: пошаговая инструкция

Как обнаружить изменения в Angular: пошаговая инструкция

Как разработчики на Angular, мы нередко задумываемся, как фреймворк отслеживает изменения в данных и затем отображает их во вьюхе. Этот процесс называется стратегией обнаружения изменений в Angular. В этом материале мы разберёмся, как это работает, и научимся выбирать подходящую стратегию для разных сценариев.

Читать дальше
JS
  • 24 мая 2025
Компоненты в Angular 18: пошаговое руководство

Компоненты в Angular 18: пошаговое руководство

Angular развивается стремительно, и с выходом версии 18 появились новые возможности, которые разработчики могут использовать в своей работе. Одним из ключевых изменений в Angular 18 стало удаление традиционного файла app.module.ts — ему на смену пришли standalone-компоненты. Если вы только начинаете работать с Angular или переходите с более ранней версии, это пошаговое руководство поможет вам разобраться в базовых принципах компонентов в Angular 18. Независимо от вашего уровня — новичок вы или опытный разработчик — этот туториал покажет, как создавать, управлять и эффективно использовать компоненты в Angular.

Читать дальше
JS
  • 19 мая 2025
Полное руководство по Angular @if

Полное руководство по Angular @if

Одно из самых заметных нововведений в Angular — это встроенный синтаксис для управляющих конструкций, который появился в версии 17. Он решает одну из самых частых задач, с которой сталкивается каждый разработчик: показывать или скрывать элементы на странице в зависимости от условия. Раньше для этого использовали привычную структурную директиву *ngIf. Теперь у нас есть более современная альтернатива — синтаксис @if, часть нового подхода к управлению шаблоном.

В этом гайде мы сравним оба варианта, разберёмся, чем @if лучше, и покажем, как можно перейти на него автоматически. Также поговорим об одной распространённой ошибке — о том, как не стоит использовать @if вместе с пайпом async.

Читать дальше
JS
  • 18 мая 2025
Модули Angular для организации кода и ленивой загрузки

Модули Angular для организации кода и ленивой загрузки

Модули — один из ключевых инструментов Angular для построения масштабируемых и поддерживаемых приложений. В этой статье мы подробно рассмотрим:

  • что такое модули в Angular;
  • зачем они нужны;
  • как их использовать для структурирования кода;
  • как реализовать «ленивую» загрузку модулей;
  • и чем отличаются Feature, Core и Shared модули.

Если вы только начинаете изучать Angular или хотите углубить свои знания, эта статья поможет вам лучше понять, как правильно организовать архитектуру Angular-приложения.

Читать дальше
JS
  • 12 мая 2025
Навигация в Angular: RouterLink, Router.navigate и Router.navigateByUrl

Навигация в Angular: RouterLink, Router.navigate и Router.navigateByUrl

Директива RouterLink позволяет настраивать переходы между маршрутами прямо в шаблоне Angular. А методы Router.navigate и Router.navigateByUrl, доступные в классе Router, дают возможность управлять навигацией программно — прямо из кода компонентов.

Разберёмся, как работают RouterLink, Router.navigate и Router.navigateByUrl.

Читать дальше
JS
  • 11 мая 2025
Полное руководство по Lazy Loading в Angular

Полное руководство по Lazy Loading в Angular

Если вы создаёте большое Angular-приложение, вам наверняка важно, чтобы оно загружалось быстро. Представьте, что вы устраиваете вечеринку и хотите подавать закуски не сразу, а по мере прихода гостей, чтобы не перегрузить кухню. «Ленивая» загрузка в Angular работает примерно так же: вместо того чтобы загружать всё приложение целиком сразу, вы подгружаете только те части, которые нужны — и только когда они нужны.

В этом пошаговом руководстве мы разберём, как реализовать lazy loading в Angular.

Читать дальше
JS
  • 10 мая 2025
Все (ну или почти все) способы автоматически перезагрузить страницу раз в N секунд

Все (ну или почти все) способы автоматически перезагрузить страницу раз в N секунд

Иногда страницу нужно просто перезагрузить. Полностью. Не компонент, не блок, а именно целиком. Без обсуждений, без лишней логики. Например, чтобы:

  • экран с результатами обновлялся каждые 10 секунд;
  • интерфейс на стенде показывал последние данные без кнопок;
  • страницы в интранете не устаревали, пока никто не смотрит.

Это можно сделать в любой связке: HTML, JS, Python, PHP, Go, Node.js — не важно. Ну и если говорить совсем прямо, то совсем разных способов всего три, а остальное просто вариации.

Читать дальше
JS
  • 5 мая 2025
Vite 6: Новый этап в развитии фронтенд-разработки

Vite 6: Новый этап в развитии фронтенд-разработки

Vite — это современный инструмент сборки, который значительно ускоряет процесс разработки фронтенда, благодаря своим невероятно быстрым и удобным функциям. И вот, наконец, вышел новый релиз Vite 6, который приносит массу улучшений и новых возможностей для разработчиков. Давайте посмотрим, что нового появилось в Vite 6 и как это может повлиять на вашу работу.

Читать дальше
JS
  • 16 января 2025