@inspark/inspark-components 14.0.59 → 14.0.60

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,11 +1,11 @@
1
- import { ChangeDetectorRef, EventEmitter, OnChanges, OnInit } from '@angular/core';
1
+ import { ChangeDetectorRef, EventEmitter, OnChanges, OnInit, SimpleChanges } from '@angular/core';
2
2
  import { ControlValueAccessor } from '@angular/forms';
3
3
  import * as i0 from "@angular/core";
4
4
  declare type IconPos = 'left' | 'right';
5
- interface SelectOption {
5
+ export interface SelectOption {
6
6
  label: string;
7
7
  title?: string;
8
- value: any;
8
+ value: string | number | boolean;
9
9
  icon?: string;
10
10
  readonly?: boolean;
11
11
  }
@@ -21,37 +21,28 @@ export declare class SelectButtonComponent implements ControlValueAccessor, OnIn
21
21
  disabled: boolean;
22
22
  required: boolean;
23
23
  multiple: boolean;
24
- initial?: any[];
24
+ initial?: Array<string | number | boolean>;
25
25
  focus: EventEmitter<any>;
26
26
  blur: EventEmitter<any>;
27
27
  click: EventEmitter<any>;
28
28
  change: EventEmitter<any>;
29
- selected: any[];
30
- value: Object | Object[];
31
- bindingtype: any;
29
+ selected: Array<string | number | boolean>;
30
+ value: string | number | boolean | Array<string | number | boolean> | null;
31
+ private propagateChange;
32
+ private propagateTouched;
32
33
  constructor(cdRef: ChangeDetectorRef);
33
34
  ngOnInit(): void;
34
- ngOnChanges(): void;
35
- clickButton(option: any): boolean;
36
- /**
37
- * Write form value to the DOM element (model => view)
38
- */
35
+ ngOnChanges(changes: SimpleChanges): void;
36
+ clickButton(option: SelectOption): void;
39
37
  writeValue(value: any): void;
40
- /**
41
- * Write form disabled state to the DOM element (model => view)
42
- */
43
- setDisabledState(isDisabled: boolean): void;
44
- /**
45
- * Update form when DOM element value changes (view => model)
46
- */
47
38
  registerOnChange(fn: any): void;
48
- /**
49
- * Update form when DOM element is blurred (view => model)
50
- */
51
39
  registerOnTouched(fn: any): void;
52
- private propagateChange;
40
+ setDisabledState(isDisabled: boolean): void;
53
41
  private onChange;
54
- private onTouched;
42
+ onBlur(): void;
43
+ onFocus(): void;
44
+ private syncSelectedWithOptions;
45
+ trackByValue(index: number, item: SelectOption): string | number | boolean;
55
46
  static ɵfac: i0.ɵɵFactoryDeclaration<SelectButtonComponent, never>;
56
47
  static ɵcmp: i0.ɵɵComponentDeclaration<SelectButtonComponent, "in-select-button", never, { "name": "name"; "testId": "testId"; "label": "label"; "icon": "icon"; "error": "error"; "iconPos": "iconPos"; "options": "options"; "disabled": "disabled"; "required": "required"; "multiple": "multiple"; "initial": "initial"; }, { "focus": "focus"; "blur": "blur"; "click": "click"; "change": "change"; }, never, never, false>;
57
48
  }
@@ -1,4 +1,4 @@
1
- import { ChangeDetectorRef, Component, EventEmitter, forwardRef, Input, Output } from '@angular/core';
1
+ import { ChangeDetectorRef, Component, EventEmitter, forwardRef, Input, Output, } from '@angular/core';
2
2
  import { NG_VALUE_ACCESSOR } from '@angular/forms';
3
3
  import * as i0 from "@angular/core";
4
4
  import * as i1 from "@angular/common";
@@ -14,38 +14,48 @@ export class SelectButtonComponent {
14
14
  this.error = '';
15
15
  this.iconPos = 'left';
16
16
  this.options = [];
17
+ this.disabled = false;
17
18
  this.required = false;
18
19
  this.multiple = false;
19
20
  this.focus = new EventEmitter();
20
21
  this.blur = new EventEmitter();
21
22
  this.click = new EventEmitter();
22
23
  this.change = new EventEmitter();
24
+ // хранит выбранные ЗНАЧЕНИЯ, а не объекты
23
25
  this.selected = [];
24
- this.propagateChange = (_) => {
25
- };
26
+ // то, что отдаем наружу в форму
27
+ this.value = null;
28
+ this.propagateChange = () => { };
29
+ this.propagateTouched = () => { };
26
30
  }
