node-red/packages/node_modules/@node-red/editor-client/src/js/ui/common/popover.js

260 lines
9.5 KiB
JavaScript
Raw Normal View History

2015-07-18 16:33:31 +02:00
/**
* Copyright JS Foundation and other contributors, http://js.foundation
2015-07-18 16:33:31 +02:00
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
**/
RED.popover = (function() {
var deltaSizes = {
"default": {
top: 10,
topTop: 30,
leftRight: 17,
2018-05-23 11:25:10 +02:00
leftLeft: 25,
leftBottom: 8,
leftTop: 11
},
"small": {
top: 6,
topTop: 20,
leftRight: 8,
leftLeft: 26,
leftBottom: 8,
leftTop: 9
}
}
2015-07-18 16:33:31 +02:00
function createPopover(options) {
var target = options.target;
var direction = options.direction || "right";
var trigger = options.trigger;
2015-07-18 16:33:31 +02:00
var content = options.content;
var delay = options.delay || { show: 750, hide: 50 };
var autoClose = options.autoClose;
var width = options.width||"auto";
var size = options.size||"default";
if (!deltaSizes[size]) {
throw new Error("Invalid RED.popover size value:",size);
}
2015-07-18 16:33:31 +02:00
var timer = null;
var active;
var div;
var openPopup = function(instant) {
2015-07-18 16:33:31 +02:00
if (active) {
var existingPopover = target.data("red-ui-popover");
if (options.tooltip && existingPopover) {
active = false;
return;
}
div = $('<div class="red-ui-popover"></div>');
if (size !== "default") {
div.addClass("red-ui-popover-size-"+size);
}
if (typeof content === 'function') {
var result = content.call(res);
2018-07-03 22:18:15 +02:00
if (result === null) {
return;
}
if (typeof result === 'string') {
div.text(result);
} else {
div.append(result);
}
} else {
div.html(content);
}
if (width !== "auto") {
div.width(width);
}
2018-07-03 22:18:15 +02:00
div.appendTo("body");
2015-07-18 16:33:31 +02:00
var targetPos = target.offset();
var targetWidth = target.outerWidth();
var targetHeight = target.outerHeight();
2015-07-18 16:33:31 +02:00
var divHeight = div.height();
var divWidth = div.width();
var viewportTop = $(window).scrollTop();
2018-10-05 18:56:42 +02:00
var viewportLeft = $(window).scrollLeft();
var viewportBottom = viewportTop + $(window).height();
2018-10-05 18:56:42 +02:00
var viewportRight = viewportLeft + $(window).width();
var top = 0;
var left = 0;
2018-10-05 18:56:42 +02:00
var d = direction;
if (d === 'right') {
top = targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top;
left = targetPos.left+targetWidth+deltaSizes[size].leftRight;
2018-10-05 18:56:42 +02:00
} else if (d === 'left') {
top = targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top;
left = targetPos.left-deltaSizes[size].leftLeft-divWidth;
2018-10-05 18:56:42 +02:00
} else if (d === 'bottom') {
top = targetPos.top+targetHeight+deltaSizes[size].top;
left = targetPos.left+targetWidth/2-divWidth/2 - deltaSizes[size].leftBottom;
2018-10-24 23:57:16 +02:00
if (left < 0) {
d = "right";
top = targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top;
left = targetPos.left+targetWidth+deltaSizes[size].leftRight;
} else if (left+divWidth > viewportRight) {
2018-10-05 18:56:42 +02:00
d = "left";
top = targetPos.top+targetHeight/2-divHeight/2-deltaSizes[size].top;
left = targetPos.left-deltaSizes[size].leftLeft-divWidth;
if (top+divHeight+targetHeight/2 + 5 > viewportBottom) {
top -= (top+divHeight+targetHeight/2 - viewportBottom + 5)
}
} else if (top+divHeight > viewportBottom) {
d = 'top';
top = targetPos.top-deltaSizes[size].topTop-divHeight;
left = targetPos.left+targetWidth/2-divWidth/2 - deltaSizes[size].leftTop;
}
2018-10-05 18:56:42 +02:00
} else if (d === 'top') {
top = targetPos.top-deltaSizes[size].topTop-divHeight;
left = targetPos.left+targetWidth/2-divWidth/2 - deltaSizes[size].leftTop;
if (top < 0) {
2018-10-05 18:56:42 +02:00
d = 'bottom';
top = targetPos.top+targetHeight+deltaSizes[size].top;
left = targetPos.left+targetWidth/2-divWidth/2 - deltaSizes[size].leftBottom;
}
}
2018-10-05 18:56:42 +02:00
div.addClass('red-ui-popover-'+d).css({top: top, left: left});
if (existingPopover) {
existingPopover.close(true);
}
target.data("red-ui-popover",res)
if (instant) {
div.show();
} else {
div.fadeIn("fast");
}
2015-07-18 16:33:31 +02:00
}
}
var closePopup = function(instant) {
$(document).off('mousedown.red-ui-popover');
2015-07-18 16:33:31 +02:00
if (!active) {
if (div) {
if (instant) {
div.remove();
} else {
div.fadeOut("fast",function() {
$(this).remove();
});
}
2015-07-18 16:33:31 +02:00
div = null;
target.removeData("red-ui-popover",res)
2015-07-18 16:33:31 +02:00
}
}
}
if (trigger === 'hover') {
target.on('mouseenter',function(e) {
2015-07-18 16:33:31 +02:00
clearTimeout(timer);
active = true;
timer = setTimeout(openPopup,delay.show);
});
2018-10-24 23:57:16 +02:00
target.on('mouseleave disabled', function(e) {
if (timer) {
clearTimeout(timer);
}
2018-10-24 23:57:16 +02:00
if (active) {
active = false;
setTimeout(closePopup,delay.hide);
}
});
} else if (trigger === 'click') {
target.on("click", function(e) {
e.preventDefault();
e.stopPropagation();
active = !active;
if (!active) {
closePopup();
} else {
openPopup();
}
});
if (autoClose) {
target.on('mouseleave disabled', function(e) {
if (timer) {
clearTimeout(timer);
}
if (active) {
active = false;
setTimeout(closePopup,autoClose);
}
});
}
} else if (trigger === 'modal') {
$(document).on('mousedown.red-ui-popover', function (event) {
var target = event.target;
while (target.nodeName !== 'BODY' && target !== div[0]) {
target = target.parentElement;
}
if (target.nodeName === 'BODY') {
active = false;
closePopup();
}
});
} else if (autoClose) {
setTimeout(function() {
active = false;
closePopup();
},autoClose);
}
2015-07-18 16:33:31 +02:00
var res = {
setContent: function(_content) {
content = _content;
return res;
},
open: function (instant) {
active = true;
openPopup(instant);
return res;
},
close: function (instant) {
active = false;
closePopup(instant);
return res;
2015-07-18 16:33:31 +02:00
}
}
return res;
}
return {
create: createPopover,
tooltip: function(target,content, action) {
var label = content;
if (action) {
label = function() {
var label = content;
var shortcut = RED.keyboard.getShortcut(action);
if (shortcut && shortcut.key) {
label = $('<span>'+content+' <span class="red-ui-popover-key">'+RED.keyboard.formatKey(shortcut.key, true)+'</span></span>');
}
return label;
}
}
2018-10-24 23:57:16 +02:00
return RED.popover.create({
tooltip: true,
target:target,
trigger: "hover",
size: "small",
direction: "bottom",
content: label,
2018-10-24 23:57:16 +02:00
delay: { show: 750, hide: 50 }
});
}
2015-07-18 16:33:31 +02:00
}
})();