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 | */
|
---|
8 | import { normalizePassiveListenerOptions, _getEventTarget, _getShadowRoot, } from '@angular/cdk/platform';
|
---|
9 | import { coerceBooleanProperty, coerceElement } from '@angular/cdk/coercion';
|
---|
10 | import { isFakeMousedownFromScreenReader, isFakeTouchstartFromScreenReader, } from '@angular/cdk/a11y';
|
---|
11 | import { Subscription, Subject } from 'rxjs';
|
---|
12 | import { combineTransforms, extendStyles, toggleNativeDragInteractions, toggleVisibility, } from './drag-styling';
|
---|
13 | import { getTransformTransitionDurationInMs } from './transition-duration';
|
---|
14 | import { getMutableClientRect, adjustClientRect } from './client-rect';
|
---|
15 | import { ParentPositionTracker } from './parent-position-tracker';
|
---|
16 | import { deepCloneNode } from './clone-node';
|
---|
17 | /** Options that can be used to bind a passive event listener. */
|
---|
18 | const passiveEventListenerOptions = normalizePassiveListenerOptions({ passive: true });
|
---|
19 | /** Options that can be used to bind an active event listener. */
|
---|
20 | const 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 | */
|
---|
27 | const MOUSE_EVENT_IGNORE_TIME = 800;
|
---|
28 | /** Inline styles to be set as `!important` while dragging. */
|
---|
29 | const 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 | */
|
---|
36 | export 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 | */
|
---|
1070 | function 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. */
|
---|
1076 | function 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 | */
|
---|
1083 | function removeNode(node) {
|
---|
1084 | if (node && node.parentNode) {
|
---|
1085 | node.parentNode.removeChild(node);
|
---|
1086 | }
|
---|
1087 | }
|
---|
1088 | /** Determines whether an event is a touch event. */
|
---|
1089 | function 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 | */
|
---|
1099 | function 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 | */
|
---|
1113 | function 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":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\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"]} |
---|