define([ 'jquery', '/customize/messages.js', '/common/common-util.js', '/common/common-hash.js', '/common/common-notifier.js', '/customize/application_config.js', '/bower_components/alertifyjs/dist/js/alertify.js', '/common/tippy/tippy.min.js', '/customize/pages.js', '/common/hyperscript.js', '/customize/loading.js', '/common/test.js', '/bower_components/bootstrap-tokenfield/dist/bootstrap-tokenfield.js', 'css!/common/tippy/tippy.css', ], function ($, Messages, Util, Hash, Notifier, AppConfig, Alertify, Tippy, Pages, h, Loading, Test) { 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, el) { var handler = function (e) { e.stopPropagation(); switch (e.which) { case 27: // cancel if (typeof(no) === 'function') { no(e); } break; case 13: // enter if (typeof(yes) === 'function') { yes(e); } break; } }; $(el || window).keydown(handler); return handler; }; var customListenForKeys = function (keys, cb, el) { if (!keys || !keys.length || typeof cb !== "function") { return; } var handler = function (e) { e.stopPropagation(); keys.some(function (k) { // k is number or array // if it's an array, it should be [keyCode, "{ctrl|alt|shift|meta}"] if (Array.isArray(k) && e.which === k[0] && e[k[1] + 'Key']) { cb(); return true; } if (e.which === k && !e.shiftKey && !e.altKey && !e.metaKey && !e.ctrlKey) { cb(); return true; } }); }; $(el || window).keydown(handler); return handler; }; var stopListening = UI.stopListening = function (handler) { if (!handler) { return; } // we don't want to stop all the 'keyup' listeners $(window).off('keyup', handler); }; var dialog = UI.dialog = {}; var merge = function (a, b) { var c = {}; if (a) { Object.keys(a).forEach(function (k) { c[k] = a[k]; }); } if (b) { Object.keys(b).forEach(function (k) { c[k] = b[k]; }); } return c; }; dialog.selectable = function (value, opt) { var attrs = merge({ type: 'text', readonly: 'readonly', }, opt); var input = h('input', attrs); $(input).val(value).click(function () { input.select(); }); return input; }; dialog.okButton = function (content, classString) { var sel = typeof(classString) === 'string'? 'button.ok.' + classString:'button.ok.primary'; return h(sel, { tabindex: '2', }, content || Messages.okButton); }; dialog.cancelButton = function (content, classString) { var sel = typeof(classString) === 'string'? 'button.' + classString:'button.cancel'; return h(sel, { tabindex: '1'}, content || Messages.cancelButton); }; dialog.message = function (text) { return h('p.msg', text); }; dialog.textInput = function (opt) { var attrs = merge({ type: 'text', 'class': 'cp-text-input', }, opt); return h('input', attrs); }; dialog.nav = function (content) { return h('nav', content || [ dialog.cancelButton(), dialog.okButton(), ]); }; dialog.frame = function (content) { return h('div.alertify', { tabindex: 1, }, [ h('div.dialog', [ h('div', content), ]) ]); }; /** * tabs is an array containing objects * each object must have the following attributes: * - title: String * - content: DOMElement */ dialog.tabs = function (tabs) { var contents = []; var titles = []; tabs.forEach(function (tab) { if (!tab.content || !tab.title) { return; } var content = h('div.alertify-tabs-content', tab.content); var title = h('span.alertify-tabs-title', tab.title); $(title).click(function () { titles.forEach(function (t) { $(t).removeClass('alertify-tabs-active'); }); contents.forEach(function (c) { $(c).removeClass('alertify-tabs-content-active'); }); $(title).addClass('alertify-tabs-active'); $(content).addClass('alertify-tabs-content-active'); }); titles.push(title); contents.push(content); }); if (contents.length) { $(contents[0]).addClass('alertify-tabs-content-active'); $(titles[0]).addClass('alertify-tabs-active'); } return h('div.alertify-tabs', [ h('div.alertify-tabs-titles', titles), h('div.alertify-tabs-contents', contents), ]); }; UI.tokenField = function (target) { var t = { element: target || h('input'), }; var $t = t.tokenfield = $(t.element).tokenfield(); t.getTokens = function (ignorePending) { var tokens = $t.tokenfield('getTokens').map(function (token) { return token.value.toLowerCase(); }); if (ignorePending) { return tokens; } var $pendingEl = $($t.parent().find('.token-input')[0]); var val = ($pendingEl.val() || "").trim(); if (val && tokens.indexOf(val) === -1) { return tokens.concat(val); } return tokens; }; var $root = $t.parent(); $t.on('tokenfield:removetoken', function () { $root.find('.token-input').focus(); }); t.preventDuplicates = function (cb) { $t.on('tokenfield:createtoken', function (ev) { var val; ev.attrs.value = ev.attrs.value.toLowerCase(); if (t.getTokens(true).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.toLowerCase(), label: token.toLowerCase(), }; })); }; t.focus = function () { var $temp = $t.closest('.tokenfield').find('.token-input'); $temp.css('width', '20%'); $t.tokenfield('focusInput', $temp[0]); }; return t; }; dialog.tagPrompt = function (tags, cb) { var input = dialog.textInput(); var tagger = dialog.frame([ dialog.message([ Messages.tags_add, h('br'), Messages.tags_searchHint, ]), input, h('center', h('small', Messages.tags_notShared)), dialog.nav(), ]); var field = UI.tokenField(input).preventDuplicates(function (val) { UI.warn(Messages._getKey('tags_duplicate', [val])); }); var listener; var close = Util.once(function (result, ev) { ev.stopPropagation(); ev.preventDefault(); var $frame = $(tagger).fadeOut(150, function () { stopListening(listener); $frame.remove(); cb(result, ev); }); }); var $ok = findOKButton(tagger).click(function (e) { var tokens = field.getTokens(); close(tokens, e); }); var $cancel = findCancelButton(tagger).click(function (e) { close(null, e); }); listener = listenForKeys(function () { $ok.click(); }, function () { $cancel.click(); }, tagger); $(tagger).on('click submit', function (e) { e.stopPropagation(); }); document.body.appendChild(tagger); // :( setTimeout(function () { field.setTokens(tags); field.focus(); }); return tagger; }; dialog.customModal = function (msg, opt) { var force = false; if (typeof(opt) === 'object') { force = opt.force || false; } else if (typeof(opt) === 'boolean') { force = opt; } if (typeof(opt) !== 'object') { opt = {}; } var message; if (typeof(msg) === 'string') { // sanitize if (!force) { msg = Util.fixHTML(msg); } message = dialog.message(); message.innerHTML = msg; } else { message = dialog.message(msg); } var close = function (el) { var $el = $(el).fadeOut(150, function () { $el.detach(); }); }; var navs = []; opt.buttons.forEach(function (b) { if (!b.name || !b.onClick) { return; } var button = h('button', { tabindex: '1', 'class': b.className || '' }, b.name); $(button).click(function () { b.onClick(); close($(button).parents('.alertify').first()); }); if (b.keys && b.keys.length) { $(button).attr('data-keys', JSON.stringify(b.keys)); } navs.push(button); }); var frame = h('div', [ message, dialog.nav(navs), ]); if (opt.forefront) { $(frame).addClass('forefront'); } return frame; }; UI.openCustomModal = function (content) { var frame = dialog.frame([ content ]); $(frame).find('button[data-keys]').each(function (i, el) { var keys = JSON.parse($(el).attr('data-keys')); customListenForKeys(keys, function () { if (!$(el).is(':visible')) { return; } $(el).click(); }, frame); }); document.body.appendChild(frame); $(frame).focus(); setTimeout(function () { Notifier.notify(); }); }; UI.alert = function (msg, cb, opt) { var force = false; if (typeof(opt) === 'object') { force = opt.force || false; } else if (typeof(opt) === 'boolean') { force = opt; } if (typeof(opt) !== 'object') { opt = {}; } cb = cb || function () {}; var message; if (typeof(msg) === 'string') { // sanitize if (!force) { msg = Util.fixHTML(msg); } message = dialog.message(); message.innerHTML = msg; } else { message = dialog.message(msg); } var ok = dialog.okButton(); var frame = dialog.frame([ message, dialog.nav(ok), ]); if (opt.forefront) { $(frame).addClass('forefront'); } var listener; var close = Util.once(function () { $(frame).fadeOut(150, function () { $(this).remove(); }); stopListening(listener); cb(); }); listener = listenForKeys(close, close, ok); var $ok = $(ok).click(close); document.body.appendChild(frame); setTimeout(function () { $ok.focus(); Notifier.notify(); }); }; UI.prompt = function (msg, def, cb, opt, force) { cb = cb || function () {}; opt = opt || {}; var input = dialog.textInput(); input.value = typeof(def) === 'string'? def: ''; var message; if (typeof(msg) === 'string') { if (!force) { msg = Util.fixHTML(msg); } message = dialog.message(); message.innerHTML = msg; } else { message = dialog.message(msg); } var ok = dialog.okButton(opt.ok); var cancel = dialog.cancelButton(opt.cancel); var frame = dialog.frame([ message, input, dialog.nav([ cancel, ok, ]), ]); var listener; var close = Util.once(function (result, ev) { var $frame = $(frame).fadeOut(150, function () { stopListening(listener); $frame.remove(); cb(result, ev); }); }); var $ok = $(ok).click(function (ev) { close(input.value, ev); }); var $cancel = $(cancel).click(function (ev) { close(null, ev); }); listener = listenForKeys(function () { // yes $ok.click(); }, function () { // no $cancel.click(); }, input); document.body.appendChild(frame); setTimeout(function () { $(input).select().focus(); Notifier.notify(); }); }; UI.confirm = function (msg, cb, opt, force) { cb = cb || function () {}; opt = opt || {}; var message; if (typeof(msg) === 'string') { if (!force) { msg = Util.fixHTML(msg); } message = dialog.message(); message.innerHTML = msg; } else { message = dialog.message(msg); } var ok = dialog.okButton(opt.ok, opt.okClass); var cancel = dialog.cancelButton(opt.cancel, opt.cancelClass); var frame = dialog.frame([ message, dialog.nav(opt.reverseOrder? [ok, cancel]: [cancel, ok]), ]); var listener; var close = Util.once(function (bool, ev) { $(frame).fadeOut(150, function () { $(this).remove(); }); stopListening(listener); cb(bool, ev); }); var $ok = $(ok).click(function (ev) { close(true, ev); }); var $cancel = $(cancel).click(function (ev) { close(false, ev); }); if (opt.cancelClass) { $cancel.addClass(opt.cancelClass); } if (opt.okClass) { $ok.addClass(opt.okClass); } listener = listenForKeys(function () { $ok.click(); }, function () { $cancel.click(); }, ok); document.body.appendChild(frame); setTimeout(function () { Notifier.notify(); $(frame).find('.ok').focus(); if (typeof(opt.done) === 'function') { opt.done($ok.closest('.dialog')); } }); }; UI.log = function (msg) { Alertify.success(Util.fixHTML(msg)); }; UI.warn = function (msg) { Alertify.error(Util.fixHTML(msg)); }; UI.passwordInput = function (opts, displayEye) { opts = opts || {}; var attributes = merge({ type: 'password' }, opts); var input = h('input.cp-password-input', attributes); var reveal = UI.createCheckbox('cp-password-reveal', Messages.password_show); var eye = h('span.fa.fa-eye.cp-password-reveal'); $(reveal).find('input').on('change', function () { if($(this).is(':checked')) { $(input).prop('type', 'text'); $(input).focus(); return; } $(input).prop('type', 'password'); $(input).focus(); }); $(eye).mousedown(function () { $(input).prop('type', 'text'); $(input).focus(); }).mouseup(function(){ $(input).prop('type', 'password'); $(input).focus(); }).mouseout(function(){ $(input).prop('type', 'password'); $(input).focus(); }); if (displayEye) { $(reveal).hide(); } else { $(eye).hide(); } return h('span.cp-password-container', [ input, reveal, eye ]); }; /* * spinner */ UI.spinner = function (parent) { var $target = $('', { 'class': 'fa fa-circle-o-notch fa-spin 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 = 'cp-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]]; };*/ var loading = { error: false, driveState: 0, padState: 0 }; UI.addLoadingScreen = function (config) { config = config || {}; var loadingText = config.loadingText; var todo = function () { var $loading = $('#' + LOADING); //.show(); $loading.css('display', ''); $loading.removeClass('cp-loading-hidden'); $('.cp-loading-spinner-container').show(); if (!config.noProgress && !$loading.find('.cp-loading-progress').length) { var progress = h('div.cp-loading-progress', [ h('p.cp-loading-progress-drive'), h('p.cp-loading-progress-pad') ]); $loading.find('.cp-loading-container').append(progress); } else if (config.noProgress) { $loading.find('.cp-loading-progress').remove(); } if (loadingText) { $('#' + LOADING).find('#cp-loading-message').show().text(loadingText); } else { $('#' + LOADING).find('#cp-loading-message').hide().text(''); } loading.error = false; }; if ($('#' + LOADING).length) { todo(); } else { Loading(); todo(); } }; UI.updateLoadingProgress = function (data, isDrive) { var $loading = $('#' + LOADING); if (!$loading.length || loading.error) { return; } var $progress; if (isDrive) { // Drive state if (loading.driveState === -1) { return; } // Already loaded $progress = $loading.find('.cp-loading-progress-drive'); if (!$progress.length) { return; } // Can't find the box to display data // If state is -1, remove the box, drive is loaded if (data.state === -1) { loading.driveState = -1; $progress.remove(); } else { if (data.state < loading.driveState) { return; } // We should not display old data // Update the current state loading.driveState = data.state; data.progress = data.progress || 100; data.msg = Messages['loading_drive_'+data.state] || ''; $progress.html(data.msg); if (data.progress) { $progress.append(h('div.cp-loading-progress-bar', [ h('div.cp-loading-progress-bar-value', {style: 'width:'+data.progress+'%;'}) ])); } } } else { // Pad state if (loading.padState === -1) { return; } // Already loaded $progress = $loading.find('.cp-loading-progress-pad'); if (!$progress.length) { return; } // Can't find the box to display data // If state is -1, remove the box, pad is loaded if (data.state === -1) { loading.padState = -1; $progress.remove(); } else { if (data.state < loading.padState) { return; } // We should not display old data // Update the current state loading.padState = data.state; data.progress = data.progress || 100; data.msg = Messages['loading_pad_'+data.state] || ''; $progress.html(data.msg); if (data.progress) { $progress.append(h('div.cp-loading-progress-bar', [ h('div.cp-loading-progress-bar-value', {style: 'width:'+data.progress+'%;'}) ])); } } } }; UI.removeLoadingScreen = function (cb) { // Release the test blocker, hopefully every test has been registered. // This test is created in sframe-boot2.js cb = cb || function () {}; if (Test.__ASYNC_BLOCKER__) { Test.__ASYNC_BLOCKER__.pass(); } $('#' + LOADING).addClass("cp-loading-hidden"); setTimeout(cb, 750); loading.error = false; var $tip = $('#cp-loading-tip').css('top', '') // loading.less sets transition-delay: $wait-time // and transition: opacity $fadeout-time .css({ 'opacity': 0, 'pointer-events': 'none', }); window.setTimeout(function () { $tip.remove(); }, 3750); // jquery.fadeout can get stuck }; UI.errorLoadingScreen = function (error, transparent, exitable) { var $loading = $('#' + LOADING); if (!$loading.is(':visible') || $loading.hasClass('cp-loading-hidden')) { UI.addLoadingScreen({hideTips: true}); } loading.error = true; $loading.find('.cp-loading-progress').remove(); $('.cp-loading-spinner-container').hide(); $('#cp-loading-tip').remove(); if (transparent) { $loading.css('opacity', 0.9); } var $error = $loading.find('#cp-loading-message').show(); if (error instanceof Element) { $error.html('').append(error); } else { $error.html(error || Messages.error); } if (exitable) { $(window).focus(); $(window).keydown(function (e) { if (e.which === 27) { $loading.hide(); loading.error = false; if (typeof(exitable) === "function") { exitable(); } } }); } }; 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 cp-icon-color-'+type; $icon = $('', {'class': 'fa ' + AppConfig.applicationsIcon[type] + appClass}); } return $icon; }; UI.getFileIcon = function (data) { var $icon = UI.getIcon(); if (!data) { return $icon; } var href = data.href; var type = data.type; if (!href && !type) { return $icon; } if (!type) { type = Hash.parsePadUrl(href).type; } $icon = UI.getIcon(type); 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. $('.tippy-popper').each(function (i, el) { if ($('[aria-describedby=' + el.getAttribute('id') + ']').length === 0) { el.remove(); } }); }; var delay = typeof(AppConfig.tooltipDelay) === "number" ? AppConfig.tooltipDelay : 500; $.extend(true, Tippy.defaults, { placement: 'bottom', performance: true, delay: [delay, 0], //sticky: true, theme: 'cryptpad', arrow: true, maxWidth: '200px', flip: true, popperOptions: { modifiers: { preventOverflow: { boundariesElement: 'window' } } }, //arrowType: 'round', dynamicTitle: true, arrowTransform: 'scale(2)', zIndex: 100000001 }); UI.addTooltips = function () { var MutationObserver = window.MutationObserver; var addTippy = function (i, el) { if (el._tippy) { return; } if (el.nodeName === 'IFRAME') { return; } var opts = { distance: 15 }; Array.prototype.slice.apply(el.attributes).filter(function (obj) { return /^data-tippy-/.test(obj.name); }).forEach(function (obj) { opts[obj.name.slice(11)] = obj.value; }); Tippy(el, opts); }; // This is the robust solution to remove dangling tooltips // The mutation observer does not always find removed nodes. //setInterval(UI.clearTooltips, delay); var checkRemoved = function (x) { var out = false; var xId = $(x).attr('aria-describedby'); if (xId) { if (xId.indexOf('tippy-tooltip-') === 0) { return true; } } $(x).find('[aria-describedby]').each(function (i, el) { var id = el.getAttribute('aria-describedby'); if (id.indexOf('tippy-tooltip-') !== 0) { return; } out = true; }); return out; }; $('[title]').each(addTippy); var observer = new MutationObserver(function(mutations) { var removed = false; mutations.forEach(function(mutation) { if (mutation.type === "childList") { for (var i = 0; i < mutation.addedNodes.length; i++) { if ($(mutation.addedNodes[i]).attr('title')) { addTippy(0, mutation.addedNodes[i]); } $(mutation.addedNodes[i]).find('[title]').each(addTippy); } for (var j = 0; j < mutation.removedNodes.length; j++) { removed |= checkRemoved(mutation.removedNodes[j]); } } if (mutation.type === "attributes" && mutation.attributeName === "title") { addTippy(0, mutation.target); } }); if (removed) { UI.clearTooltips(); } }); observer.observe($('body')[0], { attributes: true, childList: true, characterData: false, subtree: true }); }; UI.createCheckbox = Pages.createCheckbox; UI.createRadio = Pages.createRadio; return UI; });