276 lines
8.6 KiB
JavaScript
276 lines
8.6 KiB
JavaScript
/*
|
|
* Copyright (C) 2013, 2015 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.
|
|
*/
|
|
|
|
WI.HoverMenu = class HoverMenu extends WI.Object
|
|
{
|
|
constructor(delegate)
|
|
{
|
|
super();
|
|
|
|
this.delegate = delegate;
|
|
|
|
this._element = document.createElement("div");
|
|
this._element.className = "hover-menu";
|
|
this._element.addEventListener("transitionend", this, true);
|
|
|
|
this._outlineElement = this._element.appendChild(document.createElementNS("http://www.w3.org/2000/svg", "svg"));
|
|
|
|
this._button = this._element.appendChild(document.createElement("img"));
|
|
this._button.addEventListener("click", this);
|
|
}
|
|
|
|
// Public
|
|
|
|
get element()
|
|
{
|
|
return this._element;
|
|
}
|
|
|
|
present(rects)
|
|
{
|
|
this._outlineElement.textContent = "";
|
|
|
|
document.body.appendChild(this._element);
|
|
this._drawOutline(rects);
|
|
this._element.classList.add(WI.HoverMenu.VisibleClassName);
|
|
|
|
window.addEventListener("scroll", this, true);
|
|
}
|
|
|
|
dismiss(discrete)
|
|
{
|
|
if (this._element.parentNode !== document.body)
|
|
return;
|
|
|
|
if (discrete)
|
|
this._element.remove();
|
|
|
|
this._element.classList.remove(WI.HoverMenu.VisibleClassName);
|
|
|
|
window.removeEventListener("scroll", this, true);
|
|
}
|
|
|
|
// Protected
|
|
|
|
handleEvent(event)
|
|
{
|
|
switch (event.type) {
|
|
case "scroll":
|
|
if (!this._element.contains(event.target))
|
|
this.dismiss(true);
|
|
break;
|
|
case "click":
|
|
this._handleClickEvent(event);
|
|
break;
|
|
case "transitionend":
|
|
if (!this._element.classList.contains(WI.HoverMenu.VisibleClassName))
|
|
this._element.remove();
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Private
|
|
|
|
_handleClickEvent(event)
|
|
{
|
|
if (this.delegate && typeof this.delegate.hoverMenuButtonWasPressed === "function")
|
|
this.delegate.hoverMenuButtonWasPressed(this);
|
|
}
|
|
|
|
_drawOutline(rects)
|
|
{
|
|
var buttonWidth = this._button.width;
|
|
var buttonHeight = this._button.height;
|
|
|
|
// Add room for the button on the last line.
|
|
var lastRect = rects.pop();
|
|
lastRect.size.width += buttonWidth;
|
|
rects.push(lastRect);
|
|
|
|
if (rects.length === 1)
|
|
this._drawSingleLine(rects[0]);
|
|
else if (rects.length === 2 && rects[0].minX() >= rects[1].maxX())
|
|
this._drawTwoNonOverlappingLines(rects);
|
|
else
|
|
this._drawOverlappingLines(rects);
|
|
|
|
var bounds = WI.Rect.unionOfRects(rects).pad(3); // padding + 1/2 stroke-width
|
|
|
|
var style = this._element.style;
|
|
style.left = bounds.minX() + "px";
|
|
style.top = bounds.minY() + "px";
|
|
style.width = bounds.size.width + "px";
|
|
style.height = bounds.size.height + "px";
|
|
|
|
this._outlineElement.style.width = bounds.size.width + "px";
|
|
this._outlineElement.style.height = bounds.size.height + "px";
|
|
|
|
this._button.style.left = (lastRect.maxX() - bounds.minX() - buttonWidth) + "px";
|
|
this._button.style.top = (lastRect.maxY() - bounds.minY() - buttonHeight) + "px";
|
|
}
|
|
|
|
_addRect(rect)
|
|
{
|
|
var r = 4;
|
|
|
|
var svgRect = document.createElementNS("http://www.w3.org/2000/svg", "rect");
|
|
svgRect.setAttribute("x", 1);
|
|
svgRect.setAttribute("y", 1);
|
|
svgRect.setAttribute("width", rect.size.width);
|
|
svgRect.setAttribute("height", rect.size.height);
|
|
svgRect.setAttribute("rx", r);
|
|
svgRect.setAttribute("ry", r);
|
|
return this._outlineElement.appendChild(svgRect);
|
|
}
|
|
|
|
_addPath(commands, tx, ty)
|
|
{
|
|
var path = document.createElementNS("http://www.w3.org/2000/svg", "path");
|
|
path.setAttribute("d", commands.join(" "));
|
|
path.setAttribute("transform", "translate(" + (tx + 1) + "," + (ty + 1) + ")");
|
|
return this._outlineElement.appendChild(path);
|
|
}
|
|
|
|
_drawSingleLine(rect)
|
|
{
|
|
this._addRect(rect.pad(2));
|
|
}
|
|
|
|
_drawTwoNonOverlappingLines(rects)
|
|
{
|
|
var r = 4;
|
|
|
|
var firstRect = rects[0].pad(2);
|
|
var secondRect = rects[1].pad(2);
|
|
|
|
var tx = -secondRect.minX();
|
|
var ty = -firstRect.minY();
|
|
|
|
var rect = firstRect;
|
|
this._addPath([
|
|
"M", rect.maxX(), rect.minY(),
|
|
"H", rect.minX() + r,
|
|
"q", -r, 0, -r, r,
|
|
"V", rect.maxY() - r,
|
|
"q", 0, r, r, r,
|
|
"H", rect.maxX()
|
|
], tx, ty);
|
|
|
|
rect = secondRect;
|
|
this._addPath([
|
|
"M", rect.minX(), rect.minY(),
|
|
"H", rect.maxX() - r,
|
|
"q", r, 0, r, r,
|
|
"V", rect.maxY() - r,
|
|
"q", 0, r, -r, r,
|
|
"H", rect.minX()
|
|
], tx, ty);
|
|
}
|
|
|
|
_drawOverlappingLines(rects)
|
|
{
|
|
var PADDING = 2;
|
|
var r = 4;
|
|
|
|
var minX = Number.MAX_VALUE;
|
|
var maxX = -Number.MAX_VALUE;
|
|
for (var rect of rects) {
|
|
var minX = Math.min(rect.minX(), minX);
|
|
var maxX = Math.max(rect.maxX(), maxX);
|
|
}
|
|
|
|
minX -= PADDING;
|
|
maxX += PADDING;
|
|
|
|
var minY = rects[0].minY() - PADDING;
|
|
var maxY = rects.lastValue.maxY() + PADDING;
|
|
var firstLineMinX = rects[0].minX() - PADDING;
|
|
var lastLineMaxX = rects.lastValue.maxX() + PADDING;
|
|
|
|
if (firstLineMinX === minX && lastLineMaxX === maxX)
|
|
return this._addRect(new WI.Rect(minX, minY, maxX - minX, maxY - minY));
|
|
|
|
var lastLineMinY = rects.lastValue.minY() + PADDING;
|
|
if (rects[0].minX() === minX + PADDING) {
|
|
return this._addPath([
|
|
"M", minX + r, minY,
|
|
"H", maxX - r,
|
|
"q", r, 0, r, r,
|
|
"V", lastLineMinY - r,
|
|
"q", 0, r, -r, r,
|
|
"H", lastLineMaxX + r,
|
|
"q", -r, 0, -r, r,
|
|
"V", maxY - r,
|
|
"q", 0, r, -r, r,
|
|
"H", minX + r,
|
|
"q", -r, 0, -r, -r,
|
|
"V", minY + r,
|
|
"q", 0, -r, r, -r
|
|
], -minX, -minY);
|
|
}
|
|
|
|
var firstLineMaxY = rects[0].maxY() - PADDING;
|
|
if (rects.lastValue.maxX() === maxX - PADDING) {
|
|
return this._addPath([
|
|
"M", firstLineMinX + r, minY,
|
|
"H", maxX - r,
|
|
"q", r, 0, r, r,
|
|
"V", maxY - r,
|
|
"q", 0, r, -r, r,
|
|
"H", minX + r,
|
|
"q", -r, 0, -r, -r,
|
|
"V", firstLineMaxY + r,
|
|
"q", 0, -r, r, -r,
|
|
"H", firstLineMinX - r,
|
|
"q", r, 0, r, -r,
|
|
"V", minY + r,
|
|
"q", 0, -r, r, -r
|
|
], -minX, -minY);
|
|
}
|
|
|
|
return this._addPath([
|
|
"M", firstLineMinX + r, minY,
|
|
"H", maxX - r,
|
|
"q", r, 0, r, r,
|
|
"V", lastLineMinY - r,
|
|
"q", 0, r, -r, r,
|
|
"H", lastLineMaxX + r,
|
|
"q", -r, 0, -r, r,
|
|
"V", maxY - r,
|
|
"q", 0, r, -r, r,
|
|
"H", minX + r,
|
|
"q", -r, 0, -r, -r,
|
|
"V", firstLineMaxY + r,
|
|
"q", 0, -r, r, -r,
|
|
"H", firstLineMinX - r,
|
|
"q", r, 0, r, -r,
|
|
"V", minY + r,
|
|
"q", 0, -r, r, -r
|
|
], -minX, -minY);
|
|
}
|
|
};
|
|
|
|
WI.HoverMenu.VisibleClassName = "visible";
|