blob: a772dd74bdcbb562af83d6fd2d7d8969b870cc42 [file] [log] [blame]
/*
* Copyright (C) 2019 Apple Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
*
* THIS SOFTWARE IS PROVIDED BY APPLE INC. AND ITS CONTRIBUTORS ``AS IS''
* AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO,
* THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
* PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL APPLE INC. OR ITS CONTRIBUTORS
* BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
* CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
* SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
* INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
* CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF
* THE POSSIBILITY OF SUCH DAMAGE.
*/
// DropZoneView creates an invisible drop zone when a drag enters a target element.
// There are delegate methods for deciding if the drop zone should appear, drag
// progress such as entering and leaving, and the drop itself. Clients should
// always initialize with a delegate and set a target element before showing.
WI.DropZoneView = class DropZoneView extends WI.View
{
constructor(delegate)
{
console.assert(delegate);
console.assert(typeof delegate.dropZoneShouldAppearForDragEvent === "function");
super();
this._delegate = delegate;
this._targetElement = null;
this._activelyHandlingDrag = false;
this.element.classList.add("drop-zone");
}
// Public
get delegate() { return this._delegate; }
get targetElement()
{
return this._targetElement;
}
set targetElement(element)
{
console.assert(!this._activelyHandlingDrag);
if (this._targetElement === element)
return;
if (!this._boundHandleDragEnter)
this._boundHandleDragEnter = this._handleDragEnter.bind(this);
if (this._targetElement)
this._targetElement.removeEventListener("dragenter", this._boundHandleDragEnter);
this._targetElement = element;
if (this._targetElement)
this._targetElement.addEventListener("dragenter", this._boundHandleDragEnter);
}
set text(text)
{
this.element.textContent = text;
}
// Protected
initialLayout()
{
super.initialLayout();
console.assert(this._targetElement);
this.element.addEventListener("dragover", this._handleDragOver.bind(this));
this.element.addEventListener("dragleave", this._handleDragLeave.bind(this));
this.element.addEventListener("drop", this._handleDrop.bind(this));
}
// Private
_startActiveDrag()
{
console.assert(!this._activelyHandlingDrag);
this._activelyHandlingDrag = true;
this.element.classList.add("visible");
}
_stopActiveDrag()
{
console.assert(this._activelyHandlingDrag);
this._activelyHandlingDrag = false;
this.element.classList.remove("visible");
}
_handleDragEnter(event)
{
console.assert(this.isAttached);
if (this._activelyHandlingDrag)
return;
if (!this._delegate.dropZoneShouldAppearForDragEvent(this, event))
return;
this._startActiveDrag();
if (this._delegate.dropZoneHandleDragEnter)
this._delegate.dropZoneHandleDragEnter(this, event);
}
_handleDragLeave(event)
{
if (!this._activelyHandlingDrag)
return;
this._stopActiveDrag();
if (this._delegate.dropZoneHandleDragLeave)
this._delegate.dropZoneHandleDragLeave(this, event);
}
_handleDragOver(event)
{
if (!this._activelyHandlingDrag)
return;
event.preventDefault();
event.dataTransfer.dropEffect = "copy";
}
_handleDrop(event)
{
if (!this._activelyHandlingDrag)
return;
event.preventDefault();
this._stopActiveDrag();
if (this._delegate.dropZoneHandleDrop)
this._delegate.dropZoneHandleDrop(this, event);
}
};