New create entity dialogs (team and robot)

Fixes https://github.com/coreos-inc/design/issues/230
This commit is contained in:
Joseph Schorr 2016-05-12 17:59:49 -04:00
parent 2274d6ff84
commit 4a543be7a7
31 changed files with 687 additions and 232 deletions

View file

@ -1527,13 +1527,13 @@ a:focus {
-webkit-filter: grayscale(100%); -webkit-filter: grayscale(100%);
} }
.co-dialog .co-tab-content { .co-dialog .co-tab-content {
padding: 16px; padding: 16px;
padding-bottom: 30px; padding-bottom: 30px;
} }
.co-dialog .co-tab-content h3 { .co-dialog .co-tab-content h3 {
margin-top: 0px;
margin-bottom: 0px; margin-bottom: 0px;
} }
@ -1574,3 +1574,9 @@ a:focus {
height: auto; height: auto;
} }
} }
.co-modal-body-scrollable {
overflow-y: auto;
overflow-x: hidden;
max-height: 400px;
}

View file

@ -0,0 +1,53 @@
.create-entity-dialog-element .modal-body {
min-height: 300px;
}
.create-entity-dialog-element form {
padding: 10px;
}
.create-entity-dialog-element .help-text {
color: #aaa;
margin-top: 10px;
}
.create-entity-dialog-element h4 .fa {
margin-left: 4px;
margin-right: 4px;
}
.create-entity-dialog-element label {
margin-top: 4px;
}
.create-entity-dialog-element .co-table {
margin-top: 20px;
}
.create-entity-dialog-element .fa-hdd-o {
margin-right: 4px;
vertical-align: middle;
}
.create-entity-dialog-element .co-filter-box {
display: block;
float: right;
margin-bottom: 20px;
}
.create-entity-dialog-element .co-filter-box .filter-message {
left: -180px;
top: 4px;
}
.create-entity-dialog-element .co-filter-box input {
width: 100%;
padding-top: 2px;
padding-bottom: 2px;
height: 28px;
}
.create-entity-dialog-element label .avatar {
vertical-align: text-bottom;
margin-left: 4px;
}

View file

