@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.
Files changed (231) hide show
  1. package/.eslintignore +1 -0
  2. package/.eslintrc.cjs +29 -0
  3. package/.github/workflows/playwright.yml +27 -0
  4. package/CHANGELOG.md +17974 -0
  5. package/CODE_OF_CONDUCT.md +3 -0
  6. package/CONTRIBUTING.md +246 -0
  7. package/DEVELOPERS.md +488 -0
  8. package/LICENSE +22 -0
  9. package/Makefile +31 -0
  10. package/README.md +115 -0
  11. package/RELEASE.md +98 -0
  12. package/SECURITY.md +16 -0
  13. package/TRIAGING.md +135 -0
  14. package/css/angular.css +22 -0
  15. package/dist/angular-ts.cjs.js +36843 -0
  16. package/dist/angular-ts.esm.js +36841 -0
  17. package/dist/angular-ts.umd.js +36848 -0
  18. package/dist/build/angular-animate.js +4272 -0
  19. package/dist/build/angular-aria.js +426 -0
  20. package/dist/build/angular-message-format.js +1072 -0
  21. package/dist/build/angular-messages.js +829 -0
  22. package/dist/build/angular-mocks.js +3757 -0
  23. package/dist/build/angular-parse-ext.js +1275 -0
  24. package/dist/build/angular-resource.js +911 -0
  25. package/dist/build/angular-route.js +1266 -0
  26. package/dist/build/angular-sanitize.js +891 -0
  27. package/dist/build/angular-touch.js +368 -0
  28. package/dist/build/angular.js +36600 -0
  29. package/e2e/unit.spec.ts +15 -0
  30. package/images/android-chrome-192x192.png +0 -0
  31. package/images/android-chrome-512x512.png +0 -0
  32. package/images/apple-touch-icon.png +0 -0
  33. package/images/favicon-16x16.png +0 -0
  34. package/images/favicon-32x32.png +0 -0
  35. package/images/favicon.ico +0 -0
  36. package/images/site.webmanifest +1 -0
  37. package/index.html +104 -0
  38. package/package.json +47 -0
  39. package/playwright.config.ts +78 -0
  40. package/public/circle.html +1 -0
  41. package/public/my_child_directive.html +1 -0
  42. package/public/my_directive.html +1 -0
  43. package/public/my_other_directive.html +1 -0
  44. package/public/test.html +1 -0
  45. package/rollup.config.js +31 -0
  46. package/src/animations/animateCache.js +55 -0
  47. package/src/animations/animateChildrenDirective.js +105 -0
  48. package/src/animations/animateCss.js +1139 -0
  49. package/src/animations/animateCssDriver.js +291 -0
  50. package/src/animations/animateJs.js +367 -0
  51. package/src/animations/animateJsDriver.js +67 -0
  52. package/src/animations/animateQueue.js +851 -0
  53. package/src/animations/animation.js +506 -0
  54. package/src/animations/module.js +779 -0
  55. package/src/animations/ngAnimateSwap.js +119 -0
  56. package/src/animations/rafScheduler.js +50 -0
  57. package/src/animations/shared.js +378 -0
  58. package/src/constants.js +20 -0
  59. package/src/core/animate.js +845 -0
  60. package/src/core/animateCss.js +73 -0
  61. package/src/core/animateRunner.js +195 -0
  62. package/src/core/attributes.js +199 -0
  63. package/src/core/cache.js +45 -0
  64. package/src/core/compile.js +4727 -0
  65. package/src/core/controller.js +225 -0
  66. package/src/core/exceptionHandler.js +63 -0
  67. package/src/core/filter.js +146 -0
  68. package/src/core/interpolate.js +442 -0
  69. package/src/core/interval.js +188 -0
  70. package/src/core/intervalFactory.js +57 -0
  71. package/src/core/location.js +1086 -0
  72. package/src/core/parser/parse.js +2562 -0
  73. package/src/core/parser/parse.md +13 -0
  74. package/src/core/q.js +746 -0
  75. package/src/core/rootScope.js +1596 -0
  76. package/src/core/sanitizeUri.js +85 -0
  77. package/src/core/sce.js +1161 -0
  78. package/src/core/taskTrackerFactory.js +125 -0
  79. package/src/core/timeout.js +121 -0
  80. package/src/core/urlUtils.js +187 -0
  81. package/src/core/utils.js +1349 -0
  82. package/src/directive/a.js +37 -0
  83. package/src/directive/attrs.js +283 -0
  84. package/src/directive/bind.js +51 -0
  85. package/src/directive/bind.md +142 -0
  86. package/src/directive/change.js +12 -0
  87. package/src/directive/change.md +25 -0
  88. package/src/directive/cloak.js +12 -0
  89. package/src/directive/cloak.md +24 -0
  90. package/src/directive/events.js +75 -0
  91. package/src/directive/events.md +166 -0
  92. package/src/directive/form.js +725 -0
  93. package/src/directive/init.js +15 -0
  94. package/src/directive/init.md +41 -0
  95. package/src/directive/input.js +1783 -0
  96. package/src/directive/list.js +46 -0
  97. package/src/directive/list.md +22 -0
  98. package/src/directive/ngClass.js +249 -0
  99. package/src/directive/ngController.js +64 -0
  100. package/src/directive/ngCsp.js +82 -0
  101. package/src/directive/ngIf.js +134 -0
  102. package/src/directive/ngInclude.js +217 -0
  103. package/src/directive/ngModel.js +1356 -0
  104. package/src/directive/ngModelOptions.js +509 -0
  105. package/src/directive/ngOptions.js +670 -0
  106. package/src/directive/ngRef.js +90 -0
  107. package/src/directive/ngRepeat.js +650 -0
  108. package/src/directive/ngShowHide.js +255 -0
  109. package/src/directive/ngSwitch.js +178 -0
  110. package/src/directive/ngTransclude.js +98 -0
  111. package/src/directive/non-bindable.js +11 -0
  112. package/src/directive/non-bindable.md +17 -0
  113. package/src/directive/script.js +30 -0
  114. package/src/directive/select.js +624 -0
  115. package/src/directive/style.js +25 -0
  116. package/src/directive/style.md +23 -0
  117. package/src/directive/validators.js +329 -0
  118. package/src/exts/aria.js +544 -0
  119. package/src/exts/messages.js +852 -0
  120. package/src/filters/filter.js +207 -0
  121. package/src/filters/filter.md +69 -0
  122. package/src/filters/filters.js +239 -0
  123. package/src/filters/json.md +16 -0
  124. package/src/filters/limit-to.js +43 -0
  125. package/src/filters/limit-to.md +19 -0
  126. package/src/filters/order-by.js +183 -0
  127. package/src/filters/order-by.md +83 -0
  128. package/src/index.js +13 -0
  129. package/src/injector.js +1034 -0
  130. package/src/jqLite.js +1117 -0
  131. package/src/loader.js +1320 -0
  132. package/src/public.js +215 -0
  133. package/src/routeToRegExp.js +41 -0
  134. package/src/services/anchorScroll.js +135 -0
  135. package/src/services/browser.js +321 -0
  136. package/src/services/cacheFactory.js +398 -0
  137. package/src/services/cookieReader.js +72 -0
  138. package/src/services/document.js +64 -0
  139. package/src/services/http.js +1537 -0
  140. package/src/services/httpBackend.js +206 -0
  141. package/src/services/log.js +160 -0
  142. package/src/services/templateRequest.js +139 -0
  143. package/test/angular.spec.js +2153 -0
  144. package/test/aria/aria.spec.js +1245 -0
  145. package/test/binding.spec.js +504 -0
  146. package/test/build-test.html +14 -0
  147. package/test/injector.spec.js +2327 -0
  148. package/test/jasmine/jasmine-5.1.2/boot0.js +65 -0
  149. package/test/jasmine/jasmine-5.1.2/boot1.js +133 -0
  150. package/test/jasmine/jasmine-5.1.2/jasmine-html.js +963 -0
  151. package/test/jasmine/jasmine-5.1.2/jasmine.css +320 -0
  152. package/test/jasmine/jasmine-5.1.2/jasmine.js +10824 -0
  153. package/test/jasmine/jasmine-5.1.2/jasmine_favicon.png +0 -0
  154. package/test/jasmine/jasmine-browser.json +17 -0
  155. package/test/jasmine/jasmine.json +9 -0
  156. package/test/jqlite.spec.js +2133 -0
  157. package/test/loader.spec.js +219 -0
  158. package/test/messages/messages.spec.js +1146 -0
  159. package/test/min-err.spec.js +174 -0
  160. package/test/mock-test.html +13 -0
  161. package/test/module-test.html +15 -0
  162. package/test/ng/anomate.spec.js +606 -0
  163. package/test/ng/cache-factor.spec.js +334 -0
  164. package/test/ng/compile.spec.js +17956 -0
  165. package/test/ng/controller-provider.spec.js +227 -0
  166. package/test/ng/cookie-reader.spec.js +98 -0
  167. package/test/ng/directive/a.spec.js +192 -0
  168. package/test/ng/directive/bind.spec.js +334 -0
  169. package/test/ng/directive/boolean.spec.js +136 -0
  170. package/test/ng/directive/change.spec.js +71 -0
  171. package/test/ng/directive/class.spec.js +858 -0
  172. package/test/ng/directive/click.spec.js +38 -0
  173. package/test/ng/directive/cloak.spec.js +44 -0
  174. package/test/ng/directive/constoller.spec.js +194 -0
  175. package/test/ng/directive/element-style.spec.js +92 -0
  176. package/test/ng/directive/event.spec.js +282 -0
  177. package/test/ng/directive/form.spec.js +1518 -0
  178. package/test/ng/directive/href.spec.js +143 -0
  179. package/test/ng/directive/if.spec.js +402 -0
  180. package/test/ng/directive/include.spec.js +828 -0
  181. package/test/ng/directive/init.spec.js +68 -0
  182. package/test/ng/directive/input.spec.js +3810 -0
  183. package/test/ng/directive/list.spec.js +170 -0
  184. package/test/ng/directive/model-options.spec.js +1008 -0
  185. package/test/ng/directive/model.spec.js +1905 -0
  186. package/test/ng/directive/non-bindable.spec.js +55 -0
  187. package/test/ng/directive/options.spec.js +3583 -0
  188. package/test/ng/directive/ref.spec.js +575 -0
  189. package/test/ng/directive/repeat.spec.js +1675 -0
  190. package/test/ng/directive/script.spec.js +52 -0
  191. package/test/ng/directive/scrset.spec.js +67 -0
  192. package/test/ng/directive/select.spec.js +2541 -0
  193. package/test/ng/directive/show-hide.spec.js +253 -0
  194. package/test/ng/directive/src.spec.js +157 -0
  195. package/test/ng/directive/style.spec.js +178 -0
  196. package/test/ng/directive/switch.spec.js +647 -0
  197. package/test/ng/directive/validators.spec.js +717 -0
  198. package/test/ng/document.spec.js +52 -0
  199. package/test/ng/filter/filter.spec.js +714 -0
  200. package/test/ng/filter/filters.spec.js +35 -0
  201. package/test/ng/filter/limit-to.spec.js +251 -0
  202. package/test/ng/filter/order-by.spec.js +891 -0
  203. package/test/ng/filter.spec.js +149 -0
  204. package/test/ng/http-backend.spec.js +398 -0
  205. package/test/ng/http.spec.js +4071 -0
  206. package/test/ng/interpolate.spec.js +642 -0
  207. package/test/ng/interval.spec.js +343 -0
  208. package/test/ng/location.spec.js +3488 -0
  209. package/test/ng/on.spec.js +229 -0
  210. package/test/ng/parse.spec.js +4655 -0
  211. package/test/ng/prop.spec.js +805 -0
  212. package/test/ng/q.spec.js +2904 -0
  213. package/test/ng/root-element.spec.js +16 -0
  214. package/test/ng/sanitize-uri.spec.js +249 -0
  215. package/test/ng/sce.spec.js +660 -0
  216. package/test/ng/scope.spec.js +3442 -0
  217. package/test/ng/template-request.spec.js +236 -0
  218. package/test/ng/timeout.spec.js +351 -0
  219. package/test/ng/url-utils.spec.js +156 -0
  220. package/test/ng/utils.spec.js +144 -0
  221. package/test/original-test.html +21 -0
  222. package/test/public.spec.js +34 -0
  223. package/test/sanitize/bing-html.spec.js +36 -0
  224. package/test/server/express.js +158 -0
  225. package/test/test-utils.js +11 -0
  226. package/tsconfig.json +17 -0
  227. package/types/angular.d.ts +138 -0
  228. package/types/global.d.ts +9 -0
  229. package/types/index.d.ts +2357 -0
  230. package/types/jqlite.d.ts +558 -0
  231. 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
+ });