chromium/chrome/browser/resources/commerce/product_specifications/drag_and_drop_manager.ts

// Copyright 2024 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.

import {assert} from 'chrome://resources/js/assert.js';
import {EventTracker} from 'chrome://resources/js/event_tracker.js';
import type {DomRepeat} from 'chrome://resources/polymer/v3_0/polymer/lib/elements/dom-repeat.js';

import type {TableElement} from './table.js';
import {$$} from './utils.js';

/**
 * @fileoverview
 * DragAndDropManager facilitates drag-and-drop interactions within a
 * `product-specifications-table` HTMLElement, allowing users to reorder columns
 * by dragging them over one another.
 *
 * Note: DragAndDropManager relies on the table being a CSS Grid.
 *
 * Upon the table's owner document firing a 'dragstart' event,
 * DragAndDropManager adds listeners to the owner document and table for
 * relevant drag events.
 *
 * When the dragging column hovers over another column, its potential new
 * position is visually indicated by updating its 'order' CSS property.
 *
 * Releasing the dragging column over a valid drop target (another column),
 * notifies the `product-specifications-table` to update its internal column
 * order and, subsequently, their DOM positions. Listeners are then removed from
 * the document and table for non-'dragstart' drag events.
 *
 * Leaving the table mid-drag drops the dragging column at the position where
 * the 'dragleave' event was triggered.
 *
 * Once drag-and-drop ends regardless of its outcome (success, failure, or
 * cancellation), each column's CSS 'order' property is removed, to ensure the
 * visual layout matches the underlying data structure.
 */

function getColumnByComposedPath(path?: EventTarget[]): HTMLElement|null {
  if (!path) {
    return null;
  }

  for (let i = 0; i < path!.length; i++) {
    const element = path![i] as Element;
    if (element.className === 'col') {
      return path![i] as HTMLElement;
    }
  }
  return null;
}

function getVisualOrderIndex(col: HTMLElement) {
  assert(col.style.order !== '');
  return parseInt(col.style.order);
}

function syncVisualOrderWithDOM(columnElements: HTMLElement[]) {
  columnElements.forEach((column, index) => {
    // `is-first-column` ensures the first column has necessary styling.
    column.toggleAttribute('is-first-column', index === 0);
    column.style.order = `${index}`;
  });
}

function resetVisualOrder(columnElements: HTMLElement[]) {
  columnElements.forEach((column) => {
    column.style.order = '';
  });
}

export class DragAndDropManager {
  private dragImage_: HTMLImageElement;
  private eventTracker_: EventTracker = new EventTracker();
  private tableElement_: TableElement;

  private get columnElements_(): HTMLElement[] {
    const table = $$<HTMLElement>(this.tableElement_, '#table');
    assert(table);
    return Array.from(
        table.querySelectorAll<HTMLElement>('.col:not([hidden])'));
  }

  init(tableElement: TableElement): void {
    this.tableElement_ = tableElement;
    // Create a transparent 1x1 pixel image that will replace the default drag
    // "ghost" image. The image is preloaded to ensure it's available when
    // dragging starts.
    this.dragImage_ = new Image(1, 1);
    this.dragImage_.src =
        '' +
        'ABAAEAAAICTAEAOw==';
    this.eventTracker_.add(
        document, 'dragstart', (e: DragEvent) => this.onDragStart_(e));
  }

  destroy(): void {
    this.eventTracker_.removeAll();
  }

  private onDragStart_(e: DragEvent) {
    if (e.dataTransfer) {
      // Replace the ghost image that appears when dragging with a transparent
      // 1x1 pixel image.
      e.dataTransfer.setDragImage(this.dragImage_, 0, 0);
    }

    const dragElement = getColumnByComposedPath(e.composedPath());
    if (!dragElement) {
      return;
    }

    this.dragStart_(dragElement);

    this.eventTracker_.add(
        document, 'dragover', (e: DragEvent) => this.dragOver_(e));
    this.eventTracker_.add(document, 'drop', (e: DragEvent) => this.drop_(e));
    // Drops the dragging column if it leaves the table. This ensures
    // that 'dragover' events only fire for adjacent columns.
    this.eventTracker_.add(
        this.tableElement_, 'dragleave', (e: DragEvent) => this.drop_(e));
    this.eventTracker_.add(document, 'dragend', () => {
      this.eventTracker_.remove(document, 'dragover');
      this.eventTracker_.remove(document, 'drop');
      this.eventTracker_.remove(this.tableElement_, 'dragleave');
      this.dragEnd_();
    });
  }

  // Sets up column reordering for drag events.
  private dragStart_(dragElement: HTMLElement) {
    this.tableElement_.draggingColumn = dragElement;
    // Set initial column order for later visual reordering.
    syncVisualOrderWithDOM(this.columnElements_);
  }

  // Swaps the visual order of the dragging column with the adjacent column it
  // drags over.
  // Note: DOM node positions are unaffected.
  // TODO(b/354729553): Handle fast column drags.
  private dragOver_(e: DragEvent) {
    e.preventDefault();

    const dragElement = this.tableElement_.draggingColumn;
    if (!dragElement) {
      return;
    }

    const dropTarget = getColumnByComposedPath(e.composedPath());
    if (!dropTarget) {
      return;
    }

    const fromIndex = getVisualOrderIndex(dragElement);
    const toIndex = getVisualOrderIndex(dropTarget);
    if (toIndex !== fromIndex) {
      if (e.dataTransfer) {
        e.dataTransfer.dropEffect = 'move';
      }
      dropTarget.style.order = `${fromIndex}`;
      dragElement.style.order = `${toIndex}`;
      // `is-first-column` ensures the first column has necessary styling.
      dropTarget.toggleAttribute('is-first-column', fromIndex === 0);
      dragElement.toggleAttribute('is-first-column', toIndex === 0);
    }
  }

  // Updates DOM node positions to reflect drag/drop result.
  private drop_(e: DragEvent) {
    const dragElement = this.tableElement_.draggingColumn;
    if (!dragElement) {
      return;
    }

    const dropTarget = getColumnByComposedPath(e.composedPath());
    if (!dropTarget) {
      return;
    }

    const columnRepeat = $$<DomRepeat>(this.tableElement_, '#columnRepeat');
    assert(columnRepeat);
    const fromIndex = (columnRepeat.modelForElement(dragElement) as unknown as {
                        columnIndex: number,
                      }).columnIndex;
    const toIndex = getVisualOrderIndex(dropTarget);
    if (toIndex !== fromIndex) {
      this.tableElement_.moveColumnOnDrop(fromIndex, toIndex);
    }

    this.dragEnd_();
  }

  // Called when drag-and-drop is finished, even if canceled.
  private dragEnd_() {
    if (!this.tableElement_.draggingColumn) {
      return;
    }

    this.tableElement_.draggingColumn = null;
    // Remove each column's CSS 'order' property, to ensure the visual
    // layout matches the underlying data structure.
    resetVisualOrder(this.columnElements_);
  }
}