UNPKG

15.2 kBJavaScriptView Raw
1import * as tslib_1 from "tslib";
2import { HostListener } from '@angular/core';
3import { NgControl } from '@angular/forms';
4import { raf } from '../../util/util';
5export class ValueAccessor {
6 constructor(injector, el) {
7 this.injector = injector;
8 this.el = el;
9 this.onChange = () => { };
10 this.onTouched = () => { };
11 }
12 writeValue(value) {
13 /**
14 * TODO for Ionic 6:
15 * Change `value == null ? '' : value;`
16 * to `value`. This was a fix for IE9, but IE9
17 * is no longer supported; however, this change
18 * is potentially a breaking change
19 */
20 this.el.nativeElement.value = this.lastValue = value == null ? '' : value;
21 setIonicClasses(this.el);
22 }
23 handleChangeEvent(el, value) {
24 if (el === this.el.nativeElement) {
25 if (value !== this.lastValue) {
26 this.lastValue = value;
27 this.onChange(value);
28 }
29 setIonicClasses(this.el);
30 }
31 }
32 _handleBlurEvent(el) {
33 if (el === this.el.nativeElement) {
34 this.onTouched();
35 setIonicClasses(this.el);
36 }
37 }
38 registerOnChange(fn) {
39 this.onChange = fn;
40 }
41 registerOnTouched(fn) {
42 this.onTouched = fn;
43 }
44 setDisabledState(isDisabled) {
45 this.el.nativeElement.disabled = isDisabled;
46 }
47 ngOnDestroy() {
48 if (this.statusChanges) {
49 this.statusChanges.unsubscribe();
50 }
51 }
52 ngAfterViewInit() {
53 let ngControl;
54 try {
55 ngControl = this.injector.get(NgControl);
56 }
57 catch ( /* No FormControl or ngModel binding */_a) { /* No FormControl or ngModel binding */ }
58 if (!ngControl) {
59 return;
60 }
61 // Listen for changes in validity, disabled, or pending states
62 if (ngControl.statusChanges) {
63 this.statusChanges = ngControl.statusChanges.subscribe(() => setIonicClasses(this.el));
64 }
65 /**
66 * TODO Remove this in favor of https://github.com/angular/angular/issues/10887
67 * whenever it is implemented. Currently, Ionic's form status classes
68 * do not react to changes when developers manually call
69 * Angular form control methods such as markAsTouched.
70 * This results in Ionic's form status classes being out
71 * of sync with the ng form status classes.
72 * This patches the methods to manually sync
73 * the classes until this feature is implemented in Angular.
74 */
75 const formControl = ngControl.control;
76 if (formControl) {
77 const methodsToPatch = ['markAsTouched', 'markAllAsTouched', 'markAsUntouched', 'markAsDirty', 'markAsPristine'];
78 methodsToPatch.forEach(method => {
79 if (formControl[method]) {
80 const oldFn = formControl[method].bind(formControl);
81 formControl[method] = (...params) => {
82 oldFn(...params);
83 setIonicClasses(this.el);
84 };
85 }
86 });
87 }
88 }
89}
90tslib_1.__decorate([
91 HostListener('ionBlur', ['$event.target'])
92], ValueAccessor.prototype, "_handleBlurEvent", null);
93export const setIonicClasses = (element) => {
94 raf(() => {
95 const input = element.nativeElement;
96 const classes = getClasses(input);
97 setClasses(input, classes);
98 const item = input.closest('ion-item');
99 if (item) {
100 setClasses(item, classes);
101 }
102 });
103};
104const getClasses = (element) => {
105 const classList = element.classList;
106 const classes = [];
107 for (let i = 0; i < classList.length; i++) {
108 const item = classList.item(i);
109 if (item !== null && startsWith(item, 'ng-')) {
110 classes.push(`ion-${item.substr(3)}`);
111 }
112 }
113 return classes;
114};
115const ɵ0 = getClasses;
116const setClasses = (element, classes) => {
117 const classList = element.classList;
118 [
119 'ion-valid',
120 'ion-invalid',
121 'ion-touched',
122 'ion-untouched',
123 'ion-dirty',
124 'ion-pristine'
125 ].forEach(c => classList.remove(c));
126 classes.forEach(c => classList.add(c));
127};
128const ɵ1 = setClasses;
129const startsWith = (input, search) => {
130 return input.substr(0, search.length) === search;
131};
132const ɵ2 = startsWith;
133export { ɵ0, ɵ1, ɵ2 };
134//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"value-accessor.js","sourceRoot":"ng://@ionic/angular/","sources":["directives/control-value-accessors/value-accessor.ts"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ElementRef, HostListener, Injector, OnDestroy, Type } from '@angular/core';\nimport { ControlValueAccessor, NgControl } from '@angular/forms';\nimport { Subscription } from 'rxjs';\n\nimport { raf } from '../../util/util';\n\nexport class ValueAccessor implements ControlValueAccessor, AfterViewInit, OnDestroy {\n\n  private onChange: (value: any) => void = () => {/**/};\n  private onTouched: () => void = () => {/**/};\n  protected lastValue: any;\n  private statusChanges?: Subscription;\n\n  constructor(protected injector: Injector, protected el: ElementRef) {}\n\n  writeValue(value: any) {\n    /**\n     * TODO for Ionic 6:\n     * Change `value == null ? '' : value;`\n     * to `value`. This was a fix for IE9, but IE9\n     * is no longer supported; however, this change\n     * is potentially a breaking change\n     */\n    this.el.nativeElement.value = this.lastValue = value == null ? '' : value;\n    setIonicClasses(this.el);\n  }\n\n  handleChangeEvent(el: HTMLElement, value: any) {\n    if (el === this.el.nativeElement) {\n      if (value !== this.lastValue) {\n        this.lastValue = value;\n        this.onChange(value);\n      }\n      setIonicClasses(this.el);\n    }\n  }\n\n  @HostListener('ionBlur', ['$event.target'])\n  _handleBlurEvent(el: any) {\n    if (el === this.el.nativeElement) {\n      this.onTouched();\n      setIonicClasses(this.el);\n    }\n  }\n\n  registerOnChange(fn: (value: any) => void) {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: () => void) {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(isDisabled: boolean) {\n    this.el.nativeElement.disabled = isDisabled;\n  }\n\n  ngOnDestroy() {\n    if (this.statusChanges) {\n      this.statusChanges.unsubscribe();\n    }\n  }\n\n  ngAfterViewInit() {\n    let ngControl;\n    try {\n      ngControl = this.injector.get<NgControl>(NgControl as Type<NgControl>);\n    } catch { /* No FormControl or ngModel binding */ }\n\n    if (!ngControl) { return; }\n\n    // Listen for changes in validity, disabled, or pending states\n    if (ngControl.statusChanges) {\n      this.statusChanges = ngControl.statusChanges.subscribe(() => setIonicClasses(this.el));\n    }\n\n    /**\n     * TODO Remove this in favor of https://github.com/angular/angular/issues/10887\n     * whenever it is implemented. Currently, Ionic's form status classes\n     * do not react to changes when developers manually call\n     * Angular form control methods such as markAsTouched.\n     * This results in Ionic's form status classes being out\n     * of sync with the ng form status classes.\n     * This patches the methods to manually sync\n     * the classes until this feature is implemented in Angular.\n     */\n    const formControl = ngControl.control;\n    if (formControl) {\n      const methodsToPatch = ['markAsTouched', 'markAllAsTouched', 'markAsUntouched', 'markAsDirty', 'markAsPristine'];\n      methodsToPatch.forEach(method => {\n       if (formControl[method]) {\n         const oldFn = formControl[method].bind(formControl);\n         formControl[method] = (...params) => {\n           oldFn(...params);\n           setIonicClasses(this.el);\n          };\n        }\n      });\n    }\n  }\n}\n\nexport const setIonicClasses = (element: ElementRef) => {\n  raf(() => {\n    const input = element.nativeElement as HTMLElement;\n    const classes = getClasses(input);\n    setClasses(input, classes);\n\n    const item = input.closest('ion-item');\n    if (item) {\n      setClasses(item, classes);\n    }\n  });\n};\n\nconst getClasses = (element: HTMLElement) => {\n  const classList = element.classList;\n  const classes = [];\n  for (let i = 0; i < classList.length; i++) {\n    const item = classList.item(i);\n    if (item !== null && startsWith(item, 'ng-')) {\n      classes.push(`ion-${item.substr(3)}`);\n    }\n  }\n  return classes;\n};\n\nconst setClasses = (element: HTMLElement, classes: string[]) => {\n  const classList = element.classList;\n  [\n    'ion-valid',\n    'ion-invalid',\n    'ion-touched',\n    'ion-untouched',\n    'ion-dirty',\n    'ion-pristine'\n  ].forEach(c => classList.remove(c));\n\n  classes.forEach(c => classList.add(c));\n};\n\nconst startsWith = (input: string, search: string): boolean => {\n  return input.substr(0, search.length) === search;\n};\n"]}
\No newline at end of file