@true-directive/grid
Version:
Angular Data Grid from Yopsilon.
223 lines • 28.7 kB
JavaScript
import * as tslib_1 from "tslib";
/**
* Copyright (c) 2018-2019 Aleksey Melnikov, True Directive Company.
* @link https://truedirective.com/
* @license MIT
*/
import { Output, Directive, ElementRef, Renderer2, EventEmitter } from '@angular/core';
import { Mask } from '@true-directive/base';
import { Keys } from '@true-directive/base';
import { MaskSectionAction, MaskResult } from '@true-directive/base';
import { InternationalizationService } from '../internationalization/internationalization.service';
var MaskBaseDirective = /** @class */ (function () {
function MaskBaseDirective(_renderer, _elementRef, intl) {
var _this = this;
this._renderer = _renderer;
this._elementRef = _elementRef;
this.intl = intl;
this._undo = [];
this._redo = [];
// Current text value
this._txtValue = '';
this._mask = new Mask();
this.android_behavior = false;
// On state change
this.stateChange = new EventEmitter();
this.maskValueChanged = new EventEmitter();
// Fetching mask state
this._state = null;
this._localeSubscription = this.intl.onLocaleChanged.subscribe(function (locale) { return _this.setLocale(locale); });
}
Object.defineProperty(MaskBaseDirective.prototype, "state", {
get: function () {
return this._state;
},
set: function (v) {
if (this._state !== v) {
this._state = v;
this.stateChange.emit(this._state); // Emitting event
}
},
enumerable: true,
configurable: true
});
MaskBaseDirective.prototype.updateState = function () {
//
};
MaskBaseDirective.prototype.processAndroid = function (txt) {
var res = this.currentRes();
// Possibly we have carriage position
var key = Keys.whichKeyHasBeenPressed(this.last_res.newValue, txt, this.last_res.selStart, res.selStart, this.last_res.selLength);
var r = this.processKey({
keyCode: -1,
key: key.code,
char: key.char,
shiftKey: false,
ctrlKey: false,
target: { selectionStart: this.last_res.selStart, selectionEnd: 0 },
preventDefault: function (_) { }
});
if (!r) {
this.setRes(this.last_res); // Reversing, value has not been accepted
}
this.android_behavior = false;
return;
};
MaskBaseDirective.prototype.doInput = function (txt) {
if (this.android_behavior) {
this.processAndroid(txt);
return;
}
// Thus we're trying to apply a mask to value entered
var masked = this._mask.applyMask(txt);
if (masked !== this._txtValue) {
this.setText(masked, true);
}
};
MaskBaseDirective.prototype.processKey = function (e) {
if (e.keyCode === 229 || e.keyCode === 0 || e.keyCode === undefined) {
// Android detected
this.android_behavior = true;
this.last_res = this.currentRes();
return;
}
var c = e.char;
if (c === undefined) {
c = e.key;
}
var selStart = e.target.selectionStart;
var selEnd = e.target.selectionEnd;
var s = this._txtValue;
if (Keys.isFunctional(e.keyCode)) {
return true;
}
if (e.keyCode === Keys.TAB || e.keyCode === Keys.ESCAPE) {
return true;
}
if (e.keyCode === Keys.HOME || e.keyCode === Keys.END) {
return true;
}
if (e.shiftKey && (e.keyCode === Keys.DELETE || e.keyCode === Keys.INSERT)) {
return true;
}
if (e.altKey && (e.keyCode === Keys.DOWN || e.keyCode === Keys.UP)) {
return true;
}
if (e.ctrlKey && e.keyCode === Keys.Z) {
// UNDO
var undoRes = this._undo.pop();
if (undoRes) {
this._redo.push(this.getRes(s, selStart, selEnd));
this.setRes(undoRes);
}
e.preventDefault();
return false;
}
if (e.ctrlKey && e.keyCode === Keys.Y) {
// REDO
var redoRes = this._redo.pop();
if (redoRes) {
this._undo.push(this.getRes(s, selStart, selEnd));
this.setRes(redoRes);
}
e.preventDefault();
return false;
}
// Остальные Ctrl+* нас не интересуют
if (e.ctrlKey) {
return true;
}
// If everything is selected
if (selStart === 0 && selEnd === this._txtValue.length) {
if (e.keyCode === Keys.DELETE || e.keyCode === Keys.BACKSPACE) {
return true;
}
// If ArrowLeft key has been pressed, result should equal to pressing of Home
if (e.keyCode === Keys.LEFT) {
return true;
}
if (e.keyCode === Keys.RIGHT) {
return true;
}
}
if (selStart === 0 && selEnd === this._txtValue.length) {
s = '';
selStart = 0;
selEnd = 0;
}
// Applying everything that's left
var res = this._mask.applyKeyAtPos(s, e.keyCode, c, selStart, selEnd);
if (res !== null && res.action === MaskSectionAction.APPLY) {
// If value has been changed we'll add it to UNDO stack
if (res.newValue !== s) {
this._undo.push(this.getRes(s, selStart, selEnd));
this._redo = [];
}
this.setRes(res);
if (this.android_behavior) {
return true;
}
e.preventDefault();
}
return false;
};
// Setting value and carriage position
MaskBaseDirective.prototype.setRes = function (res) {
if (this.android_behavior) {
res.selLength = 0;
}
this.setText(res.newValue);
this._renderer.setProperty(this._elementRef.nativeElement, 'selectionStart', res.selStart);
this._renderer.setProperty(this._elementRef.nativeElement, 'selectionEnd', res.selStart + res.selLength);
};
MaskBaseDirective.prototype.currentRes = function () {
var res = new MaskResult(this._txtValue, MaskSectionAction.APPLY, 0);
res.selStart = this._elementRef.nativeElement.selectionStart;
res.selLength = this._elementRef.nativeElement.selectionEnd - res.selStart;
return res;
};
// Retrieving current mask value and carriage position
MaskBaseDirective.prototype.getRes = function (s, selStart, selEnd) {
var res = new MaskResult(s, MaskSectionAction.APPLY, 0);
res.selStart = selStart;
res.selLength = selEnd - selStart;
return res;
};
// Following method should be overridden
MaskBaseDirective.prototype.toModel = function () {
};
// Writing a text to control
MaskBaseDirective.prototype.setText = function (displayedValue, toModel) {
if (toModel === void 0) { toModel = true; }
// Displaying
this._txtValue = displayedValue;
this._renderer.setProperty(this._elementRef.nativeElement, 'value', this._txtValue);
// Sending to model
if (toModel) {
this.toModel();
}
};
MaskBaseDirective.prototype.setLocale = function (locale) {
this._mask.setLocale(locale);
};
MaskBaseDirective.prototype.ngOnDestroy = function () {
this._localeSubscription.unsubscribe();
};
tslib_1.__decorate([
Output('stateChange'),
tslib_1.__metadata("design:type", Object)
], MaskBaseDirective.prototype, "stateChange", void 0);
tslib_1.__decorate([
Output('maskValueChanged'),
tslib_1.__metadata("design:type", Object)
], MaskBaseDirective.prototype, "maskValueChanged", void 0);
MaskBaseDirective = tslib_1.__decorate([
Directive({
selector: 'true-mask-base'
}),
tslib_1.__metadata("design:paramtypes", [Renderer2, ElementRef, InternationalizationService])
], MaskBaseDirective);
return MaskBaseDirective;
}());
export { MaskBaseDirective };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mask-base.directive.js","sourceRoot":"ng://@true-directive/grid/","sources":["src/mask/mask-base.directive.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 { Output, Directive, ElementRef, Renderer2, EventEmitter, OnDestroy } from '@angular/core';\r\nimport { Mask } from '@true-directive/base';\r\nimport { Keys, KeyInfo } from '@true-directive/base';\r\nimport { MaskSectionAction, MaskResult } from '@true-directive/base';\r\nimport { MaskSettings } from '@true-directive/base';\r\nimport { MaskState } from '@true-directive/base';\r\n\r\nimport { InternationalizationService } from '../internationalization/internationalization.service';\r\nimport { Locale } from '@true-directive/base';\r\n\r\n@Directive({\r\n  selector: 'true-mask-base'\r\n})\r\nexport class MaskBaseDirective implements OnDestroy {\r\n\r\n    private _undo: Array<MaskResult> = [];\r\n    private _redo: Array<MaskResult> = [];\r\n\r\n    // Current text value\r\n    protected _txtValue = '';\r\n    protected readonly _mask: Mask = new Mask();\r\n\r\n    private _localeSubscription: any;\r\n\r\n    protected android_behavior = false;\r\n    protected last_res: MaskResult;\r\n\r\n    // On state change\r\n    @Output('stateChange')\r\n    stateChange = new EventEmitter<MaskState>();\r\n\r\n    @Output('maskValueChanged')\r\n    maskValueChanged = new EventEmitter<any>();\r\n\r\n    // Fetching mask state\r\n    private _state: MaskState = null;\r\n\r\n    public get state(): MaskState {\r\n      return this._state;\r\n    }\r\n\r\n    public set state(v: MaskState) {\r\n      if (this._state !== v) {\r\n        this._state = v;\r\n        this.stateChange.emit(this._state); // Emitting event\r\n      }\r\n    }\r\n\r\n    protected updateState() {\r\n      //\r\n    }\r\n\r\n    protected processAndroid(txt: any): void {\r\n\r\n      const res = this.currentRes();\r\n\r\n      // Possibly we have carriage position\r\n      const key: KeyInfo = Keys.whichKeyHasBeenPressed(this.last_res.newValue, txt,\r\n          this.last_res.selStart, res.selStart, this.last_res.selLength);\r\n\r\n      const r = this.processKey(\r\n        {\r\n          keyCode: -1,\r\n          key: key.code,\r\n          char: key.char,\r\n          shiftKey: false,\r\n          ctrlKey: false,\r\n          target: { selectionStart: this.last_res.selStart, selectionEnd: 0 },\r\n          preventDefault: (_: any) => {}\r\n        });\r\n\r\n      if (!r) {\r\n        this.setRes(this.last_res); // Reversing, value has not been accepted\r\n      }\r\n\r\n      this.android_behavior = false;\r\n      return;\r\n    }\r\n\r\n    protected doInput(txt: any) {\r\n      if (this.android_behavior) {\r\n        this.processAndroid(txt);\r\n        return;\r\n      }\r\n\r\n      // Thus we're trying to apply a mask to value entered\r\n      const masked = this._mask.applyMask(txt);\r\n      if (masked !== this._txtValue) {\r\n        this.setText(masked, true);\r\n      }\r\n    }\r\n\r\n    public processKey(e: any): boolean {\r\n\r\n      if (e.keyCode === 229 || e.keyCode === 0 || e.keyCode === undefined) {\r\n        // Android detected\r\n        this.android_behavior = true;\r\n        this.last_res = this.currentRes();\r\n        return;\r\n      }\r\n\r\n      let c: string = e.char;\r\n      if (c === undefined) {\r\n        c = e.key;\r\n      }\r\n\r\n      let selStart: number = e.target.selectionStart;\r\n      let selEnd: number = e.target.selectionEnd;\r\n      let s = this._txtValue;\r\n\r\n      if (Keys.isFunctional(e.keyCode)) {\r\n        return true;\r\n      }\r\n\r\n      if (e.keyCode === Keys.TAB || e.keyCode === Keys.ESCAPE) {\r\n        return true;\r\n      }\r\n\r\n      if (e.keyCode === Keys.HOME || e.keyCode === Keys.END) {\r\n        return true;\r\n      }\r\n\r\n      if (e.shiftKey && (e.keyCode === Keys.DELETE || e.keyCode === Keys.INSERT)) {\r\n        return true;\r\n      }\r\n\r\n      if (e.altKey && (e.keyCode === Keys.DOWN || e.keyCode === Keys.UP)) {\r\n        return true;\r\n      }\r\n\r\n      if (e.ctrlKey && e.keyCode === Keys.Z) {\r\n        // UNDO\r\n        const undoRes = this._undo.pop();\r\n        if (undoRes) {\r\n          this._redo.push(this.getRes(s, selStart, selEnd));\r\n          this.setRes(undoRes);\r\n        }\r\n        e.preventDefault();\r\n        return false;\r\n      }\r\n\r\n      if (e.ctrlKey && e.keyCode === Keys.Y) {\r\n        // REDO\r\n        const redoRes = this._redo.pop();\r\n        if (redoRes) {\r\n          this._undo.push(this.getRes(s, selStart, selEnd));\r\n          this.setRes(redoRes);\r\n        }\r\n        e.preventDefault();\r\n        return false;\r\n      }\r\n\r\n      // Остальные Ctrl+* нас не интересуют\r\n      if (e.ctrlKey) {\r\n        return true;\r\n      }\r\n\r\n      // If everything is selected\r\n      if (selStart === 0 && selEnd === this._txtValue.length) {\r\n        if (e.keyCode === Keys.DELETE || e.keyCode === Keys.BACKSPACE) {\r\n          return true;\r\n        }\r\n\r\n        // If ArrowLeft key has been pressed, result should equal to pressing of Home\r\n        if (e.keyCode === Keys.LEFT) {\r\n          return true;\r\n        }\r\n\r\n        if (e.keyCode === Keys.RIGHT) {\r\n          return true;\r\n        }\r\n      }\r\n\r\n      if (selStart === 0 && selEnd === this._txtValue.length) {\r\n        s = '';\r\n        selStart = 0;\r\n        selEnd = 0;\r\n      }\r\n\r\n      // Applying everything that's left\r\n      const res: MaskResult = this._mask.applyKeyAtPos(s, e.keyCode, c, selStart, selEnd);\r\n      if (res !== null && res.action === MaskSectionAction.APPLY) {\r\n        // If value has been changed we'll add it to UNDO stack\r\n        if (res.newValue !== s) {\r\n          this._undo.push(this.getRes(s, selStart, selEnd));\r\n          this._redo = [];\r\n        }\r\n\r\n        this.setRes(res);\r\n\r\n        if (this.android_behavior) {\r\n          return true;\r\n        }\r\n\r\n        e.preventDefault();\r\n      }\r\n\r\n      return false;\r\n    }\r\n\r\n    // Setting value and carriage position\r\n    protected setRes(res: MaskResult) {\r\n\r\n      if (this.android_behavior) {\r\n        res.selLength = 0;\r\n      }\r\n\r\n      this.setText(res.newValue);\r\n      this._renderer.setProperty(this._elementRef.nativeElement, 'selectionStart', res.selStart);\r\n      this._renderer.setProperty(this._elementRef.nativeElement, 'selectionEnd', res.selStart + res.selLength);\r\n    }\r\n\r\n    protected currentRes() {\r\n      const res = new MaskResult(this._txtValue, MaskSectionAction.APPLY, 0);\r\n      res.selStart = this._elementRef.nativeElement.selectionStart;\r\n      res.selLength = this._elementRef.nativeElement.selectionEnd - res.selStart;\r\n      return res;\r\n    }\r\n\r\n    // Retrieving current mask value and carriage position\r\n    protected getRes(s: string, selStart: number, selEnd: number): MaskResult {\r\n      const res = new MaskResult(s, MaskSectionAction.APPLY, 0);\r\n      res.selStart = selStart;\r\n      res.selLength = selEnd - selStart;\r\n      return res;\r\n    }\r\n\r\n    // Following method should be overridden\r\n    protected toModel(): void {\r\n      \r\n    }\r\n\r\n    // Writing a text to control\r\n    protected setText(displayedValue: string, toModel: boolean = true) {\r\n\r\n      // Displaying\r\n      this._txtValue = displayedValue;\r\n      this._renderer.setProperty(this._elementRef.nativeElement, 'value', this._txtValue);\r\n\r\n      // Sending to model\r\n      if (toModel) {\r\n        this.toModel();\r\n      }\r\n    }\r\n\r\n    protected setLocale(locale: Locale) {\r\n      this._mask.setLocale(locale);\r\n    }\r\n\r\n    ngOnDestroy() {\r\n      this._localeSubscription.unsubscribe();\r\n    }\r\n\r\n    constructor(protected _renderer: Renderer2, protected _elementRef: ElementRef, protected intl: InternationalizationService) {\r\n      this._localeSubscription = this.intl.onLocaleChanged.subscribe(locale => this.setLocale(locale));\r\n    }\r\n}\r\n"]}