タイトルの通りです。割と遭遇している人いるんじゃないかなと勝手に想像しているんですが、あまり情報ないので書いてみます。
ユースケース的には、WebAPI アクセスで日付を JSON シリアライズしたものを受け取って、input[type=date]の ng−model に直接データバインドするような想定です。
サンプルコード
(適当に書いているので動かなかったらすみません。雰囲気だけでも。)
WebAPI 側から JSON データ
{
"id": 1,
"name": "ググレカス",
"createdAt": "2015-02-05T00:00:00.000Z"
}
AngularJS の DataService 的コード(今回は factory)
angular.module("app").factory("ggrks", function ($resource) {
return $resource("/api/ggrks");
});
AngularJS の Controller
angular.module("app").controller("ggrksController", function ($scope, ggrks) {
// 初回データアクセス
ggrks.query(function (data) {
$scope.ggrks = data;
});
});
AngularJS のテンプレート
<input type="date" ng-model="ggrks.createdAt" />
このまま実行すると、このようなエラーが発生します。しょぼーん。
Error: [ngModel:datefmt] Expected `2015-02-05T00:00:00.000Z` to be a date
....
input[type=date]にセットする model は Date 型でなければならない
公式ドキュメントをよく見ると書いてました。有効な ISO-8601 フォーマット(yyyy-MM-dd)と書いてますが、Javascript 的には Date 型に Parse してセットしなければならないようです。
WebAPI から受け取る日付は JSON シリアライズされているので、いちいち Date 型に Perse しなければならないのか。。。本当に面倒くさい。
現在のベターな回答
いちいち Controller にて Parse するのはナンセンスなので、DataService(factory)で Parse することにしました。
ビルトインの$resource
サービスには、受け取ったレスポンスを変形させて後続処理に渡すtransformResponse
があるので、今回はこちらを使ってみました。
(Backbone.Model や Collection のparse
相当です。)
変更後の DataService 的コード
angular.module("app").factory("ggrks", function ($resource) {
return $resource("/api/ggrks", {
query: {
transformResponse: transformResponse,
},
});
// 受け取ったJSONをObjectにParseする
function transformResponse(data) {
data = angular.fromJson(data);
data.createdAt = new Date(data.createdAt);
return data;
}
});
無理やり感がはんぱないですが、、、これで無事、input[type=date]に対して日付型をデータバインドさせることができました。
[W3C の仕様](http://www.w3.org/TR/html5/forms.html#date-state-(type=date)でも、日付として有効な文字列は自動的にParseされるようですし、できればフレームワーク側で判断してPerseして欲しいところですが。。。
これより良いやり方ご存知の方いらっしゃいましたら、ぜひ教えてください。
参考文献
- javascript - How to set right date format for editable-date - Stack Overflow
- javascript - Why is my date input field in AngularJS throwing type error? - Stack Overflow
- Why is my date input field in AngularJS throwing type error? | Arianna
少し古いですがこちらに書かれている、dateInput 用のカスタム directive を作成する方法が良さそうな気がします。(未検証)