@ -7,6 +7,7 @@
overflow: hidden; overflow: hidden;
text-align: center; text-align: center;
height: 42px; height: 42px;
pointer-events: none;
} }
@keyframes flow-down-up { @keyframes flow-down-up {

View file

@ -7,12 +7,21 @@
padding-left: 10px; padding-left: 10px;
} }
.role-group.small .btn {
padding: 2px;
padding-left: 10px;
}
.new-role-group .btn .caret { .new-role-group .btn .caret {
position: absolute; position: absolute;
top: 13px; top: 13px;
right: 7px; right: 7px;
} }
.role-group.small .btn .caret {
top: 11px;
}
.new-role-group .role-help-text { .new-role-group .role-help-text {
font-size: 12px; font-size: 12px;
color: #ccc; color: #ccc;

View file

@ -0,0 +1,98 @@
<div class="create-entity-dialog-element">
<div class="modal fade co-dialog wider">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" ng-click="hide()" aria-hidden="true">&times;</button>
<h4 class="modal-title" ng-show="view == 'enterName' || view == 'creating'">
<i class="fa {{ entityIcon }}"></i>
Create {{ entityTitle }}
</h4>
<h4 class="modal-title" ng-show="view == 'addperms' || view == 'addingperms'">
Add permissions for <i class="fa {{ entityIcon }}"></i> {{ entity.name }}
</h4>
</div> <!-- /.model-header -->
<div class="modal-body" ng-show="view == 'creating' || view == 'addingperms'">
<div class="cor-loader"></div>
</div>
<div class="modal-body co-modal-body-scrollable" ng-show="view == 'addperms'">
<span class="co-filter-box">
<span class="filter-message" ng-if="options.filter">
Showing {{ orderedRepositories.entries.length }} of {{ repositories.length }} repositories
</span>
<input class="form-control" type="text" ng-model="options.filter" placeholder="Filter Repositories...">
</span>
<label>
Select repositories in
<span class="avatar" size="16" data="namespace.avatar"></span>
{{ info.namespace }}:
</label>
<table class="co-table" style="margin-bottom: 210px;">
<thead>
<td class="checkbox-col"></td>
<td ng-class="TableService.tablePredicateClass('name', options.predicate, options.reverse)">
<a ng-click="TableService.orderBy('name', options)">Repository Name</a>
</td>
<td>Permission</td>
<td ng-class="TableService.tablePredicateClass('last_modified_datetime', options.predicate, options.reverse)">
<a ng-click="TableService.orderBy('last_modified_datetime', options)">Last Updated</a>
</td>
</thead>
<tr class="co-checkable-row"
ng-repeat="repo in orderedRepositories.visibleEntries"
ng-class="checkedRepos.isChecked(repo, checkedRepos.checked) ? 'checked' : ''"
bindonce>
<td>
<span class="cor-checkable-item" controller="checkedRepos" item="repo"></span>
</td>
<td>
<i class="fa fa-hdd-o"></i>
<span bo-text="repo.name"></span>
</td>
<td>
<span class="role-group small" current-role="repo.permission"
roles="repoRolesOrNone"
role-changed="setRole(role, repo)"></span>
</td>
<td>
<span ng-if="repo.last_modified">
{{ repo.last_modified * 1000 | amCalendar }}
</span>
<span class="empty" ng-if="!repo.last_modified">(Empty Repository)</span>
</td>
</tr>
</table>
<div class="empty" ng-if="!orderedRepositories.entries.length"
style="margin-top: 20px;">
<div class="empty-primary-msg">No matching repositories found.</div>
<div class="empty-secondary-msg">Try expanding your filtering terms.</div>
</div>
</div>
<div class="modal-body" ng-show="view == 'enterName'">
<form name="enterNameForm" ng-submit="createEntity()">
<label>Provide a name for your new {{ entityTitle }}:</label>
<input type="text" class="form-control" ng-model="entityName" ng-pattern="entityNameRegexObj" required>
<div class="help-text">
Choose a name to inform your teammates
about this {{ entityTitle }}. Must match {{ entityNameRegex }}.
</div>
</form>
</div> <!-- /.modal-body -->
<div class="modal-footer" ng-show="view == 'addperms'">
<button type="button" class="btn btn-primary" ng-click="addPermissions()"
ng-show="checkedRepos.checked.length">Add permissions</button>
<button type="button" class="btn btn-default" ng-click="hide()">Close</button>
</div> <!-- /.footer-body -->
<div class="modal-footer" ng-show="view == 'enterName'">
<button type="button" class="btn btn-primary" ng-click="createEntity()"
ng-disabled="enterNameForm.$invalid">Create {{ entityTitle }}</button>
<button type="button" class="btn btn-default" ng-click="hide()">Cancel</button>
</div> <!-- /.footer-body -->
</div>
</div>
</div>
</div>

View file

@ -0,0 +1,9 @@
<div class="create-robot-dialog-element">
<div ng-if="info">
<div class="create-entity-dialog" info="info" entity-title="robot account"
entity-kind="user"
entity-icon="ci-robot" entity-name-regex="{{ ROBOT_PATTERN }}"
entity-create-requested="createRobot(name, callback)"
entity-create-completed="robotFinished(entity)"></div>
</div>
</div>

View file

@ -0,0 +1,9 @@
<div class="create-team-dialog-element">
<div ng-if="info">
<div class="create-entity-dialog" info="info" entity-title="team"
entity-kind="team"
entity-icon="fa-group" entity-name-regex="{{ TEAM_PATTERN }}"
entity-create-requested="createTeam(name, callback)"
entity-create-completed="teamFinished(entity)"></div>
</div>
</div>

View file

@ -14,12 +14,12 @@
</li> </li>
<li role="presentation" ng-show="includeTeams && isOrganization && !lazyLoading && isAdmin"> <li role="presentation" ng-show="includeTeams && isOrganization && !lazyLoading && isAdmin">
<a role="menuitem" class="new-action" tabindex="-1" ng-click="createTeam()"> <a role="menuitem" class="new-action" tabindex="-1" ng-click="askCreateTeam()">
<i class="fa fa-group"></i> Create team <i class="fa fa-group"></i> Create team
</a> </a>
</li> </li>
<li role="presentation" ng-show="includeRobots && !lazyLoading && isAdmin"> <li role="presentation" ng-show="includeRobots && !lazyLoading && isAdmin">
<a role="menuitem" class="new-action" tabindex="-1" ng-click="createRobot()"> <a role="menuitem" class="new-action" tabindex="-1" ng-click="askCreateRobot()">
<i class="fa ci-robot"></i> <i class="fa ci-robot"></i>
Create robot account Create robot account
</a> </a>
@ -70,4 +70,8 @@
</li> </li>
</ul> </ul>
</div> </div>
<div class="create-team-dialog" info="createTeamInfo"
team-created="handleTeamCreated(team)"></div>
<div class="create-robot-dialog" info="createRobotInfo"
robot-created="handleRobotCreated(robot)"></div>
</span> </span>

View file

@ -38,6 +38,7 @@
<td class="first-col">Pull Credentials:</td> <td class="first-col">Pull Credentials:</td>
<td> <td>
<div class="entity-search" namespace="repository.namespace" <div class="entity-search" namespace="repository.namespace"
for-repository="repository"
placeholder="'Choose Pull Credentials'" placeholder="'Choose Pull Credentials'"
allowed-entities="['robot']" allowed-entities="['robot']"
clear-value="clearCounter" clear-value="clearCounter"

View file

@ -85,12 +85,12 @@
Namespace {{ getNamespace(currentPageContext) }} Namespace {{ getNamespace(currentPageContext) }}
</li> </li>
<li ng-if="isOrganization(getNamespace(currentPageContext)) && canAdmin(getNamespace(currentPageContext))"> <li ng-if="isOrganization(getNamespace(currentPageContext)) && canAdmin(getNamespace(currentPageContext))">
<a ng-click="createTeam(currentPageContext)"> <a ng-click="askCreateTeam(currentPageContext)">
<i class="fa fa-group"></i> New Team <i class="fa fa-group"></i> New Team
</a> </a>
</li> </li>
<li ng-if="canAdmin(getNamespace(currentPageContext))"> <li ng-if="canAdmin(getNamespace(currentPageContext))">
<a ng-click="createRobot(currentPageContext)"> <a ng-click="askCreateRobot(currentPageContext)">
<i class="fa ci-robot"></i> New Robot Account <i class="fa ci-robot"></i> New Robot Account
</a> </a>
</li> </li>
@ -207,6 +207,14 @@
</ul> </ul>
</div> </div>
<div class="create-robot-dialog" info="createRobotInfo"
robot-created="handleRobotCreated(robot, currentPageContext)">
</div>
<div class="create-team-dialog" info="createTeamInfo"
team-created="handleTeamCreated(team, currentPageContext)">
</div>
<div class="dockerfile-build-dialog" <div class="dockerfile-build-dialog"
show-now="showBuildDialogCounter" show-now="showBuildDialogCounter"
repository="currentPageContext.repository" repository="currentPageContext.repository"

View file

@ -30,7 +30,7 @@
</thead> </thead>
<tbody> <tbody>
<tr ng-repeat="repository in orderedRepositories"> <tr ng-repeat="repository in orderedRepositories.entries">
<td class="repo-name-icon"> <td class="repo-name-icon">
<span class="avatar" size="24" data="getAvatarData(repository.namespace)"></span> <span class="avatar" size="24" data="getAvatarData(repository.namespace)"></span>
<a href="/repository/{{ repository.namespace }}/{{ repository.name }}"> <a href="/repository/{{ repository.namespace }}/{{ repository.name }}">

View file

@ -61,7 +61,7 @@
<span class="co-filter-box"> <span class="co-filter-box">
<span class="page-controls" total-count="tags.length" current-page="options.page" page-size="tagsPerPage"></span> <span class="page-controls" total-count="tags.length" current-page="options.page" page-size="tagsPerPage"></span>
<input class="form-control" type="text" ng-model="options.tagFilter" placeholder="Filter Tags..."> <input class="form-control" type="text" ng-model="options.filter" placeholder="Filter Tags...">
</span> </span>
</div> </div>

View file

@ -106,6 +106,8 @@
<tr class="add-row" ng-if-media="'(min-width: 768px)'"> <tr class="add-row" ng-if-media="'(min-width: 768px)'">
<td id="add-entity-permission" class="admin-search"> <td id="add-entity-permission" class="admin-search">
<span class="entity-search" namespace="repository.namespace" <span class="entity-search" namespace="repository.namespace"
for-repository="repository"
skip-permissions="true"
placeholder="'Select a ' + (repository.is_organization ? 'team or ' : '') + 'user...'" placeholder="'Select a ' + (repository.is_organization ? 'team or ' : '') + 'user...'"
current-entity="addPermissionInfo.entity"></span> current-entity="addPermissionInfo.entity"></span>
</td> </td>
@ -125,6 +127,8 @@
<!-- Mobile add permissions --> <!-- Mobile add permissions -->
<div class="mobile-add-row" ng-if-media="'(max-width: 767px)'"> <div class="mobile-add-row" ng-if-media="'(max-width: 767px)'">
<span class="entity-search" namespace="repository.namespace" <span class="entity-search" namespace="repository.namespace"
for-repository="repository"
skip-permissions="true"
placeholder="'Select a ' + (repository.is_organization ? 'team or ' : '') + 'user...'" placeholder="'Select a ' + (repository.is_organization ? 'team or ' : '') + 'user...'"
current-entity="addPermissionInfo.entity" current-entity="addPermissionInfo.entity"
pull-right="true"></span> pull-right="true"></span>

View file

@ -4,12 +4,9 @@
<div ng-show="!loading"> <div ng-show="!loading">
<div class="manager-header" header-title="Robot Accounts"> <div class="manager-header" header-title="Robot Accounts">
<span class="popup-input-button" pattern="ROBOT_PATTERN" <button class="btn btn-primary" ng-click="askCreateRobot()" ng-show="isEnabled">
placeholder="'Robot Account Name'" <i class="fa fa-plus" style="margin-right: 4px;"></i> Create Robot Account
submitted="createRobot(value)" </button>
ng-show="isEnabled">
<i class="fa fa-plus"></i> Create Robot Account
</span>
</div> </div>
<div class="section-description-header"> <div class="section-description-header">
@ -134,5 +131,6 @@
</table> </table>
</div> </div>
<div class="create-robot-dialog" info="createRobotInfo" robot-created="robotCreated()"></div>
<div class="robot-credentials-dialog" info="robotDisplayInfo"></div> <div class="robot-credentials-dialog" info="robotDisplayInfo"></div>
</div> </div>

View file

@ -13,23 +13,17 @@
</button> </button>
</div> </div>
</div> </div>
<span class="popup-input-button visible-xs" ng-if="!showingMembers"
pattern="TEAM_PATTERN" placeholder="'Team Name'"
submitted="createTeam(value)" ng-show="organization.is_admin">
<i class="fa fa-plus" style="margin-right: 6px;"></i> Create New Team
</span>
</div> </div>
<!-- Teams List --> <!-- Teams List -->
<div ng-show="!showingMembers"> <div ng-show="!showingMembers">
<div class="row" style="margin-left: 0px; margin-right: 0px;"> <div class="row" style="margin-left: 0px; margin-right: 0px;">
<span class="popup-input-button hidden-xs" <button class="btn btn-primary hidden-xs"
pattern="TEAM_PATTERN" placeholder="'Team Name'" ng-show="organization.is_admin"
submitted="createTeam(value)" ng-show="organization.is_admin" style="margin-bottom: 10px; float: right;"
style="margin-bottom: 10px;"> ng-click="askCreateTeam()">
<i class="fa fa-plus" style="margin-right: 6px;"></i> Create New Team <i class="fa fa-plus" style="margin-right: 4px;"></i> Create New Team
</span> </button>
</div> </div>
<div class="row hidden-xs"> <div class="row hidden-xs">
@ -155,6 +149,8 @@
</table> </table>
</div> </div>
<div class="create-team-dialog" info="createTeamInfo" team-created="handleTeamCreated(team)"></div>
<!-- Remove member confirm --> <!-- Remove member confirm -->
<div class="cor-confirm-dialog" <div class="cor-confirm-dialog"
dialog-context="removeMemberInfo" dialog-context="removeMemberInfo"

View file

@ -18,9 +18,7 @@ angular.module('quay').directive('repoPanelTags', function () {
'getImages': '&getImages' 'getImages': '&getImages'
}, },
controller: function($scope, $element, $filter, $location, ApiService, UIService, VulnerabilityService) { controller: function($scope, $element, $filter, $location, ApiService, UIService, VulnerabilityService, TableService) {
var orderBy = $filter('orderBy');
$scope.maxTrackCount = 5; $scope.maxTrackCount = 5;
$scope.checkedTags = UIService.createCheckStateController([], 'name'); $scope.checkedTags = UIService.createCheckStateController([], 'name');
@ -45,32 +43,26 @@ angular.module('quay').directive('repoPanelTags', function () {
var setTagState = function() { var setTagState = function() {
if (!$scope.repository || !$scope.selectedTags) { return; } if (!$scope.repository || !$scope.selectedTags) { return; }
var tags = []; // Build a list of all the tags, with extending information.
var allTags = []; var allTags = [];
// Build a list of tags and filtered tags.
for (var tag in $scope.repository.tags) { for (var tag in $scope.repository.tags) {
if (!$scope.repository.tags.hasOwnProperty(tag)) { continue; } if (!$scope.repository.tags.hasOwnProperty(tag)) { continue; }
var tagData = $scope.repository.tags[tag]; var tagData = $scope.repository.tags[tag];
var tagInfo = $.extend(tagData, { var tagInfo = $.extend(tagData, {
'name': tag, 'name': tag,
'last_modified_datetime': (new Date(tagData.last_modified || 0)).valueOf() * (-1) 'last_modified_datetime': TableService.getReversedTimestamp(tagData.last_modified)
}); });
allTags.push(tagInfo); allTags.push(tagInfo);
if (!$scope.options.tagFilter || tag.indexOf($scope.options.tagFilter) >= 0 ||
tagInfo.image_id.indexOf($scope.options.tagFilter) >= 0) {
tags.push(tagInfo);
}
} }
// Sort the tags by the predicate and the reverse, and map the information. // Sort the tags by the predicate and the reverse, and map the information.
var imageIDs = []; var imageIDs = [];
var ordered = orderBy(tags, $scope.options.predicate, $scope.options.reverse); var ordered = TableService.buildOrderedItems(allTags, $scope.options,
var checked = []; ['name'], ['last_modified_datetime', 'size']).entries;
var checked = [];
var imageMap = {}; var imageMap = {};
var imageIndexMap = {}; var imageIndexMap = {};
for (var i = 0; i < ordered.length; ++i) { for (var i = 0; i < ordered.length; ++i) {
@ -175,7 +167,7 @@ angular.module('quay').directive('repoPanelTags', function () {
$scope.$watch('options.predicate', setTagState); $scope.$watch('options.predicate', setTagState);
$scope.$watch('options.reverse', setTagState); $scope.$watch('options.reverse', setTagState);
$scope.$watch('options.tagFilter', setTagState); $scope.$watch('options.filter', setTagState);
$scope.$watch('options.page', function(page) { $scope.$watch('options.page', function(page) {
if (page != null && $scope.checkedTags) { if (page != null && $scope.checkedTags) {

View file

@ -0,0 +1,202 @@
/**
* An element which displays a create entity dialog.
*/
angular.module('quay').directive('createEntityDialog', function () {
var directiveDefinitionObject = {
priority: 0,
templateUrl: '/static/directives/create-entity-dialog.html',
replace: false,
transclude: true,
restrict: 'C',
scope: {
'info': '=info',
'entityKind': '@entityKind',
'entityTitle': '@entityTitle',
'entityIcon': '@entityIcon',
'entityNameRegex': '@entityNameRegex',
'entityCreateRequested': '&entityCreateRequested',
'entityCreateCompleted': '&entityCreateCompleted'
},
controller: function($scope, $element, ApiService, UIService, TableService, RolesService, UserService) {
$scope.TableService = TableService;
$scope.options = {
'predicate': 'last_modified_datetime',
'reverse': false,
'filter': ''
};
var handleRepoCheckChange = function() {
$scope.repositories.forEach(function(repo) {
if ($scope.checkedRepos.isChecked(repo)) {
if (repo['permission'] == 'none') {
repo['permission'] = 'read';
}
} else {
repo['permission'] = 'none';
}
});
};
$scope.$on('$destroy', function() {
if ($scope.inBody) {
document.body.removeChild($element[0]);
}
});
$scope.setRole = function(role, repo) {
repo['permission'] = role;
if (role == 'none') {
$scope.checkedRepos.uncheckItem(repo);
} else {
$scope.checkedRepos.checkItem(repo);
}
};
$scope.hide = function() {
$element.find('.modal').modal('hide');
if ($scope.entity) {
$scope.entityCreateCompleted({'entity': $scope.entity});
$scope.entity = null;
}
};
$scope.show = function() {
$scope.entityName = null;
$scope.entity = null;
$scope.creating = false;
$scope.view = 'enterName';
$scope.enterNameForm.$setPristine(true);
// Move the dialog to the body to prevent it from nesting if called
// from within another dialog.
$element.find('.modal').modal({});
$scope.inBody = true;
document.body.appendChild($element[0]);
};
var setRepoState = function() {
if (!$scope.repositories) {
return;
}
$scope.orderedRepositories = TableService.buildOrderedItems($scope.repositories, $scope.options,
['name', 'permission'],
['last_modified_datetime']);
};
var entityCreateCallback = function(entity) {
if (!entity || $scope.info.skip_permissions) {
$scope.entity = entity;
$scope.hide();
return;
}
// Load the repositories under the entity's namespace.
var params = {
'namespace': $scope.info.namespace,
'last_modified': true
};
ApiService.listRepos(null, params).then(function(resp) {
$scope.view = 'addperms';
$scope.entity = entity;
var repos = [];
resp['repositories'].forEach(function(repo) {
repos.push({
'namespace': repo.namespace,
'name': repo.name,
'last_modified': repo.last_modified,
'last_modified_datetime': TableService.getReversedTimestamp(repo.last_modified),
'permission': 'none'
});
});
if (repos.length == 0) {
$scope.hide();
return;
}
$scope.repositories = repos;
$scope.checkedRepos = UIService.createCheckStateController($scope.repositories, 'name');
$scope.checkedRepos.listen(handleRepoCheckChange);
if ($scope.info.repository) {
repos.forEach(function(repo) {
if (repo['namespace'] == $scope.info.repository.namespace &&
repo['name'] == $scope.info.repository.name) {
$scope.checkedRepos.checkItem(repo);
$scope.options.filter = $scope.info.repository.name;
}
});
}
setRepoState();
}, ApiService.errorDisplay('Could not load repositories'));
};
$scope.addPermissions = function() {
$scope.view = 'addingperms';
var repos = $scope.checkedRepos.checked;
var counter = 0;
var addPerm = function() {
if (counter >= repos.length) {
$scope.hide();
return;
}
var repo = repos[counter];
RolesService.setRepositoryRole(repo, repo.permission, $scope.entityKind, $scope.entity.name,
function(status) {
if (status) {
counter++;
addPerm();
} else {
$scope.hide();
}
});
};
addPerm();
};
$scope.createEntity = function() {
$scope.view = 'creating';
$scope.entityCreateRequested({
'name': $scope.entityName,
'callback': entityCreateCallback
});
};
$scope.$watch('options.predicate', setRepoState);
$scope.$watch('options.reverse', setRepoState);
$scope.$watch('options.filter', setRepoState);
$scope.$watch('entityNameRegex', function(r) {
if (r) {
$scope.entityNameRegexObj = new RegExp(r);
}
});
$scope.$watch('info', function(info) {
if (!info || !info.namespace) {
$scope.hide();
return;
}
$scope.namespace = UserService.getNamespace(info.namespace);
if ($scope.namespace) {
$scope.show();
}
});
}
};
return directiveDefinitionObject;
});

View file

@ -0,0 +1,43 @@
/**
* An element which displays a dialog for creating a robot account.
*/
angular.module('quay').directive('createRobotDialog', function () {
var directiveDefinitionObject = {
priority: 0,
templateUrl: '/static/directives/create-robot-dialog.html',
replace: false,
transclude: true,
restrict: 'C',
scope: {
'info': '=info',
'robotCreated': '&robotCreated'
},
controller: function($scope, $element, ApiService, UserService) {
$scope.ROBOT_PATTERN = ROBOT_PATTERN;
$scope.robotFinished = function(robot) {
$scope.robotCreated({'robot': robot});
};
$scope.createRobot = function(name, callback) {
var organization = $scope.info.namespace;
if (!UserService.isOrganization(organization)) {
organization = null;
}
var params = {
'robot_shortname': name
};
var errorDisplay = ApiService.errorDisplay('Cannot create robot account', function() {
callback(null);
});
ApiService.createRobot(organization, null, params).then(function(resp) {
callback(resp);
}, errorDisplay);
};
}
};
return directiveDefinitionObject;
});

View file

@ -0,0 +1,44 @@
/**
* An element which displays a dialog for creating a team.
*/
angular.module('quay').directive('createTeamDialog', function () {
var directiveDefinitionObject = {
priority: 0,
templateUrl: '/static/directives/create-team-dialog.html',
replace: false,
transclude: true,
restrict: 'C',
scope: {
'info': '=info',
'teamCreated': '&teamCreated'
},
controller: function($scope, $element, ApiService, UserService) {
$scope.TEAM_PATTERN = TEAM_PATTERN;
$scope.teamFinished = function(team) {
$scope.teamCreated({'team': team});
};
$scope.createTeam = function(name, callback) {
var data = {
'name': name,
'role': 'member'
};
var params = {
'orgname': $scope.info.namespace,
'teamname': name
};
var errorDisplay = ApiService.errorDisplay('Cannot create team', function() {
callback(null);
});
ApiService.updateOrganizationTeam(data, params).then(function(resp) {
callback(resp);
}, errorDisplay);
};
}
};
return directiveDefinitionObject;
});

View file

@ -18,6 +18,8 @@ angular.module('quay').directive('entitySearch', function () {
scope: { scope: {
'namespace': '=namespace', 'namespace': '=namespace',
'placeholder': '=placeholder', 'placeholder': '=placeholder',
'forRepository': '=forRepository',
'skipPermissions': '=skipPermissions',
// Default: ['user', 'team', 'robot'] // Default: ['user', 'team', 'robot']
'allowedEntities': '=allowedEntities', 'allowedEntities': '=allowedEntities',
@ -41,7 +43,7 @@ angular.module('quay').directive('entitySearch', function () {
// True if the menu should pull right. // True if the menu should pull right.
'pullRight': '@pullRight' 'pullRight': '@pullRight'
}, },
controller: function($rootScope, $scope, $element, Restangular, UserService, ApiService, UtilService, Config, CreateService) { controller: function($rootScope, $scope, $element, Restangular, UserService, ApiService, UtilService, Config) {
$scope.lazyLoading = true; $scope.lazyLoading = true;
$scope.teams = null; $scope.teams = null;
@ -55,6 +57,8 @@ angular.module('quay').directive('entitySearch', function () {
$scope.includeOrgs = false; $scope.includeOrgs = false;
$scope.currentEntityInternal = $scope.currentEntity; $scope.currentEntityInternal = $scope.currentEntity;
$scope.createRobotInfo = null;
$scope.createTeamInfo = null;
$scope.Config = Config; $scope.Config = Config;
@ -91,18 +95,30 @@ angular.module('quay').directive('entitySearch', function () {
} }
}; };
$scope.createTeam = function() { $scope.askCreateTeam = function() {
CreateService.askCreateTeam($scope.namespace, function(created) { $scope.createTeamInfo = {
$scope.setEntity(created.name, 'team', false, created.avatar); 'namespace': $scope.namespace,
$scope.teams[created.name] = created; 'repository': $scope.forRepository,
}); 'skip_permissions': $scope.skipPermissions
};
}; };
$scope.createRobot = function() { $scope.askCreateRobot = function() {
CreateService.askCreateRobot($scope.namespace, function(created) { $scope.createRobotInfo = {
$scope.setEntity(created.name, 'user', true, created.avatar); 'namespace': $scope.namespace,
$scope.robots.push(created); 'repository': $scope.forRepository,
}); 'skip_permissions': $scope.skipPermissions
};
};
$scope.handleTeamCreated = function(created) {
$scope.setEntity(created.name, 'team', false, created.avatar);
$scope.teams[created.name] = created;
};
$scope.handleRobotCreated = function(created) {
$scope.setEntity(created.name, 'user', true, created.avatar);
$scope.robots.push(created);
}; };
$scope.setEntity = function(name, kind, is_robot, avatar) { $scope.setEntity = function(name, kind, is_robot, avatar) {

View file

@ -13,7 +13,7 @@ angular.module('quay').directive('headerBar', function () {
scope: { scope: {
}, },
controller: function($rootScope, $scope, $element, $location, $timeout, hotkeys, UserService, controller: function($rootScope, $scope, $element, $location, $timeout, hotkeys, UserService,
PlanService, ApiService, NotificationService, Config, CreateService, Features, PlanService, ApiService, NotificationService, Config, Features,
DocumentationService, ExternalLoginService) { DocumentationService, ExternalLoginService) {
$scope.externalSigninUrl = ExternalLoginService.getSingleSigninUrl(); $scope.externalSigninUrl = ExternalLoginService.getSingleSigninUrl();
@ -268,24 +268,36 @@ angular.module('quay').directive('headerBar', function () {
$location.url('/repository/' + context.repository.namespace + '/' + context.repository.name + '/build/' + build.id); $location.url('/repository/' + context.repository.namespace + '/' + context.repository.name + '/build/' + build.id);
}; };
$scope.createRobot = function(context) { $scope.handleRobotCreated = function(created, context) {
var namespace = $scope.getNamespace(context); var namespace = $scope.getNamespace(context);
CreateService.askCreateRobot(namespace, function(created) { if (UserService.isOrganization(namespace)) {
if (UserService.isOrganization(namespace)) { $location.url('/organization/' + namespace + '?tab=robots&showRobot=' + created.name);
$location.url('/organization/' + namespace + '?tab=robots&showRobot=' + created.name); } else {
} else { $location.url('/user/' + namespace + '?tab=robots&showRobot=' + created.name);
$location.url('/user/' + namespace + '?tab=robots&showRobot=' + created.name); }
}
});
}; };
$scope.createTeam = function(context) { $scope.handleTeamCreated = function(created, context) {
var namespace = $scope.getNamespace(context);
$location.url('/organization/' + namespace + '/teams/' + created.name);
};
$scope.askCreateRobot = function(context) {
var namespace = $scope.getNamespace(context); var namespace = $scope.getNamespace(context);
if (!namespace || !UserService.isNamespaceAdmin(namespace)) { return; } if (!namespace || !UserService.isNamespaceAdmin(namespace)) { return; }
CreateService.askCreateTeam(namespace, function(created) { $scope.createRobotInfo = {
$location.url('/organization/' + namespace + '/teams/' + created.name); 'namespace': namespace
}); };
};
$scope.askCreateTeam = function(context) {
var namespace = $scope.getNamespace(context);
if (!namespace || !UserService.isNamespaceAdmin(namespace)) { return; }
$scope.createTeamInfo = {
'namespace': namespace
};
}; };
} }
}; };

View file

@ -13,30 +13,22 @@ angular.module('quay').directive('repoListTable', function () {
'namespaces': '=namespaces', 'namespaces': '=namespaces',
'starToggled': '&starToggled' 'starToggled': '&starToggled'
}, },
controller: function($scope, $element, $filter) { controller: function($scope, $element, $filter, TableService) {
var orderBy = $filter('orderBy');
$scope.repositories = null; $scope.repositories = null;
$scope.orderedRepositories = []; $scope.orderedRepositories = [];
$scope.maxPopularity = 0; $scope.maxPopularity = 0;
$scope.options = { $scope.options = {
'predicate': 'popularity', 'predicate': 'popularity',
'reverse': true 'reverse': false,
'filter': null
}; };
var buildOrderedRepositories = function() { var buildOrderedRepositories = function() {
if (!$scope.repositories) { return; } if (!$scope.repositories) { return; }
var modifier = $scope.options.reverse ? '-' : '';
var fields = [modifier + $scope.options.predicate];
// Secondary ordering by full name. $scope.orderedRepositories = TableService.buildOrderedItems($scope.repositories, $scope.options,
if ($scope.options.predicate != 'full_name') { [], ['last_modified_datetime', 'popularity'])
fields.push('full_name');
}
var ordered = orderBy($scope.repositories, fields, false);
$scope.orderedRepositories = ordered;
}; };
$scope.tablePredicateClass = function(name, predicate, reverse) { $scope.tablePredicateClass = function(name, predicate, reverse) {
@ -92,7 +84,7 @@ angular.module('quay').directive('repoListTable', function () {
(resource.value || []).forEach(function(repository) { (resource.value || []).forEach(function(repository) {
var repositoryInfo = $.extend(repository, { var repositoryInfo = $.extend(repository, {
'full_name': repository.namespace + '/' + repository.name, 'full_name': repository.namespace + '/' + repository.name,
'last_modified_datetime': (new Date(repository.last_modified || 0)).valueOf() * (-1) 'last_modified_datetime': TableService.getReversedTimestamp(repository.last_modified),
}); });
$scope.repositories.push(repositoryInfo); $scope.repositories.push(repositoryInfo);

View file

@ -28,7 +28,7 @@ angular.module('quay').directive('repositoryPermissionsTable', function () {
'repository': '=repository', 'repository': '=repository',
'isEnabled': '=isEnabled' 'isEnabled': '=isEnabled'
}, },
controller: function($scope, $element, ApiService, Restangular, UtilService, RolesService) { controller: function($scope, $element, ApiService, RolesService) {
$scope.permissionResources = {'team': {}, 'user': {}}; $scope.permissionResources = {'team': {}, 'user': {}};
$scope.permissionCache = {}; $scope.permissionCache = {};
$scope.permissions = {}; $scope.permissions = {};
@ -69,13 +69,6 @@ angular.module('quay').directive('repositoryPermissionsTable', function () {
loadAllPermissions(); loadAllPermissions();
var getPermissionEndpoint = function(entityName, kind) {
var namespace = $scope.repository.namespace;
var name = $scope.repository.name;
var url = UtilService.getRestUrl('repository', namespace, name, 'permissions', kind, entityName);
return Restangular.one(url);
};
$scope.buildEntityForPermission = function(permission, kind) { $scope.buildEntityForPermission = function(permission, kind) {
var key = permission.name + ':' + kind; var key = permission.name + ':' + kind;
if ($scope.permissionCache[key]) { if ($scope.permissionCache[key]) {
@ -146,51 +139,36 @@ angular.module('quay').directive('repositoryPermissionsTable', function () {
}; };
$scope.deleteRole = function(entityName, kind) { $scope.deleteRole = function(entityName, kind) {
var errorHandler = ApiService.errorDisplay('Cannot change permission', function(resp) { RolesService.deleteRepositoryRole($scope.repository, kind, entityName, function(status) {
if (resp.status == 409) { if (status) {
return 'Cannot change permission as you do not have the authority'; delete $scope.permissions[kind][entityName];
} }
}); });
var endpoint = getPermissionEndpoint(entityName, kind);
endpoint.customDELETE().then(function() {
delete $scope.permissions[kind][entityName];
}, errorHandler);
}; };
$scope.addRole = function(entityName, role, kind, opt_callback) { $scope.addRole = function(entityName, role, kind, opt_callback) {
var permission = { RolesService.setRepositoryRole($scope.repository, role, kind, entityName, function(status, result) {
'role': role,
};
var errorHandler = ApiService.errorDisplay('Cannot change permission', function() {
opt_callback && opt_callback(false);
$scope.addPermissionInfo = { $scope.addPermissionInfo = {
'role': readRole 'role': readRole
}; };
if (status) {
$scope.permissions[kind][entityName] = result;
}
opt_callback && opt_callback(status);
}); });
var endpoint = getPermissionEndpoint(entityName, kind);
endpoint.customPUT(permission).then(function(result) {
$scope.permissions[kind][entityName] = result;
$scope.addPermissionInfo = {
'role': readRole
};
opt_callback && opt_callback(true)
}, errorHandler);
}; };
$scope.setRole = function(role, entityName, kind) { $scope.setRole = function(role, entityName, kind) {
var errorDisplay = ApiService.errorDisplay(function(resp) { var currentRole = $scope.permissions[kind][entityName].role;
$scope.permissions[kind][entityName] = {'role': currentRole}; RolesService.setRepositoryRole($scope.repository, role, kind, entityName, function(status) {
if (status) {
$scope.permissions[kind][entityName]['role'] = role;
} else {
$scope.permissions[kind][entityName]['role'] = currentRole;
}
}); });
var permission = $scope.permissions[kind][entityName];
var currentRole = permission.role;
permission.role = role;
var endpoint = getPermissionEndpoint(entityName, kind);
endpoint.customPUT(permission).then(function() {}, errorDisplay);
}; };
} }
}; };

View file

@ -13,15 +13,14 @@ angular.module('quay').directive('robotsManager', function () {
'user': '=user', 'user': '=user',
'isEnabled': '=isEnabled' 'isEnabled': '=isEnabled'
}, },
controller: function($scope, $element, ApiService, $routeParams, $location, CreateService, controller: function($scope, $element, ApiService, $routeParams, $location, Config, $rootScope) {
Config, $rootScope) {
$scope.ROBOT_PATTERN = ROBOT_PATTERN;
$scope.robots = null; $scope.robots = null;
$scope.loading = false; $scope.loading = false;
$scope.Config = Config; $scope.Config = Config;
$scope.feedback = null; $scope.feedback = null;
$scope.robotDisplayInfo = null; $scope.robotDisplayInfo = null;
$scope.createRobotInfo = null;
// Listen for route changes and update the tabs accordingly. // Listen for route changes and update the tabs accordingly.
var locationListener = $rootScope.$on('$routeUpdate', function(){ var locationListener = $rootScope.$on('$routeUpdate', function(){
@ -96,22 +95,10 @@ angular.module('quay').directive('robotsManager', function () {
return name.substr(0, plus); return name.substr(0, plus);
}; };
$scope.createRobot = function(name) { $scope.askCreateRobot = function() {
if (!name) { return; } $scope.createRobotInfo = {
'namespace': $scope.organization ? $scope.organization.name : $scope.user.username
CreateService.createRobotAccount(ApiService, !!$scope.organization, $scope.organization ? $scope.organization.name : '', name, };
function(created) {
created.teams = [];
created.repositories = [];
$scope.robots.push(created);
$scope.feedback = {
'kind': 'success',
'message': 'Robot account {robot} was created',
'data': {
'robot': name
}
};
});
}; };
$scope.deleteRobot = function(info) { $scope.deleteRobot = function(info) {
@ -131,7 +118,6 @@ angular.module('quay').directive('robotsManager', function () {
}, ApiService.errorDisplay('Cannot delete robot account')); }, ApiService.errorDisplay('Cannot delete robot account'));
}; };
$scope.askDeleteRobot = function(info) { $scope.askDeleteRobot = function(info) {
bootbox.confirm('Are you sure you want to delete robot ' + info.name + '?', function(resp) { bootbox.confirm('Are you sure you want to delete robot ' + info.name + '?', function(resp) {
if (resp) { if (resp) {
@ -140,6 +126,10 @@ angular.module('quay').directive('robotsManager', function () {
}); });
}; };
$scope.robotCreated = function() {
update();
};
var update = function() { var update = function() {
if (!$scope.user && !$scope.organization) { return; } if (!$scope.user && !$scope.organization) { return; }
if ($scope.loading || !$scope.isEnabled) { return; } if ($scope.loading || !$scope.isEnabled) { return; }

View file

@ -12,8 +12,7 @@ angular.module('quay').directive('teamsManager', function () {
'organization': '=organization', 'organization': '=organization',
'isEnabled': '=isEnabled' 'isEnabled': '=isEnabled'
}, },
controller: function($scope, $element, ApiService, CreateService, $timeout, UserService) { controller: function($scope, $element, ApiService, $timeout, UserService) {
$scope.TEAM_PATTERN = TEAM_PATTERN;
$scope.teamRoles = [ $scope.teamRoles = [
{ 'id': 'member', 'title': 'Member', 'kind': 'default' }, { 'id': 'member', 'title': 'Member', 'kind': 'default' },
{ 'id': 'creator', 'title': 'Creator', 'kind': 'success' }, { 'id': 'creator', 'title': 'Creator', 'kind': 'success' },
@ -27,6 +26,7 @@ angular.module('quay').directive('teamsManager', function () {
$scope.showingMembers = false; $scope.showingMembers = false;
$scope.fullMemberList = null; $scope.fullMemberList = null;
$scope.feedback = null; $scope.feedback = null;
$scope.createTeamInfo = null;
var loadTeamMembers = function() { var loadTeamMembers = function() {
if (!$scope.organization || !$scope.isEnabled) { return; } if (!$scope.organization || !$scope.isEnabled) { return; }
@ -107,35 +107,27 @@ angular.module('quay').directive('teamsManager', function () {
}, errorHandler); }, errorHandler);
}; };
$scope.createTeam = function(teamname) { $scope.askCreateTeam = function(teamname) {
if (!teamname) { $scope.createTeamInfo = {
return; 'namespace': $scope.organization.name
} };
};
if ($scope.organization.teams[teamname]) { $scope.handleTeamCreated = function(created) {
$('#team-' + teamname).removeClass('highlight'); var teamname = created.name;
setTimeout(function() { $scope.organization.teams[teamname] = created;
$('#team-' + teamname).addClass('highlight'); $scope.members[teamname] = {};
}, 10); $scope.members[teamname].members = [];
return; $scope.organization.ordered_teams.push(teamname);
} $scope.orderedTeams.push(created);
var orgname = $scope.organization.name; $scope.feedback = {
CreateService.createOrganizationTeam(ApiService, orgname, teamname, function(created) { 'kind': 'success',
$scope.organization.teams[teamname] = created; 'message': 'Team {team} created',
$scope.members[teamname] = {}; 'data': {
$scope.members[teamname].members = []; 'team': teamname
$scope.organization.ordered_teams.push(teamname); }
$scope.orderedTeams.push(created); };
$scope.feedback = {
'kind': 'success',
'message': 'Team {team} created',
'data': {
'team': teamname
}
};
});
}; };
$scope.askDeleteTeam = function(teamname) { $scope.askDeleteTeam = function(teamname) {

View file

@ -41,6 +41,7 @@
var loadUser = function() { var loadUser = function() {
$scope.userResource = ApiService.getUserInformationAsResource({'username': username}).get(function(user) { $scope.userResource = ApiService.getUserInformationAsResource({'username': username}).get(function(user) {
$scope.context.viewuser = user; $scope.context.viewuser = user;
$scope.viewuser = user;
// Load the repositories. // Load the repositories.
$timeout(function() { $timeout(function() {

View file

@ -1,61 +0,0 @@
/**
* Service which exposes various methods for creating entities on the backend.
*/
angular.module('quay').factory('CreateService', ['ApiService', 'UserService', function(ApiService, UserService) {
var createService = {};
createService.createRobotAccount = function(ApiService, is_org, orgname, name, callback) {
ApiService.createRobot(is_org ? orgname : null, null, {'robot_shortname': name})
.then(callback, ApiService.errorDisplay('Cannot create robot account'));
};
createService.createOrganizationTeam = function(ApiService, orgname, teamname, callback) {
var data = {
'name': teamname,
'role': 'member'
};
var params = {
'orgname': orgname,
'teamname': teamname
};
ApiService.updateOrganizationTeam(data, params)
.then(callback, ApiService.errorDisplay('Cannot create team'));
};
createService.askCreateRobot = function(namespace, callback) {
if (!namespace || !UserService.isNamespaceAdmin(namespace)) { return; }
var isorg = UserService.isOrganization(namespace);
bootbox.prompt('Enter the name of the new robot account', function(robotname) {
if (!robotname) { return; }
var regex = new RegExp(ROBOT_PATTERN);
if (!regex.test(robotname)) {
bootbox.alert('Invalid robot account name');
return;
}
createService.createRobotAccount(ApiService, isorg, namespace, robotname, callback);
});
};
createService.askCreateTeam = function(namespace, callback) {
if (!namespace || !UserService.isNamespaceAdmin(namespace)) { return; }
bootbox.prompt('Enter the name of the new team', function(teamname) {
if (!teamname) { return; }
var regex = new RegExp(TEAM_PATTERN);
if (!regex.test(teamname)) {
bootbox.alert('Invalid team name');
return;
}
createService.createOrganizationTeam(ApiService, namespace, teamname, callback);
});
};
return createService;
}]);

View file

@ -1,20 +1,57 @@
/** /**
* Service which defines the various role groups. * Service which defines the various role groups.
*/ */
angular.module('quay').factory('RolesService', [function() { angular.module('quay').factory('RolesService', ['UtilService', 'Restangular', 'ApiService', function(UtilService, Restangular, ApiService) {
var roleService = {}; var roleService = {};
roleService.repoRoles = [ roleService.repoRolesOrNone = [
{ 'id': 'none', 'title': 'None', 'kind': 'default', 'description': 'No permissions on the repository' },
{ 'id': 'read', 'title': 'Read', 'kind': 'success', 'description': 'Can view and pull from the repository' }, { 'id': 'read', 'title': 'Read', 'kind': 'success', 'description': 'Can view and pull from the repository' },
{ 'id': 'write', 'title': 'Write', 'kind': 'success', 'description': 'Can view, pull and push to the repository' }, { 'id': 'write', 'title': 'Write', 'kind': 'success', 'description': 'Can view, pull and push to the repository' },
{ 'id': 'admin', 'title': 'Admin', 'kind': 'primary', 'description': 'Full admin access, pull and push on the repository' } { 'id': 'admin', 'title': 'Admin', 'kind': 'primary', 'description': 'Full admin access, pull and push on the repository' }
]; ];
roleService.repoRoles = roleService.repoRolesOrNone.slice(1);
roleService.teamRoles = [ roleService.teamRoles = [
{ 'id': 'member', 'title': 'Member', 'kind': 'default', 'description': 'Inherits all permissions of the team' }, { 'id': 'member', 'title': 'Member', 'kind': 'default', 'description': 'Inherits all permissions of the team' },
{ 'id': 'creator', 'title': 'Creator', 'kind': 'success', 'description': 'Member and can create new repositories' }, { 'id': 'creator', 'title': 'Creator', 'kind': 'success', 'description': 'Member and can create new repositories' },
{ 'id': 'admin', 'title': 'Admin', 'kind': 'primary', 'description': 'Full admin access to the organization' } { 'id': 'admin', 'title': 'Admin', 'kind': 'primary', 'description': 'Full admin access to the organization' }
]; ];
var getPermissionEndpoint = function(repository, entityName, kind) {
var namespace = repository.namespace;
var name = repository.name;
var url = UtilService.getRestUrl('repository', namespace, name, 'permissions', kind, entityName);
return Restangular.one(url);
};
roleService.deleteRepositoryRole = function(repository, entityKind, entityName, callback) {
var errorDisplay = ApiService.errorDisplay('Cannot change permission', function(resp) {
callback(false);
});
var endpoint = getPermissionEndpoint(repository, entityName, kind);
endpoint.customDELETE().then(function() {
callback(true);
}, errorHandler);
};
roleService.setRepositoryRole = function(repository, role, entityKind, entityName, callback) {
var errorDisplay = ApiService.errorDisplay('Cannot change permission', function(resp) {
callback(false);
});
var permission = {
'role': role
};
var endpoint = getPermissionEndpoint(repository, entityName, entityKind);
endpoint.customPUT(permission).then(function(resp) {
callback(true, resp);
}, errorDisplay);
};
return roleService; return roleService;
}]); }]);

View file

@ -22,6 +22,14 @@ angular.module('quay').factory('TableService', ['AngularViewArray', function(Ang
options.predicate = predicate; options.predicate = predicate;
}; };
tableService.getReversedTimestamp = function(datetime) {
if (!datetime) {
return -Number.MAX_VALUE;
}
return (new Date(datetime)).valueOf() * (-1);
};
tableService.buildOrderedItems = function(items, options, filterFields, numericFields, opt_extrafilter) { tableService.buildOrderedItems = function(items, options, filterFields, numericFields, opt_extrafilter) {
var orderedItems = AngularViewArray.create(); var orderedItems = AngularViewArray.create();

View file

@ -82,7 +82,7 @@ angular.module('quay').factory('UIService', ['$timeout', '$rootScope', '$locatio
CheckStateController.prototype.rebuildCheckedList_ = function() { CheckStateController.prototype.rebuildCheckedList_ = function() {
var that = this; var that = this;
this.checked = []; this.checked = [];
this.items.forEach(function(item) { this.allItems_.forEach(function(item) {
if (that.allCheckedMap_[item[that.itemKey_]]) { if (that.allCheckedMap_[item[that.itemKey_]]) {
that.checked.push(item); that.checked.push(item);
} }

View file

@ -122,6 +122,19 @@ function(ApiService, CookieService, $rootScope, Config) {
return !!org; return !!org;
}; };
userService.getNamespace = function(namespace) {
var org = userService.getOrganization(namespace);
if (org) {
return org;
}
if (namespace == userResponse.username) {
return userResponse;
}
return null;
};
userService.currentUser = function() { userService.currentUser = function() {
return userResponse; return userResponse;
}; };