27
- ngOnInit() {
28
- }
29
- ngOnChanges() {
30
- if (this.initial) {
31
- this.selected = this.initial;
31
+ ngOnInit() { }
32
+ ngOnChanges(changes) {
33
+ // применяем initial только когда он реально поменялся
34
+ if (changes['initial'] && this.initial) {
35
+ this.selected = [...this.initial];
36
+ this.value = this.multiple ? [...this.initial] : this.initial[0] ?? null;
37
+ this.cdRef.detectChanges();
32
38
  }
33
- else {
34
- this.selected = [];
39
+ // если меняется список опций, надо попытаться восстановить выделение
40
+ if (changes['options'] && this.options && this.options.length > 0) {
41
+ this.syncSelectedWithOptions();
35
42
  }
36
43
  }
37
44
  clickButton(option) {
38
- if (option.readonly) {
39
- return false;
45
+ if (option.readonly || this.disabled) {
46
+ return;
40
47
  }
48
+ this.click.emit(option);
41
49
  if (this.multiple) {
42
- if (this.selected.indexOf(option.value) !== -1) {
43
- this.selected = this.selected.filter(item => item !== option.value);
50
+ if (this.selected.includes(option.value)) {
51
+ // убираем
52
+ this.selected = this.selected.filter(v => v !== option.value);
44
53
  }
45
54
  else {
46
- this.selected.push(option.value);
55
+ // добавляем
56
+ this.selected = [...this.selected, option.value];
47
57
  }
48
- this.value = this.options.filter(item => this.selected.indexOf(item.value) !== -1).map(val => val.value);
58
+ this.value = [...this.selected];
49
59
  }
50
60
  else {
51
61
  this.selected = [option.value];
@@ -54,46 +64,79 @@ export class SelectButtonComponent {
54
64
  this.cdRef.detectChanges();
55
65
  this.onChange();
56
66
  }
57
- /**
58
- * Write form value to the DOM element (model => view)
59
- */
67
+ // ControlValueAccessor: model -> view
60
68
  writeValue(value) {
61
69
  this.value = value;
62
- if (value !== null && value !== undefined) {
63
- if (Array.isArray(value)) {
64
- this.selected = value.map(val => val);
65
- }
66
- else {
67
- this.selected = [value];
68
- }
70
+ if (value === null || value === undefined) {
71
+ this.selected = [];
69
72
  }
73
+ else if (Array.isArray(value)) {
74
+ // multiple
75
+ this.selected = [...value];
76
+ }
77
+ else {
78
+ // single
79
+ this.selected = [value];
80
+ }
81
+ // когда значение пришло извне, надо убедиться что такие опции есть
82
+ this.syncSelectedWithOptions();
70
83
  this.cdRef.detectChanges();
71
84
  }
72
- /**
73
- * Write form disabled state to the DOM element (model => view)
74
- */
75
- setDisabledState(isDisabled) {
76
- this.disabled = isDisabled;
77
- }
78
- /**
79
- * Update form when DOM element value changes (view => model)
80
- */
85
+ // ControlValueAccessor
81
86
  registerOnChange(fn) {
82
- // Store the provided function as an internal method.
83
87
  this.propagateChange = fn;
84
88
  }
85
- /**
86
- * Update form when DOM element is blurred (view => model)
87
- */
89
+ // ControlValueAccessor
88
90
  registerOnTouched(fn) {
89
- // Store the provided function as an internal method.
90
- this.onTouched = fn;
91
+ this.propagateTouched = fn;
91
92
  }
93
+ // ControlValueAccessor: disabled
94
+ setDisabledState(isDisabled) {
95
+ this.disabled = isDisabled;
96
+ this.cdRef.detectChanges();
97
+ }
98
+ // пробросить наружу
92
99
  onChange() {
93
100
  this.propagateChange(this.value);
94
101
  this.change.emit(this.value);
95
102
  }
96
- onTouched() {
103
+ onBlur() {
104
+ this.propagateTouched();
105
+ this.blur.emit();
106
+ }
107
+ onFocus() {
108
+ this.focus.emit();
109
+ }
110
+ // синхронизируем текущее выбранное со свежими options
111
+ syncSelectedWithOptions() {
112
+ if (!this.options || this.options.length === 0) {
113
+ return;
114
+ }
115
+ if (this.multiple) {
116
+ // оставляем только те выбранные, которые реально есть в новых options
117
+ const optionValues = this.options.map(o => o.value);
118
+ this.selected = this.selected.filter(v => optionValues.includes(v));
119
+ this.value = [...this.selected];
120
+ }
121
+ else {
122
+ if (this.selected.length > 0) {
123
+ const current = this.selected[0];
124
+ const found = this.options.find(o => o.value === current);
125
+ if (!found) {
126
+ // выбранного больше нет в списке
127
+ this.selected = [];
128
+ this.value = null;
129
+ }
130
+ else {
131
+ this.selected = [found.value];
132
+ this.value = found.value;
133
+ }
134
+ }
135
+ }
136
+ }
137
+ // для *ngFor trackBy
138
+ trackByValue(index, item) {
139
+ return item.value;
97
140
  }
98
141
  }
99
142
  SelectButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectButtonComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
@@ -102,8 +145,8 @@ SelectButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0",
102
145
  provide: NG_VALUE_ACCESSOR,
103
146
  useExisting: forwardRef(() => SelectButtonComponent),
104
147
  multi: true,
105
- }
106
- ], usesOnChanges: true, ngImport: i0, template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{label}}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{error}}</span>\n </span>\n<div class=\"container\">\n <in-button class=\"button-none\"></in-button>\n <in-button (click)=\"clickButton(option)\"\n *ngFor=\"let option of options\"\n [color]=\"selected.indexOf(option.value) === -1 ? 'normal': 'primary' \"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name] = option.label\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.ButtonComponent, selector: "in-button", inputs: ["size", "color", "type", "name", "testId", "label", "icon", "className", "notify", "svg", "width", "link", "isLoading", "shadow", "isIconOnly", "linkType", "linkTarget", "pressed", "disabled", "iconPos", "selectedItem", "menuPlacement", "menuItems"], outputs: ["focus", "blur", "click"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }] });
148
+ },
149
+ ], usesOnChanges: true, ngImport: i0, template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{ label }}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{ error }}</span>\n</span>\n\n<div class=\"container\">\n <!-- \u0444\u0438\u043A\u0442\u0438\u0432\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430 \u0435\u0441\u043B\u0438 \u043D\u0443\u0436\u043D\u0430 \u0440\u0430\u0437\u043C\u0435\u0442\u043A\u0430 -->\n <in-button class=\"button-none\"></in-button>\n\n <in-button\n *ngFor=\"let option of options; trackBy: trackByValue\"\n (click)=\"clickButton(option)\"\n [color]=\"selected.includes(option.value) ? 'primary' : 'normal'\"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name]=\"option.label\"\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.ButtonComponent, selector: "in-button", inputs: ["size", "color", "type", "name", "testId", "label", "icon", "className", "notify", "svg", "width", "link", "isLoading", "shadow", "isIconOnly", "linkType", "linkTarget", "pressed", "disabled", "iconPos", "selectedItem", "menuPlacement", "menuItems"], outputs: ["focus", "blur", "click"] }, { kind: "pipe", type: i3.TranslatePipe, name: "translate" }] });
107
150
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectButtonComponent, decorators: [{
108
151
  type: Component,
109
152
  args: [{ selector: 'in-select-button', providers: [
@@ -111,8 +154,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
111
154
  provide: NG_VALUE_ACCESSOR,
112
155
  useExisting: forwardRef(() => SelectButtonComponent),
113
156
  multi: true,
114
- }
115
- ], template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{label}}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{error}}</span>\n </span>\n<div class=\"container\">\n <in-button class=\"button-none\"></in-button>\n <in-button (click)=\"clickButton(option)\"\n *ngFor=\"let option of options\"\n [color]=\"selected.indexOf(option.value) === -1 ? 'normal': 'primary' \"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name] = option.label\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"] }]
157
+ },
158
+ ], template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{ label }}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{ error }}</span>\n</span>\n\n<div class=\"container\">\n <!-- \u0444\u0438\u043A\u0442\u0438\u0432\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430 \u0435\u0441\u043B\u0438 \u043D\u0443\u0436\u043D\u0430 \u0440\u0430\u0437\u043C\u0435\u0442\u043A\u0430 -->\n <in-button class=\"button-none\"></in-button>\n\n <in-button\n *ngFor=\"let option of options; trackBy: trackByValue\"\n (click)=\"clickButton(option)\"\n [color]=\"selected.includes(option.value) ? 'primary' : 'normal'\"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name]=\"option.label\"\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"] }]
116
159
  }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { name: [{
117
160
  type: Input
118
161
  }], testId: [{
@@ -144,4 +187,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
144
187
  }], change: [{
145
188
  type: Output
146
189
  }] } });
