Skip to content

Commit 9860382

Browse files
PatrickJSmhevery
authored andcommitted
feat(change_detection): json pipe
Closes #1957
1 parent 8e84f8a commit 9860382

File tree

5 files changed

+202
-3
lines changed

5 files changed

+202
-3
lines changed

modules/angular2/src/change_detection/change_detection.ts

+10-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ import {ObservablePipeFactory} from './pipes/observable_pipe';
77
import {PromisePipeFactory} from './pipes/promise_pipe';
88
import {UpperCaseFactory} from './pipes/uppercase_pipe';
99
import {LowerCaseFactory} from './pipes/lowercase_pipe';
10+
import {JsonPipeFactory} from './pipes/json_pipe';
1011
import {NullPipeFactory} from './pipes/null_pipe';
1112
import {ChangeDetection, ProtoChangeDetector, ChangeDetectorDefinition} from './interfaces';
1213
import {Injectable} from 'angular2/src/di/decorators';
@@ -55,12 +56,20 @@ export var uppercase: List < PipeFactory >= [new UpperCaseFactory(), new NullPip
5556
*/
5657
export var lowercase: List < PipeFactory >= [new LowerCaseFactory(), new NullPipeFactory()];
5758

59+
/**
60+
* Json stringify transform.
61+
*
62+
* @exportedAs angular2/pipes
63+
*/
64+
export var json: List < PipeFactory >= [new JsonPipeFactory(), new NullPipeFactory()];
65+
5866
export var defaultPipes = {
5967
"iterableDiff": iterableDiff,
6068
"keyValDiff": keyValDiff,
6169
"async": async,
6270
"uppercase": uppercase,
63-
"lowercase": lowercase
71+
"lowercase": lowercase,
72+
"json": json
6473
};
6574

6675
export var preGeneratedProtoDetectors = {};
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,80 @@
1+
import {isBlank, isPresent, CONST, Json} from 'angular2/src/facade/lang';
2+
import {Pipe, PipeFactory} from './pipe';
3+
4+
// HACK: workaround for Traceur behavior.
5+
// It expects all transpiled modules to contain this marker.
6+
// TODO: remove this when we no longer use traceur
7+
export var __esModule = true;
8+
9+
10+
/**
11+
* Implements json transforms to any object.
12+
*
13+
* # Example
14+
*
15+
* In this example we transform the user object to json.
16+
*
17+
* ```
18+
* @Component({
19+
* selector: "user-cmp"
20+
* })
21+
* @View({
22+
* template: "User: {{ user | json }}"
23+
* })
24+
* class Username {
25+
* user:Object
26+
* constructor() {
27+
* this.user = { name: "PatrickJS" };
28+
* }
29+
* }
30+
*
31+
* ```
32+
*
33+
* @exportedAs angular2/pipes
34+
*/
35+
export class JsonPipe extends Pipe {
36+
_latestRef: any;
37+
_latestValue: any;
38+
constructor() {
39+
super();
40+
this._latestRef = null;
41+
this._latestValue = null;
42+
}
43+
44+
onDestroy(): void {
45+
if (isPresent(this._latestValue)) {
46+
this._latestRef = null;
47+
this._latestValue = null;
48+
}
49+
}
50+
51+
supports(obj): boolean { return true; }
52+
53+
transform(value): any {
54+
if (value === this._latestRef) {
55+
return this._latestValue;
56+
} else {
57+
return this._prettyPrint(value);
58+
}
59+
}
60+
61+
_prettyPrint(value) {
62+
this._latestRef = value;
63+
this._latestValue = Json.stringify(value);
64+
return this._latestValue;
65+
}
66+
}
67+
68+
/**
69+
* Provides a factory for [JsonPipeFactory].
70+
*
71+
* @exportedAs angular2/pipes
72+
*/
73+
@CONST()
74+
export class JsonPipeFactory extends PipeFactory {
75+
constructor() { super(); }
76+
77+
supports(obj): boolean { return true; }
78+
79+
create(cdRef): Pipe { return new JsonPipe(); }
80+
}

modules/angular2/src/facade/lang.dart

+4-1
Original file line numberDiff line numberDiff line change
@@ -209,7 +209,10 @@ bool assertionsEnabled() {
209209
// Can't be all uppercase as our transpiler would think it is a special directive...
210210
class Json {
211211
static parse(String s) => convert.JSON.decode(s);
212-
static String stringify(data) => convert.JSON.encode(data);
212+
static String stringify(data) {
213+
var encoder = new convert.JsonEncoder.withIndent(" ");
214+
return encoder.convert(data);
215+
}
213216
}
214217

215218
class DateWrapper {

modules/angular2/src/facade/lang.ts

+11-1
Original file line numberDiff line numberDiff line change
@@ -38,17 +38,21 @@ if (assertionsEnabled_) {
3838
int = {};
3939
_global.assert = function() {};
4040
}
41+
4142
export {int};
4243

4344
// This function is needed only to properly support Dart's const expressions
4445
// see https://github.com/angular/ts2dart/pull/151 for more info
4546
export function CONST_EXPR<T>(expr: T): T {
4647
return expr;
4748
}
49+
4850
export function CONST() {
4951
return (target) => target;
5052
}
53+
5154
export class ABSTRACT {}
55+
5256
export class IMPLEMENTS {}
5357

5458
export function isPresent(obj): boolean {
@@ -246,7 +250,13 @@ export function print(obj) {
246250
}
247251

248252
// Can't be all uppercase as our transpiler would think it is a special directive...
249-
export var Json = _global.JSON;
253+
export class Json {
254+
static parse(s: string) { return _global.JSON.parse(s); }
255+
static stringify(data): string {
256+
// Dart doesn't take 3 arguments
257+
return _global.JSON.stringify(data, null, 2);
258+
}
259+
}
250260

251261
export class DateWrapper {
252262
static fromMillis(ms) { return new Date(ms); }
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,97 @@
1+
import {ddescribe, describe, it, iit, xit, expect, beforeEach, afterEach,
2+
AsyncTestCompleter, inject, proxy, SpyObject, IS_DARTIUM} from 'angular2/test_lib';
3+
import {Json, RegExp, NumberWrapper, StringWrapper} from 'angular2/src/facade/lang';
4+
5+
import {JsonPipe} from 'angular2/src/change_detection/pipes/json_pipe';
6+
7+
export function main() {
8+
describe("JsonPipe", () => {
9+
var regNewLine = new RegExp('\n');
10+
var canHasUndefined; // because Dart doesn't like undefined;
11+
var inceptionObj;
12+
var inceptionObjString;
13+
var catString;
14+
var pipe;
15+
16+
function normalize(obj: string): string {
17+
return StringWrapper.replace(obj, regNewLine, '');
18+
}
19+
20+
beforeEach(() => {
21+
inceptionObj = {
22+
dream: {
23+
dream: {
24+
dream: 'Limbo'
25+
}
26+
}
27+
};
28+
inceptionObjString = "{\n" +
29+
" \"dream\": {\n" +
30+
" \"dream\": {\n" +
31+
" \"dream\": \"Limbo\"\n" +
32+
" }\n" +
33+
" }\n" +
34+
"}";
35+
36+
37+
catString = 'Inception Cat';
38+
pipe = new JsonPipe();
39+
});
40+
41+
describe("supports", () => {
42+
it("should support objects", () => {
43+
expect(pipe.supports(inceptionObj)).toBe(true);
44+
});
45+
46+
it("should support strings", () => {
47+
expect(pipe.supports(catString)).toBe(true);
48+
});
49+
50+
it("should support null", () => {
51+
expect(pipe.supports(null)).toBe(true);
52+
});
53+
54+
it("should support NaN", () => {
55+
expect(pipe.supports(NumberWrapper.NaN)).toBe(true);
56+
});
57+
58+
if (!IS_DARTIUM) {
59+
it("should support undefined", () => {
60+
expect(pipe.supports(canHasUndefined)).toBe(true);
61+
});
62+
}
63+
64+
});
65+
66+
describe("transform", () => {
67+
it("should return JSON-formatted string", () => {
68+
expect(pipe.transform(inceptionObj)).toEqual(inceptionObjString);
69+
});
70+
71+
it("should return JSON-formatted string even when normalized", () => {
72+
var dream1 = normalize(pipe.transform(inceptionObj));
73+
var dream2 = normalize(inceptionObjString);
74+
expect(dream1).toEqual(dream2);
75+
});
76+
77+
it("should return JSON-formatted string similar to Json.stringify", () => {
78+
var dream1 = normalize(pipe.transform(inceptionObj));
79+
var dream2 = normalize(Json.stringify(inceptionObj));
80+
expect(dream1).toEqual(dream2);
81+
});
82+
83+
it("should return same value when nothing has changed since the last call", () => {
84+
expect(pipe.transform(inceptionObj)).toEqual(inceptionObjString);
85+
expect(pipe.transform(inceptionObj)).toEqual(inceptionObjString);
86+
});
87+
88+
});
89+
90+
describe("onDestroy", () => {
91+
it("should do nothing when no latest value", () => {
92+
expect(() => pipe.onDestroy()).not.toThrow();
93+
});
94+
});
95+
96+
});
97+
}

0 commit comments

Comments
 (0)