2015-02-19 21:21:54 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An element which displays a box to search for an entity (org, user, robot, team). This control
|
|
|
|
* allows for filtering of the entities found and whether to allow selection by e-mail.
|
|
|
|
*/
|
|
|
|
angular.module('quay').directive('entitySearch', function () {
|
|
|
|
var number = 0;
|
|
|
|
var directiveDefinitionObject = {
|
|
|
|
priority: 0,
|
|
|
|
templateUrl: '/static/directives/entity-search.html',
|
|
|
|
replace: false,
|
|
|
|
transclude: false,
|
|
|
|
restrict: 'C',
|
|
|
|
require: '?ngModel',
|
|
|
|
link: function(scope, element, attr, ctrl) {
|
|
|
|
scope.ngModel = ctrl;
|
|
|
|
},
|
|
|
|
scope: {
|
|
|
|
'namespace': '=namespace',
|
|
|
|
'placeholder': '=placeholder',
|
2016-05-12 21:59:49 +00:00
|
|
|
'forRepository': '=forRepository',
|
|
|
|
'skipPermissions': '=skipPermissions',
|
2015-02-19 21:21:54 +00:00
|
|
|
|
|
|
|
// Default: ['user', 'team', 'robot']
|
|
|
|
'allowedEntities': '=allowedEntities',
|
|
|
|
|
|
|
|
'currentEntity': '=currentEntity',
|
|
|
|
|
|
|
|
'entitySelected': '&entitySelected',
|
|
|
|
'emailSelected': '&emailSelected',
|
|
|
|
|
|
|
|
// When set to true, the contents of the control will be cleared as soon
|
|
|
|
// as an entity is selected.
|
|
|
|
'autoClear': '=autoClear',
|
|
|
|
|
|
|
|
// Set this property to immediately clear the contents of the control.
|
|
|
|
'clearValue': '=clearValue',
|
|
|
|
|
|
|
|
// Whether e-mail addresses are allowed.
|
|
|
|
'allowEmails': '=allowEmails',
|
|
|
|
'emailMessage': '@emailMessage',
|
|
|
|
|
|
|
|
// True if the menu should pull right.
|
|
|
|
'pullRight': '@pullRight'
|
|
|
|
},
|
2017-02-21 21:29:01 +00:00
|
|
|
controller: function($rootScope, $scope, $element, Restangular, UserService, ApiService, UtilService, AvatarService, Config) {
|
2015-02-19 21:21:54 +00:00
|
|
|
$scope.lazyLoading = true;
|
|
|
|
|
|
|
|
$scope.teams = null;
|
|
|
|
$scope.robots = null;
|
|
|
|
|
|
|
|
$scope.isAdmin = false;
|
|
|
|
$scope.isOrganization = false;
|
|
|
|
|
|
|
|
$scope.includeTeams = true;
|
|
|
|
$scope.includeRobots = true;
|
|
|
|
$scope.includeOrgs = false;
|
|
|
|
|
|
|
|
$scope.currentEntityInternal = $scope.currentEntity;
|
2016-05-12 21:59:49 +00:00
|
|
|
$scope.createRobotInfo = null;
|
|
|
|
$scope.createTeamInfo = null;
|
2015-02-19 21:21:54 +00:00
|
|
|
|
2015-03-30 21:55:04 +00:00
|
|
|
$scope.Config = Config;
|
|
|
|
|
2015-02-19 21:21:54 +00:00
|
|
|
var isSupported = function(kind, opt_array) {
|
|
|
|
return $.inArray(kind, opt_array || $scope.allowedEntities || ['user', 'team', 'robot']) >= 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.lazyLoad = function() {
|
|
|
|
if (!$scope.namespace || !$scope.lazyLoading) { return; }
|
|
|
|
|
|
|
|
// Reset the cached teams and robots.
|
|
|
|
$scope.teams = null;
|
|
|
|
$scope.robots = null;
|
|
|
|
|
|
|
|
// Load the organization's teams (if applicable).
|
|
|
|
if ($scope.isOrganization && isSupported('team')) {
|
|
|
|
// Note: We load the org here again so that we always have the fully up-to-date
|
|
|
|
// teams list.
|
|
|
|
ApiService.getOrganization(null, {'orgname': $scope.namespace}).then(function(resp) {
|
2016-06-06 15:47:53 +00:00
|
|
|
$scope.teams = Object.keys(resp.teams).map(function(key) {
|
|
|
|
return resp.teams[key];
|
|
|
|
});
|
2015-02-19 21:21:54 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
// Load the user/organization's robots (if applicable).
|
|
|
|
if ($scope.isAdmin && isSupported('robot')) {
|
|
|
|
ApiService.getRobots($scope.isOrganization ? $scope.namespace : null).then(function(resp) {
|
|
|
|
$scope.robots = resp.robots;
|
|
|
|
$scope.lazyLoading = false;
|
|
|
|
}, function() {
|
|
|
|
$scope.lazyLoading = false;
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
$scope.lazyLoading = false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2016-05-12 21:59:49 +00:00
|
|
|
$scope.askCreateTeam = function() {
|
|
|
|
$scope.createTeamInfo = {
|
|
|
|
'namespace': $scope.namespace,
|
|
|
|
'repository': $scope.forRepository,
|
|
|
|
'skip_permissions': $scope.skipPermissions
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.askCreateRobot = function() {
|
|
|
|
$scope.createRobotInfo = {
|
|
|
|
'namespace': $scope.namespace,
|
|
|
|
'repository': $scope.forRepository,
|
|
|
|
'skip_permissions': $scope.skipPermissions
|
|
|
|
};
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.handleTeamCreated = function(created) {
|
|
|
|
$scope.setEntity(created.name, 'team', false, created.avatar);
|
2016-06-06 15:47:53 +00:00
|
|
|
$scope.teams.push(created);
|
2015-02-19 21:21:54 +00:00
|
|
|
};
|
|
|
|
|
2016-05-12 21:59:49 +00:00
|
|
|
$scope.handleRobotCreated = function(created) {
|
|
|
|
$scope.setEntity(created.name, 'user', true, created.avatar);
|
|
|
|
$scope.robots.push(created);
|
2015-02-19 21:21:54 +00:00
|
|
|
};
|
|
|
|
|
2015-03-30 21:55:04 +00:00
|
|
|
$scope.setEntity = function(name, kind, is_robot, avatar) {
|
2015-02-19 21:21:54 +00:00
|
|
|
var entity = {
|
|
|
|
'name': name,
|
|
|
|
'kind': kind,
|
2015-03-30 21:55:04 +00:00
|
|
|
'is_robot': is_robot,
|
|
|
|
'avatar': avatar
|
2015-02-19 21:21:54 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if ($scope.isOrganization) {
|
|
|
|
entity['is_org_member'] = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.setEntityInternal(entity, false);
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.clearEntityInternal = function() {
|
|
|
|
$scope.currentEntityInternal = null;
|
|
|
|
$scope.currentEntity = null;
|
|
|
|
$scope.entitySelected({'entity': null});
|
|
|
|
if ($scope.ngModel) {
|
|
|
|
$scope.ngModel.$setValidity('entity', false);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
$scope.setEntityInternal = function(entity, updateTypeahead) {
|
2016-10-27 19:31:32 +00:00
|
|
|
// If the entity is an external entity, convert it to a known user via an API call.
|
|
|
|
if (entity.kind == 'external') {
|
|
|
|
var params = {
|
|
|
|
'username': entity.name
|
|
|
|
};
|
|
|
|
|
|
|
|
ApiService.linkExternalUser(null, params).then(function(resp) {
|
|
|
|
$scope.setEntityInternal(resp['entity'], updateTypeahead);
|
|
|
|
}, ApiService.errorDisplay('Could not link external user'));
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2015-02-19 21:21:54 +00:00
|
|
|
if (updateTypeahead) {
|
|
|
|
$(input).typeahead('val', $scope.autoClear ? '' : entity.name);
|
|
|
|
} else {
|
|
|
|
$(input).val($scope.autoClear ? '' : entity.name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!$scope.autoClear) {
|
|
|
|
$scope.currentEntityInternal = entity;
|
|
|
|
$scope.currentEntity = entity;
|
|
|
|
}
|
|
|
|
|
|
|
|
$scope.entitySelected({'entity': entity});
|
|
|
|
if ($scope.ngModel) {
|
|
|
|
$scope.ngModel.$setValidity('entity', !!entity);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
// Setup the typeahead.
|
|
|
|
var input = $element[0].firstChild.firstChild;
|
|
|
|
|
|
|
|
(function() {
|
|
|
|
// Create the bloodhound search query system.
|
|
|
|
$rootScope.__entity_search_counter = (($rootScope.__entity_search_counter || 0) + 1);
|
|
|
|
var entitySearchB = new Bloodhound({
|
|
|
|
name: 'entities' + $rootScope.__entity_search_counter,
|
|
|
|
remote: {
|
|
|
|
url: '/api/v1/entities/%QUERY',
|
|
|
|
replace: function (url, uriEncodedQuery) {
|
|
|
|
var namespace = $scope.namespace || '';
|
|
|
|
url = url.replace('%QUERY', uriEncodedQuery);
|
|
|
|
url += '?namespace=' + encodeURIComponent(namespace);
|
|
|
|
if ($scope.isOrganization && isSupported('team')) {
|
|
|
|
url += '&includeTeams=true'
|
|
|
|
}
|
|
|
|
if (isSupported('org')) {
|
|
|
|
url += '&includeOrgs=true'
|
|
|
|
}
|
|
|
|
return url;
|
|
|
|
},
|
|
|
|
filter: function(data) {
|
|
|
|
var datums = [];
|
|
|
|
for (var i = 0; i < data.results.length; ++i) {
|
|
|
|
var entity = data.results[i];
|
|
|
|
|
|
|
|
var found = 'user';
|
2016-10-27 19:31:32 +00:00
|
|
|
if (entity.kind == 'user' || entity.kind == 'external') {
|
2015-02-19 21:21:54 +00:00
|
|
|
found = entity.is_robot ? 'robot' : 'user';
|
|
|
|
} else if (entity.kind == 'team') {
|
|
|
|
found = 'team';
|
|
|
|
} else if (entity.kind == 'org') {
|
|
|
|
found = 'org';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!isSupported(found)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
datums.push({
|
|
|
|
'value': entity.name,
|
|
|
|
'tokens': [entity.name],
|
|
|
|
'entity': entity
|
|
|
|
});
|
|
|
|
}
|
|
|
|
return datums;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
datumTokenizer: function(d) {
|
|
|
|
return Bloodhound.tokenizers.whitespace(d.val);
|
|
|
|
},
|
|
|
|
queryTokenizer: Bloodhound.tokenizers.whitespace
|
|
|
|
});
|
|
|
|
entitySearchB.initialize();
|
|
|
|
|
|
|
|
// Setup the typeahead.
|
|
|
|
$(input).typeahead({
|
2017-04-07 21:25:44 +00:00
|
|
|
'highlight': true,
|
|
|
|
'hint': false,
|
2015-02-19 21:21:54 +00:00
|
|
|
}, {
|
2017-04-07 21:25:44 +00:00
|
|
|
display: 'value',
|
2015-02-19 21:21:54 +00:00
|
|
|
source: entitySearchB.ttAdapter(),
|
|
|
|
templates: {
|
2017-04-07 21:25:44 +00:00
|
|
|
'notFound': function(info) {
|
2015-02-19 21:21:54 +00:00
|
|
|
// Only display the empty dialog if the server load has finished.
|
|
|
|
if (info.resultKind == 'remote') {
|
|
|
|
var val = $(input).val();
|
|
|
|
if (!val) {
|
|
|
|
return null;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (UtilService.isEmailAddress(val)) {
|
|
|
|
if ($scope.allowEmails) {
|
|
|
|
return '<div class="tt-message">' + $scope.emailMessage + '</div>';
|
|
|
|
} else {
|
|
|
|
return '<div class="tt-empty">A ' + Config.REGISTRY_TITLE_SHORT + ' username (not an e-mail address) must be specified</div>';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
var classes = [];
|
|
|
|
|
|
|
|
if (isSupported('user')) { classes.push('users'); }
|
|
|
|
if (isSupported('org')) { classes.push('organizations'); }
|
|
|
|
if ($scope.isAdmin && isSupported('robot')) { classes.push('robot accounts'); }
|
|
|
|
if ($scope.isOrganization && isSupported('team')) { classes.push('teams'); }
|
|
|
|
|
2016-08-31 18:19:28 +00:00
|
|
|
if (classes.length == 0) {
|
|
|
|
return '<div class="tt-empty">No matching entities found</div>';
|
2015-02-19 21:21:54 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
var class_string = '';
|
|
|
|
for (var i = 0; i < classes.length; ++i) {
|
|
|
|
if (i > 0) {
|
|
|
|
if (i == classes.length - 1) {
|
|
|
|
class_string += ' or ';
|
|
|
|
} else {
|
|
|
|
class_string += ', ';
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
class_string += classes[i];
|
|
|
|
}
|
|
|
|
|
|
|
|
return '<div class="tt-empty">No matching ' + Config.REGISTRY_TITLE_SHORT + ' ' + class_string + ' found</div>';
|
|
|
|
}
|
|
|
|
|
|
|
|
return null;
|
|
|
|
},
|
|
|
|
'suggestion': function (datum) {
|
|
|
|
template = '<div class="entity-mini-listing">';
|
2017-04-17 22:54:27 +00:00
|
|
|
if (Config['AVATAR_KIND'] === 'gravatar' &&
|
|
|
|
((datum.entity.kind == 'user' && !datum.entity.is_robot) || (datum.entity.kind == 'org'))) {
|
2017-02-22 02:16:15 +00:00
|
|
|
template += '<i class="fa"><img class="avatar-image" src="' +
|
|
|
|
AvatarService.getAvatar(datum.entity.avatar.hash, 20, 'mm') +
|
2017-02-21 21:29:01 +00:00
|
|
|
'"></i>';
|
2016-10-27 19:31:32 +00:00
|
|
|
} else if (datum.entity.kind == 'external') {
|
|
|
|
template += '<i class="fa fa-user fa-lg"></i>';
|
2015-02-19 21:21:54 +00:00
|
|
|
} else if (datum.entity.kind == 'user' && datum.entity.is_robot) {
|
2015-04-23 20:41:47 +00:00
|
|
|
template += '<i class="fa ci-robot fa-lg"></i>';
|
2015-02-19 21:21:54 +00:00
|
|
|
} else if (datum.entity.kind == 'team') {
|
|
|
|
template += '<i class="fa fa-group fa-lg"></i>';
|
|
|
|
}
|
|
|
|
|
|
|
|
template += '<span class="name">' + datum.value + '</span>';
|
|
|
|
|
2016-09-08 15:23:37 +00:00
|
|
|
if (datum.entity.title) {
|
|
|
|
template += '<span class="title">' + datum.entity.title + '</span>';
|
|
|
|
}
|
|
|
|
|
2015-02-19 21:21:54 +00:00
|
|
|
if (datum.entity.is_org_member === false && datum.entity.kind == 'user') {
|
|
|
|
template += '<i class="fa fa-exclamation-triangle" title="User is outside the organization"></i>';
|
|
|
|
}
|
|
|
|
|
|
|
|
template += '</div>';
|
|
|
|
return template;
|
|
|
|
}}
|
|
|
|
});
|
|
|
|
|
|
|
|
$(input).on('keypress', function(e) {
|
|
|
|
var val = $(input).val();
|
|
|
|
var code = e.keyCode || e.which;
|
|
|
|
if (code == 13 && $scope.allowEmails && UtilService.isEmailAddress(val)) {
|
|
|
|
$scope.$apply(function() {
|
|
|
|
$scope.emailSelected({'email': val});
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$(input).on('input', function(e) {
|
|
|
|
$scope.$apply(function() {
|
|
|
|
$scope.clearEntityInternal();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
$(input).on('typeahead:selected', function(e, datum) {
|
|
|
|
$scope.$apply(function() {
|
|
|
|
$scope.setEntityInternal(datum.entity, true);
|
|
|
|
});
|
|
|
|
});
|
|
|
|
})();
|
|
|
|
|
|
|
|
$scope.$watch('clearValue', function() {
|
|
|
|
if (!input) { return; }
|
|
|
|
|
|
|
|
$(input).typeahead('val', '');
|
|
|
|
$scope.clearEntityInternal();
|
|
|
|
});
|
|
|
|
|
|
|
|
$scope.$watch('placeholder', function(title) {
|
|
|
|
input.setAttribute('placeholder', title);
|
|
|
|
});
|
|
|
|
|
|
|
|
$scope.$watch('allowedEntities', function(allowed) {
|
|
|
|
if (!allowed) { return; }
|
|
|
|
$scope.includeTeams = isSupported('team', allowed);
|
|
|
|
$scope.includeRobots = isSupported('robot', allowed);
|
|
|
|
});
|
|
|
|
|
|
|
|
$scope.$watch('namespace', function(namespace) {
|
|
|
|
if (!namespace) { return; }
|
|
|
|
$scope.isAdmin = UserService.isNamespaceAdmin(namespace);
|
|
|
|
$scope.isOrganization = !!UserService.getOrganization(namespace);
|
|
|
|
});
|
|
|
|
|
|
|
|
$scope.$watch('currentEntity', function(entity) {
|
|
|
|
if ($scope.currentEntityInternal != entity) {
|
|
|
|
if (entity) {
|
|
|
|
$scope.setEntityInternal(entity, false);
|
|
|
|
} else {
|
|
|
|
$scope.clearEntityInternal();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
|
|
|
return directiveDefinitionObject;
|
|
|
|
});
|