@junte/ui
Version:
Quality Angular UI components kit
204 lines • 21 kB
JavaScript
import { __decorate, __metadata } from "tslib";
import { animate, state, style, transition, trigger } from '@angular/animations';
import { Component, ContentChild, ContentChildren, EventEmitter, HostBinding, HostListener, Input, Output, QueryList, TemplateRef } from '@angular/core';
import { FormArray, FormGroup } from '@angular/forms';
import { NGXLogger } from 'ngx-logger';
import { MethodApi, PropertyApi } from '../../core/decorators/api';
import { Height } from '../../core/enums/height';
import { State } from '../../core/enums/state';
import { UI } from '../../core/enums/ui';
import { FormControlComponent } from './control/form-control.component';
let FormComponent = class FormComponent {
constructor(logger) {
this.logger = logger;
this.host = 'jnt-form-host';
this.ui = UI;
this._state = { success: false };
this._height = Height.default;
this.submitted = new EventEmitter();
this.checked = new EventEmitter();
}
set height(height) {
this._height = height || Height.default;
}
success() {
this._state.success = true;
setTimeout(() => this._state.success = false, 2100);
}
ngOnInit() {
if (!!this.form) {
this.form.statusChanges.subscribe(() => {
this.render();
this.checked.emit(this.check(this.form));
});
}
}
render() {
this.logger.debug('render form');
if (!!this.controls) {
this.controls.filter(component => !!component.name && !!component.messages.length)
.forEach(component => component.check());
}
}
check(form) {
this.logger.debug('check form');
let errors = [];
for (const key in form.controls) {
const control = form.controls[key];
if (control instanceof FormGroup || control instanceof FormArray) {
errors = errors.concat(this.check(control));
}
else {
if (!!control.errors && control.dirty) {
errors.push(control);
}
}
}
this.logger.debug('controls have errors = ', errors.map(e => e.name));
return errors;
}
/**
* @deprecated please use submit() instead
*/
onSubmit() {
this.submit();
}
submit() {
this.logger.debug('submit form');
if (!!this.form) {
this.validate(this.form);
if (this.form.valid) {
this.submitted.emit();
this.refresh(this.form);
}
}
}
validate(form) {
this.logger.debug('validate form');
for (const key in form.controls) {
const control = form.controls[key];
if (control instanceof FormGroup || control instanceof FormArray) {
this.validate(control);
}
else {
control.markAsDirty();
control.updateValueAndValidity({ emitEvent: false });
}
}
this.render();
this.checked.emit(this.check(this.form));
}
refresh(form) {
this.logger.debug('refresh form');
for (const key in form.controls) {
const control = form.controls[key];
if (control instanceof FormGroup || control instanceof FormArray) {
this.refresh(control);
}
else {
control.markAsPristine();
}
}
}
};
FormComponent.ctorParameters = () => [
{ type: NGXLogger }
];
__decorate([
HostBinding('attr.host'),
__metadata("design:type", Object)
], FormComponent.prototype, "host", void 0);
__decorate([
HostBinding('attr.data-height'),
__metadata("design:type", String)
], FormComponent.prototype, "_height", void 0);
__decorate([
PropertyApi({
description: 'Name form group',
type: 'FormGroup',
}),
Input('formGroup'),
__metadata("design:type", FormGroup)
], FormComponent.prototype, "form", void 0);
__decorate([
PropertyApi({
description: 'Title for form',
type: 'string',
}),
Input(),
__metadata("design:type", String)
], FormComponent.prototype, "title", void 0);
__decorate([
PropertyApi({
description: 'State of form',
path: 'ui.state',
options: [State.error, State.loading]
}),
Input(),
__metadata("design:type", String)
], FormComponent.prototype, "state", void 0);
__decorate([
PropertyApi({
description: 'Height of form',
path: 'ui.height',
options: [Height.default, Height.fluid],
default: Height.default
}),
Input(),
__metadata("design:type", String),
__metadata("design:paramtypes", [String])
], FormComponent.prototype, "height", null);
__decorate([
ContentChild('formTitleTemplate'),
__metadata("design:type", TemplateRef)
], FormComponent.prototype, "titleTemplate", void 0);
__decorate([
ContentChild('formFooterTemplate'),
__metadata("design:type", TemplateRef)
], FormComponent.prototype, "footerTemplate", void 0);
__decorate([
ContentChildren(FormControlComponent, { descendants: true }),
__metadata("design:type", QueryList)
], FormComponent.prototype, "controls", void 0);
__decorate([
Output(),
__metadata("design:type", Object)
], FormComponent.prototype, "submitted", void 0);
__decorate([
Output(),
__metadata("design:type", Object)
], FormComponent.prototype, "checked", void 0);
__decorate([
MethodApi({ description: 'show success animation' }),
__metadata("design:type", Function),
__metadata("design:paramtypes", []),
__metadata("design:returntype", void 0)
], FormComponent.prototype, "success", null);
__decorate([
HostListener('submit'),
__metadata("design:type", Function),
__metadata("design:paramtypes", []),
__metadata("design:returntype", void 0)
], FormComponent.prototype, "submit", null);
FormComponent = __decorate([
Component({
selector: 'jnt-form',
template: "<form child-of=\"jnt-form-host\" >\n <header child-of=\"jnt-form-host\" *ngIf=\"!!title || !!titleTemplate\">\n <h3 child-of=\"jnt-form-host\" *ngIf=\"!!title && !titleTemplate\">{{title}}</h3>\n <ng-container *ngIf=\"!!titleTemplate\">\n <ng-container *ngTemplateOutlet=\"titleTemplate\"></ng-container>\n </ng-container>\n </header>\n\n <div child-of=\"jnt-form-host\" data-content>\n <ng-content></ng-content>\n </div>\n\n <footer child-of=\"jnt-form-host\" *ngIf=\"!!footerTemplate\">\n <ng-container *ngTemplateOutlet=\"footerTemplate\"></ng-container>\n </footer>\n\n <div child-of=\"jnt-form-host\" data-loading *ngIf=\"state === ui.state.loading\">\n <jnt-spinner child-of=\"jnt-form-host\" [size]=\"ui.size.large\"></jnt-spinner>\n </div>\n <div child-of=\"jnt-form-host\" data-error *ngIf=\"state === ui.state.error\">\n <jnt-icon child-of=\"jnt-form-host\" [icon]=\"ui.icons.sad\"></jnt-icon>\n </div>\n <div child-of=\"jnt-form-host\" data-success *ngIf=\"_state.success\" [@success]=\"_state.success\">\n <jnt-icon child-of=\"jnt-form-host\" [icon]=\"ui.icons.animated.success\"></jnt-icon>\n </div>\n</form>",
animations: [
trigger('success', [
state('void', style({
opacity: 0
})),
state('*', style({
opacity: 1
})),
transition('void <=> *', [
animate('.3s ease-in-out')
]),
]),
]
}),
__metadata("design:paramtypes", [NGXLogger])
], FormComponent);
export { FormComponent };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"ng://@junte/ui/","sources":["lib/forms/form/form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { animate, state, style, transition, trigger } from '@angular/animations';\nimport {\n  Component,\n  ContentChild,\n  ContentChildren,\n  EventEmitter,\n  HostBinding,\n  HostListener,\n  Input,\n  OnInit,\n  Output,\n  QueryList,\n  TemplateRef\n} from '@angular/core';\nimport { AbstractControl, FormArray, FormGroup } from '@angular/forms';\nimport { NGXLogger } from 'ngx-logger';\nimport { MethodApi, PropertyApi } from '../../core/decorators/api';\nimport { Height } from '../../core/enums/height';\nimport { State } from '../../core/enums/state';\nimport { UI } from '../../core/enums/ui';\nimport { FormControlComponent } from './control/form-control.component';\n\n@Component({\n  selector: 'jnt-form',\n  templateUrl: './form.encapsulated.html',\n  animations: [\n    trigger('success', [\n        state(\n          'void',\n          style({\n            opacity: 0\n          })\n        ),\n        state(\n          '*',\n          style({\n            opacity: 1\n          })\n        ),\n        transition(\n          'void <=> *',\n          [\n            animate('.3s ease-in-out')\n          ]\n        ),\n      ]\n    ),\n  ]\n})\nexport class FormComponent implements OnInit {\n\n  @HostBinding('attr.host')\n  readonly host = 'jnt-form-host';\n\n  ui = UI;\n\n  _state = {success: false};\n\n  @HostBinding('attr.data-height')\n  _height: Height = Height.default;\n\n  @PropertyApi({\n    description: 'Name form group',\n    type: 'FormGroup',\n  })\n  @Input('formGroup')\n  form: FormGroup;\n\n  @PropertyApi({\n    description: 'Title for form',\n    type: 'string',\n  })\n  @Input()\n  title: string;\n\n  @PropertyApi({\n    description: 'State of form',\n    path: 'ui.state',\n    options: [State.error, State.loading]\n  })\n  @Input()\n  state: State;\n\n  @PropertyApi({\n    description: 'Height of form',\n    path: 'ui.height',\n    options: [Height.default, Height.fluid],\n    default: Height.default\n  })\n  @Input() set height(height: Height) {\n    this._height = height || Height.default;\n  }\n\n  @ContentChild('formTitleTemplate')\n  titleTemplate: TemplateRef<any>;\n\n  @ContentChild('formFooterTemplate')\n  footerTemplate: TemplateRef<any>;\n\n  @ContentChildren(FormControlComponent, {descendants: true})\n  controls: QueryList<FormControlComponent>;\n\n  @Output()\n  submitted = new EventEmitter();\n\n  @Output()\n  checked = new EventEmitter<AbstractControl[]>();\n\n  @MethodApi({description: 'show success animation'})\n  success() {\n    this._state.success = true;\n    setTimeout(() => this._state.success = false, 2100);\n  }\n\n  constructor(private logger: NGXLogger) {\n  }\n\n  ngOnInit() {\n    if (!!this.form) {\n      this.form.statusChanges.subscribe(() => {\n        this.render();\n        this.checked.emit(this.check(this.form));\n      });\n    }\n  }\n\n  render() {\n    this.logger.debug('render form');\n    if (!!this.controls) {\n      this.controls.filter(component => !!component.name && !!component.messages.length)\n        .forEach(component => component.check());\n    }\n  }\n\n  private check(form: FormGroup | FormArray): AbstractControl[] {\n    this.logger.debug('check form');\n    let errors = [];\n    for (const key in form.controls) {\n      const control = form.controls[key];\n\n      if (control instanceof FormGroup || control instanceof FormArray) {\n        errors = errors.concat(this.check(control));\n      } else {\n        if (!!control.errors && control.dirty) {\n          errors.push(control);\n        }\n      }\n    }\n    this.logger.debug('controls have errors = ', errors.map(e => e.name));\n    return errors;\n  }\n\n  /**\n   * @deprecated please use submit() instead\n   */\n  onSubmit() {\n    this.submit();\n  }\n\n  @HostListener('submit')\n  submit() {\n    this.logger.debug('submit form');\n    if (!!this.form) {\n      this.validate(this.form);\n\n      if (this.form.valid) {\n        this.submitted.emit();\n        this.refresh(this.form);\n      }\n    }\n  }\n\n  private validate(form: FormGroup | FormArray) {\n    this.logger.debug('validate form');\n    for (const key in form.controls) {\n      const control = form.controls[key];\n\n      if (control instanceof FormGroup || control instanceof FormArray) {\n        this.validate(control);\n      } else {\n        control.markAsDirty();\n        control.updateValueAndValidity({emitEvent: false});\n      }\n    }\n\n    this.render();\n    this.checked.emit(this.check(this.form));\n  }\n\n  private refresh(form: FormGroup | FormArray) {\n    this.logger.debug('refresh form');\n    for (const key in form.controls) {\n      const control = form.controls[key];\n\n      if (control instanceof FormGroup || control instanceof FormArray) {\n        this.refresh(control);\n      } else {\n        control.markAsPristine();\n      }\n    }\n  }\n}\n"]}