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 { coerceElement } from '@angular/cdk/coercion';
|
---|
9 | import { _getShadowRoot } from '@angular/cdk/platform';
|
---|
10 | import { Subject, Subscription, interval, animationFrameScheduler } from 'rxjs';
|
---|
11 | import { takeUntil } from 'rxjs/operators';
|
---|
12 | import { moveItemInArray } from './drag-utils';
|
---|
13 | import { isPointerNearClientRect, adjustClientRect, getMutableClientRect, isInsideClientRect, } from './client-rect';
|
---|
14 | import { ParentPositionTracker } from './parent-position-tracker';
|
---|
15 | import { combineTransforms } from './drag-styling';
|
---|
16 | /**
|
---|
17 | * Proximity, as a ratio to width/height, at which a
|
---|
18 | * dragged item will affect the drop container.
|
---|
19 | */
|
---|
20 | const DROP_PROXIMITY_THRESHOLD = 0.05;
|
---|
21 | /**
|
---|
22 | * Proximity, as a ratio to width/height at which to start auto-scrolling the drop list or the
|
---|
23 | * viewport. The value comes from trying it out manually until it feels right.
|
---|
24 | */
|
---|
25 | const SCROLL_PROXIMITY_THRESHOLD = 0.05;
|
---|
26 | /**
|
---|
27 | * Reference to a drop list. Used to manipulate or dispose of the container.
|
---|
28 | */
|
---|
29 | export class DropListRef {
|
---|
30 | constructor(element, _dragDropRegistry, _document, _ngZone, _viewportRuler) {
|
---|
31 | this._dragDropRegistry = _dragDropRegistry;
|
---|
32 | this._ngZone = _ngZone;
|
---|
33 | this._viewportRuler = _viewportRuler;
|
---|
34 | /** Whether starting a dragging sequence from this container is disabled. */
|
---|
35 | this.disabled = false;
|
---|
36 | /** Whether sorting items within the list is disabled. */
|
---|
37 | this.sortingDisabled = false;
|
---|
38 | /**
|
---|
39 | * Whether auto-scrolling the view when the user
|
---|
40 | * moves their pointer close to the edges is disabled.
|
---|
41 | */
|
---|
42 | this.autoScrollDisabled = false;
|
---|
43 | /** Number of pixels to scroll for each frame when auto-scrolling an element. */
|
---|
44 | this.autoScrollStep = 2;
|
---|
45 | /**
|
---|
46 | * Function that is used to determine whether an item
|
---|
47 | * is allowed to be moved into a drop container.
|
---|
48 | */
|
---|
49 | this.enterPredicate = () => true;
|
---|
50 | /** Functions that is used to determine whether an item can be sorted into a particular index. */
|
---|
51 | this.sortPredicate = () => true;
|
---|
52 | /** Emits right before dragging has started. */
|
---|
53 | this.beforeStarted = new Subject();
|
---|
54 | /**
|
---|
55 | * Emits when the user has moved a new drag item into this container.
|
---|
56 | */
|
---|
57 | this.entered = new Subject();
|
---|
58 | /**
|
---|
59 | * Emits when the user removes an item from the container
|
---|
60 | * by dragging it into another container.
|
---|
61 | */
|
---|
62 | this.exited = new Subject();
|
---|
63 | /** Emits when the user drops an item inside the container. */
|
---|
64 | this.dropped = new Subject();
|
---|
65 | /** Emits as the user is swapping items while actively dragging. */
|
---|
66 | this.sorted = new Subject();
|
---|
67 | /** Whether an item in the list is being dragged. */
|
---|
68 | this._isDragging = false;
|
---|
69 | /** Cache of the dimensions of all the items inside the container. */
|
---|
70 | this._itemPositions = [];
|
---|
71 | /**
|
---|
72 | * Keeps track of the item that was last swapped with the dragged item, as well as what direction
|
---|
73 | * the pointer was moving in when the swap occured and whether the user's pointer continued to
|
---|
74 | * overlap with the swapped item after the swapping occurred.
|
---|
75 | */
|
---|
76 | this._previousSwap = { drag: null, delta: 0, overlaps: false };
|
---|
77 | /** Draggable items in the container. */
|
---|
78 | this._draggables = [];
|
---|
79 | /** Drop lists that are connected to the current one. */
|
---|
80 | this._siblings = [];
|
---|
81 | /** Direction in which the list is oriented. */
|
---|
82 | this._orientation = 'vertical';
|
---|
83 | /** Connected siblings that currently have a dragged item. */
|
---|
84 | this._activeSiblings = new Set();
|
---|
85 | /** Layout direction of the drop list. */
|
---|
86 | this._direction = 'ltr';
|
---|
87 | /** Subscription to the window being scrolled. */
|
---|
88 | this._viewportScrollSubscription = Subscription.EMPTY;
|
---|
89 | /** Vertical direction in which the list is currently scrolling. */
|
---|
90 | this._verticalScrollDirection = 0 /* NONE */;
|
---|
91 | /** Horizontal direction in which the list is currently scrolling. */
|
---|
92 | this._horizontalScrollDirection = 0 /* NONE */;
|
---|
93 | /** Used to signal to the current auto-scroll sequence when to stop. */
|
---|
94 | this._stopScrollTimers = new Subject();
|
---|
95 | /** Shadow root of the current element. Necessary for `elementFromPoint` to resolve correctly. */
|
---|
96 | this._cachedShadowRoot = null;
|
---|
97 | /** Starts the interval that'll auto-scroll the element. */
|
---|
98 | this._startScrollInterval = () => {
|
---|
99 | this._stopScrolling();
|
---|
100 | interval(0, animationFrameScheduler)
|
---|
101 | .pipe(takeUntil(this._stopScrollTimers))
|
---|
102 | .subscribe(() => {
|
---|
103 | const node = this._scrollNode;
|
---|
104 | const scrollStep = this.autoScrollStep;
|
---|
105 | if (this._verticalScrollDirection === 1 /* UP */) {
|
---|
106 | incrementVerticalScroll(node, -scrollStep);
|
---|
107 | }
|
---|
108 | else if (this._verticalScrollDirection === 2 /* DOWN */) {
|
---|
109 | incrementVerticalScroll(node, scrollStep);
|
---|
110 | }
|
---|
111 | if (this._horizontalScrollDirection === 1 /* LEFT */) {
|
---|
112 | incrementHorizontalScroll(node, -scrollStep);
|
---|
113 | }
|
---|
114 | else if (this._horizontalScrollDirection === 2 /* RIGHT */) {
|
---|
115 | incrementHorizontalScroll(node, scrollStep);
|
---|
116 | }
|
---|
117 | });
|
---|
118 | };
|
---|
119 | this.element = coerceElement(element);
|
---|
120 | this._document = _document;
|
---|
121 | this.withScrollableParents([this.element]);
|
---|
122 | _dragDropRegistry.registerDropContainer(this);
|
---|
123 | this._parentPositions = new ParentPositionTracker(_document, _viewportRuler);
|
---|
124 | }
|
---|
125 | /** Removes the drop list functionality from the DOM element. */
|
---|
126 | dispose() {
|
---|
127 | this._stopScrolling();
|
---|
128 | this._stopScrollTimers.complete();
|
---|
129 | this._viewportScrollSubscription.unsubscribe();
|
---|
130 | this.beforeStarted.complete();
|
---|
131 | this.entered.complete();
|
---|
132 | this.exited.complete();
|
---|
133 | this.dropped.complete();
|
---|
134 | this.sorted.complete();
|
---|
135 | this._activeSiblings.clear();
|
---|
136 | this._scrollNode = null;
|
---|
137 | this._parentPositions.clear();
|
---|
138 | this._dragDropRegistry.removeDropContainer(this);
|
---|
139 | }
|
---|
140 | /** Whether an item from this list is currently being dragged. */
|
---|
141 | isDragging() {
|
---|
142 | return this._isDragging;
|
---|
143 | }
|
---|
144 | /** Starts dragging an item. */
|
---|
145 | start() {
|
---|
146 | this._draggingStarted();
|
---|
147 | this._notifyReceivingSiblings();
|
---|
148 | }
|
---|
149 | /**
|
---|
150 | * Emits an event to indicate that the user moved an item into the container.
|
---|
151 | * @param item Item that was moved into the container.
|
---|
152 | * @param pointerX Position of the item along the X axis.
|
---|
153 | * @param pointerY Position of the item along the Y axis.
|
---|
154 | * @param index Index at which the item entered. If omitted, the container will try to figure it
|
---|
155 | * out automatically.
|
---|
156 | */
|
---|
157 | enter(item, pointerX, pointerY, index) {
|
---|
158 | this._draggingStarted();
|
---|
159 | // If sorting is disabled, we want the item to return to its starting
|
---|
160 | // position if the user is returning it to its initial container.
|
---|
161 | let newIndex;
|
---|
162 | if (index == null) {
|
---|
163 | newIndex = this.sortingDisabled ? this._draggables.indexOf(item) : -1;
|
---|
164 | if (newIndex === -1) {
|
---|
165 | // We use the coordinates of where the item entered the drop
|
---|
166 | // zone to figure out at which index it should be inserted.
|
---|
167 | newIndex = this._getItemIndexFromPointerPosition(item, pointerX, pointerY);
|
---|
168 | }
|
---|
169 | }
|
---|
170 | else {
|
---|
171 | newIndex = index;
|
---|
172 | }
|
---|
173 | const activeDraggables = this._activeDraggables;
|
---|
174 | const currentIndex = activeDraggables.indexOf(item);
|
---|
175 | const placeholder = item.getPlaceholderElement();
|
---|
176 | let newPositionReference = activeDraggables[newIndex];
|
---|
177 | // If the item at the new position is the same as the item that is being dragged,
|
---|
178 | // it means that we're trying to restore the item to its initial position. In this
|
---|
179 | // case we should use the next item from the list as the reference.
|
---|
180 | if (newPositionReference === item) {
|
---|
181 | newPositionReference = activeDraggables[newIndex + 1];
|
---|
182 | }
|
---|
183 | // Since the item may be in the `activeDraggables` already (e.g. if the user dragged it
|
---|
184 | // into another container and back again), we have to ensure that it isn't duplicated.
|
---|
185 | if (currentIndex > -1) {
|
---|
186 | activeDraggables.splice(currentIndex, 1);
|
---|
187 | }
|
---|
188 | // Don't use items that are being dragged as a reference, because
|
---|
189 | // their element has been moved down to the bottom of the body.
|
---|
190 | if (newPositionReference && !this._dragDropRegistry.isDragging(newPositionReference)) {
|
---|
191 | const element = newPositionReference.getRootElement();
|
---|
192 | element.parentElement.insertBefore(placeholder, element);
|
---|
193 | activeDraggables.splice(newIndex, 0, item);
|
---|
194 | }
|
---|
195 | else if (this._shouldEnterAsFirstChild(pointerX, pointerY)) {
|
---|
196 | const reference = activeDraggables[0].getRootElement();
|
---|
197 | reference.parentNode.insertBefore(placeholder, reference);
|
---|
198 | activeDraggables.unshift(item);
|
---|
199 | }
|
---|
200 | else {
|
---|
201 | coerceElement(this.element).appendChild(placeholder);
|
---|
202 | activeDraggables.push(item);
|
---|
203 | }
|
---|
204 | // The transform needs to be cleared so it doesn't throw off the measurements.
|
---|
205 | placeholder.style.transform = '';
|
---|
206 | // Note that the positions were already cached when we called `start` above,
|
---|
207 | // but we need to refresh them since the amount of items has changed and also parent rects.
|
---|
208 | this._cacheItemPositions();
|
---|
209 | this._cacheParentPositions();
|
---|
210 | // Notify siblings at the end so that the item has been inserted into the `activeDraggables`.
|
---|
211 | this._notifyReceivingSiblings();
|
---|
212 | this.entered.next({ item, container: this, currentIndex: this.getItemIndex(item) });
|
---|
213 | }
|
---|
214 | /**
|
---|
215 | * Removes an item from the container after it was dragged into another container by the user.
|
---|
216 | * @param item Item that was dragged out.
|
---|
217 | */
|
---|
218 | exit(item) {
|
---|
219 | this._reset();
|
---|
220 | this.exited.next({ item, container: this });
|
---|
221 | }
|
---|
222 | /**
|
---|
223 | * Drops an item into this container.
|
---|
224 | * @param item Item being dropped into the container.
|
---|
225 | * @param currentIndex Index at which the item should be inserted.
|
---|
226 | * @param previousIndex Index of the item when dragging started.
|
---|
227 | * @param previousContainer Container from which the item got dragged in.
|
---|
228 | * @param isPointerOverContainer Whether the user's pointer was over the
|
---|
229 | * container when the item was dropped.
|
---|
230 | * @param distance Distance the user has dragged since the start of the dragging sequence.
|
---|
231 | */
|
---|
232 | drop(item, currentIndex, previousIndex, previousContainer, isPointerOverContainer, distance, dropPoint) {
|
---|
233 | this._reset();
|
---|
234 | this.dropped.next({
|
---|
235 | item,
|
---|
236 | currentIndex,
|
---|
237 | previousIndex,
|
---|
238 | container: this,
|
---|
239 | previousContainer,
|
---|
240 | isPointerOverContainer,
|
---|
241 | distance,
|
---|
242 | dropPoint
|
---|
243 | });
|
---|
244 | }
|
---|
245 | /**
|
---|
246 | * Sets the draggable items that are a part of this list.
|
---|
247 | * @param items Items that are a part of this list.
|
---|
248 | */
|
---|
249 | withItems(items) {
|
---|
250 | const previousItems = this._draggables;
|
---|
251 | this._draggables = items;
|
---|
252 | items.forEach(item => item._withDropContainer(this));
|
---|
253 | if (this.isDragging()) {
|
---|
254 | const draggedItems = previousItems.filter(item => item.isDragging());
|
---|
255 | // If all of the items being dragged were removed
|
---|
256 | // from the list, abort the current drag sequence.
|
---|
257 | if (draggedItems.every(item => items.indexOf(item) === -1)) {
|
---|
258 | this._reset();
|
---|
259 | }
|
---|
260 | else {
|
---|
261 | this._cacheItems();
|
---|
262 | }
|
---|
263 | }
|
---|
264 | return this;
|
---|
265 | }
|
---|
266 | /** Sets the layout direction of the drop list. */
|
---|
267 | withDirection(direction) {
|
---|
268 | this._direction = direction;
|
---|
269 | return this;
|
---|
270 | }
|
---|
271 | /**
|
---|
272 | * Sets the containers that are connected to this one. When two or more containers are
|
---|
273 | * connected, the user will be allowed to transfer items between them.
|
---|
274 | * @param connectedTo Other containers that the current containers should be connected to.
|
---|
275 | */
|
---|
276 | connectedTo(connectedTo) {
|
---|
277 | this._siblings = connectedTo.slice();
|
---|
278 | return this;
|
---|
279 | }
|
---|
280 | /**
|
---|
281 | * Sets the orientation of the container.
|
---|
282 | * @param orientation New orientation for the container.
|
---|
283 | */
|
---|
284 | withOrientation(orientation) {
|
---|
285 | this._orientation = orientation;
|
---|
286 | return this;
|
---|
287 | }
|
---|
288 | /**
|
---|
289 | * Sets which parent elements are can be scrolled while the user is dragging.
|
---|
290 | * @param elements Elements that can be scrolled.
|
---|
291 | */
|
---|
292 | withScrollableParents(elements) {
|
---|
293 | const element = coerceElement(this.element);
|
---|
294 | // We always allow the current element to be scrollable
|
---|
295 | // so we need to ensure that it's in the array.
|
---|
296 | this._scrollableElements =
|
---|
297 | elements.indexOf(element) === -1 ? [element, ...elements] : elements.slice();
|
---|
298 | return this;
|
---|
299 | }
|
---|
300 | /** Gets the scrollable parents that are registered with this drop container. */
|
---|
301 | getScrollableParents() {
|
---|
302 | return this._scrollableElements;
|
---|
303 | }
|
---|
304 | /**
|
---|
305 | * Figures out the index of an item in the container.
|
---|
306 | * @param item Item whose index should be determined.
|
---|
307 | */
|
---|
308 | getItemIndex(item) {
|
---|
309 | if (!this._isDragging) {
|
---|
310 | return this._draggables.indexOf(item);
|
---|
311 | }
|
---|
312 | // Items are sorted always by top/left in the cache, however they flow differently in RTL.
|
---|
313 | // The rest of the logic still stands no matter what orientation we're in, however
|
---|
314 | // we need to invert the array when determining the index.
|
---|
315 | const items = this._orientation === 'horizontal' && this._direction === 'rtl' ?
|
---|
316 | this._itemPositions.slice().reverse() : this._itemPositions;
|
---|
317 | return findIndex(items, currentItem => currentItem.drag === item);
|
---|
318 | }
|
---|
319 | /**
|
---|
320 | * Whether the list is able to receive the item that
|
---|
321 | * is currently being dragged inside a connected drop list.
|
---|
322 | */
|
---|
323 | isReceiving() {
|
---|
324 | return this._activeSiblings.size > 0;
|
---|
325 | }
|
---|
326 | /**
|
---|
327 | * Sorts an item inside the container based on its position.
|
---|
328 | * @param item Item to be sorted.
|
---|
329 | * @param pointerX Position of the item along the X axis.
|
---|
330 | * @param pointerY Position of the item along the Y axis.
|
---|
331 | * @param pointerDelta Direction in which the pointer is moving along each axis.
|
---|
332 | */
|
---|
333 | _sortItem(item, pointerX, pointerY, pointerDelta) {
|
---|
334 | // Don't sort the item if sorting is disabled or it's out of range.
|
---|
335 | if (this.sortingDisabled || !this._clientRect ||
|
---|
336 | !isPointerNearClientRect(this._clientRect, DROP_PROXIMITY_THRESHOLD, pointerX, pointerY)) {
|
---|
337 | return;
|
---|
338 | }
|
---|
339 | const siblings = this._itemPositions;
|
---|
340 | const newIndex = this._getItemIndexFromPointerPosition(item, pointerX, pointerY, pointerDelta);
|
---|
341 | if (newIndex === -1 && siblings.length > 0) {
|
---|
342 | return;
|
---|
343 | }
|
---|
344 | const isHorizontal = this._orientation === 'horizontal';
|
---|
345 | const currentIndex = findIndex(siblings, currentItem => currentItem.drag === item);
|
---|
346 | const siblingAtNewPosition = siblings[newIndex];
|
---|
347 | const currentPosition = siblings[currentIndex].clientRect;
|
---|
348 | const newPosition = siblingAtNewPosition.clientRect;
|
---|
349 | const delta = currentIndex > newIndex ? 1 : -1;
|
---|
350 | // How many pixels the item's placeholder should be offset.
|
---|
351 | const itemOffset = this._getItemOffsetPx(currentPosition, newPosition, delta);
|
---|
352 | // How many pixels all the other items should be offset.
|
---|
353 | const siblingOffset = this._getSiblingOffsetPx(currentIndex, siblings, delta);
|
---|
354 | // Save the previous order of the items before moving the item to its new index.
|
---|
355 | // We use this to check whether an item has been moved as a result of the sorting.
|
---|
356 | const oldOrder = siblings.slice();
|
---|
357 | // Shuffle the array in place.
|
---|
358 | moveItemInArray(siblings, currentIndex, newIndex);
|
---|
359 | this.sorted.next({
|
---|
360 | previousIndex: currentIndex,
|
---|
361 | currentIndex: newIndex,
|
---|
362 | container: this,
|
---|
363 | item
|
---|
364 | });
|
---|
365 | siblings.forEach((sibling, index) => {
|
---|
366 | // Don't do anything if the position hasn't changed.
|
---|
367 | if (oldOrder[index] === sibling) {
|
---|
368 | return;
|
---|
369 | }
|
---|
370 | const isDraggedItem = sibling.drag === item;
|
---|
371 | const offset = isDraggedItem ? itemOffset : siblingOffset;
|
---|
372 | const elementToOffset = isDraggedItem ? item.getPlaceholderElement() :
|
---|
373 | sibling.drag.getRootElement();
|
---|
374 | // Update the offset to reflect the new position.
|
---|
375 | sibling.offset += offset;
|
---|
376 | // Since we're moving the items with a `transform`, we need to adjust their cached
|
---|
377 | // client rects to reflect their new position, as well as swap their positions in the cache.
|
---|
378 | // Note that we shouldn't use `getBoundingClientRect` here to update the cache, because the
|
---|
379 | // elements may be mid-animation which will give us a wrong result.
|
---|
380 | if (isHorizontal) {
|
---|
381 | // Round the transforms since some browsers will
|
---|
382 | // blur the elements, for sub-pixel transforms.
|
---|
383 | elementToOffset.style.transform = combineTransforms(`translate3d(${Math.round(sibling.offset)}px, 0, 0)`, sibling.initialTransform);
|
---|
384 | adjustClientRect(sibling.clientRect, 0, offset);
|
---|
385 | }
|
---|
386 | else {
|
---|
387 | elementToOffset.style.transform = combineTransforms(`translate3d(0, ${Math.round(sibling.offset)}px, 0)`, sibling.initialTransform);
|
---|
388 | adjustClientRect(sibling.clientRect, offset, 0);
|
---|
389 | }
|
---|
390 | });
|
---|
391 | // Note that it's important that we do this after the client rects have been adjusted.
|
---|
392 | this._previousSwap.overlaps = isInsideClientRect(newPosition, pointerX, pointerY);
|
---|
393 | this._previousSwap.drag = siblingAtNewPosition.drag;
|
---|
394 | this._previousSwap.delta = isHorizontal ? pointerDelta.x : pointerDelta.y;
|
---|
395 | }
|
---|
396 | /**
|
---|
397 | * Checks whether the user's pointer is close to the edges of either the
|
---|
398 | * viewport or the drop list and starts the auto-scroll sequence.
|
---|
399 | * @param pointerX User's pointer position along the x axis.
|
---|
400 | * @param pointerY User's pointer position along the y axis.
|
---|
401 | */
|
---|
402 | _startScrollingIfNecessary(pointerX, pointerY) {
|
---|
403 | if (this.autoScrollDisabled) {
|
---|
404 | return;
|
---|
405 | }
|
---|
406 | let scrollNode;
|
---|
407 | let verticalScrollDirection = 0 /* NONE */;
|
---|
408 | let horizontalScrollDirection = 0 /* NONE */;
|
---|
409 | // Check whether we should start scrolling any of the parent containers.
|
---|
410 | this._parentPositions.positions.forEach((position, element) => {
|
---|
411 | // We have special handling for the `document` below. Also this would be
|
---|
412 | // nicer with a for...of loop, but it requires changing a compiler flag.
|
---|
413 | if (element === this._document || !position.clientRect || scrollNode) {
|
---|
414 | return;
|
---|
415 | }
|
---|
416 | if (isPointerNearClientRect(position.clientRect, DROP_PROXIMITY_THRESHOLD, pointerX, pointerY)) {
|
---|
417 | [verticalScrollDirection, horizontalScrollDirection] = getElementScrollDirections(element, position.clientRect, pointerX, pointerY);
|
---|
418 | if (verticalScrollDirection || horizontalScrollDirection) {
|
---|
419 | scrollNode = element;
|
---|
420 | }
|
---|
421 | }
|
---|
422 | });
|
---|
423 | // Otherwise check if we can start scrolling the viewport.
|
---|
424 | if (!verticalScrollDirection && !horizontalScrollDirection) {
|
---|
425 | const { width, height } = this._viewportRuler.getViewportSize();
|
---|
426 | const clientRect = { width, height, top: 0, right: width, bottom: height, left: 0 };
|
---|
427 | verticalScrollDirection = getVerticalScrollDirection(clientRect, pointerY);
|
---|
428 | horizontalScrollDirection = getHorizontalScrollDirection(clientRect, pointerX);
|
---|
429 | scrollNode = window;
|
---|
430 | }
|
---|
431 | if (scrollNode && (verticalScrollDirection !== this._verticalScrollDirection ||
|
---|
432 | horizontalScrollDirection !== this._horizontalScrollDirection ||
|
---|
433 | scrollNode !== this._scrollNode)) {
|
---|
434 | this._verticalScrollDirection = verticalScrollDirection;
|
---|
435 | this._horizontalScrollDirection = horizontalScrollDirection;
|
---|
436 | this._scrollNode = scrollNode;
|
---|
437 | if ((verticalScrollDirection || horizontalScrollDirection) && scrollNode) {
|
---|
438 | this._ngZone.runOutsideAngular(this._startScrollInterval);
|
---|
439 | }
|
---|
440 | else {
|
---|
441 | this._stopScrolling();
|
---|
442 | }
|
---|
443 | }
|
---|
444 | }
|
---|
445 | /** Stops any currently-running auto-scroll sequences. */
|
---|
446 | _stopScrolling() {
|
---|
447 | this._stopScrollTimers.next();
|
---|
448 | }
|
---|
449 | /** Starts the dragging sequence within the list. */
|
---|
450 | _draggingStarted() {
|
---|
451 | const styles = coerceElement(this.element).style;
|
---|
452 | this.beforeStarted.next();
|
---|
453 | this._isDragging = true;
|
---|
454 | // We need to disable scroll snapping while the user is dragging, because it breaks automatic
|
---|
455 | // scrolling. The browser seems to round the value based on the snapping points which means
|
---|
456 | // that we can't increment/decrement the scroll position.
|
---|
457 | this._initialScrollSnap = styles.msScrollSnapType || styles.scrollSnapType || '';
|
---|
458 | styles.scrollSnapType = styles.msScrollSnapType = 'none';
|
---|
459 | this._cacheItems();
|
---|
460 | this._viewportScrollSubscription.unsubscribe();
|
---|
461 | this._listenToScrollEvents();
|
---|
462 | }
|
---|
463 | /** Caches the positions of the configured scrollable parents. */
|
---|
464 | _cacheParentPositions() {
|
---|
465 | const element = coerceElement(this.element);
|
---|
466 | this._parentPositions.cache(this._scrollableElements);
|
---|
467 | // The list element is always in the `scrollableElements`
|
---|
468 | // so we can take advantage of the cached `ClientRect`.
|
---|
469 | this._clientRect = this._parentPositions.positions.get(element).clientRect;
|
---|
470 | }
|
---|
471 | /** Refreshes the position cache of the items and sibling containers. */
|
---|
472 | _cacheItemPositions() {
|
---|
473 | const isHorizontal = this._orientation === 'horizontal';
|
---|
474 | this._itemPositions = this._activeDraggables.map(drag => {
|
---|
475 | const elementToMeasure = drag.getVisibleElement();
|
---|
476 | return {
|
---|
477 | drag,
|
---|
478 | offset: 0,
|
---|
479 | initialTransform: elementToMeasure.style.transform || '',
|
---|
480 | clientRect: getMutableClientRect(elementToMeasure),
|
---|
481 | };
|
---|
482 | }).sort((a, b) => {
|
---|
483 | return isHorizontal ? a.clientRect.left - b.clientRect.left :
|
---|
484 | a.clientRect.top - b.clientRect.top;
|
---|
485 | });
|
---|
486 | }
|
---|
487 | /** Resets the container to its initial state. */
|
---|
488 | _reset() {
|
---|
489 | this._isDragging = false;
|
---|
490 | const styles = coerceElement(this.element).style;
|
---|
491 | styles.scrollSnapType = styles.msScrollSnapType = this._initialScrollSnap;
|
---|
492 | // TODO(crisbeto): may have to wait for the animations to finish.
|
---|
493 | this._activeDraggables.forEach(item => {
|
---|
494 | var _a;
|
---|
495 | const rootElement = item.getRootElement();
|
---|
496 | if (rootElement) {
|
---|
497 | const initialTransform = (_a = this._itemPositions
|
---|
498 | .find(current => current.drag === item)) === null || _a === void 0 ? void 0 : _a.initialTransform;
|
---|
499 | rootElement.style.transform = initialTransform || '';
|
---|
500 | }
|
---|
501 | });
|
---|
502 | this._siblings.forEach(sibling => sibling._stopReceiving(this));
|
---|
503 | this._activeDraggables = [];
|
---|
504 | this._itemPositions = [];
|
---|
505 | this._previousSwap.drag = null;
|
---|
506 | this._previousSwap.delta = 0;
|
---|
507 | this._previousSwap.overlaps = false;
|
---|
508 | this._stopScrolling();
|
---|
509 | this._viewportScrollSubscription.unsubscribe();
|
---|
510 | this._parentPositions.clear();
|
---|
511 | }
|
---|
512 | /**
|
---|
513 | * Gets the offset in pixels by which the items that aren't being dragged should be moved.
|
---|
514 | * @param currentIndex Index of the item currently being dragged.
|
---|
515 | * @param siblings All of the items in the list.
|
---|
516 | * @param delta Direction in which the user is moving.
|
---|
517 | */
|
---|
518 | _getSiblingOffsetPx(currentIndex, siblings, delta) {
|
---|
519 | const isHorizontal = this._orientation === 'horizontal';
|
---|
520 | const currentPosition = siblings[currentIndex].clientRect;
|
---|
521 | const immediateSibling = siblings[currentIndex + delta * -1];
|
---|
522 | let siblingOffset = currentPosition[isHorizontal ? 'width' : 'height'] * delta;
|
---|
523 | if (immediateSibling) {
|
---|
524 | const start = isHorizontal ? 'left' : 'top';
|
---|
525 | const end = isHorizontal ? 'right' : 'bottom';
|
---|
526 | // Get the spacing between the start of the current item and the end of the one immediately
|
---|
527 | // after it in the direction in which the user is dragging, or vice versa. We add it to the
|
---|
528 | // offset in order to push the element to where it will be when it's inline and is influenced
|
---|
529 | // by the `margin` of its siblings.
|
---|
530 | if (delta === -1) {
|
---|
531 | siblingOffset -= immediateSibling.clientRect[start] - currentPosition[end];
|
---|
532 | }
|
---|
533 | else {
|
---|
534 | siblingOffset += currentPosition[start] - immediateSibling.clientRect[end];
|
---|
535 | }
|
---|
536 | }
|
---|
537 | return siblingOffset;
|
---|
538 | }
|
---|
539 | /**
|
---|
540 | * Gets the offset in pixels by which the item that is being dragged should be moved.
|
---|
541 | * @param currentPosition Current position of the item.
|
---|
542 | * @param newPosition Position of the item where the current item should be moved.
|
---|
543 | * @param delta Direction in which the user is moving.
|
---|
544 | */
|
---|
545 | _getItemOffsetPx(currentPosition, newPosition, delta) {
|
---|
546 | const isHorizontal = this._orientation === 'horizontal';
|
---|
547 | let itemOffset = isHorizontal ? newPosition.left - currentPosition.left :
|
---|
548 | newPosition.top - currentPosition.top;
|
---|
549 | // Account for differences in the item width/height.
|
---|
550 | if (delta === -1) {
|
---|
551 | itemOffset += isHorizontal ? newPosition.width - currentPosition.width :
|
---|
552 | newPosition.height - currentPosition.height;
|
---|
553 | }
|
---|
554 | return itemOffset;
|
---|
555 | }
|
---|
556 | /**
|
---|
557 | * Checks if pointer is entering in the first position
|
---|
558 | * @param pointerX Position of the user's pointer along the X axis.
|
---|
559 | * @param pointerY Position of the user's pointer along the Y axis.
|
---|
560 | */
|
---|
561 | _shouldEnterAsFirstChild(pointerX, pointerY) {
|
---|
562 | if (!this._activeDraggables.length) {
|
---|
563 | return false;
|
---|
564 | }
|
---|
565 | const itemPositions = this._itemPositions;
|
---|
566 | const isHorizontal = this._orientation === 'horizontal';
|
---|
567 | // `itemPositions` are sorted by position while `activeDraggables` are sorted by child index
|
---|
568 | // check if container is using some sort of "reverse" ordering (eg: flex-direction: row-reverse)
|
---|
569 | const reversed = itemPositions[0].drag !== this._activeDraggables[0];
|
---|
570 | if (reversed) {
|
---|
571 | const lastItemRect = itemPositions[itemPositions.length - 1].clientRect;
|
---|
572 | return isHorizontal ? pointerX >= lastItemRect.right : pointerY >= lastItemRect.bottom;
|
---|
573 | }
|
---|
574 | else {
|
---|
575 | const firstItemRect = itemPositions[0].clientRect;
|
---|
576 | return isHorizontal ? pointerX <= firstItemRect.left : pointerY <= firstItemRect.top;
|
---|
577 | }
|
---|
578 | }
|
---|
579 | /**
|
---|
580 | * Gets the index of an item in the drop container, based on the position of the user's pointer.
|
---|
581 | * @param item Item that is being sorted.
|
---|
582 | * @param pointerX Position of the user's pointer along the X axis.
|
---|
583 | * @param pointerY Position of the user's pointer along the Y axis.
|
---|
584 | * @param delta Direction in which the user is moving their pointer.
|
---|
585 | */
|
---|
586 | _getItemIndexFromPointerPosition(item, pointerX, pointerY, delta) {
|
---|
587 | const isHorizontal = this._orientation === 'horizontal';
|
---|
588 | const index = findIndex(this._itemPositions, ({ drag, clientRect }, _, array) => {
|
---|
589 | if (drag === item) {
|
---|
590 | // If there's only one item left in the container, it must be
|
---|
591 | // the dragged item itself so we use it as a reference.
|
---|
592 | return array.length < 2;
|
---|
593 | }
|
---|
594 | if (delta) {
|
---|
595 | const direction = isHorizontal ? delta.x : delta.y;
|
---|
596 | // If the user is still hovering over the same item as last time, their cursor hasn't left
|
---|
597 | // the item after we made the swap, and they didn't change the direction in which they're
|
---|
598 | // dragging, we don't consider it a direction swap.
|
---|
599 | if (drag === this._previousSwap.drag && this._previousSwap.overlaps &&
|
---|
600 | direction === this._previousSwap.delta) {
|
---|
601 | return false;
|
---|
602 | }
|
---|
603 | }
|
---|
604 | return isHorizontal ?
|
---|
605 | // Round these down since most browsers report client rects with
|
---|
606 | // sub-pixel precision, whereas the pointer coordinates are rounded to pixels.
|
---|
607 | pointerX >= Math.floor(clientRect.left) && pointerX < Math.floor(clientRect.right) :
|
---|
608 | pointerY >= Math.floor(clientRect.top) && pointerY < Math.floor(clientRect.bottom);
|
---|
609 | });
|
---|
610 | return (index === -1 || !this.sortPredicate(index, item, this)) ? -1 : index;
|
---|
611 | }
|
---|
612 | /** Caches the current items in the list and their positions. */
|
---|
613 | _cacheItems() {
|
---|
614 | this._activeDraggables = this._draggables.slice();
|
---|
615 | this._cacheItemPositions();
|
---|
616 | this._cacheParentPositions();
|
---|
617 | }
|
---|
618 | /**
|
---|
619 | * Checks whether the user's pointer is positioned over the container.
|
---|
620 | * @param x Pointer position along the X axis.
|
---|
621 | * @param y Pointer position along the Y axis.
|
---|
622 | */
|
---|
623 | _isOverContainer(x, y) {
|
---|
624 | return this._clientRect != null && isInsideClientRect(this._clientRect, x, y);
|
---|
625 | }
|
---|
626 | /**
|
---|
627 | * Figures out whether an item should be moved into a sibling
|
---|
628 | * drop container, based on its current position.
|
---|
629 | * @param item Drag item that is being moved.
|
---|
630 | * @param x Position of the item along the X axis.
|
---|
631 | * @param y Position of the item along the Y axis.
|
---|
632 | */
|
---|
633 | _getSiblingContainerFromPosition(item, x, y) {
|
---|
634 | return this._siblings.find(sibling => sibling._canReceive(item, x, y));
|
---|
635 | }
|
---|
636 | /**
|
---|
637 | * Checks whether the drop list can receive the passed-in item.
|
---|
638 | * @param item Item that is being dragged into the list.
|
---|
639 | * @param x Position of the item along the X axis.
|
---|
640 | * @param y Position of the item along the Y axis.
|
---|
641 | */
|
---|
642 | _canReceive(item, x, y) {
|
---|
643 | if (!this._clientRect || !isInsideClientRect(this._clientRect, x, y) ||
|
---|
644 | !this.enterPredicate(item, this)) {
|
---|
645 | return false;
|
---|
646 | }
|
---|
647 | const elementFromPoint = this._getShadowRoot().elementFromPoint(x, y);
|
---|
648 | // If there's no element at the pointer position, then
|
---|
649 | // the client rect is probably scrolled out of the view.
|
---|
650 | if (!elementFromPoint) {
|
---|
651 | return false;
|
---|
652 | }
|
---|
653 | const nativeElement = coerceElement(this.element);
|
---|
654 | // The `ClientRect`, that we're using to find the container over which the user is
|
---|
655 | // hovering, doesn't give us any information on whether the element has been scrolled
|
---|
656 | // out of the view or whether it's overlapping with other containers. This means that
|
---|
657 | // we could end up transferring the item into a container that's invisible or is positioned
|
---|
658 | // below another one. We use the result from `elementFromPoint` to get the top-most element
|
---|
659 | // at the pointer position and to find whether it's one of the intersecting drop containers.
|
---|
660 | return elementFromPoint === nativeElement || nativeElement.contains(elementFromPoint);
|
---|
661 | }
|
---|
662 | /**
|
---|
663 | * Called by one of the connected drop lists when a dragging sequence has started.
|
---|
664 | * @param sibling Sibling in which dragging has started.
|
---|
665 | */
|
---|
666 | _startReceiving(sibling, items) {
|
---|
667 | const activeSiblings = this._activeSiblings;
|
---|
668 | if (!activeSiblings.has(sibling) && items.every(item => {
|
---|
669 | // Note that we have to add an exception to the `enterPredicate` for items that started off
|
---|
670 | // in this drop list. The drag ref has logic that allows an item to return to its initial
|
---|
671 | // container, if it has left the initial container and none of the connected containers
|
---|
672 | // allow it to enter. See `DragRef._updateActiveDropContainer` for more context.
|
---|
673 | return this.enterPredicate(item, this) || this._draggables.indexOf(item) > -1;
|
---|
674 | })) {
|
---|
675 | activeSiblings.add(sibling);
|
---|
676 | this._cacheParentPositions();
|
---|
677 | this._listenToScrollEvents();
|
---|
678 | }
|
---|
679 | }
|
---|
680 | /**
|
---|
681 | * Called by a connected drop list when dragging has stopped.
|
---|
682 | * @param sibling Sibling whose dragging has stopped.
|
---|
683 | */
|
---|
684 | _stopReceiving(sibling) {
|
---|
685 | this._activeSiblings.delete(sibling);
|
---|
686 | this._viewportScrollSubscription.unsubscribe();
|
---|
687 | }
|
---|
688 | /**
|
---|
689 | * Starts listening to scroll events on the viewport.
|
---|
690 | * Used for updating the internal state of the list.
|
---|
691 | */
|
---|
692 | _listenToScrollEvents() {
|
---|
693 | this._viewportScrollSubscription = this._dragDropRegistry
|
---|
694 | .scrolled(this._getShadowRoot())
|
---|
695 | .subscribe(event => {
|
---|
696 | if (this.isDragging()) {
|
---|
697 | const scrollDifference = this._parentPositions.handleScroll(event);
|
---|
698 | if (scrollDifference) {
|
---|
699 | // Since we know the amount that the user has scrolled we can shift all of the
|
---|
700 | // client rectangles ourselves. This is cheaper than re-measuring everything and
|
---|
701 | // we can avoid inconsistent behavior where we might be measuring the element before
|
---|
702 | // its position has changed.
|
---|
703 | this._itemPositions.forEach(({ clientRect }) => {
|
---|
704 | adjustClientRect(clientRect, scrollDifference.top, scrollDifference.left);
|
---|
705 | });
|
---|
706 | // We need two loops for this, because we want all of the cached
|
---|
707 | // positions to be up-to-date before we re-sort the item.
|
---|
708 | this._itemPositions.forEach(({ drag }) => {
|
---|
709 | if (this._dragDropRegistry.isDragging(drag)) {
|
---|
710 | // We need to re-sort the item manually, because the pointer move
|
---|
711 | // events won't be dispatched while the user is scrolling.
|
---|
712 | drag._sortFromLastPointerPosition();
|
---|
713 | }
|
---|
714 | });
|
---|
715 | }
|
---|
716 | }
|
---|
717 | else if (this.isReceiving()) {
|
---|
718 | this._cacheParentPositions();
|
---|
719 | }
|
---|
720 | });
|
---|
721 | }
|
---|
722 | /**
|
---|
723 | * Lazily resolves and returns the shadow root of the element. We do this in a function, rather
|
---|
724 | * than saving it in property directly on init, because we want to resolve it as late as possible
|
---|
725 | * in order to ensure that the element has been moved into the shadow DOM. Doing it inside the
|
---|
726 | * constructor might be too early if the element is inside of something like `ngFor` or `ngIf`.
|
---|
727 | */
|
---|
728 | _getShadowRoot() {
|
---|
729 | if (!this._cachedShadowRoot) {
|
---|
730 | const shadowRoot = _getShadowRoot(coerceElement(this.element));
|
---|
731 | this._cachedShadowRoot = shadowRoot || this._document;
|
---|
732 | }
|
---|
733 | return this._cachedShadowRoot;
|
---|
734 | }
|
---|
735 | /** Notifies any siblings that may potentially receive the item. */
|
---|
736 | _notifyReceivingSiblings() {
|
---|
737 | const draggedItems = this._activeDraggables.filter(item => item.isDragging());
|
---|
738 | this._siblings.forEach(sibling => sibling._startReceiving(this, draggedItems));
|
---|
739 | }
|
---|
740 | }
|
---|
741 | /**
|
---|
742 | * Finds the index of an item that matches a predicate function. Used as an equivalent
|
---|
743 | * of `Array.prototype.findIndex` which isn't part of the standard Google typings.
|
---|
744 | * @param array Array in which to look for matches.
|
---|
745 | * @param predicate Function used to determine whether an item is a match.
|
---|
746 | */
|
---|
747 | function findIndex(array, predicate) {
|
---|
748 | for (let i = 0; i < array.length; i++) {
|
---|
749 | if (predicate(array[i], i, array)) {
|
---|
750 | return i;
|
---|
751 | }
|
---|
752 | }
|
---|
753 | return -1;
|
---|
754 | }
|
---|
755 | /**
|
---|
756 | * Increments the vertical scroll position of a node.
|
---|
757 | * @param node Node whose scroll position should change.
|
---|
758 | * @param amount Amount of pixels that the `node` should be scrolled.
|
---|
759 | */
|
---|
760 | function incrementVerticalScroll(node, amount) {
|
---|
761 | if (node === window) {
|
---|
762 | node.scrollBy(0, amount);
|
---|
763 | }
|
---|
764 | else {
|
---|
765 | // Ideally we could use `Element.scrollBy` here as well, but IE and Edge don't support it.
|
---|
766 | node.scrollTop += amount;
|
---|
767 | }
|
---|
768 | }
|
---|
769 | /**
|
---|
770 | * Increments the horizontal scroll position of a node.
|
---|
771 | * @param node Node whose scroll position should change.
|
---|
772 | * @param amount Amount of pixels that the `node` should be scrolled.
|
---|
773 | */
|
---|
774 | function incrementHorizontalScroll(node, amount) {
|
---|
775 | if (node === window) {
|
---|
776 | node.scrollBy(amount, 0);
|
---|
777 | }
|
---|
778 | else {
|
---|
779 | // Ideally we could use `Element.scrollBy` here as well, but IE and Edge don't support it.
|
---|
780 | node.scrollLeft += amount;
|
---|
781 | }
|
---|
782 | }
|
---|
783 | /**
|
---|
784 | * Gets whether the vertical auto-scroll direction of a node.
|
---|
785 | * @param clientRect Dimensions of the node.
|
---|
786 | * @param pointerY Position of the user's pointer along the y axis.
|
---|
787 | */
|
---|
788 | function getVerticalScrollDirection(clientRect, pointerY) {
|
---|
789 | const { top, bottom, height } = clientRect;
|
---|
790 | const yThreshold = height * SCROLL_PROXIMITY_THRESHOLD;
|
---|
791 | if (pointerY >= top - yThreshold && pointerY <= top + yThreshold) {
|
---|
792 | return 1 /* UP */;
|
---|
793 | }
|
---|
794 | else if (pointerY >= bottom - yThreshold && pointerY <= bottom + yThreshold) {
|
---|
795 | return 2 /* DOWN */;
|
---|
796 | }
|
---|
797 | return 0 /* NONE */;
|
---|
798 | }
|
---|
799 | /**
|
---|
800 | * Gets whether the horizontal auto-scroll direction of a node.
|
---|
801 | * @param clientRect Dimensions of the node.
|
---|
802 | * @param pointerX Position of the user's pointer along the x axis.
|
---|
803 | */
|
---|
804 | function getHorizontalScrollDirection(clientRect, pointerX) {
|
---|
805 | const { left, right, width } = clientRect;
|
---|
806 | const xThreshold = width * SCROLL_PROXIMITY_THRESHOLD;
|
---|
807 | if (pointerX >= left - xThreshold && pointerX <= left + xThreshold) {
|
---|
808 | return 1 /* LEFT */;
|
---|
809 | }
|
---|
810 | else if (pointerX >= right - xThreshold && pointerX <= right + xThreshold) {
|
---|
811 | return 2 /* RIGHT */;
|
---|
812 | }
|
---|
813 | return 0 /* NONE */;
|
---|
814 | }
|
---|
815 | /**
|
---|
816 | * Gets the directions in which an element node should be scrolled,
|
---|
817 | * assuming that the user's pointer is already within it scrollable region.
|
---|
818 | * @param element Element for which we should calculate the scroll direction.
|
---|
819 | * @param clientRect Bounding client rectangle of the element.
|
---|
820 | * @param pointerX Position of the user's pointer along the x axis.
|
---|
821 | * @param pointerY Position of the user's pointer along the y axis.
|
---|
822 | */
|
---|
823 | function getElementScrollDirections(element, clientRect, pointerX, pointerY) {
|
---|
824 | const computedVertical = getVerticalScrollDirection(clientRect, pointerY);
|
---|
825 | const computedHorizontal = getHorizontalScrollDirection(clientRect, pointerX);
|
---|
826 | let verticalScrollDirection = 0 /* NONE */;
|
---|
827 | let horizontalScrollDirection = 0 /* NONE */;
|
---|
828 | // Note that we here we do some extra checks for whether the element is actually scrollable in
|
---|
829 | // a certain direction and we only assign the scroll direction if it is. We do this so that we
|
---|
830 | // can allow other elements to be scrolled, if the current element can't be scrolled anymore.
|
---|
831 | // This allows us to handle cases where the scroll regions of two scrollable elements overlap.
|
---|
832 | if (computedVertical) {
|
---|
833 | const scrollTop = element.scrollTop;
|
---|
834 | if (computedVertical === 1 /* UP */) {
|
---|
835 | if (scrollTop > 0) {
|
---|
836 | verticalScrollDirection = 1 /* UP */;
|
---|
837 | }
|
---|
838 | }
|
---|
839 | else if (element.scrollHeight - scrollTop > element.clientHeight) {
|
---|
840 | verticalScrollDirection = 2 /* DOWN */;
|
---|
841 | }
|
---|
842 | }
|
---|
843 | if (computedHorizontal) {
|
---|
844 | const scrollLeft = element.scrollLeft;
|
---|
845 | if (computedHorizontal === 1 /* LEFT */) {
|
---|
846 | if (scrollLeft > 0) {
|
---|
847 | horizontalScrollDirection = 1 /* LEFT */;
|
---|
848 | }
|
---|
849 | }
|
---|
850 | else if (element.scrollWidth - scrollLeft > element.clientWidth) {
|
---|
851 | horizontalScrollDirection = 2 /* RIGHT */;
|
---|
852 | }
|
---|
853 | }
|
---|
854 | return [verticalScrollDirection, horizontalScrollDirection];
|
---|
855 | }
|
---|
856 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"drop-list-ref.js","sourceRoot":"","sources":["../../../../../../src/cdk/drag-drop/drop-list-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 {ElementRef, NgZone} from '@angular/core';\nimport {Direction} from '@angular/cdk/bidi';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {ViewportRuler} from '@angular/cdk/scrolling';\nimport {_getShadowRoot} from '@angular/cdk/platform';\nimport {Subject, Subscription, interval, animationFrameScheduler} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\nimport {moveItemInArray} from './drag-utils';\nimport {DragDropRegistry} from './drag-drop-registry';\nimport {DragRefInternal as DragRef, Point} from './drag-ref';\nimport {\n  isPointerNearClientRect,\n  adjustClientRect,\n  getMutableClientRect,\n  isInsideClientRect,\n} from './client-rect';\nimport {ParentPositionTracker} from './parent-position-tracker';\nimport {combineTransforms, DragCSSStyleDeclaration} from './drag-styling';\n\n/**\n * Proximity, as a ratio to width/height, at which a\n * dragged item will affect the drop container.\n */\nconst DROP_PROXIMITY_THRESHOLD = 0.05;\n\n/**\n * Proximity, as a ratio to width/height at which to start auto-scrolling the drop list or the\n * viewport. The value comes from trying it out manually until it feels right.\n */\nconst SCROLL_PROXIMITY_THRESHOLD = 0.05;\n\n/**\n * Entry in the position cache for draggable items.\n * @docs-private\n */\ninterface CachedItemPosition {\n  /** Instance of the drag item. */\n  drag: DragRef;\n  /** Dimensions of the item. */\n  clientRect: ClientRect;\n  /** Amount by which the item has been moved since dragging started. */\n  offset: number;\n  /** Inline transform that the drag item had when dragging started. */\n  initialTransform: string;\n}\n\n/** Vertical direction in which we can auto-scroll. */\nconst enum AutoScrollVerticalDirection {NONE, UP, DOWN}\n\n/** Horizontal direction in which we can auto-scroll. */\nconst enum AutoScrollHorizontalDirection {NONE, LEFT, RIGHT}\n\n/**\n * Internal compile-time-only representation of a `DropListRef`.\n * Used to avoid circular import issues between the `DropListRef` and the `DragRef`.\n * @docs-private\n */\nexport interface DropListRefInternal extends DropListRef {}\n\n/**\n * Reference to a drop list. Used to manipulate or dispose of the container.\n */\nexport class DropListRef<T = any> {\n  /** Element that the drop list is attached to. */\n  element: HTMLElement | ElementRef<HTMLElement>;\n\n  /** Whether starting a dragging sequence from this container is disabled. */\n  disabled: boolean = false;\n\n  /** Whether sorting items within the list is disabled. */\n  sortingDisabled: boolean = false;\n\n  /** Locks the position of the draggable elements inside the container along the specified axis. */\n  lockAxis: 'x' | 'y';\n\n  /**\n   * Whether auto-scrolling the view when the user\n   * moves their pointer close to the edges is disabled.\n   */\n  autoScrollDisabled: boolean = false;\n\n  /** Number of pixels to scroll for each frame when auto-scrolling an element. */\n  autoScrollStep: number = 2;\n\n  /**\n   * Function that is used to determine whether an item\n   * is allowed to be moved into a drop container.\n   */\n  enterPredicate: (drag: DragRef, drop: DropListRef) => boolean = () => true;\n\n  /** Functions that is used to determine whether an item can be sorted into a particular index. */\n  sortPredicate: (index: number, drag: DragRef, drop: DropListRef) => boolean = () => true;\n\n  /** Emits right before dragging has started. */\n  readonly beforeStarted = new Subject<void>();\n\n  /**\n   * Emits when the user has moved a new drag item into this container.\n   */\n  readonly entered = new Subject<{item: DragRef, container: DropListRef, currentIndex: number}>();\n\n  /**\n   * Emits when the user removes an item from the container\n   * by dragging it into another container.\n   */\n  readonly exited = new Subject<{item: DragRef, container: DropListRef}>();\n\n  /** Emits when the user drops an item inside the container. */\n  readonly dropped = new Subject<{\n    item: DragRef,\n    currentIndex: number,\n    previousIndex: number,\n    container: DropListRef,\n    previousContainer: DropListRef,\n    isPointerOverContainer: boolean,\n    distance: Point;\n    dropPoint: Point;\n  }>();\n\n  /** Emits as the user is swapping items while actively dragging. */\n  readonly sorted = new Subject<{\n    previousIndex: number,\n    currentIndex: number,\n    container: DropListRef,\n    item: DragRef\n  }>();\n\n  /** Arbitrary data that can be attached to the drop list. */\n  data: T;\n\n  /** Whether an item in the list is being dragged. */\n  private _isDragging = false;\n\n  /** Cache of the dimensions of all the items inside the container. */\n  private _itemPositions: CachedItemPosition[] = [];\n\n  /** Keeps track of the positions of any parent scrollable elements. */\n  private _parentPositions: ParentPositionTracker;\n\n  /** Cached `ClientRect` of the drop list. */\n  private _clientRect: ClientRect | undefined;\n\n  /**\n   * Draggable items that are currently active inside the container. Includes the items\n   * from `_draggables`, as well as any items that have been dragged in, but haven't\n   * been dropped yet.\n   */\n  private _activeDraggables: DragRef[];\n\n  /**\n   * Keeps track of the item that was last swapped with the dragged item, as well as what direction\n   * the pointer was moving in when the swap occured and whether the user's pointer continued to\n   * overlap with the swapped item after the swapping occurred.\n   */\n  private _previousSwap = {drag: null as DragRef | null, delta: 0, overlaps: false};\n\n  /** Draggable items in the container. */\n  private _draggables: readonly DragRef[] = [];\n\n  /** Drop lists that are connected to the current one. */\n  private _siblings: readonly DropListRef[] = [];\n\n  /** Direction in which the list is oriented. */\n  private _orientation: 'horizontal' | 'vertical' = 'vertical';\n\n  /** Connected siblings that currently have a dragged item. */\n  private _activeSiblings = new Set<DropListRef>();\n\n  /** Layout direction of the drop list. */\n  private _direction: Direction = 'ltr';\n\n  /** Subscription to the window being scrolled. */\n  private _viewportScrollSubscription = Subscription.EMPTY;\n\n  /** Vertical direction in which the list is currently scrolling. */\n  private _verticalScrollDirection = AutoScrollVerticalDirection.NONE;\n\n  /** Horizontal direction in which the list is currently scrolling. */\n  private _horizontalScrollDirection = AutoScrollHorizontalDirection.NONE;\n\n  /** Node that is being auto-scrolled. */\n  private _scrollNode: HTMLElement | Window;\n\n  /** Used to signal to the current auto-scroll sequence when to stop. */\n  private readonly _stopScrollTimers = new Subject<void>();\n\n  /** Shadow root of the current element. Necessary for `elementFromPoint` to resolve correctly. */\n  private _cachedShadowRoot: DocumentOrShadowRoot | null = null;\n\n  /** Reference to the document. */\n  private _document: Document;\n\n  /** Elements that can be scrolled while the user is dragging. */\n  private _scrollableElements: HTMLElement[];\n\n  /** Initial value for the element's `scroll-snap-type` style. */\n  private _initialScrollSnap: string;\n\n  constructor(\n    element: ElementRef<HTMLElement> | HTMLElement,\n    private _dragDropRegistry: DragDropRegistry<DragRef, DropListRef>,\n    _document: any,\n    private _ngZone: NgZone,\n    private _viewportRuler: ViewportRuler) {\n    this.element = coerceElement(element);\n    this._document = _document;\n    this.withScrollableParents([this.element]);\n    _dragDropRegistry.registerDropContainer(this);\n    this._parentPositions = new ParentPositionTracker(_document, _viewportRuler);\n  }\n\n  /** Removes the drop list functionality from the DOM element. */\n  dispose() {\n    this._stopScrolling();\n    this._stopScrollTimers.complete();\n    this._viewportScrollSubscription.unsubscribe();\n    this.beforeStarted.complete();\n    this.entered.complete();\n    this.exited.complete();\n    this.dropped.complete();\n    this.sorted.complete();\n    this._activeSiblings.clear();\n    this._scrollNode = null!;\n    this._parentPositions.clear();\n    this._dragDropRegistry.removeDropContainer(this);\n  }\n\n  /** Whether an item from this list is currently being dragged. */\n  isDragging() {\n    return this._isDragging;\n  }\n\n  /** Starts dragging an item. */\n  start(): void {\n    this._draggingStarted();\n    this._notifyReceivingSiblings();\n  }\n\n  /**\n   * Emits an event to indicate that the user moved an item into the container.\n   * @param item Item that was moved into the container.\n   * @param pointerX Position of the item along the X axis.\n   * @param pointerY Position of the item along the Y axis.\n   * @param index Index at which the item entered. If omitted, the container will try to figure it\n   *   out automatically.\n   */\n  enter(item: DragRef, pointerX: number, pointerY: number, index?: number): void {\n    this._draggingStarted();\n\n    // If sorting is disabled, we want the item to return to its starting\n    // position if the user is returning it to its initial container.\n    let newIndex: number;\n\n    if (index == null) {\n      newIndex = this.sortingDisabled ? this._draggables.indexOf(item) : -1;\n\n      if (newIndex === -1) {\n        // We use the coordinates of where the item entered the drop\n        // zone to figure out at which index it should be inserted.\n        newIndex = this._getItemIndexFromPointerPosition(item, pointerX, pointerY);\n      }\n    } else {\n      newIndex = index;\n    }\n\n    const activeDraggables = this._activeDraggables;\n    const currentIndex = activeDraggables.indexOf(item);\n    const placeholder = item.getPlaceholderElement();\n    let newPositionReference: DragRef | undefined = activeDraggables[newIndex];\n\n    // If the item at the new position is the same as the item that is being dragged,\n    // it means that we're trying to restore the item to its initial position. In this\n    // case we should use the next item from the list as the reference.\n    if (newPositionReference === item) {\n      newPositionReference = activeDraggables[newIndex + 1];\n    }\n\n    // Since the item may be in the `activeDraggables` already (e.g. if the user dragged it\n    // into another container and back again), we have to ensure that it isn't duplicated.\n    if (currentIndex > -1) {\n      activeDraggables.splice(currentIndex, 1);\n    }\n\n    // Don't use items that are being dragged as a reference, because\n    // their element has been moved down to the bottom of the body.\n    if (newPositionReference && !this._dragDropRegistry.isDragging(newPositionReference)) {\n      const element = newPositionReference.getRootElement();\n      element.parentElement!.insertBefore(placeholder, element);\n      activeDraggables.splice(newIndex, 0, item);\n    } else if (this._shouldEnterAsFirstChild(pointerX, pointerY)) {\n      const reference = activeDraggables[0].getRootElement();\n      reference.parentNode!.insertBefore(placeholder, reference);\n      activeDraggables.unshift(item);\n    } else {\n      coerceElement(this.element).appendChild(placeholder);\n      activeDraggables.push(item);\n    }\n\n    // The transform needs to be cleared so it doesn't throw off the measurements.\n    placeholder.style.transform = '';\n\n    // Note that the positions were already cached when we called `start` above,\n    // but we need to refresh them since the amount of items has changed and also parent rects.\n    this._cacheItemPositions();\n    this._cacheParentPositions();\n\n    // Notify siblings at the end so that the item has been inserted into the `activeDraggables`.\n    this._notifyReceivingSiblings();\n    this.entered.next({item, container: this, currentIndex: this.getItemIndex(item)});\n  }\n\n  /**\n   * Removes an item from the container after it was dragged into another container by the user.\n   * @param item Item that was dragged out.\n   */\n  exit(item: DragRef): void {\n    this._reset();\n    this.exited.next({item, container: this});\n  }\n\n  /**\n   * Drops an item into this container.\n   * @param item Item being dropped into the container.\n   * @param currentIndex Index at which the item should be inserted.\n   * @param previousIndex Index of the item when dragging started.\n   * @param previousContainer Container from which the item got dragged in.\n   * @param isPointerOverContainer Whether the user's pointer was over the\n   *    container when the item was dropped.\n   * @param distance Distance the user has dragged since the start of the dragging sequence.\n   */\n  drop(item: DragRef, currentIndex: number, previousIndex: number, previousContainer: DropListRef,\n    isPointerOverContainer: boolean, distance: Point, dropPoint: Point): void {\n    this._reset();\n    this.dropped.next({\n      item,\n      currentIndex,\n      previousIndex,\n      container: this,\n      previousContainer,\n      isPointerOverContainer,\n      distance,\n      dropPoint\n    });\n  }\n\n  /**\n   * Sets the draggable items that are a part of this list.\n   * @param items Items that are a part of this list.\n   */\n  withItems(items: DragRef[]): this {\n    const previousItems = this._draggables;\n    this._draggables = items;\n    items.forEach(item => item._withDropContainer(this));\n\n    if (this.isDragging()) {\n      const draggedItems = previousItems.filter(item => item.isDragging());\n\n      // If all of the items being dragged were removed\n      // from the list, abort the current drag sequence.\n      if (draggedItems.every(item => items.indexOf(item) === -1)) {\n        this._reset();\n      } else {\n        this._cacheItems();\n      }\n    }\n\n    return this;\n  }\n\n  /** Sets the layout direction of the drop list. */\n  withDirection(direction: Direction): this {\n    this._direction = direction;\n    return this;\n  }\n\n  /**\n   * Sets the containers that are connected to this one. When two or more containers are\n   * connected, the user will be allowed to transfer items between them.\n   * @param connectedTo Other containers that the current containers should be connected to.\n   */\n  connectedTo(connectedTo: DropListRef[]): this {\n    this._siblings = connectedTo.slice();\n    return this;\n  }\n\n  /**\n   * Sets the orientation of the container.\n   * @param orientation New orientation for the container.\n   */\n  withOrientation(orientation: 'vertical' | 'horizontal'): this {\n    this._orientation = orientation;\n    return this;\n  }\n\n  /**\n   * Sets which parent elements are can be scrolled while the user is dragging.\n   * @param elements Elements that can be scrolled.\n   */\n  withScrollableParents(elements: HTMLElement[]): this {\n    const element = coerceElement(this.element);\n\n    // We always allow the current element to be scrollable\n    // so we need to ensure that it's in the array.\n    this._scrollableElements =\n        elements.indexOf(element) === -1 ? [element, ...elements] : elements.slice();\n    return this;\n  }\n\n  /** Gets the scrollable parents that are registered with this drop container. */\n  getScrollableParents(): readonly HTMLElement[] {\n    return this._scrollableElements;\n  }\n\n  /**\n   * Figures out the index of an item in the container.\n   * @param item Item whose index should be determined.\n   */\n  getItemIndex(item: DragRef): number {\n    if (!this._isDragging) {\n      return this._draggables.indexOf(item);\n    }\n\n    // Items are sorted always by top/left in the cache, however they flow differently in RTL.\n    // The rest of the logic still stands no matter what orientation we're in, however\n    // we need to invert the array when determining the index.\n    const items = this._orientation === 'horizontal' && this._direction === 'rtl' ?\n        this._itemPositions.slice().reverse() : this._itemPositions;\n\n    return findIndex(items, currentItem => currentItem.drag === item);\n  }\n\n  /**\n   * Whether the list is able to receive the item that\n   * is currently being dragged inside a connected drop list.\n   */\n  isReceiving(): boolean {\n    return this._activeSiblings.size > 0;\n  }\n\n  /**\n   * Sorts an item inside the container based on its position.\n   * @param item Item to be sorted.\n   * @param pointerX Position of the item along the X axis.\n   * @param pointerY Position of the item along the Y axis.\n   * @param pointerDelta Direction in which the pointer is moving along each axis.\n   */\n  _sortItem(item: DragRef, pointerX: number, pointerY: number,\n            pointerDelta: {x: number, y: number}): void {\n    // Don't sort the item if sorting is disabled or it's out of range.\n    if (this.sortingDisabled || !this._clientRect ||\n        !isPointerNearClientRect(this._clientRect, DROP_PROXIMITY_THRESHOLD, pointerX, pointerY)) {\n      return;\n    }\n\n    const siblings = this._itemPositions;\n    const newIndex = this._getItemIndexFromPointerPosition(item, pointerX, pointerY, pointerDelta);\n\n    if (newIndex === -1 && siblings.length > 0) {\n      return;\n    }\n\n    const isHorizontal = this._orientation === 'horizontal';\n    const currentIndex = findIndex(siblings, currentItem => currentItem.drag === item);\n    const siblingAtNewPosition = siblings[newIndex];\n    const currentPosition = siblings[currentIndex].clientRect;\n    const newPosition = siblingAtNewPosition.clientRect;\n    const delta = currentIndex > newIndex ? 1 : -1;\n\n    // How many pixels the item's placeholder should be offset.\n    const itemOffset = this._getItemOffsetPx(currentPosition, newPosition, delta);\n\n    // How many pixels all the other items should be offset.\n    const siblingOffset = this._getSiblingOffsetPx(currentIndex, siblings, delta);\n\n    // Save the previous order of the items before moving the item to its new index.\n    // We use this to check whether an item has been moved as a result of the sorting.\n    const oldOrder = siblings.slice();\n\n    // Shuffle the array in place.\n    moveItemInArray(siblings, currentIndex, newIndex);\n\n    this.sorted.next({\n      previousIndex: currentIndex,\n      currentIndex: newIndex,\n      container: this,\n      item\n    });\n\n    siblings.forEach((sibling, index) => {\n      // Don't do anything if the position hasn't changed.\n      if (oldOrder[index] === sibling) {\n        return;\n      }\n\n      const isDraggedItem = sibling.drag === item;\n      const offset = isDraggedItem ? itemOffset : siblingOffset;\n      const elementToOffset = isDraggedItem ? item.getPlaceholderElement() :\n                                              sibling.drag.getRootElement();\n\n      // Update the offset to reflect the new position.\n      sibling.offset += offset;\n\n      // Since we're moving the items with a `transform`, we need to adjust their cached\n      // client rects to reflect their new position, as well as swap their positions in the cache.\n      // Note that we shouldn't use `getBoundingClientRect` here to update the cache, because the\n      // elements may be mid-animation which will give us a wrong result.\n      if (isHorizontal) {\n        // Round the transforms since some browsers will\n        // blur the elements, for sub-pixel transforms.\n        elementToOffset.style.transform = combineTransforms(\n          `translate3d(${Math.round(sibling.offset)}px, 0, 0)`, sibling.initialTransform);\n        adjustClientRect(sibling.clientRect, 0, offset);\n      } else {\n        elementToOffset.style.transform = combineTransforms(\n          `translate3d(0, ${Math.round(sibling.offset)}px, 0)`, sibling.initialTransform);\n        adjustClientRect(sibling.clientRect, offset, 0);\n      }\n    });\n\n    // Note that it's important that we do this after the client rects have been adjusted.\n    this._previousSwap.overlaps = isInsideClientRect(newPosition, pointerX, pointerY);\n    this._previousSwap.drag = siblingAtNewPosition.drag;\n    this._previousSwap.delta = isHorizontal ? pointerDelta.x : pointerDelta.y;\n  }\n\n  /**\n   * Checks whether the user's pointer is close to the edges of either the\n   * viewport or the drop list and starts the auto-scroll sequence.\n   * @param pointerX User's pointer position along the x axis.\n   * @param pointerY User's pointer position along the y axis.\n   */\n  _startScrollingIfNecessary(pointerX: number, pointerY: number) {\n    if (this.autoScrollDisabled) {\n      return;\n    }\n\n    let scrollNode: HTMLElement | Window | undefined;\n    let verticalScrollDirection = AutoScrollVerticalDirection.NONE;\n    let horizontalScrollDirection = AutoScrollHorizontalDirection.NONE;\n\n    // Check whether we should start scrolling any of the parent containers.\n    this._parentPositions.positions.forEach((position, element) => {\n      // We have special handling for the `document` below. Also this would be\n      // nicer with a  for...of loop, but it requires changing a compiler flag.\n      if (element === this._document || !position.clientRect || scrollNode) {\n        return;\n      }\n\n      if (isPointerNearClientRect(position.clientRect, DROP_PROXIMITY_THRESHOLD,\n          pointerX, pointerY)) {\n        [verticalScrollDirection, horizontalScrollDirection] = getElementScrollDirections(\n            element as HTMLElement, position.clientRect, pointerX, pointerY);\n\n        if (verticalScrollDirection || horizontalScrollDirection) {\n          scrollNode = element as HTMLElement;\n        }\n      }\n    });\n\n    // Otherwise check if we can start scrolling the viewport.\n    if (!verticalScrollDirection && !horizontalScrollDirection) {\n      const {width, height} = this._viewportRuler.getViewportSize();\n      const clientRect = {width, height, top: 0, right: width, bottom: height, left: 0};\n      verticalScrollDirection = getVerticalScrollDirection(clientRect, pointerY);\n      horizontalScrollDirection = getHorizontalScrollDirection(clientRect, pointerX);\n      scrollNode = window;\n    }\n\n    if (scrollNode && (verticalScrollDirection !== this._verticalScrollDirection ||\n        horizontalScrollDirection !== this._horizontalScrollDirection ||\n        scrollNode !== this._scrollNode)) {\n      this._verticalScrollDirection = verticalScrollDirection;\n      this._horizontalScrollDirection = horizontalScrollDirection;\n      this._scrollNode = scrollNode;\n\n      if ((verticalScrollDirection || horizontalScrollDirection) && scrollNode) {\n        this._ngZone.runOutsideAngular(this._startScrollInterval);\n      } else {\n        this._stopScrolling();\n      }\n    }\n  }\n\n  /** Stops any currently-running auto-scroll sequences. */\n  _stopScrolling() {\n    this._stopScrollTimers.next();\n  }\n\n  /** Starts the dragging sequence within the list. */\n  private _draggingStarted() {\n    const styles = coerceElement(this.element).style as DragCSSStyleDeclaration;\n    this.beforeStarted.next();\n    this._isDragging = true;\n\n    // We need to disable scroll snapping while the user is dragging, because it breaks automatic\n    // scrolling. The browser seems to round the value based on the snapping points which means\n    // that we can't increment/decrement the scroll position.\n    this._initialScrollSnap = styles.msScrollSnapType || styles.scrollSnapType || '';\n    styles.scrollSnapType = styles.msScrollSnapType = 'none';\n    this._cacheItems();\n    this._viewportScrollSubscription.unsubscribe();\n    this._listenToScrollEvents();\n  }\n\n  /** Caches the positions of the configured scrollable parents. */\n  private _cacheParentPositions() {\n    const element = coerceElement(this.element);\n    this._parentPositions.cache(this._scrollableElements);\n\n    // The list element is always in the `scrollableElements`\n    // so we can take advantage of the cached `ClientRect`.\n    this._clientRect = this._parentPositions.positions.get(element)!.clientRect!;\n  }\n\n  /** Refreshes the position cache of the items and sibling containers. */\n  private _cacheItemPositions() {\n    const isHorizontal = this._orientation === 'horizontal';\n\n    this._itemPositions = this._activeDraggables.map(drag => {\n      const elementToMeasure = drag.getVisibleElement();\n      return {\n        drag,\n        offset: 0,\n        initialTransform: elementToMeasure.style.transform || '',\n        clientRect: getMutableClientRect(elementToMeasure),\n      };\n    }).sort((a, b) => {\n      return isHorizontal ? a.clientRect.left - b.clientRect.left :\n                            a.clientRect.top - b.clientRect.top;\n    });\n  }\n\n  /** Resets the container to its initial state. */\n  private _reset() {\n    this._isDragging = false;\n\n    const styles = coerceElement(this.element).style as DragCSSStyleDeclaration;\n    styles.scrollSnapType = styles.msScrollSnapType = this._initialScrollSnap;\n\n    // TODO(crisbeto): may have to wait for the animations to finish.\n    this._activeDraggables.forEach(item => {\n      const rootElement = item.getRootElement();\n\n      if (rootElement) {\n        const initialTransform = this._itemPositions\n          .find(current => current.drag === item)?.initialTransform;\n        rootElement.style.transform = initialTransform || '';\n      }\n    });\n    this._siblings.forEach(sibling => sibling._stopReceiving(this));\n    this._activeDraggables = [];\n    this._itemPositions = [];\n    this._previousSwap.drag = null;\n    this._previousSwap.delta = 0;\n    this._previousSwap.overlaps = false;\n    this._stopScrolling();\n    this._viewportScrollSubscription.unsubscribe();\n    this._parentPositions.clear();\n  }\n\n  /**\n   * Gets the offset in pixels by which the items that aren't being dragged should be moved.\n   * @param currentIndex Index of the item currently being dragged.\n   * @param siblings All of the items in the list.\n   * @param delta Direction in which the user is moving.\n   */\n  private _getSiblingOffsetPx(currentIndex: number,\n                              siblings: CachedItemPosition[],\n                              delta: 1 | -1) {\n\n    const isHorizontal = this._orientation === 'horizontal';\n    const currentPosition = siblings[currentIndex].clientRect;\n    const immediateSibling = siblings[currentIndex + delta * -1];\n    let siblingOffset = currentPosition[isHorizontal ? 'width' : 'height'] * delta;\n\n    if (immediateSibling) {\n      const start = isHorizontal ? 'left' : 'top';\n      const end = isHorizontal ? 'right' : 'bottom';\n\n      // Get the spacing between the start of the current item and the end of the one immediately\n      // after it in the direction in which the user is dragging, or vice versa. We add it to the\n      // offset in order to push the element to where it will be when it's inline and is influenced\n      // by the `margin` of its siblings.\n      if (delta === -1) {\n        siblingOffset -= immediateSibling.clientRect[start] - currentPosition[end];\n      } else {\n        siblingOffset += currentPosition[start] - immediateSibling.clientRect[end];\n      }\n    }\n\n    return siblingOffset;\n  }\n\n  /**\n   * Gets the offset in pixels by which the item that is being dragged should be moved.\n   * @param currentPosition Current position of the item.\n   * @param newPosition Position of the item where the current item should be moved.\n   * @param delta Direction in which the user is moving.\n   */\n  private _getItemOffsetPx(currentPosition: ClientRect, newPosition: ClientRect, delta: 1 | -1) {\n    const isHorizontal = this._orientation === 'horizontal';\n    let itemOffset = isHorizontal ? newPosition.left - currentPosition.left :\n                                    newPosition.top - currentPosition.top;\n\n    // Account for differences in the item width/height.\n    if (delta === -1) {\n      itemOffset += isHorizontal ? newPosition.width - currentPosition.width :\n                                   newPosition.height - currentPosition.height;\n    }\n\n    return itemOffset;\n  }\n\n  /**\n   * Checks if pointer is entering in the first position\n   * @param pointerX Position of the user's pointer along the X axis.\n   * @param pointerY Position of the user's pointer along the Y axis.\n   */\n  private _shouldEnterAsFirstChild(pointerX: number, pointerY: number) {\n    if (!this._activeDraggables.length) {\n      return false;\n    }\n\n    const itemPositions = this._itemPositions;\n    const isHorizontal = this._orientation === 'horizontal';\n\n    // `itemPositions` are sorted by position while `activeDraggables` are sorted by child index\n    // check if container is using some sort of \"reverse\" ordering (eg: flex-direction: row-reverse)\n    const reversed = itemPositions[0].drag !== this._activeDraggables[0];\n    if (reversed) {\n      const lastItemRect = itemPositions[itemPositions.length - 1].clientRect;\n      return isHorizontal ? pointerX >= lastItemRect.right : pointerY >= lastItemRect.bottom;\n    } else {\n      const firstItemRect = itemPositions[0].clientRect;\n      return isHorizontal ? pointerX <= firstItemRect.left : pointerY <= firstItemRect.top;\n    }\n  }\n\n  /**\n   * Gets the index of an item in the drop container, based on the position of the user's pointer.\n   * @param item Item that is being sorted.\n   * @param pointerX Position of the user's pointer along the X axis.\n   * @param pointerY Position of the user's pointer along the Y axis.\n   * @param delta Direction in which the user is moving their pointer.\n   */\n  private _getItemIndexFromPointerPosition(item: DragRef, pointerX: number, pointerY: number,\n                                           delta?: {x: number, y: number}): number {\n    const isHorizontal = this._orientation === 'horizontal';\n    const index = findIndex(this._itemPositions, ({drag, clientRect}, _, array) => {\n      if (drag === item) {\n        // If there's only one item left in the container, it must be\n        // the dragged item itself so we use it as a reference.\n        return array.length < 2;\n      }\n\n      if (delta) {\n        const direction = isHorizontal ? delta.x : delta.y;\n\n        // If the user is still hovering over the same item as last time, their cursor hasn't left\n        // the item after we made the swap, and they didn't change the direction in which they're\n        // dragging, we don't consider it a direction swap.\n        if (drag === this._previousSwap.drag && this._previousSwap.overlaps &&\n            direction === this._previousSwap.delta) {\n          return false;\n        }\n      }\n\n      return isHorizontal ?\n          // Round these down since most browsers report client rects with\n          // sub-pixel precision, whereas the pointer coordinates are rounded to pixels.\n          pointerX >= Math.floor(clientRect.left) && pointerX < Math.floor(clientRect.right) :\n          pointerY >= Math.floor(clientRect.top) && pointerY < Math.floor(clientRect.bottom);\n    });\n\n    return (index === -1 || !this.sortPredicate(index, item, this)) ? -1 : index;\n  }\n\n  /** Caches the current items in the list and their positions. */\n  private _cacheItems(): void {\n    this._activeDraggables = this._draggables.slice();\n    this._cacheItemPositions();\n    this._cacheParentPositions();\n  }\n\n  /** Starts the interval that'll auto-scroll the element. */\n  private _startScrollInterval = () => {\n    this._stopScrolling();\n\n    interval(0, animationFrameScheduler)\n      .pipe(takeUntil(this._stopScrollTimers))\n      .subscribe(() => {\n        const node = this._scrollNode;\n        const scrollStep = this.autoScrollStep;\n\n        if (this._verticalScrollDirection === AutoScrollVerticalDirection.UP) {\n          incrementVerticalScroll(node, -scrollStep);\n        } else if (this._verticalScrollDirection === AutoScrollVerticalDirection.DOWN) {\n          incrementVerticalScroll(node, scrollStep);\n        }\n\n        if (this._horizontalScrollDirection === AutoScrollHorizontalDirection.LEFT) {\n          incrementHorizontalScroll(node, -scrollStep);\n        } else if (this._horizontalScrollDirection === AutoScrollHorizontalDirection.RIGHT) {\n          incrementHorizontalScroll(node, scrollStep);\n        }\n      });\n  }\n\n  /**\n   * Checks whether the user's pointer is positioned over the container.\n   * @param x Pointer position along the X axis.\n   * @param y Pointer position along the Y axis.\n   */\n  _isOverContainer(x: number, y: number): boolean {\n    return this._clientRect != null && isInsideClientRect(this._clientRect, x, y);\n  }\n\n  /**\n   * Figures out whether an item should be moved into a sibling\n   * drop container, based on its current position.\n   * @param item Drag item that is being moved.\n   * @param x Position of the item along the X axis.\n   * @param y Position of the item along the Y axis.\n   */\n  _getSiblingContainerFromPosition(item: DragRef, x: number, y: number): DropListRef | undefined {\n    return this._siblings.find(sibling => sibling._canReceive(item, x, y));\n  }\n\n  /**\n   * Checks whether the drop list can receive the passed-in item.\n   * @param item Item that is being dragged into the list.\n   * @param x Position of the item along the X axis.\n   * @param y Position of the item along the Y axis.\n   */\n  _canReceive(item: DragRef, x: number, y: number): boolean {\n    if (!this._clientRect || !isInsideClientRect(this._clientRect, x, y) ||\n        !this.enterPredicate(item, this)) {\n      return false;\n    }\n\n    const elementFromPoint = this._getShadowRoot().elementFromPoint(x, y) as HTMLElement | null;\n\n    // If there's no element at the pointer position, then\n    // the client rect is probably scrolled out of the view.\n    if (!elementFromPoint) {\n      return false;\n    }\n\n    const nativeElement = coerceElement(this.element);\n\n    // The `ClientRect`, that we're using to find the container over which the user is\n    // hovering, doesn't give us any information on whether the element has been scrolled\n    // out of the view or whether it's overlapping with other containers. This means that\n    // we could end up transferring the item into a container that's invisible or is positioned\n    // below another one. We use the result from `elementFromPoint` to get the top-most element\n    // at the pointer position and to find whether it's one of the intersecting drop containers.\n    return elementFromPoint === nativeElement || nativeElement.contains(elementFromPoint);\n  }\n\n  /**\n   * Called by one of the connected drop lists when a dragging sequence has started.\n   * @param sibling Sibling in which dragging has started.\n   */\n  _startReceiving(sibling: DropListRef, items: DragRef[]) {\n    const activeSiblings = this._activeSiblings;\n\n    if (!activeSiblings.has(sibling) && items.every(item => {\n      // Note that we have to add an exception to the `enterPredicate` for items that started off\n      // in this drop list. The drag ref has logic that allows an item to return to its initial\n      // container, if it has left the initial container and none of the connected containers\n      // allow it to enter. See `DragRef._updateActiveDropContainer` for more context.\n      return this.enterPredicate(item, this) || this._draggables.indexOf(item) > -1;\n    })) {\n      activeSiblings.add(sibling);\n      this._cacheParentPositions();\n      this._listenToScrollEvents();\n    }\n  }\n\n  /**\n   * Called by a connected drop list when dragging has stopped.\n   * @param sibling Sibling whose dragging has stopped.\n   */\n  _stopReceiving(sibling: DropListRef) {\n    this._activeSiblings.delete(sibling);\n    this._viewportScrollSubscription.unsubscribe();\n  }\n\n  /**\n   * Starts listening to scroll events on the viewport.\n   * Used for updating the internal state of the list.\n   */\n  private _listenToScrollEvents() {\n    this._viewportScrollSubscription = this._dragDropRegistry\n      .scrolled(this._getShadowRoot())\n      .subscribe(event => {\n        if (this.isDragging()) {\n          const scrollDifference = this._parentPositions.handleScroll(event);\n\n          if (scrollDifference) {\n            // Since we know the amount that the user has scrolled we can shift all of the\n            // client rectangles ourselves. This is cheaper than re-measuring everything and\n            // we can avoid inconsistent behavior where we might be measuring the element before\n            // its position has changed.\n            this._itemPositions.forEach(({clientRect}) => {\n              adjustClientRect(clientRect, scrollDifference.top, scrollDifference.left);\n            });\n\n            // We need two loops for this, because we want all of the cached\n            // positions to be up-to-date before we re-sort the item.\n            this._itemPositions.forEach(({drag}) => {\n              if (this._dragDropRegistry.isDragging(drag)) {\n                // We need to re-sort the item manually, because the pointer move\n                // events won't be dispatched while the user is scrolling.\n                drag._sortFromLastPointerPosition();\n              }\n            });\n          }\n        } else if (this.isReceiving()) {\n          this._cacheParentPositions();\n        }\n      });\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(): DocumentOrShadowRoot {\n    if (!this._cachedShadowRoot) {\n      const shadowRoot = _getShadowRoot(coerceElement(this.element));\n      this._cachedShadowRoot = shadowRoot || this._document;\n    }\n\n    return this._cachedShadowRoot;\n  }\n\n  /** Notifies any siblings that may potentially receive the item. */\n  private _notifyReceivingSiblings() {\n    const draggedItems = this._activeDraggables.filter(item => item.isDragging());\n    this._siblings.forEach(sibling => sibling._startReceiving(this, draggedItems));\n  }\n}\n\n\n/**\n * Finds the index of an item that matches a predicate function. Used as an equivalent\n * of `Array.prototype.findIndex` which isn't part of the standard Google typings.\n * @param array Array in which to look for matches.\n * @param predicate Function used to determine whether an item is a match.\n */\nfunction findIndex<T>(array: T[],\n                      predicate: (value: T, index: number, obj: T[]) => boolean): number {\n\n  for (let i = 0; i < array.length; i++) {\n    if (predicate(array[i], i, array)) {\n      return i;\n    }\n  }\n\n  return -1;\n}\n\n/**\n * Increments the vertical scroll position of a node.\n * @param node Node whose scroll position should change.\n * @param amount Amount of pixels that the `node` should be scrolled.\n */\nfunction incrementVerticalScroll(node: HTMLElement | Window, amount: number) {\n  if (node === window) {\n    (node as Window).scrollBy(0, amount);\n  } else {\n    // Ideally we could use `Element.scrollBy` here as well, but IE and Edge don't support it.\n    (node as HTMLElement).scrollTop += amount;\n  }\n}\n\n/**\n * Increments the horizontal scroll position of a node.\n * @param node Node whose scroll position should change.\n * @param amount Amount of pixels that the `node` should be scrolled.\n */\nfunction incrementHorizontalScroll(node: HTMLElement | Window, amount: number) {\n  if (node === window) {\n    (node as Window).scrollBy(amount, 0);\n  } else {\n    // Ideally we could use `Element.scrollBy` here as well, but IE and Edge don't support it.\n    (node as HTMLElement).scrollLeft += amount;\n  }\n}\n\n/**\n * Gets whether the vertical auto-scroll direction of a node.\n * @param clientRect Dimensions of the node.\n * @param pointerY Position of the user's pointer along the y axis.\n */\nfunction getVerticalScrollDirection(clientRect: ClientRect, pointerY: number) {\n  const {top, bottom, height} = clientRect;\n  const yThreshold = height * SCROLL_PROXIMITY_THRESHOLD;\n\n  if (pointerY >= top - yThreshold && pointerY <= top + yThreshold) {\n    return AutoScrollVerticalDirection.UP;\n  } else if (pointerY >= bottom - yThreshold && pointerY <= bottom + yThreshold) {\n    return AutoScrollVerticalDirection.DOWN;\n  }\n\n  return AutoScrollVerticalDirection.NONE;\n}\n\n/**\n * Gets whether the horizontal auto-scroll direction of a node.\n * @param clientRect Dimensions of the node.\n * @param pointerX Position of the user's pointer along the x axis.\n */\nfunction getHorizontalScrollDirection(clientRect: ClientRect, pointerX: number) {\n  const {left, right, width} = clientRect;\n  const xThreshold = width * SCROLL_PROXIMITY_THRESHOLD;\n\n  if (pointerX >= left - xThreshold && pointerX <= left + xThreshold) {\n    return AutoScrollHorizontalDirection.LEFT;\n  } else if (pointerX >= right - xThreshold && pointerX <= right + xThreshold) {\n    return AutoScrollHorizontalDirection.RIGHT;\n  }\n\n  return AutoScrollHorizontalDirection.NONE;\n}\n\n/**\n * Gets the directions in which an element node should be scrolled,\n * assuming that the user's pointer is already within it scrollable region.\n * @param element Element for which we should calculate the scroll direction.\n * @param clientRect Bounding client rectangle of the element.\n * @param pointerX Position of the user's pointer along the x axis.\n * @param pointerY Position of the user's pointer along the y axis.\n */\nfunction getElementScrollDirections(element: HTMLElement, clientRect: ClientRect, pointerX: number,\n  pointerY: number): [AutoScrollVerticalDirection, AutoScrollHorizontalDirection] {\n  const computedVertical = getVerticalScrollDirection(clientRect, pointerY);\n  const computedHorizontal = getHorizontalScrollDirection(clientRect, pointerX);\n  let verticalScrollDirection = AutoScrollVerticalDirection.NONE;\n  let horizontalScrollDirection = AutoScrollHorizontalDirection.NONE;\n\n  // Note that we here we do some extra checks for whether the element is actually scrollable in\n  // a certain direction and we only assign the scroll direction if it is. We do this so that we\n  // can allow other elements to be scrolled, if the current element can't be scrolled anymore.\n  // This allows us to handle cases where the scroll regions of two scrollable elements overlap.\n  if (computedVertical) {\n    const scrollTop = element.scrollTop;\n\n    if (computedVertical === AutoScrollVerticalDirection.UP) {\n      if (scrollTop > 0) {\n        verticalScrollDirection = AutoScrollVerticalDirection.UP;\n      }\n    } else if (element.scrollHeight - scrollTop > element.clientHeight) {\n      verticalScrollDirection = AutoScrollVerticalDirection.DOWN;\n    }\n  }\n\n  if (computedHorizontal) {\n    const scrollLeft = element.scrollLeft;\n\n    if (computedHorizontal === AutoScrollHorizontalDirection.LEFT) {\n      if (scrollLeft > 0) {\n        horizontalScrollDirection = AutoScrollHorizontalDirection.LEFT;\n      }\n    } else if (element.scrollWidth - scrollLeft > element.clientWidth) {\n      horizontalScrollDirection = AutoScrollHorizontalDirection.RIGHT;\n    }\n  }\n\n  return [verticalScrollDirection, horizontalScrollDirection];\n}\n"]} |
---|