angular-dynamic-forms-ai
Version:
AI-enhanced dynamic form generator for Angular using Template-Driven Forms
313 lines (297 loc) • 36 kB
JavaScript
import { Component, Input, Output, EventEmitter } from '@angular/core';
import * as i0 from "@angular/core";
import * as i1 from "../services/ai-validation.service";
import * as i2 from "@angular/common";
import * as i3 from "@angular/forms";
class DynamicFormComponent {
aiValidationService;
formConfig;
formSubmit = new EventEmitter();
constructor(aiValidationService) {
this.aiValidationService = aiValidationService;
}
ngOnInit() {
// Apply AI-based validation rules to each field
this.formConfig.fields = this.formConfig.fields.map(field => this.aiValidationService.suggestValidations(field));
}
isFieldVisible(field) {
if (!field.conditionalDisplay) {
return true;
}
const dependentField = this.formConfig.fields.find(f => f.name === field.conditionalDisplay?.dependsOn);
return dependentField?.value === field.conditionalDisplay.showWhen;
}
onSubmit(form) {
if (form.valid) {
const formData = this.formConfig.fields.reduce((acc, field) => {
if (this.isFieldVisible(field)) {
acc[field.name] = field.value;
}
return acc;
}, {});
this.formSubmit.emit(formData);
}
}
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: DynamicFormComponent, deps: [{ token: i1.AiValidationService }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.0.6", type: DynamicFormComponent, selector: "lib-dynamic-form", inputs: { formConfig: "formConfig" }, outputs: { formSubmit: "formSubmit" }, ngImport: i0, template: `
<form
<div *ngFor="let field of formConfig.fields" class="form-field" [ngClass]="{'hidden': !isFieldVisible(field)}">
<label [for]="field.name" class="form-label">
{{ field.label }}
<span *ngIf="field.validations?.required" class="required-marker">*</span>
</label>
<ng-container [ngSwitch]="field.type">
<!-- Text, Email, Number, Password inputs -->
<input *ngSwitchCase="'text'"
[]="field.type"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.maxLength"
[]="field.validations?.pattern"
[]="field.validations?.email"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'email'"
type="email"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="true"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'number'"
type="number"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'password'"
type="password"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.pattern"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<!-- Select dropdown -->
<select *ngSwitchCase="'select'"
[]="field.name"
[]="field.name"
[(ngModel)]="field.value"
[]="field.validations?.required"
class="form-select"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<option value="">Select {{ field.label }}</option>
<option *ngFor="let option of field.options" [value]="option.value">
{{ option.label }}
</option>
</select>
<!-- Textarea -->
<textarea *ngSwitchCase="'textarea'"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.maxLength"
class="form-textarea"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
</textarea>
</ng-container>
<!-- Error messages -->
<div *ngIf="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
class="error-messages"
[]="field.name + '-error'"
role="alert">
<div *ngIf="fieldModel.errors?.required">
{{ field.errorMessages?.required }}
</div>
<div *ngIf="fieldModel.errors?.minlength">
{{ field.errorMessages?.minlength }}
</div>
<div *ngIf="fieldModel.errors?.maxlength">
{{ field.errorMessages?.maxlength }}
</div>
<div *ngIf="fieldModel.errors?.email">
{{ field.errorMessages?.email }}
</div>
<div *ngIf="fieldModel.errors?.pattern">
{{ field.errorMessages?.pattern }}
</div>
</div>
</div>
<button type="submit"
[]="!form.valid"
[]="formConfig.submitButton?.class || 'submit-button'"
[]="!form.valid">
{{ formConfig.submitButton?.text || 'Submit' }}
</button>
</form>
`, isInline: true, styles: [".dynamic-form{max-width:600px;margin:0 auto;padding:20px;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Arial,sans-serif}.form-field{margin-bottom:20px}.form-label{display:block;margin-bottom:8px;font-weight:500;color:#333}.required-marker{color:#dc3545;margin-left:4px}.form-input,.form-select,.form-textarea{width:100%;padding:8px 12px;border:1px solid #ced4da;border-radius:4px;font-size:16px;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}.form-textarea{min-height:100px;resize:vertical}.form-input:focus,.form-select:focus,.form-textarea:focus{border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem #007bff40}.error{border-color:#dc3545}.error:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem #dc354540}.error-messages{margin-top:5px;color:#dc3545;font-size:14px}.hidden{display:none}.submit-button{background-color:#007bff;color:#fff;padding:10px 20px;border:none;border-radius:4px;font-size:16px;cursor:pointer;transition:background-color .15s ease-in-out}.submit-button:hover:not(:disabled){background-color:#0056b3}.submit-button:disabled{background-color:#6c757d;cursor:not-allowed;opacity:.65}@media (max-width: 768px){.dynamic-form{padding:15px}}\n"], dependencies: [{ kind: "directive", type: i2.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i2.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i2.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i3.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i3.NgSelectOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.ɵNgSelectMultipleOption, selector: "option", inputs: ["ngValue", "value"] }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NumberValueAccessor, selector: "input[type=number][formControlName],input[type=number][formControl],input[type=number][ngModel]" }, { kind: "directive", type: i3.SelectControlValueAccessor, selector: "select:not([multiple])[formControlName],select:not([multiple])[formControl],select:not([multiple])[ngModel]", inputs: ["compareWith"] }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i3.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }, { kind: "directive", type: i3.MinLengthValidator, selector: "[minlength][formControlName],[minlength][formControl],[minlength][ngModel]", inputs: ["minlength"] }, { kind: "directive", type: i3.MaxLengthValidator, selector: "[maxlength][formControlName],[maxlength][formControl],[maxlength][ngModel]", inputs: ["maxlength"] }, { kind: "directive", type: i3.PatternValidator, selector: "[pattern][formControlName],[pattern][formControl],[pattern][ngModel]", inputs: ["pattern"] }, { kind: "directive", type: i3.EmailValidator, selector: "[email][formControlName],[email][formControl],[email][ngModel]", inputs: ["email"] }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i3.NgForm, selector: "form:not([ngNoForm]):not([formGroup]),ng-form,[ngForm]", inputs: ["ngFormOptions"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }] });
}
export { DynamicFormComponent };
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.0.6", ngImport: i0, type: DynamicFormComponent, decorators: [{
type: Component,
args: [{ selector: 'lib-dynamic-form', template: `
<form
<div *ngFor="let field of formConfig.fields" class="form-field" [ngClass]="{'hidden': !isFieldVisible(field)}">
<label [for]="field.name" class="form-label">
{{ field.label }}
<span *ngIf="field.validations?.required" class="required-marker">*</span>
</label>
<ng-container [ngSwitch]="field.type">
<!-- Text, Email, Number, Password inputs -->
<input *ngSwitchCase="'text'"
[]="field.type"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.maxLength"
[]="field.validations?.pattern"
[]="field.validations?.email"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'email'"
type="email"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="true"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'number'"
type="number"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<input *ngSwitchCase="'password'"
type="password"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.pattern"
class="form-input"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<!-- Select dropdown -->
<select *ngSwitchCase="'select'"
[]="field.name"
[]="field.name"
[(ngModel)]="field.value"
[]="field.validations?.required"
class="form-select"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
<option value="">Select {{ field.label }}</option>
<option *ngFor="let option of field.options" [value]="option.value">
{{ option.label }}
</option>
</select>
<!-- Textarea -->
<textarea *ngSwitchCase="'textarea'"
[]="field.name"
[]="field.name"
[]="field.placeholder || ''"
[(ngModel)]="field.value"
[]="field.validations?.required"
[]="field.validations?.minLength"
[]="field.validations?.maxLength"
class="form-textarea"
[]="{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}"
[]="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
[]="field.name + '-error'">
</textarea>
</ng-container>
<!-- Error messages -->
<div *ngIf="fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)"
class="error-messages"
[]="field.name + '-error'"
role="alert">
<div *ngIf="fieldModel.errors?.required">
{{ field.errorMessages?.required }}
</div>
<div *ngIf="fieldModel.errors?.minlength">
{{ field.errorMessages?.minlength }}
</div>
<div *ngIf="fieldModel.errors?.maxlength">
{{ field.errorMessages?.maxlength }}
</div>
<div *ngIf="fieldModel.errors?.email">
{{ field.errorMessages?.email }}
</div>
<div *ngIf="fieldModel.errors?.pattern">
{{ field.errorMessages?.pattern }}
</div>
</div>
</div>
<button type="submit"
[]="!form.valid"
[]="formConfig.submitButton?.class || 'submit-button'"
[]="!form.valid">
{{ formConfig.submitButton?.text || 'Submit' }}
</button>
</form>
`, styles: [".dynamic-form{max-width:600px;margin:0 auto;padding:20px;font-family:-apple-system,BlinkMacSystemFont,Segoe UI,Roboto,Helvetica Neue,Arial,sans-serif}.form-field{margin-bottom:20px}.form-label{display:block;margin-bottom:8px;font-weight:500;color:#333}.required-marker{color:#dc3545;margin-left:4px}.form-input,.form-select,.form-textarea{width:100%;padding:8px 12px;border:1px solid #ced4da;border-radius:4px;font-size:16px;transition:border-color .15s ease-in-out,box-shadow .15s ease-in-out}.form-textarea{min-height:100px;resize:vertical}.form-input:focus,.form-select:focus,.form-textarea:focus{border-color:#80bdff;outline:0;box-shadow:0 0 0 .2rem #007bff40}.error{border-color:#dc3545}.error:focus{border-color:#dc3545;box-shadow:0 0 0 .2rem #dc354540}.error-messages{margin-top:5px;color:#dc3545;font-size:14px}.hidden{display:none}.submit-button{background-color:#007bff;color:#fff;padding:10px 20px;border:none;border-radius:4px;font-size:16px;cursor:pointer;transition:background-color .15s ease-in-out}.submit-button:hover:not(:disabled){background-color:#0056b3}.submit-button:disabled{background-color:#6c757d;cursor:not-allowed;opacity:.65}@media (max-width: 768px){.dynamic-form{padding:15px}}\n"] }]
}], ctorParameters: function () { return [{ type: i1.AiValidationService }]; }, propDecorators: { formConfig: [{
type: Input
}], formSubmit: [{
type: Output
}] } });
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"dynamic-form.component.js","sourceRoot":"","sources":["../../../../../projects/dynamic-forms/src/lib/components/dynamic-form.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,EAAE,YAAY,EAAU,MAAM,eAAe,CAAC;;;;;AAK/E,MAuOa,oBAAoB;IAIX;IAHX,UAAU,CAAc;IACvB,UAAU,GAAG,IAAI,YAAY,EAAO,CAAC;IAE/C,YAAoB,mBAAwC;QAAxC,wBAAmB,GAAnB,mBAAmB,CAAqB;IAAG,CAAC;IAEhE,QAAQ;QACN,gDAAgD;QAChD,IAAI,CAAC,UAAU,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,GAAG,CAAC,KAAK,CAAC,EAAE,CAC1D,IAAI,CAAC,mBAAmB,CAAC,kBAAkB,CAAC,KAAK,CAAC,CACnD,CAAC;IACJ,CAAC;IAED,cAAc,CAAC,KAAsB;QACnC,IAAI,CAAC,KAAK,CAAC,kBAAkB,EAAE;YAC7B,OAAO,IAAI,CAAC;SACb;QAED,MAAM,cAAc,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,IAAI,CAChD,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CAAC,kBAAkB,EAAE,SAAS,CACpD,CAAC;QAEF,OAAO,cAAc,EAAE,KAAK,KAAK,KAAK,CAAC,kBAAkB,CAAC,QAAQ,CAAC;IACrE,CAAC;IAED,QAAQ,CAAC,IAAY;QACnB,IAAI,IAAI,CAAC,KAAK,EAAE;YACd,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC,GAAG,EAAE,KAAK,EAAE,EAAE;gBAC5D,IAAI,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,EAAE;oBAC9B,GAAG,CAAC,KAAK,CAAC,IAAI,CAAC,GAAG,KAAK,CAAC,KAAK,CAAC;iBAC/B;gBACD,OAAO,GAAG,CAAC;YACb,CAAC,EAAE,EAAS,CAAC,CAAC;YAEd,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAChC;IACH,CAAC;uGApCU,oBAAoB;2FAApB,oBAAoB,qIArOrB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqIT;;SAgGU,oBAAoB;2FAApB,oBAAoB;kBAvOhC,SAAS;+BACE,kBAAkB,YAClB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqIT;0GAiGQ,UAAU;sBAAlB,KAAK;gBACI,UAAU;sBAAnB,MAAM","sourcesContent":["import { Component, Input, Output, EventEmitter, OnInit } from '@angular/core';\nimport { NgForm } from '@angular/forms';\nimport { FormConfig, FormFieldConfig } from '../interfaces/form-field.interface';\nimport { AiValidationService } from '../services/ai-validation.service';\n\n@Component({\n  selector: 'lib-dynamic-form',\n  template: `\n    <form #form=\"ngForm\" (ngSubmit)=\"onSubmit(form)\" class=\"dynamic-form\" [attr.aria-label]=\"formConfig.fields[0]?.label + ' form'\">\n      <div *ngFor=\"let field of formConfig.fields\" class=\"form-field\" [ngClass]=\"{'hidden': !isFieldVisible(field)}\">\n        <label [for]=\"field.name\" class=\"form-label\">\n          {{ field.label }}\n          <span *ngIf=\"field.validations?.required\" class=\"required-marker\">*</span>\n        </label>\n\n        <ng-container [ngSwitch]=\"field.type\">\n          <!-- Text, Email, Number, Password inputs -->\n          <input *ngSwitchCase=\"'text'\" \n                 [type]=\"field.type\"\n                 [id]=\"field.name\"\n                 [name]=\"field.name\"\n                 [placeholder]=\"field.placeholder || ''\"\n                 [(ngModel)]=\"field.value\"\n                 #fieldModel=\"ngModel\"\n                 [required]=\"field.validations?.required\"\n                 [minlength]=\"field.validations?.minLength\"\n                 [maxlength]=\"field.validations?.maxLength\"\n                 [pattern]=\"field.validations?.pattern\"\n                 [email]=\"field.validations?.email\"\n                 class=\"form-input\"\n                 [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                 [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                 [attr.aria-describedby]=\"field.name + '-error'\">\n\n          <input *ngSwitchCase=\"'email'\"\n                 type=\"email\"\n                 [id]=\"field.name\"\n                 [name]=\"field.name\"\n                 [placeholder]=\"field.placeholder || ''\"\n                 [(ngModel)]=\"field.value\"\n                 #fieldModel=\"ngModel\"\n                 [required]=\"field.validations?.required\"\n                 [email]=\"true\"\n                 class=\"form-input\"\n                 [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                 [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                 [attr.aria-describedby]=\"field.name + '-error'\">\n\n          <input *ngSwitchCase=\"'number'\"\n                 type=\"number\"\n                 [id]=\"field.name\"\n                 [name]=\"field.name\"\n                 [placeholder]=\"field.placeholder || ''\"\n                 [(ngModel)]=\"field.value\"\n                 #fieldModel=\"ngModel\"\n                 [required]=\"field.validations?.required\"\n                 class=\"form-input\"\n                 [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                 [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                 [attr.aria-describedby]=\"field.name + '-error'\">\n\n          <input *ngSwitchCase=\"'password'\"\n                 type=\"password\"\n                 [id]=\"field.name\"\n                 [name]=\"field.name\"\n                 [placeholder]=\"field.placeholder || ''\"\n                 [(ngModel)]=\"field.value\"\n                 #fieldModel=\"ngModel\"\n                 [required]=\"field.validations?.required\"\n                 [minlength]=\"field.validations?.minLength\"\n                 [pattern]=\"field.validations?.pattern\"\n                 class=\"form-input\"\n                 [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                 [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                 [attr.aria-describedby]=\"field.name + '-error'\">\n\n          <!-- Select dropdown -->\n          <select *ngSwitchCase=\"'select'\"\n                  [id]=\"field.name\"\n                  [name]=\"field.name\"\n                  [(ngModel)]=\"field.value\"\n                  #fieldModel=\"ngModel\"\n                  [required]=\"field.validations?.required\"\n                  class=\"form-select\"\n                  [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                  [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                  [attr.aria-describedby]=\"field.name + '-error'\">\n            <option value=\"\">Select {{ field.label }}</option>\n            <option *ngFor=\"let option of field.options\" [value]=\"option.value\">\n              {{ option.label }}\n            </option>\n          </select>\n\n          <!-- Textarea -->\n          <textarea *ngSwitchCase=\"'textarea'\"\n                    [id]=\"field.name\"\n                    [name]=\"field.name\"\n                    [placeholder]=\"field.placeholder || ''\"\n                    [(ngModel)]=\"field.value\"\n                    #fieldModel=\"ngModel\"\n                    [required]=\"field.validations?.required\"\n                    [minlength]=\"field.validations?.minLength\"\n                    [maxlength]=\"field.validations?.maxLength\"\n                    class=\"form-textarea\"\n                    [ngClass]=\"{'error': fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)}\"\n                    [attr.aria-invalid]=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n                    [attr.aria-describedby]=\"field.name + '-error'\">\n          </textarea>\n        </ng-container>\n\n        <!-- Error messages -->\n        <div *ngIf=\"fieldModel.invalid && (fieldModel.dirty || fieldModel.touched)\"\n             class=\"error-messages\"\n             [id]=\"field.name + '-error'\"\n             role=\"alert\">\n          <div *ngIf=\"fieldModel.errors?.required\">\n            {{ field.errorMessages?.required }}\n          </div>\n          <div *ngIf=\"fieldModel.errors?.minlength\">\n            {{ field.errorMessages?.minlength }}\n          </div>\n          <div *ngIf=\"fieldModel.errors?.maxlength\">\n            {{ field.errorMessages?.maxlength }}\n          </div>\n          <div *ngIf=\"fieldModel.errors?.email\">\n            {{ field.errorMessages?.email }}\n          </div>\n          <div *ngIf=\"fieldModel.errors?.pattern\">\n            {{ field.errorMessages?.pattern }}\n          </div>\n        </div>\n      </div>\n\n      <button type=\"submit\"\n              [disabled]=\"!form.valid\"\n              [class]=\"formConfig.submitButton?.class || 'submit-button'\"\n              [attr.aria-disabled]=\"!form.valid\">\n        {{ formConfig.submitButton?.text || 'Submit' }}\n      </button>\n    </form>\n  `,\n  styles: [`\n    .dynamic-form {\n      max-width: 600px;\n      margin: 0 auto;\n      padding: 20px;\n      font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif;\n    }\n\n    .form-field {\n      margin-bottom: 20px;\n    }\n\n    .form-label {\n      display: block;\n      margin-bottom: 8px;\n      font-weight: 500;\n      color: #333;\n    }\n\n    .required-marker {\n      color: #dc3545;\n      margin-left: 4px;\n    }\n\n    .form-input,\n    .form-select,\n    .form-textarea {\n      width: 100%;\n      padding: 8px 12px;\n      border: 1px solid #ced4da;\n      border-radius: 4px;\n      font-size: 16px;\n      transition: border-color 0.15s ease-in-out, box-shadow 0.15s ease-in-out;\n    }\n\n    .form-textarea {\n      min-height: 100px;\n      resize: vertical;\n    }\n\n    .form-input:focus,\n    .form-select:focus,\n    .form-textarea:focus {\n      border-color: #80bdff;\n      outline: 0;\n      box-shadow: 0 0 0 0.2rem rgba(0, 123, 255, 0.25);\n    }\n\n    .error {\n      border-color: #dc3545;\n    }\n\n    .error:focus {\n      border-color: #dc3545;\n      box-shadow: 0 0 0 0.2rem rgba(220, 53, 69, 0.25);\n    }\n\n    .error-messages {\n      margin-top: 5px;\n      color: #dc3545;\n      font-size: 14px;\n    }\n\n    .hidden {\n      display: none;\n    }\n\n    .submit-button {\n      background-color: #007bff;\n      color: white;\n      padding: 10px 20px;\n      border: none;\n      border-radius: 4px;\n      font-size: 16px;\n      cursor: pointer;\n      transition: background-color 0.15s ease-in-out;\n    }\n\n    .submit-button:hover:not(:disabled) {\n      background-color: #0056b3;\n    }\n\n    .submit-button:disabled {\n      background-color: #6c757d;\n      cursor: not-allowed;\n      opacity: 0.65;\n    }\n\n    @media (max-width: 768px) {\n      .dynamic-form {\n        padding: 15px;\n      }\n    }\n  `]\n})\nexport class DynamicFormComponent implements OnInit {\n  @Input() formConfig!: FormConfig;\n  @Output() formSubmit = new EventEmitter<any>();\n\n  constructor(private aiValidationService: AiValidationService) {}\n\n  ngOnInit() {\n    // Apply AI-based validation rules to each field\n    this.formConfig.fields = this.formConfig.fields.map(field => \n      this.aiValidationService.suggestValidations(field)\n    );\n  }\n\n  isFieldVisible(field: FormFieldConfig): boolean {\n    if (!field.conditionalDisplay) {\n      return true;\n    }\n\n    const dependentField = this.formConfig.fields.find(\n      f => f.name === field.conditionalDisplay?.dependsOn\n    );\n\n    return dependentField?.value === field.conditionalDisplay.showWhen;\n  }\n\n  onSubmit(form: NgForm) {\n    if (form.valid) {\n      const formData = this.formConfig.fields.reduce((acc, field) => {\n        if (this.isFieldVisible(field)) {\n          acc[field.name] = field.value;\n        }\n        return acc;\n      }, {} as any);\n\n      this.formSubmit.emit(formData);\n    }\n  }\n}\n"]}