UNPKG

@true-directive/grid

Version:

Angular Data Grid from Yopsilon.

434 lines (429 loc) 47.5 kB
import * as tslib_1 from "tslib"; var SelectComponent_1; /** * Copyright (c) 2018-2019 Aleksey Melnikov, True Directive Company. * @link https://truedirective.com/ * @license MIT */ import { Component, Input, Output, EventEmitter, Renderer2, ElementRef, ViewChild, forwardRef } from '@angular/core'; import { NG_VALUE_ACCESSOR } from '@angular/forms'; import { DropdownBaseComponent } from './dropdown-base.component'; import { GridSettings, GridState } from '@true-directive/base'; import { GridComponent } from '../grid.component'; import { Keys, Utils } from '@true-directive/base'; let SelectComponent = SelectComponent_1 = class SelectComponent extends DropdownBaseComponent { constructor(_elementRef, _renderer) { super(_elementRef, _renderer); this._elementRef = _elementRef; this._renderer = _renderer; /** * Current text value */ this.displayValue = ''; // Запрос данных this.dataQuery = new EventEmitter(); this._focusedValue = null; this._valueField = null; this._displayField = null; this.columns = []; this.parentState = null; this._settings = null; this.itemSelect = new EventEmitter(); // Выбираем ли мы из грида this._onGrid = false; this._empty = false; this._updating = false; this.useAltDown = false; } /** * Current value */ set value(v) { if (v !== this._value) { this._value = v; this.writeValue(this._value); this.onChange(v); } } get value() { return this._value; } set valueField(fieldName) { this._valueField = fieldName; if (this._settings) { // Настройки могут быть еще не созданы this._settings.keyField = this._valueField; } } get valueField() { if (this._valueField === null && this.columns.length > 0) { return this.columns[0].fieldName; } return this._valueField === null ? 'id' : this._valueField; } set displayField(fieldName) { this._displayField = fieldName; } get displayField() { if (this._displayField === null && this.columns.length > 0) { return this.columns[0].fieldName; } return this._displayField === null ? 'name' : this._displayField; } set settings(v) { this._settings = v; this._settings.enableTouchScroll = false; } get settings() { if (!this._settings) { this._settings = GridSettings.minimal(); this._settings.keyField = this.valueField; this._settings.widthUnit = 'em'; this._settings.showHeader = false; this._settings.columnAutoWidth = false; this._settings.enableTouchScroll = false; this._settings.appearance.enableHoverAppearance = true; } return this._settings; } gridStartProcess(e) { if (this._empty && this.settings.requestData) { this._empty = false; this._updating = true; } } gridEndProcess(e) { setTimeout(() => { if (!this.popup.visible) { this.showByTarget(); } else { // Let's update position of popup window. When list reduce, it may move // up or down regarding input. this.popup.updatePosition(); } }); this._updating = false; if (this.grid.resultRows.length === 0) { this._empty = true; } else { this._empty = false; } } gridDataQuery(e) { this.dataQuery.emit(e); } // Отображаем значение в компоненте. Formatter: Ctrl --> View writeValue(v) { this._value = v; let found = this.items.find(item => item[this.valueField] === v); if (!found) { this.displayValue = ''; } else { this.displayValue = found[this.displayField]; } } // За один раз устанавливаем и значение и отображение setValue(value, displayValue) { if (value !== this._value || displayValue !== this.displayValue) { this._focusedValue = value; this.displayValue = displayValue; this.value = value; } } // По введенному тексту ничего не найдено? empty() { return this._empty; } // Происходит обновление данных? updating() { return this._updating; } // Ответ fetchData(dataQuery, data) { let sel = this.getSelectedRow(); if (sel) { this._focusedValue = sel[this.valueField]; } if (this.grid && this.popupVisible) { this.grid.fetchData(dataQuery, data); if (this._focusedValue) { this.grid.locateByKey(this._focusedValue); } else { if (this._value !== null) { this.grid.locateByKey(this._value); } } } } /** * Общая ширина грида в выпадающем списке * @return Ширина в заданных единицах измерения */ gridWidth() { let ww = 0; this.columns.forEach(c => ww += c.width); return ww + this.settings.widthUnit; } /** * Перекрываем показывание выпадающего окна, чтобы обновить вью грида * и выделить строку, содержащую текущее значение * @param e Параметры события */ popupShow(e) { this._onGrid = false; if (this._value !== null) { this.grid.locateByKey(this._value, this.valueField); } else { if (this.shownByKey) { this.grid.processKey(Keys.generateEvent(null, Keys.DOWN)); // Будет выбрана первая запись } } this.grid.checkSize(); } /** * Обработчик клавиш вызывается из родительского класса * Возвращает false, если клавиша не обработана */ processKey(e) { if (e.keyCode === Keys.ENTER) { this.setValueFromGrid(); this.popup.closePopup(); return true; } if (!this.popupVisible) { return false; } if (e.keyCode === Keys.ESCAPE) { this.popup.closePopup(); e.preventDefault(); e.stopPropagation(); return true; } if (e.keyCode === Keys.DOWN || e.keyCode === Keys.UP) { this._onGrid = true; this.grid.processKey(e); e.preventDefault(); return true; } return false; } // Клик по записи в гриде gridRowClick(e) { this._onGrid = true; this._skipFocusOnPopupClose = false; this.value = e.row[this.valueField]; this.popup.closePopup(); this.itemSelect.emit(e.row); } // Поворот иконки iconClass() { let s = 'true-icon-angle-down true-turnable'; if (this.popupVisible) { s += ' true-turned'; } return s; } // Ввод текста пользователем inputInput(e = null) { // Окошко еще на выпало? показываем if (!this.popupVisible && this.displayValue !== '') { this.grid.searchString = this.displayValue; return; } if (this.popupVisible) { setTimeout(() => { // Устанавливаем фильтр по тексту this.grid.searchString = this.displayValue; }); } if (this.displayValue === '') { this.setValue(null, ''); return; } this._value = null; } // Возврат выбранной строки в гриде getSelectedRow() { if (this.grid && this.grid.state.selection.ranges.length > 0) { let item = this.grid.state.selection.ranges[0].fromCell.row; return item; } return null; } containsDisplayed(s) { return !this.displayValue || (s !== undefined && s.toLowerCase().indexOf(this.displayValue.toLowerCase()) >= 0); } // Установка значения по выделенной в гриде записи setValueFromGrid() { if (!this.popupVisible && this.grid.state.model) { const found = this.grid.state.model.find(item => item[this.displayField].toLowerCase() === this.displayValue.toLowerCase()); if (found) { this.setValue(found[this.valueField], found[this.displayField]); } else { this.setValue(null, ''); } return; } const sel = this.getSelectedRow(); if (sel && (this._onGrid || this.containsDisplayed(sel[this.displayField]))) { this.setValue(sel[this.valueField], sel[this.displayField]); } else { this.setValueFromDisplayed(); } } /** * Установка значения по введенному тексту */ setValueFromDisplayed() { // По запросу. Смотрим в гриде. Хотя, если подумать, можно всегда в гриде смотреть.. let f = null; if (this.grid && this.grid.resultRows) { f = this.grid.resultRows.find(item => item[this.displayField].toLowerCase() === this.displayValue.toLowerCase()); } if (f) { this.setValue(f[this.valueField], f[this.displayField]); } else { this.setValue(null, ''); } return; } /** * Потеря фокуса инпутом * @param e Параметры события */ inputBlur(e) { this._skipFocusOnPopupClose = true; if ((this.value === null || this.value === undefined) && this.displayValue !== '') { if (e.relatedTarget && Utils.isAncestor(this.popup.elementRef.nativeElement, e.relatedTarget)) { // Если фокус перешел на... return; } this.closePopup(); this.setValueFromDisplayed(); } super.inputBlur(e); } }; tslib_1.__decorate([ Output('dataQuery'), tslib_1.__metadata("design:type", EventEmitter) ], SelectComponent.prototype, "dataQuery", void 0); tslib_1.__decorate([ Input('valueField'), tslib_1.__metadata("design:type", String), tslib_1.__metadata("design:paramtypes", [String]) ], SelectComponent.prototype, "valueField", null); tslib_1.__decorate([ Input('displayField'), tslib_1.__metadata("design:type", String), tslib_1.__metadata("design:paramtypes", [String]) ], SelectComponent.prototype, "displayField", null); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Array) ], SelectComponent.prototype, "columns", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", Array) ], SelectComponent.prototype, "items", void 0); tslib_1.__decorate([ Input(), tslib_1.__metadata("design:type", GridState) ], SelectComponent.prototype, "parentState", void 0); tslib_1.__decorate([ Input('settings'), tslib_1.__metadata("design:type", Object), tslib_1.__metadata("design:paramtypes", [Object]) ], SelectComponent.prototype, "settings", null); tslib_1.__decorate([ Output('itemSelect'), tslib_1.__metadata("design:type", EventEmitter) ], SelectComponent.prototype, "itemSelect", void 0); tslib_1.__decorate([ ViewChild('grid', { static: true }), tslib_1.__metadata("design:type", GridComponent) ], SelectComponent.prototype, "grid", void 0); SelectComponent = SelectComponent_1 = tslib_1.__decorate([ Component({ selector: 'true-select', template: ` <true-input-wrapper (btnClick)="btnClick($event)" [disabled]="disabled" class="true-select__input" [class.true-input_popup-visible]="popupVisible" [icon]="iconClass()"> <input #input [(ngModel)]="displayValue" [readonly]="disableTextEditor" [attr.disabled]="disabled" [class.true-disable-te]="disableTextEditor" (input)="inputInput($event)" (click)="inputClick($event)" (touchstart)="inputTouchStart($event)" (touchmove)="inputTouchMove($event)" (touchend)="inputTouchEnd($event)" (blur)="inputBlur($event)" (focus)="inputFocus($event)" (keydown)="inputKeyDown($event)"/> </true-input-wrapper> <true-popup #popup (close)="popupClose($event)" (show)="popupShow($event)" > <true-grid #grid [class.true-select__grid_hidden]="empty()" class="true-select__grid" [maxHeight]="maxDropDownHeight" [columns]="columns" [data]="items" [settings]="settings" (dataQuery)="gridDataQuery($event)" (startProcess)="gridStartProcess($event)" (endProcess)="gridEndProcess($event)" (rowClick)="gridRowClick($event)"> </true-grid> <div *ngIf="empty()" class="true-select__notfound" [style.width]="gridWidth()">Ничего не найдено</div> <div *ngIf="updating()" class="true-select__updating">Загрузка...</div> </true-popup> `, providers: [{ provide: NG_VALUE_ACCESSOR, useExisting: forwardRef(() => SelectComponent_1), multi: true }], styles: [` .true-select__input { width: 100%; } input { box-sizing: border-box; width: 100%; height: 100%; } .true-select__grid_hidden { display: none; } .true-select__grid { border: none !important; cursor: pointer; } .true-select__notfound, .true-select__updating { padding: 0.6em; box-sizing: border-box; color: #aaa; } `] }), tslib_1.__metadata("design:paramtypes", [ElementRef, Renderer2]) ], SelectComponent); export { SelectComponent }; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select.component.js","sourceRoot":"ng://@true-directive/grid/","sources":["src/controls/select.component.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Copyright (c) 2018-2019 Aleksey Melnikov, True Directive Company.\r\n * @link https://truedirective.com/\r\n * @license MIT\r\n*/\r\nimport { Component, Input, Output, HostBinding, EventEmitter, Renderer2, ElementRef,\r\n         ChangeDetectorRef,\r\n         ViewChild, forwardRef } from '@angular/core';\r\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';\r\n\r\nimport { PopupComponent } from './popup.component';\r\nimport { DropdownBaseComponent } from './dropdown-base.component';\r\n\r\nimport { GridSettings, GridState } from '@true-directive/base';\r\nimport { Column, DataQuery } from '@true-directive/base';\r\nimport { GridComponent } from '../grid.component';\r\nimport { Keys, Utils } from '@true-directive/base';\r\n\r\n@Component({\r\n  selector: 'true-select',\r\n  template:`\r\n      <true-input-wrapper\r\n        (btnClick)=\"btnClick($event)\"\r\n        [disabled]=\"disabled\"\r\n        class=\"true-select__input\"\r\n        [class.true-input_popup-visible]=\"popupVisible\"\r\n        [icon]=\"iconClass()\">\r\n        <input #input\r\n               [(ngModel)]=\"displayValue\"\r\n               [readonly]=\"disableTextEditor\"\r\n               [attr.disabled]=\"disabled\"\r\n               [class.true-disable-te]=\"disableTextEditor\"\r\n               (input)=\"inputInput($event)\"\r\n               (click)=\"inputClick($event)\"\r\n               (touchstart)=\"inputTouchStart($event)\"\r\n               (touchmove)=\"inputTouchMove($event)\"\r\n               (touchend)=\"inputTouchEnd($event)\"\r\n               (blur)=\"inputBlur($event)\"\r\n               (focus)=\"inputFocus($event)\"\r\n               (keydown)=\"inputKeyDown($event)\"/>\r\n      </true-input-wrapper>\r\n      <true-popup #popup\r\n        (close)=\"popupClose($event)\"\r\n        (show)=\"popupShow($event)\" >\r\n        <true-grid #grid\r\n                 [class.true-select__grid_hidden]=\"empty()\"\r\n                 class=\"true-select__grid\"\r\n                 [maxHeight]=\"maxDropDownHeight\"\r\n                 [columns]=\"columns\"\r\n                 [data]=\"items\"\r\n                 [settings]=\"settings\"\r\n                 (dataQuery)=\"gridDataQuery($event)\"\r\n                 (startProcess)=\"gridStartProcess($event)\"\r\n                 (endProcess)=\"gridEndProcess($event)\"\r\n                 (rowClick)=\"gridRowClick($event)\">\r\n        </true-grid>\r\n        <div *ngIf=\"empty()\" class=\"true-select__notfound\" [style.width]=\"gridWidth()\">Ничего не найдено</div>\r\n        <div *ngIf=\"updating()\" class=\"true-select__updating\">Загрузка...</div>\r\n      </true-popup>\r\n    `,\r\n    /*\r\n      Removed: :host { display: inline-block; } - invalid ios rendering.\r\n     */\r\n  styles: [`\r\n\r\n    .true-select__input {\r\n      width: 100%;\r\n    }\r\n\r\n    input {\r\n      box-sizing: border-box;\r\n      width: 100%;\r\n      height: 100%;\r\n    }\r\n\r\n    .true-select__grid_hidden {\r\n      display: none;\r\n    }\r\n\r\n    .true-select__grid {\r\n      border: none !important;\r\n      cursor: pointer;\r\n    }\r\n\r\n    .true-select__notfound, .true-select__updating {\r\n      padding: 0.6em;\r\n      box-sizing: border-box;\r\n      color: #aaa;\r\n    }\r\n  `],\r\n  providers: [{\r\n      provide: NG_VALUE_ACCESSOR,\r\n      useExisting: forwardRef(() => SelectComponent),\r\n      multi: true}]\r\n  })\r\nexport class SelectComponent extends DropdownBaseComponent {\r\n\r\n  /**\r\n   * Current text value\r\n   */\r\n  public displayValue: string = '';\r\n\r\n  /**\r\n   * Current value\r\n   */\r\n  public set value(v: any) {\r\n    if (v !== this._value) {\r\n      this._value = v;\r\n      this.writeValue(this._value);\r\n      this.onChange(v);\r\n    }\r\n  }\r\n\r\n  public get value() {\r\n    return this._value;\r\n  }\r\n\r\n  // Запрос данных\r\n  @Output('dataQuery')\r\n  dataQuery: EventEmitter<any> = new EventEmitter<any>();\r\n\r\n  private _focusedValue: any = null;\r\n  private _valueField: string = null;\r\n\r\n  @Input('valueField')\r\n  public set valueField(fieldName: string) {\r\n    this._valueField = fieldName;\r\n    if (this._settings) {\r\n      // Настройки могут быть еще не созданы\r\n      this._settings.keyField = this._valueField;\r\n    }\r\n  }\r\n\r\n  public get valueField() {\r\n    if (this._valueField === null && this.columns.length > 0) {\r\n      return this.columns[0].fieldName;\r\n    }\r\n    return this._valueField === null ? 'id' : this._valueField;\r\n  }\r\n\r\n  private _displayField: string = null;\r\n  @Input('displayField')\r\n  public set displayField(fieldName: string) {\r\n    this._displayField = fieldName;\r\n  }\r\n\r\n  public get displayField() {\r\n    if (this._displayField === null && this.columns.length > 0) {\r\n      return this.columns[0].fieldName;\r\n    }\r\n    return this._displayField === null ? 'name' : this._displayField;\r\n  }\r\n\r\n  @Input()\r\n  columns: Array<Column> = [];\r\n\r\n  @Input()\r\n  items: Array<any>;\r\n\r\n  @Input()\r\n  parentState: GridState = null;\r\n\r\n  private _settings: GridSettings = null;\r\n\r\n  @Input('settings')\r\n  set settings(v) {\r\n    this._settings = v;\r\n    this._settings.enableTouchScroll = false;\r\n  }\r\n\r\n  get settings() {\r\n    if (!this._settings) {\r\n      this._settings = GridSettings.minimal();\r\n      this._settings.keyField = this.valueField;\r\n      this._settings.widthUnit = 'em';\r\n      this._settings.showHeader = false;\r\n      this._settings.columnAutoWidth = false;\r\n      this._settings.enableTouchScroll = false;\r\n      this._settings.appearance.enableHoverAppearance = true;\r\n    }\r\n    return this._settings;\r\n  }\r\n\r\n  @Output('itemSelect')\r\n  itemSelect: EventEmitter<any> = new EventEmitter();\r\n\r\n  @ViewChild('grid', {static: true})\r\n  grid: GridComponent;\r\n\r\n  // Выбираем ли мы из грида\r\n  _onGrid: boolean = false;\r\n  _empty = false;\r\n\r\n  gridStartProcess(e: any) {\r\n    if (this._empty && this.settings.requestData) {\r\n      this._empty = false;\r\n      this._updating = true;\r\n    }\r\n  }\r\n\r\n  _updating = false;\r\n  gridEndProcess(e: any) {\r\n    setTimeout(() => {\r\n      if (!this.popup.visible) {\r\n        this.showByTarget();\r\n      } else {\r\n        // Let's update position of popup window. When list reduce, it may move\r\n        // up or down regarding input.\r\n        this.popup.updatePosition();\r\n      }\r\n    });\r\n\r\n    this._updating = false;\r\n    if (this.grid.resultRows.length === 0) {\r\n      this._empty = true;\r\n    } else {\r\n      this._empty = false;\r\n    }\r\n  }\r\n\r\n  gridDataQuery(e: any) {\r\n    this.dataQuery.emit(e);\r\n  }\r\n\r\n  // Отображаем значение в компоненте. Formatter: Ctrl --> View\r\n  writeValue(v: any) {\r\n    this._value = v;\r\n    let found = this.items.find(item => item[this.valueField] === v);\r\n    if (!found) {\r\n      this.displayValue = '';\r\n    } else {\r\n      this.displayValue = found[this.displayField];\r\n    }\r\n  }\r\n\r\n  // За один раз устанавливаем и значение и отображение\r\n  setValue(value: any, displayValue: string) {\r\n    if (value !== this._value || displayValue !== this.displayValue) {\r\n      this._focusedValue = value;\r\n      this.displayValue = displayValue;\r\n      this.value = value;\r\n    }\r\n  }\r\n\r\n  // По введенному тексту ничего не найдено?\r\n  empty(): boolean {\r\n    return this._empty;\r\n  }\r\n\r\n  // Происходит обновление данных?\r\n  updating(): boolean {\r\n    return this._updating;\r\n  }\r\n\r\n  // Ответ\r\n  fetchData(dataQuery: DataQuery, data: Array<any>) {\r\n    let sel = this.getSelectedRow();\r\n    if (sel) {\r\n      this._focusedValue = sel[this.valueField];\r\n    }\r\n\r\n    if (this.grid && this.popupVisible) {\r\n      this.grid.fetchData(dataQuery, data);\r\n\r\n      if (this._focusedValue) {\r\n        this.grid.locateByKey(this._focusedValue);\r\n      } else {\r\n        if (this._value !== null) {\r\n          this.grid.locateByKey(this._value);\r\n        }\r\n      }\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Общая ширина грида в выпадающем списке\r\n   * @return Ширина в заданных единицах измерения\r\n   */\r\n  gridWidth() {\r\n    let ww = 0;\r\n    this.columns.forEach(c => ww += c.width);\r\n    return ww + this.settings.widthUnit;\r\n  }\r\n\r\n  /**\r\n   * Перекрываем показывание выпадающего окна, чтобы обновить вью грида\r\n   * и выделить строку, содержащую текущее значение\r\n   * @param  e Параметры события\r\n   */\r\n  popupShow(e: any) {\r\n    this._onGrid = false;\r\n\r\n    if (this._value !== null) {\r\n        this.grid.locateByKey(this._value, this.valueField);\r\n    } else {\r\n      if (this.shownByKey) {\r\n        this.grid.processKey(Keys.generateEvent(null, Keys.DOWN)); // Будет выбрана первая запись\r\n      }\r\n    }\r\n    this.grid.checkSize();\r\n  }\r\n\r\n  /**\r\n   * Обработчик клавиш вызывается из родительского класса\r\n   * Возвращает false, если клавиша не обработана\r\n   */\r\n  processKey(e: any): boolean {\r\n\r\n    if (e.keyCode === Keys.ENTER) {\r\n      this.setValueFromGrid();\r\n      this.popup.closePopup();\r\n      return true;\r\n    }\r\n\r\n    if (!this.popupVisible) {\r\n      return false;\r\n    }\r\n\r\n    if (e.keyCode === Keys.ESCAPE) {\r\n      this.popup.closePopup();\r\n      e.preventDefault();\r\n      e.stopPropagation();\r\n      return true;\r\n    }\r\n\r\n    if (e.keyCode === Keys.DOWN || e.keyCode === Keys.UP) {\r\n      this._onGrid = true;\r\n      this.grid.processKey(e);\r\n      e.preventDefault();\r\n      return true;\r\n    }\r\n\r\n    return false;\r\n  }\r\n\r\n  // Клик по записи в гриде\r\n  public gridRowClick(e: any) {\r\n    this._onGrid = true;\r\n    this._skipFocusOnPopupClose = false;\r\n    this.value = e.row[this.valueField];\r\n    this.popup.closePopup();\r\n    this.itemSelect.emit(e.row);\r\n  }\r\n\r\n  // Поворот иконки\r\n  public iconClass() {\r\n    let s = 'true-icon-angle-down true-turnable';\r\n    if (this.popupVisible) {\r\n      s += ' true-turned';\r\n    }\r\n    return s;\r\n  }\r\n\r\n  // Ввод текста пользователем\r\n  public inputInput(e: any = null) {\r\n    // Окошко еще на выпало? показываем\r\n    if (!this.popupVisible && this.displayValue !== '') {\r\n      this.grid.searchString = this.displayValue;\r\n      return;\r\n    }\r\n\r\n    if (this.popupVisible) {\r\n      setTimeout(() => {\r\n        // Устанавливаем фильтр по тексту\r\n        this.grid.searchString = this.displayValue;\r\n      });\r\n    }\r\n\r\n    if (this.displayValue === '') {\r\n      this.setValue(null, '');\r\n      return;\r\n    }\r\n\r\n    this._value = null;\r\n  }\r\n\r\n  // Возврат выбранной строки в гриде\r\n  private getSelectedRow(): any {\r\n    if (this.grid && this.grid.state.selection.ranges.length > 0) {\r\n      let item = this.grid.state.selection.ranges[0].fromCell.row;\r\n      return item;\r\n    }\r\n    return null;\r\n  }\r\n\r\n  private containsDisplayed(s: string): boolean {\r\n    return !this.displayValue || (s !== undefined && s.toLowerCase().indexOf(this.displayValue.toLowerCase()) >= 0);\r\n  }\r\n\r\n  // Установка значения по выделенной в гриде записи\r\n  private setValueFromGrid() {\r\n\r\n    if (!this.popupVisible && this.grid.state.model) {\r\n\r\n      const found = this.grid.state.model.find(\r\n          item => item[this.displayField].toLowerCase() === this.displayValue.toLowerCase()\r\n      );\r\n\r\n      if (found) {\r\n        this.setValue(found[this.valueField], found[this.displayField]);\r\n      } else {\r\n        this.setValue(null, '');\r\n      }\r\n      return;\r\n    }\r\n\r\n    const sel = this.getSelectedRow();\r\n    if (sel && (this._onGrid || this.containsDisplayed(sel[this.displayField]))) {\r\n      this.setValue(sel[this.valueField], sel[this.displayField]);\r\n    } else {\r\n      this.setValueFromDisplayed();\r\n    }\r\n  }\r\n\r\n  /**\r\n   * Установка значения по введенному тексту\r\n   */\r\n  public setValueFromDisplayed() {\r\n    // По запросу. Смотрим в гриде. Хотя, если подумать, можно всегда в гриде смотреть..\r\n    let f = null;\r\n    if (this.grid && this.grid.resultRows) {\r\n      f = this.grid.resultRows.find(item => item[this.displayField].toLowerCase() === this.displayValue.toLowerCase());\r\n    }\r\n    if (f) {\r\n      this.setValue(f[this.valueField], f[this.displayField]);\r\n    } else {\r\n      this.setValue(null, '');\r\n    }\r\n    return;\r\n  }\r\n\r\n  /**\r\n   * Потеря фокуса инпутом\r\n   * @param  e Параметры события\r\n   */\r\n  inputBlur(e: FocusEvent) {\r\n    this._skipFocusOnPopupClose = true;\r\n    if ((this.value === null || this.value === undefined) && this.displayValue !== '') {\r\n\r\n      if (e.relatedTarget && Utils.isAncestor(this.popup.elementRef.nativeElement, e.relatedTarget)) {\r\n        // Если фокус перешел на...\r\n        return;\r\n      }\r\n\r\n      this.closePopup();\r\n      this.setValueFromDisplayed();\r\n    }\r\n    super.inputBlur(e);\r\n  }\r\n\r\n  constructor(\r\n    protected _elementRef: ElementRef,\r\n    protected _renderer: Renderer2) {\r\n      super(_elementRef, _renderer);\r\n      this.useAltDown = false;\r\n  }\r\n}\r\n"]}