?i»?

Your IP : 3.142.252.191


Current Path : /home/scgforma/www/cloud/settings/js/
Upload File :
Current File : /home/scgforma/www/cloud/settings/js/apps.js

/* 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('<img src="' + OC.imagePath('core', imageName) + '">');
	}
	return new Handlebars.SafeString('');
});
Handlebars.registerHelper('level', function() {
	if(typeof this.level !== 'undefined') {
		if(this.level === 200) {
			return new Handlebars.SafeString('<span class="official icon-checkmark">' + t('settings', 'Official') + '</span>');
		}
	}
});

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', 'Your apps'), ident: 'installed', id: '0'},
			{displayName: t('settings', 'Enabled apps'), ident: 'enabled', id: '1'},
			{displayName: t('settings', 'Disabled apps'), ident: 'disabled', 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);
				var updateCategory = $.grep(jsondata, function(element, index) {
					return element.ident === 'updates'
				});
				$('#apps-categories').html(html);
				$('#app-category-' + OC.Settings.Apps.State.currentCategory).addClass('active');
				if (updateCategory.length === 1) {
					OC.Settings.Apps.State.availableUpdates = updateCategory[0].counter;
					OC.Settings.Apps.refreshUpdateCounter();
				}
			},
			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;

		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 === 'updates' || 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) {
					if(categoryId !== 'app-bundles') {
						appList.sort(function (a, b) {
							if (a.active !== b.active) {
								return (a.active ? -1 : 1)
							}
							if (a.update !== b.update) {
								return (a.update ? -1 : 1)
							}
							return OC.Util.naturalSortCompare(a.name, b.name);
						});
					}

					var firstExperimental = false;
					var hasNewUpdates = 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) {
							hasNewUpdates = true;
							var $update = $('#app-' + app.id + ' .update');
							$update.removeClass('hidden');
							$update.val(t('settings', 'Update to %s').replace(/%s/g, app.update));
						}
					});
					// reload updates if a list with new updates is loaded
					if (hasNewUpdates) {
						OC.Settings.Apps.reloadUpdates();
					} else {
						// hide update category after all updates are installed
						// and the user is switching away from the empty updates view
						OC.Settings.Apps.refreshUpdateCounter();
					}
				} else {
					if (categoryId === 'updates') {
						OC.Settings.Apps.showEmptyUpdates();
					} else {
						$('#apps-list').addClass('hidden');
						$('#apps-list-empty').removeClass('hidden').find('h2').text(t('settings', 'No apps found for your version'));
						$('#app-list-empty-icon').addClass('icon-search').removeClass('icon-download');
					}
				}

				$('.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);

		if (app.preview) {
			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 = '<svg viewBox="0 0 72 72">';
			img += '<image x="0" y="0" width="72" height="72" preserveAspectRatio="xMinYMin meet" xlink:href="' + url  + '"  class="app-icon" /></svg>';
		} else {
			var rnd = Math.floor((Math.random() * 100 )) + new Date().getSeconds() + new Date().getMilliseconds();
			img = '<svg width="32" height="32" viewBox="0 0 32 32">';
			img += '<defs><filter id="invertIconApps-' + rnd + '"><feColorMatrix in="SourceGraphic" type="matrix" values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0"></feColorMatrix></filter></defs>'
			img += '<image x="0" y="0" width="32" height="32" preserveAspectRatio="xMinYMin meet" filter="url(#invertIconApps-' + rnd + ')" xlink:href="' + url + '?v=' + oc_config.version + '" class="app-icon"></image></svg>';
		}
		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 apps = OC.Settings.Apps.State.currentCategoryElements;
		var appsToEnable = [];
		apps.forEach(function(app) {
			if(app['bundleId'] === bundleId) {
				if(app['active'] === false) {
					appsToEnable.push(app['id']);
				}
			}
		});

		OC.Settings.Apps.enableApp(appsToEnable, false, groups);
	},

		/**
		 * @param {string[]} appId
		 * @param {boolean} active
		 * @param {array} groups
		 */
	enableApp:function(appId, active, groups) {
		if (OC.PasswordConfirmation.requiresPasswordConfirmation()) {
			OC.PasswordConfirmation.requirePasswordConfirmation(_.bind(this.enableApp, this, appId, active, groups));
			return;
		}

		var elements = [];
		appId.forEach(function(appId) {
			elements.push($('#app-'+appId+' .enable'));
		});

		var self = this;
		appId.forEach(function(appId) {
			OC.Settings.Apps.hideErrorMessage(appId);
		});
		groups = groups || [];
		var appItems = [];
		appId.forEach(function(appId) {
			appItems.push($('div#app-'+appId+''));
		});

		if(active && !groups.length) {
			elements.forEach(function(element) {
				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);
						appItems.forEach(function(appItem) {
							appItem.data('errormsg', result.data.message);
						})
					} else {
						OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while disabling app'));
						appItems.forEach(function(appItem) {
							appItem.data('errormsg', t('settings', 'Error while disabling app'));
						});
					}
					elements.forEach(function(element) {
						element.val(t('settings','Disable'));
					});
					appItems.forEach(function(appItem) {
						appItem.addClass('appwarning');
					});
				} else {
					OC.Settings.Apps.rebuildNavigation();
					appItems.forEach(function(appItem) {
						appItem.data('active', false);
						appItem.data('groups', '');
					});
					elements.forEach(function(element) {
						element.data('active', false);
					});
					appItems.forEach(function(appItem) {
						appItem.removeClass('active');
					});
					elements.forEach(function(element) {
						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
			elements.forEach(function(element) {
				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);
						appItems.forEach(function(appItem) {
							appItem.data('errormsg', result.data.message);
						});
					} else {
						OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while enabling app'));
						appItems.forEach(function(appItem) {
							appItem.data('errormsg', t('settings', 'Error while disabling app'));
						});
					}
					elements.forEach(function(element) {
						element.val(t('settings', 'Enable'));
					});
					appItems.forEach(function(appItem) {
						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();
						appItems.forEach(function(appItem) {
							appItem.data('active', true);
						});
						elements.forEach(function(element) {
							element.data('active', true);
						});
						appItems.forEach(function(appItem) {
							appItem.addClass('active');
						});
						elements.forEach(function(element) {
							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')) {
							elements.forEach(function(element) {
								element.parent().find(".groups-enable").prop('checked', true);
								element.parent().find(".groups-enable").hide();
								element.parent().find('.group_select').hide().val(null);
							});
						} else {
							elements.forEach(function(element) {
								element.parent().find("#groups-enable").show();
							});
							if (groups) {
								appItems.forEach(function(appItem) {
									appItem.data('groups', JSON.stringify(groups));
								});
							} else {
								appItems.forEach(function(appItem) {
									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 can not be enabled because it makes the server unstable')
							);
							appItems.forEach(function(appItem) {
								appItem.data('errormsg', t('settings', 'Error while enabling app'));
							});
							elements.forEach(function(element) {
								element.val(t('settings', 'Enable'));
							});
							appItems.forEach(function(appItem) {
								appItem.addClass('appwarning');
							});
						}).fail(function() {
							OC.Settings.Apps.showErrorMessage(
								appId,
								t('settings', 'Error: Could not disable broken app')
							);
							appItems.forEach(function(appItem) {
								appItem.data('errormsg', t('settings', 'Error while disabling broken app'));
							});
							elements.forEach(function(element) {
								element.val(t('settings', 'Enable'));
							});
						});
					});
				}
			},'json')
				.fail(function() {
					OC.Settings.Apps.showErrorMessage(appId, t('settings', 'Error while enabling app'));
					appItems.forEach(function(appItem) {
						appItem.data('errormsg', t('settings', 'Error while enabling app'));
						appItem.data('active', false);
						appItem.addClass('appwarning');
					});
					elements.forEach(function(element) {
						element.val(t('settings', 'Enable'));
					});
				});
		}
	},

	showEmptyUpdates: function() {
		$('#apps-list').addClass('hidden');
		$('#apps-list-empty').removeClass('hidden').find('h2').text(t('settings', 'App up to date'));
		$('#app-list-empty-icon').removeClass('icon-search').addClass('icon-download');
	},

	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','Could not update 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']);

				OC.Settings.Apps.State.availableUpdates--;
				OC.Settings.Apps.refreshUpdateCounter();

				if (OC.Settings.Apps.State.currentCategory === 'updates') {
					$('#app-' + appId).remove();
					if (OC.Settings.Apps.State.availableUpdates === 0) {
						OC.Settings.Apps.showEmptyUpdates();
					}
				}
			}
		},'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','Could not remove 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></li>');
						li.attr('data-id', entry.id);
						var img = '<svg width="16" height="16" viewBox="0 0 16 16">';
						img += '<defs><filter id="invert"><feColorMatrix in="SourceGraphic" type="matrix" values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /></filter></defs>';
						img += '<image x="0" y="0" width="16" height="16" preserveAspectRatio="xMinYMin meet" filter="url(#invert)" xlink:href="' + entry.icon + '"  class="app-icon" /></svg>';
						var a = $('<a></a>').attr('href', entry.href);
						var filename = $('<span></span>');
						var loading = $('<div class="icon-loading-dark"></div>').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></li>');
						li.attr('data-id', entry.id);
						var img = '<img src="' + entry.icon + '" class="app-icon">';
						if (OCA.Theming && OCA.Theming.inverted) {
							img = '<svg width="20" height="20" viewBox="0 0 20 20">';
							img += '<defs><filter id="invert"><feColorMatrix in="SourceGraphic" type="matrix" values="-1 0 0 0 1 0 -1 0 0 1 0 0 -1 0 1 0 0 0 1 0" /></filter></defs>';
							img += '<image x="0" y="0" width="20" height="20" preserveAspectRatio="xMinYMin meet" filter="url(#invert)" xlink:href="' + entry.icon + '"  class="app-icon" /></svg>';
						}
						var a = $('<a></a>').attr('href', entry.href);
						var filename = $('<span></span>');
						var loading = $('<div class="icon-loading-dark"></div>').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');
			}
		});
	},

	reloadUpdates: function() {
		if (this._loadUpdatesCall) {
			this._loadUpdatesCall.abort();
		}
		this._loadUpdatesCall = $.ajax(OC.generateUrl('settings/apps/list?category=updates'), {
			type:'GET',
			success: function (apps) {
				OC.Settings.Apps.State.availableUpdates = apps.apps.length;
				OC.Settings.Apps.refreshUpdateCounter();
			}
		});
	},

	refreshUpdateCounter: function() {
		var $appCategoryUpdates = $('#app-category-updates');
		var $updateCount = $appCategoryUpdates.find('.app-navigation-entry-utils-counter');
		if (OC.Settings.Apps.State.availableUpdates > 0) {
			$updateCount.html(OC.Settings.Apps.State.availableUpdates);
			$appCategoryUpdates.show();
		} else {
			$updateCount.empty();
			if (OC.Settings.Apps.State.currentCategory !== 'updates') {
				$appCategoryUpdates.hide();
			}
		}
	},

	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');
		$('#app-list-empty-icon').addClass('icon-search').removeClass('icon-download');
		$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 = '<a href="' + href + '" rel="noreferrer noopener"';
			if (title) {
				out += ' title="' + title + '"';
			}
			out += '>' + text + '</a>';
			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 bundleId = $(this).data('bundleid');
			var element = $(this);
			var active = $(this).data('active');

			var category = $('#app-navigation').attr('data-category');
			if(bundleId) {
				OC.Settings.Apps.enableAppBundle(bundleId, active, element);
				element.val(t('settings', 'Enable all'));
			} else {
				OC.Settings.Apps.enableApp([appId], active);
			}
		});

		$(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).closest('.section').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, 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'));
	}
});