Merge pull request #1409 from coreos-inc/user-setup

Implement new design for user and org settings
This commit is contained in:
josephschorr 2016-04-28 14:32:30 -04:00
commit ac22658ee7
25 changed files with 784 additions and 614 deletions

View file

@ -866,6 +866,14 @@ a:focus {
margin-right: 6px; margin-right: 6px;
} }
.co-dialog .co-single-field-dialog {
padding: 10px;
}
.co-dialog .co-single-field-dialog input {
margin-top: 10px;
}
.co-step-bar .co-step-element { .co-step-bar .co-step-element {
cursor: default; cursor: default;
display: inline-block; display: inline-block;
@ -1370,3 +1378,46 @@ a:focus {
float: none; float: none;
} }
} }
.co-list-table tr td:first-child {
font-weight: bold;
padding-right: 10px;
vertical-align: top;
width: 120px;
padding-left: 0px;
}
.co-list-table tr td {
padding: 10px;
font-size: 15px;
}
.co-list-table .help-text {
margin-top: 6px;
font-size: 14px;
color: #aaa;
}
.co-modify-link:after {
font-family: FontAwesome;
content: "\f054";
color: #ccc;
vertical-align: middle;
display: inline-block;
margin-left: 10px;
font-size: 10px;
line-height: 16px;
}
.co-option-table tr td:first-child {
padding-left: 16px;
padding-right: 16px;
padding-top: 0px;
vertical-align: top;
}
.co-option-table tr td:last-child {
padding-bottom: 10px;
}

View file

@ -0,0 +1,32 @@
.billing-management-panel-element .credit-card-image {
width: 33px;
margin-right: 10px;
}
.billing-management-panel-element .credit-card-number {
display: inline-block;
font-size: 16px;
margin-right: 10px;
}
.billing-management-panel-element .receipt-form {
padding-top: 20px;
padding-bottom: 12px;
}
.billing-management-panel-element .sub-usage {
margin-bottom: 10px;
font-size: 16px;
}
.billing-management-panel-element .sub-usage .fa {
margin-right: 4px;
}
.billing-management-panel-element .sub-usage .red {
color: #D64456;
}
.billing-management-panel-element .sub-usage .yellow {
color: #FCA657;
}

View file

@ -1,15 +0,0 @@
.billing-options .settings-option {
padding: 4px;
font-size: 18px;
margin-bottom: 10px;
}
.billing-options .settings-option label {
margin-left: 6px;
}
.billing-options .settings-option .settings-description {
font-size: 16px;
color: #888;
padding-left: 26px;
}

View file

@ -2,6 +2,10 @@
padding: 10px; padding: 10px;
} }
.convert-user-to-org .convert-form {
padding: 20px;
}
.convert-user-to-org .convert-form h3 { .convert-user-to-org .convert-form h3 {
margin-bottom: 20px; margin-bottom: 20px;
} }
@ -66,4 +70,8 @@
.convert-user-to-org .form-group-content .co-table { .convert-user-to-org .form-group-content .co-table {
margin: 0px; margin: 0px;
}
.convert-user-to-org .co-option-table {
margin-top: 12px;
} }

View file

@ -0,0 +1,3 @@
.billing-page .co-main-content-panel {
padding: 40px;
}

View file

@ -6,4 +6,9 @@
.org-view h3 { .org-view h3 {
margin-bottom: 20px; margin-bottom: 20px;
margin-top: 0px; margin-top: 0px;
} }
.org-view .settings-section {
margin-bottom: 50px;
}

View file

@ -8,27 +8,15 @@
} }
.user-view h3 { .user-view h3 {
margin-bottom: 20px;
margin-top: 0px; margin-top: 0px;
} }
.user-view .section-description-header { .user-view .section-description-header {
padding-left: 40px;
position: relative; position: relative;
margin-bottom: 20px; margin-bottom: 20px;
min-height: 50px; min-height: 50px;
} }
.user-view .section-description-header:before {
font-family: FontAwesome;
content: "\f05a";
position: absolute;
top: -4px;
left: 6px;
font-size: 27px;
color: #888;
}
.user-view .user-settings-form .row { .user-view .user-settings-form .row {
padding: 10px; padding: 10px;
margin: 0px; margin: 0px;
@ -53,3 +41,6 @@
box-shadow: none; box-shadow: none;
} }
.user-view .settings-section {
margin-bottom: 50px;
}

View file

