Shows the images that will be deleted when removing a tag
This commit is contained in:
parent
de8f1ef776
commit
d2b9e0d65a
3 changed files with 145 additions and 2 deletions
|
@ -1210,6 +1210,61 @@ p.editable:hover i {
|
||||||
border: 0px;
|
border: 0px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings {
|
||||||
|
margin: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing {
|
||||||
|
margin: 4px;
|
||||||
|
padding: 2px;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing .image-listing-id {
|
||||||
|
display: inline-block;
|
||||||
|
margin-left: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing .image-listing-line {
|
||||||
|
border-left: 2px solid steelblue;
|
||||||
|
display: inline-block;
|
||||||
|
position: absolute;
|
||||||
|
top: -2px;
|
||||||
|
bottom: 8px;
|
||||||
|
left: 6px;
|
||||||
|
width: 1px;
|
||||||
|
z-index: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing.tag-image .image-listing-line {
|
||||||
|
top: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing.child .image-listing-line {
|
||||||
|
bottom: -2px;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing .image-listing-circle {
|
||||||
|
position: absolute;
|
||||||
|
top: 8px;
|
||||||
|
|
||||||
|
border-radius: 50%;
|
||||||
|
border: 2px solid steelblue;
|
||||||
|
width: 10px;
|
||||||
|
height: 10px;
|
||||||
|
display: inline-block;
|
||||||
|
background: white;
|
||||||
|
z-index: 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .image-listings .image-listing.tag-image .image-listing-circle {
|
||||||
|
background: steelblue;
|
||||||
|
}
|
||||||
|
|
||||||
|
#confirmdeleteTagModal .more-changes {
|
||||||
|
margin-left: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
.repo .header {
|
.repo .header {
|
||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
position: relative;
|
position: relative;
|
||||||
|
|
|
@ -194,6 +194,60 @@ function RepoCtrl($scope, Restangular, ApiService, $routeParams, $rootScope, $lo
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
$scope.tagSpecificImages = function(tagName) {
|
||||||
|
if (!tagName) { return []; }
|
||||||
|
|
||||||
|
var tag = $scope.repo.tags[tagName];
|
||||||
|
if (!tag) { return []; }
|
||||||
|
|
||||||
|
if ($scope.specificImages && $scope.specificImages[tagName]) {
|
||||||
|
return $scope.specificImages[tagName];
|
||||||
|
}
|
||||||
|
|
||||||
|
var getIdsForTag = function(currentTag) {
|
||||||
|
var ancestors = currentTag.image.ancestors.split('/');
|
||||||
|
var dbid = currentTag.image.dbid;
|
||||||
|
var ids = {};
|
||||||
|
|
||||||
|
ids[dbid] = true;
|
||||||
|
for (var i = 0; i < ancestors.length; ++i) {
|
||||||
|
if (ancestors[i]) {
|
||||||
|
ids[ancestors[i]] = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ids;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// Remove any IDs that match other tags.
|
||||||
|
var toDelete = getIdsForTag(tag);
|
||||||
|
for (var currentTagName in $scope.repo.tags) {
|
||||||
|
var currentTag = $scope.repo.tags[currentTagName];
|
||||||
|
if (currentTag != tag) {
|
||||||
|
for (var dbid in getIdsForTag(currentTag)) {
|
||||||
|
delete toDelete[dbid];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Return the matching list of images.
|
||||||
|
var images = [];
|
||||||
|
for (var i = 0; i < $scope.images.length; ++i) {
|
||||||
|
var image = $scope.images[i];
|
||||||
|
if (toDelete[image.dbid]) {
|
||||||
|
images.push(image);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
images.sort(function(a, b) {
|
||||||
|
return b.dbid - a.dbid;
|
||||||
|
});
|
||||||
|
|
||||||
|
$scope.specificImages[tagName] = images;
|
||||||
|
return images;
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
$scope.askDeleteTag = function(tagName) {
|
$scope.askDeleteTag = function(tagName) {
|
||||||
if (!$scope.repo.can_admin) { return; }
|
if (!$scope.repo.can_admin) { return; }
|
||||||
|
|
||||||
|
@ -257,6 +311,22 @@ function RepoCtrl($scope, Restangular, ApiService, $routeParams, $rootScope, $lo
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
|
$scope.getFirstTextLine = getFirstTextLine;
|
||||||
|
|
||||||
|
$scope.getImageListingClasses = function(image, tagName) {
|
||||||
|
var classes = '';
|
||||||
|
if (image.ancestors.length > 1) {
|
||||||
|
classes += 'child ';
|
||||||
|
}
|
||||||
|
|
||||||
|
var currentTag = $scope.repo.tags[tagName];
|
||||||
|
if (image.dbid == currentTag.image.dbid) {
|
||||||
|
classes += 'tag-image ';
|
||||||
|
}
|
||||||
|
|
||||||
|
return classes;
|
||||||
|
};
|
||||||
|
|
||||||
$scope.getTagCount = function(repo) {
|
$scope.getTagCount = function(repo) {
|
||||||
if (!repo) { return 0; }
|
if (!repo) { return 0; }
|
||||||
var count = 0;
|
var count = 0;
|
||||||
|
@ -389,6 +459,9 @@ function RepoCtrl($scope, Restangular, ApiService, $routeParams, $rootScope, $lo
|
||||||
var listImages = function() {
|
var listImages = function() {
|
||||||
var params = {'repository': namespace + '/' + name};
|
var params = {'repository': namespace + '/' + name};
|
||||||
$scope.imageHistory = ApiService.listRepositoryImagesAsResource(params).get(function(resp) {
|
$scope.imageHistory = ApiService.listRepositoryImagesAsResource(params).get(function(resp) {
|
||||||
|
$scope.images = resp.images;
|
||||||
|
$scope.specificImages = [];
|
||||||
|
|
||||||
// Dispose of any existing tree.
|
// Dispose of any existing tree.
|
||||||
if ($scope.tree) {
|
if ($scope.tree) {
|
||||||
$scope.tree.dispose();
|
$scope.tree.dispose();
|
||||||
|
|
|
@ -213,8 +213,23 @@ sudo docker push quay.io/{{repo.namespace}}/{{repo.name}}</pre>
|
||||||
{{ tagToDelete }}
|
{{ tagToDelete }}
|
||||||
</span>?
|
</span>?
|
||||||
|
|
||||||
<br><br>
|
<div ng-show="tagSpecificImages(tagToDelete).length" style="margin-top: 20px">
|
||||||
Doing so will delete any images not attached to another tag.
|
The following images will also be deleted:
|
||||||
|
<div class="image-listings">
|
||||||
|
<div class="image-listing" ng-repeat="image in tagSpecificImages(tagToDelete) | limitTo:5"
|
||||||
|
ng-class="getImageListingClasses(image, tagToDelete)">
|
||||||
|
<!--<i class="fa fa-archive"></i>-->
|
||||||
|
<span class="image-listing-circle"></span>
|
||||||
|
<span class="image-listing-line"></span>
|
||||||
|
<span class="context-tooltip image-listing-id" bs-tooltip="getFirstTextLine(image.comment)">
|
||||||
|
{{ image.id.substr(0, 12) }}
|
||||||
|
</span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="more-changes" ng-show="tagSpecificImages(tagToDelete).length > 5">
|
||||||
|
And {{ tagSpecificImages(tagToDelete).length - 5 }} more...
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<div class="modal-footer">
|
<div class="modal-footer">
|
||||||
<button type="button" class="btn btn-primary" ng-click="deleteTag(tagToDelete)">Delete Tag</button>
|
<button type="button" class="btn btn-primary" ng-click="deleteTag(tagToDelete)">Delete Tag</button>
|
||||||
|
|
Reference in a new issue