Files
cloudron-box/dashboard/src/views/apps.js

385 lines
15 KiB
JavaScript
Raw Normal View History

2018-01-22 13:01:38 -08:00
'use strict';
/* global angular:false */
/* global $:false */
2022-07-11 17:24:07 +02:00
/* global APP_TYPES */
/* global onAppClick */
2018-01-22 13:01:38 -08:00
2021-01-28 10:07:35 -08:00
angular.module('Application').controller('AppsController', ['$scope', '$translate', '$interval', '$location', 'Client', function ($scope, $translate, $interval, $location, Client) {
2019-05-20 23:40:02 +02:00
var ALL_DOMAINS_DOMAIN = { _alldomains: true, domain: 'All Domains' }; // dummy record for the single select filter
2021-03-02 17:30:01 -08:00
var GROUP_ACCESS_UNSET = { _unset: true, name: 'Select Group' }; // dummy record for the single select filter
2019-05-20 23:40:02 +02:00
2018-01-22 13:01:38 -08:00
$scope.installedApps = Client.getInstalledApps();
2019-04-12 11:06:56 +02:00
$scope.tags = Client.getAppTags();
2020-09-26 17:50:23 +02:00
$scope.states = [
{ state: '', label: 'All States' },
{ state: 'running', label: 'Running' },
2020-10-06 13:01:01 -07:00
{ state: 'stopped', label: 'Stopped' },
2022-05-14 14:45:22 +02:00
{ state: 'update_available', label: 'Update Available' },
2020-10-06 13:01:01 -07:00
{ state: 'not_responding', label: 'Not Responding' }
2020-09-26 17:50:23 +02:00
];
$scope.selectedState = $scope.states[0];
2019-04-12 11:06:56 +02:00
$scope.selectedTags = [];
2021-02-16 20:10:11 +01:00
$scope.selectedGroup = GROUP_ACCESS_UNSET;
2019-05-20 23:40:02 +02:00
$scope.selectedDomain = ALL_DOMAINS_DOMAIN;
$scope.filterDomains = [ ALL_DOMAINS_DOMAIN ];
2018-01-22 13:01:38 -08:00
$scope.config = Client.getConfig();
$scope.user = Client.getUserInfo();
$scope.domains = [];
2020-01-06 15:27:31 +01:00
$scope.appSearch = '';
2021-02-16 20:10:11 +01:00
$scope.groups = [ GROUP_ACCESS_UNSET ];
2022-07-11 17:24:07 +02:00
$scope.APP_TYPES = APP_TYPES;
$scope.showFilter = false;
$scope.filterActive = false;
2024-04-29 09:32:00 +02:00
$scope.VIEWS = {
GRID: 'grid',
LIST: 'list'
};
$scope.view = $scope.VIEWS.GRID;
$scope.orderBy = 'location'; // or app, status, sso
$scope.orderByReverse = false;
2018-01-22 13:01:38 -08:00
$scope.allUsers = [];
$scope.allGroups = [];
2022-05-14 14:45:22 +02:00
$translate(['apps.stateFilterHeader', 'apps.domainsFilterHeader', 'apps.groupsFilterHeader', 'app.states.running', 'app.states.stopped', 'app.states.notResponding', 'app.states.updateAvailable']).then(function (tr) {
2020-12-15 15:48:25 +01:00
if (tr['apps.domainsFilterHeader']) ALL_DOMAINS_DOMAIN.domain = tr['apps.domainsFilterHeader'];
2021-03-02 17:30:01 -08:00
if (tr['apps.groupsFilterHeader']) GROUP_ACCESS_UNSET.name = tr['apps.groupsFilterHeader'];
2020-12-15 15:48:25 +01:00
if (tr['apps.stateFilterHeader']) $scope.states[0].label = tr['apps.stateFilterHeader'];
if (tr['app.states.running']) $scope.states[1].label = tr['app.states.running'];
if (tr['app.states.stopped']) $scope.states[2].label = tr['app.states.stopped'];
2022-05-14 14:45:22 +02:00
if (tr['app.states.notResponding']) $scope.states[4].label = tr['app.states.notResponding'];
if (tr['app.states.updateAvailable']) $scope.states[3].label = tr['app.states.updateAvailable'];
2020-12-15 15:48:25 +01:00
});
$scope.pendingChecklistItems = function (app) {
if (!app.checklist) return 0;
return Object.keys(app.checklist).filter(function (key) { return !app.checklist[key].acknowledged; }).length;
};
$scope.setOrderBy = function (by) {
if (by === $scope.orderBy) {
$scope.orderByReverse = !$scope.orderByReverse;
} else {
$scope.orderBy = by;
$scope.orderByReverse = false;
}
localStorage.appsOrderBy = by;
if ($scope.orderByReverse) localStorage.appsOrderByReverse = true;
else localStorage.removeItem('appsOrderByReverse');
};
// for sorting/grouping
$scope.orderByFilter = function (item) {
if ($scope.orderBy === 'app') return item.manifest.title || 'App Link';
if ($scope.orderBy === 'status') return item.installationState + '-' + item.runState;
if ($scope.orderBy === 'sso') return item.sso;
return item.label || item.fqdn;
};
$scope.setView = function (view) {
2024-04-29 09:32:00 +02:00
if (view !== $scope.VIEWS.LIST && view !== $scope.VIEWS.GRID) return;
$scope.view = view;
localStorage.appsView = view;
};
$scope.toggleView = function () {
2024-04-29 09:32:00 +02:00
$scope.view = $scope.view === $scope.VIEWS.GRID ? $scope.VIEWS.LIST : $scope.VIEWS.GRID;
localStorage.appsView = $scope.view;
};
$scope.toggleFilter = function () {
$scope.showFilter = !$scope.showFilter;
if ($scope.showFilter) localStorage.appsShowFilter = true;
else localStorage.removeItem('appsShowFilter');
// clear on hide
if (!$scope.showFilter) {
$scope.selectedState = $scope.states[0];
$scope.selectedTags = [];
$scope.selectedGroup = GROUP_ACCESS_UNSET;
$scope.selectedDomain = ALL_DOMAINS_DOMAIN;
}
};
$scope.$watch('selectedTags', function (newVal, oldVal) {
if (newVal === oldVal) return;
localStorage.selectedTags = newVal.join(',');
});
2021-11-11 15:11:09 +01:00
$scope.$watch('selectedState', function (newVal, oldVal) {
if (newVal === oldVal) return;
if (newVal === $scope.states[0]) localStorage.removeItem('selectedState');
else localStorage.selectedState = newVal.state;
});
$scope.$watch('selectedGroup', function (newVal, oldVal) {
if (newVal === oldVal) return;
if (newVal === GROUP_ACCESS_UNSET) localStorage.removeItem('selectedGroup');
else localStorage.selectedGroup = newVal.id;
});
$scope.$watch('selectedDomain', function (newVal, oldVal) {
if (newVal === oldVal) return;
if (newVal._alldomains) localStorage.removeItem('selectedDomain');
else localStorage.selectedDomain = newVal.domain;
});
$scope.onAppClick = function (app, $event) { onAppClick(app, $event, $scope.isOperator(app), $scope); };
2022-09-11 17:40:59 +02:00
$scope.appPostInstallConfirm = {
app: {},
message: '',
show: function (app) {
$scope.appPostInstallConfirm.app = app;
$scope.appPostInstallConfirm.message = app.manifest.postInstallMessage;
$('#appsPostInstallConfirmModal').modal('show');
return false; // prevent propagation and default
},
submit: function () {
$scope.appPostInstallConfirm.app.pendingPostInstallConfirmation = false;
delete localStorage['confirmPostInstall_' + $scope.appPostInstallConfirm.app.id];
$('#appsPostInstallConfirmModal').modal('hide');
}
};
2022-07-07 16:25:04 +02:00
$scope.applinksEdit = {
error: {},
2022-07-07 16:41:12 +02:00
busyEdit: false,
busyRemove: false,
2022-07-08 17:44:20 +02:00
applink: {},
2022-07-07 16:25:04 +02:00
id: '',
upstreamUri: '',
2022-07-07 18:53:14 +02:00
label: '',
2022-07-07 19:11:54 +02:00
tags: '',
accessRestrictionOption: '',
accessRestriction: { users: [], groups: [] },
2022-07-08 17:44:20 +02:00
icon: { data: null },
iconUrl: function () {
if ($scope.applinksEdit.icon.data === '__original__') { // user clicked reset
// https://png-pixel.com/ white pixel placeholder
return 'data:image/png;base64,iVBORw0KGgoAAAANSUhEUgAAAAEAAAABCAQAAAC1HAwCAAAAC0lEQVR42mP8/x8AAwMCAO+ip1sAAAAASUVORK5CYII=';
} else if ($scope.applinksEdit.icon.data) { // user uploaded icon
return $scope.applinksEdit.icon.data;
} else { // current icon
return $scope.applinksEdit.applink.iconUrl;
}
},
resetCustomIcon: function () {
$scope.applinksEdit.icon.data = '__original__';
},
showCustomIconSelector: function () {
$('#applinksEditIconFileInput').click();
},
isAccessRestrictionValid: function () {
return !!($scope.applinksEdit.accessRestriction.users.length || $scope.applinksEdit.accessRestriction.groups.length);
},
2022-07-07 16:25:04 +02:00
show: function (applink) {
$scope.applinksEdit.error = {};
2022-07-07 16:41:12 +02:00
$scope.applinksEdit.busyEdit = false;
$scope.applinksEdit.busyRemove = false;
2022-07-08 17:44:20 +02:00
$scope.applinksEdit.applink = applink;
2022-07-07 16:25:04 +02:00
$scope.applinksEdit.id = applink.id;
$scope.applinksEdit.upstreamUri = applink.upstreamUri;
2022-07-07 18:53:14 +02:00
$scope.applinksEdit.label = applink.label;
$scope.applinksEdit.accessRestrictionOption = applink.accessRestriction ? 'groups' : 'any';
$scope.applinksEdit.accessRestriction = { users: [], groups: [] };
$scope.applinksEdit.icon = { data: null };
var userSet, groupSet;
if (applink.accessRestriction) {
userSet = {};
applink.accessRestriction.users.forEach(function (uid) { userSet[uid] = true; });
2022-07-08 15:09:02 +02:00
$scope.allUsers.forEach(function (u) { if (userSet[u.id] === true) $scope.applinksEdit.accessRestriction.users.push(u); });
groupSet = {};
if (applink.accessRestriction.groups) applink.accessRestriction.groups.forEach(function (gid) { groupSet[gid] = true; });
2022-07-08 15:09:02 +02:00
$scope.allGroups.forEach(function (g) { if (groupSet[g.id] === true) $scope.applinksEdit.accessRestriction.groups.push(g); });
}
2022-07-07 16:25:04 +02:00
2022-07-07 19:11:54 +02:00
// translate for tag-input
$scope.applinksEdit.tags = applink.tags ? applink.tags.join(' ') : '';
2022-07-07 16:41:12 +02:00
$scope.applinksEditForm.$setUntouched();
$scope.applinksEditForm.$setPristine();
2022-07-07 16:25:04 +02:00
$('#applinksEditModal').modal('show');
return false; // prevent propagation and default
},
submit: function () {
2022-07-07 16:41:12 +02:00
$scope.applinksEdit.busyEdit = true;
2022-10-06 19:50:47 +02:00
$scope.applinksEdit.error = {};
2022-07-07 16:41:12 +02:00
var accessRestriction = null;
if ($scope.applinksEdit.accessRestrictionOption === 'groups') {
accessRestriction = { users: [], groups: [] };
accessRestriction.users = $scope.applinksEdit.accessRestriction.users.map(function (u) { return u.id; });
accessRestriction.groups = $scope.applinksEdit.accessRestriction.groups.map(function (g) { return g.id; });
}
2022-07-07 16:41:12 +02:00
var data = {
2022-07-07 18:53:14 +02:00
upstreamUri: $scope.applinksEdit.upstreamUri,
2022-07-07 19:11:54 +02:00
label: $scope.applinksEdit.label,
accessRestriction: accessRestriction,
2022-07-07 19:11:54 +02:00
tags: $scope.applinksEdit.tags.split(' ').map(function (t) { return t.trim(); }).filter(function (t) { return !!t; })
2022-07-07 16:41:12 +02:00
};
if ($scope.applinksEdit.icon.data === '__original__') { // user reset the icon
data.icon = '';
} else if ($scope.applinksEdit.icon.data) { // user loaded custom icon
data.icon = $scope.applinksEdit.icon.data.replace(/^data:image\/[a-z]+;base64,/, '');
}
2022-07-07 16:41:12 +02:00
Client.updateApplink($scope.applinksEdit.id, data, function (error) {
$scope.applinksEdit.busyEdit = false;
2022-10-06 19:50:47 +02:00
if (error && error.statusCode === 400 && error.message.includes('upstreamUri')) {
$scope.applinksEdit.error.upstreamUri = error.message;
$scope.applinksEditForm.$setUntouched();
$scope.applinksEditForm.$setPristine();
return;
}
2022-07-07 16:41:12 +02:00
if (error) return console.error('Failed to update applink', error);
2022-07-07 16:55:52 +02:00
Client.refreshInstalledApps();
2022-07-07 16:41:12 +02:00
$('#applinksEditModal').modal('hide');
});
2022-07-07 16:25:04 +02:00
},
remove: function () {
2022-07-07 16:41:12 +02:00
$scope.applinksEdit.busyRemove = true;
2022-07-07 16:25:04 +02:00
Client.removeApplink($scope.applinksEdit.id, function (error) {
2022-07-07 16:41:12 +02:00
$scope.applinksEdit.busyRemove = false;
2022-07-07 16:25:04 +02:00
if (error) return console.error('Failed to remove applink', error);
Client.refreshInstalledApps();
$('#applinksEditModal').modal('hide');
});
}
};
$scope.showAppConfigure = function (app, view) {
$location.path('/app/' + app.id + '/' + view);
2018-01-22 13:01:38 -08:00
};
2021-09-21 15:26:05 -07:00
$scope.isOperator = function (app) {
return app.accessLevel === 'operator' || app.accessLevel === 'admin';
};
Client.onReady(function () {
setTimeout(function () { $('#appSearch').focus(); }, 1);
// refresh the new list immediately when switching from another view (appstore)
Client.refreshInstalledApps(function () {
var refreshAppsTimer = $interval(Client.refreshInstalledApps.bind(Client, function () {}), 5000);
$scope.$on('$destroy', function () {
$interval.cancel(refreshAppsTimer);
});
});
$scope.setView(localStorage.appsView);
$scope.orderBy = localStorage.appsOrderBy || 'location';
$scope.orderByReverse = !!localStorage.appsOrderByReverse;
$scope.showFilter = !!localStorage.appsShowFilter;
if (!$scope.user.isAtLeastAdmin) return;
// load local settings and apply tag filter
if (localStorage.selectedTags) {
if (!$scope.tags.length) localStorage.removeItem('selectedTags');
else $scope.selectedTags = localStorage.selectedTags.split(',');
}
2021-11-11 15:11:09 +01:00
if (localStorage.selectedState) $scope.selectedState = $scope.states.find(function (s) { return s.state === localStorage.selectedState; }) || $scope.states[0];
2021-02-16 20:10:11 +01:00
Client.getGroups(function (error, result) {
if (error) Client.error(error);
$scope.groups = [ GROUP_ACCESS_UNSET ].concat(result);
$scope.allGroups = result;
2021-11-11 15:11:09 +01:00
if (localStorage.selectedGroup) $scope.selectedGroup = $scope.groups.find(function (g) { return g.id === localStorage.selectedGroup; }) || GROUP_ACCESS_UNSET;
2021-02-16 20:10:11 +01:00
});
Client.getDomains(function (error, result) {
if (error) Client.error(error);
$scope.domains = result;
$scope.filterDomains = [ALL_DOMAINS_DOMAIN].concat(result);
if (localStorage.selectedDomain) $scope.selectedDomain = $scope.filterDomains.find(function (d) { return d.domain === localStorage.selectedDomain; }) || ALL_DOMAINS_DOMAIN;
2018-01-22 13:01:38 -08:00
});
Client.getAllUsers(function (error, users) {
if (error) Client.error(error);
$scope.allUsers = users;
});
2018-01-22 13:01:38 -08:00
});
2022-07-08 17:44:20 +02:00
$('#applinksEditIconFileInput').get(0).onchange = function (event) {
var fr = new FileReader();
fr.onload = function () {
$scope.$apply(function () {
// var file = event.target.files[0];
$scope.applinksEdit.icon.data = fr.result;
});
};
fr.readAsDataURL(event.target.files[0]);
};
2022-07-07 13:32:20 +02:00
// setup all the dialog focus handling
2022-07-07 18:53:14 +02:00
['applinksAddModal', 'applinksEditModal'].forEach(function (id) {
2022-07-07 13:32:20 +02:00
$('#' + id).on('shown.bs.modal', function () {
$(this).find('autofocus]:first').focus();
2022-07-07 13:32:20 +02:00
});
});
2020-06-24 23:06:12 -07:00
$('.collapse').on('shown.bs.collapse', function(){
$(this).parent().find('.fa-angle-right').removeClass('fa-angle-right').addClass('fa-angle-down');
}).on('hidden.bs.collapse', function(){
$(this).parent().find('.fa-angle-down').removeClass('fa-angle-down').addClass('fa-angle-right');
});
2018-01-22 13:01:38 -08:00
$('.modal-backdrop').remove();
function keyboardHandler(event) {
if (event.key === '/') {
document.getElementById('appSearch').focus();
event.preventDefault();
}
}
document.addEventListener('keydown', keyboardHandler);
$scope.$on('$destroy', function () {
document.removeEventListener('keydown', keyboardHandler);
});
2018-01-22 13:01:38 -08:00
}]);