@ -449,48 +449,6 @@ i.toggle-icon:hover {
visibility: hidden; visibility: hidden;
} }
.billing-options-element .current-card {
font-size: 16px;
margin-bottom: 20px;
}
.billing-options-element .current-card .no-card-outline {
display: inline-block;
width: 73px;
height: 44px;
vertical-align: middle;
margin-right: 10px;
border: 1px dashed #aaa;
border-radius: 4px;
}
.billing-options-element .current-card .last4 {
color: #aaa;
}
.billing-options-element .current-card .last4 b {
color: black;
}
.billing-options-element .current-card .expires:before {
content: "Expires:";
color: #aaa;
font-size: 12px;
}
.billing-options-element .current-card .expires {
margin-left: 20px;
font-size: 12px;
}
.billing-options-element .current-card img {
margin-right: 10px;
vertical-align: middle;
}
.organization-header-element { .organization-header-element {
padding: 20px; padding: 20px;
margin-bottom: 20px; margin-bottom: 20px;
@ -4077,24 +4035,12 @@ i.rocket-icon {
} }
.section-description-header { .section-description-header {
padding-left: 40px;
position: relative; position: relative;
margin-bottom: 20px; margin-bottom: 10px;
min-height: 50px; min-height: 50px;
border-bottom: 1px solid #eee;
padding-bottom: 10px; padding-bottom: 10px;
} }
.section-description-header:before {
font-family: FontAwesome;
content: "\f05a";
position: absolute;
top: -4px;
left: 6px;
font-size: 27px;
color: #888;
}
.nvtooltip h3 { .nvtooltip h3 {
margin: 0; margin: 0;
padding: 4px 14px; padding: 4px 14px;

View file

@ -0,0 +1,80 @@
<div class="billing-management-panel-element">
<div class="cor-loader-inline" ng-show="updating"></div>
<div ng-show="!updating">
<table class="co-list-table">
<tr>
<td>Current Plan:</td>
<td>
<div class="sub-usage" ng-if="subscription.usedPrivateRepos > currentPlan.privateRepos">
<i class="fa fa-exclamation-triangle red"></i> <strong>{{ subscription.usedPrivateRepos }}</strong> private repositories exceeds the amount allowed by your plan. Upgrade your plan to avoid service disruptions.
</div>
<div class="sub-usage" ng-if="subscription.usedPrivateRepos == currentPlan.privateRepos">
<i class="fa fa-exclamation-triangle yellow"></i> <strong>{{ subscription.usedPrivateRepos }}</strong> private repositories is the maximum allowed by your plan. Upgrade your plan to create more private repositories.
</div>
<a class="co-modify-link" ng-href="{{ getEntityPrefix() }}/billing">{{ currentPlan.privateRepos }} private repositories</a>
<div class="help-text">Up to {{ currentPlan.privateRepos }} private repositories, unlimited public repositories</div>
</td>
</tr>
<tr ng-show="currentCard">
<td>Credit Card:</td>
<td>
<img class="credit-card-image" ng-src="/static/img/creditcards/{{ getCreditImage(currentCard) }}">
<span class="credit-card-number">
&#8226;&#8226;&#8226;&#8226;&nbsp;
&#8226;&#8226;&#8226;&#8226;&nbsp;
&#8226;&#8226;&#8226;&#8226;&nbsp;
{{ currentCard.last4 }}
</span>
<a class="co-modify-link" ng-click="changeCreditCard()">Change card</a>
<div class="help-text">Expires {{ currentCard.exp_month }}/{{ currentCard.exp_year }}</div>
</td>
</tr>
<tr>
<td>Invoices:</td>
<td>
<a ng-href="{{ getEntityPrefix() }}/billing/invoices">View Invoices</a>
</td>
</tr>
<tr>
<td>Receipts:</td>
<td>
<a class="co-modify-link" ng-click="showChangeReceipts()" ng-show="!invoice_email">Do not email after successful charges</a>
<a class="co-modify-link" ng-click="showChangeReceipts()" ng-show="invoice_email">Email receipts to {{ invoice_email_address }}</a>
</td>
</tr>
</table>
</div>
<!-- Change receipts dialog -->
<div class="cor-confirm-dialog"
dialog-context="changeReceiptsInfo"
dialog-action="changeReceipts(info, callback)"
dialog-title="Receipts Settings"
dialog-action-title="Update Setting"
dialog-form="context.receiptform">
<form class="receipt-form" name="context.receiptform">
<table class="co-option-table">
<tr>
<td><input type="radio" id="emailReceiptNo" ng-model="changeReceiptsInfo.sendOption" ng-value="false"></td>
<td>
<label for="emailReceiptNo">Do not send email receipts</label>
<div class="help-text">Log into your account to view invoices</div>
</td>
</tr>
<tr>
<td><input type="radio" id="emailReceiptYes" ng-model="changeReceiptsInfo.sendOption" ng-value="true"></td>
<td>
<label for="emailReceiptYes">Send receipts via email</label>
<div class="help-text">
After every successful charge send an email to:
<div style="margin-top: 6px;"><input type="email" class="form-control" ng-model="changeReceiptsInfo.address" required></div>
</div>
</td>
</tr>
</table>
</form>
</div>
</div>

View file

@ -1,65 +0,0 @@
<div class="billing-options-element">
<!-- Credit Card -->
<div style="margin-bottom: 20px">
<div class="panel-title">
Credit Card
</div>
<div class="panel-body">
<div class="cor-loader-inline" ng-show="!currentCard || changingCard"></div>
<div class="current-card" ng-show="currentCard && !changingCard">
<div class="alert alert-warning" ng-if="currentCard.last4 && isExpiringSoon(currentCard)">
Your current credit card is expiring soon!
</div>
<img ng-src="{{ '/static/img/creditcards/' + getCreditImage(currentCard) }}" ng-show="currentCard.last4">
<span class="no-card-outline" ng-show="!currentCard.last4"></span>
<span class="last4" ng-show="currentCard.last4">****-****-****-<b>{{ currentCard.last4 }}</b></span>
<span class="expires" ng-show="currentCard.last4">
{{ currentCard.exp_month }} / {{ currentCard.exp_year }}
</span>
<span class="not-found" ng-show="!currentCard.last4">No credit card found</span>
</div>
<button class="btn btn-primary" ng-show="currentCard && !changingCard" ng-click="changeCard()">
Change Credit Card
</button>
</div>
</div>
<!-- Options -->
<div style="margin-bottom: 20px">
<div class="panel-title">
Billing Receipts
<div class="cor-loader-inline" ng-show="working"></div>
</div>
<div class="panel-body">
<div class="settings-option">
<input id="invoiceEmail" type="checkbox" ng-model="invoice_email">
<label for="invoiceEmail">Send Receipt Emails</label>
<div class="settings-description">
If checked, a receipt email will be sent to <a ng-click="changeInvoiceEmailAddress()">{{ obj.invoice_email_address || obj.email }}</a> on every successful charge
</div>
</div>
</div>
</div>
<!-- Modal message dialog -->
<div class="modal fade" id="cannotchangecardModal">
<div class="modal-dialog">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
<h4 class="modal-title">Cannot change credit card</h4>
</div>
<div class="modal-body">
Your credit card could not be changed
</div>
<div class="modal-footer">
<button type="button" class="btn btn-default" data-dismiss="modal">Close</button>
</div>
</div><!-- /.modal-content -->
</div><!-- /.modal-dialog -->
</div><!-- /.modal -->
</div>

View file

@ -1,112 +1,108 @@
<div class="convert-user-to-org-element"> <div class="convert-user-to-org-element">
<!-- Step 0 --> <div class="co-dialog modal fade" id="convertAccountModal">
<div ng-show="convertStep == 0"> <div class="modal-dialog">
<div ng-show="user.organizations.length > 0"> <div class="modal-content">
Cannot convert this account into an organization, as it is a member of {{user.organizations.length}} other <div class="modal-header">
organization{{user.organizations.length > 1 ? 's' : ''}}. <button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
<br><br> <h4 class="modal-title">Change Account Type</h4>
Please leave the following organizations first: </div>
<ul class="org-list"> <div class="modal-body">
<li ng-repeat="org in user.organizations"> <!-- Step 0 -->
<span class="avatar" size="avatarSize || 16" data="org.avatar"></span> <div ng-show="convertStep == 0">
<a href="/organization/{{ org.name }}">{{ org.name }}</a> <div ng-show="user.organizations.length > 0">
</li> This account cannot be converted into an organization, as it is a member of {{user.organizations.length}} other
</ul> organization{{user.organizations.length > 1 ? 's' : ''}}.
</div> <br><br>
Please leave the following organizations first:
<ul class="org-list">
<li ng-repeat="org in user.organizations">
<span class="avatar" size="avatarSize || 16" data="org.avatar"></span>
<a href="/organization/{{ org.name }}">{{ org.name }}</a>
</li>
</ul>
</div>
<div ng-show="user.organizations.length == 0"> <div ng-show="user.organizations.length == 0">
<button class="btn btn-primary" ng-click="showConvertForm()">Start conversion process <i class="fa fa-arrow-circle-right" aria-hidden="true"></i></button> <table class="co-option-table">
</div> <tr>
</div> <td><input type="radio" id="accountTypeI" ng-model="accountType" value="user"></td>
<td>
<!-- Step 1 --> <label for="accountTypeI">Individual account (current)</label>
<div class="convert-form" ng-show="convertStep == 1"> <div class="help-text">Single account with multiple repositories</div>
Fill out the form below to convert your current user account into an organization. Your existing repositories will be maintained under the </td>
namespace. All <strong>direct</strong> permissions delegated to {{ user.username }} will be deleted. </tr>
<tr>
<form method="post" name="convertForm" id="convertForm" ng-submit="convertToOrg()"> <td><input type="radio" id="accountTypeO" ng-model="accountType" value="organization"></td>
<div class="form-group"> <td>
<label for="orgName">Organization Name</label> <label for="accountTypeO">Organization</label>
<div class="form-group-content"> <div class="help-text">Multiple users and teams that share access and billing under a single namespace</div>
<div class="existing-data"> </td>
<span class="avatar" size="24" data="user.avatar"></span> </tr>
<span class="username">{{ user.username }}</span> </table>
</div>
</div> </div>
<span class="description">This will continue to be the namespace for your repositories</span>
</div>
</div>
<div class="form-group"> <!-- Step 1 -->
<label for="orgName">Admin User</label> <div class="convert-form" ng-show="convertStep == 1">
<div class="form-group-content"> Fill out the form below to convert your current user account into an organization. Your existing repositories will be maintained under the
<input id="adminUsername" name="adminUsername" type="text" class="form-control" placeholder="Admin Username" namespace. All <strong>direct</strong> permissions delegated to {{ user.username }} will be deleted.
ng-model="org.adminUser" required autofocus>
<input id="adminPassword" name="adminPassword" type="password" class="form-control" placeholder="Admin Password"
ng-model="org.adminPassword" required>
<span class="description">
The username and password for the account that will become an administrator of the organization.
Note that this account <b>must be a separate registered account</b> from the account that you are
trying to convert, and <b>must already exist</b>.
</span>
</div>
</div>
<!-- Plans Table --> <form method="post" name="convertForm" id="convertForm" ng-submit="nextStep()">
<div class="form-group plan-group" quay-require="['BILLING']"> <div class="form-group">
<label>Organization Plan</label> <label for="orgName">Organization Name</label>
<div class="form-group-content"> <div class="form-group-content">
<div class="plans-table" plans="orgPlans" current-plan="org.plan"></div> <div class="existing-data">
<span class="description">The billing plan for the new organization. If private repositories are unneeded, select "Open Source".</span> <span class="avatar" size="24" data="user.avatar"></span>
</div> <span class="username">{{ user.username }}</span>
</div> </div>
<span class="description">This will continue to be the namespace for your repositories</span>
</div>
</div>
<div class="button-bar"> <div class="form-group">
<button class="btn btn-large btn-danger" type="submit" <label for="orgName">Admin User</label>
ng-disabled="convertForm.$invalid || (Features.BILLING && !org.plan)" <div class="form-group-content">
analytics-on analytics-event="convert_to_organization"> <input id="adminUsername" name="adminUsername" type="text" class="form-control" placeholder="Admin Username"
Convert To Organization ng-model="org.adminUser" required autofocus>
</button> <input id="adminPassword" name="adminPassword" type="password" class="form-control" placeholder="Admin Password"
</div> ng-model="org.adminPassword" required>
</form> <span class="description">
</div> The username and password for the account that will become an administrator of the organization.
Note that this account <b>must be a separate registered account</b> from the account that you are
trying to convert, and <b>must already exist</b>.
</span>
</div>
</div>
</form>
</div>
<!-- Step 2 -->
<div class="convert-form" ng-show="convertStep == 2">
Please select the billing plan to use for the new organization. Select "Open Source" to create an organization without
private repositories.
<div class="plans-table" plans="orgPlans" current-plan="org.plan"></div>
</div>
<!-- Step 3 (conversion) -->
<div class="convert-form" ng-show="convertStep == 3">
<div class="cor-loader"></div>
</div>
<!-- Modal message dialog -->
<div class="modal fade" id="cannotconvertModal">
<div class="modal-dialog co-dialog">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
<h4 class="modal-title">Cannot convert account</h4>
</div> </div>
<div class="modal-body"> <div class="modal-footer" ng-show="convertStep < 3">
Your account could not be converted. Please try again in a moment. <button class="btn btn-default" ng-show="convertStep == 0 && accountType == 'user'" data-dismiss="modal">Close</button>
</div> <button class="btn btn-primary" ng-show="convertStep == 0 && accountType == 'organization'" ng-click="showConvertForm()">Convert Account</button>
<div class="modal-footer"> <button class="btn btn-primary" ng-show="convertStep == 1" ng-disabled="convertForm.$invalid" ng-click="nextStep()">
<button type="button" class="btn btn-default" data-dismiss="modal">Close</button> <span ng-if="Features.BILLING">Choose billing</span>
<span ng-if="!Features.BILLING">Convert Account</span>
</button>
<button class="btn btn-primary" ng-show="convertStep == 2" ng-disabled="!org.plan" ng-click="performConversion()">
Convert Account
</button>
</div> </div>
</div><!-- /.modal-content --> </div><!-- /.modal-content -->
</div><!-- /.modal-dialog --> </div><!-- /.modal-dialog -->
</div><!-- /.modal --> </div><!-- /.modal -->
<!-- Modal message dialog -->
<div class="modal co-modal fade" id="reallyconvertModal">
<div class="modal-dialog co-dialog">
<div class="modal-content">
<div class="modal-header">
<button type="button" class="close" data-dismiss="modal" aria-hidden="true">&times;</button>
<h4 class="modal-title">Convert to organization?</h4>
</div>
<div class="modal-body">
<div class="co-alert co-alert-danger">You will not be able to login to this account once converted!</div>
<div>Are you <b>absolutely sure</b> you would like to convert this account to an organization? Once done, there is no going back.</div>
</div>
<div class="modal-footer">
<button type="button" class="btn btn-danger" data-dismiss="modal" ng-click="reallyConvert()">Absolutely: Convert Now</button>
<button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button>
</div>
</div><!-- /.modal-content -->
</div><!-- /.modal-dialog -->
</div><!-- /.modal -->
</div> </div>

View file

@ -14,7 +14,7 @@
<span ng-transclude/> <span ng-transclude/>
</div> </div>
<div class="modal-footer" ng-show="!working"> <div class="modal-footer" ng-show="!working">
<button type="button" class="btn btn-primary" ng-click="performAction()"> <button type="button" class="btn btn-primary" ng-click="performAction()" ng-disabled="dialogForm && dialogForm.$invalid">
{{ dialogActionTitle }} {{ dialogActionTitle }}
</button> </button>
<button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button> <button type="button" class="btn btn-default" data-dismiss="modal">Cancel</button>

View file

@ -121,9 +121,21 @@ quayApp.config(['$routeProvider', '$locationProvider', 'pages', function($routeP
// Organization View Application // Organization View Application
.route('/organization/:orgname/application/:clientid', 'manage-application') .route('/organization/:orgname/application/:clientid', 'manage-application')
// View Organization Billing
.route('/organization/:orgname/billing', 'billing')
// View Organization Billing Invoices
.route('/organization/:orgname/billing/invoices', 'invoices')
// View User // View User
.route('/user/:username', 'user-view') .route('/user/:username', 'user-view')
// View User Billing
.route('/user/:username/billing', 'billing')
// View User Billing Invoices
.route('/user/:username/billing/invoices', 'invoices')
// Sign In // Sign In
.route('/signin/', 'signin') .route('/signin/', 'signin')

View file

@ -172,10 +172,12 @@ angular.module("core-ui", [])
scope: { scope: {
'dialogTitle': '@dialogTitle', 'dialogTitle': '@dialogTitle',
'dialogActionTitle': '@dialogActionTitle', 'dialogActionTitle': '@dialogActionTitle',
'dialogForm': '=dialogForm',
'dialogContext': '=dialogContext', 'dialogContext': '=dialogContext',
'dialogAction': '&dialogAction' 'dialogAction': '&dialogAction'
}, },
controller: function($rootScope, $scope, $element) { controller: function($rootScope, $scope, $element) {
$scope.working = false; $scope.working = false;
@ -195,6 +197,10 @@ angular.module("core-ui", [])
}; };
$scope.show = function() { $scope.show = function() {
if ($scope.dialogForm) {
$scope.dialogForm.$setPristine();
}
$scope.working = false; $scope.working = false;
$element.find('.modal').modal({}); $element.find('.modal').modal({});
}; };

View file

@ -0,0 +1,140 @@
/**
* An element which displays the billing options for a user or an organization.
*/
angular.module('quay').directive('billingManagementPanel', function () {
var directiveDefinitionObject = {
priority: 0,
templateUrl: '/static/directives/billing-management-panel.html',
replace: false,
transclude: false,
restrict: 'C',
scope: {
'user': '=user',
'organization': '=organization',
'isEnabled': '=isEnabled'
},
controller: function($scope, $element, PlanService, ApiService) {
$scope.currentCard = null;
$scope.subscription = null;
$scope.updating = true;
$scope.changeReceiptsInfo = null;
$scope.context = {};
var setSubscription = function(sub) {
$scope.subscription = sub;
// Load the plan info.
PlanService.getPlan(sub['plan'], function(plan) {
$scope.currentPlan = plan;
if (!sub.hasSubscription) {
$scope.updating = false;
return;
}
// Load credit card information.
PlanService.getCardInfo($scope.organization ? $scope.organization.name : null, function(card) {
$scope.currentCard = card;
$scope.updating = false;
});
});
};
var update = function() {
if (!$scope.isEnabled || !($scope.user || $scope.organization)) {
return;
}
$scope.entity = $scope.user ? $scope.user : $scope.organization;
$scope.invoice_email = $scope.entity.invoice_email;
$scope.invoice_email_address = $scope.entity.invoice_email_address || $scope.entity.email;
$scope.updating = true;
// Load plan information.
PlanService.getSubscription($scope.organization, setSubscription, function() {
setSubscription({ 'plan': PlanService.getFreePlan() });
});
};
// Listen to plan changes.
PlanService.registerListener(this, function(plan) {
if (plan && plan.price > 0) {
update();
}
});
$scope.$on('$destroy', function() {
PlanService.unregisterListener(this);
});
$scope.$watch('isEnabled', update);
$scope.$watch('organization', update);
$scope.$watch('user', update);
$scope.getEntityPrefix = function() {
if ($scope.organization) {
return '/organization/' + $scope.organization.name;
} else {
return '/user/' + $scope.user.username;
}
};
$scope.changeCreditCard = function() {
var callbacks = {
'opened': function() { },
'closed': function() { },
'started': function() { },
'success': function(resp) {
$scope.currentCard = resp.card;
update();
},
'failure': function(resp) {
if (!PlanService.isCardError(resp)) {
bootbox.alert('Could not change credit card. Please try again later.');
}
}
};
PlanService.changeCreditCard($scope, $scope.organization ? $scope.organization.name : null, callbacks);
};
$scope.getCreditImage = function(creditInfo) {
if (!creditInfo || !creditInfo.type) { return 'credit.png'; }
var kind = creditInfo.type.toLowerCase() || 'credit';
var supported = {
'american express': 'amex',
'credit': 'credit',
'diners club': 'diners',
'discover': 'discover',
'jcb': 'jcb',
'mastercard': 'mastercard',
'visa': 'visa'
};
kind = supported[kind] || 'credit';
return kind + '.png';
};
$scope.changeReceipts = function(info, callback) {
$scope.entity['invoice_email'] = info['sendOption'] || false;
$scope.entity['invoice_email_address'] = info['address'] || $scope.entity.email;
var errorHandler = ApiService.errorDisplay('Could not change billing options', callback);
ApiService.changeDetails($scope.organization, $scope.entity).then(function(resp) {
callback(true);
update();
}, errorHandler);
};
$scope.showChangeReceipts = function() {
$scope.changeReceiptsInfo = {
'sendOption': $scope.invoice_email,
'address': $scope.invoice_email_address
};
};
}
};
return directiveDefinitionObject;
});

View file

@ -1,130 +0,0 @@
/**
* An element which displays the billing options for a user or an organization.
*/
angular.module('quay').directive('billingOptions', function () {
var directiveDefinitionObject = {
priority: 0,
templateUrl: '/static/directives/billing-options.html',
replace: false,
transclude: false,
restrict: 'C',
scope: {
'user': '=user',
'organization': '=organization'
},
controller: function($scope, $element, PlanService, ApiService) {
$scope.invoice_email = false;
$scope.currentCard = null;
// Listen to plan changes.
PlanService.registerListener(this, function(plan) {
if (plan && plan.price > 0) {
update();
}
});
$scope.$on('$destroy', function() {
PlanService.unregisterListener(this);
});
$scope.isExpiringSoon = function(cardInfo) {
var current = new Date();
var expires = new Date(cardInfo.exp_year, cardInfo.exp_month, 1);
var difference = expires - current;
return difference < (60 * 60 * 24 * 60 * 1000 /* 60 days */);
};
$scope.changeInvoiceEmailAddress = function() {
bootbox.prompt('Enter the email address for receiving receipts', function(email) {
// Copied from Angular.
var EMAIL_REGEXP = /^[a-z0-9!#$%&'*+/=?^_`{|}~.-]+@[a-z0-9-]+(\.[a-z0-9-]+)*$/i;
if (!email || !EMAIL_REGEXP.test(email)) {
return;
}
$scope.obj['invoice_email_address'] = email;
var errorHandler = ApiService.errorDisplay('Could not change user details');
ApiService.changeDetails($scope.organization, $scope.obj).then(function(resp) {
$scope.working = false;
}, errorHandler);
});
};
$scope.changeCard = function() {
var previousCard = $scope.currentCard;
$scope.changingCard = true;
var callbacks = {
'opened': function() { $scope.changingCard = true; },
'closed': function() { $scope.changingCard = false; },
'started': function() { $scope.currentCard = null; },
'success': function(resp) {
$scope.currentCard = resp.card;
$scope.changingCard = false;
},
'failure': function(resp) {
$scope.changingCard = false;
$scope.currentCard = previousCard;
if (!PlanService.isCardError(resp)) {
$('#cannotchangecardModal').modal({});
}
}
};
PlanService.changeCreditCard($scope, $scope.organization ? $scope.organization.name : null, callbacks);
};
$scope.getCreditImage = function(creditInfo) {
if (!creditInfo || !creditInfo.type) { return 'credit.png'; }
var kind = creditInfo.type.toLowerCase() || 'credit';
var supported = {
'american express': 'amex',
'credit': 'credit',
'diners club': 'diners',
'discover': 'discover',
'jcb': 'jcb',
'mastercard': 'mastercard',
'visa': 'visa'
};
kind = supported[kind] || 'credit';
return kind + '.png';
};
var update = function() {
if (!$scope.user && !$scope.organization) { return; }
$scope.obj = $scope.user ? $scope.user : $scope.organization;
$scope.invoice_email = $scope.obj.invoice_email;
// Load the credit card information.
PlanService.getCardInfo($scope.organization ? $scope.organization.name : null, function(card) {
$scope.currentCard = card;
});
};
var save = function() {
$scope.working = true;
var errorHandler = ApiService.errorDisplay('Could not change user details');
ApiService.changeDetails($scope.organization, $scope.obj).then(function(resp) {
$scope.working = false;
}, errorHandler);
};
var checkSave = function() {
if (!$scope.obj) { return; }
if ($scope.obj.invoice_email != $scope.invoice_email) {
$scope.obj.invoice_email = $scope.invoice_email;
save();
}
};
$scope.$watch('invoice_email', checkSave);
$scope.$watch('organization', update);
$scope.$watch('user', update);
}
};
return directiveDefinitionObject;
});

View file

@ -9,14 +9,29 @@ angular.module('quay').directive('convertUserToOrg', function () {
transclude: true, transclude: true,
restrict: 'C', restrict: 'C',
scope: { scope: {
'user': '=user' 'info': '=info'
}, },
controller: function($scope, $element, $location, Features, PlanService, Config, ApiService, CookieService, UserService) { controller: function($scope, $element, $location, Features, PlanService, Config, ApiService, CookieService, UserService) {
$scope.convertStep = 0; $scope.convertStep = 0;
$scope.org = {}; $scope.org = {};
$scope.loading = false; $scope.loading = false;
$scope.user = null;
$scope.Features = Features;
$scope.$watch('info', function(info) {
if (info && info.user) {
$scope.user = info.user;
$scope.accountType = 'user';
$scope.convertStep = 0;
$('#convertAccountModal').modal({});
}
});
$scope.showConvertForm = function() { $scope.showConvertForm = function() {
$scope.convertStep = 1;
};
$scope.nextStep = function() {
if (Features.BILLING) { if (Features.BILLING) {
PlanService.getMatchingBusinessPlan(function(plan) { PlanService.getMatchingBusinessPlan(function(plan) {
$scope.org.plan = plan; $scope.org.plan = plan;
@ -25,22 +40,19 @@ angular.module('quay').directive('convertUserToOrg', function () {
PlanService.getPlans(function(plans) { PlanService.getPlans(function(plans) {
$scope.orgPlans = plans; $scope.orgPlans = plans;
}); });
$scope.convertStep = 2;
} else {
$scope.performConversion();
} }
$scope.convertStep = 1;
}; };
$scope.convertToOrg = function() { $scope.performConversion = function() {
$('#reallyconvertModal').modal({});
};
$scope.reallyConvert = function() {
if (Config.AUTHENTICATION_TYPE != 'Database') { return; } if (Config.AUTHENTICATION_TYPE != 'Database') { return; }
$scope.convertStep = 3;
$scope.loading = true;
var errorHandler = ApiService.errorDisplay(function() { var errorHandler = ApiService.errorDisplay(function() {
$scope.loading = false; $('#convertAccountModal').modal('hide');
}); });
var data = { var data = {
@ -52,6 +64,7 @@ angular.module('quay').directive('convertUserToOrg', function () {
ApiService.convertUserToOrganization(data).then(function(resp) { ApiService.convertUserToOrganization(data).then(function(resp) {
CookieService.putPermanent('quay.namespace', $scope.user.username); CookieService.putPermanent('quay.namespace', $scope.user.username);
UserService.load(); UserService.load();
$('#convertAccountModal').modal('hide');
$location.path('/'); $location.path('/');
}, errorHandler); }, errorHandler);
}; };

View file

@ -0,0 +1,44 @@
(function() {
/**
* Billing plans page.
*/
angular.module('quayPages').config(['pages', function(pages) {
pages.create('billing', 'billing.html', BillingCtrl, {
'title': 'Billing',
'description': 'Billing',
'newLayout': true
});
}]);
/**
* Billing invoices page.
*/
angular.module('quayPages').config(['pages', function(pages) {
pages.create('invoices', 'invoices.html', BillingCtrl, {
'title': 'Billing Invoices',
'description': 'Billing Invoices',
'newLayout': true
});
}]);
function BillingCtrl($scope, ApiService, $routeParams) {
$scope.orgname = $routeParams['orgname'];
$scope.username = $routeParams['username'];
var loadEntity = function() {
if ($scope.orgname) {
$scope.entityResource = ApiService.getOrganizationAsResource({'orgname': $scope.orgname}).get(function(org) {
$scope.organization = org;
});
} else {
$scope.entityResource = ApiService.getUserInformationAsResource({'username': $scope.username}).get(function(user) {
$scope.viewuser = user;
});
}
};
// Load the user or organization.
loadEntity();
}
}());

View file

@ -16,9 +16,10 @@
$scope.namespace = orgname; $scope.namespace = orgname;
$scope.showLogsCounter = 0; $scope.showLogsCounter = 0;
$scope.showApplicationsCounter = 0; $scope.showApplicationsCounter = 0;
$scope.showInvoicesCounter = 0; $scope.showBillingCounter = 0;
$scope.showRobotsCounter = 0; $scope.showRobotsCounter = 0;
$scope.showTeamsCounter = 0; $scope.showTeamsCounter = 0;
$scope.changeEmailInfo = null;
$scope.orgScope = { $scope.orgScope = {
'changingOrganization': false, 'changingOrganization': false,
@ -67,8 +68,8 @@
$scope.showTeamsCounter++; $scope.showTeamsCounter++;
}; };
$scope.showInvoices = function() { $scope.showBilling = function() {
$scope.showInvoicesCounter++; $scope.showBillingCounter = true;
}; };
$scope.showApplications = function() { $scope.showApplications = function() {
@ -79,25 +80,27 @@
$scope.showLogsCounter++; $scope.showLogsCounter++;
}; };
$scope.changeEmail = function() { $scope.showChangeEmail = function() {
UIService.hidePopover('#changeEmailForm input'); $scope.changeEmailInfo = {
'email': $scope.organization.email
};
};
$scope.orgScope.changingOrganization = true; $scope.changeEmail = function(info, callback) {
var params = { var params = {
'orgname': orgname 'orgname': orgname
}; };
var data = { var details = {
'email': $scope.orgScope.organizationEmail 'email': $scope.changeEmailInfo.email
}; };
ApiService.changeOrganizationDetails(data, params).then(function(org) { var errorDisplay = ApiService.errorDisplay('Could not change email address', callback);
$scope.orgScope.changingOrganization = false;
$scope.organization = org; ApiService.changeOrganizationDetails(details, params).then(function() {
}, function(result) { $scope.organization.email = $scope.changeEmailInfo.email;
$scope.orgScope.changingOrganization = false; callback(true);
UIService.showFormError('#changeEmailForm input', result, 'right'); }, errorDisplay);
});
}; };
} }
})(); })();

View file

@ -13,11 +13,13 @@
function UserViewCtrl($scope, $routeParams, $timeout, ApiService, UserService, UIService, AvatarService, Config, ExternalLoginService) { function UserViewCtrl($scope, $routeParams, $timeout, ApiService, UserService, UIService, AvatarService, Config, ExternalLoginService) {
var username = $routeParams.username; var username = $routeParams.username;
$scope.showInvoicesCounter = 0;
$scope.showAppsCounter = 0; $scope.showAppsCounter = 0;
$scope.showRobotsCounter = 0; $scope.showRobotsCounter = 0;
$scope.changeEmailInfo = {}; $scope.showBillingCounter = 0;
$scope.changePasswordInfo = {}; $scope.showLogsCounter = 0;
$scope.changeEmailInfo = null;
$scope.changePasswordInfo = null;
$scope.hasSingleSignin = ExternalLoginService.hasSingleSignin(); $scope.hasSingleSignin = ExternalLoginService.hasSingleSignin();
$scope.context = {}; $scope.context = {};
@ -54,37 +56,32 @@
$scope.showRobotsCounter++; $scope.showRobotsCounter++;
}; };
$scope.showInvoices = function() { $scope.showLogs = function() {
$scope.showInvoicesCounter++; $scope.showLogsCounter++;
}; };
$scope.showApplications = function() { $scope.showApplications = function() {
$scope.showAppsCounter++; $scope.showAppsCounter++;
}; };
$scope.changePassword = function() { $scope.showChangePassword = function() {
if (Config.AUTHENTICATION_TYPE != 'Database') { return; } $scope.changePasswordInfo = {};
};
UIService.hidePopover('#changePasswordForm'); $scope.changePassword = function(info, callback) {
$scope.changePasswordInfo.state = 'changing'; if (Config.AUTHENTICATION_TYPE != 'Database') { return; }
var data = { var data = {
'password': $scope.changePasswordInfo.password 'password': $scope.changePasswordInfo.password
}; };
var errorDisplay = ApiService.errorDisplay('Could not change password', callback);
ApiService.changeUserDetails(data).then(function(resp) { ApiService.changeUserDetails(data).then(function(resp) {
$scope.changePasswordInfo.state = 'changed';
// Reset the form
delete $scope.changePasswordInfo['password']
delete $scope.changePasswordInfo['repeatPassword']
// Reload the user. // Reload the user.
UserService.load(); UserService.load();
}, function(result) { callback(true);
$scope.changePasswordInfo.state = 'change-error'; }, errorDisplay);
UIService.showFormError('#changePasswordForm', result);
});
}; };
$scope.generateClientToken = function() { $scope.generateClientToken = function() {
@ -102,21 +99,32 @@
UIService.showPasswordDialog('Enter your password to generate an encrypted version:', generateToken); UIService.showPasswordDialog('Enter your password to generate an encrypted version:', generateToken);
}; };
$scope.changeEmail = function() { $scope.showChangeEmail = function() {
UIService.hidePopover('#changeEmailForm'); $scope.changeEmailInfo = {
'email': $scope.context.viewuser.email
};
};
$scope.changeEmail = function(info, callback) {
var details = { var details = {
'email': $scope.changeEmailInfo.email 'email': $scope.changeEmailInfo.email
}; };
$scope.changeEmailInfo.state = 'sending'; var errorDisplay = ApiService.errorDisplay('Could not change email address', callback);
ApiService.changeUserDetails(details).then(function() { ApiService.changeUserDetails(details).then(function() {
$scope.changeEmailInfo.state = 'sent'; callback(true);
delete $scope.changeEmailInfo['email']; }, errorDisplay);
}, function(result) { };
$scope.changeEmailInfo.state = 'send-error';
UIService.showFormError('#changeEmailForm', result); $scope.showChangeAccount = function() {
}); $scope.convertAccountInfo = {
'user': $scope.context.viewuser
};
};
$scope.showBilling = function() {
$scope.showBillingCounter++;
}; };
} }
})(); })();

View file

@ -313,7 +313,6 @@ angular.module('quay').factory('ApiService', ['Restangular', '$q', 'UtilService'
} }
message = UtilService.stringToHTML(message); message = UtilService.stringToHTML(message);
bootbox.dialog({ bootbox.dialog({
"message": message, "message": message,
"title": defaultMessage || 'Request Failure', "title": defaultMessage || 'Request Failure',

View file

@ -0,0 +1,32 @@
<div class="billing-page page-content">
<div class="resource-view" resource="entityResource" error-message="'Could not load entity'">
<div class="cor-title" ng-if="organization">
<span class="cor-title-link">
<a class="back-link" href="/organization/{{ organization.name }}?tab=settings">
<span class="avatar" size="24" data="organization.avatar"></span>
{{ organization.name }}
</a>
</span>
<span class="cor-title-content">
Account Plan
</span>
</div>
<div class="cor-title" ng-if="viewuser">
<span class="cor-title-link">
<a class="back-link" href="/user/{{ viewuser.username }}?tab=settings">
<span class="avatar" size="24" data="viewuser.avatar"></span>
{{ viewuser.username }}
</a>
</span>
<span class="cor-title-content">
Account Plan
</span>
</div>
<div class="co-main-content-panel" style="min-height: 500px;">
<div class="plan-manager" organization="organization.name" has-subscription="hasSubscription" ng-if="organization"></div>
<div class="plan-manager" user="viewuser" has-subscription="hasSubscription" ng-if="!organization"></div>
</div>
</div>
</div>

View file

@ -0,0 +1,32 @@
<div class="billing-page page-content">
<div class="resource-view" resource="entityResource" error-message="'Could not load entity'">
<div class="cor-title" ng-if="organization">
<span class="cor-title-link">
<a class="back-link" href="/organization/{{ organization.name }}?tab=settings">
<span class="avatar" size="24" data="organization.avatar"></span>
{{ organization.name }}
</a>
</span>
<span class="cor-title-content">
Billing Invoices
</span>
</div>
<div class="cor-title" ng-if="viewuser">
<span class="cor-title-link">
<a class="back-link" href="/user/{{ viewuser.username }}?tab=settings">
<span class="avatar" size="24" data="viewuser.avatar"></span>
{{ viewuser.username }}
</a>
</span>
<span class="cor-title-content">
Billing Invoices
</span>
</div>
<div class="co-main-content-panel" style="min-height: 500px;">
<div class="billing-invoices" user="viewuser" makevisible="true" ng-if="!organization"></div>
<div class="billing-invoices" organization="organization" makevisible="true" ng-if="organization"></div>
</div>
</div>
</div>

View file

@ -35,14 +35,6 @@
<span class="cor-tab" tab-title="Default Permissions" tab-target="#default" ng-show="isAdmin"> <span class="cor-tab" tab-title="Default Permissions" tab-target="#default" ng-show="isAdmin">
<i class="fa ci-stamp"></i> <i class="fa ci-stamp"></i>
</span> </span>
<span class="cor-tab" tab-title="Billing" tab-target="#usage"
quay-show="isAdmin && Features.BILLING">
<i class="fa fa-credit-card"></i>
</span>
<span class="cor-tab" tab-title="Billing Invoices" tab-target="#invoices"
tab-init="showInvoices()" quay-show="isAdmin && Features.BILLING">
<i class="fa ci-invoice"></i>
</span>
<span class="cor-tab" tab-title="Usage Logs" tab-target="#logs" <span class="cor-tab" tab-title="Usage Logs" tab-target="#logs"
tab-init="showLogs()" ng-show="isAdmin"> tab-init="showLogs()" ng-show="isAdmin">
<i class="fa fa-bar-chart"></i> <i class="fa fa-bar-chart"></i>
@ -52,7 +44,7 @@
<i class="fa ci-application"></i> <i class="fa ci-application"></i>
</span> </span>
<span class="cor-tab" tab-title="Organization Settings" tab-target="#settings" <span class="cor-tab" tab-title="Organization Settings" tab-target="#settings"
ng-show="isAdmin"> ng-show="isAdmin" tab-init="showBilling()">
<i class="fa fa-gears"></i> <i class="fa fa-gears"></i>
</span> </span>
</div> <!-- /cor-tabs --> </div> <!-- /cor-tabs -->
@ -95,45 +87,60 @@
makevisible="showApplicationsCounter"></div> makevisible="showApplicationsCounter"></div>
</div> </div>
<!-- Plan and Usage -->
<div id="usage" class="tab-pane" quay-require="['BILLING']">
<h3>Plan Usage and Billing</h3>
<div ng-if="isAdmin">
<div class="plan-manager" organization="organization.name"
has-subscription="hasSubscription"></div>
<hr ng-show="hasSubscription">
<div class="billing-options" organization="organization" ng-show="hasSubscription"></div>
</div>
</div>
<!-- Billing Invoices -->
<div id="invoices" class="tab-pane" quay-require="['BILLING']">
<h3>Billing Invoices</h3>
<div class="billing-invoices" organization="organization"
makevisible="showInvoicesCounter"></div>
</div>
<!-- Settings --> <!-- Settings -->
<div id="settings" class="tab-pane"> <div id="settings" class="tab-pane">
<div ng-if="isAdmin"> <div ng-if="isAdmin">
<h3>Organization Settings</h3> <!-- Org Settings -->
<div class="settings-section">
<h3>Organization Settings</h3>
<table class="co-list-table">
<tr>
<td>Namespace:</td>
<td>
{{ organization.name }}
<div class="help-text">Organization names cannot currently be changed. Please <a href="/contact">contact support</a> to migrate accounts.</div>
</td>
</tr>
<tr>
<td>Avatar:</td>
<td>
<span class="avatar" size="48" data="organization.avatar"></span>
<div class="help-text" ng-if="Config.AVATAR_KIND == 'local'">Avatar is generated based off the organization's name.</div>
<div class="help-text" ng-if="Config.AVATAR_KIND == 'gravatar'">Avatar is served by <a href="http://gravatar.com" rel="nofollow" target="_blank">Gravatar</a> based on the {{ organization.email }} e-mail address.</div>
</td>
</tr>
<tr quay-show="Features.MAILING">
<td>Email Address:</td>
<td>
<a class="co-modify-link" ng-click="showChangeEmail()">{{ organization.email }}</a>
</td>
</tr>
</table>
</div>
<div class="panel" ng-show="!orgScope.changingOrganization"> <!-- Billing Information -->
<div class="panel-title">Organization's e-mail address</div> <div class="settings-section">
<div class="panel-content" style="padding-left: 20px; margin-top: 10px;"> <h3>Billing Information</h3>
<form class="form-change" id="changeEmailForm" name="changeEmailForm" ng-submit="changeEmail()"> <div class="billing-management-panel" organization="organization" is-enabled="showBillingCounter"></div>
<input type="email" class="form-control" style="max-width: 500px;"
ng-model="orgScope.organizationEmail" required>
<button class="btn btn-primary" type="submit"
ng-disabled="changeEmailForm.$invalid || orgScope.organizationEmail == organization.email">
Save
</button>
</form>
</div>
</div> </div>
</div> </div>
</div> </div>
</div> <!-- /cor-tab-content --> </div> <!-- /cor-tab-content -->
</div> </div>
</div> </div>
<!-- Change email dialog -->
<div class="cor-confirm-dialog"
dialog-context="changeEmailInfo"
dialog-action="changeEmail(info, callback)"
dialog-title="Change E-mail Address"
dialog-action-title="Change Email"
dialog-form="context.emailform">
<form name="context.emailform" class="co-single-field-dialog">
Please enter a new email address.
<input type="email" class="form-control" placeholder="Your new e-mail address"
ng-model="changeEmailInfo.email" required>
</form>
</div>
</div> </div>

View file

@ -28,24 +28,16 @@
<span class="cor-tab" tab-title="Robot Accounts" tab-init="showRobots()" tab-target="#robots"> <span class="cor-tab" tab-title="Robot Accounts" tab-init="showRobots()" tab-target="#robots">
<i class="fa ci-robot"></i> <i class="fa ci-robot"></i>
</span> </span>
<span class="cor-tab" tab-title="User Settings" tab-target="#settings"> <span class="cor-tab" tab-title="External Logins And Applications" tab-target="#external"
<i class="fa fa-gears"></i> tab-init="showApplications()">
</span>
<span class="cor-tab" tab-title="Billing" tab-target="#usage"
quay-show="Features.BILLING">
<i class="fa fa-credit-card"></i>
</span>
<span class="cor-tab" tab-title="Billing Invoices" tab-target="#invoices"
tab-init="showInvoices()" quay-show="Features.BILLING">
<i class="fa ci-invoice"></i>
</span>
<span class="cor-tab" tab-title="External Logins" tab-target="#external"
quay-show="!hasSingleSignin">
<i class="fa fa-external-link-square"></i> <i class="fa fa-external-link-square"></i>
</span> </span>
<span class="cor-tab" tab-title="Authorized Applications" tab-target="#applications" <span class="cor-tab" tab-title="Usage Logs" tab-target="#logs" tab-init="showLogs()"
tab-init="showApplications()"> quay-show="Features.USER_LOG_ACCESS">
<i class="fa ci-application"></i> <i class="fa fa-bar-chart"></i>
</span>
<span class="cor-tab" tab-title="User Settings" tab-target="#settings" tab-init="showBilling()">
<i class="fa fa-gears"></i>
</span> </span>
</div> <!-- /cor-tabs --> </div> <!-- /cor-tabs -->
@ -60,149 +52,129 @@
<div class="robots-manager" user="viewuser" is-enabled="showRobotsCounter"></div> <div class="robots-manager" user="viewuser" is-enabled="showRobotsCounter"></div>
</div> </div>
<!-- External Logins --> <!-- Usage Logs -->
<div id="logs" class="tab-pane">
<div class="logs-view" user="viewuser" makevisible="showLogsCounter"></div>
</div>
<!-- External Logins And Applications -->
<div id="external" class="tab-pane" quay-show="!hasSingleSignin"> <div id="external" class="tab-pane" quay-show="!hasSingleSignin">
<div class="external-logins-manager" user="viewuser"></div> <div class="external-logins-manager" user="viewuser"
</div> quay-show="!hasSingleSignin"></div>
<div style="margin: 50px" quay-show="!hasSingleSignin">
<!-- Applications --> </div>
<div id="applications" class="tab-pane">
<div class="authorized-apps-manager" user="viewuser" is-enabled="showAppsCounter"></div> <div class="authorized-apps-manager" user="viewuser" is-enabled="showAppsCounter"></div>
</div> </div>
<!-- Plan and Usage -->
<div id="usage" class="tab-pane" quay-require="['BILLING']">
<h3>Plan Usage and Billing</h3>
<div class="plan-manager" user="viewuser" has-subscription="hasSubscription"></div>
<hr ng-show="hasSubscription">
<div class="billing-options" user="viewuser" ng-show="hasSubscription"></div>
</div>
<!-- Billing Invoices -->
<div id="invoices" class="tab-pane" quay-require="['BILLING']">
<h3>Billing Invoices</h3>
<div class="billing-invoices" user="viewuser"
makevisible="showInvoicesCounter"></div>
</div>
<!-- Settings --> <!-- Settings -->
<div id="settings" class="tab-pane"> <div id="settings" class="tab-pane">
<h3>User Settings</h3> <!-- Encrypted Password -->
<div class="settings-section">
<!-- E-mail address --> <h3>Docker CLI Password</h3>
<div class="co-panel" quay-show="Features.MAILING"> <div ng-if="!Features.REQUIRE_ENCRYPTED_BASIC_AUTH">
<div class="co-panel-heading"><i class="fa fa-envelope-o"></i> E-mail Address</div> The Docker CLI stores passwords entered on the command line in <strong>plaintext</strong>. It is therefore highly recommended to generate an an encrypted version of your password to use for <code>docker login</code>.
<div class="panel-body" style="padding-top: 5px;">
<div class="co-alert co-alert-success" ng-show="changeEmailInfo.state == 'sent'">
An e-mail has been sent to {{ sentEmail }} to verify the change.
</div>
<div class="cor-loader" ng-show="changeEmailInfo.state == 'sending'"></div>
<div ng-show="changeEmailInfo.state != 'sending'">
<form class="form-change user-settings-form"
id="changeEmailForm" name="changeEmailForm"
ng-submit="changeEmail(); changeEmailForm.$setPristine()"
ng-show="!awaitingConfirmation && !registering">
<div class="row">
<table class="col-md-6">
<tr>
<td>Current E-mail Address:</td>
<td>{{ context.viewuser.email }}</td>
</tr>
<tr>
<td>New E-mail Address:</td>
<td>
<input type="email" class="form-control"
placeholder="Your new e-mail address"
ng-model="changeEmailInfo.email" required>
</td>
</tr>
</table>
</div>
<button class="btn btn-primary"
ng-disabled="changeEmailForm.$invalid || changeEmail.email == context.viewuser.email"
type="submit">
Change E-mail Address
</button>
</form>
</div>
</div> </div>
</div> <!-- /E-mail -->
<!-- Password --> <div ng-if="Features.REQUIRE_ENCRYPTED_BASIC_AUTH">
<div class="co-panel" style="margin-bottom: 0px"> This installation is set to <strong>require</strong> encrypted passwords when
<div class="co-panel-heading"><i class="fa fa-lock"></i> Password</div> using the Docker command line interface.
<div class="panel-body" style="padding-top: 5px;">
<div class="cor-loader" ng-show="changePasswordInfo.state == 'changing'"></div>
<!-- Encrypted Password -->
<div class="row" ng-show="changePasswordInfo.state !='changing'">
<div class="panel">
<div class="panel-title">Generate Encrypted Password</div>
<div class="panel-body">
<div class="co-alert co-alert-info" ng-if="!Features.REQUIRE_ENCRYPTED_BASIC_AUTH">
Due to Docker storing passwords entered on the command line in <strong>plaintext</strong>, it is highly recommended to use the button below to generate an an encrypted version of your password.
</div>
<div class="co-alert co-alert-warning" ng-if="Features.REQUIRE_ENCRYPTED_BASIC_AUTH">
This installation is set to <strong>require</strong> encrypted passwords when
using the Docker command line interface. To generate an encrypted password, click the button below.
</div>
<button class="btn btn-primary" ng-click="generateClientToken()">
<i class="fa fa-key" style="margin-right: 6px;"></i>Generate Encrypted Password
</button>
</div>
</div>
</div>
<!-- Change Password -->
<div class="row" quay-show="Config.AUTHENTICATION_TYPE == 'Database' && changePasswordInfo.state !='changing'">
<div class="panel">
<div class="panel-title">Change Password</div>
<span class="help-block" ng-show="changePasswordInfo.state == 'changed'">
Password changed successfully
</span>
<div class="panel-body">
<div class="co-alert co-alert-warning">Note: Changing your password will also invalidate any generated encrypted passwords.</div>
<form class="form-change col-md-6" id="changePasswordForm" name="changePasswordForm" ng-submit="changePassword(); changePasswordForm.$setPristine()"
ng-show="!awaitingConfirmation && !registering">
<input type="password" class="form-control" placeholder="Your new password" ng-model="changePasswordInfo.password" required
ng-pattern="/^.{8,}$/">
<input type="password" class="form-control" placeholder="Verify your new password" ng-model="changePasswordInfo.repeatPassword"
match="changePasswordInfo.password" required ng-pattern="/^.{8,}$/">
<button class="btn btn-danger" ng-disabled="changePasswordForm.$invalid" type="submit"
analytics-on analytics-event="change_pass">Change Password</button>
</form>
</div>
</div>
</div>
</div> </div>
</div> <!-- /Password -->
<!-- Convert --> <table class="co-list-table" style="margin-top: 10px;">
<div class="co-panel" quay-show="Config.AUTHENTICATION_TYPE == 'Database'"> <tr>
<div class="co-panel-heading"><i class="fa fa-group"></i> Convert to organization</div> <td>CLI Password:</td>
<div class="panel-body" style="padding-top: 5px;"> <td>
<div class="convert-user-to-org" user="viewuser"></div> <a ng-click="generateClientToken()">Generate Encrypted Password</a>
</div> </td>
</tr>
</table>
</div> </div>
</div> <!-- /Convert -->
</div> <!-- /cor-tab-content --> <!-- User Settings -->
<div class="settings-section">
<h3>User Settings</h3>
<table class="co-list-table">
<tr>
<td>Username:</td>
<td>
{{ context.viewuser.username }}
<div class="help-text">Usernames cannot currently be changed. Please <a href="/contact">contact support</a> to migrate accounts.</div>
</td>
</tr>
<tr>
<td>Avatar:</td>
<td>
<span class="avatar" size="48" data="context.viewuser.avatar"></span>
<div class="help-text" ng-if="Config.AVATAR_KIND == 'local'">Avatar is generated based off of your username.</div>
<div class="help-text" ng-if="Config.AVATAR_KIND == 'gravatar'">Avatar is served by <a href="http://gravatar.com" rel="nofollow" target="_blank">Gravatar</a> based on the {{ context.viewuser.email }} e-mail address.</div>
</td>
</tr>
<tr quay-show="Features.MAILING">
<td>Email Address:</td>
<td>
<a class="co-modify-link" ng-click="showChangeEmail()">{{ context.viewuser.email }}</a>
</td>
</tr>
<tr quay-show="Config.AUTHENTICATION_TYPE == 'Database'">
<td>Password:</td>
<td>
<a class="co-modify-link" ng-click="showChangePassword()">Change password</a>
</td>
</tr>
<tr quay-show="Config.AUTHENTICATION_TYPE == 'Database'">
<td>Account Type:</td>
<td>
<a class="co-modify-link" ng-click="showChangeAccount()">Individual account</a>
</td>
</tr>
</table>
</div>
<!-- Billing Information -->
<div class="settings-section">
<h3>Billing Information</h3>
<div class="billing-management-panel" user="context.viewuser" is-enabled="showBillingCounter"></div>
</div>
</div> <!-- /cor-tab-content -->
</div> </div>
</div> </div>
<!-- Change email dialog -->
<div class="cor-confirm-dialog"
dialog-context="changeEmailInfo"
dialog-action="changeEmail(info, callback)"
dialog-title="Change E-mail Address"
dialog-action-title="Change Email"
dialog-form="context.emailform">
<form name="context.emailform" class="co-single-field-dialog">
Please enter a new email address. A verification email will be sent before being applied.
<input type="email" class="form-control" placeholder="Your new e-mail address"
ng-model="changeEmailInfo.email" required>
</form>
</div>
<!-- Change password dialog -->
<div class="cor-confirm-dialog"
dialog-context="changePasswordInfo"
dialog-action="changePassword(info, callback)"
dialog-title="Change Password"
dialog-action-title="Change Password"
dialog-form="context.passwordform">
<form name="context.passwordform" class="co-single-field-dialog">
Enter a new password. Passwords must be at least 8 characters in length.
<input type="password" class="form-control" placeholder="Your new password" ng-model="changePasswordInfo.password" required
ng-pattern="/^.{8,}$/">
<input type="password" class="form-control" placeholder="Verify your new password" ng-model="changePasswordInfo.repeatPassword"
match="changePasswordInfo.password" required ng-pattern="/^.{8,}$/">
</form>
</div>
<!-- Convert account dialog -->
<div class="convert-user-to-org" info="convertAccountInfo"></div>
<!-- Modal message dialog --> <!-- Modal message dialog -->
<div class="modal fade" id="clientTokenModal"> <div class="co-dialog modal fade" id="clientTokenModal">
<div class="modal-dialog"> <div class="modal-dialog">
<div class="modal-content"> <div class="modal-content">
<div class="modal-header"> <div class="modal-header">