source: trip-planner-front/node_modules/@angular/cdk/esm2015/drag-drop/drag-ref.js

Last change on this file was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 180.6 KB
Line 
1/**
2 * @license
3 * Copyright Google LLC All Rights Reserved.
4 *
5 * Use of this source code is governed by an MIT-style license that can be
6 * found in the LICENSE file at https://angular.io/license
7 */
8import { normalizePassiveListenerOptions, _getEventTarget, _getShadowRoot, } from '@angular/cdk/platform';
9import { coerceBooleanProperty, coerceElement } from '@angular/cdk/coercion';
10import { isFakeMousedownFromScreenReader, isFakeTouchstartFromScreenReader, } from '@angular/cdk/a11y';
11import { Subscription, Subject } from 'rxjs';
12import { combineTransforms, extendStyles, toggleNativeDragInteractions, toggleVisibility, } from './drag-styling';
13import { getTransformTransitionDurationInMs } from './transition-duration';
14import { getMutableClientRect, adjustClientRect } from './client-rect';
15import { ParentPositionTracker } from './parent-position-tracker';
16import { deepCloneNode } from './clone-node';
17/** Options that can be used to bind a passive event listener. */
18const passiveEventListenerOptions = normalizePassiveListenerOptions({ passive: true });
19/** Options that can be used to bind an active event listener. */
20const activeEventListenerOptions = normalizePassiveListenerOptions({ passive: false });
21/**
22 * Time in milliseconds for which to ignore mouse events, after
23 * receiving a touch event. Used to avoid doing double work for
24 * touch devices where the browser fires fake mouse events, in
25 * addition to touch events.
26 */
27const MOUSE_EVENT_IGNORE_TIME = 800;
28/** Inline styles to be set as `!important` while dragging. */
29const dragImportantProperties = new Set([
30 // Needs to be important, because some `mat-table` sets `position: sticky !important`. See #22781.
31 'position'
32]);
33/**
34 * Reference to a draggable item. Used to manipulate or dispose of the item.
35 */
36export class DragRef {
37 constructor(element, _config, _document, _ngZone, _viewportRuler, _dragDropRegistry) {
38 this._config = _config;
39 this._document = _document;
40 this._ngZone = _ngZone;
41 this._viewportRuler = _viewportRuler;
42 this._dragDropRegistry = _dragDropRegistry;
43 /**
44 * CSS `transform` applied to the element when it isn't being dragged. We need a
45 * passive transform in order for the dragged element to retain its new position
46 * after the user has stopped dragging and because we need to know the relative
47 * position in case they start dragging again. This corresponds to `element.style.transform`.
48 */
49 this._passiveTransform = { x: 0, y: 0 };
50 /** CSS `transform` that is applied to the element while it's being dragged. */
51 this._activeTransform = { x: 0, y: 0 };
52 /**
53 * Whether the dragging sequence has been started. Doesn't
54 * necessarily mean that the element has been moved.
55 */
56 this._hasStartedDragging = false;
57 /** Emits when the item is being moved. */
58 this._moveEvents = new Subject();
59 /** Subscription to pointer movement events. */
60 this._pointerMoveSubscription = Subscription.EMPTY;
61 /** Subscription to the event that is dispatched when the user lifts their pointer. */
62 this._pointerUpSubscription = Subscription.EMPTY;
63 /** Subscription to the viewport being scrolled. */
64 this._scrollSubscription = Subscription.EMPTY;
65 /** Subscription to the viewport being resized. */
66 this._resizeSubscription = Subscription.EMPTY;
67 /** Cached reference to the boundary element. */
68 this._boundaryElement = null;
69 /** Whether the native dragging interactions have been enabled on the root element. */
70 this._nativeInteractionsEnabled = true;
71 /** Elements that can be used to drag the draggable item. */
72 this._handles = [];
73 /** Registered handles that are currently disabled. */
74 this._disabledHandles = new Set();
75 /** Layout direction of the item. */
76 this._direction = 'ltr';
77 /**
78 * Amount of milliseconds to wait after the user has put their
79 * pointer down before starting to drag the element.
80 */
81 this.dragStartDelay = 0;
82 this._disabled = false;
83 /** Emits as the drag sequence is being prepared. */
84 this.beforeStarted = new Subject();
85 /** Emits when the user starts dragging the item. */
86 this.started = new Subject();
87 /** Emits when the user has released a drag item, before any animations have started. */
88 this.released = new Subject();
89 /** Emits when the user stops dragging an item in the container. */
90 this.ended = new Subject();
91 /** Emits when the user has moved the item into a new container. */
92 this.entered = new Subject();
93 /** Emits when the user removes the item its container by dragging it into another container. */
94 this.exited = new Subject();
95 /** Emits when the user drops the item inside a container. */
96 this.dropped = new Subject();
97 /**
98 * Emits as the user is dragging the item. Use with caution,
99 * because this event will fire for every pixel that the user has dragged.
100 */
101 this.moved = this._moveEvents;
102 /** Handler for the `mousedown`/`touchstart` events. */
103 this._pointerDown = (event) => {
104 this.beforeStarted.next();
105 // Delegate the event based on whether it started from a handle or the element itself.
106 if (this._handles.length) {
107 const targetHandle = this._handles.find(handle => {
108 const target = _getEventTarget(event);
109 return !!target && (target === handle || handle.contains(target));
110 });
111 if (targetHandle && !this._disabledHandles.has(targetHandle) && !this.disabled) {
112 this._initializeDragSequence(targetHandle, event);
113 }
114 }
115 else if (!this.disabled) {
116 this._initializeDragSequence(this._rootElement, event);
117 }
118 };
119 /** Handler that is invoked when the user moves their pointer after they've initiated a drag. */
120 this._pointerMove = (event) => {
121 const pointerPosition = this._getPointerPositionOnPage(event);
122 if (!this._hasStartedDragging) {
123 const distanceX = Math.abs(pointerPosition.x - this._pickupPositionOnPage.x);
124 const distanceY = Math.abs(pointerPosition.y - this._pickupPositionOnPage.y);
125 const isOverThreshold = distanceX + distanceY >= this._config.dragStartThreshold;
126 // Only start dragging after the user has moved more than the minimum distance in either
127 // direction. Note that this is preferrable over doing something like `skip(minimumDistance)`
128 // in the `pointerMove` subscription, because we're not guaranteed to have one move event
129 // per pixel of movement (e.g. if the user moves their pointer quickly).
130 if (isOverThreshold) {
131 const isDelayElapsed = Date.now() >= this._dragStartTime + this._getDragStartDelay(event);
132 const container = this._dropContainer;
133 if (!isDelayElapsed) {
134 this._endDragSequence(event);
135 return;
136 }
137 // Prevent other drag sequences from starting while something in the container is still
138 // being dragged. This can happen while we're waiting for the drop animation to finish
139 // and can cause errors, because some elements might still be moving around.
140 if (!container || (!container.isDragging() && !container.isReceiving())) {
141 // Prevent the default action as soon as the dragging sequence is considered as
142 // "started" since waiting for the next event can allow the device to begin scrolling.
143 event.preventDefault();
144 this._hasStartedDragging = true;
145 this._ngZone.run(() => this._startDragSequence(event));
146 }
147 }
148 return;
149 }
150 // We only need the preview dimensions if we have a boundary element.
151 if (this._boundaryElement) {
152 // Cache the preview element rect if we haven't cached it already or if
153 // we cached it too early before the element dimensions were computed.
154 if (!this._previewRect || (!this._previewRect.width && !this._previewRect.height)) {
155 this._previewRect = (this._preview || this._rootElement).getBoundingClientRect();
156 }
157 }
158 // We prevent the default action down here so that we know that dragging has started. This is
159 // important for touch devices where doing this too early can unnecessarily block scrolling,
160 // if there's a dragging delay.
161 event.preventDefault();
162 const constrainedPointerPosition = this._getConstrainedPointerPosition(pointerPosition);
163 this._hasMoved = true;
164 this._lastKnownPointerPosition = pointerPosition;
165 this._updatePointerDirectionDelta(constrainedPointerPosition);
166 if (this._dropContainer) {
167 this._updateActiveDropContainer(constrainedPointerPosition, pointerPosition);
168 }
169 else {
170 const activeTransform = this._activeTransform;
171 activeTransform.x =
172 constrainedPointerPosition.x - this._pickupPositionOnPage.x + this._passiveTransform.x;
173 activeTransform.y =
174 constrainedPointerPosition.y - this._pickupPositionOnPage.y + this._passiveTransform.y;
175 this._applyRootElementTransform(activeTransform.x, activeTransform.y);
176 // Apply transform as attribute if dragging and svg element to work for IE
177 if (typeof SVGElement !== 'undefined' && this._rootElement instanceof SVGElement) {
178 const appliedTransform = `translate(${activeTransform.x} ${activeTransform.y})`;
179 this._rootElement.setAttribute('transform', appliedTransform);
180 }
181 }
182 // Since this event gets fired for every pixel while dragging, we only
183 // want to fire it if the consumer opted into it. Also we have to
184 // re-enter the zone because we run all of the events on the outside.
185 if (this._moveEvents.observers.length) {
186 this._ngZone.run(() => {
187 this._moveEvents.next({
188 source: this,
189 pointerPosition: constrainedPointerPosition,
190 event,
191 distance: this._getDragDistance(constrainedPointerPosition),
192 delta: this._pointerDirectionDelta
193 });
194 });
195 }
196 };
197 /** Handler that is invoked when the user lifts their pointer up, after initiating a drag. */
198 this._pointerUp = (event) => {
199 this._endDragSequence(event);
200 };
201 this.withRootElement(element).withParent(_config.parentDragRef || null);
202 this._parentPositions = new ParentPositionTracker(_document, _viewportRuler);
203 _dragDropRegistry.registerDragItem(this);
204 }
205 /** Whether starting to drag this element is disabled. */
206 get disabled() {
207 return this._disabled || !!(this._dropContainer && this._dropContainer.disabled);
208 }
209 set disabled(value) {
210 const newValue = coerceBooleanProperty(value);
211 if (newValue !== this._disabled) {
212 this._disabled = newValue;
213 this._toggleNativeDragInteractions();
214 this._handles.forEach(handle => toggleNativeDragInteractions(handle, newValue));
215 }
216 }
217 /**
218 * Returns the element that is being used as a placeholder
219 * while the current element is being dragged.
220 */
221 getPlaceholderElement() {
222 return this._placeholder;
223 }
224 /** Returns the root draggable element. */
225 getRootElement() {
226 return this._rootElement;
227 }
228 /**
229 * Gets the currently-visible element that represents the drag item.
230 * While dragging this is the placeholder, otherwise it's the root element.
231 */
232 getVisibleElement() {
233 return this.isDragging() ? this.getPlaceholderElement() : this.getRootElement();
234 }
235 /** Registers the handles that can be used to drag the element. */
236 withHandles(handles) {
237 this._handles = handles.map(handle => coerceElement(handle));
238 this._handles.forEach(handle => toggleNativeDragInteractions(handle, this.disabled));
239 this._toggleNativeDragInteractions();
240 // Delete any lingering disabled handles that may have been destroyed. Note that we re-create
241 // the set, rather than iterate over it and filter out the destroyed handles, because while
242 // the ES spec allows for sets to be modified while they're being iterated over, some polyfills
243 // use an array internally which may throw an error.
244 const disabledHandles = new Set();
245 this._disabledHandles.forEach(handle => {
246 if (this._handles.indexOf(handle) > -1) {
247 disabledHandles.add(handle);
248 }
249 });
250 this._disabledHandles = disabledHandles;
251 return this;
252 }
253 /**
254 * Registers the template that should be used for the drag preview.
255 * @param template Template that from which to stamp out the preview.
256 */
257 withPreviewTemplate(template) {
258 this._previewTemplate = template;
259 return this;
260 }
261 /**
262 * Registers the template that should be used for the drag placeholder.
263 * @param template Template that from which to stamp out the placeholder.
264 */
265 withPlaceholderTemplate(template) {
266 this._placeholderTemplate = template;
267 return this;
268 }
269 /**
270 * Sets an alternate drag root element. The root element is the element that will be moved as
271 * the user is dragging. Passing an alternate root element is useful when trying to enable
272 * dragging on an element that you might not have access to.
273 */
274 withRootElement(rootElement) {
275 const element = coerceElement(rootElement);
276 if (element !== this._rootElement) {
277 if (this._rootElement) {
278 this._removeRootElementListeners(this._rootElement);
279 }
280 this._ngZone.runOutsideAngular(() => {
281 element.addEventListener('mousedown', this._pointerDown, activeEventListenerOptions);
282 element.addEventListener('touchstart', this._pointerDown, passiveEventListenerOptions);
283 });
284 this._initialTransform = undefined;
285 this._rootElement = element;
286 }
287 if (typeof SVGElement !== 'undefined' && this._rootElement instanceof SVGElement) {
288 this._ownerSVGElement = this._rootElement.ownerSVGElement;
289 }
290 return this;
291 }
292 /**
293 * Element to which the draggable's position will be constrained.
294 */
295 withBoundaryElement(boundaryElement) {
296 this._boundaryElement = boundaryElement ? coerceElement(boundaryElement) : null;
297 this._resizeSubscription.unsubscribe();
298 if (boundaryElement) {
299 this._resizeSubscription = this._viewportRuler
300 .change(10)
301 .subscribe(() => this._containInsideBoundaryOnResize());
302 }
303 return this;
304 }
305 /** Sets the parent ref that the ref is nested in. */
306 withParent(parent) {
307 this._parentDragRef = parent;
308 return this;
309 }
310 /** Removes the dragging functionality from the DOM element. */
311 dispose() {
312 this._removeRootElementListeners(this._rootElement);
313 // Do this check before removing from the registry since it'll
314 // stop being considered as dragged once it is removed.
315 if (this.isDragging()) {
316 // Since we move out the element to the end of the body while it's being
317 // dragged, we have to make sure that it's removed if it gets destroyed.
318 removeNode(this._rootElement);
319 }
320 removeNode(this._anchor);
321 this._destroyPreview();
322 this._destroyPlaceholder();
323 this._dragDropRegistry.removeDragItem(this);
324 this._removeSubscriptions();
325 this.beforeStarted.complete();
326 this.started.complete();
327 this.released.complete();
328 this.ended.complete();
329 this.entered.complete();
330 this.exited.complete();
331 this.dropped.complete();
332 this._moveEvents.complete();
333 this._handles = [];
334 this._disabledHandles.clear();
335 this._dropContainer = undefined;
336 this._resizeSubscription.unsubscribe();
337 this._parentPositions.clear();
338 this._boundaryElement = this._rootElement = this._ownerSVGElement = this._placeholderTemplate =
339 this._previewTemplate = this._anchor = this._parentDragRef = null;
340 }
341 /** Checks whether the element is currently being dragged. */
342 isDragging() {
343 return this._hasStartedDragging && this._dragDropRegistry.isDragging(this);
344 }
345 /** Resets a standalone drag item to its initial position. */
346 reset() {
347 this._rootElement.style.transform = this._initialTransform || '';
348 this._activeTransform = { x: 0, y: 0 };
349 this._passiveTransform = { x: 0, y: 0 };
350 }
351 /**
352 * Sets a handle as disabled. While a handle is disabled, it'll capture and interrupt dragging.
353 * @param handle Handle element that should be disabled.
354 */
355 disableHandle(handle) {
356 if (!this._disabledHandles.has(handle) && this._handles.indexOf(handle) > -1) {
357 this._disabledHandles.add(handle);
358 toggleNativeDragInteractions(handle, true);
359 }
360 }
361 /**
362 * Enables a handle, if it has been disabled.
363 * @param handle Handle element to be enabled.
364 */
365 enableHandle(handle) {
366 if (this._disabledHandles.has(handle)) {
367 this._disabledHandles.delete(handle);
368 toggleNativeDragInteractions(handle, this.disabled);
369 }
370 }
371 /** Sets the layout direction of the draggable item. */
372 withDirection(direction) {
373 this._direction = direction;
374 return this;
375 }
376 /** Sets the container that the item is part of. */
377 _withDropContainer(container) {
378 this._dropContainer = container;
379 }
380 /**
381 * Gets the current position in pixels the draggable outside of a drop container.
382 */
383 getFreeDragPosition() {
384 const position = this.isDragging() ? this._activeTransform : this._passiveTransform;
385 return { x: position.x, y: position.y };
386 }
387 /**
388 * Sets the current position in pixels the draggable outside of a drop container.
389 * @param value New position to be set.
390 */
391 setFreeDragPosition(value) {
392 this._activeTransform = { x: 0, y: 0 };
393 this._passiveTransform.x = value.x;
394 this._passiveTransform.y = value.y;
395 if (!this._dropContainer) {
396 this._applyRootElementTransform(value.x, value.y);
397 }
398 return this;
399 }
400 /**
401 * Sets the container into which to insert the preview element.
402 * @param value Container into which to insert the preview.
403 */
404 withPreviewContainer(value) {
405 this._previewContainer = value;
406 return this;
407 }
408 /** Updates the item's sort order based on the last-known pointer position. */
409 _sortFromLastPointerPosition() {
410 const position = this._lastKnownPointerPosition;
411 if (position && this._dropContainer) {
412 this._updateActiveDropContainer(this._getConstrainedPointerPosition(position), position);
413 }
414 }
415 /** Unsubscribes from the global subscriptions. */
416 _removeSubscriptions() {
417 this._pointerMoveSubscription.unsubscribe();
418 this._pointerUpSubscription.unsubscribe();
419 this._scrollSubscription.unsubscribe();
420 }
421 /** Destroys the preview element and its ViewRef. */
422 _destroyPreview() {
423 if (this._preview) {
424 removeNode(this._preview);
425 }
426 if (this._previewRef) {
427 this._previewRef.destroy();
428 }
429 this._preview = this._previewRef = null;
430 }
431 /** Destroys the placeholder element and its ViewRef. */
432 _destroyPlaceholder() {
433 if (this._placeholder) {
434 removeNode(this._placeholder);
435 }
436 if (this._placeholderRef) {
437 this._placeholderRef.destroy();
438 }
439 this._placeholder = this._placeholderRef = null;
440 }
441 /**
442 * Clears subscriptions and stops the dragging sequence.
443 * @param event Browser event object that ended the sequence.
444 */
445 _endDragSequence(event) {
446 // Note that here we use `isDragging` from the service, rather than from `this`.
447 // The difference is that the one from the service reflects whether a dragging sequence
448 // has been initiated, whereas the one on `this` includes whether the user has passed
449 // the minimum dragging threshold.
450 if (!this._dragDropRegistry.isDragging(this)) {
451 return;
452 }
453 this._removeSubscriptions();
454 this._dragDropRegistry.stopDragging(this);
455 this._toggleNativeDragInteractions();
456 if (this._handles) {
457 this._rootElement.style.webkitTapHighlightColor = this._rootElementTapHighlight;
458 }
459 if (!this._hasStartedDragging) {
460 return;
461 }
462 this.released.next({ source: this });
463 if (this._dropContainer) {
464 // Stop scrolling immediately, instead of waiting for the animation to finish.
465 this._dropContainer._stopScrolling();
466 this._animatePreviewToPlaceholder().then(() => {
467 this._cleanupDragArtifacts(event);
468 this._cleanupCachedDimensions();
469 this._dragDropRegistry.stopDragging(this);
470 });
471 }
472 else {
473 // Convert the active transform into a passive one. This means that next time
474 // the user starts dragging the item, its position will be calculated relatively
475 // to the new passive transform.
476 this._passiveTransform.x = this._activeTransform.x;
477 const pointerPosition = this._getPointerPositionOnPage(event);
478 this._passiveTransform.y = this._activeTransform.y;
479 this._ngZone.run(() => {
480 this.ended.next({
481 source: this,
482 distance: this._getDragDistance(pointerPosition),
483 dropPoint: pointerPosition
484 });
485 });
486 this._cleanupCachedDimensions();
487 this._dragDropRegistry.stopDragging(this);
488 }
489 }
490 /** Starts the dragging sequence. */
491 _startDragSequence(event) {
492 if (isTouchEvent(event)) {
493 this._lastTouchEventTime = Date.now();
494 }
495 this._toggleNativeDragInteractions();
496 const dropContainer = this._dropContainer;
497 if (dropContainer) {
498 const element = this._rootElement;
499 const parent = element.parentNode;
500 const placeholder = this._placeholder = this._createPlaceholderElement();
501 const anchor = this._anchor = this._anchor || this._document.createComment('');
502 // Needs to happen before the root element is moved.
503 const shadowRoot = this._getShadowRoot();
504 // Insert an anchor node so that we can restore the element's position in the DOM.
505 parent.insertBefore(anchor, element);
506 // There's no risk of transforms stacking when inside a drop container so
507 // we can keep the initial transform up to date any time dragging starts.
508 this._initialTransform = element.style.transform || '';
509 // Create the preview after the initial transform has
510 // been cached, because it can be affected by the transform.
511 this._preview = this._createPreviewElement();
512 // We move the element out at the end of the body and we make it hidden, because keeping it in
513 // place will throw off the consumer's `:last-child` selectors. We can't remove the element
514 // from the DOM completely, because iOS will stop firing all subsequent events in the chain.
515 toggleVisibility(element, false, dragImportantProperties);
516 this._document.body.appendChild(parent.replaceChild(placeholder, element));
517 this._getPreviewInsertionPoint(parent, shadowRoot).appendChild(this._preview);
518 this.started.next({ source: this }); // Emit before notifying the container.
519 dropContainer.start();
520 this._initialContainer = dropContainer;
521 this._initialIndex = dropContainer.getItemIndex(this);
522 }
523 else {
524 this.started.next({ source: this });
525 this._initialContainer = this._initialIndex = undefined;
526 }
527 // Important to run after we've called `start` on the parent container
528 // so that it has had time to resolve its scrollable parents.
529 this._parentPositions.cache(dropContainer ? dropContainer.getScrollableParents() : []);
530 }
531 /**
532 * Sets up the different variables and subscriptions
533 * that will be necessary for the dragging sequence.
534 * @param referenceElement Element that started the drag sequence.
535 * @param event Browser event object that started the sequence.
536 */
537 _initializeDragSequence(referenceElement, event) {
538 // Stop propagation if the item is inside another
539 // draggable so we don't start multiple drag sequences.
540 if (this._parentDragRef) {
541 event.stopPropagation();
542 }
543 const isDragging = this.isDragging();
544 const isTouchSequence = isTouchEvent(event);
545 const isAuxiliaryMouseButton = !isTouchSequence && event.button !== 0;
546 const rootElement = this._rootElement;
547 const target = _getEventTarget(event);
548 const isSyntheticEvent = !isTouchSequence && this._lastTouchEventTime &&
549 this._lastTouchEventTime + MOUSE_EVENT_IGNORE_TIME > Date.now();
550 const isFakeEvent = isTouchSequence ? isFakeTouchstartFromScreenReader(event) :
551 isFakeMousedownFromScreenReader(event);
552 // If the event started from an element with the native HTML drag&drop, it'll interfere
553 // with our own dragging (e.g. `img` tags do it by default). Prevent the default action
554 // to stop it from happening. Note that preventing on `dragstart` also seems to work, but
555 // it's flaky and it fails if the user drags it away quickly. Also note that we only want
556 // to do this for `mousedown` since doing the same for `touchstart` will stop any `click`
557 // events from firing on touch devices.
558 if (target && target.draggable && event.type === 'mousedown') {
559 event.preventDefault();
560 }
561 // Abort if the user is already dragging or is using a mouse button other than the primary one.
562 if (isDragging || isAuxiliaryMouseButton || isSyntheticEvent || isFakeEvent) {
563 return;
564 }
565 // If we've got handles, we need to disable the tap highlight on the entire root element,
566 // otherwise iOS will still add it, even though all the drag interactions on the handle
567 // are disabled.
568 if (this._handles.length) {
569 this._rootElementTapHighlight = rootElement.style.webkitTapHighlightColor || '';
570 rootElement.style.webkitTapHighlightColor = 'transparent';
571 }
572 this._hasStartedDragging = this._hasMoved = false;
573 // Avoid multiple subscriptions and memory leaks when multi touch
574 // (isDragging check above isn't enough because of possible temporal and/or dimensional delays)
575 this._removeSubscriptions();
576 this._pointerMoveSubscription = this._dragDropRegistry.pointerMove.subscribe(this._pointerMove);
577 this._pointerUpSubscription = this._dragDropRegistry.pointerUp.subscribe(this._pointerUp);
578 this._scrollSubscription = this._dragDropRegistry
579 .scrolled(this._getShadowRoot())
580 .subscribe(scrollEvent => this._updateOnScroll(scrollEvent));
581 if (this._boundaryElement) {
582 this._boundaryRect = getMutableClientRect(this._boundaryElement);
583 }
584 // If we have a custom preview we can't know ahead of time how large it'll be so we position
585 // it next to the cursor. The exception is when the consumer has opted into making the preview
586 // the same size as the root element, in which case we do know the size.
587 const previewTemplate = this._previewTemplate;
588 this._pickupPositionInElement = previewTemplate && previewTemplate.template &&
589 !previewTemplate.matchSize ? { x: 0, y: 0 } :
590 this._getPointerPositionInElement(referenceElement, event);
591 const pointerPosition = this._pickupPositionOnPage = this._lastKnownPointerPosition =
592 this._getPointerPositionOnPage(event);
593 this._pointerDirectionDelta = { x: 0, y: 0 };
594 this._pointerPositionAtLastDirectionChange = { x: pointerPosition.x, y: pointerPosition.y };
595 this._dragStartTime = Date.now();
596 this._dragDropRegistry.startDragging(this, event);
597 }
598 /** Cleans up the DOM artifacts that were added to facilitate the element being dragged. */
599 _cleanupDragArtifacts(event) {
600 // Restore the element's visibility and insert it at its old position in the DOM.
601 // It's important that we maintain the position, because moving the element around in the DOM
602 // can throw off `NgFor` which does smart diffing and re-creates elements only when necessary,
603 // while moving the existing elements in all other cases.
604 toggleVisibility(this._rootElement, true, dragImportantProperties);
605 this._anchor.parentNode.replaceChild(this._rootElement, this._anchor);
606 this._destroyPreview();
607 this._destroyPlaceholder();
608 this._boundaryRect = this._previewRect = this._initialTransform = undefined;
609 // Re-enter the NgZone since we bound `document` events on the outside.
610 this._ngZone.run(() => {
611 const container = this._dropContainer;
612 const currentIndex = container.getItemIndex(this);
613 const pointerPosition = this._getPointerPositionOnPage(event);
614 const distance = this._getDragDistance(pointerPosition);
615 const isPointerOverContainer = container._isOverContainer(pointerPosition.x, pointerPosition.y);
616 this.ended.next({ source: this, distance, dropPoint: pointerPosition });
617 this.dropped.next({
618 item: this,
619 currentIndex,
620 previousIndex: this._initialIndex,
621 container: container,
622 previousContainer: this._initialContainer,
623 isPointerOverContainer,
624 distance,
625 dropPoint: pointerPosition
626 });
627 container.drop(this, currentIndex, this._initialIndex, this._initialContainer, isPointerOverContainer, distance, pointerPosition);
628 this._dropContainer = this._initialContainer;
629 });
630 }
631 /**
632 * Updates the item's position in its drop container, or moves it
633 * into a new one, depending on its current drag position.
634 */
635 _updateActiveDropContainer({ x, y }, { x: rawX, y: rawY }) {
636 // Drop container that draggable has been moved into.
637 let newContainer = this._initialContainer._getSiblingContainerFromPosition(this, x, y);
638 // If we couldn't find a new container to move the item into, and the item has left its
639 // initial container, check whether the it's over the initial container. This handles the
640 // case where two containers are connected one way and the user tries to undo dragging an
641 // item into a new container.
642 if (!newContainer && this._dropContainer !== this._initialContainer &&
643 this._initialContainer._isOverContainer(x, y)) {
644 newContainer = this._initialContainer;
645 }
646 if (newContainer && newContainer !== this._dropContainer) {
647 this._ngZone.run(() => {
648 // Notify the old container that the item has left.
649 this.exited.next({ item: this, container: this._dropContainer });
650 this._dropContainer.exit(this);
651 // Notify the new container that the item has entered.
652 this._dropContainer = newContainer;
653 this._dropContainer.enter(this, x, y, newContainer === this._initialContainer &&
654 // If we're re-entering the initial container and sorting is disabled,
655 // put item the into its starting index to begin with.
656 newContainer.sortingDisabled ? this._initialIndex : undefined);
657 this.entered.next({
658 item: this,
659 container: newContainer,
660 currentIndex: newContainer.getItemIndex(this)
661 });
662 });
663 }
664 // Dragging may have been interrupted as a result of the events above.
665 if (this.isDragging()) {
666 this._dropContainer._startScrollingIfNecessary(rawX, rawY);
667 this._dropContainer._sortItem(this, x, y, this._pointerDirectionDelta);
668 this._applyPreviewTransform(x - this._pickupPositionInElement.x, y - this._pickupPositionInElement.y);
669 }
670 }
671 /**
672 * Creates the element that will be rendered next to the user's pointer
673 * and will be used as a preview of the element that is being dragged.
674 */
675 _createPreviewElement() {
676 const previewConfig = this._previewTemplate;
677 const previewClass = this.previewClass;
678 const previewTemplate = previewConfig ? previewConfig.template : null;
679 let preview;
680 if (previewTemplate && previewConfig) {
681 // Measure the element before we've inserted the preview
682 // since the insertion could throw off the measurement.
683 const rootRect = previewConfig.matchSize ? this._rootElement.getBoundingClientRect() : null;
684 const viewRef = previewConfig.viewContainer.createEmbeddedView(previewTemplate, previewConfig.context);
685 viewRef.detectChanges();
686 preview = getRootNode(viewRef, this._document);
687 this._previewRef = viewRef;
688 if (previewConfig.matchSize) {
689 matchElementSize(preview, rootRect);
690 }
691 else {
692 preview.style.transform =
693 getTransform(this._pickupPositionOnPage.x, this._pickupPositionOnPage.y);
694 }
695 }
696 else {
697 const element = this._rootElement;
698 preview = deepCloneNode(element);
699 matchElementSize(preview, element.getBoundingClientRect());
700 if (this._initialTransform) {
701 preview.style.transform = this._initialTransform;
702 }
703 }
704 extendStyles(preview.style, {
705 // It's important that we disable the pointer events on the preview, because
706 // it can throw off the `document.elementFromPoint` calls in the `CdkDropList`.
707 'pointer-events': 'none',
708 // We have to reset the margin, because it can throw off positioning relative to the viewport.
709 'margin': '0',
710 'position': 'fixed',
711 'top': '0',
712 'left': '0',
713 'z-index': `${this._config.zIndex || 1000}`
714 }, dragImportantProperties);
715 toggleNativeDragInteractions(preview, false);
716 preview.classList.add('cdk-drag-preview');
717 preview.setAttribute('dir', this._direction);
718 if (previewClass) {
719 if (Array.isArray(previewClass)) {
720 previewClass.forEach(className => preview.classList.add(className));
721 }
722 else {
723 preview.classList.add(previewClass);
724 }
725 }
726 return preview;
727 }
728 /**
729 * Animates the preview element from its current position to the location of the drop placeholder.
730 * @returns Promise that resolves when the animation completes.
731 */
732 _animatePreviewToPlaceholder() {
733 // If the user hasn't moved yet, the transitionend event won't fire.
734 if (!this._hasMoved) {
735 return Promise.resolve();
736 }
737 const placeholderRect = this._placeholder.getBoundingClientRect();
738 // Apply the class that adds a transition to the preview.
739 this._preview.classList.add('cdk-drag-animating');
740 // Move the preview to the placeholder position.
741 this._applyPreviewTransform(placeholderRect.left, placeholderRect.top);
742 // If the element doesn't have a `transition`, the `transitionend` event won't fire. Since
743 // we need to trigger a style recalculation in order for the `cdk-drag-animating` class to
744 // apply its style, we take advantage of the available info to figure out whether we need to
745 // bind the event in the first place.
746 const duration = getTransformTransitionDurationInMs(this._preview);
747 if (duration === 0) {
748 return Promise.resolve();
749 }
750 return this._ngZone.runOutsideAngular(() => {
751 return new Promise(resolve => {
752 const handler = ((event) => {
753 var _a;
754 if (!event || (_getEventTarget(event) === this._preview &&
755 event.propertyName === 'transform')) {
756 (_a = this._preview) === null || _a === void 0 ? void 0 : _a.removeEventListener('transitionend', handler);
757 resolve();
758 clearTimeout(timeout);
759 }
760 });
761 // If a transition is short enough, the browser might not fire the `transitionend` event.
762 // Since we know how long it's supposed to take, add a timeout with a 50% buffer that'll
763 // fire if the transition hasn't completed when it was supposed to.
764 const timeout = setTimeout(handler, duration * 1.5);
765 this._preview.addEventListener('transitionend', handler);
766 });
767 });
768 }
769 /** Creates an element that will be shown instead of the current element while dragging. */
770 _createPlaceholderElement() {
771 const placeholderConfig = this._placeholderTemplate;
772 const placeholderTemplate = placeholderConfig ? placeholderConfig.template : null;
773 let placeholder;
774 if (placeholderTemplate) {
775 this._placeholderRef = placeholderConfig.viewContainer.createEmbeddedView(placeholderTemplate, placeholderConfig.context);
776 this._placeholderRef.detectChanges();
777 placeholder = getRootNode(this._placeholderRef, this._document);
778 }
779 else {
780 placeholder = deepCloneNode(this._rootElement);
781 }
782 placeholder.classList.add('cdk-drag-placeholder');
783 return placeholder;
784 }
785 /**
786 * Figures out the coordinates at which an element was picked up.
787 * @param referenceElement Element that initiated the dragging.
788 * @param event Event that initiated the dragging.
789 */
790 _getPointerPositionInElement(referenceElement, event) {
791 const elementRect = this._rootElement.getBoundingClientRect();
792 const handleElement = referenceElement === this._rootElement ? null : referenceElement;
793 const referenceRect = handleElement ? handleElement.getBoundingClientRect() : elementRect;
794 const point = isTouchEvent(event) ? event.targetTouches[0] : event;
795 const scrollPosition = this._getViewportScrollPosition();
796 const x = point.pageX - referenceRect.left - scrollPosition.left;
797 const y = point.pageY - referenceRect.top - scrollPosition.top;
798 return {
799 x: referenceRect.left - elementRect.left + x,
800 y: referenceRect.top - elementRect.top + y
801 };
802 }
803 /** Determines the point of the page that was touched by the user. */
804 _getPointerPositionOnPage(event) {
805 const scrollPosition = this._getViewportScrollPosition();
806 const point = isTouchEvent(event) ?
807 // `touches` will be empty for start/end events so we have to fall back to `changedTouches`.
808 // Also note that on real devices we're guaranteed for either `touches` or `changedTouches`
809 // to have a value, but Firefox in device emulation mode has a bug where both can be empty
810 // for `touchstart` and `touchend` so we fall back to a dummy object in order to avoid
811 // throwing an error. The value returned here will be incorrect, but since this only
812 // breaks inside a developer tool and the value is only used for secondary information,
813 // we can get away with it. See https://bugzilla.mozilla.org/show_bug.cgi?id=1615824.
814 (event.touches[0] || event.changedTouches[0] || { pageX: 0, pageY: 0 }) : event;
815 const x = point.pageX - scrollPosition.left;
816 const y = point.pageY - scrollPosition.top;
817 // if dragging SVG element, try to convert from the screen coordinate system to the SVG
818 // coordinate system
819 if (this._ownerSVGElement) {
820 const svgMatrix = this._ownerSVGElement.getScreenCTM();
821 if (svgMatrix) {
822 const svgPoint = this._ownerSVGElement.createSVGPoint();
823 svgPoint.x = x;
824 svgPoint.y = y;
825 return svgPoint.matrixTransform(svgMatrix.inverse());
826 }
827 }
828 return { x, y };
829 }
830 /** Gets the pointer position on the page, accounting for any position constraints. */
831 _getConstrainedPointerPosition(point) {
832 const dropContainerLock = this._dropContainer ? this._dropContainer.lockAxis : null;
833 let { x, y } = this.constrainPosition ? this.constrainPosition(point, this) : point;
834 if (this.lockAxis === 'x' || dropContainerLock === 'x') {
835 y = this._pickupPositionOnPage.y;
836 }
837 else if (this.lockAxis === 'y' || dropContainerLock === 'y') {
838 x = this._pickupPositionOnPage.x;
839 }
840 if (this._boundaryRect) {
841 const { x: pickupX, y: pickupY } = this._pickupPositionInElement;
842 const boundaryRect = this._boundaryRect;
843 const previewRect = this._previewRect;
844 const minY = boundaryRect.top + pickupY;
845 const maxY = boundaryRect.bottom - (previewRect.height - pickupY);
846 const minX = boundaryRect.left + pickupX;
847 const maxX = boundaryRect.right - (previewRect.width - pickupX);
848 x = clamp(x, minX, maxX);
849 y = clamp(y, minY, maxY);
850 }
851 return { x, y };
852 }
853 /** Updates the current drag delta, based on the user's current pointer position on the page. */
854 _updatePointerDirectionDelta(pointerPositionOnPage) {
855 const { x, y } = pointerPositionOnPage;
856 const delta = this._pointerDirectionDelta;
857 const positionSinceLastChange = this._pointerPositionAtLastDirectionChange;
858 // Amount of pixels the user has dragged since the last time the direction changed.
859 const changeX = Math.abs(x - positionSinceLastChange.x);
860 const changeY = Math.abs(y - positionSinceLastChange.y);
861 // Because we handle pointer events on a per-pixel basis, we don't want the delta
862 // to change for every pixel, otherwise anything that depends on it can look erratic.
863 // To make the delta more consistent, we track how much the user has moved since the last
864 // delta change and we only update it after it has reached a certain threshold.
865 if (changeX > this._config.pointerDirectionChangeThreshold) {
866 delta.x = x > positionSinceLastChange.x ? 1 : -1;
867 positionSinceLastChange.x = x;
868 }
869 if (changeY > this._config.pointerDirectionChangeThreshold) {
870 delta.y = y > positionSinceLastChange.y ? 1 : -1;
871 positionSinceLastChange.y = y;
872 }
873 return delta;
874 }
875 /** Toggles the native drag interactions, based on how many handles are registered. */
876 _toggleNativeDragInteractions() {
877 if (!this._rootElement || !this._handles) {
878 return;
879 }
880 const shouldEnable = this._handles.length > 0 || !this.isDragging();
881 if (shouldEnable !== this._nativeInteractionsEnabled) {
882 this._nativeInteractionsEnabled = shouldEnable;
883 toggleNativeDragInteractions(this._rootElement, shouldEnable);
884 }
885 }
886 /** Removes the manually-added event listeners from the root element. */
887 _removeRootElementListeners(element) {
888 element.removeEventListener('mousedown', this._pointerDown, activeEventListenerOptions);
889 element.removeEventListener('touchstart', this._pointerDown, passiveEventListenerOptions);
890 }
891 /**
892 * Applies a `transform` to the root element, taking into account any existing transforms on it.
893 * @param x New transform value along the X axis.
894 * @param y New transform value along the Y axis.
895 */
896 _applyRootElementTransform(x, y) {
897 const transform = getTransform(x, y);
898 // Cache the previous transform amount only after the first drag sequence, because
899 // we don't want our own transforms to stack on top of each other.
900 // Should be excluded none because none + translate3d(x, y, x) is invalid css
901 if (this._initialTransform == null) {
902 this._initialTransform = this._rootElement.style.transform
903 && this._rootElement.style.transform != 'none'
904 ? this._rootElement.style.transform
905 : '';
906 }
907 // Preserve the previous `transform` value, if there was one. Note that we apply our own
908 // transform before the user's, because things like rotation can affect which direction
909 // the element will be translated towards.
910 this._rootElement.style.transform = combineTransforms(transform, this._initialTransform);
911 }
912 /**
913 * Applies a `transform` to the preview, taking into account any existing transforms on it.
914 * @param x New transform value along the X axis.
915 * @param y New transform value along the Y axis.
916 */
917 _applyPreviewTransform(x, y) {
918 var _a;
919 // Only apply the initial transform if the preview is a clone of the original element, otherwise
920 // it could be completely different and the transform might not make sense anymore.
921 const initialTransform = ((_a = this._previewTemplate) === null || _a === void 0 ? void 0 : _a.template) ? undefined : this._initialTransform;
922 const transform = getTransform(x, y);
923 this._preview.style.transform = combineTransforms(transform, initialTransform);
924 }
925 /**
926 * Gets the distance that the user has dragged during the current drag sequence.
927 * @param currentPosition Current position of the user's pointer.
928 */
929 _getDragDistance(currentPosition) {
930 const pickupPosition = this._pickupPositionOnPage;
931 if (pickupPosition) {
932 return { x: currentPosition.x - pickupPosition.x, y: currentPosition.y - pickupPosition.y };
933 }
934 return { x: 0, y: 0 };
935 }
936 /** Cleans up any cached element dimensions that we don't need after dragging has stopped. */
937 _cleanupCachedDimensions() {
938 this._boundaryRect = this._previewRect = undefined;
939 this._parentPositions.clear();
940 }
941 /**
942 * Checks whether the element is still inside its boundary after the viewport has been resized.
943 * If not, the position is adjusted so that the element fits again.
944 */
945 _containInsideBoundaryOnResize() {
946 let { x, y } = this._passiveTransform;
947 if ((x === 0 && y === 0) || this.isDragging() || !this._boundaryElement) {
948 return;
949 }
950 const boundaryRect = this._boundaryElement.getBoundingClientRect();
951 const elementRect = this._rootElement.getBoundingClientRect();
952 // It's possible that the element got hidden away after dragging (e.g. by switching to a
953 // different tab). Don't do anything in this case so we don't clear the user's position.
954 if ((boundaryRect.width === 0 && boundaryRect.height === 0) ||
955 (elementRect.width === 0 && elementRect.height === 0)) {
956 return;
957 }
958 const leftOverflow = boundaryRect.left - elementRect.left;
959 const rightOverflow = elementRect.right - boundaryRect.right;
960 const topOverflow = boundaryRect.top - elementRect.top;
961 const bottomOverflow = elementRect.bottom - boundaryRect.bottom;
962 // If the element has become wider than the boundary, we can't
963 // do much to make it fit so we just anchor it to the left.
964 if (boundaryRect.width > elementRect.width) {
965 if (leftOverflow > 0) {
966 x += leftOverflow;
967 }
968 if (rightOverflow > 0) {
969 x -= rightOverflow;
970 }
971 }
972 else {
973 x = 0;
974 }
975 // If the element has become taller than the boundary, we can't
976 // do much to make it fit so we just anchor it to the top.
977 if (boundaryRect.height > elementRect.height) {
978 if (topOverflow > 0) {
979 y += topOverflow;
980 }
981 if (bottomOverflow > 0) {
982 y -= bottomOverflow;
983 }
984 }
985 else {
986 y = 0;
987 }
988 if (x !== this._passiveTransform.x || y !== this._passiveTransform.y) {
989 this.setFreeDragPosition({ y, x });
990 }
991 }
992 /** Gets the drag start delay, based on the event type. */
993 _getDragStartDelay(event) {
994 const value = this.dragStartDelay;
995 if (typeof value === 'number') {
996 return value;
997 }
998 else if (isTouchEvent(event)) {
999 return value.touch;
1000 }
1001 return value ? value.mouse : 0;
1002 }
1003 /** Updates the internal state of the draggable element when scrolling has occurred. */
1004 _updateOnScroll(event) {
1005 const scrollDifference = this._parentPositions.handleScroll(event);
1006 if (scrollDifference) {
1007 const target = _getEventTarget(event);
1008 // ClientRect dimensions are based on the scroll position of the page and its parent node so
1009 // we have to update the cached boundary ClientRect if the user has scrolled. Check for
1010 // the `document` specifically since IE doesn't support `contains` on it.
1011 if (this._boundaryRect && (target === this._document ||
1012 (target !== this._boundaryElement && target.contains(this._boundaryElement)))) {
1013 adjustClientRect(this._boundaryRect, scrollDifference.top, scrollDifference.left);
1014 }
1015 this._pickupPositionOnPage.x += scrollDifference.left;
1016 this._pickupPositionOnPage.y += scrollDifference.top;
1017 // If we're in free drag mode, we have to update the active transform, because
1018 // it isn't relative to the viewport like the preview inside a drop list.
1019 if (!this._dropContainer) {
1020 this._activeTransform.x -= scrollDifference.left;
1021 this._activeTransform.y -= scrollDifference.top;
1022 this._applyRootElementTransform(this._activeTransform.x, this._activeTransform.y);
1023 }
1024 }
1025 }
1026 /** Gets the scroll position of the viewport. */
1027 _getViewportScrollPosition() {
1028 const cachedPosition = this._parentPositions.positions.get(this._document);
1029 return cachedPosition ? cachedPosition.scrollPosition :
1030 this._viewportRuler.getViewportScrollPosition();
1031 }
1032 /**
1033 * Lazily resolves and returns the shadow root of the element. We do this in a function, rather
1034 * than saving it in property directly on init, because we want to resolve it as late as possible
1035 * in order to ensure that the element has been moved into the shadow DOM. Doing it inside the
1036 * constructor might be too early if the element is inside of something like `ngFor` or `ngIf`.
1037 */
1038 _getShadowRoot() {
1039 if (this._cachedShadowRoot === undefined) {
1040 this._cachedShadowRoot = _getShadowRoot(this._rootElement);
1041 }
1042 return this._cachedShadowRoot;
1043 }
1044 /** Gets the element into which the drag preview should be inserted. */
1045 _getPreviewInsertionPoint(initialParent, shadowRoot) {
1046 const previewContainer = this._previewContainer || 'global';
1047 if (previewContainer === 'parent') {
1048 return initialParent;
1049 }
1050 if (previewContainer === 'global') {
1051 const documentRef = this._document;
1052 // We can't use the body if the user is in fullscreen mode,
1053 // because the preview will render under the fullscreen element.
1054 // TODO(crisbeto): dedupe this with the `FullscreenOverlayContainer` eventually.
1055 return shadowRoot ||
1056 documentRef.fullscreenElement ||
1057 documentRef.webkitFullscreenElement ||
1058 documentRef.mozFullScreenElement ||
1059 documentRef.msFullscreenElement ||
1060 documentRef.body;
1061 }
1062 return coerceElement(previewContainer);
1063 }
1064}
1065/**
1066 * Gets a 3d `transform` that can be applied to an element.
1067 * @param x Desired position of the element along the X axis.
1068 * @param y Desired position of the element along the Y axis.
1069 */
1070function getTransform(x, y) {
1071 // Round the transforms since some browsers will
1072 // blur the elements for sub-pixel transforms.
1073 return `translate3d(${Math.round(x)}px, ${Math.round(y)}px, 0)`;
1074}
1075/** Clamps a value between a minimum and a maximum. */
1076function clamp(value, min, max) {
1077 return Math.max(min, Math.min(max, value));
1078}
1079/**
1080 * Helper to remove a node from the DOM and to do all the necessary null checks.
1081 * @param node Node to be removed.
1082 */
1083function removeNode(node) {
1084 if (node && node.parentNode) {
1085 node.parentNode.removeChild(node);
1086 }
1087}
1088/** Determines whether an event is a touch event. */
1089function isTouchEvent(event) {
1090 // This function is called for every pixel that the user has dragged so we need it to be
1091 // as fast as possible. Since we only bind mouse events and touch events, we can assume
1092 // that if the event's name starts with `t`, it's a touch event.
1093 return event.type[0] === 't';
1094}
1095/**
1096 * Gets the root HTML element of an embedded view.
1097 * If the root is not an HTML element it gets wrapped in one.
1098 */
1099function getRootNode(viewRef, _document) {
1100 const rootNodes = viewRef.rootNodes;
1101 if (rootNodes.length === 1 && rootNodes[0].nodeType === _document.ELEMENT_NODE) {
1102 return rootNodes[0];
1103 }
1104 const wrapper = _document.createElement('div');
1105 rootNodes.forEach(node => wrapper.appendChild(node));
1106 return wrapper;
1107}
1108/**
1109 * Matches the target element's size to the source's size.
1110 * @param target Element that needs to be resized.
1111 * @param sourceRect Dimensions of the source element.
1112 */
1113function matchElementSize(target, sourceRect) {
1114 target.style.width = `${sourceRect.width}px`;
1115 target.style.height = `${sourceRect.height}px`;
1116 target.style.transform = getTransform(sourceRect.left, sourceRect.top);
1117}
1118//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drag-ref.js","sourceRoot":"","sources":["../../../../../../src/cdk/drag-drop/drag-ref.ts"],"names":[],"mappings":"AAAA;;;;;;GAMG;AAKH,OAAO,EACL,+BAA+B,EAC/B,eAAe,EACf,cAAc,GACf,MAAM,uBAAuB,CAAC;AAC/B,OAAO,EAAC,qBAAqB,EAAE,aAAa,EAAC,MAAM,uBAAuB,CAAC;AAC3E,OAAO,EACL,+BAA+B,EAC/B,gCAAgC,GACjC,MAAM,mBAAmB,CAAC;AAC3B,OAAO,EAAC,YAAY,EAAE,OAAO,EAAa,MAAM,MAAM,CAAC;AAGvD,OAAO,EACL,iBAAiB,EACjB,YAAY,EACZ,4BAA4B,EAC5B,gBAAgB,GACjB,MAAM,gBAAgB,CAAC;AACxB,OAAO,EAAC,kCAAkC,EAAC,MAAM,uBAAuB,CAAC;AACzE,OAAO,EAAC,oBAAoB,EAAE,gBAAgB,EAAC,MAAM,eAAe,CAAC;AACrE,OAAO,EAAC,qBAAqB,EAAC,MAAM,2BAA2B,CAAC;AAChE,OAAO,EAAC,aAAa,EAAC,MAAM,cAAc,CAAC;AAuB3C,iEAAiE;AACjE,MAAM,2BAA2B,GAAG,+BAA+B,CAAC,EAAC,OAAO,EAAE,IAAI,EAAC,CAAC,CAAC;AAErF,iEAAiE;AACjE,MAAM,0BAA0B,GAAG,+BAA+B,CAAC,EAAC,OAAO,EAAE,KAAK,EAAC,CAAC,CAAC;AAErF;;;;;GAKG;AACH,MAAM,uBAAuB,GAAG,GAAG,CAAC;AA8BpC,8DAA8D;AAC9D,MAAM,uBAAuB,GAAG,IAAI,GAAG,CAAC;IACtC,kGAAkG;IAClG,UAAU;CACX,CAAC,CAAC;AAgBH;;GAEG;AACH,MAAM,OAAO,OAAO;IA6OlB,YACE,OAA8C,EACtC,OAAsB,EACtB,SAAmB,EACnB,OAAe,EACf,cAA6B,EAC7B,iBAAyD;QAJzD,YAAO,GAAP,OAAO,CAAe;QACtB,cAAS,GAAT,SAAS,CAAU;QACnB,YAAO,GAAP,OAAO,CAAQ;QACf,mBAAc,GAAd,cAAc,CAAe;QAC7B,sBAAiB,GAAjB,iBAAiB,CAAwC;QAvNnE;;;;;WAKG;QACK,sBAAiB,GAAU,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;QAEhD,+EAA+E;QACvE,qBAAgB,GAAU,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;QAK/C;;;WAGG;QACK,wBAAmB,GAAG,KAAK,CAAC;QAcpC,0CAA0C;QACzB,gBAAW,GAAG,IAAI,OAAO,EAMtC,CAAC;QA4BL,+CAA+C;QACvC,6BAAwB,GAAG,YAAY,CAAC,KAAK,CAAC;QAEtD,sFAAsF;QAC9E,2BAAsB,GAAG,YAAY,CAAC,KAAK,CAAC;QAEpD,mDAAmD;QAC3C,wBAAmB,GAAG,YAAY,CAAC,KAAK,CAAC;QAEjD,kDAAkD;QAC1C,wBAAmB,GAAG,YAAY,CAAC,KAAK,CAAC;QAYjD,gDAAgD;QACxC,qBAAgB,GAAuB,IAAI,CAAC;QAEpD,sFAAsF;QAC9E,+BAA0B,GAAG,IAAI,CAAC;QAc1C,4DAA4D;QACpD,aAAQ,GAAkB,EAAE,CAAC;QAErC,sDAAsD;QAC9C,qBAAgB,GAAG,IAAI,GAAG,EAAe,CAAC;QAKlD,oCAAoC;QAC5B,eAAU,GAAc,KAAK,CAAC;QAetC;;;WAGG;QACH,mBAAc,GAA4C,CAAC,CAAC;QAkBpD,cAAS,GAAG,KAAK,CAAC;QAE1B,oDAAoD;QAC3C,kBAAa,GAAG,IAAI,OAAO,EAAQ,CAAC;QAE7C,oDAAoD;QAC3C,YAAO,GAAG,IAAI,OAAO,EAAqB,CAAC;QAEpD,wFAAwF;QAC/E,aAAQ,GAAG,IAAI,OAAO,EAAqB,CAAC;QAErD,mEAAmE;QAC1D,UAAK,GAAG,IAAI,OAAO,EAAwD,CAAC;QAErF,mEAAmE;QAC1D,YAAO,GAAG,IAAI,OAAO,EAAiE,CAAC;QAEhG,gGAAgG;QACvF,WAAM,GAAG,IAAI,OAAO,EAA2C,CAAC;QAEzE,6DAA6D;QACpD,YAAO,GAAG,IAAI,OAAO,EAS1B,CAAC;QAEL;;;WAGG;QACM,UAAK,GAMT,IAAI,CAAC,WAAW,CAAC;QA+RtB,uDAAuD;QAC/C,iBAAY,GAAG,CAAC,KAA8B,EAAE,EAAE;YACxD,IAAI,CAAC,aAAa,CAAC,IAAI,EAAE,CAAC;YAE1B,sFAAsF;YACtF,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;gBACxB,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE;oBAC/C,MAAM,MAAM,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;oBACtC,OAAO,CAAC,CAAC,MAAM,IAAI,CAAC,MAAM,KAAK,MAAM,IAAI,MAAM,CAAC,QAAQ,CAAC,MAAqB,CAAC,CAAC,CAAC;gBACnF,CAAC,CAAC,CAAC;gBAEH,IAAI,YAAY,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;oBAC9E,IAAI,CAAC,uBAAuB,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;iBACnD;aACF;iBAAM,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;gBACzB,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;aACxD;QACH,CAAC,CAAA;QAED,gGAAgG;QACxF,iBAAY,GAAG,CAAC,KAA8B,EAAE,EAAE;YACxD,MAAM,eAAe,GAAG,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;YAE9D,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE;gBAC7B,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC7E,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,CAAC,eAAe,CAAC,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;gBAC7E,MAAM,eAAe,GAAG,SAAS,GAAG,SAAS,IAAI,IAAI,CAAC,OAAO,CAAC,kBAAkB,CAAC;gBAEjF,wFAAwF;gBACxF,6FAA6F;gBAC7F,yFAAyF;gBACzF,wEAAwE;gBACxE,IAAI,eAAe,EAAE;oBACnB,MAAM,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC;oBAC1F,MAAM,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC;oBAEtC,IAAI,CAAC,cAAc,EAAE;wBACnB,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;wBAC7B,OAAO;qBACR;oBAED,uFAAuF;oBACvF,sFAAsF;oBACtF,4EAA4E;oBAC5E,IAAI,CAAC,SAAS,IAAI,CAAC,CAAC,SAAS,CAAC,UAAU,EAAE,IAAI,CAAC,SAAS,CAAC,WAAW,EAAE,CAAC,EAAE;wBACvE,+EAA+E;wBAC/E,sFAAsF;wBACtF,KAAK,CAAC,cAAc,EAAE,CAAC;wBACvB,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;wBAChC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,CAAC,CAAC;qBACxD;iBACF;gBAED,OAAO;aACR;YAED,qEAAqE;YACrE,IAAI,IAAI,CAAC,gBAAgB,EAAE;gBACzB,uEAAuE;gBACvE,sEAAsE;gBACtE,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,EAAE;oBACjF,IAAI,CAAC,YAAY,GAAG,CAAC,IAAI,CAAC,QAAQ,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC,qBAAqB,EAAE,CAAC;iBAClF;aACF;YAED,6FAA6F;YAC7F,4FAA4F;YAC5F,+BAA+B;YAC/B,KAAK,CAAC,cAAc,EAAE,CAAC;YAEvB,MAAM,0BAA0B,GAAG,IAAI,CAAC,8BAA8B,CAAC,eAAe,CAAC,CAAC;YACxF,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;YACtB,IAAI,CAAC,yBAAyB,GAAG,eAAe,CAAC;YACjD,IAAI,CAAC,4BAA4B,CAAC,0BAA0B,CAAC,CAAC;YAE9D,IAAI,IAAI,CAAC,cAAc,EAAE;gBACvB,IAAI,CAAC,0BAA0B,CAAC,0BAA0B,EAAE,eAAe,CAAC,CAAC;aAC9E;iBAAM;gBACL,MAAM,eAAe,GAAG,IAAI,CAAC,gBAAgB,CAAC;gBAC9C,eAAe,CAAC,CAAC;oBACb,0BAA0B,CAAC,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBAC3F,eAAe,CAAC,CAAC;oBACb,0BAA0B,CAAC,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC;gBAE3F,IAAI,CAAC,0BAA0B,CAAC,eAAe,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC;gBAEtE,0EAA0E;gBAC1E,IAAI,OAAO,UAAU,KAAK,WAAW,IAAI,IAAI,CAAC,YAAY,YAAY,UAAU,EAAE;oBAChF,MAAM,gBAAgB,GAAG,aAAa,eAAe,CAAC,CAAC,IAAI,eAAe,CAAC,CAAC,GAAG,CAAC;oBAChF,IAAI,CAAC,YAAY,CAAC,YAAY,CAAC,WAAW,EAAE,gBAAgB,CAAC,CAAC;iBAC/D;aACF;YAED,sEAAsE;YACtE,iEAAiE;YACjE,qEAAqE;YACrE,IAAI,IAAI,CAAC,WAAW,CAAC,SAAS,CAAC,MAAM,EAAE;gBACrC,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE;oBACpB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC;wBACpB,MAAM,EAAE,IAAI;wBACZ,eAAe,EAAE,0BAA0B;wBAC3C,KAAK;wBACL,QAAQ,EAAE,IAAI,CAAC,gBAAgB,CAAC,0BAA0B,CAAC;wBAC3D,KAAK,EAAE,IAAI,CAAC,sBAAsB;qBACnC,CAAC,CAAC;gBACL,CAAC,CAAC,CAAC;aACJ;QACH,CAAC,CAAA;QAED,6FAA6F;QACrF,eAAU,GAAG,CAAC,KAA8B,EAAE,EAAE;YACtD,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,CAAC;QAC/B,CAAC,CAAA;QA1XC,IAAI,CAAC,eAAe,CAAC,OAAO,CAAC,CAAC,UAAU,CAAC,OAAO,CAAC,aAAa,IAAI,IAAI,CAAC,CAAC;QACxE,IAAI,CAAC,gBAAgB,GAAG,IAAI,qBAAqB,CAAC,SAAS,EAAE,cAAc,CAAC,CAAC;QAC7E,iBAAiB,CAAC,gBAAgB,CAAC,IAAI,CAAC,CAAC;IAC3C,CAAC;IA/ED,yDAAyD;IACzD,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,SAAS,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,IAAI,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC;IACnF,CAAC;IACD,IAAI,QAAQ,CAAC,KAAc;QACzB,MAAM,QAAQ,GAAG,qBAAqB,CAAC,KAAK,CAAC,CAAC;QAE9C,IAAI,QAAQ,KAAK,IAAI,CAAC,SAAS,EAAE;YAC/B,IAAI,CAAC,SAAS,GAAG,QAAQ,CAAC;YAC1B,IAAI,CAAC,6BAA6B,EAAE,CAAC;YACrC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,4BAA4B,CAAC,MAAM,EAAE,QAAQ,CAAC,CAAC,CAAC;SACjF;IACH,CAAC;IAqED;;;OAGG;IACH,qBAAqB;QACnB,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED,0CAA0C;IAC1C,cAAc;QACZ,OAAO,IAAI,CAAC,YAAY,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,iBAAiB;QACf,OAAO,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,qBAAqB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;IAClF,CAAC;IAED,kEAAkE;IAClE,WAAW,CAAC,OAAkD;QAC5D,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,aAAa,CAAC,MAAM,CAAC,CAAC,CAAC;QAC7D,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC,4BAA4B,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC,CAAC;QACrF,IAAI,CAAC,6BAA6B,EAAE,CAAC;QAErC,6FAA6F;QAC7F,2FAA2F;QAC3F,+FAA+F;QAC/F,oDAAoD;QACpD,MAAM,eAAe,GAAG,IAAI,GAAG,EAAe,CAAC;QAC/C,IAAI,CAAC,gBAAgB,CAAC,OAAO,CAAC,MAAM,CAAC,EAAE;YACrC,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,EAAE;gBACtC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;aAC7B;QACH,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,gBAAgB,GAAG,eAAe,CAAC;QACxC,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,mBAAmB,CAAC,QAAoC;QACtD,IAAI,CAAC,gBAAgB,GAAG,QAAQ,CAAC;QACjC,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,uBAAuB,CAAC,QAAmC;QACzD,IAAI,CAAC,oBAAoB,GAAG,QAAQ,CAAC;QACrC,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;;OAIG;IACH,eAAe,CAAC,WAAkD;QAChE,MAAM,OAAO,GAAG,aAAa,CAAC,WAAW,CAAC,CAAC;QAE3C,IAAI,OAAO,KAAK,IAAI,CAAC,YAAY,EAAE;YACjC,IAAI,IAAI,CAAC,YAAY,EAAE;gBACrB,IAAI,CAAC,2BAA2B,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;aACrD;YAED,IAAI,CAAC,OAAO,CAAC,iBAAiB,CAAC,GAAG,EAAE;gBAClC,OAAO,CAAC,gBAAgB,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,EAAE,0BAA0B,CAAC,CAAC;gBACrF,OAAO,CAAC,gBAAgB,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,EAAE,2BAA2B,CAAC,CAAC;YACzF,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,iBAAiB,GAAG,SAAS,CAAC;YACnC,IAAI,CAAC,YAAY,GAAG,OAAO,CAAC;SAC7B;QAED,IAAI,OAAO,UAAU,KAAK,WAAW,IAAI,IAAI,CAAC,YAAY,YAAY,UAAU,EAAE;YAChF,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,YAAY,CAAC,eAAe,CAAC;SAC3D;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;OAEG;IACH,mBAAmB,CAAC,eAA6D;QAC/E,IAAI,CAAC,gBAAgB,GAAG,eAAe,CAAC,CAAC,CAAC,aAAa,CAAC,eAAe,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC;QAChF,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,CAAC;QACvC,IAAI,eAAe,EAAE;YACnB,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,cAAc;iBAC3C,MAAM,CAAC,EAAE,CAAC;iBACV,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,8BAA8B,EAAE,CAAC,CAAC;SAC3D;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAED,sDAAsD;IACtD,UAAU,CAAC,MAA+B;QACxC,IAAI,CAAC,cAAc,GAAG,MAAM,CAAC;QAC7B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,+DAA+D;IAC/D,OAAO;QACL,IAAI,CAAC,2BAA2B,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAEpD,8DAA8D;QAC9D,uDAAuD;QACvD,IAAI,IAAI,CAAC,UAAU,EAAE,EAAE;YACrB,wEAAwE;YACxE,wEAAwE;YACxE,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SAC/B;QAED,UAAU,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACzB,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,iBAAiB,CAAC,cAAc,CAAC,IAAI,CAAC,CAAC;QAC5C,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,aAAa,CAAC,QAAQ,EAAE,CAAC;QAC9B,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;QACxB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QACzB,IAAI,CAAC,KAAK,CAAC,QAAQ,EAAE,CAAC;QACtB,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;QACxB,IAAI,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;QACvB,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;QACxB,IAAI,CAAC,WAAW,CAAC,QAAQ,EAAE,CAAC;QAC5B,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;QACnB,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC;QAC9B,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;QAChC,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,CAAC;QACvC,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC;QAC9B,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,oBAAoB;YACzF,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,cAAc,GAAG,IAAK,CAAC;IACzE,CAAC;IAED,6DAA6D;IAC7D,UAAU;QACR,OAAO,IAAI,CAAC,mBAAmB,IAAI,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC;IAC7E,CAAC;IAED,6DAA6D;IAC7D,KAAK;QACH,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,iBAAiB,IAAI,EAAE,CAAC;QACjE,IAAI,CAAC,gBAAgB,GAAG,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;QACrC,IAAI,CAAC,iBAAiB,GAAG,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;IACxC,CAAC;IAED;;;OAGG;IACH,aAAa,CAAC,MAAmB;QAC/B,IAAI,CAAC,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,MAAM,CAAC,IAAI,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC,EAAE;YAC5E,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YAClC,4BAA4B,CAAC,MAAM,EAAE,IAAI,CAAC,CAAC;SAC5C;IACH,CAAC;IAED;;;OAGG;IACH,YAAY,CAAC,MAAmB;QAC9B,IAAI,IAAI,CAAC,gBAAgB,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE;YACrC,IAAI,CAAC,gBAAgB,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;YACrC,4BAA4B,CAAC,MAAM,EAAE,IAAI,CAAC,QAAQ,CAAC,CAAC;SACrD;IACH,CAAC;IAED,uDAAuD;IACvD,aAAa,CAAC,SAAoB;QAChC,IAAI,CAAC,UAAU,GAAG,SAAS,CAAC;QAC5B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,mDAAmD;IACnD,kBAAkB,CAAC,SAAsB;QACvC,IAAI,CAAC,cAAc,GAAG,SAAS,CAAC;IAClC,CAAC;IAED;;OAEG;IACH,mBAAmB;QACjB,MAAM,QAAQ,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;QACpF,OAAO,EAAC,CAAC,EAAE,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,QAAQ,CAAC,CAAC,EAAC,CAAC;IACxC,CAAC;IAED;;;OAGG;IACH,mBAAmB,CAAC,KAAY;QAC9B,IAAI,CAAC,gBAAgB,GAAG,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;QACrC,IAAI,CAAC,iBAAiB,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC;QACnC,IAAI,CAAC,iBAAiB,CAAC,CAAC,GAAG,KAAK,CAAC,CAAC,CAAC;QAEnC,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;YACxB,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC;SACnD;QAED,OAAO,IAAI,CAAC;IACd,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,KAAuB;QAC1C,IAAI,CAAC,iBAAiB,GAAG,KAAK,CAAC;QAC/B,OAAO,IAAI,CAAC;IACd,CAAC;IAED,8EAA8E;IAC9E,4BAA4B;QAC1B,MAAM,QAAQ,GAAG,IAAI,CAAC,yBAAyB,CAAC;QAEhD,IAAI,QAAQ,IAAI,IAAI,CAAC,cAAc,EAAE;YACnC,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,8BAA8B,CAAC,QAAQ,CAAC,EAAE,QAAQ,CAAC,CAAC;SAC1F;IACH,CAAC;IAED,kDAAkD;IAC1C,oBAAoB;QAC1B,IAAI,CAAC,wBAAwB,CAAC,WAAW,EAAE,CAAC;QAC5C,IAAI,CAAC,sBAAsB,CAAC,WAAW,EAAE,CAAC;QAC1C,IAAI,CAAC,mBAAmB,CAAC,WAAW,EAAE,CAAC;IACzC,CAAC;IAED,oDAAoD;IAC5C,eAAe;QACrB,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,UAAU,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;SAC3B;QAED,IAAI,IAAI,CAAC,WAAW,EAAE;YACpB,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE,CAAC;SAC5B;QAED,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,WAAW,GAAG,IAAK,CAAC;IAC3C,CAAC;IAED,wDAAwD;IAChD,mBAAmB;QACzB,IAAI,IAAI,CAAC,YAAY,EAAE;YACrB,UAAU,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SAC/B;QAED,IAAI,IAAI,CAAC,eAAe,EAAE;YACxB,IAAI,CAAC,eAAe,CAAC,OAAO,EAAE,CAAC;SAChC;QAED,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,eAAe,GAAG,IAAK,CAAC;IACnD,CAAC;IAoHD;;;OAGG;IACK,gBAAgB,CAAC,KAA8B;QACrD,gFAAgF;QAChF,uFAAuF;QACvF,qFAAqF;QACrF,kCAAkC;QAClC,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE;YAC5C,OAAO;SACR;QAED,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;QAC1C,IAAI,CAAC,6BAA6B,EAAE,CAAC;QAErC,IAAI,IAAI,CAAC,QAAQ,EAAE;YACjB,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,uBAAuB,GAAG,IAAI,CAAC,wBAAwB,CAAC;SACjF;QAED,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE;YAC7B,OAAO;SACR;QAED,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,IAAI,EAAC,CAAC,CAAC;QAEnC,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,8EAA8E;YAC9E,IAAI,CAAC,cAAc,CAAC,cAAc,EAAE,CAAC;YACrC,IAAI,CAAC,4BAA4B,EAAE,CAAC,IAAI,CAAC,GAAG,EAAE;gBAC5C,IAAI,CAAC,qBAAqB,CAAC,KAAK,CAAC,CAAC;gBAClC,IAAI,CAAC,wBAAwB,EAAE,CAAC;gBAChC,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;YAC5C,CAAC,CAAC,CAAC;SACJ;aAAM;YACL,6EAA6E;YAC7E,gFAAgF;YAChF,gCAAgC;YAChC,IAAI,CAAC,iBAAiB,CAAC,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC;YACnD,MAAM,eAAe,GAAG,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;YAC9D,IAAI,CAAC,iBAAiB,CAAC,CAAC,GAAG,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC;YACnD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE;gBACpB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC;oBACd,MAAM,EAAE,IAAI;oBACZ,QAAQ,EAAE,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAAC;oBAChD,SAAS,EAAE,eAAe;iBAC3B,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;YACH,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAChC,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;SAC3C;IACH,CAAC;IAED,oCAAoC;IAC5B,kBAAkB,CAAC,KAA8B;QACvD,IAAI,YAAY,CAAC,KAAK,CAAC,EAAE;YACvB,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;SACvC;QAED,IAAI,CAAC,6BAA6B,EAAE,CAAC;QAErC,MAAM,aAAa,GAAG,IAAI,CAAC,cAAc,CAAC;QAE1C,IAAI,aAAa,EAAE;YACjB,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC;YAClC,MAAM,MAAM,GAAG,OAAO,CAAC,UAAyB,CAAC;YACjD,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,yBAAyB,EAAE,CAAC;YACzE,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,EAAE,CAAC,CAAC;YAE/E,oDAAoD;YACpD,MAAM,UAAU,GAAG,IAAI,CAAC,cAAc,EAAE,CAAC;YAEzC,kFAAkF;YAClF,MAAM,CAAC,YAAY,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;YAErC,yEAAyE;YACzE,yEAAyE;YACzE,IAAI,CAAC,iBAAiB,GAAG,OAAO,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE,CAAC;YAEvD,qDAAqD;YACrD,4DAA4D;YAC5D,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,qBAAqB,EAAE,CAAC;YAE7C,8FAA8F;YAC9F,2FAA2F;YAC3F,4FAA4F;YAC5F,gBAAgB,CAAC,OAAO,EAAE,KAAK,EAAE,uBAAuB,CAAC,CAAC;YAC1D,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,WAAW,CAAC,MAAM,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC,CAAC;YAC3E,IAAI,CAAC,yBAAyB,CAAC,MAAM,EAAE,UAAU,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;YAC9E,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,IAAI,EAAC,CAAC,CAAC,CAAC,uCAAuC;YAC1E,aAAa,CAAC,KAAK,EAAE,CAAC;YACtB,IAAI,CAAC,iBAAiB,GAAG,aAAa,CAAC;YACvC,IAAI,CAAC,aAAa,GAAG,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;SACvD;aAAM;YACL,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,IAAI,EAAC,CAAC,CAAC;YAClC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,aAAa,GAAG,SAAU,CAAC;SAC1D;QAED,sEAAsE;QACtE,6DAA6D;QAC7D,IAAI,CAAC,gBAAgB,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,oBAAoB,EAAE,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;IACzF,CAAC;IAED;;;;;OAKG;IACK,uBAAuB,CAAC,gBAA6B,EAAE,KAA8B;QAC3F,iDAAiD;QACjD,uDAAuD;QACvD,IAAI,IAAI,CAAC,cAAc,EAAE;YACvB,KAAK,CAAC,eAAe,EAAE,CAAC;SACzB;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,EAAE,CAAC;QACrC,MAAM,eAAe,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC;QAC5C,MAAM,sBAAsB,GAAG,CAAC,eAAe,IAAK,KAAoB,CAAC,MAAM,KAAK,CAAC,CAAC;QACtF,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC;QACtC,MAAM,MAAM,GAAG,eAAe,CAAC,KAAK,CAAC,CAAC;QACtC,MAAM,gBAAgB,GAAG,CAAC,eAAe,IAAI,IAAI,CAAC,mBAAmB;YACnE,IAAI,CAAC,mBAAmB,GAAG,uBAAuB,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QAClE,MAAM,WAAW,GAAG,eAAe,CAAC,CAAC,CAAC,gCAAgC,CAAC,KAAmB,CAAC,CAAC,CAAC;YAC3F,+BAA+B,CAAC,KAAmB,CAAC,CAAC;QAEvD,uFAAuF;QACvF,uFAAuF;QACvF,yFAAyF;QACzF,yFAAyF;QACzF,yFAAyF;QACzF,uCAAuC;QACvC,IAAI,MAAM,IAAK,MAAsB,CAAC,SAAS,IAAI,KAAK,CAAC,IAAI,KAAK,WAAW,EAAE;YAC7E,KAAK,CAAC,cAAc,EAAE,CAAC;SACxB;QAED,+FAA+F;QAC/F,IAAI,UAAU,IAAI,sBAAsB,IAAI,gBAAgB,IAAI,WAAW,EAAE;YAC3E,OAAO;SACR;QAED,yFAAyF;QACzF,uFAAuF;QACvF,gBAAgB;QAChB,IAAI,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE;YACxB,IAAI,CAAC,wBAAwB,GAAG,WAAW,CAAC,KAAK,CAAC,uBAAuB,IAAI,EAAE,CAAC;YAChF,WAAW,CAAC,KAAK,CAAC,uBAAuB,GAAG,aAAa,CAAC;SAC3D;QAED,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QAElD,iEAAiE;QACjE,+FAA+F;QAC/F,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,CAAC,SAAS,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QAChG,IAAI,CAAC,sBAAsB,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;QAC1F,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC,iBAAiB;aAC9C,QAAQ,CAAC,IAAI,CAAC,cAAc,EAAE,CAAC;aAC/B,SAAS,CAAC,WAAW,CAAC,EAAE,CAAC,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,CAAC,CAAC;QAE/D,IAAI,IAAI,CAAC,gBAAgB,EAAE;YACzB,IAAI,CAAC,aAAa,GAAG,oBAAoB,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;SAClE;QAED,4FAA4F;QAC5F,8FAA8F;QAC9F,wEAAwE;QACxE,MAAM,eAAe,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAC9C,IAAI,CAAC,wBAAwB,GAAG,eAAe,IAAI,eAAe,CAAC,QAAQ;YACzE,CAAC,eAAe,CAAC,SAAS,CAAC,CAAC,CAAC,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC,CAAC;YAC3C,IAAI,CAAC,4BAA4B,CAAC,gBAAgB,EAAE,KAAK,CAAC,CAAC;QAC7D,MAAM,eAAe,GAAG,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,yBAAyB;YAC/E,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;QAC1C,IAAI,CAAC,sBAAsB,GAAG,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;QAC3C,IAAI,CAAC,qCAAqC,GAAG,EAAC,CAAC,EAAE,eAAe,CAAC,CAAC,EAAE,CAAC,EAAE,eAAe,CAAC,CAAC,EAAC,CAAC;QAC1F,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;QACjC,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC,IAAI,EAAE,KAAK,CAAC,CAAC;IACpD,CAAC;IAED,2FAA2F;IACnF,qBAAqB,CAAC,KAA8B;QAC1D,iFAAiF;QACjF,6FAA6F;QAC7F,8FAA8F;QAC9F,yDAAyD;QACzD,gBAAgB,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,EAAE,uBAAuB,CAAC,CAAC;QACnE,IAAI,CAAC,OAAO,CAAC,UAAW,CAAC,YAAY,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,CAAC,OAAO,CAAC,CAAC;QAEvE,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,iBAAiB,GAAG,SAAS,CAAC;QAE5E,uEAAuE;QACvE,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE;YACpB,MAAM,SAAS,GAAG,IAAI,CAAC,cAAe,CAAC;YACvC,MAAM,YAAY,GAAG,SAAS,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;YAClD,MAAM,eAAe,GAAG,IAAI,CAAC,yBAAyB,CAAC,KAAK,CAAC,CAAC;YAC9D,MAAM,QAAQ,GAAG,IAAI,CAAC,gBAAgB,CAAC,eAAe,CAAC,CAAC;YACxD,MAAM,sBAAsB,GAAG,SAAS,CAAC,gBAAgB,CACvD,eAAe,CAAC,CAAC,EAAE,eAAe,CAAC,CAAC,CAAC,CAAC;YAExC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,EAAC,MAAM,EAAE,IAAI,EAAE,QAAQ,EAAE,SAAS,EAAE,eAAe,EAAC,CAAC,CAAC;YACtE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;gBAChB,IAAI,EAAE,IAAI;gBACV,YAAY;gBACZ,aAAa,EAAE,IAAI,CAAC,aAAa;gBACjC,SAAS,EAAE,SAAS;gBACpB,iBAAiB,EAAE,IAAI,CAAC,iBAAiB;gBACzC,sBAAsB;gBACtB,QAAQ;gBACR,SAAS,EAAE,eAAe;aAC3B,CAAC,CAAC;YACH,SAAS,CAAC,IAAI,CAAC,IAAI,EAAE,YAAY,EAAE,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,iBAAiB,EAC3E,sBAAsB,EAAE,QAAQ,EAAE,eAAe,CAAC,CAAC;YACrD,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,iBAAiB,CAAC;QAC/C,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;OAGG;IACK,0BAA0B,CAAC,EAAC,CAAC,EAAE,CAAC,EAAQ,EAAE,EAAC,CAAC,EAAE,IAAI,EAAE,CAAC,EAAE,IAAI,EAAQ;QACzE,qDAAqD;QACrD,IAAI,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC,gCAAgC,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAEvF,uFAAuF;QACvF,yFAAyF;QACzF,yFAAyF;QACzF,6BAA6B;QAC7B,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,cAAc,KAAK,IAAI,CAAC,iBAAiB;YAC/D,IAAI,CAAC,iBAAiB,CAAC,gBAAgB,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE;YACjD,YAAY,GAAG,IAAI,CAAC,iBAAiB,CAAC;SACvC;QAED,IAAI,YAAY,IAAI,YAAY,KAAK,IAAI,CAAC,cAAc,EAAE;YACxD,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,EAAE;gBACpB,mDAAmD;gBACnD,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,EAAC,IAAI,EAAE,IAAI,EAAE,SAAS,EAAE,IAAI,CAAC,cAAe,EAAC,CAAC,CAAC;gBAChE,IAAI,CAAC,cAAe,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;gBAChC,sDAAsD;gBACtD,IAAI,CAAC,cAAc,GAAG,YAAa,CAAC;gBACpC,IAAI,CAAC,cAAc,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,EAAE,YAAY,KAAK,IAAI,CAAC,iBAAiB;oBACzE,sEAAsE;oBACtE,sDAAsD;oBACtD,YAAY,CAAC,eAAe,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;gBACnE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC;oBAChB,IAAI,EAAE,IAAI;oBACV,SAAS,EAAE,YAAa;oBACxB,YAAY,EAAE,YAAa,CAAC,YAAY,CAAC,IAAI,CAAC;iBAC/C,CAAC,CAAC;YACL,CAAC,CAAC,CAAC;SACJ;QAED,sEAAsE;QACtE,IAAI,IAAI,CAAC,UAAU,EAAE,EAAE;YACrB,IAAI,CAAC,cAAe,CAAC,0BAA0B,CAAC,IAAI,EAAE,IAAI,CAAC,CAAC;YAC5D,IAAI,CAAC,cAAe,CAAC,SAAS,CAAC,IAAI,EAAE,CAAC,EAAE,CAAC,EAAE,IAAI,CAAC,sBAAsB,CAAC,CAAC;YACxE,IAAI,CAAC,sBAAsB,CACzB,CAAC,GAAG,IAAI,CAAC,wBAAwB,CAAC,CAAC,EAAE,CAAC,GAAG,IAAI,CAAC,wBAAwB,CAAC,CAAC,CAAC,CAAC;SAC7E;IACH,CAAC;IAED;;;OAGG;IACK,qBAAqB;QAC3B,MAAM,aAAa,GAAG,IAAI,CAAC,gBAAgB,CAAC;QAC5C,MAAM,YAAY,GAAG,IAAI,CAAC,YAAY,CAAC;QACvC,MAAM,eAAe,GAAG,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC;QACtE,IAAI,OAAoB,CAAC;QAEzB,IAAI,eAAe,IAAI,aAAa,EAAE;YACpC,wDAAwD;YACxD,uDAAuD;YACvD,MAAM,QAAQ,GAAG,aAAa,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,qBAAqB,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC;YAC5F,MAAM,OAAO,GAAG,aAAa,CAAC,aAAa,CAAC,kBAAkB,CAAC,eAAe,EACf,aAAa,CAAC,OAAO,CAAC,CAAC;YACtF,OAAO,CAAC,aAAa,EAAE,CAAC;YACxB,OAAO,GAAG,WAAW,CAAC,OAAO,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;YAC/C,IAAI,CAAC,WAAW,GAAG,OAAO,CAAC;YAC3B,IAAI,aAAa,CAAC,SAAS,EAAE;gBAC3B,gBAAgB,CAAC,OAAO,EAAE,QAAS,CAAC,CAAC;aACtC;iBAAM;gBACL,OAAO,CAAC,KAAK,CAAC,SAAS;oBACnB,YAAY,CAAC,IAAI,CAAC,qBAAqB,CAAC,CAAC,EAAE,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;aAC9E;SACF;aAAM;YACL,MAAM,OAAO,GAAG,IAAI,CAAC,YAAY,CAAC;YAClC,OAAO,GAAG,aAAa,CAAC,OAAO,CAAC,CAAC;YACjC,gBAAgB,CAAC,OAAO,EAAE,OAAO,CAAC,qBAAqB,EAAE,CAAC,CAAC;YAE3D,IAAI,IAAI,CAAC,iBAAiB,EAAE;gBAC1B,OAAO,CAAC,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,iBAAiB,CAAC;aAClD;SACF;QAED,YAAY,CAAC,OAAO,CAAC,KAAK,EAAE;YAC1B,4EAA4E;YAC5E,+EAA+E;YAC/E,gBAAgB,EAAE,MAAM;YACxB,8FAA8F;YAC9F,QAAQ,EAAE,GAAG;YACb,UAAU,EAAE,OAAO;YACnB,KAAK,EAAE,GAAG;YACV,MAAM,EAAE,GAAG;YACX,SAAS,EAAE,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,IAAI,IAAI,EAAE;SAC5C,EAAE,uBAAuB,CAAC,CAAC;QAE5B,4BAA4B,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;QAC7C,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;QAC1C,OAAO,CAAC,YAAY,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QAE7C,IAAI,YAAY,EAAE;YAChB,IAAI,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE;gBAC/B,YAAY,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC,CAAC;aACrE;iBAAM;gBACL,OAAO,CAAC,SAAS,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;aACrC;SACF;QAED,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;;OAGG;IACK,4BAA4B;QAClC,oEAAoE;QACpE,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE;YACnB,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;SAC1B;QAED,MAAM,eAAe,GAAG,IAAI,CAAC,YAAY,CAAC,qBAAqB,EAAE,CAAC;QAElE,yDAAyD;QACzD,IAAI,CAAC,QAAQ,CAAC,SAAS,CAAC,GAAG,CAAC,oBAAoB,CAAC,CAAC;QAElD,gDAAgD;QAChD,IAAI,CAAC,sBAAsB,CAAC,eAAe,CAAC,IAAI,EAAE,eAAe,CAAC,GAAG,CAAC,CAAC;QAEvE,0FAA0F;QAC1F,0FAA0F;QAC1F,4FAA4F;QAC5F,qCAAqC;QACrC,MAAM,QAAQ,GAAG,kCAAkC,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;QAEnE,IAAI,QAAQ,KAAK,CAAC,EAAE;YAClB,OAAO,OAAO,CAAC,OAAO,EAAE,CAAC;SAC1B;QAED,OAAO,IAAI,CAAC,OAAO,CAAC,iBAAiB,CAAC,GAAG,EAAE;YACzC,OAAO,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE;gBAC3B,MAAM,OAAO,GAAG,CAAC,CAAC,KAAsB,EAAE,EAAE;;oBAC1C,IAAI,CAAC,KAAK,IAAI,CAAC,eAAe,CAAC,KAAK,CAAC,KAAK,IAAI,CAAC,QAAQ;wBACnD,KAAK,CAAC,YAAY,KAAK,WAAW,CAAC,EAAE;wBACvC,MAAA,IAAI,CAAC,QAAQ,0CAAE,mBAAmB,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;wBAC7D,OAAO,EAAE,CAAC;wBACV,YAAY,CAAC,OAAO,CAAC,CAAC;qBACvB;gBACH,CAAC,CAAuC,CAAC;gBAEzC,yFAAyF;gBACzF,wFAAwF;gBACxF,mEAAmE;gBACnE,MAAM,OAAO,GAAG,UAAU,CAAC,OAAmB,EAAE,QAAQ,GAAG,GAAG,CAAC,CAAC;gBAChE,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,eAAe,EAAE,OAAO,CAAC,CAAC;YAC3D,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED,2FAA2F;IACnF,yBAAyB;QAC/B,MAAM,iBAAiB,GAAG,IAAI,CAAC,oBAAoB,CAAC;QACpD,MAAM,mBAAmB,GAAG,iBAAiB,CAAC,CAAC,CAAC,iBAAiB,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC;QAClF,IAAI,WAAwB,CAAC;QAE7B,IAAI,mBAAmB,EAAE;YACvB,IAAI,CAAC,eAAe,GAAG,iBAAkB,CAAC,aAAa,CAAC,kBAAkB,CACxE,mBAAmB,EACnB,iBAAkB,CAAC,OAAO,CAC3B,CAAC;YACF,IAAI,CAAC,eAAe,CAAC,aAAa,EAAE,CAAC;YACrC,WAAW,GAAG,WAAW,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;SACjE;aAAM;YACL,WAAW,GAAG,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SAChD;QAED,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,sBAAsB,CAAC,CAAC;QAClD,OAAO,WAAW,CAAC;IACrB,CAAC;IAED;;;;OAIG;IACK,4BAA4B,CAAC,gBAA6B,EAC7B,KAA8B;QACjE,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,qBAAqB,EAAE,CAAC;QAC9D,MAAM,aAAa,GAAG,gBAAgB,KAAK,IAAI,CAAC,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,gBAAgB,CAAC;QACvF,MAAM,aAAa,GAAG,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC,CAAC,CAAC,WAAW,CAAC;QAC1F,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QACnE,MAAM,cAAc,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QACzD,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,GAAG,aAAa,CAAC,IAAI,GAAG,cAAc,CAAC,IAAI,CAAC;QACjE,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,GAAG,aAAa,CAAC,GAAG,GAAG,cAAc,CAAC,GAAG,CAAC;QAE/D,OAAO;YACL,CAAC,EAAE,aAAa,CAAC,IAAI,GAAG,WAAW,CAAC,IAAI,GAAG,CAAC;YAC5C,CAAC,EAAE,aAAa,CAAC,GAAG,GAAG,WAAW,CAAC,GAAG,GAAG,CAAC;SAC3C,CAAC;IACJ,CAAC;IAED,qEAAqE;IAC7D,yBAAyB,CAAC,KAA8B;QAC9D,MAAM,cAAc,GAAG,IAAI,CAAC,0BAA0B,EAAE,CAAC;QACzD,MAAM,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC,CAAC,CAAC;YAC/B,4FAA4F;YAC5F,2FAA2F;YAC3F,0FAA0F;YAC1F,sFAAsF;YACtF,oFAAoF;YACpF,uFAAuF;YACvF,qFAAqF;YACrF,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,KAAK,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,EAAC,KAAK,EAAE,CAAC,EAAE,KAAK,EAAE,CAAC,EAAC,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QAElF,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,GAAG,cAAc,CAAC,IAAI,CAAC;QAC5C,MAAM,CAAC,GAAG,KAAK,CAAC,KAAK,GAAG,cAAc,CAAC,GAAG,CAAC;QAE3C,uFAAuF;QACvF,oBAAoB;QACpB,IAAI,IAAI,CAAC,gBAAgB,EAAE;YACzB,MAAM,SAAS,GAAG,IAAI,CAAC,gBAAgB,CAAC,YAAY,EAAE,CAAC;YACvD,IAAI,SAAS,EAAE;gBACb,MAAM,QAAQ,GAAG,IAAI,CAAC,gBAAgB,CAAC,cAAc,EAAE,CAAC;gBACxD,QAAQ,CAAC,CAAC,GAAG,CAAC,CAAC;gBACf,QAAQ,CAAC,CAAC,GAAG,CAAC,CAAC;gBACf,OAAO,QAAQ,CAAC,eAAe,CAAC,SAAS,CAAC,OAAO,EAAE,CAAC,CAAC;aACtD;SACF;QAED,OAAO,EAAC,CAAC,EAAE,CAAC,EAAC,CAAC;IAChB,CAAC;IAGD,sFAAsF;IAC9E,8BAA8B,CAAC,KAAY;QACjD,MAAM,iBAAiB,GAAG,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC;QACpF,IAAI,EAAC,CAAC,EAAE,CAAC,EAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,KAAK,EAAE,IAAI,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC;QAElF,IAAI,IAAI,CAAC,QAAQ,KAAK,GAAG,IAAI,iBAAiB,KAAK,GAAG,EAAE;YACtD,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;SAClC;aAAM,IAAI,IAAI,CAAC,QAAQ,KAAK,GAAG,IAAI,iBAAiB,KAAK,GAAG,EAAE;YAC7D,CAAC,GAAG,IAAI,CAAC,qBAAqB,CAAC,CAAC,CAAC;SAClC;QAED,IAAI,IAAI,CAAC,aAAa,EAAE;YACtB,MAAM,EAAC,CAAC,EAAE,OAAO,EAAE,CAAC,EAAE,OAAO,EAAC,GAAG,IAAI,CAAC,wBAAwB,CAAC;YAC/D,MAAM,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC;YACxC,MAAM,WAAW,GAAG,IAAI,CAAC,YAAa,CAAC;YACvC,MAAM,IAAI,GAAG,YAAY,CAAC,GAAG,GAAG,OAAO,CAAC;YACxC,MAAM,IAAI,GAAG,YAAY,CAAC,MAAM,GAAG,CAAC,WAAW,CAAC,MAAM,GAAG,OAAO,CAAC,CAAC;YAClE,MAAM,IAAI,GAAG,YAAY,CAAC,IAAI,GAAG,OAAO,CAAC;YACzC,MAAM,IAAI,GAAG,YAAY,CAAC,KAAK,GAAG,CAAC,WAAW,CAAC,KAAK,GAAG,OAAO,CAAC,CAAC;YAEhE,CAAC,GAAG,KAAK,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;YACzB,CAAC,GAAG,KAAK,CAAC,CAAC,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;SAC1B;QAED,OAAO,EAAC,CAAC,EAAE,CAAC,EAAC,CAAC;IAChB,CAAC;IAGD,gGAAgG;IACxF,4BAA4B,CAAC,qBAA4B;QAC/D,MAAM,EAAC,CAAC,EAAE,CAAC,EAAC,GAAG,qBAAqB,CAAC;QACrC,MAAM,KAAK,GAAG,IAAI,CAAC,sBAAsB,CAAC;QAC1C,MAAM,uBAAuB,GAAG,IAAI,CAAC,qCAAqC,CAAC;QAE3E,mFAAmF;QACnF,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,uBAAuB,CAAC,CAAC,CAAC,CAAC;QACxD,MAAM,OAAO,GAAG,IAAI,CAAC,GAAG,CAAC,CAAC,GAAG,uBAAuB,CAAC,CAAC,CAAC,CAAC;QAExD,iFAAiF;QACjF,qFAAqF;QACrF,yFAAyF;QACzF,+EAA+E;QAC/E,IAAI,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,+BAA+B,EAAE;YAC1D,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,uBAAuB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACjD,uBAAuB,CAAC,CAAC,GAAG,CAAC,CAAC;SAC/B;QAED,IAAI,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,+BAA+B,EAAE;YAC1D,KAAK,CAAC,CAAC,GAAG,CAAC,GAAG,uBAAuB,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACjD,uBAAuB,CAAC,CAAC,GAAG,CAAC,CAAC;SAC/B;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED,sFAAsF;IAC9E,6BAA6B;QACnC,IAAI,CAAC,IAAI,CAAC,YAAY,IAAI,CAAC,IAAI,CAAC,QAAQ,EAAE;YACxC,OAAO;SACR;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;QAEpE,IAAI,YAAY,KAAK,IAAI,CAAC,0BAA0B,EAAE;YACpD,IAAI,CAAC,0BAA0B,GAAG,YAAY,CAAC;YAC/C,4BAA4B,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,CAAC,CAAC;SAC/D;IACH,CAAC;IAED,wEAAwE;IAChE,2BAA2B,CAAC,OAAoB;QACtD,OAAO,CAAC,mBAAmB,CAAC,WAAW,EAAE,IAAI,CAAC,YAAY,EAAE,0BAA0B,CAAC,CAAC;QACxF,OAAO,CAAC,mBAAmB,CAAC,YAAY,EAAE,IAAI,CAAC,YAAY,EAAE,2BAA2B,CAAC,CAAC;IAC5F,CAAC;IAED;;;;OAIG;IACK,0BAA0B,CAAC,CAAS,EAAE,CAAS;QACrD,MAAM,SAAS,GAAG,YAAY,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAErC,kFAAkF;QAClF,kEAAkE;QAClE,6EAA6E;QAC7E,IAAI,IAAI,CAAC,iBAAiB,IAAI,IAAI,EAAE;YAClC,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS;mBAC9B,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,IAAI,MAAM;gBAC9C,CAAC,CAAC,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS;gBACnC,CAAC,CAAC,EAAE,CAAC;SAC/B;QAED,wFAAwF;QACxF,uFAAuF;QACvF,0CAA0C;QAC1C,IAAI,CAAC,YAAY,CAAC,KAAK,CAAC,SAAS,GAAG,iBAAiB,CAAC,SAAS,EAAE,IAAI,CAAC,iBAAiB,CAAC,CAAC;IAC3F,CAAC;IAED;;;;OAIG;IACK,sBAAsB,CAAC,CAAS,EAAE,CAAS;;QACjD,gGAAgG;QAChG,mFAAmF;QACnF,MAAM,gBAAgB,GAAG,CAAA,MAAA,IAAI,CAAC,gBAAgB,0CAAE,QAAQ,EAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC;QAC9F,MAAM,SAAS,GAAG,YAAY,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QACrC,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,SAAS,GAAG,iBAAiB,CAAC,SAAS,EAAE,gBAAgB,CAAC,CAAC;IACjF,CAAC;IAED;;;OAGG;IACK,gBAAgB,CAAC,eAAsB;QAC7C,MAAM,cAAc,GAAG,IAAI,CAAC,qBAAqB,CAAC;QAElD,IAAI,cAAc,EAAE;YAClB,OAAO,EAAC,CAAC,EAAE,eAAe,CAAC,CAAC,GAAG,cAAc,CAAC,CAAC,EAAE,CAAC,EAAE,eAAe,CAAC,CAAC,GAAG,cAAc,CAAC,CAAC,EAAC,CAAC;SAC3F;QAED,OAAO,EAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAAC,CAAC;IACtB,CAAC;IAED,6FAA6F;IACrF,wBAAwB;QAC9B,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,YAAY,GAAG,SAAS,CAAC;QACnD,IAAI,CAAC,gBAAgB,CAAC,KAAK,EAAE,CAAC;IAChC,CAAC;IAED;;;OAGG;IACK,8BAA8B;QACpC,IAAI,EAAC,CAAC,EAAE,CAAC,EAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC;QAEpC,IAAI,CAAC,CAAC,KAAK,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,IAAI,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,IAAI,CAAC,gBAAgB,EAAE;YACvE,OAAO;SACR;QAED,MAAM,YAAY,GAAG,IAAI,CAAC,gBAAgB,CAAC,qBAAqB,EAAE,CAAC;QACnE,MAAM,WAAW,GAAG,IAAI,CAAC,YAAY,CAAC,qBAAqB,EAAE,CAAC;QAE9D,wFAAwF;QACxF,wFAAwF;QACxF,IAAI,CAAC,YAAY,CAAC,KAAK,KAAK,CAAC,IAAI,YAAY,CAAC,MAAM,KAAK,CAAC,CAAC;YACvD,CAAC,WAAW,CAAC,KAAK,KAAK,CAAC,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,CAAC,EAAE;YACzD,OAAO;SACR;QAED,MAAM,YAAY,GAAG,YAAY,CAAC,IAAI,GAAG,WAAW,CAAC,IAAI,CAAC;QAC1D,MAAM,aAAa,GAAG,WAAW,CAAC,KAAK,GAAG,YAAY,CAAC,KAAK,CAAC;QAC7D,MAAM,WAAW,GAAG,YAAY,CAAC,GAAG,GAAG,WAAW,CAAC,GAAG,CAAC;QACvD,MAAM,cAAc,GAAG,WAAW,CAAC,MAAM,GAAG,YAAY,CAAC,MAAM,CAAC;QAEhE,8DAA8D;QAC9D,2DAA2D;QAC3D,IAAI,YAAY,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,EAAE;YAC1C,IAAI,YAAY,GAAG,CAAC,EAAE;gBACpB,CAAC,IAAI,YAAY,CAAC;aACnB;YAED,IAAI,aAAa,GAAG,CAAC,EAAE;gBACrB,CAAC,IAAI,aAAa,CAAC;aACpB;SACF;aAAM;YACL,CAAC,GAAG,CAAC,CAAC;SACP;QAED,+DAA+D;QAC/D,0DAA0D;QAC1D,IAAI,YAAY,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM,EAAE;YAC5C,IAAI,WAAW,GAAG,CAAC,EAAE;gBACnB,CAAC,IAAI,WAAW,CAAC;aAClB;YAED,IAAI,cAAc,GAAG,CAAC,EAAE;gBACtB,CAAC,IAAI,cAAc,CAAC;aACrB;SACF;aAAM;YACL,CAAC,GAAG,CAAC,CAAC;SACP;QAED,IAAI,CAAC,KAAK,IAAI,CAAC,iBAAiB,CAAC,CAAC,IAAI,CAAC,KAAK,IAAI,CAAC,iBAAiB,CAAC,CAAC,EAAE;YACpE,IAAI,CAAC,mBAAmB,CAAC,EAAC,CAAC,EAAE,CAAC,EAAC,CAAC,CAAC;SAClC;IACH,CAAC;IAED,0DAA0D;IAClD,kBAAkB,CAAC,KAA8B;QACvD,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC;QAElC,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE;YAC7B,OAAO,KAAK,CAAC;SACd;aAAM,IAAI,YAAY,CAAC,KAAK,CAAC,EAAE;YAC9B,OAAO,KAAK,CAAC,KAAK,CAAC;SACpB;QAED,OAAO,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;IACjC,CAAC;IAED,uFAAuF;IAC/E,eAAe,CAAC,KAAY;QAClC,MAAM,gBAAgB,GAAG,IAAI,CAAC,gBAAgB,CAAC,YAAY,CAAC,KAAK,CAAC,CAAC;QAEnE,IAAI,gBAAgB,EAAE;YACpB,MAAM,MAAM,GAAG,eAAe,CAAuB,KAAK,CAAE,CAAC;YAE7D,4FAA4F;YAC5F,uFAAuF;YACvF,yEAAyE;YACzE,IAAI,IAAI,CAAC,aAAa,IAAI,CAAC,MAAM,KAAK,IAAI,CAAC,SAAS;gBAChD,CAAC,MAAM,KAAK,IAAI,CAAC,gBAAgB,IAAI,MAAM,CAAC,QAAQ,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,EAAE;gBACjF,gBAAgB,CAAC,IAAI,CAAC,aAAa,EAAE,gBAAgB,CAAC,GAAG,EAAE,gBAAgB,CAAC,IAAI,CAAC,CAAC;aACnF;YAED,IAAI,CAAC,qBAAqB,CAAC,CAAC,IAAI,gBAAgB,CAAC,IAAI,CAAC;YACtD,IAAI,CAAC,qBAAqB,CAAC,CAAC,IAAI,gBAAgB,CAAC,GAAG,CAAC;YAErD,8EAA8E;YAC9E,yEAAyE;YACzE,IAAI,CAAC,IAAI,CAAC,cAAc,EAAE;gBACxB,IAAI,CAAC,gBAAgB,CAAC,CAAC,IAAI,gBAAgB,CAAC,IAAI,CAAC;gBACjD,IAAI,CAAC,gBAAgB,CAAC,CAAC,IAAI,gBAAgB,CAAC,GAAG,CAAC;gBAChD,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC,EAAE,IAAI,CAAC,gBAAgB,CAAC,CAAC,CAAC,CAAC;aACnF;SACF;IACH,CAAC;IAED,gDAAgD;IACxC,0BAA0B;QAChC,MAAM,cAAc,GAAG,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;QAC3E,OAAO,cAAc,CAAC,CAAC,CAAC,cAAc,CAAC,cAAc,CAAC,CAAC;YACnD,IAAI,CAAC,cAAc,CAAC,yBAAyB,EAAE,CAAC;IACtD,CAAC;IAED;;;;;OAKG;IACK,cAAc;QACpB,IAAI,IAAI,CAAC,iBAAiB,KAAK,SAAS,EAAE;YACxC,IAAI,CAAC,iBAAiB,GAAG,cAAc,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;SAC5D;QAED,OAAO,IAAI,CAAC,iBAAiB,CAAC;IAChC,CAAC;IAED,uEAAuE;IAC/D,yBAAyB,CAAC,aAA0B,EAC1B,UAA6B;QAC7D,MAAM,gBAAgB,GAAG,IAAI,CAAC,iBAAiB,IAAI,QAAQ,CAAC;QAE5D,IAAI,gBAAgB,KAAK,QAAQ,EAAE;YACjC,OAAO,aAAa,CAAC;SACtB;QAED,IAAI,gBAAgB,KAAK,QAAQ,EAAE;YACjC,MAAM,WAAW,GAAG,IAAI,CAAC,SAAS,CAAC;YAEnC,2DAA2D;YAC3D,gEAAgE;YAChE,gFAAgF;YAChF,OAAO,UAAU;gBACV,WAAW,CAAC,iBAAiB;gBAC5B,WAAmB,CAAC,uBAAuB;gBAC3C,WAAmB,CAAC,oBAAoB;gBACxC,WAAmB,CAAC,mBAAmB;gBACxC,WAAW,CAAC,IAAI,CAAC;SACzB;QAED,OAAO,aAAa,CAAC,gBAAgB,CAAC,CAAC;IACzC,CAAC;CACF;AAED;;;;GAIG;AACH,SAAS,YAAY,CAAC,CAAS,EAAE,CAAS;IACxC,gDAAgD;IAChD,8CAA8C;IAC9C,OAAO,eAAe,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,QAAQ,CAAC;AAClE,CAAC;AAED,sDAAsD;AACtD,SAAS,KAAK,CAAC,KAAa,EAAE,GAAW,EAAE,GAAW;IACpD,OAAO,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,KAAK,CAAC,CAAC,CAAC;AAC7C,CAAC;AAED;;;GAGG;AACH,SAAS,UAAU,CAAC,IAAiB;IACnC,IAAI,IAAI,IAAI,IAAI,CAAC,UAAU,EAAE;QAC3B,IAAI,CAAC,UAAU,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC;KACnC;AACH,CAAC;AAED,oDAAoD;AACpD,SAAS,YAAY,CAAC,KAA8B;IAClD,wFAAwF;IACxF,uFAAuF;IACvF,gEAAgE;IAChE,OAAO,KAAK,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK,GAAG,CAAC;AAC/B,CAAC;AAED;;;GAGG;AACH,SAAS,WAAW,CAAC,OAA6B,EAAE,SAAmB;IACrE,MAAM,SAAS,GAAW,OAAO,CAAC,SAAS,CAAC;IAE5C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,IAAI,SAAS,CAAC,CAAC,CAAC,CAAC,QAAQ,KAAK,SAAS,CAAC,YAAY,EAAE;QAC9E,OAAO,SAAS,CAAC,CAAC,CAAgB,CAAC;KACpC;IAED,MAAM,OAAO,GAAG,SAAS,CAAC,aAAa,CAAC,KAAK,CAAC,CAAC;IAC/C,SAAS,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC;IACrD,OAAO,OAAO,CAAC;AACjB,CAAC;AAED;;;;GAIG;AACH,SAAS,gBAAgB,CAAC,MAAmB,EAAE,UAAsB;IACnE,MAAM,CAAC,KAAK,CAAC,KAAK,GAAG,GAAG,UAAU,CAAC,KAAK,IAAI,CAAC;IAC7C,MAAM,CAAC,KAAK,CAAC,MAAM,GAAG,GAAG,UAAU,CAAC,MAAM,IAAI,CAAC;IAC/C,MAAM,CAAC,KAAK,CAAC,SAAS,GAAG,YAAY,CAAC,UAAU,CAAC,IAAI,EAAE,UAAU,CAAC,GAAG,CAAC,CAAC;AACzE,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\nimport {EmbeddedViewRef, ElementRef, NgZone, ViewContainerRef, TemplateRef} from '@angular/core';\nimport {ViewportRuler} from '@angular/cdk/scrolling';\nimport {Direction} from '@angular/cdk/bidi';\nimport {\n  normalizePassiveListenerOptions,\n  _getEventTarget,\n  _getShadowRoot,\n} from '@angular/cdk/platform';\nimport {coerceBooleanProperty, coerceElement} from '@angular/cdk/coercion';\nimport {\n  isFakeMousedownFromScreenReader,\n  isFakeTouchstartFromScreenReader,\n} from '@angular/cdk/a11y';\nimport {Subscription, Subject, Observable} from 'rxjs';\nimport {DropListRefInternal as DropListRef} from './drop-list-ref';\nimport {DragDropRegistry} from './drag-drop-registry';\nimport {\n  combineTransforms,\n  extendStyles,\n  toggleNativeDragInteractions,\n  toggleVisibility,\n} from './drag-styling';\nimport {getTransformTransitionDurationInMs} from './transition-duration';\nimport {getMutableClientRect, adjustClientRect} from './client-rect';\nimport {ParentPositionTracker} from './parent-position-tracker';\nimport {deepCloneNode} from './clone-node';\n\n/** Object that can be used to configure the behavior of DragRef. */\nexport interface DragRefConfig {\n  /**\n   * Minimum amount of pixels that the user should\n   * drag, before the CDK initiates a drag sequence.\n   */\n  dragStartThreshold: number;\n\n  /**\n   * Amount the pixels the user should drag before the CDK\n   * considers them to have changed the drag direction.\n   */\n  pointerDirectionChangeThreshold: number;\n\n  /** `z-index` for the absolutely-positioned elements that are created by the drag item. */\n  zIndex?: number;\n\n  /** Ref that the current drag item is nested in. */\n  parentDragRef?: DragRef;\n}\n\n/** Options that can be used to bind a passive event listener. */\nconst passiveEventListenerOptions = normalizePassiveListenerOptions({passive: true});\n\n/** Options that can be used to bind an active event listener. */\nconst activeEventListenerOptions = normalizePassiveListenerOptions({passive: false});\n\n/**\n * Time in milliseconds for which to ignore mouse events, after\n * receiving a touch event. Used to avoid doing double work for\n * touch devices where the browser fires fake mouse events, in\n * addition to touch events.\n */\nconst MOUSE_EVENT_IGNORE_TIME = 800;\n\n// TODO(crisbeto): add an API for moving a draggable up/down the\n// list programmatically. Useful for keyboard controls.\n\n/**\n * Internal compile-time-only representation of a `DragRef`.\n * Used to avoid circular import issues between the `DragRef` and the `DropListRef`.\n * @docs-private\n */\nexport interface DragRefInternal extends DragRef {}\n\n/** Template that can be used to create a drag helper element (e.g. a preview or a placeholder). */\ninterface DragHelperTemplate<T = any> {\n  template: TemplateRef<T> | null;\n  viewContainer: ViewContainerRef;\n  context: T;\n}\n\n/** Template that can be used to create a drag preview element. */\ninterface DragPreviewTemplate<T = any> extends DragHelperTemplate<T> {\n  matchSize?: boolean;\n}\n\n/** Point on the page or within an element. */\nexport interface Point {\n  x: number;\n  y: number;\n}\n\n/** Inline styles to be set as `!important` while dragging. */\nconst dragImportantProperties = new Set([\n  // Needs to be important, because some `mat-table` sets `position: sticky !important`. See #22781.\n  'position'\n]);\n\n/**\n * Possible places into which the preview of a drag item can be inserted.\n * - `global` - Preview will be inserted at the bottom of the `<body>`. The advantage is that\n * you don't have to worry about `overflow: hidden` or `z-index`, but the item won't retain\n * its inherited styles.\n * - `parent` - Preview will be inserted into the parent of the drag item. The advantage is that\n * inherited styles will be preserved, but it may be clipped by `overflow: hidden` or not be\n * visible due to `z-index`. Furthermore, the preview is going to have an effect over selectors\n * like `:nth-child` and some flexbox configurations.\n * - `ElementRef<HTMLElement> | HTMLElement` - Preview will be inserted into a specific element.\n * Same advantages and disadvantages as `parent`.\n */\nexport type PreviewContainer = 'global' | 'parent' | ElementRef<HTMLElement> | HTMLElement;\n\n/**\n * Reference to a draggable item. Used to manipulate or dispose of the item.\n */\nexport class DragRef<T = any> {\n  /** Element displayed next to the user's pointer while the element is dragged. */\n  private _preview: HTMLElement;\n\n  /** Reference to the view of the preview element. */\n  private _previewRef: EmbeddedViewRef<any> | null;\n\n  /** Container into which to insert the preview. */\n  private _previewContainer: PreviewContainer | undefined;\n\n  /** Reference to the view of the placeholder element. */\n  private _placeholderRef: EmbeddedViewRef<any> | null;\n\n  /** Element that is rendered instead of the draggable item while it is being sorted. */\n  private _placeholder: HTMLElement;\n\n  /** Coordinates within the element at which the user picked up the element. */\n  private _pickupPositionInElement: Point;\n\n  /** Coordinates on the page at which the user picked up the element. */\n  private _pickupPositionOnPage: Point;\n\n  /**\n   * Anchor node used to save the place in the DOM where the element was\n   * picked up so that it can be restored at the end of the drag sequence.\n   */\n  private _anchor: Comment;\n\n  /**\n   * CSS `transform` applied to the element when it isn't being dragged. We need a\n   * passive transform in order for the dragged element to retain its new position\n   * after the user has stopped dragging and because we need to know the relative\n   * position in case they start dragging again. This corresponds to `element.style.transform`.\n   */\n  private _passiveTransform: Point = {x: 0, y: 0};\n\n  /** CSS `transform` that is applied to the element while it's being dragged. */\n  private _activeTransform: Point = {x: 0, y: 0};\n\n  /** Inline `transform` value that the element had before the first dragging sequence. */\n  private _initialTransform?: string;\n\n  /**\n   * Whether the dragging sequence has been started. Doesn't\n   * necessarily mean that the element has been moved.\n   */\n  private _hasStartedDragging = false;\n\n  /** Whether the element has moved since the user started dragging it. */\n  private _hasMoved: boolean;\n\n  /** Drop container in which the DragRef resided when dragging began. */\n  private _initialContainer: DropListRef;\n\n  /** Index at which the item started in its initial container. */\n  private _initialIndex: number;\n\n  /** Cached positions of scrollable parent elements. */\n  private _parentPositions: ParentPositionTracker;\n\n  /** Emits when the item is being moved. */\n  private readonly _moveEvents = new Subject<{\n    source: DragRef;\n    pointerPosition: {x: number, y: number};\n    event: MouseEvent | TouchEvent;\n    distance: Point;\n    delta: {x: -1 | 0 | 1, y: -1 | 0 | 1};\n  }>();\n\n  /** Keeps track of the direction in which the user is dragging along each axis. */\n  private _pointerDirectionDelta: {x: -1 | 0 | 1, y: -1 | 0 | 1};\n\n  /** Pointer position at which the last change in the delta occurred. */\n  private _pointerPositionAtLastDirectionChange: Point;\n\n  /** Position of the pointer at the last pointer event. */\n  private _lastKnownPointerPosition: Point;\n\n  /**\n   * Root DOM node of the drag instance. This is the element that will\n   * be moved around as the user is dragging.\n   */\n  private _rootElement: HTMLElement;\n\n  /**\n   * Nearest ancestor SVG, relative to which coordinates are calculated if dragging SVGElement\n   */\n  private _ownerSVGElement: SVGSVGElement | null;\n\n  /**\n   * Inline style value of `-webkit-tap-highlight-color` at the time the\n   * dragging was started. Used to restore the value once we're done dragging.\n   */\n  private _rootElementTapHighlight: string;\n\n  /** Subscription to pointer movement events. */\n  private _pointerMoveSubscription = Subscription.EMPTY;\n\n  /** Subscription to the event that is dispatched when the user lifts their pointer. */\n  private _pointerUpSubscription = Subscription.EMPTY;\n\n  /** Subscription to the viewport being scrolled. */\n  private _scrollSubscription = Subscription.EMPTY;\n\n  /** Subscription to the viewport being resized. */\n  private _resizeSubscription = Subscription.EMPTY;\n\n  /**\n   * Time at which the last touch event occurred. Used to avoid firing the same\n   * events multiple times on touch devices where the browser will fire a fake\n   * mouse event for each touch event, after a certain time.\n   */\n  private _lastTouchEventTime: number;\n\n  /** Time at which the last dragging sequence was started. */\n  private _dragStartTime: number;\n\n  /** Cached reference to the boundary element. */\n  private _boundaryElement: HTMLElement | null = null;\n\n  /** Whether the native dragging interactions have been enabled on the root element. */\n  private _nativeInteractionsEnabled = true;\n\n  /** Cached dimensions of the preview element. */\n  private _previewRect?: ClientRect;\n\n  /** Cached dimensions of the boundary element. */\n  private _boundaryRect?: ClientRect;\n\n  /** Element that will be used as a template to create the draggable item's preview. */\n  private _previewTemplate?: DragPreviewTemplate | null;\n\n  /** Template for placeholder element rendered to show where a draggable would be dropped. */\n  private _placeholderTemplate?: DragHelperTemplate | null;\n\n  /** Elements that can be used to drag the draggable item. */\n  private _handles: HTMLElement[] = [];\n\n  /** Registered handles that are currently disabled. */\n  private _disabledHandles = new Set<HTMLElement>();\n\n  /** Droppable container that the draggable is a part of. */\n  private _dropContainer?: DropListRef;\n\n  /** Layout direction of the item. */\n  private _direction: Direction = 'ltr';\n\n  /** Ref that the current drag item is nested in. */\n  private _parentDragRef: DragRef<unknown> | null;\n\n  /**\n   * Cached shadow root that the element is placed in. `null` means that the element isn't in\n   * the shadow DOM and `undefined` means that it hasn't been resolved yet. Should be read via\n   * `_getShadowRoot`, not directly.\n   */\n  private _cachedShadowRoot: ShadowRoot | null | undefined;\n\n  /** Axis along which dragging is locked. */\n  lockAxis: 'x' | 'y';\n\n  /**\n   * Amount of milliseconds to wait after the user has put their\n   * pointer down before starting to drag the element.\n   */\n  dragStartDelay: number | {touch: number, mouse: number} = 0;\n\n  /** Class to be added to the preview element. */\n  previewClass: string|string[]|undefined;\n\n  /** Whether starting to drag this element is disabled. */\n  get disabled(): boolean {\n    return this._disabled || !!(this._dropContainer && this._dropContainer.disabled);\n  }\n  set disabled(value: boolean) {\n    const newValue = coerceBooleanProperty(value);\n\n    if (newValue !== this._disabled) {\n      this._disabled = newValue;\n      this._toggleNativeDragInteractions();\n      this._handles.forEach(handle => toggleNativeDragInteractions(handle, newValue));\n    }\n  }\n  private _disabled = false;\n\n  /** Emits as the drag sequence is being prepared. */\n  readonly beforeStarted = new Subject<void>();\n\n  /** Emits when the user starts dragging the item. */\n  readonly started = new Subject<{source: DragRef}>();\n\n  /** Emits when the user has released a drag item, before any animations have started. */\n  readonly released = new Subject<{source: DragRef}>();\n\n  /** Emits when the user stops dragging an item in the container. */\n  readonly ended = new Subject<{source: DragRef, distance: Point, dropPoint: Point}>();\n\n  /** Emits when the user has moved the item into a new container. */\n  readonly entered = new Subject<{container: DropListRef, item: DragRef, currentIndex: number}>();\n\n  /** Emits when the user removes the item its container by dragging it into another container. */\n  readonly exited = new Subject<{container: DropListRef, item: DragRef}>();\n\n  /** Emits when the user drops the item inside a container. */\n  readonly dropped = new Subject<{\n    previousIndex: number;\n    currentIndex: number;\n    item: DragRef;\n    container: DropListRef;\n    previousContainer: DropListRef;\n    distance: Point;\n    dropPoint: Point;\n    isPointerOverContainer: boolean;\n  }>();\n\n  /**\n   * Emits as the user is dragging the item. Use with caution,\n   * because this event will fire for every pixel that the user has dragged.\n   */\n  readonly moved: Observable<{\n    source: DragRef;\n    pointerPosition: {x: number, y: number};\n    event: MouseEvent | TouchEvent;\n    distance: Point;\n    delta: {x: -1 | 0 | 1, y: -1 | 0 | 1};\n  }> = this._moveEvents;\n\n  /** Arbitrary data that can be attached to the drag item. */\n  data: T;\n\n  /**\n   * Function that can be used to customize the logic of how the position of the drag item\n   * is limited while it's being dragged. Gets called with a point containing the current position\n   * of the user's pointer on the page and should return a point describing where the item should\n   * be rendered.\n   */\n  constrainPosition?: (point: Point, dragRef: DragRef) => Point;\n\n  constructor(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    private _config: DragRefConfig,\n    private _document: Document,\n    private _ngZone: NgZone,\n    private _viewportRuler: ViewportRuler,\n    private _dragDropRegistry: DragDropRegistry<DragRef, DropListRef>) {\n\n    this.withRootElement(element).withParent(_config.parentDragRef || null);\n    this._parentPositions = new ParentPositionTracker(_document, _viewportRuler);\n    _dragDropRegistry.registerDragItem(this);\n  }\n\n  /**\n   * Returns the element that is being used as a placeholder\n   * while the current element is being dragged.\n   */\n  getPlaceholderElement(): HTMLElement {\n    return this._placeholder;\n  }\n\n  /** Returns the root draggable element. */\n  getRootElement(): HTMLElement {\n    return this._rootElement;\n  }\n\n  /**\n   * Gets the currently-visible element that represents the drag item.\n   * While dragging this is the placeholder, otherwise it's the root element.\n   */\n  getVisibleElement(): HTMLElement {\n    return this.isDragging() ? this.getPlaceholderElement() : this.getRootElement();\n  }\n\n  /** Registers the handles that can be used to drag the element. */\n  withHandles(handles: (HTMLElement | ElementRef<HTMLElement>)[]): this {\n    this._handles = handles.map(handle => coerceElement(handle));\n    this._handles.forEach(handle => toggleNativeDragInteractions(handle, this.disabled));\n    this._toggleNativeDragInteractions();\n\n    // Delete any lingering disabled handles that may have been destroyed. Note that we re-create\n    // the set, rather than iterate over it and filter out the destroyed handles, because while\n    // the ES spec allows for sets to be modified while they're being iterated over, some polyfills\n    // use an array internally which may throw an error.\n    const disabledHandles = new Set<HTMLElement>();\n    this._disabledHandles.forEach(handle => {\n      if (this._handles.indexOf(handle) > -1) {\n        disabledHandles.add(handle);\n      }\n    });\n    this._disabledHandles = disabledHandles;\n    return this;\n  }\n\n  /**\n   * Registers the template that should be used for the drag preview.\n   * @param template Template that from which to stamp out the preview.\n   */\n  withPreviewTemplate(template: DragPreviewTemplate | null): this {\n    this._previewTemplate = template;\n    return this;\n  }\n\n  /**\n   * Registers the template that should be used for the drag placeholder.\n   * @param template Template that from which to stamp out the placeholder.\n   */\n  withPlaceholderTemplate(template: DragHelperTemplate | null): this {\n    this._placeholderTemplate = template;\n    return this;\n  }\n\n  /**\n   * Sets an alternate drag root element. The root element is the element that will be moved as\n   * the user is dragging. Passing an alternate root element is useful when trying to enable\n   * dragging on an element that you might not have access to.\n   */\n  withRootElement(rootElement: ElementRef<HTMLElement> | HTMLElement): this {\n    const element = coerceElement(rootElement);\n\n    if (element !== this._rootElement) {\n      if (this._rootElement) {\n        this._removeRootElementListeners(this._rootElement);\n      }\n\n      this._ngZone.runOutsideAngular(() => {\n        element.addEventListener('mousedown', this._pointerDown, activeEventListenerOptions);\n        element.addEventListener('touchstart', this._pointerDown, passiveEventListenerOptions);\n      });\n      this._initialTransform = undefined;\n      this._rootElement = element;\n    }\n\n    if (typeof SVGElement !== 'undefined' && this._rootElement instanceof SVGElement) {\n      this._ownerSVGElement = this._rootElement.ownerSVGElement;\n    }\n\n    return this;\n  }\n\n  /**\n   * Element to which the draggable's position will be constrained.\n   */\n  withBoundaryElement(boundaryElement: ElementRef<HTMLElement> | HTMLElement | null): this {\n    this._boundaryElement = boundaryElement ? coerceElement(boundaryElement) : null;\n    this._resizeSubscription.unsubscribe();\n    if (boundaryElement) {\n      this._resizeSubscription = this._viewportRuler\n        .change(10)\n        .subscribe(() => this._containInsideBoundaryOnResize());\n    }\n    return this;\n  }\n\n  /** Sets the parent ref that the ref is nested in.  */\n  withParent(parent: DragRef<unknown> | null): this {\n    this._parentDragRef = parent;\n    return this;\n  }\n\n  /** Removes the dragging functionality from the DOM element. */\n  dispose() {\n    this._removeRootElementListeners(this._rootElement);\n\n    // Do this check before removing from the registry since it'll\n    // stop being considered as dragged once it is removed.\n    if (this.isDragging()) {\n      // Since we move out the element to the end of the body while it's being\n      // dragged, we have to make sure that it's removed if it gets destroyed.\n      removeNode(this._rootElement);\n    }\n\n    removeNode(this._anchor);\n    this._destroyPreview();\n    this._destroyPlaceholder();\n    this._dragDropRegistry.removeDragItem(this);\n    this._removeSubscriptions();\n    this.beforeStarted.complete();\n    this.started.complete();\n    this.released.complete();\n    this.ended.complete();\n    this.entered.complete();\n    this.exited.complete();\n    this.dropped.complete();\n    this._moveEvents.complete();\n    this._handles = [];\n    this._disabledHandles.clear();\n    this._dropContainer = undefined;\n    this._resizeSubscription.unsubscribe();\n    this._parentPositions.clear();\n    this._boundaryElement = this._rootElement = this._ownerSVGElement = this._placeholderTemplate =\n        this._previewTemplate = this._anchor = this._parentDragRef = null!;\n  }\n\n  /** Checks whether the element is currently being dragged. */\n  isDragging(): boolean {\n    return this._hasStartedDragging && this._dragDropRegistry.isDragging(this);\n  }\n\n  /** Resets a standalone drag item to its initial position. */\n  reset(): void {\n    this._rootElement.style.transform = this._initialTransform || '';\n    this._activeTransform = {x: 0, y: 0};\n    this._passiveTransform = {x: 0, y: 0};\n  }\n\n  /**\n   * Sets a handle as disabled. While a handle is disabled, it'll capture and interrupt dragging.\n   * @param handle Handle element that should be disabled.\n   */\n  disableHandle(handle: HTMLElement) {\n    if (!this._disabledHandles.has(handle) && this._handles.indexOf(handle) > -1) {\n      this._disabledHandles.add(handle);\n      toggleNativeDragInteractions(handle, true);\n    }\n  }\n\n  /**\n   * Enables a handle, if it has been disabled.\n   * @param handle Handle element to be enabled.\n   */\n  enableHandle(handle: HTMLElement) {\n    if (this._disabledHandles.has(handle)) {\n      this._disabledHandles.delete(handle);\n      toggleNativeDragInteractions(handle, this.disabled);\n    }\n  }\n\n  /** Sets the layout direction of the draggable item. */\n  withDirection(direction: Direction): this {\n    this._direction = direction;\n    return this;\n  }\n\n  /** Sets the container that the item is part of. */\n  _withDropContainer(container: DropListRef) {\n    this._dropContainer = container;\n  }\n\n  /**\n   * Gets the current position in pixels the draggable outside of a drop container.\n   */\n  getFreeDragPosition(): Readonly<Point> {\n    const position = this.isDragging() ? this._activeTransform : this._passiveTransform;\n    return {x: position.x, y: position.y};\n  }\n\n  /**\n   * Sets the current position in pixels the draggable outside of a drop container.\n   * @param value New position to be set.\n   */\n  setFreeDragPosition(value: Point): this {\n    this._activeTransform = {x: 0, y: 0};\n    this._passiveTransform.x = value.x;\n    this._passiveTransform.y = value.y;\n\n    if (!this._dropContainer) {\n      this._applyRootElementTransform(value.x, value.y);\n    }\n\n    return this;\n  }\n\n  /**\n   * Sets the container into which to insert the preview element.\n   * @param value Container into which to insert the preview.\n   */\n  withPreviewContainer(value: PreviewContainer): this {\n    this._previewContainer = value;\n    return this;\n  }\n\n  /** Updates the item's sort order based on the last-known pointer position. */\n  _sortFromLastPointerPosition() {\n    const position = this._lastKnownPointerPosition;\n\n    if (position && this._dropContainer) {\n      this._updateActiveDropContainer(this._getConstrainedPointerPosition(position), position);\n    }\n  }\n\n  /** Unsubscribes from the global subscriptions. */\n  private _removeSubscriptions() {\n    this._pointerMoveSubscription.unsubscribe();\n    this._pointerUpSubscription.unsubscribe();\n    this._scrollSubscription.unsubscribe();\n  }\n\n  /** Destroys the preview element and its ViewRef. */\n  private _destroyPreview() {\n    if (this._preview) {\n      removeNode(this._preview);\n    }\n\n    if (this._previewRef) {\n      this._previewRef.destroy();\n    }\n\n    this._preview = this._previewRef = null!;\n  }\n\n  /** Destroys the placeholder element and its ViewRef. */\n  private _destroyPlaceholder() {\n    if (this._placeholder) {\n      removeNode(this._placeholder);\n    }\n\n    if (this._placeholderRef) {\n      this._placeholderRef.destroy();\n    }\n\n    this._placeholder = this._placeholderRef = null!;\n  }\n\n  /** Handler for the `mousedown`/`touchstart` events. */\n  private _pointerDown = (event: MouseEvent | TouchEvent) => {\n    this.beforeStarted.next();\n\n    // Delegate the event based on whether it started from a handle or the element itself.\n    if (this._handles.length) {\n      const targetHandle = this._handles.find(handle => {\n        const target = _getEventTarget(event);\n        return !!target && (target === handle || handle.contains(target as HTMLElement));\n      });\n\n      if (targetHandle && !this._disabledHandles.has(targetHandle) && !this.disabled) {\n        this._initializeDragSequence(targetHandle, event);\n      }\n    } else if (!this.disabled) {\n      this._initializeDragSequence(this._rootElement, event);\n    }\n  }\n\n  /** Handler that is invoked when the user moves their pointer after they've initiated a drag. */\n  private _pointerMove = (event: MouseEvent | TouchEvent) => {\n    const pointerPosition = this._getPointerPositionOnPage(event);\n\n    if (!this._hasStartedDragging) {\n      const distanceX = Math.abs(pointerPosition.x - this._pickupPositionOnPage.x);\n      const distanceY = Math.abs(pointerPosition.y - this._pickupPositionOnPage.y);\n      const isOverThreshold = distanceX + distanceY >= this._config.dragStartThreshold;\n\n      // Only start dragging after the user has moved more than the minimum distance in either\n      // direction. Note that this is preferrable over doing something like `skip(minimumDistance)`\n      // in the `pointerMove` subscription, because we're not guaranteed to have one move event\n      // per pixel of movement (e.g. if the user moves their pointer quickly).\n      if (isOverThreshold) {\n        const isDelayElapsed = Date.now() >= this._dragStartTime + this._getDragStartDelay(event);\n        const container = this._dropContainer;\n\n        if (!isDelayElapsed) {\n          this._endDragSequence(event);\n          return;\n        }\n\n        // Prevent other drag sequences from starting while something in the container is still\n        // being dragged. This can happen while we're waiting for the drop animation to finish\n        // and can cause errors, because some elements might still be moving around.\n        if (!container || (!container.isDragging() && !container.isReceiving())) {\n          // Prevent the default action as soon as the dragging sequence is considered as\n          // \"started\" since waiting for the next event can allow the device to begin scrolling.\n          event.preventDefault();\n          this._hasStartedDragging = true;\n          this._ngZone.run(() => this._startDragSequence(event));\n        }\n      }\n\n      return;\n    }\n\n    // We only need the preview dimensions if we have a boundary element.\n    if (this._boundaryElement) {\n      // Cache the preview element rect if we haven't cached it already or if\n      // we cached it too early before the element dimensions were computed.\n      if (!this._previewRect || (!this._previewRect.width && !this._previewRect.height)) {\n        this._previewRect = (this._preview || this._rootElement).getBoundingClientRect();\n      }\n    }\n\n    // We prevent the default action down here so that we know that dragging has started. This is\n    // important for touch devices where doing this too early can unnecessarily block scrolling,\n    // if there's a dragging delay.\n    event.preventDefault();\n\n    const constrainedPointerPosition = this._getConstrainedPointerPosition(pointerPosition);\n    this._hasMoved = true;\n    this._lastKnownPointerPosition = pointerPosition;\n    this._updatePointerDirectionDelta(constrainedPointerPosition);\n\n    if (this._dropContainer) {\n      this._updateActiveDropContainer(constrainedPointerPosition, pointerPosition);\n    } else {\n      const activeTransform = this._activeTransform;\n      activeTransform.x =\n          constrainedPointerPosition.x - this._pickupPositionOnPage.x + this._passiveTransform.x;\n      activeTransform.y =\n          constrainedPointerPosition.y - this._pickupPositionOnPage.y + this._passiveTransform.y;\n\n      this._applyRootElementTransform(activeTransform.x, activeTransform.y);\n\n      // Apply transform as attribute if dragging and svg element to work for IE\n      if (typeof SVGElement !== 'undefined' && this._rootElement instanceof SVGElement) {\n        const appliedTransform = `translate(${activeTransform.x} ${activeTransform.y})`;\n        this._rootElement.setAttribute('transform', appliedTransform);\n      }\n    }\n\n    // Since this event gets fired for every pixel while dragging, we only\n    // want to fire it if the consumer opted into it. Also we have to\n    // re-enter the zone because we run all of the events on the outside.\n    if (this._moveEvents.observers.length) {\n      this._ngZone.run(() => {\n        this._moveEvents.next({\n          source: this,\n          pointerPosition: constrainedPointerPosition,\n          event,\n          distance: this._getDragDistance(constrainedPointerPosition),\n          delta: this._pointerDirectionDelta\n        });\n      });\n    }\n  }\n\n  /** Handler that is invoked when the user lifts their pointer up, after initiating a drag. */\n  private _pointerUp = (event: MouseEvent | TouchEvent) => {\n    this._endDragSequence(event);\n  }\n\n  /**\n   * Clears subscriptions and stops the dragging sequence.\n   * @param event Browser event object that ended the sequence.\n   */\n  private _endDragSequence(event: MouseEvent | TouchEvent) {\n    // Note that here we use `isDragging` from the service, rather than from `this`.\n    // The difference is that the one from the service reflects whether a dragging sequence\n    // has been initiated, whereas the one on `this` includes whether the user has passed\n    // the minimum dragging threshold.\n    if (!this._dragDropRegistry.isDragging(this)) {\n      return;\n    }\n\n    this._removeSubscriptions();\n    this._dragDropRegistry.stopDragging(this);\n    this._toggleNativeDragInteractions();\n\n    if (this._handles) {\n      this._rootElement.style.webkitTapHighlightColor = this._rootElementTapHighlight;\n    }\n\n    if (!this._hasStartedDragging) {\n      return;\n    }\n\n    this.released.next({source: this});\n\n    if (this._dropContainer) {\n      // Stop scrolling immediately, instead of waiting for the animation to finish.\n      this._dropContainer._stopScrolling();\n      this._animatePreviewToPlaceholder().then(() => {\n        this._cleanupDragArtifacts(event);\n        this._cleanupCachedDimensions();\n        this._dragDropRegistry.stopDragging(this);\n      });\n    } else {\n      // Convert the active transform into a passive one. This means that next time\n      // the user starts dragging the item, its position will be calculated relatively\n      // to the new passive transform.\n      this._passiveTransform.x = this._activeTransform.x;\n      const pointerPosition = this._getPointerPositionOnPage(event);\n      this._passiveTransform.y = this._activeTransform.y;\n      this._ngZone.run(() => {\n        this.ended.next({\n          source: this,\n          distance: this._getDragDistance(pointerPosition),\n          dropPoint: pointerPosition\n        });\n      });\n      this._cleanupCachedDimensions();\n      this._dragDropRegistry.stopDragging(this);\n    }\n  }\n\n  /** Starts the dragging sequence. */\n  private _startDragSequence(event: MouseEvent | TouchEvent) {\n    if (isTouchEvent(event)) {\n      this._lastTouchEventTime = Date.now();\n    }\n\n    this._toggleNativeDragInteractions();\n\n    const dropContainer = this._dropContainer;\n\n    if (dropContainer) {\n      const element = this._rootElement;\n      const parent = element.parentNode as HTMLElement;\n      const placeholder = this._placeholder = this._createPlaceholderElement();\n      const anchor = this._anchor = this._anchor || this._document.createComment('');\n\n      // Needs to happen before the root element is moved.\n      const shadowRoot = this._getShadowRoot();\n\n      // Insert an anchor node so that we can restore the element's position in the DOM.\n      parent.insertBefore(anchor, element);\n\n      // There's no risk of transforms stacking when inside a drop container so\n      // we can keep the initial transform up to date any time dragging starts.\n      this._initialTransform = element.style.transform || '';\n\n      // Create the preview after the initial transform has\n      // been cached, because it can be affected by the transform.\n      this._preview = this._createPreviewElement();\n\n      // We move the element out at the end of the body and we make it hidden, because keeping it in\n      // place will throw off the consumer's `:last-child` selectors. We can't remove the element\n      // from the DOM completely, because iOS will stop firing all subsequent events in the chain.\n      toggleVisibility(element, false, dragImportantProperties);\n      this._document.body.appendChild(parent.replaceChild(placeholder, element));\n      this._getPreviewInsertionPoint(parent, shadowRoot).appendChild(this._preview);\n      this.started.next({source: this}); // Emit before notifying the container.\n      dropContainer.start();\n      this._initialContainer = dropContainer;\n      this._initialIndex = dropContainer.getItemIndex(this);\n    } else {\n      this.started.next({source: this});\n      this._initialContainer = this._initialIndex = undefined!;\n    }\n\n    // Important to run after we've called `start` on the parent container\n    // so that it has had time to resolve its scrollable parents.\n    this._parentPositions.cache(dropContainer ? dropContainer.getScrollableParents() : []);\n  }\n\n  /**\n   * Sets up the different variables and subscriptions\n   * that will be necessary for the dragging sequence.\n   * @param referenceElement Element that started the drag sequence.\n   * @param event Browser event object that started the sequence.\n   */\n  private _initializeDragSequence(referenceElement: HTMLElement, event: MouseEvent | TouchEvent) {\n    // Stop propagation if the item is inside another\n    // draggable so we don't start multiple drag sequences.\n    if (this._parentDragRef) {\n      event.stopPropagation();\n    }\n\n    const isDragging = this.isDragging();\n    const isTouchSequence = isTouchEvent(event);\n    const isAuxiliaryMouseButton = !isTouchSequence && (event as MouseEvent).button !== 0;\n    const rootElement = this._rootElement;\n    const target = _getEventTarget(event);\n    const isSyntheticEvent = !isTouchSequence && this._lastTouchEventTime &&\n      this._lastTouchEventTime + MOUSE_EVENT_IGNORE_TIME > Date.now();\n    const isFakeEvent = isTouchSequence ? isFakeTouchstartFromScreenReader(event as TouchEvent) :\n      isFakeMousedownFromScreenReader(event as MouseEvent);\n\n    // If the event started from an element with the native HTML drag&drop, it'll interfere\n    // with our own dragging (e.g. `img` tags do it by default). Prevent the default action\n    // to stop it from happening. Note that preventing on `dragstart` also seems to work, but\n    // it's flaky and it fails if the user drags it away quickly. Also note that we only want\n    // to do this for `mousedown` since doing the same for `touchstart` will stop any `click`\n    // events from firing on touch devices.\n    if (target && (target as HTMLElement).draggable && event.type === 'mousedown') {\n      event.preventDefault();\n    }\n\n    // Abort if the user is already dragging or is using a mouse button other than the primary one.\n    if (isDragging || isAuxiliaryMouseButton || isSyntheticEvent || isFakeEvent) {\n      return;\n    }\n\n    // If we've got handles, we need to disable the tap highlight on the entire root element,\n    // otherwise iOS will still add it, even though all the drag interactions on the handle\n    // are disabled.\n    if (this._handles.length) {\n      this._rootElementTapHighlight = rootElement.style.webkitTapHighlightColor || '';\n      rootElement.style.webkitTapHighlightColor = 'transparent';\n    }\n\n    this._hasStartedDragging = this._hasMoved = false;\n\n    // Avoid multiple subscriptions and memory leaks when multi touch\n    // (isDragging check above isn't enough because of possible temporal and/or dimensional delays)\n    this._removeSubscriptions();\n    this._pointerMoveSubscription = this._dragDropRegistry.pointerMove.subscribe(this._pointerMove);\n    this._pointerUpSubscription = this._dragDropRegistry.pointerUp.subscribe(this._pointerUp);\n    this._scrollSubscription = this._dragDropRegistry\n      .scrolled(this._getShadowRoot())\n      .subscribe(scrollEvent => this._updateOnScroll(scrollEvent));\n\n    if (this._boundaryElement) {\n      this._boundaryRect = getMutableClientRect(this._boundaryElement);\n    }\n\n    // If we have a custom preview we can't know ahead of time how large it'll be so we position\n    // it next to the cursor. The exception is when the consumer has opted into making the preview\n    // the same size as the root element, in which case we do know the size.\n    const previewTemplate = this._previewTemplate;\n    this._pickupPositionInElement = previewTemplate && previewTemplate.template &&\n      !previewTemplate.matchSize ? {x: 0, y: 0} :\n      this._getPointerPositionInElement(referenceElement, event);\n    const pointerPosition = this._pickupPositionOnPage = this._lastKnownPointerPosition =\n        this._getPointerPositionOnPage(event);\n    this._pointerDirectionDelta = {x: 0, y: 0};\n    this._pointerPositionAtLastDirectionChange = {x: pointerPosition.x, y: pointerPosition.y};\n    this._dragStartTime = Date.now();\n    this._dragDropRegistry.startDragging(this, event);\n  }\n\n  /** Cleans up the DOM artifacts that were added to facilitate the element being dragged. */\n  private _cleanupDragArtifacts(event: MouseEvent | TouchEvent) {\n    // Restore the element's visibility and insert it at its old position in the DOM.\n    // It's important that we maintain the position, because moving the element around in the DOM\n    // can throw off `NgFor` which does smart diffing and re-creates elements only when necessary,\n    // while moving the existing elements in all other cases.\n    toggleVisibility(this._rootElement, true, dragImportantProperties);\n    this._anchor.parentNode!.replaceChild(this._rootElement, this._anchor);\n\n    this._destroyPreview();\n    this._destroyPlaceholder();\n    this._boundaryRect = this._previewRect = this._initialTransform = undefined;\n\n    // Re-enter the NgZone since we bound `document` events on the outside.\n    this._ngZone.run(() => {\n      const container = this._dropContainer!;\n      const currentIndex = container.getItemIndex(this);\n      const pointerPosition = this._getPointerPositionOnPage(event);\n      const distance = this._getDragDistance(pointerPosition);\n      const isPointerOverContainer = container._isOverContainer(\n        pointerPosition.x, pointerPosition.y);\n\n      this.ended.next({source: this, distance, dropPoint: pointerPosition});\n      this.dropped.next({\n        item: this,\n        currentIndex,\n        previousIndex: this._initialIndex,\n        container: container,\n        previousContainer: this._initialContainer,\n        isPointerOverContainer,\n        distance,\n        dropPoint: pointerPosition\n      });\n      container.drop(this, currentIndex, this._initialIndex, this._initialContainer,\n        isPointerOverContainer, distance, pointerPosition);\n      this._dropContainer = this._initialContainer;\n    });\n  }\n\n  /**\n   * Updates the item's position in its drop container, or moves it\n   * into a new one, depending on its current drag position.\n   */\n  private _updateActiveDropContainer({x, y}: Point, {x: rawX, y: rawY}: Point) {\n    // Drop container that draggable has been moved into.\n    let newContainer = this._initialContainer._getSiblingContainerFromPosition(this, x, y);\n\n    // If we couldn't find a new container to move the item into, and the item has left its\n    // initial container, check whether the it's over the initial container. This handles the\n    // case where two containers are connected one way and the user tries to undo dragging an\n    // item into a new container.\n    if (!newContainer && this._dropContainer !== this._initialContainer &&\n        this._initialContainer._isOverContainer(x, y)) {\n      newContainer = this._initialContainer;\n    }\n\n    if (newContainer && newContainer !== this._dropContainer) {\n      this._ngZone.run(() => {\n        // Notify the old container that the item has left.\n        this.exited.next({item: this, container: this._dropContainer!});\n        this._dropContainer!.exit(this);\n        // Notify the new container that the item has entered.\n        this._dropContainer = newContainer!;\n        this._dropContainer.enter(this, x, y, newContainer === this._initialContainer &&\n            // If we're re-entering the initial container and sorting is disabled,\n            // put item the into its starting index to begin with.\n            newContainer.sortingDisabled ? this._initialIndex : undefined);\n        this.entered.next({\n          item: this,\n          container: newContainer!,\n          currentIndex: newContainer!.getItemIndex(this)\n        });\n      });\n    }\n\n    // Dragging may have been interrupted as a result of the events above.\n    if (this.isDragging()) {\n      this._dropContainer!._startScrollingIfNecessary(rawX, rawY);\n      this._dropContainer!._sortItem(this, x, y, this._pointerDirectionDelta);\n      this._applyPreviewTransform(\n        x - this._pickupPositionInElement.x, y - this._pickupPositionInElement.y);\n    }\n  }\n\n  /**\n   * Creates the element that will be rendered next to the user's pointer\n   * and will be used as a preview of the element that is being dragged.\n   */\n  private _createPreviewElement(): HTMLElement {\n    const previewConfig = this._previewTemplate;\n    const previewClass = this.previewClass;\n    const previewTemplate = previewConfig ? previewConfig.template : null;\n    let preview: HTMLElement;\n\n    if (previewTemplate && previewConfig) {\n      // Measure the element before we've inserted the preview\n      // since the insertion could throw off the measurement.\n      const rootRect = previewConfig.matchSize ? this._rootElement.getBoundingClientRect() : null;\n      const viewRef = previewConfig.viewContainer.createEmbeddedView(previewTemplate,\n                                                                     previewConfig.context);\n      viewRef.detectChanges();\n      preview = getRootNode(viewRef, this._document);\n      this._previewRef = viewRef;\n      if (previewConfig.matchSize) {\n        matchElementSize(preview, rootRect!);\n      } else {\n        preview.style.transform =\n            getTransform(this._pickupPositionOnPage.x, this._pickupPositionOnPage.y);\n      }\n    } else {\n      const element = this._rootElement;\n      preview = deepCloneNode(element);\n      matchElementSize(preview, element.getBoundingClientRect());\n\n      if (this._initialTransform) {\n        preview.style.transform = this._initialTransform;\n      }\n    }\n\n    extendStyles(preview.style, {\n      // It's important that we disable the pointer events on the preview, because\n      // it can throw off the `document.elementFromPoint` calls in the `CdkDropList`.\n      'pointer-events': 'none',\n      // We have to reset the margin, because it can throw off positioning relative to the viewport.\n      'margin': '0',\n      'position': 'fixed',\n      'top': '0',\n      'left': '0',\n      'z-index': `${this._config.zIndex || 1000}`\n    }, dragImportantProperties);\n\n    toggleNativeDragInteractions(preview, false);\n    preview.classList.add('cdk-drag-preview');\n    preview.setAttribute('dir', this._direction);\n\n    if (previewClass) {\n      if (Array.isArray(previewClass)) {\n        previewClass.forEach(className => preview.classList.add(className));\n      } else {\n        preview.classList.add(previewClass);\n      }\n    }\n\n    return preview;\n  }\n\n  /**\n   * Animates the preview element from its current position to the location of the drop placeholder.\n   * @returns Promise that resolves when the animation completes.\n   */\n  private _animatePreviewToPlaceholder(): Promise<void> {\n    // If the user hasn't moved yet, the transitionend event won't fire.\n    if (!this._hasMoved) {\n      return Promise.resolve();\n    }\n\n    const placeholderRect = this._placeholder.getBoundingClientRect();\n\n    // Apply the class that adds a transition to the preview.\n    this._preview.classList.add('cdk-drag-animating');\n\n    // Move the preview to the placeholder position.\n    this._applyPreviewTransform(placeholderRect.left, placeholderRect.top);\n\n    // If the element doesn't have a `transition`, the `transitionend` event won't fire. Since\n    // we need to trigger a style recalculation in order for the `cdk-drag-animating` class to\n    // apply its style, we take advantage of the available info to figure out whether we need to\n    // bind the event in the first place.\n    const duration = getTransformTransitionDurationInMs(this._preview);\n\n    if (duration === 0) {\n      return Promise.resolve();\n    }\n\n    return this._ngZone.runOutsideAngular(() => {\n      return new Promise(resolve => {\n        const handler = ((event: TransitionEvent) => {\n          if (!event || (_getEventTarget(event) === this._preview &&\n              event.propertyName === 'transform')) {\n            this._preview?.removeEventListener('transitionend', handler);\n            resolve();\n            clearTimeout(timeout);\n          }\n        }) as EventListenerOrEventListenerObject;\n\n        // If a transition is short enough, the browser might not fire the `transitionend` event.\n        // Since we know how long it's supposed to take, add a timeout with a 50% buffer that'll\n        // fire if the transition hasn't completed when it was supposed to.\n        const timeout = setTimeout(handler as Function, duration * 1.5);\n        this._preview.addEventListener('transitionend', handler);\n      });\n    });\n  }\n\n  /** Creates an element that will be shown instead of the current element while dragging. */\n  private _createPlaceholderElement(): HTMLElement {\n    const placeholderConfig = this._placeholderTemplate;\n    const placeholderTemplate = placeholderConfig ? placeholderConfig.template : null;\n    let placeholder: HTMLElement;\n\n    if (placeholderTemplate) {\n      this._placeholderRef = placeholderConfig!.viewContainer.createEmbeddedView(\n        placeholderTemplate,\n        placeholderConfig!.context\n      );\n      this._placeholderRef.detectChanges();\n      placeholder = getRootNode(this._placeholderRef, this._document);\n    } else {\n      placeholder = deepCloneNode(this._rootElement);\n    }\n\n    placeholder.classList.add('cdk-drag-placeholder');\n    return placeholder;\n  }\n\n  /**\n   * Figures out the coordinates at which an element was picked up.\n   * @param referenceElement Element that initiated the dragging.\n   * @param event Event that initiated the dragging.\n   */\n  private _getPointerPositionInElement(referenceElement: HTMLElement,\n                                       event: MouseEvent | TouchEvent): Point {\n    const elementRect = this._rootElement.getBoundingClientRect();\n    const handleElement = referenceElement === this._rootElement ? null : referenceElement;\n    const referenceRect = handleElement ? handleElement.getBoundingClientRect() : elementRect;\n    const point = isTouchEvent(event) ? event.targetTouches[0] : event;\n    const scrollPosition = this._getViewportScrollPosition();\n    const x = point.pageX - referenceRect.left - scrollPosition.left;\n    const y = point.pageY - referenceRect.top - scrollPosition.top;\n\n    return {\n      x: referenceRect.left - elementRect.left + x,\n      y: referenceRect.top - elementRect.top + y\n    };\n  }\n\n  /** Determines the point of the page that was touched by the user. */\n  private _getPointerPositionOnPage(event: MouseEvent | TouchEvent): Point {\n    const scrollPosition = this._getViewportScrollPosition();\n    const point = isTouchEvent(event) ?\n        // `touches` will be empty for start/end events so we have to fall back to `changedTouches`.\n        // Also note that on real devices we're guaranteed for either `touches` or `changedTouches`\n        // to have a value, but Firefox in device emulation mode has a bug where both can be empty\n        // for `touchstart` and `touchend` so we fall back to a dummy object in order to avoid\n        // throwing an error. The value returned here will be incorrect, but since this only\n        // breaks inside a developer tool and the value is only used for secondary information,\n        // we can get away with it. See https://bugzilla.mozilla.org/show_bug.cgi?id=1615824.\n        (event.touches[0] || event.changedTouches[0] || {pageX: 0, pageY: 0}) : event;\n\n    const x = point.pageX - scrollPosition.left;\n    const y = point.pageY - scrollPosition.top;\n\n    // if dragging SVG element, try to convert from the screen coordinate system to the SVG\n    // coordinate system\n    if (this._ownerSVGElement) {\n      const svgMatrix = this._ownerSVGElement.getScreenCTM();\n      if (svgMatrix) {\n        const svgPoint = this._ownerSVGElement.createSVGPoint();\n        svgPoint.x = x;\n        svgPoint.y = y;\n        return svgPoint.matrixTransform(svgMatrix.inverse());\n      }\n    }\n\n    return {x, y};\n  }\n\n\n  /** Gets the pointer position on the page, accounting for any position constraints. */\n  private _getConstrainedPointerPosition(point: Point): Point {\n    const dropContainerLock = this._dropContainer ? this._dropContainer.lockAxis : null;\n    let {x, y} = this.constrainPosition ? this.constrainPosition(point, this) : point;\n\n    if (this.lockAxis === 'x' || dropContainerLock === 'x') {\n      y = this._pickupPositionOnPage.y;\n    } else if (this.lockAxis === 'y' || dropContainerLock === 'y') {\n      x = this._pickupPositionOnPage.x;\n    }\n\n    if (this._boundaryRect) {\n      const {x: pickupX, y: pickupY} = this._pickupPositionInElement;\n      const boundaryRect = this._boundaryRect;\n      const previewRect = this._previewRect!;\n      const minY = boundaryRect.top + pickupY;\n      const maxY = boundaryRect.bottom - (previewRect.height - pickupY);\n      const minX = boundaryRect.left + pickupX;\n      const maxX = boundaryRect.right - (previewRect.width - pickupX);\n\n      x = clamp(x, minX, maxX);\n      y = clamp(y, minY, maxY);\n    }\n\n    return {x, y};\n  }\n\n\n  /** Updates the current drag delta, based on the user's current pointer position on the page. */\n  private _updatePointerDirectionDelta(pointerPositionOnPage: Point) {\n    const {x, y} = pointerPositionOnPage;\n    const delta = this._pointerDirectionDelta;\n    const positionSinceLastChange = this._pointerPositionAtLastDirectionChange;\n\n    // Amount of pixels the user has dragged since the last time the direction changed.\n    const changeX = Math.abs(x - positionSinceLastChange.x);\n    const changeY = Math.abs(y - positionSinceLastChange.y);\n\n    // Because we handle pointer events on a per-pixel basis, we don't want the delta\n    // to change for every pixel, otherwise anything that depends on it can look erratic.\n    // To make the delta more consistent, we track how much the user has moved since the last\n    // delta change and we only update it after it has reached a certain threshold.\n    if (changeX > this._config.pointerDirectionChangeThreshold) {\n      delta.x = x > positionSinceLastChange.x ? 1 : -1;\n      positionSinceLastChange.x = x;\n    }\n\n    if (changeY > this._config.pointerDirectionChangeThreshold) {\n      delta.y = y > positionSinceLastChange.y ? 1 : -1;\n      positionSinceLastChange.y = y;\n    }\n\n    return delta;\n  }\n\n  /** Toggles the native drag interactions, based on how many handles are registered. */\n  private _toggleNativeDragInteractions() {\n    if (!this._rootElement || !this._handles) {\n      return;\n    }\n\n    const shouldEnable = this._handles.length > 0 || !this.isDragging();\n\n    if (shouldEnable !== this._nativeInteractionsEnabled) {\n      this._nativeInteractionsEnabled = shouldEnable;\n      toggleNativeDragInteractions(this._rootElement, shouldEnable);\n    }\n  }\n\n  /** Removes the manually-added event listeners from the root element. */\n  private _removeRootElementListeners(element: HTMLElement) {\n    element.removeEventListener('mousedown', this._pointerDown, activeEventListenerOptions);\n    element.removeEventListener('touchstart', this._pointerDown, passiveEventListenerOptions);\n  }\n\n  /**\n   * Applies a `transform` to the root element, taking into account any existing transforms on it.\n   * @param x New transform value along the X axis.\n   * @param y New transform value along the Y axis.\n   */\n  private _applyRootElementTransform(x: number, y: number) {\n    const transform = getTransform(x, y);\n\n    // Cache the previous transform amount only after the first drag sequence, because\n    // we don't want our own transforms to stack on top of each other.\n    // Should be excluded none because none + translate3d(x, y, x) is invalid css\n    if (this._initialTransform == null) {\n      this._initialTransform = this._rootElement.style.transform\n                               && this._rootElement.style.transform != 'none'\n                               ? this._rootElement.style.transform\n                               : '';\n    }\n\n    // Preserve the previous `transform` value, if there was one. Note that we apply our own\n    // transform before the user's, because things like rotation can affect which direction\n    // the element will be translated towards.\n    this._rootElement.style.transform = combineTransforms(transform, this._initialTransform);\n  }\n\n  /**\n   * Applies a `transform` to the preview, taking into account any existing transforms on it.\n   * @param x New transform value along the X axis.\n   * @param y New transform value along the Y axis.\n   */\n  private _applyPreviewTransform(x: number, y: number) {\n    // Only apply the initial transform if the preview is a clone of the original element, otherwise\n    // it could be completely different and the transform might not make sense anymore.\n    const initialTransform = this._previewTemplate?.template ? undefined : this._initialTransform;\n    const transform = getTransform(x, y);\n    this._preview.style.transform = combineTransforms(transform, initialTransform);\n  }\n\n  /**\n   * Gets the distance that the user has dragged during the current drag sequence.\n   * @param currentPosition Current position of the user's pointer.\n   */\n  private _getDragDistance(currentPosition: Point): Point {\n    const pickupPosition = this._pickupPositionOnPage;\n\n    if (pickupPosition) {\n      return {x: currentPosition.x - pickupPosition.x, y: currentPosition.y - pickupPosition.y};\n    }\n\n    return {x: 0, y: 0};\n  }\n\n  /** Cleans up any cached element dimensions that we don't need after dragging has stopped. */\n  private _cleanupCachedDimensions() {\n    this._boundaryRect = this._previewRect = undefined;\n    this._parentPositions.clear();\n  }\n\n  /**\n   * Checks whether the element is still inside its boundary after the viewport has been resized.\n   * If not, the position is adjusted so that the element fits again.\n   */\n  private _containInsideBoundaryOnResize() {\n    let {x, y} = this._passiveTransform;\n\n    if ((x === 0 && y === 0) || this.isDragging() || !this._boundaryElement) {\n      return;\n    }\n\n    const boundaryRect = this._boundaryElement.getBoundingClientRect();\n    const elementRect = this._rootElement.getBoundingClientRect();\n\n    // It's possible that the element got hidden away after dragging (e.g. by switching to a\n    // different tab). Don't do anything in this case so we don't clear the user's position.\n    if ((boundaryRect.width === 0 && boundaryRect.height === 0) ||\n        (elementRect.width === 0 && elementRect.height === 0)) {\n      return;\n    }\n\n    const leftOverflow = boundaryRect.left - elementRect.left;\n    const rightOverflow = elementRect.right - boundaryRect.right;\n    const topOverflow = boundaryRect.top - elementRect.top;\n    const bottomOverflow = elementRect.bottom - boundaryRect.bottom;\n\n    // If the element has become wider than the boundary, we can't\n    // do much to make it fit so we just anchor it to the left.\n    if (boundaryRect.width > elementRect.width) {\n      if (leftOverflow > 0) {\n        x += leftOverflow;\n      }\n\n      if (rightOverflow > 0) {\n        x -= rightOverflow;\n      }\n    } else {\n      x = 0;\n    }\n\n    // If the element has become taller than the boundary, we can't\n    // do much to make it fit so we just anchor it to the top.\n    if (boundaryRect.height > elementRect.height) {\n      if (topOverflow > 0) {\n        y += topOverflow;\n      }\n\n      if (bottomOverflow > 0) {\n        y -= bottomOverflow;\n      }\n    } else {\n      y = 0;\n    }\n\n    if (x !== this._passiveTransform.x || y !== this._passiveTransform.y) {\n      this.setFreeDragPosition({y, x});\n    }\n  }\n\n  /** Gets the drag start delay, based on the event type. */\n  private _getDragStartDelay(event: MouseEvent | TouchEvent): number {\n    const value = this.dragStartDelay;\n\n    if (typeof value === 'number') {\n      return value;\n    } else if (isTouchEvent(event)) {\n      return value.touch;\n    }\n\n    return value ? value.mouse : 0;\n  }\n\n  /** Updates the internal state of the draggable element when scrolling has occurred. */\n  private _updateOnScroll(event: Event) {\n    const scrollDifference = this._parentPositions.handleScroll(event);\n\n    if (scrollDifference) {\n      const target = _getEventTarget<HTMLElement|Document>(event)!;\n\n      // ClientRect dimensions are based on the scroll position of the page and its parent node so\n      // we have to update the cached boundary ClientRect if the user has scrolled. Check for\n      // the `document` specifically since IE doesn't support `contains` on it.\n      if (this._boundaryRect && (target === this._document ||\n          (target !== this._boundaryElement && target.contains(this._boundaryElement)))) {\n        adjustClientRect(this._boundaryRect, scrollDifference.top, scrollDifference.left);\n      }\n\n      this._pickupPositionOnPage.x += scrollDifference.left;\n      this._pickupPositionOnPage.y += scrollDifference.top;\n\n      // If we're in free drag mode, we have to update the active transform, because\n      // it isn't relative to the viewport like the preview inside a drop list.\n      if (!this._dropContainer) {\n        this._activeTransform.x -= scrollDifference.left;\n        this._activeTransform.y -= scrollDifference.top;\n        this._applyRootElementTransform(this._activeTransform.x, this._activeTransform.y);\n      }\n    }\n  }\n\n  /** Gets the scroll position of the viewport. */\n  private _getViewportScrollPosition() {\n    const cachedPosition = this._parentPositions.positions.get(this._document);\n    return cachedPosition ? cachedPosition.scrollPosition :\n        this._viewportRuler.getViewportScrollPosition();\n  }\n\n  /**\n   * Lazily resolves and returns the shadow root of the element. We do this in a function, rather\n   * than saving it in property directly on init, because we want to resolve it as late as possible\n   * in order to ensure that the element has been moved into the shadow DOM. Doing it inside the\n   * constructor might be too early if the element is inside of something like `ngFor` or `ngIf`.\n   */\n  private _getShadowRoot(): ShadowRoot | null {\n    if (this._cachedShadowRoot === undefined) {\n      this._cachedShadowRoot = _getShadowRoot(this._rootElement);\n    }\n\n    return this._cachedShadowRoot;\n  }\n\n  /** Gets the element into which the drag preview should be inserted. */\n  private _getPreviewInsertionPoint(initialParent: HTMLElement,\n                                    shadowRoot: ShadowRoot | null): HTMLElement {\n    const previewContainer = this._previewContainer || 'global';\n\n    if (previewContainer === 'parent') {\n      return initialParent;\n    }\n\n    if (previewContainer === 'global') {\n      const documentRef = this._document;\n\n      // We can't use the body if the user is in fullscreen mode,\n      // because the preview will render under the fullscreen element.\n      // TODO(crisbeto): dedupe this with the `FullscreenOverlayContainer` eventually.\n      return shadowRoot ||\n             documentRef.fullscreenElement ||\n             (documentRef as any).webkitFullscreenElement ||\n             (documentRef as any).mozFullScreenElement ||\n             (documentRef as any).msFullscreenElement ||\n             documentRef.body;\n    }\n\n    return coerceElement(previewContainer);\n  }\n}\n\n/**\n * Gets a 3d `transform` that can be applied to an element.\n * @param x Desired position of the element along the X axis.\n * @param y Desired position of the element along the Y axis.\n */\nfunction getTransform(x: number, y: number): string {\n  // Round the transforms since some browsers will\n  // blur the elements for sub-pixel transforms.\n  return `translate3d(${Math.round(x)}px, ${Math.round(y)}px, 0)`;\n}\n\n/** Clamps a value between a minimum and a maximum. */\nfunction clamp(value: number, min: number, max: number) {\n  return Math.max(min, Math.min(max, value));\n}\n\n/**\n * Helper to remove a node from the DOM and to do all the necessary null checks.\n * @param node Node to be removed.\n */\nfunction removeNode(node: Node | null) {\n  if (node && node.parentNode) {\n    node.parentNode.removeChild(node);\n  }\n}\n\n/** Determines whether an event is a touch event. */\nfunction isTouchEvent(event: MouseEvent | TouchEvent): event is TouchEvent {\n  // This function is called for every pixel that the user has dragged so we need it to be\n  // as fast as possible. Since we only bind mouse events and touch events, we can assume\n  // that if the event's name starts with `t`, it's a touch event.\n  return event.type[0] === 't';\n}\n\n/**\n * Gets the root HTML element of an embedded view.\n * If the root is not an HTML element it gets wrapped in one.\n */\nfunction getRootNode(viewRef: EmbeddedViewRef<any>, _document: Document): HTMLElement {\n  const rootNodes: Node[] = viewRef.rootNodes;\n\n  if (rootNodes.length === 1 && rootNodes[0].nodeType === _document.ELEMENT_NODE) {\n    return rootNodes[0] as HTMLElement;\n  }\n\n  const wrapper = _document.createElement('div');\n  rootNodes.forEach(node => wrapper.appendChild(node));\n  return wrapper;\n}\n\n/**\n * Matches the target element's size to the source's size.\n * @param target Element that needs to be resized.\n * @param sourceRect Dimensions of the source element.\n */\nfunction matchElementSize(target: HTMLElement, sourceRect: ClientRect): void {\n  target.style.width = `${sourceRect.width}px`;\n  target.style.height = `${sourceRect.height}px`;\n  target.style.transform = getTransform(sourceRect.left, sourceRect.top);\n}\n"]}
Note: See TracBrowser for help on using the repository browser.