147
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select-button.component.js","sourceRoot":"","sources":["../../../../../../projects/inspark-components/src/components/select-button/select-button.component.ts","../../../../../../projects/inspark-components/src/components/select-button/select-button.component.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectorRef, Component, EventEmitter, forwardRef, Input, OnChanges, OnInit, Output} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\n\n\ntype IconPos = 'left' | 'right';\n\ninterface SelectOption {\n  label: string;\n  title?: string;\n  value: any;\n  icon?: string;\n  readonly?: boolean;\n}\n\n\n@Component({\n  selector: 'in-select-button',\n  templateUrl: './select-button.component.html',\n  styleUrls: ['./select-button.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => SelectButtonComponent),\n      multi: true,\n    }\n  ],\n})\nexport class SelectButtonComponent implements ControlValueAccessor, OnInit, OnChanges {\n\n  @Input() name = '';\n  @Input() testId = '';\n  @Input() label = '';\n  @Input() icon = '';\n  @Input() error = '';\n  @Input() iconPos: IconPos = 'left';\n  @Input() options: SelectOption[] = [];\n  @Input() disabled: boolean;\n  @Input() required = false;\n  @Input() multiple = false;\n  @Input() initial?: any[];\n\n  @Output() focus: EventEmitter<any> = new EventEmitter();\n  @Output() blur: EventEmitter<any> = new EventEmitter();\n  @Output() click: EventEmitter<any> = new EventEmitter();\n  @Output() change: EventEmitter<any> = new EventEmitter();\n\n  selected = [];\n\n  value: Object | Object[];\n\n  bindingtype;\n\n  constructor(private cdRef: ChangeDetectorRef) {\n\n  }\n\n  ngOnInit() {\n  }\n\n  ngOnChanges() {\n    if (this.initial) {\n      this.selected = this.initial;\n    } else {\n      this.selected = [];\n    }\n  }\n\n  clickButton(option) {\n    if (option.readonly) {\n      return false;\n    }\n    if (this.multiple) {\n      if (this.selected.indexOf(option.value) !== -1) {\n        this.selected = this.selected.filter(item => item !== option.value);\n      } else {\n        this.selected.push(option.value);\n      }\n      this.value = this.options.filter(item => this.selected.indexOf(item.value) !== -1).map(val => val.value);\n    } else {\n      this.selected = [option.value];\n      this.value = option.value;\n    }\n    this.cdRef.detectChanges();\n    this.onChange();\n  }\n\n  /**\n   * Write form value to the DOM element (model => view)\n   */\n  writeValue(value: any): void {\n    this.value = value;\n    if (value !== null && value !== undefined) {\n      if (Array.isArray(value)) {\n        this.selected = value.map(val => val);\n      } else {\n        this.selected = [value];\n      }\n    }\n    this.cdRef.detectChanges();\n  }\n\n  /**\n   * Write form disabled state to the DOM element (model => view)\n   */\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n  }\n\n  /**\n   * Update form when DOM element value changes (view => model)\n   */\n  registerOnChange(fn: any): void {\n    // Store the provided function as an internal method.\n    this.propagateChange = fn;\n  }\n\n  /**\n   * Update form when DOM element is blurred (view => model)\n   */\n  registerOnTouched(fn: any): void {\n    // Store the provided function as an internal method.\n    this.onTouched = fn;\n  }\n\n  private propagateChange = (_: any) => {\n  }\n\n  private onChange() {\n    this.propagateChange(this.value);\n    this.change.emit(this.value);\n  }\n\n  private onTouched() {\n  }\n}\n","<span *ngIf=\"label\" class=\"c-label__content\">\n        {{label}}\n  <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n        <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{error}}</span>\n    </span>\n<div class=\"container\">\n  <in-button class=\"button-none\"></in-button>\n  <in-button (click)=\"clickButton(option)\"\n             *ngFor=\"let option of options\"\n             [color]=\"selected.indexOf(option.value) === -1 ? 'normal': 'primary' \"\n             [icon]=\"option.icon\"\n             [label]=\"option.label | translate\"\n             [testId]=\"'filter-buttons-' + option.value\"\n             [title]=\"option.title ? (option.title | translate) : ''\"\n             [name] = option.label\n             [disabled]=\"option.readonly || disabled\"\n  >\n  </in-button>\n</div>\n"]}
190
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select-button.component.js","sourceRoot":"","sources":["../../../../../../projects/inspark-components/src/components/select-button/select-button.component.ts","../../../../../../projects/inspark-components/src/components/select-button/select-button.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectorRef,\n  Component,\n  EventEmitter,\n  forwardRef,\n  Input,\n  OnChanges,\n  OnInit,\n  Output,\n  SimpleChanges,\n} from '@angular/core';\nimport {ControlValueAccessor, NG_VALUE_ACCESSOR} from '@angular/forms';\n\ntype IconPos = 'left' | 'right';\n\nexport interface SelectOption {\n  label: string;\n  title?: string;\n  value: string | number | boolean; // важно: примитив\n  icon?: string;\n  readonly?: boolean;\n}\n\n@Component({\n  selector: 'in-select-button',\n  templateUrl: './select-button.component.html',\n  styleUrls: ['./select-button.component.scss'],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => SelectButtonComponent),\n      multi: true,\n    },\n  ],\n})\nexport class SelectButtonComponent\n  implements ControlValueAccessor, OnInit, OnChanges {\n\n  @Input() name = '';\n  @Input() testId = '';\n  @Input() label = '';\n  @Input() icon = '';\n  @Input() error = '';\n  @Input() iconPos: IconPos = 'left';\n  @Input() options: SelectOption[] = [];\n  @Input() disabled = false;\n  @Input() required = false;\n  @Input() multiple = false;\n  @Input() initial?: Array<string | number | boolean>;\n\n  @Output() focus = new EventEmitter<any>();\n  @Output() blur = new EventEmitter<any>();\n  @Output() click = new EventEmitter<any>();\n  @Output() change = new EventEmitter<any>();\n\n  // хранит выбранные ЗНАЧЕНИЯ, а не объекты\n  selected: Array<string | number | boolean> = [];\n\n  // то, что отдаем наружу в форму\n  value: string | number | boolean | Array<string | number | boolean> | null = null;\n\n  private propagateChange: (val: any) => void = () => {};\n  private propagateTouched: () => void = () => {};\n\n  constructor(private cdRef: ChangeDetectorRef) {}\n\n  ngOnInit(): void {}\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // применяем initial только когда он реально поменялся\n    if (changes['initial'] && this.initial) {\n      this.selected = [...this.initial];\n      this.value = this.multiple ? [...this.initial] : this.initial[0] ?? null;\n      this.cdRef.detectChanges();\n    }\n\n    // если меняется список опций, надо попытаться восстановить выделение\n    if (changes['options'] && this.options && this.options.length > 0) {\n      this.syncSelectedWithOptions();\n    }\n  }\n\n  clickButton(option: SelectOption): void {\n    if (option.readonly || this.disabled) {\n      return;\n    }\n\n    this.click.emit(option);\n\n    if (this.multiple) {\n      if (this.selected.includes(option.value)) {\n        // убираем\n        this.selected = this.selected.filter(v => v !== option.value);\n      } else {\n        // добавляем\n        this.selected = [...this.selected, option.value];\n      }\n      this.value = [...this.selected];\n    } else {\n      this.selected = [option.value];\n      this.value = option.value;\n    }\n\n    this.cdRef.detectChanges();\n    this.onChange();\n  }\n\n  // ControlValueAccessor: model -> view\n  writeValue(value: any): void {\n    this.value = value;\n\n    if (value === null || value === undefined) {\n      this.selected = [];\n    } else if (Array.isArray(value)) {\n      // multiple\n      this.selected = [...value];\n    } else {\n      // single\n      this.selected = [value];\n    }\n\n    // когда значение пришло извне, надо убедиться что такие опции есть\n    this.syncSelectedWithOptions();\n    this.cdRef.detectChanges();\n  }\n\n  // ControlValueAccessor\n  registerOnChange(fn: any): void {\n    this.propagateChange = fn;\n  }\n\n  // ControlValueAccessor\n  registerOnTouched(fn: any): void {\n    this.propagateTouched = fn;\n  }\n\n  // ControlValueAccessor: disabled\n  setDisabledState(isDisabled: boolean): void {\n    this.disabled = isDisabled;\n    this.cdRef.detectChanges();\n  }\n\n  // пробросить наружу\n  private onChange(): void {\n    this.propagateChange(this.value);\n    this.change.emit(this.value);\n  }\n\n  onBlur(): void {\n    this.propagateTouched();\n    this.blur.emit();\n  }\n\n  onFocus(): void {\n    this.focus.emit();\n  }\n\n  // синхронизируем текущее выбранное со свежими options\n  private syncSelectedWithOptions(): void {\n    if (!this.options || this.options.length === 0) {\n      return;\n    }\n\n    if (this.multiple) {\n      // оставляем только те выбранные, которые реально есть в новых options\n      const optionValues = this.options.map(o => o.value);\n      this.selected = this.selected.filter(v => optionValues.includes(v));\n      this.value = [...this.selected];\n    } else {\n      if (this.selected.length > 0) {\n        const current = this.selected[0];\n        const found = this.options.find(o => o.value === current);\n        if (!found) {\n          // выбранного больше нет в списке\n          this.selected = [];\n          this.value = null;\n        } else {\n          this.selected = [found.value];\n          this.value = found.value;\n        }\n      }\n    }\n  }\n\n  // для *ngFor trackBy\n  trackByValue(index: number, item: SelectOption): string | number | boolean {\n    return item.value;\n  }\n}\n","<span *ngIf=\"label\" class=\"c-label__content\">\n  {{ label }}\n  <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n  <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{ error }}</span>\n</span>\n\n<div class=\"container\">\n  <!-- фиктивная кнопка если нужна разметка -->\n  <in-button class=\"button-none\"></in-button>\n\n  <in-button\n    *ngFor=\"let option of options; trackBy: trackByValue\"\n    (click)=\"clickButton(option)\"\n    [color]=\"selected.includes(option.value) ? 'primary' : 'normal'\"\n    [icon]=\"option.icon\"\n    [label]=\"option.label | translate\"\n    [testId]=\"'filter-buttons-' + option.value\"\n    [title]=\"option.title ? (option.title | translate) : ''\"\n    [name]=\"option.label\"\n    [disabled]=\"option.readonly || disabled\"\n  >\n  </in-button>\n</div>\n"]}
@@ -2102,38 +2102,49 @@ class SelectButtonComponent {
2102
2102
  this.error = '';
2103
2103
  this.iconPos = 'left';
2104
2104
  this.options = [];
2105
+ this.disabled = false;
2105
2106
  this.required = false;
2106
2107
  this.multiple = false;
2107
2108
  this.focus = new EventEmitter();
2108
2109
  this.blur = new EventEmitter();
2109
2110
  this.click = new EventEmitter();
2110
2111
  this.change = new EventEmitter();
2112
+ // хранит выбранные ЗНАЧЕНИЯ, а не объекты
2111
2113
  this.selected = [];
2112
- this.propagateChange = (_) => {
2113
- };
2114
- }
2115
- ngOnInit() {
2114
+ // то, что отдаем наружу в форму
2115
+ this.value = null;
2116
+ this.propagateChange = () => { };
2117
+ this.propagateTouched = () => { };
2116
2118
  }
2117
- ngOnChanges() {
2118
- if (this.initial) {
2119
- this.selected = this.initial;
2119
+ ngOnInit() { }
2120
+ ngOnChanges(changes) {
2121
+ var _a;
2122
+ // применяем initial только когда он реально поменялся
2123
+ if (changes['initial'] && this.initial) {
2124
+ this.selected = [...this.initial];
2125
+ this.value = this.multiple ? [...this.initial] : (_a = this.initial[0]) !== null && _a !== void 0 ? _a : null;
2126
+ this.cdRef.detectChanges();
2120
2127
  }
2121
- else {
2122
- this.selected = [];
2128
+ // если меняется список опций, надо попытаться восстановить выделение
2129
+ if (changes['options'] && this.options && this.options.length > 0) {
2130
+ this.syncSelectedWithOptions();
2123
2131
  }
2124
2132
  }
2125
2133
  clickButton(option) {
2126
- if (option.readonly) {
2127
- return false;
2134
+ if (option.readonly || this.disabled) {
2135
+ return;
2128
2136
  }
2137
+ this.click.emit(option);
2129
2138
  if (this.multiple) {
2130
- if (this.selected.indexOf(option.value) !== -1) {
2131
- this.selected = this.selected.filter(item => item !== option.value);
2139
+ if (this.selected.includes(option.value)) {
2140
+ // убираем
2141
+ this.selected = this.selected.filter(v => v !== option.value);
2132
2142
  }
2133
2143
  else {
2134
- this.selected.push(option.value);
2144
+ // добавляем
2145
+ this.selected = [...this.selected, option.value];
2135
2146
  }
2136
- this.value = this.options.filter(item => this.selected.indexOf(item.value) !== -1).map(val => val.value);
2147
+ this.value = [...this.selected];
2137
2148
  }
2138
2149
  else {
2139
2150
  this.selected = [option.value];
@@ -2142,46 +2153,79 @@ class SelectButtonComponent {
2142
2153
  this.cdRef.detectChanges();
2143
2154
  this.onChange();
2144
2155
  }
2145
- /**
2146
- * Write form value to the DOM element (model => view)
2147
- */
2156
+ // ControlValueAccessor: model -> view
2148
2157
  writeValue(value) {
2149
2158
  this.value = value;
2150
- if (value !== null && value !== undefined) {
2151
- if (Array.isArray(value)) {
2152
- this.selected = value.map(val => val);
2153
- }
2154
- else {
2155
- this.selected = [value];
2156
- }
2159
+ if (value === null || value === undefined) {
2160
+ this.selected = [];
2157
2161
  }
2162
+ else if (Array.isArray(value)) {
2163
+ // multiple
2164
+ this.selected = [...value];
2165
+ }
2166
+ else {
2167
+ // single
2168
+ this.selected = [value];
2169
+ }
2170
+ // когда значение пришло извне, надо убедиться что такие опции есть
2171
+ this.syncSelectedWithOptions();
2158
2172
  this.cdRef.detectChanges();
2159
2173
  }
2160
- /**
2161
- * Write form disabled state to the DOM element (model => view)
2162
- */
2163
- setDisabledState(isDisabled) {
2164
- this.disabled = isDisabled;
2165
- }
2166
- /**
2167
- * Update form when DOM element value changes (view => model)
2168
- */
2174
+ // ControlValueAccessor
2169
2175
  registerOnChange(fn) {
2170
- // Store the provided function as an internal method.
2171
2176
  this.propagateChange = fn;
2172
2177
  }
2173
- /**
2174
- * Update form when DOM element is blurred (view => model)
2175
- */
2178
+ // ControlValueAccessor
2176
2179
  registerOnTouched(fn) {
2177
- // Store the provided function as an internal method.
2178
- this.onTouched = fn;
2180
+ this.propagateTouched = fn;
2181
+ }
2182
+ // ControlValueAccessor: disabled
2183
+ setDisabledState(isDisabled) {
2184
+ this.disabled = isDisabled;
2185
+ this.cdRef.detectChanges();
2179
2186
  }
2187
+ // пробросить наружу
2180
2188
  onChange() {
2181
2189
  this.propagateChange(this.value);
2182
2190
  this.change.emit(this.value);
2183
2191
  }
2184
- onTouched() {
2192
+ onBlur() {
2193
+ this.propagateTouched();
2194
+ this.blur.emit();
2195
+ }
2196
+ onFocus() {
2197
+ this.focus.emit();
2198
+ }
2199
+ // синхронизируем текущее выбранное со свежими options
2200
+ syncSelectedWithOptions() {
2201
+ if (!this.options || this.options.length === 0) {
2202
+ return;
2203
+ }
2204
+ if (this.multiple) {
2205
+ // оставляем только те выбранные, которые реально есть в новых options
2206
+ const optionValues = this.options.map(o => o.value);
2207
+ this.selected = this.selected.filter(v => optionValues.includes(v));
2208
+ this.value = [...this.selected];
2209
+ }
2210
+ else {
2211
+ if (this.selected.length > 0) {
2212
+ const current = this.selected[0];
2213
+ const found = this.options.find(o => o.value === current);
2214
+ if (!found) {
2215
+ // выбранного больше нет в списке
2216
+ this.selected = [];
2217
+ this.value = null;
2218
+ }
2219
+ else {
2220
+ this.selected = [found.value];
2221
+ this.value = found.value;
2222
+ }
2223
+ }
2224
+ }
2225
+ }
2226
+ // для *ngFor trackBy
2227
+ trackByValue(index, item) {
2228
+ return item.value;
2185
2229
  }
2186
2230
  }
2187
2231
  SelectButtonComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectButtonComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
@@ -2190,8 +2234,8 @@ SelectButtonComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0",
2190
2234
  provide: NG_VALUE_ACCESSOR,
2191
2235
  useExisting: forwardRef(() => SelectButtonComponent),
2192
2236
  multi: true,
2193
- }
2194
- ], usesOnChanges: true, ngImport: i0, template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{label}}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{error}}</span>\n </span>\n<div class=\"container\">\n <in-button class=\"button-none\"></in-button>\n <in-button (click)=\"clickButton(option)\"\n *ngFor=\"let option of options\"\n [color]=\"selected.indexOf(option.value) === -1 ? 'normal': 'primary' \"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name] = option.label\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"], dependencies: [{ kind: "directive", type: i1$2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1$2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "in-button", inputs: ["size", "color", "type", "name", "testId", "label", "icon", "className", "notify", "svg", "width", "link", "isLoading", "shadow", "isIconOnly", "linkType", "linkTarget", "pressed", "disabled", "iconPos", "selectedItem", "menuPlacement", "menuItems"], outputs: ["focus", "blur", "click"] }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }] });
2237
+ },
2238
+ ], usesOnChanges: true, ngImport: i0, template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{ label }}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{ error }}</span>\n</span>\n\n<div class=\"container\">\n <!-- \u0444\u0438\u043A\u0442\u0438\u0432\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430 \u0435\u0441\u043B\u0438 \u043D\u0443\u0436\u043D\u0430 \u0440\u0430\u0437\u043C\u0435\u0442\u043A\u0430 -->\n <in-button class=\"button-none\"></in-button>\n\n <in-button\n *ngFor=\"let option of options; trackBy: trackByValue\"\n (click)=\"clickButton(option)\"\n [color]=\"selected.includes(option.value) ? 'primary' : 'normal'\"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name]=\"option.label\"\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"], dependencies: [{ kind: "directive", type: i1$2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1$2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: ButtonComponent, selector: "in-button", inputs: ["size", "color", "type", "name", "testId", "label", "icon", "className", "notify", "svg", "width", "link", "isLoading", "shadow", "isIconOnly", "linkType", "linkTarget", "pressed", "disabled", "iconPos", "selectedItem", "menuPlacement", "menuItems"], outputs: ["focus", "blur", "click"] }, { kind: "pipe", type: i1.TranslatePipe, name: "translate" }] });
2195
2239
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImport: i0, type: SelectButtonComponent, decorators: [{
2196
2240
  type: Component,
2197
2241
  args: [{ selector: 'in-select-button', providers: [
@@ -2199,8 +2243,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "14.3.0", ngImpor
2199
2243
  provide: NG_VALUE_ACCESSOR,
2200
2244
  useExisting: forwardRef(() => SelectButtonComponent),
2201
2245
  multi: true,
2202
- }
2203
- ], template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{label}}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{error}}</span>\n </span>\n<div class=\"container\">\n <in-button class=\"button-none\"></in-button>\n <in-button (click)=\"clickButton(option)\"\n *ngFor=\"let option of options\"\n [color]=\"selected.indexOf(option.value) === -1 ? 'normal': 'primary' \"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name] = option.label\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"] }]
2246
+ },
2247
+ ], template: "<span *ngIf=\"label\" class=\"c-label__content\">\n {{ label }}\n <span *ngIf=\"required\" class=\"c-label__req\">*</span>\n <span *ngIf=\"error\" class=\"c-label__sub_is-error\">{{ error }}</span>\n</span>\n\n<div class=\"container\">\n <!-- \u0444\u0438\u043A\u0442\u0438\u0432\u043D\u0430\u044F \u043A\u043D\u043E\u043F\u043A\u0430 \u0435\u0441\u043B\u0438 \u043D\u0443\u0436\u043D\u0430 \u0440\u0430\u0437\u043C\u0435\u0442\u043A\u0430 -->\n <in-button class=\"button-none\"></in-button>\n\n <in-button\n *ngFor=\"let option of options; trackBy: trackByValue\"\n (click)=\"clickButton(option)\"\n [color]=\"selected.includes(option.value) ? 'primary' : 'normal'\"\n [icon]=\"option.icon\"\n [label]=\"option.label | translate\"\n [testId]=\"'filter-buttons-' + option.value\"\n [title]=\"option.title ? (option.title | translate) : ''\"\n [name]=\"option.label\"\n [disabled]=\"option.readonly || disabled\"\n >\n </in-button>\n</div>\n", styles: [".container{white-space:nowrap}in-button ::ng-deep button{outline:none}in-button:first-child ::ng-deep button{border-bottom-right-radius:0;border-top-right-radius:0;border-right:1px solid #000}in-button:not(:first-child):not(:last-child) ::ng-deep button{border-radius:0;border-right:1px solid #000}in-button:last-child ::ng-deep button{border-bottom-left-radius:0;border-top-left-radius:0}.button-none{display:none}\n"] }]
2204
2248
  }], ctorParameters: function () { return [{ type: i0.ChangeDetectorRef }]; }, propDecorators: { name: [{
2205
2249
  type: Input
2206
2250
  }], testId: [{