@angular-wave/angular.ts 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.eslintignore +1 -0
- package/.eslintrc.cjs +29 -0
- package/.github/workflows/playwright.yml +27 -0
- package/CHANGELOG.md +17974 -0
- package/CODE_OF_CONDUCT.md +3 -0
- package/CONTRIBUTING.md +246 -0
- package/DEVELOPERS.md +488 -0
- package/LICENSE +22 -0
- package/Makefile +31 -0
- package/README.md +115 -0
- package/RELEASE.md +98 -0
- package/SECURITY.md +16 -0
- package/TRIAGING.md +135 -0
- package/css/angular.css +22 -0
- package/dist/angular-ts.cjs.js +36843 -0
- package/dist/angular-ts.esm.js +36841 -0
- package/dist/angular-ts.umd.js +36848 -0
- package/dist/build/angular-animate.js +4272 -0
- package/dist/build/angular-aria.js +426 -0
- package/dist/build/angular-message-format.js +1072 -0
- package/dist/build/angular-messages.js +829 -0
- package/dist/build/angular-mocks.js +3757 -0
- package/dist/build/angular-parse-ext.js +1275 -0
- package/dist/build/angular-resource.js +911 -0
- package/dist/build/angular-route.js +1266 -0
- package/dist/build/angular-sanitize.js +891 -0
- package/dist/build/angular-touch.js +368 -0
- package/dist/build/angular.js +36600 -0
- package/e2e/unit.spec.ts +15 -0
- package/images/android-chrome-192x192.png +0 -0
- package/images/android-chrome-512x512.png +0 -0
- package/images/apple-touch-icon.png +0 -0
- package/images/favicon-16x16.png +0 -0
- package/images/favicon-32x32.png +0 -0
- package/images/favicon.ico +0 -0
- package/images/site.webmanifest +1 -0
- package/index.html +104 -0
- package/package.json +47 -0
- package/playwright.config.ts +78 -0
- package/public/circle.html +1 -0
- package/public/my_child_directive.html +1 -0
- package/public/my_directive.html +1 -0
- package/public/my_other_directive.html +1 -0
- package/public/test.html +1 -0
- package/rollup.config.js +31 -0
- package/src/animations/animateCache.js +55 -0
- package/src/animations/animateChildrenDirective.js +105 -0
- package/src/animations/animateCss.js +1139 -0
- package/src/animations/animateCssDriver.js +291 -0
- package/src/animations/animateJs.js +367 -0
- package/src/animations/animateJsDriver.js +67 -0
- package/src/animations/animateQueue.js +851 -0
- package/src/animations/animation.js +506 -0
- package/src/animations/module.js +779 -0
- package/src/animations/ngAnimateSwap.js +119 -0
- package/src/animations/rafScheduler.js +50 -0
- package/src/animations/shared.js +378 -0
- package/src/constants.js +20 -0
- package/src/core/animate.js +845 -0
- package/src/core/animateCss.js +73 -0
- package/src/core/animateRunner.js +195 -0
- package/src/core/attributes.js +199 -0
- package/src/core/cache.js +45 -0
- package/src/core/compile.js +4727 -0
- package/src/core/controller.js +225 -0
- package/src/core/exceptionHandler.js +63 -0
- package/src/core/filter.js +146 -0
- package/src/core/interpolate.js +442 -0
- package/src/core/interval.js +188 -0
- package/src/core/intervalFactory.js +57 -0
- package/src/core/location.js +1086 -0
- package/src/core/parser/parse.js +2562 -0
- package/src/core/parser/parse.md +13 -0
- package/src/core/q.js +746 -0
- package/src/core/rootScope.js +1596 -0
- package/src/core/sanitizeUri.js +85 -0
- package/src/core/sce.js +1161 -0
- package/src/core/taskTrackerFactory.js +125 -0
- package/src/core/timeout.js +121 -0
- package/src/core/urlUtils.js +187 -0
- package/src/core/utils.js +1349 -0
- package/src/directive/a.js +37 -0
- package/src/directive/attrs.js +283 -0
- package/src/directive/bind.js +51 -0
- package/src/directive/bind.md +142 -0
- package/src/directive/change.js +12 -0
- package/src/directive/change.md +25 -0
- package/src/directive/cloak.js +12 -0
- package/src/directive/cloak.md +24 -0
- package/src/directive/events.js +75 -0
- package/src/directive/events.md +166 -0
- package/src/directive/form.js +725 -0
- package/src/directive/init.js +15 -0
- package/src/directive/init.md +41 -0
- package/src/directive/input.js +1783 -0
- package/src/directive/list.js +46 -0
- package/src/directive/list.md +22 -0
- package/src/directive/ngClass.js +249 -0
- package/src/directive/ngController.js +64 -0
- package/src/directive/ngCsp.js +82 -0
- package/src/directive/ngIf.js +134 -0
- package/src/directive/ngInclude.js +217 -0
- package/src/directive/ngModel.js +1356 -0
- package/src/directive/ngModelOptions.js +509 -0
- package/src/directive/ngOptions.js +670 -0
- package/src/directive/ngRef.js +90 -0
- package/src/directive/ngRepeat.js +650 -0
- package/src/directive/ngShowHide.js +255 -0
- package/src/directive/ngSwitch.js +178 -0
- package/src/directive/ngTransclude.js +98 -0
- package/src/directive/non-bindable.js +11 -0
- package/src/directive/non-bindable.md +17 -0
- package/src/directive/script.js +30 -0
- package/src/directive/select.js +624 -0
- package/src/directive/style.js +25 -0
- package/src/directive/style.md +23 -0
- package/src/directive/validators.js +329 -0
- package/src/exts/aria.js +544 -0
- package/src/exts/messages.js +852 -0
- package/src/filters/filter.js +207 -0
- package/src/filters/filter.md +69 -0
- package/src/filters/filters.js +239 -0
- package/src/filters/json.md +16 -0
- package/src/filters/limit-to.js +43 -0
- package/src/filters/limit-to.md +19 -0
- package/src/filters/order-by.js +183 -0
- package/src/filters/order-by.md +83 -0
- package/src/index.js +13 -0
- package/src/injector.js +1034 -0
- package/src/jqLite.js +1117 -0
- package/src/loader.js +1320 -0
- package/src/public.js +215 -0
- package/src/routeToRegExp.js +41 -0
- package/src/services/anchorScroll.js +135 -0
- package/src/services/browser.js +321 -0
- package/src/services/cacheFactory.js +398 -0
- package/src/services/cookieReader.js +72 -0
- package/src/services/document.js +64 -0
- package/src/services/http.js +1537 -0
- package/src/services/httpBackend.js +206 -0
- package/src/services/log.js +160 -0
- package/src/services/templateRequest.js +139 -0
- package/test/angular.spec.js +2153 -0
- package/test/aria/aria.spec.js +1245 -0
- package/test/binding.spec.js +504 -0
- package/test/build-test.html +14 -0
- package/test/injector.spec.js +2327 -0
- package/test/jasmine/jasmine-5.1.2/boot0.js +65 -0
- package/test/jasmine/jasmine-5.1.2/boot1.js +133 -0
- package/test/jasmine/jasmine-5.1.2/jasmine-html.js +963 -0
- package/test/jasmine/jasmine-5.1.2/jasmine.css +320 -0
- package/test/jasmine/jasmine-5.1.2/jasmine.js +10824 -0
- package/test/jasmine/jasmine-5.1.2/jasmine_favicon.png +0 -0
- package/test/jasmine/jasmine-browser.json +17 -0
- package/test/jasmine/jasmine.json +9 -0
- package/test/jqlite.spec.js +2133 -0
- package/test/loader.spec.js +219 -0
- package/test/messages/messages.spec.js +1146 -0
- package/test/min-err.spec.js +174 -0
- package/test/mock-test.html +13 -0
- package/test/module-test.html +15 -0
- package/test/ng/anomate.spec.js +606 -0
- package/test/ng/cache-factor.spec.js +334 -0
- package/test/ng/compile.spec.js +17956 -0
- package/test/ng/controller-provider.spec.js +227 -0
- package/test/ng/cookie-reader.spec.js +98 -0
- package/test/ng/directive/a.spec.js +192 -0
- package/test/ng/directive/bind.spec.js +334 -0
- package/test/ng/directive/boolean.spec.js +136 -0
- package/test/ng/directive/change.spec.js +71 -0
- package/test/ng/directive/class.spec.js +858 -0
- package/test/ng/directive/click.spec.js +38 -0
- package/test/ng/directive/cloak.spec.js +44 -0
- package/test/ng/directive/constoller.spec.js +194 -0
- package/test/ng/directive/element-style.spec.js +92 -0
- package/test/ng/directive/event.spec.js +282 -0
- package/test/ng/directive/form.spec.js +1518 -0
- package/test/ng/directive/href.spec.js +143 -0
- package/test/ng/directive/if.spec.js +402 -0
- package/test/ng/directive/include.spec.js +828 -0
- package/test/ng/directive/init.spec.js +68 -0
- package/test/ng/directive/input.spec.js +3810 -0
- package/test/ng/directive/list.spec.js +170 -0
- package/test/ng/directive/model-options.spec.js +1008 -0
- package/test/ng/directive/model.spec.js +1905 -0
- package/test/ng/directive/non-bindable.spec.js +55 -0
- package/test/ng/directive/options.spec.js +3583 -0
- package/test/ng/directive/ref.spec.js +575 -0
- package/test/ng/directive/repeat.spec.js +1675 -0
- package/test/ng/directive/script.spec.js +52 -0
- package/test/ng/directive/scrset.spec.js +67 -0
- package/test/ng/directive/select.spec.js +2541 -0
- package/test/ng/directive/show-hide.spec.js +253 -0
- package/test/ng/directive/src.spec.js +157 -0
- package/test/ng/directive/style.spec.js +178 -0
- package/test/ng/directive/switch.spec.js +647 -0
- package/test/ng/directive/validators.spec.js +717 -0
- package/test/ng/document.spec.js +52 -0
- package/test/ng/filter/filter.spec.js +714 -0
- package/test/ng/filter/filters.spec.js +35 -0
- package/test/ng/filter/limit-to.spec.js +251 -0
- package/test/ng/filter/order-by.spec.js +891 -0
- package/test/ng/filter.spec.js +149 -0
- package/test/ng/http-backend.spec.js +398 -0
- package/test/ng/http.spec.js +4071 -0
- package/test/ng/interpolate.spec.js +642 -0
- package/test/ng/interval.spec.js +343 -0
- package/test/ng/location.spec.js +3488 -0
- package/test/ng/on.spec.js +229 -0
- package/test/ng/parse.spec.js +4655 -0
- package/test/ng/prop.spec.js +805 -0
- package/test/ng/q.spec.js +2904 -0
- package/test/ng/root-element.spec.js +16 -0
- package/test/ng/sanitize-uri.spec.js +249 -0
- package/test/ng/sce.spec.js +660 -0
- package/test/ng/scope.spec.js +3442 -0
- package/test/ng/template-request.spec.js +236 -0
- package/test/ng/timeout.spec.js +351 -0
- package/test/ng/url-utils.spec.js +156 -0
- package/test/ng/utils.spec.js +144 -0
- package/test/original-test.html +21 -0
- package/test/public.spec.js +34 -0
- package/test/sanitize/bing-html.spec.js +36 -0
- package/test/server/express.js +158 -0
- package/test/test-utils.js +11 -0
- package/tsconfig.json +17 -0
- package/types/angular.d.ts +138 -0
- package/types/global.d.ts +9 -0
- package/types/index.d.ts +2357 -0
- package/types/jqlite.d.ts +558 -0
- package/vite.config.js +14 -0
|
@@ -0,0 +1,1146 @@
|
|
|
1
|
+
import { createInjector } from "../../src/injector";
|
|
2
|
+
import { countWatchers } from "../../src/core/rootScope";
|
|
3
|
+
import { publishExternalAPI } from "../../src/public";
|
|
4
|
+
import { isString } from "../../src/core/utils";
|
|
5
|
+
import { jqLite } from "../../src/jqLite";
|
|
6
|
+
|
|
7
|
+
describe("ngMessages", () => {
|
|
8
|
+
let $rootScope, $compile, $templateCache;
|
|
9
|
+
|
|
10
|
+
beforeEach(() => {
|
|
11
|
+
publishExternalAPI();
|
|
12
|
+
window.angular
|
|
13
|
+
.module("app", ["ng", "ngMessages"])
|
|
14
|
+
.directive("messageWrap", () => ({
|
|
15
|
+
transclude: true,
|
|
16
|
+
scope: {
|
|
17
|
+
col: "=col",
|
|
18
|
+
},
|
|
19
|
+
template:
|
|
20
|
+
'<div ng-messages="col"><ng-transclude></ng-transclude></div>',
|
|
21
|
+
}));
|
|
22
|
+
|
|
23
|
+
createInjector(["app"]).invoke(
|
|
24
|
+
(_$rootScope_, _$compile_, _$templateCache_) => {
|
|
25
|
+
$rootScope = _$rootScope_;
|
|
26
|
+
$compile = _$compile_;
|
|
27
|
+
$templateCache = _$templateCache_;
|
|
28
|
+
},
|
|
29
|
+
);
|
|
30
|
+
});
|
|
31
|
+
|
|
32
|
+
function messageChildren(element) {
|
|
33
|
+
return (element.length ? element[0] : element).querySelectorAll(
|
|
34
|
+
"[ng-message], [ng-message-exp]",
|
|
35
|
+
);
|
|
36
|
+
}
|
|
37
|
+
|
|
38
|
+
function s(str) {
|
|
39
|
+
return str.replace(/\s+/g, "");
|
|
40
|
+
}
|
|
41
|
+
|
|
42
|
+
function trim(value) {
|
|
43
|
+
return isString(value) ? value.trim() : value;
|
|
44
|
+
}
|
|
45
|
+
|
|
46
|
+
let element;
|
|
47
|
+
|
|
48
|
+
it("should render based off of a hashmap collection", () => {
|
|
49
|
+
element = $compile(
|
|
50
|
+
'<div ng-messages="col">' +
|
|
51
|
+
' <div ng-message="val">Message is set</div>' +
|
|
52
|
+
"</div>",
|
|
53
|
+
)($rootScope);
|
|
54
|
+
$rootScope.$digest();
|
|
55
|
+
|
|
56
|
+
expect(element.text()).not.toContain("Message is set");
|
|
57
|
+
|
|
58
|
+
$rootScope.$apply(() => {
|
|
59
|
+
$rootScope.col = { val: true };
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
expect(element.text()).toContain("Message is set");
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
it("should render the same message if multiple message keys match", () => {
|
|
66
|
+
element = $compile(
|
|
67
|
+
'<div ng-messages="col">' +
|
|
68
|
+
' <div ng-message="one, two, three">Message is set</div>' +
|
|
69
|
+
"</div>",
|
|
70
|
+
)($rootScope);
|
|
71
|
+
$rootScope.$digest();
|
|
72
|
+
|
|
73
|
+
expect(element.text()).not.toContain("Message is set");
|
|
74
|
+
|
|
75
|
+
$rootScope.$apply(() => {
|
|
76
|
+
$rootScope.col = { one: true };
|
|
77
|
+
});
|
|
78
|
+
|
|
79
|
+
expect(element.text()).toContain("Message is set");
|
|
80
|
+
|
|
81
|
+
$rootScope.$apply(() => {
|
|
82
|
+
$rootScope.col = { two: true, one: false };
|
|
83
|
+
});
|
|
84
|
+
|
|
85
|
+
expect(element.text()).toContain("Message is set");
|
|
86
|
+
|
|
87
|
+
$rootScope.$apply(() => {
|
|
88
|
+
$rootScope.col = { three: true, two: false };
|
|
89
|
+
});
|
|
90
|
+
|
|
91
|
+
expect(element.text()).toContain("Message is set");
|
|
92
|
+
|
|
93
|
+
$rootScope.$apply(() => {
|
|
94
|
+
$rootScope.col = { three: false };
|
|
95
|
+
});
|
|
96
|
+
|
|
97
|
+
expect(element.text()).not.toContain("Message is set");
|
|
98
|
+
});
|
|
99
|
+
|
|
100
|
+
it("should use the when attribute when an element directive is used", () => {
|
|
101
|
+
element = $compile(
|
|
102
|
+
'<ng-messages for="col">' +
|
|
103
|
+
' <ng-message when="val">Message is set</div>' +
|
|
104
|
+
"</ng-messages>",
|
|
105
|
+
)($rootScope);
|
|
106
|
+
$rootScope.$digest();
|
|
107
|
+
|
|
108
|
+
expect(element.text()).not.toContain("Message is set");
|
|
109
|
+
|
|
110
|
+
$rootScope.$apply(() => {
|
|
111
|
+
$rootScope.col = { val: true };
|
|
112
|
+
});
|
|
113
|
+
|
|
114
|
+
expect(element.text()).toContain("Message is set");
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
it("should render the same message if multiple message keys match based on the when attribute", () => {
|
|
118
|
+
element = $compile(
|
|
119
|
+
'<ng-messages for="col">' +
|
|
120
|
+
' <ng-message when=" one two three ">Message is set</div>' +
|
|
121
|
+
"</ng-messages>",
|
|
122
|
+
)($rootScope);
|
|
123
|
+
$rootScope.$digest();
|
|
124
|
+
|
|
125
|
+
expect(element.text()).not.toContain("Message is set");
|
|
126
|
+
|
|
127
|
+
$rootScope.$apply(() => {
|
|
128
|
+
$rootScope.col = { one: true };
|
|
129
|
+
});
|
|
130
|
+
|
|
131
|
+
expect(element.text()).toContain("Message is set");
|
|
132
|
+
|
|
133
|
+
$rootScope.$apply(() => {
|
|
134
|
+
$rootScope.col = { two: true, one: false };
|
|
135
|
+
});
|
|
136
|
+
|
|
137
|
+
expect(element.text()).toContain("Message is set");
|
|
138
|
+
|
|
139
|
+
$rootScope.$apply(() => {
|
|
140
|
+
$rootScope.col = { three: true, two: false };
|
|
141
|
+
});
|
|
142
|
+
|
|
143
|
+
expect(element.text()).toContain("Message is set");
|
|
144
|
+
|
|
145
|
+
$rootScope.$apply(() => {
|
|
146
|
+
$rootScope.col = { three: false };
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
expect(element.text()).not.toContain("Message is set");
|
|
150
|
+
});
|
|
151
|
+
|
|
152
|
+
it("should allow a dynamic expression to be set when ng-message-exp is used", () => {
|
|
153
|
+
element = $compile(
|
|
154
|
+
'<div ng-messages="col">' +
|
|
155
|
+
' <div ng-message-exp="variable">Message is crazy</div>' +
|
|
156
|
+
"</div>",
|
|
157
|
+
)($rootScope);
|
|
158
|
+
$rootScope.$digest();
|
|
159
|
+
|
|
160
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
161
|
+
|
|
162
|
+
$rootScope.$apply(() => {
|
|
163
|
+
$rootScope.variable = "error";
|
|
164
|
+
$rootScope.col = { error: true };
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
expect(element.text()).toContain("Message is crazy");
|
|
168
|
+
|
|
169
|
+
$rootScope.$apply(() => {
|
|
170
|
+
$rootScope.col = { error: false, failure: true };
|
|
171
|
+
});
|
|
172
|
+
|
|
173
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
174
|
+
|
|
175
|
+
$rootScope.$apply(() => {
|
|
176
|
+
$rootScope.variable = ["failure"];
|
|
177
|
+
});
|
|
178
|
+
|
|
179
|
+
expect(element.text()).toContain("Message is crazy");
|
|
180
|
+
|
|
181
|
+
$rootScope.$apply(() => {
|
|
182
|
+
$rootScope.variable = null;
|
|
183
|
+
});
|
|
184
|
+
|
|
185
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
186
|
+
});
|
|
187
|
+
|
|
188
|
+
it("should allow a dynamic expression to be set when the when-exp attribute is used", () => {
|
|
189
|
+
element = $compile(
|
|
190
|
+
'<ng-messages for="col">' +
|
|
191
|
+
' <ng-message when-exp="variable">Message is crazy</ng-message>' +
|
|
192
|
+
"</ng-messages>",
|
|
193
|
+
)($rootScope);
|
|
194
|
+
$rootScope.$digest();
|
|
195
|
+
|
|
196
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
197
|
+
|
|
198
|
+
$rootScope.$apply(() => {
|
|
199
|
+
$rootScope.variable = "error, failure";
|
|
200
|
+
$rootScope.col = { error: true };
|
|
201
|
+
});
|
|
202
|
+
|
|
203
|
+
expect(element.text()).toContain("Message is crazy");
|
|
204
|
+
|
|
205
|
+
$rootScope.$apply(() => {
|
|
206
|
+
$rootScope.col = { error: false, failure: true };
|
|
207
|
+
});
|
|
208
|
+
|
|
209
|
+
expect(element.text()).toContain("Message is crazy");
|
|
210
|
+
|
|
211
|
+
$rootScope.$apply(() => {
|
|
212
|
+
$rootScope.variable = [];
|
|
213
|
+
});
|
|
214
|
+
|
|
215
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
216
|
+
|
|
217
|
+
$rootScope.$apply(() => {
|
|
218
|
+
$rootScope.variable = null;
|
|
219
|
+
});
|
|
220
|
+
|
|
221
|
+
expect(element.text()).not.toContain("Message is crazy");
|
|
222
|
+
});
|
|
223
|
+
|
|
224
|
+
// they(
|
|
225
|
+
// "should render empty when $prop is used as a collection value",
|
|
226
|
+
// {
|
|
227
|
+
// null: null,
|
|
228
|
+
// false: false,
|
|
229
|
+
// 0: 0,
|
|
230
|
+
// "[]": [],
|
|
231
|
+
// "[{}]": [{}],
|
|
232
|
+
// "": "",
|
|
233
|
+
// "{ val2 : true }": { val2: true },
|
|
234
|
+
// },
|
|
235
|
+
// (prop) => {
|
|
236
|
+
// () => {
|
|
237
|
+
// element = $compile(
|
|
238
|
+
// '<div ng-messages="col">' +
|
|
239
|
+
// ' <div ng-message="val">Message is set</div>' +
|
|
240
|
+
// "</div>",
|
|
241
|
+
// )($rootScope);
|
|
242
|
+
// $rootScope.$digest();
|
|
243
|
+
|
|
244
|
+
// $rootScope.$apply(() => {
|
|
245
|
+
// $rootScope.col = prop;
|
|
246
|
+
// });
|
|
247
|
+
// expect(element.text()).not.toContain("Message is set");
|
|
248
|
+
// });
|
|
249
|
+
// },
|
|
250
|
+
// );
|
|
251
|
+
|
|
252
|
+
// they(
|
|
253
|
+
// "should insert and remove matching inner elements when $prop is used as a value",
|
|
254
|
+
// { true: true, 1: 1, "{}": {}, "[]": [], "[null]": [null] },
|
|
255
|
+
// (prop) => {
|
|
256
|
+
// () => {
|
|
257
|
+
// element = $compile(
|
|
258
|
+
// '<div ng-messages="col">' +
|
|
259
|
+
// ' <div ng-message="blue">This message is blue</div>' +
|
|
260
|
+
// ' <div ng-message="red">This message is red</div>' +
|
|
261
|
+
// "</div>",
|
|
262
|
+
// )($rootScope);
|
|
263
|
+
|
|
264
|
+
// $rootScope.$apply(() => {
|
|
265
|
+
// $rootScope.col = {};
|
|
266
|
+
// });
|
|
267
|
+
|
|
268
|
+
// expect(messageChildren(element).length).toBe(0);
|
|
269
|
+
// expect(trim(element.text())).toEqual("");
|
|
270
|
+
|
|
271
|
+
// $rootScope.$apply(() => {
|
|
272
|
+
// $rootScope.col = {
|
|
273
|
+
// blue: true,
|
|
274
|
+
// red: false,
|
|
275
|
+
// };
|
|
276
|
+
// });
|
|
277
|
+
|
|
278
|
+
// expect(messageChildren(element).length).toBe(1);
|
|
279
|
+
// expect(trim(element.text())).toEqual("This message is blue");
|
|
280
|
+
|
|
281
|
+
// $rootScope.$apply(() => {
|
|
282
|
+
// $rootScope.col = {
|
|
283
|
+
// red: prop,
|
|
284
|
+
// };
|
|
285
|
+
// });
|
|
286
|
+
|
|
287
|
+
// expect(messageChildren(element).length).toBe(1);
|
|
288
|
+
// expect(trim(element.text())).toEqual("This message is red");
|
|
289
|
+
|
|
290
|
+
// $rootScope.$apply(() => {
|
|
291
|
+
// $rootScope.col = null;
|
|
292
|
+
// });
|
|
293
|
+
// expect(messageChildren(element).length).toBe(0);
|
|
294
|
+
// expect(trim(element.text())).toEqual("");
|
|
295
|
+
|
|
296
|
+
// $rootScope.$apply(() => {
|
|
297
|
+
// $rootScope.col = {
|
|
298
|
+
// blue: 0,
|
|
299
|
+
// red: null,
|
|
300
|
+
// };
|
|
301
|
+
// });
|
|
302
|
+
|
|
303
|
+
// expect(messageChildren(element).length).toBe(0);
|
|
304
|
+
// expect(trim(element.text())).toEqual("");
|
|
305
|
+
// });
|
|
306
|
+
// },
|
|
307
|
+
// );
|
|
308
|
+
|
|
309
|
+
it("should display the elements in the order defined in the DOM", () => {
|
|
310
|
+
element = $compile(
|
|
311
|
+
'<div ng-messages="col">' +
|
|
312
|
+
' <div ng-message="one">Message#one</div>' +
|
|
313
|
+
' <div ng-message="two">Message#two</div>' +
|
|
314
|
+
' <div ng-message="three">Message#three</div>' +
|
|
315
|
+
"</div>",
|
|
316
|
+
)($rootScope);
|
|
317
|
+
|
|
318
|
+
$rootScope.$apply(() => {
|
|
319
|
+
$rootScope.col = {
|
|
320
|
+
three: true,
|
|
321
|
+
one: true,
|
|
322
|
+
two: true,
|
|
323
|
+
};
|
|
324
|
+
});
|
|
325
|
+
|
|
326
|
+
["one", "two", "three"].forEach((key) => {
|
|
327
|
+
expect(s(element.text())).toEqual(`Message#${key}`);
|
|
328
|
+
|
|
329
|
+
$rootScope.$apply(() => {
|
|
330
|
+
$rootScope.col[key] = false;
|
|
331
|
+
});
|
|
332
|
+
});
|
|
333
|
+
|
|
334
|
+
expect(s(element.text())).toEqual("");
|
|
335
|
+
});
|
|
336
|
+
|
|
337
|
+
it("should add ng-active/ng-inactive CSS classes to the element when errors are/aren't displayed", () => {
|
|
338
|
+
element = $compile(
|
|
339
|
+
'<div ng-messages="col">' +
|
|
340
|
+
' <div ng-message="ready">This message is ready</div>' +
|
|
341
|
+
"</div>",
|
|
342
|
+
)($rootScope);
|
|
343
|
+
|
|
344
|
+
$rootScope.$apply(() => {
|
|
345
|
+
$rootScope.col = {};
|
|
346
|
+
});
|
|
347
|
+
|
|
348
|
+
expect(element[0].classList.contains("ng-active")).toBe(false);
|
|
349
|
+
expect(element[0].classList.contains("ng-inactive")).toBe(true);
|
|
350
|
+
|
|
351
|
+
$rootScope.$apply(() => {
|
|
352
|
+
$rootScope.col = { ready: true };
|
|
353
|
+
});
|
|
354
|
+
|
|
355
|
+
expect(element[0].classList.contains("ng-active")).toBe(true);
|
|
356
|
+
expect(element[0].classList.contains("ng-inactive")).toBe(false);
|
|
357
|
+
});
|
|
358
|
+
|
|
359
|
+
it("should automatically re-render the messages when other directives dynamically change them", () => {
|
|
360
|
+
element = $compile(
|
|
361
|
+
'<div ng-messages="col">' +
|
|
362
|
+
' <div ng-message="primary">Enter something</div>' +
|
|
363
|
+
' <div ng-repeat="item in items">' +
|
|
364
|
+
' <div ng-message-exp="item.name">{{ item.text }}</div>' +
|
|
365
|
+
" </div>" +
|
|
366
|
+
"</div>",
|
|
367
|
+
)($rootScope);
|
|
368
|
+
|
|
369
|
+
$rootScope.$apply(() => {
|
|
370
|
+
$rootScope.col = {};
|
|
371
|
+
$rootScope.items = [
|
|
372
|
+
{ text: "Your age is incorrect", name: "age" },
|
|
373
|
+
{ text: "You're too tall man!", name: "height" },
|
|
374
|
+
{ text: "Your hair is too long", name: "hair" },
|
|
375
|
+
];
|
|
376
|
+
});
|
|
377
|
+
|
|
378
|
+
expect(messageChildren(element).length).toBe(0);
|
|
379
|
+
expect(trim(element.text())).toEqual("");
|
|
380
|
+
|
|
381
|
+
$rootScope.$apply(() => {
|
|
382
|
+
$rootScope.col = { hair: true };
|
|
383
|
+
});
|
|
384
|
+
|
|
385
|
+
expect(messageChildren(element).length).toBe(1);
|
|
386
|
+
expect(trim(element.text())).toEqual("Your hair is too long");
|
|
387
|
+
|
|
388
|
+
$rootScope.$apply(() => {
|
|
389
|
+
$rootScope.col = { age: true, hair: true };
|
|
390
|
+
});
|
|
391
|
+
|
|
392
|
+
expect(messageChildren(element).length).toBe(1);
|
|
393
|
+
expect(trim(element.text())).toEqual("Your age is incorrect");
|
|
394
|
+
|
|
395
|
+
$rootScope.$apply(() => {
|
|
396
|
+
// remove the age!
|
|
397
|
+
$rootScope.items.shift();
|
|
398
|
+
});
|
|
399
|
+
|
|
400
|
+
expect(messageChildren(element).length).toBe(1);
|
|
401
|
+
expect(trim(element.text())).toEqual("Your hair is too long");
|
|
402
|
+
|
|
403
|
+
$rootScope.$apply(() => {
|
|
404
|
+
// remove the hair!
|
|
405
|
+
$rootScope.items.length = 0;
|
|
406
|
+
$rootScope.col.primary = true;
|
|
407
|
+
});
|
|
408
|
+
|
|
409
|
+
expect(messageChildren(element).length).toBe(1);
|
|
410
|
+
expect(trim(element.text())).toEqual("Enter something");
|
|
411
|
+
});
|
|
412
|
+
|
|
413
|
+
it("should be compatible with ngBind", () => {
|
|
414
|
+
element = $compile(
|
|
415
|
+
'<div ng-messages="col">' +
|
|
416
|
+
' <div ng-message="required" ng-bind="errorMessages.required"></div>' +
|
|
417
|
+
' <div ng-message="extra" ng-bind="errorMessages.extra"></div>' +
|
|
418
|
+
"</div>",
|
|
419
|
+
)($rootScope);
|
|
420
|
+
|
|
421
|
+
$rootScope.$apply(() => {
|
|
422
|
+
$rootScope.col = {
|
|
423
|
+
required: true,
|
|
424
|
+
extra: true,
|
|
425
|
+
};
|
|
426
|
+
$rootScope.errorMessages = {
|
|
427
|
+
required: "Fill in the text field.",
|
|
428
|
+
extra: "Extra error message.",
|
|
429
|
+
};
|
|
430
|
+
});
|
|
431
|
+
|
|
432
|
+
expect(messageChildren(element).length).toBe(1);
|
|
433
|
+
expect(trim(element.text())).toEqual("Fill in the text field.");
|
|
434
|
+
|
|
435
|
+
$rootScope.$apply(() => {
|
|
436
|
+
$rootScope.col.required = false;
|
|
437
|
+
$rootScope.col.extra = true;
|
|
438
|
+
});
|
|
439
|
+
|
|
440
|
+
expect(messageChildren(element).length).toBe(1);
|
|
441
|
+
expect(trim(element.text())).toEqual("Extra error message.");
|
|
442
|
+
|
|
443
|
+
$rootScope.$apply(() => {
|
|
444
|
+
$rootScope.errorMessages.extra = "New error message.";
|
|
445
|
+
});
|
|
446
|
+
|
|
447
|
+
expect(messageChildren(element).length).toBe(1);
|
|
448
|
+
expect(trim(element.text())).toEqual("New error message.");
|
|
449
|
+
});
|
|
450
|
+
|
|
451
|
+
// issue #12856
|
|
452
|
+
it("should only detach the message object that is associated with the message node being removed", () => {
|
|
453
|
+
// We are going to spy on the `leave` method to give us control over
|
|
454
|
+
// when the element is actually removed
|
|
455
|
+
//spyOn($animate, "leave");
|
|
456
|
+
|
|
457
|
+
// Create a basic ng-messages set up
|
|
458
|
+
element = $compile(
|
|
459
|
+
'<div ng-messages="col">' +
|
|
460
|
+
' <div ng-message="primary">Enter something</div>' +
|
|
461
|
+
"</div>",
|
|
462
|
+
)($rootScope);
|
|
463
|
+
|
|
464
|
+
// Trigger the message to be displayed
|
|
465
|
+
$rootScope.col = { primary: true };
|
|
466
|
+
$rootScope.$digest();
|
|
467
|
+
expect(messageChildren(element).length).toEqual(1);
|
|
468
|
+
const oldMessageNode = messageChildren(element)[0];
|
|
469
|
+
|
|
470
|
+
// Remove the message
|
|
471
|
+
$rootScope.col = { primary: undefined };
|
|
472
|
+
$rootScope.$digest();
|
|
473
|
+
|
|
474
|
+
// Since we have spied on the `leave` method, the message node is still in the DOM
|
|
475
|
+
//expect($animate.leave).toHaveBeenCalled();
|
|
476
|
+
// const nodeToRemove = $animate.leave.calls.mostRecent().args[0][0];
|
|
477
|
+
// expect(nodeToRemove).toBe(oldMessageNode);
|
|
478
|
+
|
|
479
|
+
// Add the message back in
|
|
480
|
+
$rootScope.col = { primary: true };
|
|
481
|
+
$rootScope.$digest();
|
|
482
|
+
|
|
483
|
+
// Simulate the animation completing on the node
|
|
484
|
+
// jqLite(nodeToRemove).remove();
|
|
485
|
+
|
|
486
|
+
// We should not get another call to `leave`
|
|
487
|
+
//expect($animate.leave).not.toHaveBeenCalled();
|
|
488
|
+
|
|
489
|
+
// There should only be the new message node
|
|
490
|
+
expect(messageChildren(element).length).toEqual(1);
|
|
491
|
+
const newMessageNode = messageChildren(element)[0];
|
|
492
|
+
expect(newMessageNode).not.toBe(oldMessageNode);
|
|
493
|
+
});
|
|
494
|
+
|
|
495
|
+
// it("should render animations when the active/inactive classes are added/removed", () => {
|
|
496
|
+
// // module("ngAnimate");
|
|
497
|
+
// // module("ngAnimateMock");
|
|
498
|
+
// element = $compile(
|
|
499
|
+
// '<div ng-messages="col">' +
|
|
500
|
+
// ' <div ng-message="ready">This message is ready</div>' +
|
|
501
|
+
// "</div>",
|
|
502
|
+
// )($rootScope);
|
|
503
|
+
|
|
504
|
+
// $rootScope.$apply(() => {
|
|
505
|
+
// $rootScope.col = {};
|
|
506
|
+
// });
|
|
507
|
+
|
|
508
|
+
// let event = $animate.queue.pop();
|
|
509
|
+
// expect(event.event).toBe("setClass");
|
|
510
|
+
// expect(event.args[1]).toBe("ng-inactive");
|
|
511
|
+
// expect(event.args[2]).toBe("ng-active");
|
|
512
|
+
|
|
513
|
+
// $rootScope.$apply(() => {
|
|
514
|
+
// $rootScope.col = { ready: true };
|
|
515
|
+
// });
|
|
516
|
+
|
|
517
|
+
// event = $animate.queue.pop();
|
|
518
|
+
// expect(event.event).toBe("setClass");
|
|
519
|
+
// expect(event.args[1]).toBe("ng-active");
|
|
520
|
+
// expect(event.args[2]).toBe("ng-inactive");
|
|
521
|
+
// });
|
|
522
|
+
|
|
523
|
+
describe("ngMessage nested nested inside elements", () => {
|
|
524
|
+
it(
|
|
525
|
+
"should not crash or leak memory when the messages are transcluded, the first message is " +
|
|
526
|
+
"visible, and ngMessages is removed by ngIf",
|
|
527
|
+
() => {
|
|
528
|
+
element = $compile(
|
|
529
|
+
'<div><div ng-if="show"><div message-wrap col="col">' +
|
|
530
|
+
' <div ng-message="a">A</div>' +
|
|
531
|
+
' <div ng-message="b">B</div>' +
|
|
532
|
+
"</div></div></div>",
|
|
533
|
+
)($rootScope);
|
|
534
|
+
|
|
535
|
+
$rootScope.$apply(() => {
|
|
536
|
+
$rootScope.show = true;
|
|
537
|
+
$rootScope.col = {
|
|
538
|
+
a: true,
|
|
539
|
+
b: true,
|
|
540
|
+
};
|
|
541
|
+
});
|
|
542
|
+
|
|
543
|
+
expect(messageChildren(element).length).toBe(1);
|
|
544
|
+
expect(trim(element.text())).toEqual("A");
|
|
545
|
+
|
|
546
|
+
$rootScope.$apply("show = false");
|
|
547
|
+
|
|
548
|
+
expect(messageChildren(element).length).toBe(0);
|
|
549
|
+
},
|
|
550
|
+
);
|
|
551
|
+
|
|
552
|
+
it("should not crash when the first of two nested messages is removed", () => {
|
|
553
|
+
element = $compile(
|
|
554
|
+
'<div ng-messages="col">' +
|
|
555
|
+
'<div class="wrapper">' +
|
|
556
|
+
'<div remove-me ng-message="a">A</div>' +
|
|
557
|
+
'<div ng-message="b">B</div>' +
|
|
558
|
+
"</div>" +
|
|
559
|
+
"</div>",
|
|
560
|
+
)($rootScope);
|
|
561
|
+
|
|
562
|
+
$rootScope.$apply(() => {
|
|
563
|
+
$rootScope.col = {
|
|
564
|
+
a: true,
|
|
565
|
+
b: false,
|
|
566
|
+
};
|
|
567
|
+
});
|
|
568
|
+
|
|
569
|
+
expect(messageChildren(element).length).toBe(1);
|
|
570
|
+
expect(trim(element.text())).toEqual("A");
|
|
571
|
+
|
|
572
|
+
const ctrl = element.controller("ngMessages");
|
|
573
|
+
const deregisterSpy = spyOn(ctrl, "deregister").and.callThrough();
|
|
574
|
+
|
|
575
|
+
const nodeA = element[0].querySelector('[ng-message="a"]');
|
|
576
|
+
jqLite(nodeA).remove();
|
|
577
|
+
$rootScope.$digest(); // The next digest triggers the error
|
|
578
|
+
|
|
579
|
+
// Make sure removing the element triggers the deregistration in ngMessages
|
|
580
|
+
expect(trim(deregisterSpy.calls.mostRecent().args[0].nodeValue)).toBe(
|
|
581
|
+
"ngMessage: a",
|
|
582
|
+
);
|
|
583
|
+
expect(messageChildren(element).length).toBe(0);
|
|
584
|
+
});
|
|
585
|
+
|
|
586
|
+
it(
|
|
587
|
+
"should not crash, but show deeply nested messages correctly after a message " +
|
|
588
|
+
"has been removed",
|
|
589
|
+
() => {
|
|
590
|
+
element = $compile(
|
|
591
|
+
'<div ng-messages="col" ng-messages-multiple>' +
|
|
592
|
+
'<div class="another-wrapper">' +
|
|
593
|
+
'<div ng-message="a">A</div>' +
|
|
594
|
+
'<div class="wrapper">' +
|
|
595
|
+
'<div ng-message="b">B</div>' +
|
|
596
|
+
'<div ng-message="c">C</div>' +
|
|
597
|
+
"</div>" +
|
|
598
|
+
'<div ng-message="d">D</div>' +
|
|
599
|
+
"</div>" +
|
|
600
|
+
"</div>",
|
|
601
|
+
)($rootScope);
|
|
602
|
+
|
|
603
|
+
$rootScope.$apply(() => {
|
|
604
|
+
$rootScope.col = {
|
|
605
|
+
a: true,
|
|
606
|
+
b: true,
|
|
607
|
+
};
|
|
608
|
+
});
|
|
609
|
+
|
|
610
|
+
expect(messageChildren(element).length).toBe(2);
|
|
611
|
+
expect(trim(element.text())).toEqual("AB");
|
|
612
|
+
|
|
613
|
+
const ctrl = element.controller("ngMessages");
|
|
614
|
+
const deregisterSpy = spyOn(ctrl, "deregister").and.callThrough();
|
|
615
|
+
|
|
616
|
+
const nodeB = element[0].querySelector('[ng-message="b"]');
|
|
617
|
+
jqLite(nodeB).remove();
|
|
618
|
+
$rootScope.$digest(); // The next digest triggers the error
|
|
619
|
+
|
|
620
|
+
// Make sure removing the element triggers the deregistration in ngMessages
|
|
621
|
+
expect(trim(deregisterSpy.calls.mostRecent().args[0].nodeValue)).toBe(
|
|
622
|
+
"ngMessage: b",
|
|
623
|
+
);
|
|
624
|
+
expect(messageChildren(element).length).toBe(1);
|
|
625
|
+
expect(trim(element.text())).toEqual("A");
|
|
626
|
+
},
|
|
627
|
+
);
|
|
628
|
+
});
|
|
629
|
+
|
|
630
|
+
it("should clean-up the ngMessage scope when a message is removed", () => {
|
|
631
|
+
const html =
|
|
632
|
+
'<div ng-messages="items">' +
|
|
633
|
+
'<div ng-message="a">{{forA}}</div>' +
|
|
634
|
+
"</div>";
|
|
635
|
+
|
|
636
|
+
element = $compile(html)($rootScope);
|
|
637
|
+
$rootScope.$apply(() => {
|
|
638
|
+
$rootScope.forA = "A";
|
|
639
|
+
$rootScope.items = { a: true };
|
|
640
|
+
});
|
|
641
|
+
|
|
642
|
+
expect(element.text()).toBe("A");
|
|
643
|
+
const watchers = countWatchers($rootScope);
|
|
644
|
+
|
|
645
|
+
$rootScope.$apply("items.a = false");
|
|
646
|
+
|
|
647
|
+
expect(element.text()).toBe("");
|
|
648
|
+
// We don't know exactly how many watchers are on the scope, only that there should be
|
|
649
|
+
// one less now
|
|
650
|
+
expect(countWatchers($rootScope)).toBe(watchers - 1);
|
|
651
|
+
});
|
|
652
|
+
|
|
653
|
+
it("should unregister the ngMessage even if it was never attached", () => {
|
|
654
|
+
const html =
|
|
655
|
+
'<div ng-messages="items">' +
|
|
656
|
+
'<div ng-if="show"><div ng-message="x">ERROR</div></div>' +
|
|
657
|
+
"</div>";
|
|
658
|
+
|
|
659
|
+
element = $compile(html)($rootScope);
|
|
660
|
+
|
|
661
|
+
const ctrl = element.controller("ngMessages");
|
|
662
|
+
|
|
663
|
+
expect(messageChildren(element).length).toBe(0);
|
|
664
|
+
expect(Object.keys(ctrl.messages).length).toEqual(0);
|
|
665
|
+
|
|
666
|
+
$rootScope.$apply("show = true");
|
|
667
|
+
expect(messageChildren(element).length).toBe(0);
|
|
668
|
+
expect(Object.keys(ctrl.messages).length).toEqual(1);
|
|
669
|
+
|
|
670
|
+
$rootScope.$apply("show = false");
|
|
671
|
+
expect(messageChildren(element).length).toBe(0);
|
|
672
|
+
expect(Object.keys(ctrl.messages).length).toEqual(0);
|
|
673
|
+
});
|
|
674
|
+
|
|
675
|
+
describe("default message", () => {
|
|
676
|
+
it("should render a default message when no message matches", () => {
|
|
677
|
+
element = $compile(
|
|
678
|
+
'<div ng-messages="col">' +
|
|
679
|
+
' <div ng-message="val">Message is set</div>' +
|
|
680
|
+
" <div ng-message-default>Default message is set</div>" +
|
|
681
|
+
"</div>",
|
|
682
|
+
)($rootScope);
|
|
683
|
+
$rootScope.$apply(() => {
|
|
684
|
+
$rootScope.col = { unexpected: false };
|
|
685
|
+
});
|
|
686
|
+
|
|
687
|
+
$rootScope.$digest();
|
|
688
|
+
|
|
689
|
+
expect(element.text().trim()).toBe("");
|
|
690
|
+
expect(element[0].classList.contains("ng-active")).toBeFalse();
|
|
691
|
+
|
|
692
|
+
$rootScope.$apply(() => {
|
|
693
|
+
$rootScope.col = { unexpected: true };
|
|
694
|
+
});
|
|
695
|
+
|
|
696
|
+
expect(element.text().trim()).toBe("Default message is set");
|
|
697
|
+
expect(element[0].classList.contains("ng-active")).toBeTrue();
|
|
698
|
+
|
|
699
|
+
$rootScope.$apply(() => {
|
|
700
|
+
$rootScope.col = { unexpected: false };
|
|
701
|
+
});
|
|
702
|
+
|
|
703
|
+
expect(element.text().trim()).toBe("");
|
|
704
|
+
expect(element[0].classList.contains("ng-active")).toBeFalse();
|
|
705
|
+
|
|
706
|
+
$rootScope.$apply(() => {
|
|
707
|
+
$rootScope.col = { val: true, unexpected: true };
|
|
708
|
+
});
|
|
709
|
+
|
|
710
|
+
expect(element.text().trim()).toBe("Message is set");
|
|
711
|
+
expect(element[0].classList.contains("ng-active")).toBeTrue();
|
|
712
|
+
});
|
|
713
|
+
|
|
714
|
+
it("should not render a default message with ng-messages-multiple if another error matches", () => {
|
|
715
|
+
element = $compile(
|
|
716
|
+
'<div ng-messages="col" ng-messages-multiple>' +
|
|
717
|
+
' <div ng-message="val">Message is set</div>' +
|
|
718
|
+
' <div ng-message="other">Other message is set</div>' +
|
|
719
|
+
" <div ng-message-default>Default message is set</div>" +
|
|
720
|
+
"</div>",
|
|
721
|
+
)($rootScope);
|
|
722
|
+
|
|
723
|
+
expect(element.text().trim()).toBe("");
|
|
724
|
+
|
|
725
|
+
$rootScope.$apply(() => {
|
|
726
|
+
$rootScope.col = { val: true, other: false, unexpected: false };
|
|
727
|
+
});
|
|
728
|
+
|
|
729
|
+
expect(element.text().trim()).toBe("Message is set");
|
|
730
|
+
|
|
731
|
+
$rootScope.$apply(() => {
|
|
732
|
+
$rootScope.col = { val: true, other: true, unexpected: true };
|
|
733
|
+
});
|
|
734
|
+
|
|
735
|
+
expect(element.text().trim()).toBe(
|
|
736
|
+
"Message is set Other message is set",
|
|
737
|
+
);
|
|
738
|
+
|
|
739
|
+
$rootScope.$apply(() => {
|
|
740
|
+
$rootScope.col = { val: false, other: false, unexpected: true };
|
|
741
|
+
});
|
|
742
|
+
|
|
743
|
+
expect(element.text().trim()).toBe("Default message is set");
|
|
744
|
+
});
|
|
745
|
+
|
|
746
|
+
it("should handle a default message with ngIf", () => {
|
|
747
|
+
element = $compile(
|
|
748
|
+
'<div ng-messages="col">' +
|
|
749
|
+
' <div ng-message="val">Message is set</div>' +
|
|
750
|
+
' <div ng-if="default" ng-message-default>Default message is set</div>' +
|
|
751
|
+
"</div>",
|
|
752
|
+
)($rootScope);
|
|
753
|
+
$rootScope.default = true;
|
|
754
|
+
$rootScope.col = { unexpected: true };
|
|
755
|
+
$rootScope.$digest();
|
|
756
|
+
|
|
757
|
+
expect(element.text().trim()).toBe("Default message is set");
|
|
758
|
+
|
|
759
|
+
$rootScope.$apply("default = false");
|
|
760
|
+
|
|
761
|
+
expect(element.text().trim()).toBe("");
|
|
762
|
+
|
|
763
|
+
$rootScope.$apply("default = true");
|
|
764
|
+
|
|
765
|
+
expect(element.text().trim()).toBe("Default message is set");
|
|
766
|
+
|
|
767
|
+
$rootScope.$apply(() => {
|
|
768
|
+
$rootScope.col = { val: true };
|
|
769
|
+
});
|
|
770
|
+
|
|
771
|
+
expect(element.text().trim()).toBe("Message is set");
|
|
772
|
+
});
|
|
773
|
+
});
|
|
774
|
+
|
|
775
|
+
describe("when including templates", () => {
|
|
776
|
+
// they(
|
|
777
|
+
// "should work with a dynamic collection model which is managed by ngRepeat",
|
|
778
|
+
// {
|
|
779
|
+
// '<div ng-messages-include="...">':
|
|
780
|
+
// '<div ng-messages="item">' +
|
|
781
|
+
// '<div ng-messages-include="abc.html"></div>' +
|
|
782
|
+
// "</div>",
|
|
783
|
+
// '<ng-messages-include src="...">':
|
|
784
|
+
// '<ng-messages for="item">' +
|
|
785
|
+
// '<ng-messages-include src="abc.html"></ng-messages-include>' +
|
|
786
|
+
// "</ng-messages>",
|
|
787
|
+
// },
|
|
788
|
+
// (html) => {
|
|
789
|
+
// inject(($compile, $rootScope, $templateCache) => {
|
|
790
|
+
// $templateCache.put(
|
|
791
|
+
// "abc.html",
|
|
792
|
+
// '<div ng-message="a">A</div>' +
|
|
793
|
+
// '<div ng-message="b">B</div>' +
|
|
794
|
+
// '<div ng-message="c">C</div>',
|
|
795
|
+
// );
|
|
796
|
+
|
|
797
|
+
// html = `<div><div ng-repeat="item in items">${html}</div></div>`;
|
|
798
|
+
// $rootScope.items = [{}, {}, {}];
|
|
799
|
+
|
|
800
|
+
// element = $compile(html)($rootScope);
|
|
801
|
+
// $rootScope.$apply(() => {
|
|
802
|
+
// $rootScope.items[0].a = true;
|
|
803
|
+
// $rootScope.items[1].b = true;
|
|
804
|
+
// $rootScope.items[2].c = true;
|
|
805
|
+
// });
|
|
806
|
+
|
|
807
|
+
// const elements = element[0].querySelectorAll("[ng-repeat]");
|
|
808
|
+
|
|
809
|
+
// // all three collections should have at least one error showing up
|
|
810
|
+
// expect(messageChildren(element).length).toBe(3);
|
|
811
|
+
// expect(messageChildren(elements[0]).length).toBe(1);
|
|
812
|
+
// expect(messageChildren(elements[1]).length).toBe(1);
|
|
813
|
+
// expect(messageChildren(elements[2]).length).toBe(1);
|
|
814
|
+
|
|
815
|
+
// // this is the standard order of the displayed error messages
|
|
816
|
+
// expect(element.text().trim()).toBe("ABC");
|
|
817
|
+
|
|
818
|
+
// $rootScope.$apply(() => {
|
|
819
|
+
// $rootScope.items[0].a = false;
|
|
820
|
+
// $rootScope.items[0].c = true;
|
|
821
|
+
|
|
822
|
+
// $rootScope.items[1].b = false;
|
|
823
|
+
|
|
824
|
+
// $rootScope.items[2].c = false;
|
|
825
|
+
// $rootScope.items[2].a = true;
|
|
826
|
+
// });
|
|
827
|
+
|
|
828
|
+
// // with the 2nd item gone and the values changed
|
|
829
|
+
// // we should see both 1 and 3 changed
|
|
830
|
+
// expect(element.text().trim()).toBe("CA");
|
|
831
|
+
|
|
832
|
+
// $rootScope.$apply(() => {
|
|
833
|
+
// // add the value for the 2nd item back
|
|
834
|
+
// $rootScope.items[1].b = true;
|
|
835
|
+
// $rootScope.items.reverse();
|
|
836
|
+
// });
|
|
837
|
+
|
|
838
|
+
// // when reversed we get back to our original value
|
|
839
|
+
// expect(element.text().trim()).toBe("ABC");
|
|
840
|
+
// });
|
|
841
|
+
// },
|
|
842
|
+
// );
|
|
843
|
+
|
|
844
|
+
// they(
|
|
845
|
+
// "should remove the $prop element and place a comment anchor node where it used to be",
|
|
846
|
+
// {
|
|
847
|
+
// '<div ng-messages-include="...">':
|
|
848
|
+
// '<div ng-messages="data">' +
|
|
849
|
+
// '<div ng-messages-include="abc.html"></div>' +
|
|
850
|
+
// "</div>",
|
|
851
|
+
// '<ng-messages-include src="...">':
|
|
852
|
+
// '<ng-messages for="data">' +
|
|
853
|
+
// '<ng-messages-include src="abc.html"></ng-messages-include>' +
|
|
854
|
+
// "</ng-messages>",
|
|
855
|
+
// },
|
|
856
|
+
// (html) => {
|
|
857
|
+
// inject(($compile, $rootScope, $templateCache) => {
|
|
858
|
+
// $templateCache.put("abc.html", "<div></div>");
|
|
859
|
+
|
|
860
|
+
// element = $compile(html)($rootScope);
|
|
861
|
+
// $rootScope.$digest();
|
|
862
|
+
|
|
863
|
+
// const includeElement = element[0].querySelector(
|
|
864
|
+
// "[ng-messages-include], ng-messages-include",
|
|
865
|
+
// );
|
|
866
|
+
// expect(includeElement).toBeFalsy();
|
|
867
|
+
|
|
868
|
+
// const comment = element[0].childNodes[0];
|
|
869
|
+
// expect(comment.nodeType).toBe(8);
|
|
870
|
+
// expect(comment.nodeValue).toBe(" ngMessagesInclude: abc.html ");
|
|
871
|
+
// });
|
|
872
|
+
// },
|
|
873
|
+
// );
|
|
874
|
+
|
|
875
|
+
// they(
|
|
876
|
+
// "should load a remote template using $prop",
|
|
877
|
+
// {
|
|
878
|
+
// '<div ng-messages-include="...">':
|
|
879
|
+
// '<div ng-messages="data">' +
|
|
880
|
+
// '<div ng-messages-include="abc.html"></div>' +
|
|
881
|
+
// "</div>",
|
|
882
|
+
// '<ng-messages-include src="...">':
|
|
883
|
+
// '<ng-messages for="data">' +
|
|
884
|
+
// '<ng-messages-include src="abc.html"></ng-messages-include>' +
|
|
885
|
+
// "</ng-messages>",
|
|
886
|
+
// },
|
|
887
|
+
// (html) => {
|
|
888
|
+
// inject(($compile, $rootScope, $templateCache) => {
|
|
889
|
+
// $templateCache.put(
|
|
890
|
+
// "abc.html",
|
|
891
|
+
// '<div ng-message="a">A</div>' +
|
|
892
|
+
// '<div ng-message="b">B</div>' +
|
|
893
|
+
// '<div ng-message="c">C</div>',
|
|
894
|
+
// );
|
|
895
|
+
|
|
896
|
+
// element = $compile(html)($rootScope);
|
|
897
|
+
// $rootScope.$apply(() => {
|
|
898
|
+
// $rootScope.data = {
|
|
899
|
+
// a: 1,
|
|
900
|
+
// b: 2,
|
|
901
|
+
// c: 3,
|
|
902
|
+
// };
|
|
903
|
+
// });
|
|
904
|
+
|
|
905
|
+
// expect(messageChildren(element).length).toBe(1);
|
|
906
|
+
// expect(trim(element.text())).toEqual("A");
|
|
907
|
+
|
|
908
|
+
// $rootScope.$apply(() => {
|
|
909
|
+
// $rootScope.data = {
|
|
910
|
+
// c: 3,
|
|
911
|
+
// };
|
|
912
|
+
// });
|
|
913
|
+
|
|
914
|
+
// expect(messageChildren(element).length).toBe(1);
|
|
915
|
+
// expect(trim(element.text())).toEqual("C");
|
|
916
|
+
// });
|
|
917
|
+
// },
|
|
918
|
+
// );
|
|
919
|
+
|
|
920
|
+
it("should cache the template after download", () => {
|
|
921
|
+
expect($templateCache.get("/mock/hello")).toBeUndefined();
|
|
922
|
+
element = $compile(
|
|
923
|
+
'<div ng-messages="data"><div ng-messages-include="/mock/hello"></div></div>',
|
|
924
|
+
)($rootScope);
|
|
925
|
+
|
|
926
|
+
$rootScope.$digest();
|
|
927
|
+
expect($templateCache.get("/mock/hello")).toBeDefined();
|
|
928
|
+
});
|
|
929
|
+
|
|
930
|
+
it("should re-render the messages after download without an extra digest", (done) => {
|
|
931
|
+
element = $compile(
|
|
932
|
+
'<div ng-messages="data">' +
|
|
933
|
+
' <div ng-messages-include="/mock/my-messages"></div>' +
|
|
934
|
+
' <div ng-message="failed">Your value is that of failure</div>' +
|
|
935
|
+
"</div>",
|
|
936
|
+
)($rootScope);
|
|
937
|
+
|
|
938
|
+
$rootScope.data = {
|
|
939
|
+
required: true,
|
|
940
|
+
failed: true,
|
|
941
|
+
};
|
|
942
|
+
|
|
943
|
+
$rootScope.$digest();
|
|
944
|
+
|
|
945
|
+
expect(messageChildren(element).length).toBe(1);
|
|
946
|
+
expect(trim(element.text())).toEqual("Your value is that of failure");
|
|
947
|
+
|
|
948
|
+
$rootScope.$digest();
|
|
949
|
+
setTimeout(() => {
|
|
950
|
+
expect(messageChildren(element).length).toBe(1);
|
|
951
|
+
expect(trim(element.text())).toEqual("You did not enter a value");
|
|
952
|
+
done();
|
|
953
|
+
}, 10);
|
|
954
|
+
});
|
|
955
|
+
|
|
956
|
+
it("should allow for overriding the remote template messages within the element depending on where the remote template is placed", () => {
|
|
957
|
+
$templateCache.put(
|
|
958
|
+
"abc.html",
|
|
959
|
+
'<div ng-message="a">A</div>' +
|
|
960
|
+
'<div ng-message="b">B</div>' +
|
|
961
|
+
'<div ng-message="c">C</div>',
|
|
962
|
+
);
|
|
963
|
+
|
|
964
|
+
element = $compile(
|
|
965
|
+
'<div ng-messages="data">' +
|
|
966
|
+
' <div ng-message="a">AAA</div>' +
|
|
967
|
+
' <div ng-messages-include="abc.html"></div>' +
|
|
968
|
+
' <div ng-message="c">CCC</div>' +
|
|
969
|
+
"</div>",
|
|
970
|
+
)($rootScope);
|
|
971
|
+
|
|
972
|
+
$rootScope.$apply(() => {
|
|
973
|
+
$rootScope.data = {
|
|
974
|
+
a: 1,
|
|
975
|
+
b: 2,
|
|
976
|
+
c: 3,
|
|
977
|
+
};
|
|
978
|
+
});
|
|
979
|
+
|
|
980
|
+
expect(messageChildren(element).length).toBe(1);
|
|
981
|
+
expect(trim(element.text())).toEqual("AAA");
|
|
982
|
+
|
|
983
|
+
$rootScope.$apply(() => {
|
|
984
|
+
$rootScope.data = {
|
|
985
|
+
b: 2,
|
|
986
|
+
c: 3,
|
|
987
|
+
};
|
|
988
|
+
});
|
|
989
|
+
|
|
990
|
+
expect(messageChildren(element).length).toBe(1);
|
|
991
|
+
expect(trim(element.text())).toEqual("B");
|
|
992
|
+
|
|
993
|
+
$rootScope.$apply(() => {
|
|
994
|
+
$rootScope.data = {
|
|
995
|
+
c: 3,
|
|
996
|
+
};
|
|
997
|
+
});
|
|
998
|
+
|
|
999
|
+
expect(messageChildren(element).length).toBe(1);
|
|
1000
|
+
expect(trim(element.text())).toEqual("C");
|
|
1001
|
+
});
|
|
1002
|
+
|
|
1003
|
+
// it("should properly detect a previous message, even if it was registered later", () => {
|
|
1004
|
+
// $templateCache.put("include.html", '<div ng-message="a">A</div>');
|
|
1005
|
+
// const html =
|
|
1006
|
+
// '<div ng-messages="items">' +
|
|
1007
|
+
// "<div ng-include=\"'include.html'\"></div>" +
|
|
1008
|
+
// '<div ng-message="b">B</div>' +
|
|
1009
|
+
// '<div ng-message="c">C</div>' +
|
|
1010
|
+
// "</div>";
|
|
1011
|
+
|
|
1012
|
+
// element = $compile(html)($rootScope);
|
|
1013
|
+
// $rootScope.$apply("items = {b: true, c: true}");
|
|
1014
|
+
|
|
1015
|
+
// expect(element.text()).toBe("B");
|
|
1016
|
+
|
|
1017
|
+
// const ctrl = element.controller("ngMessages");
|
|
1018
|
+
// const deregisterSpy = spyOn(ctrl, "deregister").and.callThrough();
|
|
1019
|
+
|
|
1020
|
+
// const nodeB = element[0].querySelector('[ng-message="b"]');
|
|
1021
|
+
// jqLite(nodeB).remove();
|
|
1022
|
+
|
|
1023
|
+
// // Make sure removing the element triggers the deregistration in ngMessages
|
|
1024
|
+
// expect(trim(deregisterSpy.calls.mostRecent().args[0].nodeValue)).toBe(
|
|
1025
|
+
// "ngMessage: b",
|
|
1026
|
+
// );
|
|
1027
|
+
|
|
1028
|
+
// $rootScope.$apply("items.a = true");
|
|
1029
|
+
|
|
1030
|
+
// expect(element.text()).toBe("A");
|
|
1031
|
+
// });
|
|
1032
|
+
|
|
1033
|
+
it("should not throw if the template is empty", () => {
|
|
1034
|
+
const html =
|
|
1035
|
+
'<div ng-messages="items">' +
|
|
1036
|
+
'<div ng-messages-include="messages1.html"></div>' +
|
|
1037
|
+
'<div ng-messages-include="messages2.html"></div>' +
|
|
1038
|
+
"</div>";
|
|
1039
|
+
|
|
1040
|
+
$templateCache.put("messages1.html", "");
|
|
1041
|
+
$templateCache.put("messages2.html", " ");
|
|
1042
|
+
element = $compile(html)($rootScope);
|
|
1043
|
+
$rootScope.$digest();
|
|
1044
|
+
|
|
1045
|
+
expect(element.text()).toBe("");
|
|
1046
|
+
expect(element[0].childNodes.length).toBe(2);
|
|
1047
|
+
});
|
|
1048
|
+
});
|
|
1049
|
+
|
|
1050
|
+
describe("when multiple", () => {
|
|
1051
|
+
// they(
|
|
1052
|
+
// "should show all truthy messages when the $prop attr is present",
|
|
1053
|
+
// { multiple: "multiple", "ng-messages-multiple": "ng-messages-multiple" },
|
|
1054
|
+
// (prop) => {
|
|
1055
|
+
// () => {
|
|
1056
|
+
// element = $compile(
|
|
1057
|
+
// `<div ng-messages="data" ${prop}>` +
|
|
1058
|
+
// ` <div ng-message="one">1</div>` +
|
|
1059
|
+
// ` <div ng-message="two">2</div>` +
|
|
1060
|
+
// ` <div ng-message="three">3</div>` +
|
|
1061
|
+
// `</div>`,
|
|
1062
|
+
// )($rootScope);
|
|
1063
|
+
|
|
1064
|
+
// $rootScope.$apply(() => {
|
|
1065
|
+
// $rootScope.data = {
|
|
1066
|
+
// one: true,
|
|
1067
|
+
// two: false,
|
|
1068
|
+
// three: true,
|
|
1069
|
+
// };
|
|
1070
|
+
// });
|
|
1071
|
+
|
|
1072
|
+
// expect(messageChildren(element).length).toBe(2);
|
|
1073
|
+
// expect(s(element.text())).toContain("13");
|
|
1074
|
+
// });
|
|
1075
|
+
// },
|
|
1076
|
+
// );
|
|
1077
|
+
|
|
1078
|
+
it("should render all truthy messages from a remote template", () => {
|
|
1079
|
+
$templateCache.put(
|
|
1080
|
+
"xyz.html",
|
|
1081
|
+
'<div ng-message="x">X</div>' +
|
|
1082
|
+
'<div ng-message="y">Y</div>' +
|
|
1083
|
+
'<div ng-message="z">Z</div>',
|
|
1084
|
+
);
|
|
1085
|
+
|
|
1086
|
+
element = $compile(
|
|
1087
|
+
'<div ng-messages="data" ng-messages-multiple="true">' +
|
|
1088
|
+
'<div ng-messages-include="xyz.html"></div>' +
|
|
1089
|
+
"</div>",
|
|
1090
|
+
)($rootScope);
|
|
1091
|
+
|
|
1092
|
+
$rootScope.$apply(() => {
|
|
1093
|
+
$rootScope.data = {
|
|
1094
|
+
x: "a",
|
|
1095
|
+
y: null,
|
|
1096
|
+
z: true,
|
|
1097
|
+
};
|
|
1098
|
+
});
|
|
1099
|
+
|
|
1100
|
+
expect(messageChildren(element).length).toBe(2);
|
|
1101
|
+
expect(s(element.text())).toEqual("XZ");
|
|
1102
|
+
|
|
1103
|
+
$rootScope.$apply(() => {
|
|
1104
|
+
$rootScope.data.y = {};
|
|
1105
|
+
});
|
|
1106
|
+
|
|
1107
|
+
expect(messageChildren(element).length).toBe(3);
|
|
1108
|
+
expect(s(element.text())).toEqual("XYZ");
|
|
1109
|
+
});
|
|
1110
|
+
|
|
1111
|
+
it("should render and override all truthy messages from a remote template", () => {
|
|
1112
|
+
$templateCache.put(
|
|
1113
|
+
"xyz.html",
|
|
1114
|
+
'<div ng-message="x">X</div>' +
|
|
1115
|
+
'<div ng-message="y">Y</div>' +
|
|
1116
|
+
'<div ng-message="z">Z</div>',
|
|
1117
|
+
);
|
|
1118
|
+
|
|
1119
|
+
element = $compile(
|
|
1120
|
+
'<div ng-messages="data" ng-messages-multiple="true">' +
|
|
1121
|
+
'<div ng-message="y">YYY</div>' +
|
|
1122
|
+
'<div ng-message="z">ZZZ</div>' +
|
|
1123
|
+
'<div ng-messages-include="xyz.html"></div>' +
|
|
1124
|
+
"</div>",
|
|
1125
|
+
)($rootScope);
|
|
1126
|
+
|
|
1127
|
+
$rootScope.$apply(() => {
|
|
1128
|
+
$rootScope.data = {
|
|
1129
|
+
x: "a",
|
|
1130
|
+
y: null,
|
|
1131
|
+
z: true,
|
|
1132
|
+
};
|
|
1133
|
+
});
|
|
1134
|
+
|
|
1135
|
+
expect(messageChildren(element).length).toBe(2);
|
|
1136
|
+
expect(s(element.text())).toEqual("ZZZX");
|
|
1137
|
+
|
|
1138
|
+
$rootScope.$apply(() => {
|
|
1139
|
+
$rootScope.data.y = {};
|
|
1140
|
+
});
|
|
1141
|
+
|
|
1142
|
+
expect(messageChildren(element).length).toBe(3);
|
|
1143
|
+
expect(s(element.text())).toEqual("YYYZZZX");
|
|
1144
|
+
});
|
|
1145
|
+
});
|
|
1146
|
+
});
|