@angular/core 18.0.0-rc.0 → 18.0.0-rc.2
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/esm2022/primitives/event-dispatch/contract_binary.mjs +3 -3
- package/esm2022/primitives/event-dispatch/index.mjs +2 -2
- package/esm2022/primitives/event-dispatch/src/action_resolver.mjs +221 -0
- package/esm2022/primitives/event-dispatch/src/attribute.mjs +63 -65
- package/esm2022/primitives/event-dispatch/src/cache.mjs +10 -10
- package/esm2022/primitives/event-dispatch/src/dispatcher.mjs +52 -176
- package/esm2022/primitives/event-dispatch/src/earlyeventcontract.mjs +15 -9
- package/esm2022/primitives/event-dispatch/src/event_contract_defines.mjs +1 -19
- package/esm2022/primitives/event-dispatch/src/eventcontract.mjs +24 -362
- package/esm2022/primitives/event-dispatch/src/key_code.mjs +11 -13
- package/esm2022/primitives/event-dispatch/src/legacy_dispatcher.mjs +252 -2
- package/esm2022/primitives/event-dispatch/src/property.mjs +30 -27
- package/esm2022/primitives/event-dispatch/src/register_events.mjs +16 -17
- package/esm2022/primitives/event-dispatch/src/restriction.mjs +2 -2
- package/esm2022/src/application/application_ref.mjs +6 -3
- package/esm2022/src/application/create_application.mjs +12 -5
- package/esm2022/src/change_detection/scheduling/exhaustive_check_no_changes.mjs +150 -0
- package/esm2022/src/change_detection/scheduling/ng_zone_scheduling.mjs +6 -4
- package/esm2022/src/change_detection/scheduling/zoneless_scheduling.mjs +3 -1
- package/esm2022/src/change_detection/scheduling/zoneless_scheduling_impl.mjs +9 -6
- package/esm2022/src/core.mjs +2 -1
- package/esm2022/src/core_private_export.mjs +2 -1
- package/esm2022/src/core_reactivity_export_internal.mjs +1 -3
- package/esm2022/src/core_render3_private_export.mjs +1 -3
- package/esm2022/src/defer/instructions.mjs +20 -12
- package/esm2022/src/defer/interfaces.mjs +1 -3
- package/esm2022/src/errors.mjs +1 -1
- package/esm2022/src/hydration/event_replay.mjs +67 -79
- package/esm2022/src/hydration/utils.mjs +1 -2
- package/esm2022/src/metadata/directives.mjs +1 -1
- package/esm2022/src/platform/platform_ref.mjs +10 -4
- package/esm2022/src/render3/after_render_hooks.mjs +4 -2
- package/esm2022/src/render3/component_ref.mjs +1 -1
- package/esm2022/src/render3/index.mjs +1 -3
- package/esm2022/src/render3/instructions/change_detection.mjs +13 -10
- package/esm2022/src/render3/instructions/listener.mjs +12 -1
- package/esm2022/src/render3/interfaces/public_definitions.mjs +1 -1
- package/esm2022/src/render3/state.mjs +14 -4
- package/esm2022/src/render3/view_ref.mjs +3 -2
- package/esm2022/src/util/callback_scheduler.mjs +12 -26
- package/esm2022/src/version.mjs +1 -1
- package/esm2022/src/zone/ng_zone.mjs +9 -23
- package/esm2022/testing/src/component_fixture.mjs +2 -4
- package/esm2022/testing/src/defer.mjs +1 -2
- package/esm2022/testing/src/logger.mjs +3 -3
- package/esm2022/testing/src/test_bed.mjs +1 -3
- package/esm2022/testing/src/test_bed_compiler.mjs +3 -6
- package/event-dispatch-contract.min.js +1 -1
- package/fesm2022/core.mjs +800 -660
- package/fesm2022/core.mjs.map +1 -1
- package/fesm2022/primitives/event-dispatch.mjs +484 -807
- package/fesm2022/primitives/event-dispatch.mjs.map +1 -1
- package/fesm2022/primitives/signals.mjs +1 -1
- package/fesm2022/rxjs-interop.mjs +1 -1
- package/fesm2022/testing.mjs +4 -10
- package/fesm2022/testing.mjs.map +1 -1
- package/index.d.ts +36 -4
- package/package.json +1 -1
- package/primitives/event-dispatch/index.d.ts +111 -162
- package/primitives/signals/index.d.ts +1 -1
- package/rxjs-interop/index.d.ts +1 -1
- package/schematics/migrations/http-providers/bundle.js +110 -71
- package/schematics/migrations/http-providers/bundle.js.map +3 -3
- package/schematics/migrations/invalid-two-way-bindings/bundle.js +197 -167
- package/schematics/migrations/invalid-two-way-bindings/bundle.js.map +2 -2
- package/schematics/ng-generate/control-flow-migration/bundle.js +205 -175
- package/schematics/ng-generate/control-flow-migration/bundle.js.map +2 -2
- package/schematics/ng-generate/standalone-migration/bundle.js +481 -451
- package/schematics/ng-generate/standalone-migration/bundle.js.map +2 -2
- package/testing/index.d.ts +1 -4
- package/esm2022/primitives/event-dispatch/src/base_dispatcher.mjs +0 -96
- package/esm2022/primitives/event-dispatch/src/custom_events.mjs +0 -63
- package/esm2022/primitives/event-dispatch/src/replay.mjs +0 -389
|
@@ -29,24 +29,11 @@
|
|
|
29
29
|
* possible and thus its dependencies to a minimum.
|
|
30
30
|
*/
|
|
31
31
|
import * as a11yClickLib from './a11y_click';
|
|
32
|
-
import {
|
|
33
|
-
import * as cache from './cache';
|
|
34
|
-
import { Char } from './char';
|
|
32
|
+
import { ActionResolver } from './action_resolver';
|
|
35
33
|
import * as eventLib from './event';
|
|
36
|
-
import { A11Y_CLICK_SUPPORT,
|
|
34
|
+
import { A11Y_CLICK_SUPPORT, MOUSE_SPECIAL_SUPPORT } from './event_contract_defines';
|
|
37
35
|
import * as eventInfoLib from './event_info';
|
|
38
36
|
import { EventType } from './event_type';
|
|
39
|
-
import { Property } from './property';
|
|
40
|
-
const DEFAULT_EVENT_TYPE = EventType.CLICK;
|
|
41
|
-
/**
|
|
42
|
-
* Since maps from event to action are immutable we can use a single map
|
|
43
|
-
* to represent the empty map.
|
|
44
|
-
*/
|
|
45
|
-
const EMPTY_ACTION_MAP = {};
|
|
46
|
-
/**
|
|
47
|
-
* This regular expression matches a semicolon.
|
|
48
|
-
*/
|
|
49
|
-
const REGEXP_SEMICOLON = /\s*;\s*/;
|
|
50
37
|
/**
|
|
51
38
|
* EventContract intercepts events in the bubbling phase at the
|
|
52
39
|
* boundary of a container element, and maps them to generic actions
|
|
@@ -62,12 +49,10 @@ const REGEXP_SEMICOLON = /\s*;\s*/;
|
|
|
62
49
|
* be delay loaded in a generic way.
|
|
63
50
|
*/
|
|
64
51
|
export class EventContract {
|
|
65
|
-
static { this.CUSTOM_EVENT_SUPPORT = CUSTOM_EVENT_SUPPORT; }
|
|
66
|
-
static { this.STOP_PROPAGATION = STOP_PROPAGATION; }
|
|
67
52
|
static { this.A11Y_CLICK_SUPPORT = A11Y_CLICK_SUPPORT; }
|
|
68
53
|
static { this.MOUSE_SPECIAL_SUPPORT = MOUSE_SPECIAL_SUPPORT; }
|
|
69
|
-
|
|
70
|
-
|
|
54
|
+
constructor(containerManager, useActionResolver = true) {
|
|
55
|
+
this.useActionResolver = useActionResolver;
|
|
71
56
|
/**
|
|
72
57
|
* The DOM events which this contract covers. Used to prevent double
|
|
73
58
|
* registration of event types. The value of the map is the
|
|
@@ -90,16 +75,13 @@ export class EventContract {
|
|
|
90
75
|
* as soon as the `Dispatcher` is registered.
|
|
91
76
|
*/
|
|
92
77
|
this.queuedEventInfos = [];
|
|
93
|
-
/** Whether a11y click support has been loaded or not. */
|
|
94
|
-
this.hasA11yClickSupport = false;
|
|
95
78
|
/** Whether to add an a11y click listener. */
|
|
96
79
|
this.addA11yClickListener = false;
|
|
97
|
-
this.updateEventInfoForA11yClick = undefined;
|
|
98
|
-
this.preventDefaultForA11yClick = undefined;
|
|
99
|
-
this.populateClickOnlyAction = undefined;
|
|
100
80
|
this.containerManager = containerManager;
|
|
101
|
-
if (
|
|
102
|
-
this.
|
|
81
|
+
if (this.useActionResolver) {
|
|
82
|
+
this.actionResolver = new ActionResolver({
|
|
83
|
+
syntheticMouseEventSupport: EventContract.MOUSE_SPECIAL_SUPPORT,
|
|
84
|
+
});
|
|
103
85
|
}
|
|
104
86
|
if (EventContract.A11Y_CLICK_SUPPORT) {
|
|
105
87
|
// Add a11y click support to the `EventContract`.
|
|
@@ -123,176 +105,13 @@ export class EventContract {
|
|
|
123
105
|
// All events are queued when the dispatcher isn't yet loaded.
|
|
124
106
|
eventInfoLib.setIsReplay(eventInfo, true);
|
|
125
107
|
this.queuedEventInfos?.push(eventInfo);
|
|
126
|
-
}
|
|
127
|
-
if (EventContract.CUSTOM_EVENT_SUPPORT &&
|
|
128
|
-
eventInfoLib.getEventType(eventInfo) === EventType.CUSTOM) {
|
|
129
|
-
const detail = eventInfoLib.getEvent(eventInfo).detail;
|
|
130
|
-
// For custom events, use a secondary dispatch based on the internal
|
|
131
|
-
// custom type of the event.
|
|
132
|
-
if (!detail || !detail['_type']) {
|
|
133
|
-
// This should never happen.
|
|
134
|
-
return;
|
|
135
|
-
}
|
|
136
|
-
eventInfoLib.setEventType(eventInfo, detail['_type']);
|
|
137
|
-
}
|
|
138
|
-
this.populateAction(eventInfo);
|
|
139
|
-
if (!this.dispatcher) {
|
|
140
108
|
return;
|
|
141
109
|
}
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
// `clickonly`. Revert back to a regular click, otherwise we won't be able
|
|
145
|
-
// to execute global event handlers registered on click events.
|
|
146
|
-
if (eventInfoLib.getEventType(globalEventInfo) === EventType.CLICKONLY) {
|
|
147
|
-
eventInfoLib.setEventType(globalEventInfo, EventType.CLICK);
|
|
148
|
-
}
|
|
149
|
-
this.dispatcher(globalEventInfo, /* dispatch global event */ true);
|
|
150
|
-
const action = eventInfoLib.getAction(eventInfo);
|
|
151
|
-
if (!action) {
|
|
152
|
-
return;
|
|
153
|
-
}
|
|
154
|
-
if (shouldPreventDefaultBeforeDispatching(eventInfoLib.getActionElement(action), eventInfo)) {
|
|
155
|
-
eventLib.preventDefault(eventInfoLib.getEvent(eventInfo));
|
|
110
|
+
if (this.useActionResolver) {
|
|
111
|
+
this.actionResolver.resolve(eventInfo);
|
|
156
112
|
}
|
|
157
113
|
this.dispatcher(eventInfo);
|
|
158
114
|
}
|
|
159
|
-
/**
|
|
160
|
-
* Searches for a jsaction that the DOM event maps to and creates an
|
|
161
|
-
* object containing event information used for dispatching by
|
|
162
|
-
* jsaction.Dispatcher. This method populates the `action` and `actionElement`
|
|
163
|
-
* fields of the EventInfo object passed in by finding the first
|
|
164
|
-
* jsaction attribute above the target Node of the event, and below
|
|
165
|
-
* the container Node, that specifies a jsaction for the event
|
|
166
|
-
* type. If no such jsaction is found, then action is undefined.
|
|
167
|
-
*
|
|
168
|
-
* @param eventInfo `EventInfo` to set `action` and `actionElement` if an
|
|
169
|
-
* action is found on any `Element` in the path of the `Event`.
|
|
170
|
-
*/
|
|
171
|
-
populateAction(eventInfo) {
|
|
172
|
-
// We distinguish modified and plain clicks in order to support the
|
|
173
|
-
// default browser behavior of modified clicks on links; usually to
|
|
174
|
-
// open the URL of the link in new tab or new window on ctrl/cmd
|
|
175
|
-
// click. A DOM 'click' event is mapped to the jsaction 'click'
|
|
176
|
-
// event iff there is no modifier present on the event. If there is
|
|
177
|
-
// a modifier, it's mapped to 'clickmod' instead.
|
|
178
|
-
//
|
|
179
|
-
// It's allowed to omit the event in the jsaction attribute. In that
|
|
180
|
-
// case, 'click' is assumed. Thus the following two are equivalent:
|
|
181
|
-
//
|
|
182
|
-
// <a href="someurl" jsaction="gna.fu">
|
|
183
|
-
// <a href="someurl" jsaction="click:gna.fu">
|
|
184
|
-
//
|
|
185
|
-
// For unmodified clicks, EventContract invokes the jsaction
|
|
186
|
-
// 'gna.fu'. For modified clicks, EventContract won't find a
|
|
187
|
-
// suitable action and leave the event to be handled by the
|
|
188
|
-
// browser.
|
|
189
|
-
//
|
|
190
|
-
// In order to also invoke a jsaction handler for a modifier click,
|
|
191
|
-
// 'clickmod' needs to be used:
|
|
192
|
-
//
|
|
193
|
-
// <a href="someurl" jsaction="clickmod:gna.fu">
|
|
194
|
-
//
|
|
195
|
-
// EventContract invokes the jsaction 'gna.fu' for modified
|
|
196
|
-
// clicks. Unmodified clicks are left to the browser.
|
|
197
|
-
//
|
|
198
|
-
// In order to set up the event contract to handle both clickonly and
|
|
199
|
-
// clickmod, only addEvent(EventType.CLICK) is necessary.
|
|
200
|
-
//
|
|
201
|
-
// In order to set up the event contract to handle click,
|
|
202
|
-
// addEvent() is necessary for CLICK, KEYDOWN, and KEYPRESS event types. If
|
|
203
|
-
// a11y click support is enabled, addEvent() will set up the appropriate key
|
|
204
|
-
// event handler automatically.
|
|
205
|
-
if (eventInfoLib.getEventType(eventInfo) === EventType.CLICK &&
|
|
206
|
-
eventLib.isModifiedClickEvent(eventInfoLib.getEvent(eventInfo))) {
|
|
207
|
-
eventInfoLib.setEventType(eventInfo, EventType.CLICKMOD);
|
|
208
|
-
}
|
|
209
|
-
else if (this.hasA11yClickSupport) {
|
|
210
|
-
this.updateEventInfoForA11yClick(eventInfo);
|
|
211
|
-
}
|
|
212
|
-
// Walk to the parent node, unless the node has a different owner in
|
|
213
|
-
// which case we walk to the owner. Attempt to walk to host of a
|
|
214
|
-
// shadow root if needed.
|
|
215
|
-
let actionElement = eventInfoLib.getTargetElement(eventInfo);
|
|
216
|
-
while (actionElement && actionElement !== eventInfoLib.getContainer(eventInfo)) {
|
|
217
|
-
this.populateActionOnElement(actionElement, eventInfo);
|
|
218
|
-
if (eventInfoLib.getAction(eventInfo)) {
|
|
219
|
-
// An event is handled by at most one jsaction. Thus we stop at the
|
|
220
|
-
// first matching jsaction specified in a jsaction attribute up the
|
|
221
|
-
// ancestor chain of the event target node.
|
|
222
|
-
break;
|
|
223
|
-
}
|
|
224
|
-
if (actionElement[Property.OWNER]) {
|
|
225
|
-
actionElement = actionElement[Property.OWNER];
|
|
226
|
-
continue;
|
|
227
|
-
}
|
|
228
|
-
if (actionElement.parentNode?.nodeName !== '#document-fragment') {
|
|
229
|
-
actionElement = actionElement.parentNode;
|
|
230
|
-
}
|
|
231
|
-
else {
|
|
232
|
-
actionElement = actionElement.parentNode?.host ?? null;
|
|
233
|
-
}
|
|
234
|
-
}
|
|
235
|
-
const action = eventInfoLib.getAction(eventInfo);
|
|
236
|
-
if (!action) {
|
|
237
|
-
// No action found.
|
|
238
|
-
return;
|
|
239
|
-
}
|
|
240
|
-
if (this.hasA11yClickSupport) {
|
|
241
|
-
this.preventDefaultForA11yClick(eventInfo);
|
|
242
|
-
}
|
|
243
|
-
// We attempt to handle the mouseenter/mouseleave events here by
|
|
244
|
-
// detecting whether the mouseover/mouseout events correspond to
|
|
245
|
-
// entering/leaving an element.
|
|
246
|
-
if (EventContract.MOUSE_SPECIAL_SUPPORT &&
|
|
247
|
-
(eventInfoLib.getEventType(eventInfo) === EventType.MOUSEENTER ||
|
|
248
|
-
eventInfoLib.getEventType(eventInfo) === EventType.MOUSELEAVE ||
|
|
249
|
-
eventInfoLib.getEventType(eventInfo) === EventType.POINTERENTER ||
|
|
250
|
-
eventInfoLib.getEventType(eventInfo) === EventType.POINTERLEAVE)) {
|
|
251
|
-
// We attempt to handle the mouseenter/mouseleave events here by
|
|
252
|
-
// detecting whether the mouseover/mouseout events correspond to
|
|
253
|
-
// entering/leaving an element.
|
|
254
|
-
if (eventLib.isMouseSpecialEvent(eventInfoLib.getEvent(eventInfo), eventInfoLib.getEventType(eventInfo), eventInfoLib.getActionElement(action))) {
|
|
255
|
-
// If both mouseover/mouseout and mouseenter/mouseleave events are
|
|
256
|
-
// enabled, two separate handlers for mouseover/mouseout are
|
|
257
|
-
// registered. Both handlers will see the same event instance
|
|
258
|
-
// so we create a copy to avoid interfering with the dispatching of
|
|
259
|
-
// the mouseover/mouseout event.
|
|
260
|
-
const copiedEvent = eventLib.createMouseSpecialEvent(eventInfoLib.getEvent(eventInfo), eventInfoLib.getActionElement(action));
|
|
261
|
-
eventInfoLib.setEvent(eventInfo, copiedEvent);
|
|
262
|
-
// Since the mouseenter/mouseleave events do not bubble, the target
|
|
263
|
-
// of the event is technically the `actionElement` (the node with the
|
|
264
|
-
// `jsaction` attribute)
|
|
265
|
-
eventInfoLib.setTargetElement(eventInfo, eventInfoLib.getActionElement(action));
|
|
266
|
-
}
|
|
267
|
-
else {
|
|
268
|
-
eventInfoLib.unsetAction(eventInfo);
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
}
|
|
272
|
-
/**
|
|
273
|
-
* Accesses the jsaction map on a node and retrieves the name of the
|
|
274
|
-
* action the given event is mapped to, if any. It parses the
|
|
275
|
-
* attribute value and stores it in a property on the node for
|
|
276
|
-
* subsequent retrieval without re-parsing and re-accessing the
|
|
277
|
-
* attribute. In order to fully qualify jsaction names using a
|
|
278
|
-
* namespace, the DOM is searched starting at the current node and
|
|
279
|
-
* going through ancestor nodes until a jsnamespace attribute is
|
|
280
|
-
* found.
|
|
281
|
-
*
|
|
282
|
-
* @param actionElement The DOM node to retrieve the jsaction map from.
|
|
283
|
-
* @param eventInfo `EventInfo` to set `action` and `actionElement` if an
|
|
284
|
-
* action is found on the `actionElement`.
|
|
285
|
-
*/
|
|
286
|
-
populateActionOnElement(actionElement, eventInfo) {
|
|
287
|
-
const actionMap = parseActions(actionElement, eventInfoLib.getContainer(eventInfo));
|
|
288
|
-
const actionName = actionMap[eventInfoLib.getEventType(eventInfo)];
|
|
289
|
-
if (actionName !== undefined) {
|
|
290
|
-
eventInfoLib.setAction(eventInfo, actionName, actionElement);
|
|
291
|
-
}
|
|
292
|
-
if (this.hasA11yClickSupport) {
|
|
293
|
-
this.populateClickOnlyAction(actionElement, eventInfo, actionMap);
|
|
294
|
-
}
|
|
295
|
-
}
|
|
296
115
|
/**
|
|
297
116
|
* Enables jsaction handlers to be called for the event type given by
|
|
298
117
|
* name.
|
|
@@ -344,10 +163,10 @@ export class EventContract {
|
|
|
344
163
|
* in the provided event contract. Once all the events are replayed, it cleans
|
|
345
164
|
* up the early contract.
|
|
346
165
|
*/
|
|
347
|
-
replayEarlyEvents() {
|
|
166
|
+
replayEarlyEvents(earlyJsactionContainer = window) {
|
|
348
167
|
// Check if the early contract is present and prevent calling this function
|
|
349
168
|
// more than once.
|
|
350
|
-
const earlyJsactionData =
|
|
169
|
+
const earlyJsactionData = earlyJsactionContainer._ejsa;
|
|
351
170
|
if (!earlyJsactionData) {
|
|
352
171
|
return;
|
|
353
172
|
}
|
|
@@ -365,13 +184,10 @@ export class EventContract {
|
|
|
365
184
|
}
|
|
366
185
|
}
|
|
367
186
|
// Clean up the early contract.
|
|
368
|
-
const earlyEventTypes = earlyJsactionData.et;
|
|
369
187
|
const earlyEventHandler = earlyJsactionData.h;
|
|
370
|
-
|
|
371
|
-
|
|
372
|
-
|
|
373
|
-
}
|
|
374
|
-
delete window._ejsa;
|
|
188
|
+
removeEventListeners(earlyJsactionData.c, earlyJsactionData.et, earlyEventHandler);
|
|
189
|
+
removeEventListeners(earlyJsactionData.c, earlyJsactionData.etc, earlyEventHandler, true);
|
|
190
|
+
delete earlyJsactionContainer._ejsa;
|
|
375
191
|
}
|
|
376
192
|
/**
|
|
377
193
|
* Returns all JSAction event types that have been registered for a given
|
|
@@ -452,10 +268,14 @@ export class EventContract {
|
|
|
452
268
|
*/
|
|
453
269
|
addA11yClickSupportImpl(updateEventInfoForA11yClick, preventDefaultForA11yClick, populateClickOnlyAction) {
|
|
454
270
|
this.addA11yClickListener = true;
|
|
455
|
-
this.
|
|
456
|
-
|
|
457
|
-
|
|
458
|
-
|
|
271
|
+
if (this.useActionResolver) {
|
|
272
|
+
this.actionResolver.addA11yClickSupport(updateEventInfoForA11yClick, preventDefaultForA11yClick, populateClickOnlyAction);
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
}
|
|
276
|
+
function removeEventListeners(container, eventTypes, earlyEventHandler, capture) {
|
|
277
|
+
for (let idx = 0; idx < eventTypes.length; idx++) {
|
|
278
|
+
container.removeEventListener(eventTypes[idx], earlyEventHandler, /* useCapture */ capture);
|
|
459
279
|
}
|
|
460
280
|
}
|
|
461
281
|
/**
|
|
@@ -467,162 +287,4 @@ export class EventContract {
|
|
|
467
287
|
export function addDeferredA11yClickSupport(eventContract) {
|
|
468
288
|
eventContract.ecaacs?.(a11yClickLib.updateEventInfoForA11yClick, a11yClickLib.preventDefaultForA11yClick, a11yClickLib.populateClickOnlyAction);
|
|
469
289
|
}
|
|
470
|
-
/**
|
|
471
|
-
* Returns true if the default action of this event should be prevented before
|
|
472
|
-
* this event is dispatched.
|
|
473
|
-
*/
|
|
474
|
-
function shouldPreventDefaultBeforeDispatching(actionElement, eventInfo) {
|
|
475
|
-
// Prevent browser from following <a> node links if a jsaction is present
|
|
476
|
-
// and we are dispatching the action now. Note that the targetElement may be
|
|
477
|
-
// a child of an anchor that has a jsaction attached. For that reason, we
|
|
478
|
-
// need to check the actionElement rather than the targetElement.
|
|
479
|
-
return (actionElement.tagName === 'A' &&
|
|
480
|
-
(eventInfoLib.getEventType(eventInfo) === EventType.CLICK ||
|
|
481
|
-
eventInfoLib.getEventType(eventInfo) === EventType.CLICKMOD));
|
|
482
|
-
}
|
|
483
|
-
/**
|
|
484
|
-
* Parses and caches an element's jsaction element into a map.
|
|
485
|
-
*
|
|
486
|
-
* This is primarily for internal use.
|
|
487
|
-
*
|
|
488
|
-
* @param actionElement The DOM node to retrieve the jsaction map from.
|
|
489
|
-
* @param container The node which limits the namespace lookup for a jsaction
|
|
490
|
-
* name. The container node itself will not be searched.
|
|
491
|
-
* @return Map from event to qualified name of the jsaction bound to it.
|
|
492
|
-
*/
|
|
493
|
-
export function parseActions(actionElement, container) {
|
|
494
|
-
let actionMap = cache.get(actionElement);
|
|
495
|
-
if (!actionMap) {
|
|
496
|
-
const jsactionAttribute = getAttr(actionElement, Attribute.JSACTION);
|
|
497
|
-
if (!jsactionAttribute) {
|
|
498
|
-
actionMap = EMPTY_ACTION_MAP;
|
|
499
|
-
cache.set(actionElement, actionMap);
|
|
500
|
-
}
|
|
501
|
-
else {
|
|
502
|
-
actionMap = cache.getParsed(jsactionAttribute);
|
|
503
|
-
if (!actionMap) {
|
|
504
|
-
actionMap = {};
|
|
505
|
-
const values = jsactionAttribute.split(REGEXP_SEMICOLON);
|
|
506
|
-
for (let idx = 0; idx < values.length; idx++) {
|
|
507
|
-
const value = values[idx];
|
|
508
|
-
if (!value) {
|
|
509
|
-
continue;
|
|
510
|
-
}
|
|
511
|
-
const colon = value.indexOf(Char.EVENT_ACTION_SEPARATOR);
|
|
512
|
-
const hasColon = colon !== -1;
|
|
513
|
-
const type = hasColon ? stringTrim(value.substr(0, colon)) : DEFAULT_EVENT_TYPE;
|
|
514
|
-
const action = hasColon ? stringTrim(value.substr(colon + 1)) : value;
|
|
515
|
-
actionMap[type] = action;
|
|
516
|
-
}
|
|
517
|
-
cache.setParsed(jsactionAttribute, actionMap);
|
|
518
|
-
}
|
|
519
|
-
// If namespace support is active we need to augment the (potentially
|
|
520
|
-
// cached) jsaction mapping with the namespace.
|
|
521
|
-
if (EventContract.JSNAMESPACE_SUPPORT) {
|
|
522
|
-
const noNs = actionMap;
|
|
523
|
-
actionMap = {};
|
|
524
|
-
for (const type in noNs) {
|
|
525
|
-
actionMap[type] = getFullyQualifiedAction(noNs[type], actionElement, container);
|
|
526
|
-
}
|
|
527
|
-
}
|
|
528
|
-
cache.set(actionElement, actionMap);
|
|
529
|
-
}
|
|
530
|
-
}
|
|
531
|
-
return actionMap;
|
|
532
|
-
}
|
|
533
|
-
/**
|
|
534
|
-
* Returns the fully qualified jsaction action. If the given jsaction
|
|
535
|
-
* name doesn't already contain the namespace, the function iterates
|
|
536
|
-
* over ancestor nodes until a jsnamespace attribute is found, and
|
|
537
|
-
* uses the value of that attribute as the namespace.
|
|
538
|
-
*
|
|
539
|
-
* @param action The jsaction action to resolve.
|
|
540
|
-
* @param start The node from which to start searching for a jsnamespace
|
|
541
|
-
* attribute.
|
|
542
|
-
* @param container The node which limits the search for a jsnamespace
|
|
543
|
-
* attribute. This node will be searched.
|
|
544
|
-
* @return The fully qualified name of the jsaction. If no namespace is found,
|
|
545
|
-
* returns the unqualified name in case it exists in the global namespace.
|
|
546
|
-
*/
|
|
547
|
-
function getFullyQualifiedAction(action, start, container) {
|
|
548
|
-
if (EventContract.JSNAMESPACE_SUPPORT) {
|
|
549
|
-
if (isNamespacedAction(action)) {
|
|
550
|
-
return action;
|
|
551
|
-
}
|
|
552
|
-
let node = start;
|
|
553
|
-
while (node) {
|
|
554
|
-
const namespace = getNamespaceFromElement(node);
|
|
555
|
-
if (namespace) {
|
|
556
|
-
return namespace + Char.NAMESPACE_ACTION_SEPARATOR + action;
|
|
557
|
-
}
|
|
558
|
-
// If this node is the container, stop.
|
|
559
|
-
if (node === container) {
|
|
560
|
-
break;
|
|
561
|
-
}
|
|
562
|
-
node = node.parentNode;
|
|
563
|
-
}
|
|
564
|
-
}
|
|
565
|
-
return action;
|
|
566
|
-
}
|
|
567
|
-
/**
|
|
568
|
-
* Checks if a jsaction action contains a namespace part.
|
|
569
|
-
*/
|
|
570
|
-
function isNamespacedAction(action) {
|
|
571
|
-
return action.indexOf(Char.NAMESPACE_ACTION_SEPARATOR) >= 0;
|
|
572
|
-
}
|
|
573
|
-
/**
|
|
574
|
-
* Returns the value of the jsnamespace attribute of the given node.
|
|
575
|
-
* Also caches the value for subsequent lookups.
|
|
576
|
-
* @param element The node whose jsnamespace attribute is being asked for.
|
|
577
|
-
* @return The value of the jsnamespace attribute, or null if not found.
|
|
578
|
-
*/
|
|
579
|
-
function getNamespaceFromElement(element) {
|
|
580
|
-
let namespace = cache.getNamespace(element);
|
|
581
|
-
// Only query for the attribute if it has not been queried for
|
|
582
|
-
// before. getAttr() returns null if an attribute is not present. Thus,
|
|
583
|
-
// namespace is string|null if the query took place in the past, or
|
|
584
|
-
// undefined if the query did not take place.
|
|
585
|
-
if (namespace === undefined) {
|
|
586
|
-
namespace = getAttr(element, Attribute.JSNAMESPACE);
|
|
587
|
-
cache.setNamespace(element, namespace);
|
|
588
|
-
}
|
|
589
|
-
return namespace;
|
|
590
|
-
}
|
|
591
|
-
/**
|
|
592
|
-
* Accesses the event handler attribute value of a DOM node. It guards
|
|
593
|
-
* against weird situations (described in the body) that occur in
|
|
594
|
-
* connection with nodes that are removed from their document.
|
|
595
|
-
* @param element The DOM element.
|
|
596
|
-
* @param attribute The name of the attribute to access.
|
|
597
|
-
* @return The attribute value if it was found, null otherwise.
|
|
598
|
-
*/
|
|
599
|
-
function getAttr(element, attribute) {
|
|
600
|
-
let value = null;
|
|
601
|
-
// NOTE: Nodes in IE do not always have a getAttribute
|
|
602
|
-
// method defined. This is the case where sourceElement has in
|
|
603
|
-
// fact been removed from the DOM before eventContract begins
|
|
604
|
-
// handling - where a parentNode does not have getAttribute
|
|
605
|
-
// defined.
|
|
606
|
-
// NOTE: We must use the 'in' operator instead of the regular dot
|
|
607
|
-
// notation, since the latter fails in IE8 if the getAttribute method is not
|
|
608
|
-
// defined. See b/7139109.
|
|
609
|
-
if ('getAttribute' in element) {
|
|
610
|
-
value = element.getAttribute(attribute);
|
|
611
|
-
}
|
|
612
|
-
return value;
|
|
613
|
-
}
|
|
614
|
-
/**
|
|
615
|
-
* Helper function to trim whitespace from the beginning and the end
|
|
616
|
-
* of the string. This deliberately doesn't use the closure equivalent
|
|
617
|
-
* to keep dependencies small.
|
|
618
|
-
* @param str Input string.
|
|
619
|
-
* @return Trimmed string.
|
|
620
|
-
*/
|
|
621
|
-
function stringTrim(str) {
|
|
622
|
-
if (typeof String.prototype.trim === 'function') {
|
|
623
|
-
return str.trim();
|
|
624
|
-
}
|
|
625
|
-
const trimmedLeft = str.replace(/^\s+/, '');
|
|
626
|
-
return trimmedLeft.replace(/\s+$/, '');
|
|
627
|
-
}
|
|
628
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eventcontract.js","sourceRoot":"","sources":["../../../../../../../../packages/core/primitives/event-dispatch/src/eventcontract.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @fileoverview Implements the local event handling contract. This\n * allows DOM objects in a container that enters into this contract to\n * define event handlers which are executed in a local context.\n *\n * One EventContract instance can manage the contract for multiple\n * containers, which are added using the addContainer() method.\n *\n * Events can be registered using the addEvent() method.\n *\n * A Dispatcher is added using the registerDispatcher() method. Until there is\n * a dispatcher, events are queued. The idea is that the EventContract\n * class is inlined in the HTML of the top level page and instantiated\n * right after the start of <body>. The Dispatcher class is contained\n * in the external deferred js, and instantiated and registered with\n * EventContract when the external javascript in the page loads. The\n * external javascript will also register the jsaction handlers, which\n * then pick up the queued events at the time of registration.\n *\n * Since this class is meant to be inlined in the main page HTML, the\n * size of the binary compiled from this file MUST be kept as small as\n * possible and thus its dependencies to a minimum.\n */\n\nimport * as a11yClickLib from './a11y_click';\nimport {Attribute} from './attribute';\nimport * as cache from './cache';\nimport {Char} from './char';\nimport {EarlyJsactionData} from './earlyeventcontract';\nimport * as eventLib from './event';\nimport {EventContractContainerManager} from './event_contract_container';\nimport {\n  A11Y_CLICK_SUPPORT,\n  CUSTOM_EVENT_SUPPORT,\n  JSNAMESPACE_SUPPORT,\n  MOUSE_SPECIAL_SUPPORT,\n  STOP_PROPAGATION,\n} from './event_contract_defines';\nimport * as eventInfoLib from './event_info';\nimport {EventType} from './event_type';\nimport {Property} from './property';\nimport {Restriction} from './restriction';\n\n/**\n * The API of an EventContract that is safe to call from any compilation unit.\n */\nexport declare interface UnrenamedEventContract {\n  // Alias for Jsction EventContract registerDispatcher.\n  ecrd(dispatcher: Dispatcher, restriction: Restriction): void;\n  // Unrenamed function. Abbreviation for `eventContract.addA11yClickSupport`.\n  ecaacs?: (\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) => void;\n}\n\n/** A function that is called to handle events captured by the EventContract. */\nexport type Dispatcher = (eventInfo: eventInfoLib.EventInfo, globalDispatch?: boolean) => void;\n\n/**\n * A function that handles an event dispatched from the browser.\n *\n * eventType: May differ from `event.type` if JSAction uses a\n * short-hand name or is patching over an non-bubbling event with a bubbling\n * variant.\n * event: The native browser event.\n * container: The container for this dispatch.\n */\ntype EventHandler = (eventType: string, event: Event, container: Element) => void;\n\nconst DEFAULT_EVENT_TYPE: string = EventType.CLICK;\n\n/**\n * Since maps from event to action are immutable we can use a single map\n * to represent the empty map.\n */\nconst EMPTY_ACTION_MAP: {[key: string]: string} = {};\n\n/**\n * This regular expression matches a semicolon.\n */\nconst REGEXP_SEMICOLON = /\\s*;\\s*/;\n\n/**\n * EventContract intercepts events in the bubbling phase at the\n * boundary of a container element, and maps them to generic actions\n * which are specified using the custom jsaction attribute in\n * HTML. Behavior of the application is then specified in terms of\n * handler for such actions, cf. jsaction.Dispatcher in dispatcher.js.\n *\n * This has several benefits: (1) No DOM event handlers need to be\n * registered on the specific elements in the UI. (2) The set of\n * events that the application has to handle can be specified in terms\n * of the semantics of the application, rather than in terms of DOM\n * events. (3) Invocation of handlers can be delayed and handlers can\n * be delay loaded in a generic way.\n */\nexport class EventContract implements UnrenamedEventContract {\n  static CUSTOM_EVENT_SUPPORT = CUSTOM_EVENT_SUPPORT;\n  static STOP_PROPAGATION = STOP_PROPAGATION;\n  static A11Y_CLICK_SUPPORT = A11Y_CLICK_SUPPORT;\n  static MOUSE_SPECIAL_SUPPORT = MOUSE_SPECIAL_SUPPORT;\n  static JSNAMESPACE_SUPPORT = JSNAMESPACE_SUPPORT;\n\n  private containerManager: EventContractContainerManager | null;\n\n  /**\n   * The DOM events which this contract covers. Used to prevent double\n   * registration of event types. The value of the map is the\n   * internally created DOM event handler function that handles the\n   * DOM events. See addEvent().\n   *\n   */\n  private eventHandlers: {[key: string]: EventHandler} = {};\n\n  private browserEventTypeToExtraEventTypes: {[key: string]: string[]} = {};\n\n  /**\n   * The dispatcher function. Events are passed to this function for\n   * handling once it was set using the registerDispatcher() method. This is\n   * done because the function is passed from another jsbinary, so passing the\n   * instance and invoking the method here would require to leave the method\n   * unobfuscated.\n   */\n  private dispatcher: Dispatcher | null = null;\n\n  /**\n   * The list of suspended `EventInfo` that will be dispatched\n   * as soon as the `Dispatcher` is registered.\n   */\n  private queuedEventInfos: eventInfoLib.EventInfo[] | null = [];\n\n  /** Whether a11y click support has been loaded or not. */\n  private hasA11yClickSupport = false;\n  /** Whether to add an a11y click listener. */\n  private addA11yClickListener = false;\n\n  private updateEventInfoForA11yClick?: (eventInfo: eventInfoLib.EventInfo) => void = undefined;\n\n  private preventDefaultForA11yClick?: (eventInfo: eventInfoLib.EventInfo) => void = undefined;\n\n  private populateClickOnlyAction?: (\n    actionElement: Element,\n    eventInfo: eventInfoLib.EventInfo,\n    actionMap: {[key: string]: string},\n  ) => void = undefined;\n\n  ecaacs?: (\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) => void;\n\n  constructor(containerManager: EventContractContainerManager) {\n    this.containerManager = containerManager;\n    if (EventContract.CUSTOM_EVENT_SUPPORT) {\n      this.addEvent(EventType.CUSTOM);\n    }\n    if (EventContract.A11Y_CLICK_SUPPORT) {\n      // Add a11y click support to the `EventContract`.\n      this.addA11yClickSupport();\n    }\n  }\n\n  private handleEvent(eventType: string, event: Event, container: Element) {\n    const eventInfo = eventInfoLib.createEventInfoFromParameters(\n      /* eventType= */ eventType,\n      /* event= */ event,\n      /* targetElement= */ event.target as Element,\n      /* container= */ container,\n      /* timestamp= */ Date.now(),\n    );\n    this.handleEventInfo(eventInfo);\n  }\n\n  /**\n   * Handle an `EventInfo`.\n   */\n  private handleEventInfo(eventInfo: eventInfoLib.EventInfo) {\n    if (!this.dispatcher) {\n      // All events are queued when the dispatcher isn't yet loaded.\n      eventInfoLib.setIsReplay(eventInfo, true);\n      this.queuedEventInfos?.push(eventInfo);\n    }\n    if (\n      EventContract.CUSTOM_EVENT_SUPPORT &&\n      eventInfoLib.getEventType(eventInfo) === EventType.CUSTOM\n    ) {\n      const detail = (eventInfoLib.getEvent(eventInfo) as CustomEvent).detail;\n      // For custom events, use a secondary dispatch based on the internal\n      // custom type of the event.\n      if (!detail || !detail['_type']) {\n        // This should never happen.\n        return;\n      }\n      eventInfoLib.setEventType(eventInfo, detail['_type']);\n    }\n\n    this.populateAction(eventInfo);\n\n    if (!this.dispatcher) {\n      return;\n    }\n    const globalEventInfo: eventInfoLib.EventInfo = eventInfoLib.cloneEventInfo(eventInfo);\n\n    // In some cases, `populateAction` will rewrite `click` events to\n    // `clickonly`. Revert back to a regular click, otherwise we won't be able\n    // to execute global event handlers registered on click events.\n    if (eventInfoLib.getEventType(globalEventInfo) === EventType.CLICKONLY) {\n      eventInfoLib.setEventType(globalEventInfo, EventType.CLICK);\n    }\n\n    this.dispatcher(globalEventInfo, /* dispatch global event */ true);\n\n    const action = eventInfoLib.getAction(eventInfo);\n    if (!action) {\n      return;\n    }\n    if (shouldPreventDefaultBeforeDispatching(eventInfoLib.getActionElement(action), eventInfo)) {\n      eventLib.preventDefault(eventInfoLib.getEvent(eventInfo));\n    }\n\n    this.dispatcher(eventInfo);\n  }\n\n  /**\n   * Searches for a jsaction that the DOM event maps to and creates an\n   * object containing event information used for dispatching by\n   * jsaction.Dispatcher. This method populates the `action` and `actionElement`\n   * fields of the EventInfo object passed in by finding the first\n   * jsaction attribute above the target Node of the event, and below\n   * the container Node, that specifies a jsaction for the event\n   * type. If no such jsaction is found, then action is undefined.\n   *\n   * @param eventInfo `EventInfo` to set `action` and `actionElement` if an\n   *    action is found on any `Element` in the path of the `Event`.\n   */\n  private populateAction(eventInfo: eventInfoLib.EventInfo) {\n    // We distinguish modified and plain clicks in order to support the\n    // default browser behavior of modified clicks on links; usually to\n    // open the URL of the link in new tab or new window on ctrl/cmd\n    // click. A DOM 'click' event is mapped to the jsaction 'click'\n    // event iff there is no modifier present on the event. If there is\n    // a modifier, it's mapped to 'clickmod' instead.\n    //\n    // It's allowed to omit the event in the jsaction attribute. In that\n    // case, 'click' is assumed. Thus the following two are equivalent:\n    //\n    //   <a href=\"someurl\" jsaction=\"gna.fu\">\n    //   <a href=\"someurl\" jsaction=\"click:gna.fu\">\n    //\n    // For unmodified clicks, EventContract invokes the jsaction\n    // 'gna.fu'. For modified clicks, EventContract won't find a\n    // suitable action and leave the event to be handled by the\n    // browser.\n    //\n    // In order to also invoke a jsaction handler for a modifier click,\n    // 'clickmod' needs to be used:\n    //\n    //   <a href=\"someurl\" jsaction=\"clickmod:gna.fu\">\n    //\n    // EventContract invokes the jsaction 'gna.fu' for modified\n    // clicks. Unmodified clicks are left to the browser.\n    //\n    // In order to set up the event contract to handle both clickonly and\n    // clickmod, only addEvent(EventType.CLICK) is necessary.\n    //\n    // In order to set up the event contract to handle click,\n    // addEvent() is necessary for CLICK, KEYDOWN, and KEYPRESS event types.  If\n    // a11y click support is enabled, addEvent() will set up the appropriate key\n    // event handler automatically.\n    if (\n      eventInfoLib.getEventType(eventInfo) === EventType.CLICK &&\n      eventLib.isModifiedClickEvent(eventInfoLib.getEvent(eventInfo))\n    ) {\n      eventInfoLib.setEventType(eventInfo, EventType.CLICKMOD);\n    } else if (this.hasA11yClickSupport) {\n      this.updateEventInfoForA11yClick!(eventInfo);\n    }\n\n    // Walk to the parent node, unless the node has a different owner in\n    // which case we walk to the owner. Attempt to walk to host of a\n    // shadow root if needed.\n    let actionElement: Element | null = eventInfoLib.getTargetElement(eventInfo);\n    while (actionElement && actionElement !== eventInfoLib.getContainer(eventInfo)) {\n      this.populateActionOnElement(actionElement, eventInfo);\n\n      if (eventInfoLib.getAction(eventInfo)) {\n        // An event is handled by at most one jsaction. Thus we stop at the\n        // first matching jsaction specified in a jsaction attribute up the\n        // ancestor chain of the event target node.\n        break;\n      }\n      if (actionElement[Property.OWNER]) {\n        actionElement = actionElement[Property.OWNER] as Element;\n        continue;\n      }\n      if (actionElement.parentNode?.nodeName !== '#document-fragment') {\n        actionElement = actionElement.parentNode as Element | null;\n      } else {\n        actionElement = (actionElement.parentNode as ShadowRoot | null)?.host ?? null;\n      }\n    }\n\n    const action = eventInfoLib.getAction(eventInfo);\n    if (!action) {\n      // No action found.\n      return;\n    }\n\n    if (this.hasA11yClickSupport) {\n      this.preventDefaultForA11yClick!(eventInfo);\n    }\n\n    // We attempt to handle the mouseenter/mouseleave events here by\n    // detecting whether the mouseover/mouseout events correspond to\n    // entering/leaving an element.\n    if (\n      EventContract.MOUSE_SPECIAL_SUPPORT &&\n      (eventInfoLib.getEventType(eventInfo) === EventType.MOUSEENTER ||\n        eventInfoLib.getEventType(eventInfo) === EventType.MOUSELEAVE ||\n        eventInfoLib.getEventType(eventInfo) === EventType.POINTERENTER ||\n        eventInfoLib.getEventType(eventInfo) === EventType.POINTERLEAVE)\n    ) {\n      // We attempt to handle the mouseenter/mouseleave events here by\n      // detecting whether the mouseover/mouseout events correspond to\n      // entering/leaving an element.\n      if (\n        eventLib.isMouseSpecialEvent(\n          eventInfoLib.getEvent(eventInfo),\n          eventInfoLib.getEventType(eventInfo),\n          eventInfoLib.getActionElement(action),\n        )\n      ) {\n        // If both mouseover/mouseout and mouseenter/mouseleave events are\n        // enabled, two separate handlers for mouseover/mouseout are\n        // registered. Both handlers will see the same event instance\n        // so we create a copy to avoid interfering with the dispatching of\n        // the mouseover/mouseout event.\n        const copiedEvent = eventLib.createMouseSpecialEvent(\n          eventInfoLib.getEvent(eventInfo),\n          eventInfoLib.getActionElement(action),\n        );\n        eventInfoLib.setEvent(eventInfo, copiedEvent);\n        // Since the mouseenter/mouseleave events do not bubble, the target\n        // of the event is technically the `actionElement` (the node with the\n        // `jsaction` attribute)\n        eventInfoLib.setTargetElement(eventInfo, eventInfoLib.getActionElement(action));\n      } else {\n        eventInfoLib.unsetAction(eventInfo);\n      }\n    }\n  }\n\n  /**\n   * Accesses the jsaction map on a node and retrieves the name of the\n   * action the given event is mapped to, if any. It parses the\n   * attribute value and stores it in a property on the node for\n   * subsequent retrieval without re-parsing and re-accessing the\n   * attribute. In order to fully qualify jsaction names using a\n   * namespace, the DOM is searched starting at the current node and\n   * going through ancestor nodes until a jsnamespace attribute is\n   * found.\n   *\n   * @param actionElement The DOM node to retrieve the jsaction map from.\n   * @param eventInfo `EventInfo` to set `action` and `actionElement` if an\n   *    action is found on the `actionElement`.\n   */\n  private populateActionOnElement(actionElement: Element, eventInfo: eventInfoLib.EventInfo) {\n    const actionMap = parseActions(actionElement, eventInfoLib.getContainer(eventInfo));\n\n    const actionName = actionMap[eventInfoLib.getEventType(eventInfo)];\n    if (actionName !== undefined) {\n      eventInfoLib.setAction(eventInfo, actionName, actionElement);\n    }\n\n    if (this.hasA11yClickSupport) {\n      this.populateClickOnlyAction!(actionElement, eventInfo, actionMap);\n    }\n  }\n\n  /**\n   * Enables jsaction handlers to be called for the event type given by\n   * name.\n   *\n   * If the event is already registered, this does nothing.\n   *\n   * @param prefixedEventType If supplied, this event is used in\n   *     the actual browser event registration instead of the name that is\n   *     exposed to jsaction. Use this if you e.g. want users to be able\n   *     to subscribe to jsaction=\"transitionEnd:foo\" while the underlying\n   *     event is webkitTransitionEnd in one browser and mozTransitionEnd\n   *     in another.\n   */\n  addEvent(eventType: string, prefixedEventType?: string) {\n    if (eventType in this.eventHandlers || !this.containerManager) {\n      return;\n    }\n\n    if (\n      !EventContract.MOUSE_SPECIAL_SUPPORT &&\n      (eventType === EventType.MOUSEENTER ||\n        eventType === EventType.MOUSELEAVE ||\n        eventType === EventType.POINTERENTER ||\n        eventType === EventType.POINTERLEAVE)\n    ) {\n      return;\n    }\n\n    const eventHandler = (eventType: string, event: Event, container: Element) => {\n      this.handleEvent(eventType, event, container);\n    };\n\n    // Store the callback to allow us to replay events.\n    this.eventHandlers[eventType] = eventHandler;\n\n    const browserEventType = eventLib.getBrowserEventType(prefixedEventType || eventType);\n\n    if (browserEventType !== eventType) {\n      const eventTypes = this.browserEventTypeToExtraEventTypes[browserEventType] || [];\n      eventTypes.push(eventType);\n      this.browserEventTypeToExtraEventTypes[browserEventType] = eventTypes;\n    }\n\n    this.containerManager.addEventListener(browserEventType, (element: Element) => {\n      return (event: Event) => {\n        eventHandler(eventType, event, element);\n      };\n    });\n\n    // Automatically install a keypress/keydown event handler if support for\n    // accessible clicks is turned on.\n    if (this.addA11yClickListener && eventType === EventType.CLICK) {\n      this.addEvent(EventType.KEYDOWN);\n    }\n  }\n\n  /**\n   * Gets the queued early events and replay them using the appropriate handler\n   * in the provided event contract. Once all the events are replayed, it cleans\n   * up the early contract.\n   */\n  replayEarlyEvents() {\n    // Check if the early contract is present and prevent calling this function\n    // more than once.\n    const earlyJsactionData: EarlyJsactionData | undefined = window._ejsa;\n    if (!earlyJsactionData) {\n      return;\n    }\n\n    // Replay the early contract events.\n    const earlyEventInfos: eventInfoLib.EventInfo[] = earlyJsactionData.q;\n    for (let idx = 0; idx < earlyEventInfos.length; idx++) {\n      const earlyEventInfo: eventInfoLib.EventInfo = earlyEventInfos[idx];\n      const eventTypes = this.getEventTypesForBrowserEventType(earlyEventInfo.eventType);\n      for (let i = 0; i < eventTypes.length; i++) {\n        const eventInfo = eventInfoLib.cloneEventInfo(earlyEventInfo);\n        // EventInfo eventType maps to JSAction's internal event type,\n        // rather than the browser event type.\n        eventInfoLib.setEventType(eventInfo, eventTypes[i]);\n        this.handleEventInfo(eventInfo);\n      }\n    }\n\n    // Clean up the early contract.\n    const earlyEventTypes: string[] = earlyJsactionData.et;\n    const earlyEventHandler: (event: Event) => void = earlyJsactionData.h;\n    for (let idx = 0; idx < earlyEventTypes.length; idx++) {\n      const eventType: string = earlyEventTypes[idx];\n      window.document.documentElement.removeEventListener(eventType, earlyEventHandler);\n    }\n    delete window._ejsa;\n  }\n\n  /**\n   * Returns all JSAction event types that have been registered for a given\n   * browser event type.\n   */\n  private getEventTypesForBrowserEventType(browserEventType: string) {\n    const eventTypes = [];\n    if (this.eventHandlers[browserEventType]) {\n      eventTypes.push(browserEventType);\n    }\n    if (this.browserEventTypeToExtraEventTypes[browserEventType]) {\n      eventTypes.push(...this.browserEventTypeToExtraEventTypes[browserEventType]);\n    }\n    return eventTypes;\n  }\n\n  /**\n   * Returns the event handler function for a given event type.\n   */\n  handler(eventType: string): EventHandler | undefined {\n    return this.eventHandlers[eventType];\n  }\n\n  /**\n   * Cleans up the event contract. This resets all of the `EventContract`'s\n   * internal state. Users are responsible for not using this `EventContract`\n   * after it has been cleaned up.\n   */\n  cleanUp() {\n    this.containerManager!.cleanUp();\n    this.containerManager = null;\n    this.eventHandlers = {};\n    this.browserEventTypeToExtraEventTypes = {};\n    this.dispatcher = null;\n    this.queuedEventInfos = [];\n  }\n\n  /**\n   * Register a dispatcher function. Event info of each event mapped to\n   * a jsaction is passed for handling to this callback. The queued\n   * events are passed as well to the dispatcher for later replaying\n   * once the dispatcher is registered. Clears the event queue to null.\n   *\n   * @param dispatcher The dispatcher function.\n   * @param restriction\n   */\n  registerDispatcher(dispatcher: Dispatcher, restriction: Restriction) {\n    this.ecrd(dispatcher, restriction);\n  }\n\n  /**\n   * Unrenamed alias for registerDispatcher. Necessary for any codebases that\n   * split the `EventContract` and `Dispatcher` code into different compilation\n   * units.\n   */\n  ecrd(dispatcher: Dispatcher, restriction: Restriction) {\n    this.dispatcher = dispatcher;\n\n    if (this.queuedEventInfos?.length) {\n      for (let i = 0; i < this.queuedEventInfos.length; i++) {\n        this.handleEventInfo(this.queuedEventInfos[i]);\n      }\n      this.queuedEventInfos = null;\n    }\n  }\n\n  /**\n   * Adds a11y click support to the given `EventContract`. Meant to be called in\n   * the same compilation unit as the `EventContract`.\n   */\n  addA11yClickSupport() {\n    this.addA11yClickSupportImpl(\n      a11yClickLib.updateEventInfoForA11yClick,\n      a11yClickLib.preventDefaultForA11yClick,\n      a11yClickLib.populateClickOnlyAction,\n    );\n  }\n\n  /**\n   * Enables a11y click support to be deferred. Meant to be called in the same\n   * compilation unit as the `EventContract`.\n   */\n  exportAddA11yClickSupport() {\n    this.addA11yClickListener = true;\n    this.ecaacs = this.addA11yClickSupportImpl.bind(this);\n  }\n\n  /**\n   * Unrenamed function that loads a11yClickSupport.\n   */\n  private addA11yClickSupportImpl(\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) {\n    this.addA11yClickListener = true;\n    this.hasA11yClickSupport = true;\n    this.updateEventInfoForA11yClick = updateEventInfoForA11yClick;\n    this.preventDefaultForA11yClick = preventDefaultForA11yClick;\n    this.populateClickOnlyAction = populateClickOnlyAction;\n  }\n}\n\n/**\n * Adds a11y click support to the given `EventContract`. Meant to be called\n * in a different compilation unit from the `EventContract`. The `EventContract`\n * must have called `exportAddA11yClickSupport` in its compilation unit for this\n * to have any effect.\n */\nexport function addDeferredA11yClickSupport(eventContract: EventContract) {\n  eventContract.ecaacs?.(\n    a11yClickLib.updateEventInfoForA11yClick,\n    a11yClickLib.preventDefaultForA11yClick,\n    a11yClickLib.populateClickOnlyAction,\n  );\n}\n\n/**\n * Returns true if the default action of this event should be prevented before\n * this event is dispatched.\n */\nfunction shouldPreventDefaultBeforeDispatching(\n  actionElement: Element,\n  eventInfo: eventInfoLib.EventInfo,\n): boolean {\n  // Prevent browser from following <a> node links if a jsaction is present\n  // and we are dispatching the action now. Note that the targetElement may be\n  // a child of an anchor that has a jsaction attached. For that reason, we\n  // need to check the actionElement rather than the targetElement.\n  return (\n    actionElement.tagName === 'A' &&\n    (eventInfoLib.getEventType(eventInfo) === EventType.CLICK ||\n      eventInfoLib.getEventType(eventInfo) === EventType.CLICKMOD)\n  );\n}\n\n/**\n * Parses and caches an element's jsaction element into a map.\n *\n * This is primarily for internal use.\n *\n * @param actionElement The DOM node to retrieve the jsaction map from.\n * @param container The node which limits the namespace lookup for a jsaction\n * name. The container node itself will not be searched.\n * @return Map from event to qualified name of the jsaction bound to it.\n */\nexport function parseActions(actionElement: Element, container: Node): {[key: string]: string} {\n  let actionMap: {[key: string]: string} | undefined = cache.get(actionElement);\n  if (!actionMap) {\n    const jsactionAttribute = getAttr(actionElement, Attribute.JSACTION);\n    if (!jsactionAttribute) {\n      actionMap = EMPTY_ACTION_MAP;\n      cache.set(actionElement, actionMap);\n    } else {\n      actionMap = cache.getParsed(jsactionAttribute);\n      if (!actionMap) {\n        actionMap = {};\n        const values = jsactionAttribute.split(REGEXP_SEMICOLON);\n        for (let idx = 0; idx < values.length; idx++) {\n          const value = values[idx];\n          if (!value) {\n            continue;\n          }\n          const colon = value.indexOf(Char.EVENT_ACTION_SEPARATOR);\n          const hasColon = colon !== -1;\n          const type = hasColon ? stringTrim(value.substr(0, colon)) : DEFAULT_EVENT_TYPE;\n          const action = hasColon ? stringTrim(value.substr(colon + 1)) : value;\n          actionMap[type] = action;\n        }\n        cache.setParsed(jsactionAttribute, actionMap);\n      }\n      // If namespace support is active we need to augment the (potentially\n      // cached) jsaction mapping with the namespace.\n      if (EventContract.JSNAMESPACE_SUPPORT) {\n        const noNs = actionMap;\n        actionMap = {};\n        for (const type in noNs) {\n          actionMap[type] = getFullyQualifiedAction(noNs[type], actionElement, container);\n        }\n      }\n      cache.set(actionElement, actionMap);\n    }\n  }\n  return actionMap;\n}\n\n/**\n * Returns the fully qualified jsaction action. If the given jsaction\n * name doesn't already contain the namespace, the function iterates\n * over ancestor nodes until a jsnamespace attribute is found, and\n * uses the value of that attribute as the namespace.\n *\n * @param action The jsaction action to resolve.\n * @param start The node from which to start searching for a jsnamespace\n * attribute.\n * @param container The node which limits the search for a jsnamespace\n * attribute. This node will be searched.\n * @return The fully qualified name of the jsaction. If no namespace is found,\n * returns the unqualified name in case it exists in the global namespace.\n */\nfunction getFullyQualifiedAction(action: string, start: Element, container: Node): string {\n  if (EventContract.JSNAMESPACE_SUPPORT) {\n    if (isNamespacedAction(action)) {\n      return action;\n    }\n\n    let node: Node | null = start;\n    while (node) {\n      const namespace = getNamespaceFromElement(node as Element);\n      if (namespace) {\n        return namespace + Char.NAMESPACE_ACTION_SEPARATOR + action;\n      }\n\n      // If this node is the container, stop.\n      if (node === container) {\n        break;\n      }\n\n      node = node.parentNode;\n    }\n  }\n\n  return action;\n}\n\n/**\n * Checks if a jsaction action contains a namespace part.\n */\nfunction isNamespacedAction(action: string): boolean {\n  return action.indexOf(Char.NAMESPACE_ACTION_SEPARATOR) >= 0;\n}\n\n/**\n * Returns the value of the jsnamespace attribute of the given node.\n * Also caches the value for subsequent lookups.\n * @param element The node whose jsnamespace attribute is being asked for.\n * @return The value of the jsnamespace attribute, or null if not found.\n */\nfunction getNamespaceFromElement(element: Element): string | null {\n  let namespace = cache.getNamespace(element);\n  // Only query for the attribute if it has not been queried for\n  // before. getAttr() returns null if an attribute is not present. Thus,\n  // namespace is string|null if the query took place in the past, or\n  // undefined if the query did not take place.\n  if (namespace === undefined) {\n    namespace = getAttr(element, Attribute.JSNAMESPACE);\n    cache.setNamespace(element, namespace);\n  }\n  return namespace;\n}\n\n/**\n * Accesses the event handler attribute value of a DOM node. It guards\n * against weird situations (described in the body) that occur in\n * connection with nodes that are removed from their document.\n * @param element The DOM element.\n * @param attribute The name of the attribute to access.\n * @return The attribute value if it was found, null otherwise.\n */\nfunction getAttr(element: Element, attribute: string): string | null {\n  let value = null;\n  // NOTE: Nodes in IE do not always have a getAttribute\n  // method defined. This is the case where sourceElement has in\n  // fact been removed from the DOM before eventContract begins\n  // handling - where a parentNode does not have getAttribute\n  // defined.\n  // NOTE: We must use the 'in' operator instead of the regular dot\n  // notation, since the latter fails in IE8 if the getAttribute method is not\n  // defined. See b/7139109.\n  if ('getAttribute' in element) {\n    value = element.getAttribute(attribute);\n  }\n  return value;\n}\n\n/**\n * Helper function to trim whitespace from the beginning and the end\n * of the string. This deliberately doesn't use the closure equivalent\n * to keep dependencies small.\n * @param str  Input string.\n * @return  Trimmed string.\n */\nfunction stringTrim(str: string): string {\n  if (typeof String.prototype.trim === 'function') {\n    return str.trim();\n  }\n\n  const trimmedLeft = str.replace(/^\\s+/, '');\n  return trimmedLeft.replace(/\\s+$/, '');\n}\n"]}
|
|
290
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"eventcontract.js","sourceRoot":"","sources":["../../../../../../../../packages/core/primitives/event-dispatch/src/eventcontract.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAEH;;;;;;;;;;;;;;;;;;;;;;GAsBG;AAEH,OAAO,KAAK,YAAY,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAC,cAAc,EAAC,MAAM,mBAAmB,CAAC;AAEjD,OAAO,KAAK,QAAQ,MAAM,SAAS,CAAC;AAEpC,OAAO,EAAC,kBAAkB,EAAE,qBAAqB,EAAC,MAAM,0BAA0B,CAAC;AACnF,OAAO,KAAK,YAAY,MAAM,cAAc,CAAC;AAC7C,OAAO,EAAC,SAAS,EAAC,MAAM,cAAc,CAAC;AA+BvC;;;;;;;;;;;;;GAaG;AACH,MAAM,OAAO,aAAa;aACjB,uBAAkB,GAAG,kBAAkB,AAArB,CAAsB;aACxC,0BAAqB,GAAG,qBAAqB,AAAxB,CAAyB;IAyCrD,YACE,gBAA+C,EAC9B,oBAAoB,IAAI;QAAxB,sBAAiB,GAAjB,iBAAiB,CAAO;QArC3C;;;;;;WAMG;QACK,kBAAa,GAAkC,EAAE,CAAC;QAElD,sCAAiC,GAA8B,EAAE,CAAC;QAE1E;;;;;;WAMG;QACK,eAAU,GAAsB,IAAI,CAAC;QAE7C;;;WAGG;QACK,qBAAgB,GAAoC,EAAE,CAAC;QAE/D,6CAA6C;QACrC,yBAAoB,GAAG,KAAK,CAAC;QAYnC,IAAI,CAAC,gBAAgB,GAAG,gBAAgB,CAAC;QACzC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,cAAc,GAAG,IAAI,cAAc,CAAC;gBACvC,0BAA0B,EAAE,aAAa,CAAC,qBAAqB;aAChE,CAAC,CAAC;QACL,CAAC;QACD,IAAI,aAAa,CAAC,kBAAkB,EAAE,CAAC;YACrC,iDAAiD;YACjD,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC7B,CAAC;IACH,CAAC;IAEO,WAAW,CAAC,SAAiB,EAAE,KAAY,EAAE,SAAkB;QACrE,MAAM,SAAS,GAAG,YAAY,CAAC,6BAA6B;QAC1D,gBAAgB,CAAC,SAAS;QAC1B,YAAY,CAAC,KAAK;QAClB,oBAAoB,CAAC,KAAK,CAAC,MAAiB;QAC5C,gBAAgB,CAAC,SAAS;QAC1B,gBAAgB,CAAC,IAAI,CAAC,GAAG,EAAE,CAC5B,CAAC;QACF,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC;IAClC,CAAC;IAED;;OAEG;IACK,eAAe,CAAC,SAAiC;QACvD,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACrB,8DAA8D;YAC9D,YAAY,CAAC,WAAW,CAAC,SAAS,EAAE,IAAI,CAAC,CAAC;YAC1C,IAAI,CAAC,gBAAgB,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;YACvC,OAAO;QACT,CAAC;QACD,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,cAAe,CAAC,OAAO,CAAC,SAAS,CAAC,CAAC;QAC1C,CAAC;QACD,IAAI,CAAC,UAAU,CAAC,SAAS,CAAC,CAAC;IAC7B,CAAC;IAED;;;;;;;;;;;;OAYG;IACH,QAAQ,CAAC,SAAiB,EAAE,iBAA0B;QACpD,IAAI,SAAS,IAAI,IAAI,CAAC,aAAa,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE,CAAC;YAC9D,OAAO;QACT,CAAC;QAED,IACE,CAAC,aAAa,CAAC,qBAAqB;YACpC,CAAC,SAAS,KAAK,SAAS,CAAC,UAAU;gBACjC,SAAS,KAAK,SAAS,CAAC,UAAU;gBAClC,SAAS,KAAK,SAAS,CAAC,YAAY;gBACpC,SAAS,KAAK,SAAS,CAAC,YAAY,CAAC,EACvC,CAAC;YACD,OAAO;QACT,CAAC;QAED,MAAM,YAAY,GAAG,CAAC,SAAiB,EAAE,KAAY,EAAE,SAAkB,EAAE,EAAE;YAC3E,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,KAAK,EAAE,SAAS,CAAC,CAAC;QAChD,CAAC,CAAC;QAEF,mDAAmD;QACnD,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,GAAG,YAAY,CAAC;QAE7C,MAAM,gBAAgB,GAAG,QAAQ,CAAC,mBAAmB,CAAC,iBAAiB,IAAI,SAAS,CAAC,CAAC;QAEtF,IAAI,gBAAgB,KAAK,SAAS,EAAE,CAAC;YACnC,MAAM,UAAU,GAAG,IAAI,CAAC,iCAAiC,CAAC,gBAAgB,CAAC,IAAI,EAAE,CAAC;YAClF,UAAU,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;YAC3B,IAAI,CAAC,iCAAiC,CAAC,gBAAgB,CAAC,GAAG,UAAU,CAAC;QACxE,CAAC;QAED,IAAI,CAAC,gBAAgB,CAAC,gBAAgB,CAAC,gBAAgB,EAAE,CAAC,OAAgB,EAAE,EAAE;YAC5E,OAAO,CAAC,KAAY,EAAE,EAAE;gBACtB,YAAY,CAAC,SAAS,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;YAC1C,CAAC,CAAC;QACJ,CAAC,CAAC,CAAC;QAEH,wEAAwE;QACxE,kCAAkC;QAClC,IAAI,IAAI,CAAC,oBAAoB,IAAI,SAAS,KAAK,SAAS,CAAC,KAAK,EAAE,CAAC;YAC/D,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,OAAO,CAAC,CAAC;QACnC,CAAC;IACH,CAAC;IAED;;;;OAIG;IACH,iBAAiB,CACf,yBAAqD,MAAoC;QAEzF,2EAA2E;QAC3E,kBAAkB;QAClB,MAAM,iBAAiB,GAAkC,sBAAsB,CAAC,KAAK,CAAC;QACtF,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QAED,oCAAoC;QACpC,MAAM,eAAe,GAA6B,iBAAiB,CAAC,CAAC,CAAC;QACtE,KAAK,IAAI,GAAG,GAAG,CAAC,EAAE,GAAG,GAAG,eAAe,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE,CAAC;YACtD,MAAM,cAAc,GAA2B,eAAe,CAAC,GAAG,CAAC,CAAC;YACpE,MAAM,UAAU,GAAG,IAAI,CAAC,gCAAgC,CAAC,cAAc,CAAC,SAAS,CAAC,CAAC;YACnF,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBAC3C,MAAM,SAAS,GAAG,YAAY,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;gBAC9D,8DAA8D;gBAC9D,sCAAsC;gBACtC,YAAY,CAAC,YAAY,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;gBACpD,IAAI,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC;YAClC,CAAC;QACH,CAAC;QAED,+BAA+B;QAC/B,MAAM,iBAAiB,GAA2B,iBAAiB,CAAC,CAAC,CAAC;QACtE,oBAAoB,CAAC,iBAAiB,CAAC,CAAC,EAAE,iBAAiB,CAAC,EAAE,EAAE,iBAAiB,CAAC,CAAC;QACnF,oBAAoB,CAAC,iBAAiB,CAAC,CAAC,EAAE,iBAAiB,CAAC,GAAG,EAAE,iBAAiB,EAAE,IAAI,CAAC,CAAC;QAC1F,OAAO,sBAAsB,CAAC,KAAK,CAAC;IACtC,CAAC;IAED;;;OAGG;IACK,gCAAgC,CAAC,gBAAwB;QAC/D,MAAM,UAAU,GAAG,EAAE,CAAC;QACtB,IAAI,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,EAAE,CAAC;YACzC,UAAU,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QACpC,CAAC;QACD,IAAI,IAAI,CAAC,iCAAiC,CAAC,gBAAgB,CAAC,EAAE,CAAC;YAC7D,UAAU,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,iCAAiC,CAAC,gBAAgB,CAAC,CAAC,CAAC;QAC/E,CAAC;QACD,OAAO,UAAU,CAAC;IACpB,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,SAAiB;QACvB,OAAO,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,CAAC;IACvC,CAAC;IAED;;;;OAIG;IACH,OAAO;QACL,IAAI,CAAC,gBAAiB,CAAC,OAAO,EAAE,CAAC;QACjC,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC7B,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,IAAI,CAAC,iCAAiC,GAAG,EAAE,CAAC;QAC5C,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC;QACvB,IAAI,CAAC,gBAAgB,GAAG,EAAE,CAAC;IAC7B,CAAC;IAED;;;;;;;;OAQG;IACH,kBAAkB,CAAC,UAAsB,EAAE,WAAwB;QACjE,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,WAAW,CAAC,CAAC;IACrC,CAAC;IAED;;;;OAIG;IACH,IAAI,CAAC,UAAsB,EAAE,WAAwB;QACnD,IAAI,CAAC,UAAU,GAAG,UAAU,CAAC;QAE7B,IAAI,IAAI,CAAC,gBAAgB,EAAE,MAAM,EAAE,CAAC;YAClC,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;gBACtD,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC,CAAC;YACjD,CAAC;YACD,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC;QAC/B,CAAC;IACH,CAAC;IAED;;;OAGG;IACH,mBAAmB;QACjB,IAAI,CAAC,uBAAuB,CAC1B,YAAY,CAAC,2BAA2B,EACxC,YAAY,CAAC,0BAA0B,EACvC,YAAY,CAAC,uBAAuB,CACrC,CAAC;IACJ,CAAC;IAED;;;OAGG;IACH,yBAAyB;QACvB,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;QACjC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;IACxD,CAAC;IAED;;OAEG;IACK,uBAAuB,CAC7B,2BAA4E,EAC5E,0BAA0E,EAC1E,uBAAoE;QAEpE,IAAI,CAAC,oBAAoB,GAAG,IAAI,CAAC;QACjC,IAAI,IAAI,CAAC,iBAAiB,EAAE,CAAC;YAC3B,IAAI,CAAC,cAAe,CAAC,mBAAmB,CACtC,2BAA2B,EAC3B,0BAA0B,EAC1B,uBAAuB,CACxB,CAAC;QACJ,CAAC;IACH,CAAC;;AAGH,SAAS,oBAAoB,CAC3B,SAAsB,EACtB,UAAoB,EACpB,iBAAqC,EACrC,OAAiB;IAEjB,KAAK,IAAI,GAAG,GAAG,CAAC,EAAE,GAAG,GAAG,UAAU,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE,CAAC;QACjD,SAAS,CAAC,mBAAmB,CAAC,UAAU,CAAC,GAAG,CAAC,EAAE,iBAAiB,EAAE,gBAAgB,CAAC,OAAO,CAAC,CAAC;IAC9F,CAAC;AACH,CAAC;AAED;;;;;GAKG;AACH,MAAM,UAAU,2BAA2B,CAAC,aAA4B;IACtE,aAAa,CAAC,MAAM,EAAE,CACpB,YAAY,CAAC,2BAA2B,EACxC,YAAY,CAAC,0BAA0B,EACvC,YAAY,CAAC,uBAAuB,CACrC,CAAC;AACJ,CAAC","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\n/**\n * @fileoverview Implements the local event handling contract. This\n * allows DOM objects in a container that enters into this contract to\n * define event handlers which are executed in a local context.\n *\n * One EventContract instance can manage the contract for multiple\n * containers, which are added using the addContainer() method.\n *\n * Events can be registered using the addEvent() method.\n *\n * A Dispatcher is added using the registerDispatcher() method. Until there is\n * a dispatcher, events are queued. The idea is that the EventContract\n * class is inlined in the HTML of the top level page and instantiated\n * right after the start of <body>. The Dispatcher class is contained\n * in the external deferred js, and instantiated and registered with\n * EventContract when the external javascript in the page loads. The\n * external javascript will also register the jsaction handlers, which\n * then pick up the queued events at the time of registration.\n *\n * Since this class is meant to be inlined in the main page HTML, the\n * size of the binary compiled from this file MUST be kept as small as\n * possible and thus its dependencies to a minimum.\n */\n\nimport * as a11yClickLib from './a11y_click';\nimport {ActionResolver} from './action_resolver';\nimport {EarlyJsactionData, EarlyJsactionDataContainer} from './earlyeventcontract';\nimport * as eventLib from './event';\nimport {EventContractContainerManager} from './event_contract_container';\nimport {A11Y_CLICK_SUPPORT, MOUSE_SPECIAL_SUPPORT} from './event_contract_defines';\nimport * as eventInfoLib from './event_info';\nimport {EventType} from './event_type';\nimport {Restriction} from './restriction';\n\n/**\n * The API of an EventContract that is safe to call from any compilation unit.\n */\nexport declare interface UnrenamedEventContract {\n  // Alias for Jsction EventContract registerDispatcher.\n  ecrd(dispatcher: Dispatcher, restriction: Restriction): void;\n  // Unrenamed function. Abbreviation for `eventContract.addA11yClickSupport`.\n  ecaacs?: (\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) => void;\n}\n\n/** A function that is called to handle events captured by the EventContract. */\nexport type Dispatcher = (eventInfo: eventInfoLib.EventInfo, globalDispatch?: boolean) => void;\n\n/**\n * A function that handles an event dispatched from the browser.\n *\n * eventType: May differ from `event.type` if JSAction uses a\n * short-hand name or is patching over an non-bubbling event with a bubbling\n * variant.\n * event: The native browser event.\n * container: The container for this dispatch.\n */\ntype EventHandler = (eventType: string, event: Event, container: Element) => void;\n\n/**\n * EventContract intercepts events in the bubbling phase at the\n * boundary of a container element, and maps them to generic actions\n * which are specified using the custom jsaction attribute in\n * HTML. Behavior of the application is then specified in terms of\n * handler for such actions, cf. jsaction.Dispatcher in dispatcher.js.\n *\n * This has several benefits: (1) No DOM event handlers need to be\n * registered on the specific elements in the UI. (2) The set of\n * events that the application has to handle can be specified in terms\n * of the semantics of the application, rather than in terms of DOM\n * events. (3) Invocation of handlers can be delayed and handlers can\n * be delay loaded in a generic way.\n */\nexport class EventContract implements UnrenamedEventContract {\n  static A11Y_CLICK_SUPPORT = A11Y_CLICK_SUPPORT;\n  static MOUSE_SPECIAL_SUPPORT = MOUSE_SPECIAL_SUPPORT;\n\n  private containerManager: EventContractContainerManager | null;\n\n  private readonly actionResolver?: ActionResolver;\n\n  /**\n   * The DOM events which this contract covers. Used to prevent double\n   * registration of event types. The value of the map is the\n   * internally created DOM event handler function that handles the\n   * DOM events. See addEvent().\n   *\n   */\n  private eventHandlers: {[key: string]: EventHandler} = {};\n\n  private browserEventTypeToExtraEventTypes: {[key: string]: string[]} = {};\n\n  /**\n   * The dispatcher function. Events are passed to this function for\n   * handling once it was set using the registerDispatcher() method. This is\n   * done because the function is passed from another jsbinary, so passing the\n   * instance and invoking the method here would require to leave the method\n   * unobfuscated.\n   */\n  private dispatcher: Dispatcher | null = null;\n\n  /**\n   * The list of suspended `EventInfo` that will be dispatched\n   * as soon as the `Dispatcher` is registered.\n   */\n  private queuedEventInfos: eventInfoLib.EventInfo[] | null = [];\n\n  /** Whether to add an a11y click listener. */\n  private addA11yClickListener = false;\n\n  ecaacs?: (\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) => void;\n\n  constructor(\n    containerManager: EventContractContainerManager,\n    private readonly useActionResolver = true,\n  ) {\n    this.containerManager = containerManager;\n    if (this.useActionResolver) {\n      this.actionResolver = new ActionResolver({\n        syntheticMouseEventSupport: EventContract.MOUSE_SPECIAL_SUPPORT,\n      });\n    }\n    if (EventContract.A11Y_CLICK_SUPPORT) {\n      // Add a11y click support to the `EventContract`.\n      this.addA11yClickSupport();\n    }\n  }\n\n  private handleEvent(eventType: string, event: Event, container: Element) {\n    const eventInfo = eventInfoLib.createEventInfoFromParameters(\n      /* eventType= */ eventType,\n      /* event= */ event,\n      /* targetElement= */ event.target as Element,\n      /* container= */ container,\n      /* timestamp= */ Date.now(),\n    );\n    this.handleEventInfo(eventInfo);\n  }\n\n  /**\n   * Handle an `EventInfo`.\n   */\n  private handleEventInfo(eventInfo: eventInfoLib.EventInfo) {\n    if (!this.dispatcher) {\n      // All events are queued when the dispatcher isn't yet loaded.\n      eventInfoLib.setIsReplay(eventInfo, true);\n      this.queuedEventInfos?.push(eventInfo);\n      return;\n    }\n    if (this.useActionResolver) {\n      this.actionResolver!.resolve(eventInfo);\n    }\n    this.dispatcher(eventInfo);\n  }\n\n  /**\n   * Enables jsaction handlers to be called for the event type given by\n   * name.\n   *\n   * If the event is already registered, this does nothing.\n   *\n   * @param prefixedEventType If supplied, this event is used in\n   *     the actual browser event registration instead of the name that is\n   *     exposed to jsaction. Use this if you e.g. want users to be able\n   *     to subscribe to jsaction=\"transitionEnd:foo\" while the underlying\n   *     event is webkitTransitionEnd in one browser and mozTransitionEnd\n   *     in another.\n   */\n  addEvent(eventType: string, prefixedEventType?: string) {\n    if (eventType in this.eventHandlers || !this.containerManager) {\n      return;\n    }\n\n    if (\n      !EventContract.MOUSE_SPECIAL_SUPPORT &&\n      (eventType === EventType.MOUSEENTER ||\n        eventType === EventType.MOUSELEAVE ||\n        eventType === EventType.POINTERENTER ||\n        eventType === EventType.POINTERLEAVE)\n    ) {\n      return;\n    }\n\n    const eventHandler = (eventType: string, event: Event, container: Element) => {\n      this.handleEvent(eventType, event, container);\n    };\n\n    // Store the callback to allow us to replay events.\n    this.eventHandlers[eventType] = eventHandler;\n\n    const browserEventType = eventLib.getBrowserEventType(prefixedEventType || eventType);\n\n    if (browserEventType !== eventType) {\n      const eventTypes = this.browserEventTypeToExtraEventTypes[browserEventType] || [];\n      eventTypes.push(eventType);\n      this.browserEventTypeToExtraEventTypes[browserEventType] = eventTypes;\n    }\n\n    this.containerManager.addEventListener(browserEventType, (element: Element) => {\n      return (event: Event) => {\n        eventHandler(eventType, event, element);\n      };\n    });\n\n    // Automatically install a keypress/keydown event handler if support for\n    // accessible clicks is turned on.\n    if (this.addA11yClickListener && eventType === EventType.CLICK) {\n      this.addEvent(EventType.KEYDOWN);\n    }\n  }\n\n  /**\n   * Gets the queued early events and replay them using the appropriate handler\n   * in the provided event contract. Once all the events are replayed, it cleans\n   * up the early contract.\n   */\n  replayEarlyEvents(\n    earlyJsactionContainer: EarlyJsactionDataContainer = window as EarlyJsactionDataContainer,\n  ) {\n    // Check if the early contract is present and prevent calling this function\n    // more than once.\n    const earlyJsactionData: EarlyJsactionData | undefined = earlyJsactionContainer._ejsa;\n    if (!earlyJsactionData) {\n      return;\n    }\n\n    // Replay the early contract events.\n    const earlyEventInfos: eventInfoLib.EventInfo[] = earlyJsactionData.q;\n    for (let idx = 0; idx < earlyEventInfos.length; idx++) {\n      const earlyEventInfo: eventInfoLib.EventInfo = earlyEventInfos[idx];\n      const eventTypes = this.getEventTypesForBrowserEventType(earlyEventInfo.eventType);\n      for (let i = 0; i < eventTypes.length; i++) {\n        const eventInfo = eventInfoLib.cloneEventInfo(earlyEventInfo);\n        // EventInfo eventType maps to JSAction's internal event type,\n        // rather than the browser event type.\n        eventInfoLib.setEventType(eventInfo, eventTypes[i]);\n        this.handleEventInfo(eventInfo);\n      }\n    }\n\n    // Clean up the early contract.\n    const earlyEventHandler: (event: Event) => void = earlyJsactionData.h;\n    removeEventListeners(earlyJsactionData.c, earlyJsactionData.et, earlyEventHandler);\n    removeEventListeners(earlyJsactionData.c, earlyJsactionData.etc, earlyEventHandler, true);\n    delete earlyJsactionContainer._ejsa;\n  }\n\n  /**\n   * Returns all JSAction event types that have been registered for a given\n   * browser event type.\n   */\n  private getEventTypesForBrowserEventType(browserEventType: string) {\n    const eventTypes = [];\n    if (this.eventHandlers[browserEventType]) {\n      eventTypes.push(browserEventType);\n    }\n    if (this.browserEventTypeToExtraEventTypes[browserEventType]) {\n      eventTypes.push(...this.browserEventTypeToExtraEventTypes[browserEventType]);\n    }\n    return eventTypes;\n  }\n\n  /**\n   * Returns the event handler function for a given event type.\n   */\n  handler(eventType: string): EventHandler | undefined {\n    return this.eventHandlers[eventType];\n  }\n\n  /**\n   * Cleans up the event contract. This resets all of the `EventContract`'s\n   * internal state. Users are responsible for not using this `EventContract`\n   * after it has been cleaned up.\n   */\n  cleanUp() {\n    this.containerManager!.cleanUp();\n    this.containerManager = null;\n    this.eventHandlers = {};\n    this.browserEventTypeToExtraEventTypes = {};\n    this.dispatcher = null;\n    this.queuedEventInfos = [];\n  }\n\n  /**\n   * Register a dispatcher function. Event info of each event mapped to\n   * a jsaction is passed for handling to this callback. The queued\n   * events are passed as well to the dispatcher for later replaying\n   * once the dispatcher is registered. Clears the event queue to null.\n   *\n   * @param dispatcher The dispatcher function.\n   * @param restriction\n   */\n  registerDispatcher(dispatcher: Dispatcher, restriction: Restriction) {\n    this.ecrd(dispatcher, restriction);\n  }\n\n  /**\n   * Unrenamed alias for registerDispatcher. Necessary for any codebases that\n   * split the `EventContract` and `Dispatcher` code into different compilation\n   * units.\n   */\n  ecrd(dispatcher: Dispatcher, restriction: Restriction) {\n    this.dispatcher = dispatcher;\n\n    if (this.queuedEventInfos?.length) {\n      for (let i = 0; i < this.queuedEventInfos.length; i++) {\n        this.handleEventInfo(this.queuedEventInfos[i]);\n      }\n      this.queuedEventInfos = null;\n    }\n  }\n\n  /**\n   * Adds a11y click support to the given `EventContract`. Meant to be called in\n   * the same compilation unit as the `EventContract`.\n   */\n  addA11yClickSupport() {\n    this.addA11yClickSupportImpl(\n      a11yClickLib.updateEventInfoForA11yClick,\n      a11yClickLib.preventDefaultForA11yClick,\n      a11yClickLib.populateClickOnlyAction,\n    );\n  }\n\n  /**\n   * Enables a11y click support to be deferred. Meant to be called in the same\n   * compilation unit as the `EventContract`.\n   */\n  exportAddA11yClickSupport() {\n    this.addA11yClickListener = true;\n    this.ecaacs = this.addA11yClickSupportImpl.bind(this);\n  }\n\n  /**\n   * Unrenamed function that loads a11yClickSupport.\n   */\n  private addA11yClickSupportImpl(\n    updateEventInfoForA11yClick: typeof a11yClickLib.updateEventInfoForA11yClick,\n    preventDefaultForA11yClick: typeof a11yClickLib.preventDefaultForA11yClick,\n    populateClickOnlyAction: typeof a11yClickLib.populateClickOnlyAction,\n  ) {\n    this.addA11yClickListener = true;\n    if (this.useActionResolver) {\n      this.actionResolver!.addA11yClickSupport(\n        updateEventInfoForA11yClick,\n        preventDefaultForA11yClick,\n        populateClickOnlyAction,\n      );\n    }\n  }\n}\n\nfunction removeEventListeners(\n  container: HTMLElement,\n  eventTypes: string[],\n  earlyEventHandler: (e: Event) => void,\n  capture?: boolean,\n) {\n  for (let idx = 0; idx < eventTypes.length; idx++) {\n    container.removeEventListener(eventTypes[idx], earlyEventHandler, /* useCapture */ capture);\n  }\n}\n\n/**\n * Adds a11y click support to the given `EventContract`. Meant to be called\n * in a different compilation unit from the `EventContract`. The `EventContract`\n * must have called `exportAddA11yClickSupport` in its compilation unit for this\n * to have any effect.\n */\nexport function addDeferredA11yClickSupport(eventContract: EventContract) {\n  eventContract.ecaacs?.(\n    a11yClickLib.updateEventInfoForA11yClick,\n    a11yClickLib.preventDefaultForA11yClick,\n    a11yClickLib.populateClickOnlyAction,\n  );\n}\n"]}
|