define([ 'jquery', '/customize/messages.js', '/common/common-util.js', '/customize/application_config.js', '/bower_components/alertifyjs/dist/js/alertify.js', '/common/notify.js', '/common/visible.js', '/common/tippy.min.js', '/common/hyperscript.js', '/bower_components/bootstrap-tokenfield/dist/bootstrap-tokenfield.js', 'css!/common/tippy.css', ], function ($, Messages, Util, AppConfig, Alertify, Notify, Visible, Tippy, h) { var UI = {}; /* * Alertifyjs */ UI.Alertify = Alertify; // set notification timeout Alertify._$$alertify.delay = AppConfig.notificationTimeout || 5000; var findCancelButton = UI.findCancelButton = function (root) { if (root) { return $(root).find('button.cancel').last(); } return $('button.cancel').last(); }; var findOKButton = UI.findOKButton = function (root) { if (root) { return $(root).find('button.ok').last(); } return $('button.ok').last(); }; var listenForKeys = UI.listenForKeys = function (yes, no) { var handler = function (e) { switch (e.which) { case 27: // cancel if (typeof(no) === 'function') { no(e); } break; case 13: // enter if (typeof(yes) === 'function') { yes(e); } break; } }; $(window).keyup(handler); return handler; }; var stopListening = UI.stopListening = function (handler) { $(window).off('keyup', handler); }; UI.alert = function (msg, cb, force) { cb = cb || function () {}; if (force !== true) { msg = Util.fixHTML(msg); } var close = function () { findOKButton().click(); }; var keyHandler = listenForKeys(close, close); Alertify .okBtn(Messages.okButton || 'OK') .alert(msg, function (ev) { cb(ev); stopListening(keyHandler); }); window.setTimeout(function () { findOKButton().focus(); if (typeof(UI.notify) === 'function') { UI.notify(); } }); }; UI.prompt = function (msg, def, cb, opt, force) { opt = opt || {}; cb = cb || function () {}; if (force !== true) { msg = Util.fixHTML(msg); } var keyHandler = listenForKeys(function () { // yes findOKButton().click(); }, function () { // no findCancelButton().click(); }); // Make sure we don't call both the "yes" and "no" handlers if we use "findOKButton().click()" // in the callback var isClicked = false; Alertify .defaultValue(def || '') .okBtn(opt.ok || Messages.okButton || 'OK') .cancelBtn(opt.cancel || Messages.cancelButton || 'Cancel') .prompt(msg, function (val, ev) { if (isClicked) { return; } isClicked = true; cb(val, ev); stopListening(keyHandler); }, function (ev) { if (isClicked) { return; } isClicked = true; cb(null, ev); stopListening(keyHandler); }); if (typeof(UI.notify) === 'function') { UI.notify(); } }; UI.confirm = function (msg, cb, opt, force, styleCB) { opt = opt || {}; cb = cb || function () {}; if (force !== true) { msg = Util.fixHTML(msg); } var keyHandler = listenForKeys(function () { findOKButton().click(); }, function () { findCancelButton().click(); }); // Make sure we don't call both the "yes" and "no" handlers if we use "findOKButton().click()" // in the callback var isClicked = false; Alertify .okBtn(opt.ok || Messages.okButton || 'OK') .cancelBtn(opt.cancel || Messages.cancelButton || 'Cancel') .confirm(msg, function () { if (isClicked) { return; } isClicked = true; cb(true); stopListening(keyHandler); }, function () { if (isClicked) { return; } isClicked = true; cb(false); stopListening(keyHandler); }); window.setTimeout(function () { var $ok = findOKButton(); var $cancel = findCancelButton(); if (opt.okClass) { $ok.addClass(opt.okClass); } if (opt.cancelClass) { $cancel.addClass(opt.cancelClass); } if (opt.reverseOrder) { $ok.insertBefore($ok.prev()); } if (typeof(styleCB) === 'function') { styleCB($ok.closest('.dialog')); } }, 0); if (typeof(UI.notify) === 'function') { UI.notify(); } }; UI.log = function (msg) { Alertify.success(Util.fixHTML(msg)); }; UI.warn = function (msg) { Alertify.error(Util.fixHTML(msg)); }; /* * spinner */ UI.spinner = function (parent) { var $target = $('', { 'class': 'fa fa-spinner fa-pulse fa-4x fa-fw' }).hide(); $(parent).append($target); return { show: function () { $target.css('display', 'inline'); return this; }, hide: function () { $target.hide(); return this; }, get: function () { return $target; }, }; }; var LOADING = 'loading'; var getRandomTip = function () { if (!Messages.tips || !Object.keys(Messages.tips).length) { return ''; } var keys = Object.keys(Messages.tips); var rdm = Math.floor(Math.random() * keys.length); return Messages.tips[keys[rdm]]; }; UI.addLoadingScreen = function (config) { config = config || {}; var loadingText = config.loadingText; var hideTips = config.hideTips; var hideLogo = config.hideLogo; var $loading, $container; if ($('#' + LOADING).length) { $loading = $('#' + LOADING).show(); if (loadingText) { $('#' + LOADING).find('p').text(loadingText); } $container = $loading.find('.loadingContainer'); } else { $loading = $('
', {id: LOADING}); $container = $('
', {'class': 'loadingContainer'}); if (!hideLogo) { $container.append(''); } var $spinner = $('
', {'class': 'spinnerContainer'}); UI.spinner($spinner).show(); var $text = $('

').text(loadingText || Messages.loading); $container.append($spinner).append($text); $loading.append($container); $('body').append($loading); } if (Messages.tips && !hideTips) { var $loadingTip = $('

', {'id': 'loadingTip'}); $('', {'class': 'tips'}).text(getRandomTip()).appendTo($loadingTip); $loadingTip.css({ 'bottom': $('body').height()/2 - $container.height()/2 + 20 + 'px' }); $('body').append($loadingTip); } }; UI.removeLoadingScreen = function (cb) { $('#' + LOADING).fadeOut(750, cb); var $tip = $('#loadingTip').css('top', '') // loading.less sets transition-delay: $wait-time // and transition: opacity $fadeout-time .css({ 'opacity': 0, 'pointer-events': 'none', }); setTimeout(function () { $tip.remove(); }, 3750); // jquery.fadeout can get stuck }; UI.errorLoadingScreen = function (error, transparent) { if (!$('#' + LOADING).is(':visible')) { UI.addLoadingScreen({hideTips: true}); } $('.spinnerContainer').hide(); if (transparent) { $('#' + LOADING).css('opacity', 0.8); } $('#' + LOADING).find('p').html(error || Messages.error); }; // Notify var notify = {}; UI.unnotify = function () { if (notify.tabNotification && typeof(notify.tabNotification.cancel) === 'function') { notify.tabNotification.cancel(); } }; UI.notify = function () { if (Visible.isSupported() && !Visible.currently()) { UI.unnotify(); notify.tabNotification = Notify.tab(1000, 10); } }; if (Visible.isSupported()) { Visible.onChange(function (yes) { if (yes) { UI.unnotify(); } }); } UI.importContent = function (type, f, cfg) { return function () { var $files = $('', {type:"file"}); if (cfg && cfg.accept) { $files.attr('accept', cfg.accept); } $files.click(); $files.on('change', function (e) { var file = e.target.files[0]; var reader = new FileReader(); reader.onload = function (e) { f(e.target.result, file); }; reader.readAsText(file, type); }); }; }; var $defaultIcon = $('', {"class": "fa fa-file-text-o"}); UI.getIcon = function (type) { var $icon = $defaultIcon.clone(); if (AppConfig.applicationsIcon && AppConfig.applicationsIcon[type]) { var appClass = ' cp-icon-color-'+type; $icon = $('', {'class': 'fa ' + AppConfig.applicationsIcon[type] + appClass}); } return $icon; }; // Tooltips UI.clearTooltips = function () { // If an element is removed from the UI while a tooltip is applied on that element, the tooltip will get hung // forever, this is a solution which just searches for tooltips which have no corrisponding element and removes // them. var win; $('.tippy-popper').each(function (i, el) { win = win || $('#pad-iframe')[0].contentWindow; if (win.$('[aria-describedby=' + el.getAttribute('id') + ']').length === 0) { el.remove(); } }); }; UI.addTooltips = function () { var MutationObserver = window.MutationObserver; var addTippy = function (el) { if (el.nodeName === 'IFRAME') { return; } var delay = typeof(AppConfig.tooltipDelay) === "number" ? AppConfig.tooltipDelay : 500; Tippy(el, { position: 'bottom', distance: 0, performance: true, dynamicTitle: true, delay: [delay, 0] }); }; var $body = $('body'); var $padIframe = $('#pad-iframe').contents().find('body'); $('[title]').each(function (i, el) { addTippy(el); }); $('#pad-iframe').contents().find('[title]').each(function (i, el) { addTippy(el); }); var observer = new MutationObserver(function(mutations) { mutations.forEach(function(mutation) { if (mutation.type === 'childList' && mutation.addedNodes.length) { $body.find('[title]').each(function (i, el) { addTippy(el); }); if (!$padIframe.length) { return; } $padIframe.find('[title]').each(function (i, el) { addTippy(el); }); } }); }); observer.observe($('body')[0], { attributes: false, childList: true, characterData: false, subtree: true }); if ($('#pad-iframe').length) { observer.observe($('#pad-iframe').contents().find('body')[0], { attributes: false, childList: true, characterData: false, subtree: true }); } }; UI.tokenField = function (target) { var t = { element: target || h('input'), }; var $t = t.tokenfield = $(t.element).tokenfield(); t.getTokens = function () { return $t.tokenfield('getTokens').map(function (token) { return token.value; }); }; t.preventDuplicates = function (cb) { $t.on('tokenfield:createtoken', function (ev) { var val; if (t.getTokens().some(function (t) { if (t === ev.attrs.value) { return ((val = t)); } })) { ev.preventDefault(); if (typeof(cb) === 'function') { cb(val); } } }); return t; }; t.setTokens = function (tokens) { $t.tokenfield('setTokens', tokens.map(function (token) { return { value: token, label: token, }; })); }; t.focus = function () { var $temp = $t.closest('.tokenfield').find('.token-input'); $temp.css('width', '20%'); $t.tokenfield('focusInput', $temp[0]); }; return t; }; var dialog = UI.dialog = {}; dialog.okButton = function () { return h('button.ok', { tabindex: '2', }, Messages.okButton); }; dialog.cancelButton = function () { return h('button.cancel', { tabindex: '1'}, Messages.cancelButton); }; dialog.message = function (text) { return h('p.message', text); }; dialog.textInput = function (opt) { return h('input', opt || { placeholder: '', type: 'text', 'class': 'cp-text-input', }); }; dialog.nav = function (content) { return h('nav', content || [ dialog.cancelButton(), dialog.okButton(), ]); }; dialog.frame = function (content) { return h('div.alertify', [ h('div.dialog', [ h('div', content), ]) ]); }; dialog.tagPrompt = function (tags, cb) { var input = dialog.textInput(); var tagger = dialog.frame([ dialog.message('make some tags'), // TODO translate input, dialog.nav(), ]); var field = UI.tokenField(input).preventDuplicates(function (val) { UI.warn('Duplicate tag: ' + val); // TODO translate }); var close = Util.once(function () { var $t = $(tagger).fadeOut(150, function () { $t.remove(); }); }); var listener = listenForKeys(function () {}, function () { close(); stopListening(listener); }); var CB = Util.once(cb); findOKButton(tagger).click(function () { var tokens = field.getTokens(); close(); CB(tokens); }); findCancelButton(tagger).click(function () { close(); CB(null); }); // :( setTimeout(function () { field.setTokens(tags); field.focus(); }); return tagger; }; return UI; });