/* global Handlebars */ Handlebars.registerHelper('score', function() { if(this.score) { var score = Math.round( this.score * 10 ); var imageName = 'rating/s' + score + '.svg'; return new Handlebars.SafeString(''); } return new Handlebars.SafeString(''); }); Handlebars.registerHelper('level', function() { if(typeof this.level !== 'undefined') { if(this.level === 200) { return new Handlebars.SafeString('' + t('settings', 'Official') + ''); } } }); OC.Settings = OC.Settings || {}; OC.Settings.Apps = OC.Settings.Apps || { markedOptions: {}, setupGroupsSelect: function($elements) { OC.Settings.setupGroupsSelect($elements, { placeholder: t('core', 'All') }); }, State: { currentCategory: null, currentCategoryElements: null, apps: null, $updateNotification: null, availableUpdates: 0 }, loadCategories: function() { if (this._loadCategoriesCall) { this._loadCategoriesCall.abort(); } var categories = [ {displayName: t('settings', 'Enabled apps'), ident: 'enabled', id: '0'}, {displayName: t('settings', 'Disabled apps'), ident: 'disabled', id: '1'}, {displayName: t('settings', 'Your apps'), ident: 'installed', id: '2'} ]; var source = $("#categories-template").html(); var template = Handlebars.compile(source); var html = template(categories); $('#apps-categories').html(html); OC.Settings.Apps.loadCategory($('#app-navigation').attr('data-category')); this._loadCategoriesCall = $.ajax(OC.generateUrl('settings/apps/categories'), { data:{}, type:'GET', success:function (jsondata) { var html = template(jsondata); $('#apps-categories').html(html); $('#app-category-' + OC.Settings.Apps.State.currentCategory).addClass('active'); }, complete: function() { $('#app-navigation').removeClass('icon-loading'); } }); }, loadCategory: function(categoryId) { if (OC.Settings.Apps.State.currentCategory === categoryId) { return; } if (this._loadCategoryCall) { this._loadCategoryCall.abort(); } $('#app-content').addClass('icon-loading'); $('#apps-list') .removeClass('hidden') .html(''); $('#apps-list-empty').addClass('hidden'); $('#app-category-' + OC.Settings.Apps.State.currentCategory).removeClass('active'); $('#app-category-' + categoryId).addClass('active'); OC.Settings.Apps.State.currentCategory = categoryId; OC.Settings.Apps.State.availableUpdates = 0; this._loadCategoryCall = $.ajax(OC.generateUrl('settings/apps/list?category={categoryId}', { categoryId: categoryId }), { type:'GET', success: function (apps) { OC.Settings.Apps.State.currentCategoryElements = apps.apps; var appListWithIndex = _.indexBy(apps.apps, 'id'); OC.Settings.Apps.State.apps = appListWithIndex; var appList = _.map(appListWithIndex, function(app) { // default values for missing fields return _.extend({level: 0}, app); }); var source; if (categoryId === 'enabled' || categoryId === 'disabled' || categoryId === 'installed' || categoryId === 'app-bundles') { source = $("#app-template-installed").html(); $('#apps-list').addClass('installed'); } else { source = $("#app-template").html(); $('#apps-list').removeClass('installed'); } var template = Handlebars.compile(source); if (appList.length) { appList.sort(function(a,b) { if (a.active !== b.active) { return (a.active ? -1 : 1) } else { var levelDiff = b.level - a.level; if (levelDiff === 0) { return OC.Util.naturalSortCompare(a.name, b.name); } return levelDiff; } }); var firstExperimental = false; _.each(appList, function(app) { if(app.level === 0 && firstExperimental === false) { firstExperimental = true; OC.Settings.Apps.renderApp(app, template, null, true); } else { OC.Settings.Apps.renderApp(app, template, null, false); } if (app.update) { var $update = $('#app-' + app.id + ' .update'); $update.removeClass('hidden'); $update.val(t('settings', 'Update to %s').replace(/%s/g, app.update)); OC.Settings.Apps.State.availableUpdates++; } }); if (OC.Settings.Apps.State.availableUpdates > 0) { OC.Settings.Apps.State.$updateNotification = OC.Notification.show(n('settings', 'You have %n app update pending', 'You have %n app updates pending', OC.Settings.Apps.State.availableUpdates)); } } else { $('#apps-list').addClass('hidden'); $('#apps-list-empty').removeClass('hidden').find('h2').text(t('settings', 'No apps found for your version')); } $('.enable.needs-download').tooltip({ title: t('settings', 'The app will be downloaded from the app store'), placement: 'bottom', container: 'body' }); $('.app-level .official').tooltip({ title: t('settings', 'Official apps are developed by and within the community. They offer central functionality and are ready for production use.'), placement: 'bottom', container: 'body' }); $('.app-level .approved').tooltip({ title: t('settings', 'Approved apps are developed by trusted developers and have passed a cursory security check. They are actively maintained in an open code repository and their maintainers deem them to be stable for casual to normal use.'), placement: 'bottom', container: 'body' }); $('.app-level .experimental').tooltip({ title: t('settings', 'This app is not checked for security issues and is new or known to be unstable. Install at your own risk.'), placement: 'bottom', container: 'body' }); }, complete: function() { $('#app-content').removeClass('icon-loading'); } }); }, renderApp: function(app, template, selector, firstExperimental) { if (!template) { var source = $("#app-template").html(); template = Handlebars.compile(source); } if (typeof app === 'string') { app = OC.Settings.Apps.State.apps[app]; } app.firstExperimental = firstExperimental; if (!app.preview) { app.preview = OC.imagePath('core', 'places/default-app-icon'); app.previewAsIcon = true; } if (_.isArray(app.author)) { var authors = []; _.each(app.author, function (author) { if (typeof author === 'string') { authors.push(author); } else { authors.push(author['@value']); } }); app.author = authors.join(', '); } else if (typeof app.author !== 'string') { app.author = app.author['@value']; } // Parse markdown in app description app.description = DOMPurify.sanitize( marked(app.description.trim(), OC.Settings.Apps.markedOptions), { SAFE_FOR_JQUERY: true, ALLOWED_TAGS: [ 'strong', 'p', 'a', 'ul', 'ol', 'li', 'em', 'del', 'blockquote' ] } ); var html = template(app); if (selector) { selector.html(html); } else { $('#apps-list').append(html); } var page = $('#app-' + app.id); // image loading kung-fu (IE doesn't properly scale SVGs, so disable app icons) if (app.preview && !OC.Util.isIE()) { var currentImage = new Image(); currentImage.src = app.preview; currentImage.onload = function() { /* Trigger color inversion for placeholder image too */ if(app.previewAsIcon) { page.find('.app-image') .append(OC.Settings.Apps.imageUrl(app.preview, false)) .removeClass('icon-loading'); } else { page.find('.app-image') .append(OC.Settings.Apps.imageUrl(app.preview, app.fromAppStore)) .removeClass('icon-loading'); } }; } // set group select properly if(OC.Settings.Apps.isType(app, 'filesystem') || OC.Settings.Apps.isType(app, 'prelogin') || OC.Settings.Apps.isType(app, 'authentication') || OC.Settings.Apps.isType(app, 'logging') || OC.Settings.Apps.isType(app, 'prevent_group_restriction')) { page.find(".groups-enable").hide(); page.find(".groups-enable__checkbox").prop('checked', false); } else { page.find('#group_select').val((app.groups || []).join('|')); if (app.active) { if (app.groups.length) { OC.Settings.Apps.setupGroupsSelect(page.find('#group_select')); page.find(".groups-enable__checkbox").prop('checked', true); } else { page.find(".groups-enable__checkbox").prop('checked', false); } page.find(".groups-enable").show(); } else { page.find(".groups-enable").hide(); } } }, /** * Returns the image for apps listing * url : the url of the image * appfromstore: bool to check whether the app is fetched from store or not. */ imageUrl : function (url, appfromstore) { var img; if (appfromstore) { img = ''; } else { img = ''; } return img; }, isType: function(app, type){ return app.types && app.types.indexOf(type) !== -1; }, /** * Checks the server health. * * If the promise fails, the server is broken. * * @return {Promise} promise */ _checkServerHealth: function() { return $.get(OC.generateUrl('apps/files')); }, enableAppBundle:function(bundleId, active, element, groups) { if (OC.PasswordConfirmation.requiresPasswordConfirmation()) { OC.PasswordConfirmation.requirePasswordConfirmation(_.bind(this.enableAppBundle, this, bundleId, active, element, groups)); return; } var bundles = OC.Settings.Apps.State.currentCategoryElements; bundles.forEach(function(bundle) { if(bundle['id'] === bundleId) { OC.Settings.Apps.enableApp(bundle['apps'], active, element, groups); } }); }, enableApp:function(appId, active, element, groups) { if (OC.PasswordConfirmation.requiresPasswordConfirmation()) { OC.PasswordConfirmation.requirePasswordConfirmation(_.bind(this.enableApp, this, appId, active, element, groups)); return; } var self = this; OC.Settings.Apps.hideErrorMessage(appId); groups = groups || []; var appItem = $('div#app-'+appId+''); if(active && !groups.length) { element.val(t('settings','Disabling app …')); $.post(OC.filePath('settings','ajax','disableapp.php'),{appid:appId},function(result) { if(!result || result.status !== 'success') { if (result.data && result.data.message) { OC.Settings.Apps.showErrorMessage(appId, result.data.message); appItem.data('errormsg', result.data.message); } else { OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while disabling app')); appItem.data('errormsg', t('settings', 'Error while disabling app')); } element.val(t('settings','Disable')); appItem.addClass('appwarning'); } else { OC.Settings.Apps.rebuildNavigation(); appItem.data('active',false); appItem.data('groups', ''); element.data('active',false); appItem.removeClass('active'); element.val(t('settings','Enable')); element.parent().find(".groups-enable").hide(); element.parent().find('#group_select').hide().val(null); OC.Settings.Apps.State.apps[appId].active = false; } },'json'); } else { // TODO: display message to admin to not refresh the page! // TODO: lock UI to prevent further operations element.val(t('settings','Enabling app …')); var appIdArray = []; if( typeof appId === 'string' ) { appIdArray = [appId]; } else { appIdArray = appId; } $.post(OC.filePath('settings','ajax','enableapp.php'),{appIds: appIdArray, groups: groups},function(result) { if(!result || result.status !== 'success') { if (result.data && result.data.message) { OC.Settings.Apps.showErrorMessage(appId, result.data.message); appItem.data('errormsg', result.data.message); } else { OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while enabling app')); appItem.data('errormsg', t('settings', 'Error while disabling app')); } element.val(t('settings','Enable')); appItem.addClass('appwarning'); } else { self._checkServerHealth().done(function() { if (result.data.update_required) { OC.Settings.Apps.showReloadMessage(); setTimeout(function() { location.reload(); }, 5000); } OC.Settings.Apps.rebuildNavigation(); appItem.data('active',true); element.data('active',true); appItem.addClass('active'); element.val(t('settings','Disable')); var app = OC.Settings.Apps.State.apps[appId]; app.active = true; if (OC.Settings.Apps.isType(app, 'filesystem') || OC.Settings.Apps.isType(app, 'prelogin') || OC.Settings.Apps.isType(app, 'authentication') || OC.Settings.Apps.isType(app, 'logging')) { element.parent().find(".groups-enable").prop('checked', true); element.parent().find(".groups-enable").hide(); element.parent().find('#group_select').hide().val(null); } else { element.parent().find("#groups-enable").show(); if (groups) { appItem.data('groups', JSON.stringify(groups)); } else { appItem.data('groups', ''); } } }).fail(function() { // server borked, emergency disable app $.post(OC.webroot + '/index.php/disableapp', {appid: appId}, function() { OC.Settings.Apps.showErrorMessage( appId, t('settings', 'Error: this app cannot be enabled because it makes the server unstable') ); appItem.data('errormsg', t('settings', 'Error while enabling app')); element.val(t('settings','Enable')); appItem.addClass('appwarning'); }).fail(function() { OC.Settings.Apps.showErrorMessage( appId, t('settings', 'Error: could not disable broken app') ); appItem.data('errormsg', t('settings', 'Error while disabling broken app')); element.val(t('settings','Enable')); }); }); } },'json') .fail(function() { OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while enabling app')); appItem.data('errormsg', t('settings', 'Error while enabling app')); appItem.data('active',false); appItem.addClass('appwarning'); element.val(t('settings','Enable')); }); } }, updateApp:function(appId, element) { var oldButtonText = element.val(); element.val(t('settings','Updating....')); OC.Settings.Apps.hideErrorMessage(appId); $.post(OC.filePath('settings','ajax','updateapp.php'),{appid:appId},function(result) { if(!result || result.status !== 'success') { if (result.data && result.data.message) { OC.Settings.Apps.showErrorMessage(appId, result.data.message); } else { OC.Settings.Apps.showErrorMessage(appId, t('settings','Error while updating app')); } element.val(oldButtonText); } else { element.val(t('settings','Updated')); element.hide(); var $update = $('#app-' + appId + ' .update'); $update.addClass('hidden'); var $version = $('#app-' + appId + ' .app-version'); $version.text(OC.Settings.Apps.State.apps[appId]['update']); if (OC.Settings.Apps.State.$updateNotification) { OC.Notification.hide(OC.Settings.Apps.State.$updateNotification); } OC.Settings.Apps.State.availableUpdates--; if (OC.Settings.Apps.State.availableUpdates > 0) { OC.Settings.Apps.State.$updateNotification = OC.Notification.show(n('settings', 'You have %n app update pending', 'You have %n app updates pending', OC.Settings.Apps.State.availableUpdates)); } } },'json'); }, uninstallApp:function(appId, element) { if (OC.PasswordConfirmation.requiresPasswordConfirmation()) { OC.PasswordConfirmation.requirePasswordConfirmation(_.bind(this.uninstallApp, this, appId, element)); return; } OC.Settings.Apps.hideErrorMessage(appId); element.val(t('settings','Removing …')); $.post(OC.filePath('settings','ajax','uninstallapp.php'),{appid:appId},function(result) { if(!result || result.status !== 'success') { OC.Settings.Apps.showErrorMessage(appId, t('settings','Error while removing app')); element.val(t('settings','Remove')); } else { OC.Settings.Apps.rebuildNavigation(); element.parents('#apps-list > .section').fadeOut(function() { this.remove(); }); } },'json'); }, rebuildNavigation: function() { $.getJSON(OC.filePath('settings', 'ajax', 'navigationdetect.php')).done(function(response){ if(response.status === 'success') { var addedApps = {}; var navEntries = response.nav_entries; var container = $('#apps ul'); // remove disabled apps for (var i = 0; i < navEntries.length; i++) { var entry = navEntries[i]; if(container.children('li[data-id="' + entry.id + '"]').length === 0) { addedApps[entry.id] = true; } } container.children('li[data-id]').each(function (index, el) { var id = $(el).data('id'); // remove all apps that are not in the correct order if (!navEntries[index] || (navEntries[index] && navEntries[index].id !== $(el).data('id'))) { $(el).remove(); $('#appmenu li[data-id='+id+']').remove(); } }); var previousEntry = {}; // add enabled apps to #navigation and #appmenu for (var i = 0; i < navEntries.length; i++) { var entry = navEntries[i]; if (container.children('li[data-id="' + entry.id + '"]').length === 0) { var li = $('
'); li.attr('data-id', entry.id); var img = ''; var a = $('').attr('href', entry.href); var filename = $(''); var loading = $('').css('display', 'none'); filename.text(entry.name); a.prepend(filename); a.prepend(loading); a.prepend(img); li.append(a); $('#navigation li[data-id=' + previousEntry.id + ']').after(li); // draw attention to the newly added app entry // by flashing it twice if(addedApps[entry.id]) { $('#header .menutoggle') .animate({opacity: 0.5}) .animate({opacity: 1}) .animate({opacity: 0.5}) .animate({opacity: 1}) .animate({opacity: 0.75}); } } if ($('#appmenu').children('li[data-id="' + entry.id + '"]').length === 0) { var li = $(''); li.attr('data-id', entry.id); var img = ''; var a = $('').attr('href', entry.href); var filename = $(''); var loading = $('').css('display', 'none'); filename.text(entry.name); a.prepend(filename); a.prepend(loading); a.prepend(img); li.append(a); $('#appmenu li[data-id='+ previousEntry.id+']').after(li); if(addedApps[entry.id]) { li.animate({opacity: 0.5}) .animate({opacity: 1}) .animate({opacity: 0.5}) .animate({opacity: 1}); } } previousEntry = entry; } $(window).trigger('resize'); } }); }, showErrorMessage: function(appId, message) { $('div#app-'+appId+' .warning') .show() .text(message); }, hideErrorMessage: function(appId) { $('div#app-'+appId+' .warning') .hide() .text(''); }, showReloadMessage: function() { OC.dialogs.info( t( 'settings', 'The app has been enabled but needs to be updated. You will be redirected to the update page in 5 seconds.' ), t('settings','App update'), function () { window.location.reload(); }, true ); }, /** * Splits the query by spaces and tries to find all substring in the app * @param {string} string * @param {string} query * @returns {boolean} */ _search: function(string, query) { var keywords = query.split(' '), stringLower = string.toLowerCase(), found = true; _.each(keywords, function(keyword) { found = found && stringLower.indexOf(keyword) !== -1; }); return found; }, filter: function(query) { var $appList = $('#apps-list'), $emptyList = $('#apps-list-empty'); $appList.removeClass('hidden'); $appList.find('.section').removeClass('hidden'); $emptyList.addClass('hidden'); if (query === '') { return; } query = query.toLowerCase(); $appList.find('.section').addClass('hidden'); // App Name var apps = _.filter(OC.Settings.Apps.State.apps, function (app) { return OC.Settings.Apps._search(app.name, query); }); // App ID apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { return OC.Settings.Apps._search(app.id, query); })); // App Description apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { return OC.Settings.Apps._search(app.description, query); })); // Author Name apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { var authors = []; if (_.isArray(app.author)) { _.each(app.author, function (author) { if (typeof author === 'string') { authors.push(author); } else { authors.push(author['@value']); if (!_.isUndefined(author['@attributes']['homepage'])) { authors.push(author['@attributes']['homepage']); } if (!_.isUndefined(author['@attributes']['mail'])) { authors.push(author['@attributes']['mail']); } } }); return OC.Settings.Apps._search(authors.join(' '), query); } else if (typeof app.author !== 'string') { authors.push(app.author['@value']); if (!_.isUndefined(app.author['@attributes']['homepage'])) { authors.push(app.author['@attributes']['homepage']); } if (!_.isUndefined(app.author['@attributes']['mail'])) { authors.push(app.author['@attributes']['mail']); } return OC.Settings.Apps._search(authors.join(' '), query); } return OC.Settings.Apps._search(app.author, query); })); // App status if (t('settings', 'Official').toLowerCase().indexOf(query) !== -1) { apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { return app.level === 200; })); } if (t('settings', 'Approved').toLowerCase().indexOf(query) !== -1) { apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { return app.level === 100; })); } if (t('settings', 'Experimental').toLowerCase().indexOf(query) !== -1) { apps = apps.concat(_.filter(OC.Settings.Apps.State.apps, function (app) { return app.level !== 100 && app.level !== 200; })); } apps = _.uniq(apps, function(app){return app.id;}); if (apps.length === 0) { $appList.addClass('hidden'); $emptyList.removeClass('hidden'); $emptyList.removeClass('hidden').find('h2').text(t('settings', 'No apps found for {query}', { query: query })); } else { _.each(apps, function (app) { $('#app-' + app.id).removeClass('hidden'); }); $('#searchresults').hide(); } }, _onPopState: function(params) { params = _.extend({ category: 'enabled' }, params); OC.Settings.Apps.loadCategory(params.category); }, /** * Initializes the apps list */ initialize: function($el) { var renderer = new marked.Renderer(); renderer.link = function(href, title, text) { try { var prot = decodeURIComponent(unescape(href)) .replace(/[^\w:]/g, '') .toLowerCase(); } catch (e) { return ''; } if (prot.indexOf('http:') !== 0 && prot.indexOf('https:') !== 0) { return ''; } var out = '' + text + ''; return out; }; renderer.image = function(href, title, text) { if (text) { return text; } return title; }; renderer.blockquote = function(quote) { return quote; }; OC.Settings.Apps.markedOptions = { renderer: renderer, gfm: false, highlight: false, tables: false, breaks: false, pedantic: false, sanitize: true, smartLists: true, smartypants: false }; OC.Plugins.register('OCA.Search', OC.Settings.Apps.Search); OC.Settings.Apps.loadCategories(); OC.Util.History.addOnPopStateHandler(_.bind(this._onPopState, this)); $(document).on('click', 'ul#apps-categories li', function () { var categoryId = $(this).data('categoryId'); OC.Settings.Apps.loadCategory(categoryId); OC.Util.History.pushState({ category: categoryId }); $('#searchbox').val(''); }); $(document).on('click', '.app-description-toggle-show', function () { $(this).addClass('hidden'); $(this).siblings('.app-description-toggle-hide').removeClass('hidden'); $(this).siblings('.app-description-container').slideDown(); }); $(document).on('click', '.app-description-toggle-hide', function () { $(this).addClass('hidden'); $(this).siblings('.app-description-toggle-show').removeClass('hidden'); $(this).siblings('.app-description-container').slideUp(); }); $(document).on('click', '#apps-list input.enable', function () { var appId = $(this).data('appid'); var element = $(this); var active = $(this).data('active'); var category = $('#app-navigation').attr('data-category'); if(category === 'app-bundles') { OC.Settings.Apps.enableAppBundle(appId, active, element); } else { OC.Settings.Apps.enableApp(appId, active, element); } }); $(document).on('click', '#apps-list input.uninstall', function () { var appId = $(this).data('appid'); var element = $(this); OC.Settings.Apps.uninstallApp(appId, element); }); $(document).on('click', '#apps-list input.update', function () { var appId = $(this).data('appid'); var element = $(this); OC.Settings.Apps.updateApp(appId, element); }); $(document).on('change', '#group_select', function() { var element = $(this).parent().find('input.enable'); var groups = $(this).val(); if (groups && groups !== '') { groups = groups.split('|'); } else { groups = []; } var appId = element.data('appid'); if (appId) { OC.Settings.Apps.enableApp(appId, false, element, groups); OC.Settings.Apps.State.apps[appId].groups = groups; } }); $(document).on('change', ".groups-enable__checkbox", function() { var $select = $(this).closest('.section').find('#group_select'); $select.val(''); if (this.checked) { OC.Settings.Apps.setupGroupsSelect($select); } else { $select.select2('destroy'); } $select.change(); }); $(document).on('click', '#enable-experimental-apps', function () { var state = $(this).prop('checked'); $.ajax(OC.generateUrl('settings/apps/experimental'), { data: {state: state}, type: 'POST', success:function () { location.reload(); } }); }); } }; OC.Settings.Apps.Search = { attach: function (search) { search.setFilter('settings', OC.Settings.Apps.filter); } }; $(document).ready(function () { // HACK: FIXME: use plugin approach if (!window.TESTING) { OC.Settings.Apps.initialize($('#apps-list')); } });