UNPKG

@alauda-fe/common

Version:

Alauda frontend team common codes.

239 lines 33.7 kB
import { ButtonModule, IconModule, InputModule } from '@alauda/ui'; import { CommonModule } from '@angular/common'; import { ChangeDetectionStrategy, Component, forwardRef } from '@angular/core'; import { FormsModule, NG_VALIDATORS, NG_VALUE_ACCESSOR, } from '@angular/forms'; import { mergeLeft } from 'ramda'; import { CalcPipe, K8S_RESOURCE_LABEL_KEY_NAME_PATTERN, K8S_RESOURCE_LABEL_KEY_PREFIX_PATTERN, K8S_RESOURCE_LABEL_VALUE_PATTERN, PurePipe, createNestedFormControl, } from '../../core/public-api'; import { EffectDirectiveModule } from '../../effect-directive/module'; import { TranslateModule } from '../../translate/public-api'; import { ErrorsMapperModule } from '../errors-mapper/public-api'; import { ValidatorsModule } from '../validators/public-api'; import * as i0 from "@angular/core"; import * as i1 from "@angular/common"; import * as i2 from "../../translate/translate.pipe"; import * as i3 from "@angular/forms"; import * as i4 from "@alauda/ui"; import * as i5 from "../validators/function-validator.directive"; import * as i6 from "../validators/maxlength-minlength-validator.directive"; import * as i7 from "../errors-mapper/errors-mapper-component/component"; import * as i8 from "../../effect-directive/e2e-attribute-binding.directive"; import * as i9 from "../../effect-directive/validators.directive"; const _c0 = a0 => ({ key: a0 }); const _c1 = a0 => ({ duplicated: a0 }); function LabelsEditorComponent_tr_12_acl_errors_mapper_5_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelement(0, "acl-errors-mapper", 20); i0.ɵɵpipe(1, "translate"); } if (rf & 2) { i0.ɵɵnextContext(); const key_r4 = i0.ɵɵreference(3); i0.ɵɵproperty("errors", key_r4.errors)("errorsMapper", i0.ɵɵpureFunction1(7, _c1, i0.ɵɵpipeBind2(1, 2, "label_key_already_exists", i0.ɵɵpureFunction1(5, _c0, key_r4.errors == null ? null : key_r4.errors.duplicated)))); } } function LabelsEditorComponent_tr_12_acl_errors_mapper_9_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelement(0, "acl-errors-mapper", 21); } if (rf & 2) { i0.ɵɵnextContext(); const value_r5 = i0.ɵɵreference(8); i0.ɵɵproperty("errors", value_r5.errors); } } function LabelsEditorComponent_tr_12_Template(rf, ctx) { if (rf & 1) { const _r1 = i0.ɵɵgetCurrentView(); i0.ɵɵelementStart(0, "tr")(1, "td")(2, "input", 13, 0); i0.ɵɵpipe(4, "calc"); i0.ɵɵlistener("ngModelChange", function LabelsEditorComponent_tr_12_Template_input_ngModelChange_2_listener($event) { const index_r2 = i0.ɵɵrestoreView(_r1).index; const ctx_r2 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r2.editLabel(index_r2, "key", $event)); }); i0.ɵɵelementEnd(); i0.ɵɵtemplate(5, LabelsEditorComponent_tr_12_acl_errors_mapper_5_Template, 2, 9, "acl-errors-mapper", 14); i0.ɵɵelementEnd(); i0.ɵɵelementStart(6, "td")(7, "input", 15, 1); i0.ɵɵlistener("ngModelChange", function LabelsEditorComponent_tr_12_Template_input_ngModelChange_7_listener($event) { const index_r2 = i0.ɵɵrestoreView(_r1).index; const ctx_r2 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r2.editLabel(index_r2, "value", $event)); }); i0.ɵɵelementEnd(); i0.ɵɵtemplate(9, LabelsEditorComponent_tr_12_acl_errors_mapper_9_Template, 1, 1, "acl-errors-mapper", 16); i0.ɵɵelementEnd(); i0.ɵɵelementStart(10, "td", 17)(11, "button", 18); i0.ɵɵlistener("click", function LabelsEditorComponent_tr_12_Template_button_click_11_listener() { const index_r2 = i0.ɵɵrestoreView(_r1).index; const ctx_r2 = i0.ɵɵnextContext(); return i0.ɵɵresetView(ctx_r2.removeLabel(index_r2)); }); i0.ɵɵelement(12, "aui-icon", 19); i0.ɵɵelementEnd()()(); } if (rf & 2) { const label_r6 = ctx.$implicit; const index_r2 = ctx.index; const key_r4 = i0.ɵɵreference(3); const value_r5 = i0.ɵɵreference(8); const ctx_r2 = i0.ɵɵnextContext(); i0.ɵɵclassProp("duplicated", ctx_r2.duplicatedRows[index_r2]); i0.ɵɵadvance(2); i0.ɵɵproperty("ngModel", label_r6.key)("aclValidateFn", i0.ɵɵpipeBind3(4, 10, ctx_r2.validateLabelKey, ctx_r2.duplicatedRows, index_r2)); i0.ɵɵadvance(3); i0.ɵɵproperty("ngIf", ctx_r2.submitted || key_r4.dirty); i0.ɵɵadvance(2); i0.ɵɵproperty("ngModel", label_r6.value)("pattern", ctx_r2.K8S_RESOURCE_LABEL_VALUE_PATTERN.pattern); i0.ɵɵadvance(2); i0.ɵɵproperty("ngIf", ctx_r2.submitted || value_r5.dirty); i0.ɵɵadvance(2); i0.ɵɵproperty("square", true)("plain", true); } } function LabelsEditorComponent_tr_13_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "tr")(1, "td", 9)(2, "div", 22); i0.ɵɵtext(3); i0.ɵɵpipe(4, "translate"); i0.ɵɵelementEnd()()(); } if (rf & 2) { i0.ɵɵadvance(3); i0.ɵɵtextInterpolate(i0.ɵɵpipeBind1(4, 1, "no_label")); } } export class LabelsEditorComponent extends createNestedFormControl() { constructor() { super(...arguments); this.K8S_RESOURCE_LABEL_VALUE_PATTERN = K8S_RESOURCE_LABEL_VALUE_PATTERN; this.duplicatedRows = {}; this.validateLabelKey = (duplicatedRows, index) => (control) => mergeLeft(this.keyDuplicated(duplicatedRows, index)(control), this.keyPattern(control)); } addLabel() { this.model.push({ key: '', value: '' }); } editLabel(index, key, value) { this.model[index][key] = value; this.emitModel(); } removeLabel(index) { this.model.splice(index, 1); this.emitModel(); } writeValue(value) { super.writeValue(value); this.checkDuplicatedRows(); } emitValue(value) { super.emitValue(value); this.checkDuplicatedRows(); } valueIn(value) { return Object.entries(value ?? {}).map(([key, value]) => ({ key, value, })); } modelOut(model) { return model.reduce((acc, { key, value }) => { const k = key.trim(); if (k) { acc[k] = value.trim(); } return acc; }, {}); } checkDuplicatedRows() { this.duplicatedRows = (this.model ?? []).reduce((acc, { key }, index, labels) => { if (!key) { return acc; } for (const [i, label] of labels.entries()) { if (i !== index && label.key === key) { acc[index] = true; break; } } return acc; }, {}); } keyPattern(control) { if (!control.value) { return null; } const [name, prefix = ''] = control.value.split('/').reverse(); if (name.length > 63 || K8S_RESOURCE_LABEL_KEY_NAME_PATTERN.pattern.test(name) === false) { return { pattern: true }; } if (!!prefix && (prefix.length > 253 || K8S_RESOURCE_LABEL_KEY_PREFIX_PATTERN.pattern.test(prefix) === false)) { return { pattern: true }; } return null; } keyDuplicated(duplicatedRows, index) { return (control) => { const key = control.value; if (duplicatedRows[index]) { return { duplicated: key }; } return null; }; } static { this.ɵfac = /*@__PURE__*/ (() => { let ɵLabelsEditorComponent_BaseFactory; return function LabelsEditorComponent_Factory(t) { returnLabelsEditorComponent_BaseFactory ||LabelsEditorComponent_BaseFactory = i0.ɵɵgetInheritedFactory(LabelsEditorComponent)))(t || LabelsEditorComponent); }; })(); } static { this.ɵcmp = /*@__PURE__*/ i0.ɵɵdefineComponent({ type: LabelsEditorComponent, selectors: [["acl-labels-editor"]], standalone: true, features: [i0.ɵɵProvidersFeature([ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => LabelsEditorComponent), multi: true, }, { provide: NG_VALIDATORS, useExisting: forwardRef(() => LabelsEditorComponent), multi: true, }, ]), i0.ɵɵInheritDefinitionFeature, i0.ɵɵStandaloneFeature], decls: 22, vars: 15, consts: [["key", "ngModel"], ["value", "ngModel"], [1, "acl-array-form-table", "mp-sticky-table-header", "mp-sticky-table-actions"], ["required", "", "width", "50%"], ["width", "50%"], [2, "min-width", "44px"], [3, "duplicated", 4, "ngFor", "ngForOf"], [4, "ngIf"], [1, "acl-array-form-table--row-action"], ["colspan", "100"], [1, "acl-array-form-table__bottom-control-buttons"], ["aui-button", "primary", "size", "mini", "type", "button", 3, "click", "plain", "disabled"], ["icon", "plus_circle"], ["aui-input", "", "size", "small", "name", "key", "required", "", 3, "ngModelChange", "ngModel", "aclValidateFn"], [3, "errors", "errorsMapper", 4, "ngIf"], ["aui-input", "", "size", "small", "name", "value", "aclMaxlength", "63", 3, "ngModelChange", "ngModel", "pattern"], [3, "errors", 4, "ngIf"], [1, "acl-array-form-table__action-col"], ["aui-button", "text", "size", "small", "type", "button", 3, "click", "square", "plain"], ["icon", "minus_circle"], [3, "errors", "errorsMapper"], [3, "errors"], [1, "placeholder"]], template: function LabelsEditorComponent_Template(rf, ctx) { if (rf & 1) { i0.ɵɵelementStart(0, "table", 2)(1, "thead")(2, "tr")(3, "th", 3)(4, "span"); i0.ɵɵtext(5); i0.ɵɵpipe(6, "translate"); i0.ɵɵelementEnd()(); i0.ɵɵelementStart(7, "th", 4); i0.ɵɵtext(8); i0.ɵɵpipe(9, "translate"); i0.ɵɵelementEnd(); i0.ɵɵelement(10, "th", 5); i0.ɵɵelementEnd()(); i0.ɵɵelementStart(11, "tbody"); i0.ɵɵtemplate(12, LabelsEditorComponent_tr_12_Template, 13, 14, "tr", 6)(13, LabelsEditorComponent_tr_13_Template, 5, 3, "tr", 7); i0.ɵɵelementStart(14, "tr", 8)(15, "td", 9)(16, "div", 10)(17, "button", 11); i0.ɵɵlistener("click", function LabelsEditorComponent_Template_button_click_17_listener() { return ctx.addLabel(); }); i0.ɵɵelement(18, "aui-icon", 12); i0.ɵɵelementStart(19, "span"); i0.ɵɵtext(20); i0.ɵɵpipe(21, "translate"); i0.ɵɵelementEnd()()()()()()(); } if (rf & 2) { i0.ɵɵclassProp("submitted", ctx.submitted); i0.ɵɵadvance(5); i0.ɵɵtextInterpolate(i0.ɵɵpipeBind1(6, 9, "key")); i0.ɵɵadvance(3); i0.ɵɵtextInterpolate(i0.ɵɵpipeBind1(9, 11, "value")); i0.ɵɵadvance(4); i0.ɵɵproperty("ngForOf", ctx.model); i0.ɵɵadvance(); i0.ɵɵproperty("ngIf", !(ctx.model == null ? null : ctx.model.length)); i0.ɵɵadvance(4); i0.ɵɵproperty("plain", true)("disabled", (ctx.model == null ? null : ctx.model.length) >= 50); i0.ɵɵadvance(3); i0.ɵɵtextInterpolate1(" ", i0.ɵɵpipeBind1(21, 13, "add"), " "); } }, dependencies: [CommonModule, i1.NgForOf, i1.NgIf, TranslateModule, i2.TranslatePipe, FormsModule, i3.DefaultValueAccessor, i3.NgControlStatus, i3.RequiredValidator, i3.PatternValidator, i3.NgModel, InputModule, i4.InputComponent, ValidatorsModule, i5.FunctionValidatorDirective, i6.MaxLengthValidatorDirective, ErrorsMapperModule, i7.ErrorsMapperComponent, IconModule, i4.IconComponent, ButtonModule, i4.ButtonComponent, CalcPipe, EffectDirectiveModule, i8.E2eAttributeBindingDirective, i9.ValidatorsDirective], styles: ["[_nghost-%COMP%] {display:block;width:100%}[_nghost-%COMP%] .acl-array-form-table tbody td{vertical-align:top;line-height:var(--aui-inline-height-s)}[_nghost-%COMP%] .acl-array-form-table tbody tr .placeholder{text-align:center;color:rgb(var(--aui-color-help-text))}[_nghost-%COMP%] .acl-array-form-table tbody tr.duplicated{background-color:rgb(var(--aui-color-r-7))}"], changeDetection: 0 }); } } (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassMetadata(LabelsEditorComponent, [{ type: Component, args: [{ standalone: true, selector: 'acl-labels-editor', changeDetection: ChangeDetectionStrategy.OnPush, imports: [ CommonModule, TranslateModule, FormsModule, InputModule, ValidatorsModule, ErrorsMapperModule, IconModule, ButtonModule, PurePipe, CalcPipe, EffectDirectiveModule, ], providers: [ { provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => LabelsEditorComponent), multi: true, }, { provide: NG_VALIDATORS, useExisting: forwardRef(() => LabelsEditorComponent), multi: true, }, ], template: "<table\n class=\"acl-array-form-table mp-sticky-table-header mp-sticky-table-actions\"\n [class.submitted]=\"submitted\"\n>\n <thead>\n <tr>\n <th\n required\n width=\"50%\"\n >\n <span>{{ 'key' | translate }}</span>\n </th>\n <th width=\"50%\">{{ 'value' | translate }}</th>\n <th style=\"min-width: 44px\"></th>\n </tr>\n </thead>\n <tbody>\n <tr\n *ngFor=\"let label of model; let index = index\"\n [class.duplicated]=\"duplicatedRows[index]\"\n >\n <td>\n <input\n aui-input\n size=\"small\"\n name=\"key\"\n #key=\"ngModel\"\n [ngModel]=\"label.key\"\n (ngModelChange)=\"editLabel(index, 'key', $event)\"\n required\n [aclValidateFn]=\"validateLabelKey | calc: duplicatedRows:index\"\n />\n <acl-errors-mapper\n *ngIf=\"submitted || key.dirty\"\n [errors]=\"key.errors\"\n [errorsMapper]=\"{\n duplicated:\n 'label_key_already_exists'\n | translate: { key: key.errors?.duplicated }\n }\"\n ></acl-errors-mapper>\n </td>\n <td>\n <input\n aui-input\n size=\"small\"\n name=\"value\"\n #value=\"ngModel\"\n [ngModel]=\"label.value\"\n (ngModelChange)=\"editLabel(index, 'value', $event)\"\n aclMaxlength=\"63\"\n [pattern]=\"K8S_RESOURCE_LABEL_VALUE_PATTERN.pattern\"\n />\n <acl-errors-mapper\n *ngIf=\"submitted || value.dirty\"\n [errors]=\"value.errors\"\n ></acl-errors-mapper>\n </td>\n <td class=\"acl-array-form-table__action-col\">\n <button\n aui-button=\"text\"\n size=\"small\"\n type=\"button\"\n [square]=\"true\"\n [plain]=\"true\"\n (click)=\"removeLabel(index)\"\n >\n <aui-icon icon=\"minus_circle\"></aui-icon>\n </button>\n </td>\n </tr>\n <tr *ngIf=\"!model?.length\">\n <td colspan=\"100\">\n <div class=\"placeholder\">{{ 'no_label' | translate }}</div>\n </td>\n </tr>\n <tr class=\"acl-array-form-table--row-action\">\n <td colspan=\"100\">\n <div class=\"acl-array-form-table__bottom-control-buttons\">\n <button\n aui-button=\"primary\"\n [plain]=\"true\"\n size=\"mini\"\n type=\"button\"\n (click)=\"addLabel()\"\n [disabled]=\"model?.length >= 50\"\n >\n <aui-icon icon=\"plus_circle\"></aui-icon>\n <span>\n {{ 'add' | translate }}\n </span>\n </button>\n </div>\n </td>\n </tr>\n </tbody>\n</table>\n", styles: [":host::ng-deep{display:block;width:100%}:host::ng-deep .acl-array-form-table tbody td{vertical-align:top;line-height:var(--aui-inline-height-s)}:host::ng-deep .acl-array-form-table tbody tr .placeholder{text-align:center;color:rgb(var(--aui-color-help-text))}:host::ng-deep .acl-array-form-table tbody tr.duplicated{background-color:rgb(var(--aui-color-r-7))}\n"] }] }], null, null); })(); (() => { (typeof ngDevMode === "undefined" || ngDevMode) && i0.ɵsetClassDebugInfo(LabelsEditorComponent, { className: "LabelsEditorComponent" }); })(); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"component.js","sourceRoot":"","sources":["../../../../../../libs/common/src/form/labels-editor/component.ts","../../../../../../libs/common/src/form/labels-editor/template.html"],"names":[],"mappings":"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","sourcesContent":["import { ButtonModule, IconModule, InputModule } from '@alauda/ui';\nimport { CommonModule } from '@angular/common';\nimport { ChangeDetectionStrategy, Component, forwardRef } from '@angular/core';\nimport {\n  AbstractControl,\n  FormsModule,\n  NG_VALIDATORS,\n  NG_VALUE_ACCESSOR,\n} from '@angular/forms';\nimport { mergeLeft } from 'ramda';\n\nimport {\n  CalcPipe,\n  K8S_RESOURCE_LABEL_KEY_NAME_PATTERN,\n  K8S_RESOURCE_LABEL_KEY_PREFIX_PATTERN,\n  K8S_RESOURCE_LABEL_VALUE_PATTERN,\n  PurePipe,\n  createNestedFormControl,\n} from '../../core/public-api';\nimport { EffectDirectiveModule } from '../../effect-directive/module';\nimport { TranslateModule } from '../../translate/public-api';\nimport { ErrorsMapperModule } from '../errors-mapper/public-api';\nimport { ValidatorsModule } from '../validators/public-api';\n\n@Component({\n  standalone: true,\n  selector: 'acl-labels-editor',\n  templateUrl: './template.html',\n  styleUrls: ['./style.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  imports: [\n    CommonModule,\n    TranslateModule,\n    FormsModule,\n    InputModule,\n    ValidatorsModule,\n    ErrorsMapperModule,\n    IconModule,\n    ButtonModule,\n    PurePipe,\n    CalcPipe,\n    EffectDirectiveModule,\n  ],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => LabelsEditorComponent),\n      multi: true,\n    },\n    {\n      provide: NG_VALIDATORS,\n      useExisting: forwardRef(() => LabelsEditorComponent),\n      multi: true,\n    },\n  ],\n})\nexport class LabelsEditorComponent extends createNestedFormControl<\n  Record<string, string>,\n  Pair[]\n>() {\n  K8S_RESOURCE_LABEL_VALUE_PATTERN = K8S_RESOURCE_LABEL_VALUE_PATTERN;\n  duplicatedRows: Record<number, boolean> = {};\n\n  addLabel() {\n    this.model.push({ key: '', value: '' });\n  }\n\n  editLabel(index: number, key: keyof Pair, value: string) {\n    this.model[index][key] = value;\n    this.emitModel();\n  }\n\n  removeLabel(index: number) {\n    this.model.splice(index, 1);\n    this.emitModel();\n  }\n\n  validateLabelKey =\n    (duplicatedRows: Record<number, boolean>, index: number) =>\n    (control: AbstractControl) =>\n      mergeLeft(\n        this.keyDuplicated(duplicatedRows, index)(control),\n        this.keyPattern(control),\n      );\n\n  override writeValue(value: Record<string, string>): void {\n    super.writeValue(value);\n\n    this.checkDuplicatedRows();\n  }\n\n  override emitValue(value: Record<string, string>): void {\n    super.emitValue(value);\n\n    this.checkDuplicatedRows();\n  }\n\n  protected override valueIn(value: Record<string, string>): Pair[] {\n    return Object.entries(value ?? {}).map(([key, value]) => ({\n      key,\n      value,\n    }));\n  }\n\n  protected override modelOut(model: Pair[]): Record<string, string> {\n    return model.reduce((acc, { key, value }) => {\n      const k = key.trim();\n      if (k) {\n        acc[k] = value.trim();\n      }\n      return acc;\n    }, {} as Record<string, string>);\n  }\n\n  private checkDuplicatedRows() {\n    this.duplicatedRows = (this.model ?? []).reduce(\n      (acc, { key }, index, labels) => {\n        if (!key) {\n          return acc;\n        }\n\n        for (const [i, label] of labels.entries()) {\n          if (i !== index && label.key === key) {\n            acc[index] = true;\n            break;\n          }\n        }\n        return acc;\n      },\n      {} as Record<number, boolean>,\n    );\n  }\n\n  private keyPattern(control: AbstractControl<string>) {\n    if (!control.value) {\n      return null;\n    }\n    const [name, prefix = ''] = control.value.split('/').reverse();\n    if (\n      name.length > 63 ||\n      K8S_RESOURCE_LABEL_KEY_NAME_PATTERN.pattern.test(name) === false\n    ) {\n      return { pattern: true };\n    }\n\n    if (\n      !!prefix &&\n      (prefix.length > 253 ||\n        K8S_RESOURCE_LABEL_KEY_PREFIX_PATTERN.pattern.test(prefix) === false)\n    ) {\n      return { pattern: true };\n    }\n\n    return null;\n  }\n\n  private keyDuplicated(\n    duplicatedRows: Record<number, boolean>,\n    index: number,\n  ) {\n    return (control: AbstractControl) => {\n      const key = control.value;\n      if (duplicatedRows[index]) {\n        return { duplicated: key };\n      }\n      return null;\n    };\n  }\n}\n\ninterface Pair {\n  key: string;\n  value: string;\n}\n","<table\n  class=\"acl-array-form-table mp-sticky-table-header mp-sticky-table-actions\"\n  [class.submitted]=\"submitted\"\n>\n  <thead>\n    <tr>\n      <th\n        required\n        width=\"50%\"\n      >\n        <span>{{ 'key' | translate }}</span>\n      </th>\n      <th width=\"50%\">{{ 'value' | translate }}</th>\n      <th style=\"min-width: 44px\"></th>\n    </tr>\n  </thead>\n  <tbody>\n    <tr\n      *ngFor=\"let label of model; let index = index\"\n      [class.duplicated]=\"duplicatedRows[index]\"\n    >\n      <td>\n        <input\n          aui-input\n          size=\"small\"\n          name=\"key\"\n          #key=\"ngModel\"\n          [ngModel]=\"label.key\"\n          (ngModelChange)=\"editLabel(index, 'key', $event)\"\n          required\n          [aclValidateFn]=\"validateLabelKey | calc: duplicatedRows:index\"\n        />\n        <acl-errors-mapper\n          *ngIf=\"submitted || key.dirty\"\n          [errors]=\"key.errors\"\n          [errorsMapper]=\"{\n            duplicated:\n              'label_key_already_exists'\n              | translate: { key: key.errors?.duplicated }\n          }\"\n        ></acl-errors-mapper>\n      </td>\n      <td>\n        <input\n          aui-input\n          size=\"small\"\n          name=\"value\"\n          #value=\"ngModel\"\n          [ngModel]=\"label.value\"\n          (ngModelChange)=\"editLabel(index, 'value', $event)\"\n          aclMaxlength=\"63\"\n          [pattern]=\"K8S_RESOURCE_LABEL_VALUE_PATTERN.pattern\"\n        />\n        <acl-errors-mapper\n          *ngIf=\"submitted || value.dirty\"\n          [errors]=\"value.errors\"\n        ></acl-errors-mapper>\n      </td>\n      <td class=\"acl-array-form-table__action-col\">\n        <button\n          aui-button=\"text\"\n          size=\"small\"\n          type=\"button\"\n          [square]=\"true\"\n          [plain]=\"true\"\n          (click)=\"removeLabel(index)\"\n        >\n          <aui-icon icon=\"minus_circle\"></aui-icon>\n        </button>\n      </td>\n    </tr>\n    <tr *ngIf=\"!model?.length\">\n      <td colspan=\"100\">\n        <div class=\"placeholder\">{{ 'no_label' | translate }}</div>\n      </td>\n    </tr>\n    <tr class=\"acl-array-form-table--row-action\">\n      <td colspan=\"100\">\n        <div class=\"acl-array-form-table__bottom-control-buttons\">\n          <button\n            aui-button=\"primary\"\n            [plain]=\"true\"\n            size=\"mini\"\n            type=\"button\"\n            (click)=\"addLabel()\"\n            [disabled]=\"model?.length >= 50\"\n          >\n            <aui-icon icon=\"plus_circle\"></aui-icon>\n            <span>\n              {{ 'add' | translate }}\n            </span>\n          </button>\n        </div>\n      </td>\n    </tr>\n  </tbody>\n</table>\n"]}