UNPKG

@junte/ui

Version:

Quality Angular UI components kit

204 lines 21 kB
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"]}