更新:問題はControlValueAccessorの実装にあり、その後、ControlValueAccessorを子要素に適用することに問題があると判断されました。質問は反映するように編集されました。イオン入力の通貨入力ディレクティブの実装方法
通貨の値を「ドル」形式(例:10.00)で表示するが、下位のモデルにセント(たとえば1000)形式で格納する属性ディレクティブを提供したいと考えています。
<!-- cost = 1000 would result in text input value of 10.00
<input type="text" [(ngModel)]="cost" name="cost" currencyInput>
<!-- or in Ionic 2 -->
<ion-input type="text" [(ngModel)]="cost" name="cost-ionic" currencyInput>
以前AngularJS 1.xでIは、解析を使用すると、次のように指示リンク機能でレンダリング:
(function() {
'use strict';
angular.module('app.directives').directive('ndDollarsToCents', ['$parse', function($parse) {
return {
require: 'ngModel',
link: function(scope, element, attrs, ctrl) {
var listener = function() {
element.val((value/100).toFixed(2));
};
ctrl.$parsers.push(function(viewValue) {
return Math.round(parseFloat(viewValue) * 100);
});
ctrl.$render = function() {
element.val((ctrl.$viewValue/100).toFixed(2));
};
element.bind('change', listener);
}
};
}]);
})();
イオン2で/角度2 IはControlValueAccessorインタフェースを使用してこれを実現以下有します。
import { Directive, Renderer, ElementRef, forwardRef } from '@angular/core';
import { ControlValueAccessor, NG_VALUE_ACCESSOR } from '@angular/forms';
const CURRENCY_VALUE_ACCESSOR = {
provide: NG_VALUE_ACCESSOR,
useExisting: forwardRef(() => CurrencyInputDirective),
multi: true
}
@Directive({
selector: '[currencyInput]',
host: {
'(input)': 'handleInput($event.target.value)'
},
providers: [ CURRENCY_VALUE_ACCESSOR ]
})
export class CurrencyInputDirective implements ControlValueAccessor, AfterViewInit
{
onChange = (_: any) => {};
onTouched =() => {};
inputElement: HTMLInputElement = null;
constructor(private renderer: Renderer, private elementRef: ElementRef) {}
ngAfterViewInit()
{
let element = this.elementRef.nativeElement;
if(element.tagName === 'INPUT')
{
this.inputElement = element;
}
else
{
this.inputElement = element.getElementsByTagName('input')[0];
}
}
registerOnChange(fn: (_: any) => void): void { this.onChange = fn; }
registerOnTouched(fn:() => void): void { this.onTouched = fn; }
handleInput(value : string)
{
if (value)
{
value = String(Math.round(parseFloat(value) * 100));
}
this.onChange(value);
}
writeValue(value: any): void
{
if (value)
{
value = (parseInt(value)/100).toFixed(2);
}
this.renderer.setElementProperty(this.inputElement, 'value', value);
}
}
これはイオン入力に適用したときに直線入力エレメントに適用するとうまく動作しますが、機能しません。 ControlValueAccessorを取得してイオン入力の子入力要素に適用する方法はありますか?
入出力サンプル – Aravind
私の答えを参照してください – Aravind