Merge branch 'orgs' of ssh://bitbucket.org/yackob03/quay into orgs
Conflicts: static/partials/new-organization.html
This commit is contained in:
commit
9f1bf1499d
9 changed files with 123 additions and 50 deletions
|
@ -120,6 +120,10 @@ def convert_user_to_organization():
|
||||||
error_resp.status_code = 400
|
error_resp.status_code = 400
|
||||||
return error_resp
|
return error_resp
|
||||||
|
|
||||||
|
# Subscribe the organization to the new plan.
|
||||||
|
plan = convert_data['plan']
|
||||||
|
subscribe(user, plan, None, BUSINESS_PLANS)
|
||||||
|
|
||||||
# Convert the user to an organization.
|
# Convert the user to an organization.
|
||||||
model.convert_user_to_organization(user, model.get_user(admin_username))
|
model.convert_user_to_organization(user, model.get_user(admin_username))
|
||||||
|
|
||||||
|
|
|
@ -1714,23 +1714,10 @@ p.editable:hover i {
|
||||||
padding-left: 10px;
|
padding-left: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-org .plan-group table {
|
|
||||||
margin: 20px;
|
|
||||||
border: 1px solid #eee;
|
|
||||||
}
|
|
||||||
|
|
||||||
.create-org .plan-group strong {
|
.create-org .plan-group strong {
|
||||||
margin-bottom: 10px;
|
margin-bottom: 10px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.create-org .plan-group td {
|
|
||||||
vertical-align: middle;
|
|
||||||
}
|
|
||||||
|
|
||||||
.create-org .plan-group .plan-price {
|
|
||||||
font-size: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.create-org .step-container .description {
|
.create-org .step-container .description {
|
||||||
margin-top: 10px;
|
margin-top: 10px;
|
||||||
display: block;
|
display: block;
|
||||||
|
@ -1748,26 +1735,40 @@ p.editable:hover i {
|
||||||
margin-bottom: 20px;
|
margin-bottom: 20px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.plan-manager-element .plans-table thead td {
|
.plan-manager-element .plans-list-table thead td {
|
||||||
color: #aaa;
|
color: #aaa;
|
||||||
font-weight: bold;
|
font-weight: bold;
|
||||||
}
|
}
|
||||||
|
|
||||||
.plan-manager-element .plans-table td {
|
.plan-manager-element .plans-list-table td {
|
||||||
padding: 10px;
|
padding: 10px;
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
vertical-align: middle;
|
vertical-align: middle;
|
||||||
}
|
}
|
||||||
|
|
||||||
.plan-manager-element .plans-table td.controls {
|
.plan-manager-element .plans-list-table td.controls {
|
||||||
text-align: right;
|
text-align: right;
|
||||||
}
|
}
|
||||||
|
|
||||||
.plan-manager-element .plans-table .plan-price {
|
.plan-manager-element .plans-list-table .plan-price {
|
||||||
font-size: 16px;
|
font-size: 16px;
|
||||||
margin-bottom: 0px;
|
margin-bottom: 0px;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.plans-table-element table {
|
||||||
|
margin: 20px;
|
||||||
|
border: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.plans-table-element td {
|
||||||
|
vertical-align: middle !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.plans-table-element .plan-price {
|
||||||
|
font-size: 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* Overrides for typeahead to work with bootstrap 3. */
|
/* Overrides for typeahead to work with bootstrap 3. */
|
||||||
|
|
||||||
.twitter-typeahead .tt-query,
|
.twitter-typeahead .tt-query,
|
||||||
|
|
|
@ -24,7 +24,7 @@
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<!-- Plans Table -->
|
<!-- Plans Table -->
|
||||||
<table class="table table-hover plans-table" ng-show="!planLoading">
|
<table class="table table-hover plans-list-table" ng-show="!planLoading">
|
||||||
<thead>
|
<thead>
|
||||||
<td>Plan</td>
|
<td>Plan</td>
|
||||||
<td>Private Repositories</td>
|
<td>Private Repositories</td>
|
||||||
|
|
23
static/directives/plans-table.html
Normal file
23
static/directives/plans-table.html
Normal file
|
@ -0,0 +1,23 @@
|
||||||
|
<div class="plans-table-element">
|
||||||
|
<table class="table table-hover plans-table-table" ng-show="plans">
|
||||||
|
<thead>
|
||||||
|
<th>Plan</th>
|
||||||
|
<th>Private Repositories</th>
|
||||||
|
<th style="min-width: 85px">Price</th>
|
||||||
|
<th></th>
|
||||||
|
</thead>
|
||||||
|
|
||||||
|
<tr ng-repeat="plan in plans" ng-class="currentPlan == plan ? 'active' : ''">
|
||||||
|
<td>{{ plan.title }}</td>
|
||||||
|
<td>{{ plan.privateRepos }}</td>
|
||||||
|
<td><div class="plan-price">${{ plan.price / 100 }}</div></td>
|
||||||
|
<td class="controls">
|
||||||
|
<a class="btn" href="javascript:void(0)"
|
||||||
|
ng-class="currentPlan == plan ? 'btn-primary' : 'btn-default'"
|
||||||
|
ng-click="setPlan(plan)">
|
||||||
|
{{ currentPlan == plan ? 'Selected' : 'Choose' }}
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
</table>
|
||||||
|
</div>
|
|
@ -55,6 +55,7 @@ quayApp = angular.module('quay', ['restangular', 'angularMoment', 'angulartics',
|
||||||
username: null,
|
username: null,
|
||||||
email: null,
|
email: null,
|
||||||
askForPassword: false,
|
askForPassword: false,
|
||||||
|
organizations: []
|
||||||
}
|
}
|
||||||
|
|
||||||
var userService = {}
|
var userService = {}
|
||||||
|
@ -140,6 +141,23 @@ quayApp = angular.module('quay', ['restangular', 'angularMoment', 'angulartics',
|
||||||
}, function() { callback([]); });
|
}, function() { callback([]); });
|
||||||
};
|
};
|
||||||
|
|
||||||
|
planService.getMatchingBusinessPlan = function(callback) {
|
||||||
|
planService.getPlans(function() {
|
||||||
|
planService.getSubscription(null, function(sub) {
|
||||||
|
var plan = planDict[sub.plan];
|
||||||
|
if (!plan) {
|
||||||
|
planService.getMinimumPlan(0, true, callback);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var count = Math.max(sub.usedPrivateRepos, plan.privateRepos);
|
||||||
|
planService.getMinimumPlan(count, true, callback);
|
||||||
|
}, function() {
|
||||||
|
planService.getMinimumPlan(0, true, callback);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
planService.getPlans = function(callback) {
|
planService.getPlans = function(callback) {
|
||||||
planService.verifyLoaded(callback);
|
planService.verifyLoaded(callback);
|
||||||
};
|
};
|
||||||
|
@ -387,6 +405,27 @@ quayApp.directive('signinForm', function () {
|
||||||
});
|
});
|
||||||
|
|
||||||
|
|
||||||
|
quayApp.directive('plansTable', function () {
|
||||||
|
var directiveDefinitionObject = {
|
||||||
|
priority: 0,
|
||||||
|
templateUrl: '/static/directives/plans-table.html',
|
||||||
|
replace: false,
|
||||||
|
transclude: true,
|
||||||
|
restrict: 'C',
|
||||||
|
scope: {
|
||||||
|
'plans': '=plans',
|
||||||
|
'currentPlan': '=currentPlan'
|
||||||
|
},
|
||||||
|
controller: function($scope, $element) {
|
||||||
|
$scope.setPlan = function(plan) {
|
||||||
|
$scope.currentPlan = plan;
|
||||||
|
};
|
||||||
|
}
|
||||||
|
};
|
||||||
|
return directiveDefinitionObject;
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
quayApp.directive('organizationHeader', function () {
|
quayApp.directive('organizationHeader', function () {
|
||||||
var directiveDefinitionObject = {
|
var directiveDefinitionObject = {
|
||||||
priority: 0,
|
priority: 0,
|
||||||
|
|
|
@ -15,8 +15,11 @@ $.fn.clipboardCopy = function() {
|
||||||
};
|
};
|
||||||
|
|
||||||
function HeaderCtrl($scope, $location, UserService, Restangular) {
|
function HeaderCtrl($scope, $location, UserService, Restangular) {
|
||||||
|
var searchToken = 0;
|
||||||
|
|
||||||
$scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) {
|
$scope.$watch( function () { return UserService.currentUser(); }, function (currentUser) {
|
||||||
$scope.user = currentUser;
|
$scope.user = currentUser;
|
||||||
|
++searchToken;
|
||||||
}, true);
|
}, true);
|
||||||
|
|
||||||
$scope.signout = function() {
|
$scope.signout = function() {
|
||||||
|
@ -40,6 +43,11 @@ function HeaderCtrl($scope, $location, UserService, Restangular) {
|
||||||
name: 'repositories',
|
name: 'repositories',
|
||||||
remote: {
|
remote: {
|
||||||
url: '/api/find/repository?query=%QUERY',
|
url: '/api/find/repository?query=%QUERY',
|
||||||
|
replace: function (url, uriEncodedQuery) {
|
||||||
|
url = url.replace('%QUERY', uriEncodedQuery);
|
||||||
|
url += '&cb=' + searchToken;
|
||||||
|
return url;
|
||||||
|
},
|
||||||
filter: function(data) {
|
filter: function(data) {
|
||||||
var datums = [];
|
var datums = [];
|
||||||
for (var i = 0; i < data.repositories.length; ++i) {
|
for (var i = 0; i < data.repositories.length; ++i) {
|
||||||
|
@ -215,12 +223,14 @@ function LandingCtrl($scope, $timeout, $location, Restangular, UserService, KeyS
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if ($scope.user.organizations) {
|
||||||
for (var i = 0; i < $scope.user.organizations.length; ++i) {
|
for (var i = 0; i < $scope.user.organizations.length; ++i) {
|
||||||
var org = $scope.user.organizations[i];
|
var org = $scope.user.organizations[i];
|
||||||
if (org.name == namespace) {
|
if (org.name == namespace) {
|
||||||
return org.can_create_repo;
|
return org.can_create_repo;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
return false;
|
return false;
|
||||||
};
|
};
|
||||||
|
@ -711,11 +721,19 @@ function UserAdminCtrl($scope, $timeout, $location, Restangular, PlanService, Us
|
||||||
$scope.updatingUser = false;
|
$scope.updatingUser = false;
|
||||||
$scope.changePasswordSuccess = false;
|
$scope.changePasswordSuccess = false;
|
||||||
$scope.convertStep = 0;
|
$scope.convertStep = 0;
|
||||||
|
$scope.org = {};
|
||||||
|
|
||||||
$('.form-change-pw').popover();
|
$('.form-change-pw').popover();
|
||||||
|
|
||||||
$scope.showConvertForm = function() {
|
$scope.showConvertForm = function() {
|
||||||
|
PlanService.getMatchingBusinessPlan(function(plan) {
|
||||||
|
$scope.org.plan = plan;
|
||||||
|
});
|
||||||
|
|
||||||
|
PlanService.getPlans(function(plans) {
|
||||||
|
$scope.orgPlans = plans.business;
|
||||||
|
});
|
||||||
|
|
||||||
$scope.convertStep = 1;
|
$scope.convertStep = 1;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -728,7 +746,8 @@ function UserAdminCtrl($scope, $timeout, $location, Restangular, PlanService, Us
|
||||||
|
|
||||||
var data = {
|
var data = {
|
||||||
'adminUser': $scope.org.adminUser,
|
'adminUser': $scope.org.adminUser,
|
||||||
'adminPassword': $scope.org.adminPassword
|
'adminPassword': $scope.org.adminPassword,
|
||||||
|
'plan': $scope.org.plan.stripeId
|
||||||
};
|
};
|
||||||
|
|
||||||
var convertAccount = Restangular.one('user/convert');
|
var convertAccount = Restangular.one('user/convert');
|
||||||
|
@ -1016,7 +1035,7 @@ function NewRepoCtrl($scope, $location, $http, $timeout, UserService, Restangula
|
||||||
};
|
};
|
||||||
|
|
||||||
$scope.upgradePlan = function() {
|
$scope.upgradePlan = function() {
|
||||||
PlanService.showSubscribeDialog($scope, $scope.planRequired.stripeId, null, function() {
|
PlanService.changePlan($scope, null, $scope.planRequired.stripeId, null, function() {
|
||||||
// Subscribing.
|
// Subscribing.
|
||||||
$scope.planChanging = true;
|
$scope.planChanging = true;
|
||||||
}, function(plan) {
|
}, function(plan) {
|
||||||
|
|
|
@ -1219,8 +1219,8 @@ RepositoryUsageChart.prototype.drawInternal_ = function() {
|
||||||
* Draws the chart in the given container.
|
* Draws the chart in the given container.
|
||||||
*/
|
*/
|
||||||
RepositoryUsageChart.prototype.draw = function(container) {
|
RepositoryUsageChart.prototype.draw = function(container) {
|
||||||
var cw = document.getElementById(container).clientWidth;
|
var cw = 200;
|
||||||
var ch = document.getElementById(container).clientHeight;
|
var ch = 200;
|
||||||
var radius = Math.min(cw, ch) / 2;
|
var radius = Math.min(cw, ch) / 2;
|
||||||
|
|
||||||
var pie = d3.layout.pie().sort(null);
|
var pie = d3.layout.pie().sort(null);
|
||||||
|
|
|
@ -68,26 +68,7 @@
|
||||||
<!-- Plans Table -->
|
<!-- Plans Table -->
|
||||||
<div class="form-group plan-group">
|
<div class="form-group plan-group">
|
||||||
<strong>Choose your organization's plan</strong>
|
<strong>Choose your organization's plan</strong>
|
||||||
<table class="table table-hover plans-table" ng-show="plans">
|
<div class="plans-table" plans="plans" current-plan="currentPlan"></div>
|
||||||
<thead>
|
|
||||||
<th>Plan</th>
|
|
||||||
<th>Private Repositories</th>
|
|
||||||
<th style="min-width: 70px">Price</th>
|
|
||||||
<th></th>
|
|
||||||
</thead>
|
|
||||||
|
|
||||||
<tr ng-repeat="plan in plans" ng-class="currentPlan == plan ? 'active' : ''">
|
|
||||||
<td>{{ plan.title }}</td>
|
|
||||||
<td>{{ plan.privateRepos }}</td>
|
|
||||||
<td><div class="plan-price">${{ plan.price / 100 }}</div></td>
|
|
||||||
<td class="controls">
|
|
||||||
<a class="btn" href="javascript:void(0)" ng-class="currentPlan == plan ? 'btn-primary' : 'btn-default'"
|
|
||||||
ng-click="setPlan(plan)">
|
|
||||||
{{ currentPlan == plan ? 'Selected' : 'Choose' }}
|
|
||||||
</a>
|
|
||||||
</td>
|
|
||||||
</tr>
|
|
||||||
</table>
|
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
<div class="button-bar">
|
<div class="button-bar">
|
||||||
|
|
|
@ -101,8 +101,14 @@
|
||||||
<span class="description">The username and password for an <b>existing account</b> that will become administrator of the organization</span>
|
<span class="description">The username and password for an <b>existing account</b> that will become administrator of the organization</span>
|
||||||
</div>
|
</div>
|
||||||
|
|
||||||
|
<!-- Plans Table -->
|
||||||
|
<div class="form-group plan-group">
|
||||||
|
<label>Organization Plan</label>
|
||||||
|
<div class="plans-table" plans="orgPlans" current-plan="org.plan"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
<div class="button-bar">
|
<div class="button-bar">
|
||||||
<button class="btn btn-large btn-danger" type="submit" ng-disabled="convertForm.$invalid">
|
<button class="btn btn-large btn-danger" type="submit" ng-disabled="convertForm.$invalid || !org.plan">
|
||||||
Convert To Organization
|
Convert To Organization
|
||||||
</button>
|
</button>
|
||||||
</div>
|
</div>
|
||||||
|
|
Reference in a new issue