GUAC-605: Migrate status to guacNotification. Add countdown string to en_US translation. Use 15 second reconnect countdown if appropriate for error at hand.

This commit is contained in:
Michael Jumper
2014-11-29 20:54:50 -08:00
parent 5b31b206a7
commit ed31e0c026
7 changed files with 166 additions and 248 deletions

View File

@@ -51,6 +51,18 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
0x031D: true 0x031D: true
}; };
/**
* All error codes for which automatic reconnection is appropriate when a
* client error occurs.
*/
var CLIENT_AUTO_RECONNECT = {
0x0200: true,
0x0202: true,
0x0203: true,
0x0301: true,
0x0308: true
};
/** /**
* All tunnel error codes handled and passed off for translation. Any error * All tunnel error codes handled and passed off for translation. Any error
* code not present in this list will be represented by the "DEFAULT" * code not present in this list will be represented by the "DEFAULT"
@@ -67,7 +79,18 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
0x0308: true, 0x0308: true,
0x031D: true 0x031D: true
}; };
/**
* All error codes for which automatic reconnection is appropriate when a
* tunnel error occurs.
*/
var TUNNEL_AUTO_RECONNECT = {
0x0200: true,
0x0202: true,
0x0203: true,
0x0308: true
};
/** /**
* The reconnect action to be provided along with the object sent to * The reconnect action to be provided along with the object sent to
* showStatus. * showStatus.
@@ -81,6 +104,16 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
} }
}; };
/**
* The reconnect countdown to display if an error or status warrants an
* automatic, timed reconnect.
*/
var RECONNECT_COUNTDOWN = {
text: "client.action.reconnectCountdown",
callback: RECONNECT_ACTION.callback,
remaining: 15
};
// Get DAO for reading connections and groups // Get DAO for reading connections and groups
var connectionGroupDAO = $injector.get('connectionGroupDAO'); var connectionGroupDAO = $injector.get('connectionGroupDAO');
var connectionDAO = $injector.get('connectionDAO'); var connectionDAO = $injector.get('connectionDAO');
@@ -146,7 +179,7 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
return true; return true;
} }
$scope.$watch('menuShown', function setKeyboardEnabled(menuShown, menuShownPreviousState) { $scope.$watch('menuShown', function setKeyboardEnabled(menuShown, menuShownPreviousState) {
// Send clipboard data if menu is hidden // Send clipboard data if menu is hidden
@@ -218,7 +251,10 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
$scope.id = null; $scope.id = null;
// Determine translation name of error // Determine translation name of error
var errorName = (status in CLIENT_ERRORS) ? status.toString(16) : "DEFAULT"; var errorName = (status in CLIENT_ERRORS) ? status.toString(16).toUpperCase() : "DEFAULT";
// Determine whether the reconnect countdown applies
var countdown = (status in CLIENT_AUTO_RECONNECT) ? RECONNECT_COUNTDOWN : null;
// Override any existing status // Override any existing status
$scope.showStatus(false); $scope.showStatus(false);
@@ -228,6 +264,7 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
className: "error", className: "error",
title: "client.error.connectionErrorTitle", title: "client.error.connectionErrorTitle",
text: "client.error.clientErrors." + errorName, text: "client.error.clientErrors." + errorName,
countdown: countdown,
actions: [ RECONNECT_ACTION ] actions: [ RECONNECT_ACTION ]
}); });
@@ -253,7 +290,10 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
$scope.id = null; $scope.id = null;
// Determine translation name of error // Determine translation name of error
var errorName = (status in TUNNEL_ERRORS) ? status.toString(16) : "DEFAULT"; var errorName = (status in TUNNEL_ERRORS) ? status.toString(16).toUpperCase() : "DEFAULT";
// Determine whether the reconnect countdown applies
var countdown = (status in TUNNEL_AUTO_RECONNECT) ? RECONNECT_COUNTDOWN : null;
// Override any existing status // Override any existing status
$scope.showStatus(false); $scope.showStatus(false);
@@ -263,6 +303,7 @@ angular.module('home').controller('clientController', ['$scope', '$routeParams',
className: "error", className: "error",
title: "client.error.connectionErrorTitle", title: "client.error.connectionErrorTitle",
text: "client.error.tunnelErrors." + errorName, text: "client.error.tunnelErrors." + errorName,
countdown: countdown,
actions: [ RECONNECT_ACTION ] actions: [ RECONNECT_ACTION ]
}); });

View File

@@ -0,0 +1,43 @@
/*
* Copyright (C) 2013 Glyptodon LLC
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#notificationArea {
position: fixed;
right: 0.5em;
bottom: 0.5em;
max-width: 25%;
width: 2in;
}
#notificationArea .notification {
font-size: 0.7em;
text-align: center;
width: 100%;
overflow: hidden;
}
#notificationArea .notification .text {
width: 100%;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}

View File

@@ -1,159 +0,0 @@
/*
* Copyright (C) 2013 Glyptodon LLC
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
* THE SOFTWARE.
*/
#notificationArea {
position: fixed;
right: 0.5em;
bottom: 0.5em;
max-width: 25%;
min-width: 10em;
}
.notification {
font-size: 0.7em;
text-align: center;
border: 1px solid rgba(0, 0, 0, 0.125);
box-shadow: 1px 1px 2px rgba(0, 0, 0, 0.125);
background: white;
color: black;
padding: 0.5em;
margin: 1em;
width: 2in;
max-width: 75%;
overflow: hidden;
}
.notification .buttons {
margin: 0;
}
@keyframes notification-progress {
from {background-position: 0px 0px;}
to {background-position: 64px 0px;}
}
@-webkit-keyframes notification-progress {
from {background-position: 0px 0px;}
to {background-position: 64px 0px;}
}
.notification .title-bar {
font-size: 1.25em;
font-weight: bold;
text-transform: uppercase;
border-bottom: 1px solid rgba(0, 0, 0, 0.125);
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.125);
background: rgba(0, 0, 0, 0.04);
margin-bottom: 1em;
}
.notification .text {
width: 100%;
white-space: nowrap;
overflow: hidden;
text-overflow: ellipsis;
}
.notification.upload .progress {
float: none;
width: 80%;
margin-left: auto;
margin-right: auto;
}
.notification.download .progress div,
.notification.upload .progress div {
position: relative;
}
.notification.download .progress .bar,
.notification.upload .progress .bar {
background: #A3D655;
position: absolute;
top: 0;
left: 0;
height: 100%;
width: 0;
box-shadow: inset 1px 1px 0 rgba(255, 255, 255, 0.5),
inset -1px -1px 0 rgba( 0, 0, 0, 0.1),
1px 1px 0 gray;
}
.notification.upload .progress,
.notification.download .progress {
width: 100%;
background: #C2C2C2 url('images/progress.png');
background-size: 16px 16px;
-moz-background-size: 16px 16px;
-webkit-background-size: 16px 16px;
-khtml-background-size: 16px 16px;
animation-name: notification-progress;
animation-duration: 2s;
animation-timing-function: linear;
animation-iteration-count: infinite;
-webkit-animation-name: notification-progress;
-webkit-animation-duration: 2s;
-webkit-animation-timing-function: linear;
-webkit-animation-iteration-count: infinite;
padding: 0.25em;
min-width: 5em;
border: 1px solid gray;
-moz-border-radius: 0.2em;
-webkit-border-radius: 0.2em;
-khtml-border-radius: 0.2em;
border-radius: 0.2em;
text-align: center;
float: right;
position: relative;
}
.notification.download .download {
background: rgb(16, 87, 153);
cursor: pointer;
}
.notification.message {
background: #DFD;
animation: fadein 0.125s linear, fadeout 2s 3s linear;
-webkit-animation: fadein 0.125s linear, fadeout 2s 3s linear;
}
.notification.message .caption {
vertical-align: middle;
white-space: normal;
overflow: hidden;
text-overflow: ellipsis;
}

View File

@@ -73,20 +73,30 @@ angular.module('index').controller('indexController', ['$scope', '$injector',
$location.path('/login'); $location.path('/login');
/** /**
* Shows or hides the status. If a status is currently shown, * Shows or hides the given notification as a modal status. If a status
* no further statuses will be shown until the current status is * notification is currently shown, no further statuses will be shown
* hidden. * until the current status is hidden.
* *
* @param {Boolean|Object} status The status to show, or false to hide the * @param {Object} status The status notification to show.
* current status. * @param {String} [status.title] The title of the notification.
* @param {String} [status.title] The title of the status. * @param {String} [status.text] The body text of the notification.
* @param {String} [status.text] The body text of the status.
* @param {String} [status.className] The CSS class name to apply. * @param {String} [status.className] The CSS class name to apply.
* @param {Object[]} [status.actions] Array of action objects which *
* contain an action name and callback to * @param {String} [status.countdown.text]
* be executed when that action is * In the case that a countdown applies to the notification, the text to
* invoked. * display while the countdown is active.
* *
* @param {Function} [status.countdown.callback]
* The callback to call when the countdown expires.
*
* @param {String} [status.countdown.remaining]
* The number of seconds remaining before the countdown callback is
* called.
*
* @param {Object[]} [status.actions]
* Array of action objects which contain an action name and callback to
* be executed when that action is invoked.
*
* @example * @example
* *
* // To show a status message with actions * // To show a status message with actions
@@ -114,15 +124,26 @@ angular.module('index').controller('indexController', ['$scope', '$injector',
* *
* @param {Object} notification The notification to add. * @param {Object} notification The notification to add.
* @param {String} [notification.title] The title of the notification. * @param {String} [notification.title] The title of the notification.
* @param {String} [notification.text] The body text of the status modal. * @param {String} [notification.text] The body text of the notification.
* @param {String} [notification.className] The CSS class name to apply. * @param {String} [notification.className] The CSS class name to apply.
* @param {Object[]} [notification.actions] Array of action objects which *
* contain an action name and callback to * @param {String} [notification.countdown.text]
* be executed when that action is * In the case that a countdown applies to the notification, the text to
* invoked. * display while the countdown is active.
*
* @param {Function} [notification.countdown.callback]
* The callback to call when the countdown expires.
*
* @param {String} [notification.countdown.remaining]
* The number of seconds remaining before the countdown callback is
* called.
*
* @param {Object[]} [notification.actions]
* Array of action objects which contain an action name and callback to
* be executed when that action is invoked.
*
* @returns {Number} A unique ID for the notification that's just been added. * @returns {Number} A unique ID for the notification that's just been added.
* *
*
* @example * @example
* *
* var id = $scope.addNotification({ * var id = $scope.addNotification({
@@ -138,7 +159,7 @@ angular.module('index').controller('indexController', ['$scope', '$injector',
*/ */
$scope.addNotification = function addNotification(notification) { $scope.addNotification = function addNotification(notification) {
var id = ++notificationUniqueID; var id = ++notificationUniqueID;
$scope.notifications.push({ $scope.notifications.push({
notification : notification, notification : notification,
id : id id : id

View File

@@ -20,16 +20,6 @@
* THE SOFTWARE. * THE SOFTWARE.
*/ */
.status-container {
position: fixed;
top: 0;
left: 0;
bottom: 0;
right: 0;
background: rgba(0, 0, 0, 0.5);
padding: 1em;
}
.status-outer { .status-outer {
display: table; display: table;
height: 100%; height: 100%;
@@ -48,7 +38,7 @@
vertical-align: middle; vertical-align: middle;
} }
.status { .status-middle .notification {
width: 75%; width: 75%;
max-width: 5in; max-width: 5in;
@@ -56,33 +46,22 @@
margin-right: auto; margin-right: auto;
overflow: auto; overflow: auto;
background: white;
border: 1px solid rgba(0, 0, 0, 0.25);
box-shadow: 1px 1px 2px rgba(0, 0, 0, 0.25);
text-align: left; text-align: left;
} }
.status .title { .status-middle .notification .body {
font-size: 1.25em;
font-weight: bold;
background: rgba(0, 0, 0, 0.04);
border-bottom: 1px solid rgba(0, 0, 0, 0.125);
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.125);
margin: 0;
padding: 0.5em 1em;
text-transform: uppercase;
}
.status.error {
background: #FDD;
}
.status > * {
margin: 1.25em; margin: 1.25em;
} }
.status-middle .notification .buttons {
margin: 1em;
}
.status-middle .notification.error {
background: #FDD;
}
/* Fade entire status area in/out based on shown status */ /* Fade entire status area in/out based on shown status */
.status-outer { .status-outer {
@@ -99,10 +78,10 @@
/* Hide dialog immediately based on status */ /* Hide dialog immediately based on status */
.status { .status-middle .notification {
visibility: hidden; visibility: hidden;
} }
.shown .status { .shown .status-middle .notification {
visibility: visible; visibility: visible;
} }

View File

@@ -37,20 +37,17 @@ THE SOFTWARE.
<!-- Global status/error dialog --> <!-- Global status/error dialog -->
<div ng-class="{shown: status}" class="status-outer"> <div ng-class="{shown: status}" class="status-outer">
<div class="status-middle"> <div class="status-middle">
<div class="status" ng-class="status.className">
<guac-notification
<!-- Status title --> class-name="status.className"
<p ng-show="status.title" class="title">{{status.title | translate}}</p> title="status.title"
text="status.text"
<!-- Status text --> progress="status.progress"
<p ng-show="status.text" class="text">{{status.text | translate}}</p> actions="status.actions"
countdown-text="status.countdown.text"
<!-- All action buttons, if any --> countdown="status.countdown.remaining"
<div ng-show="status.actions && status.actions.length" class="buttons"> default-callback="status.countdown.callback"/>
<button ng-repeat="action in status.actions" ng-click="action.callback()">{{action.name | translate}}</button>
</div>
</div>
</div> </div>
</div> </div>
@@ -59,24 +56,19 @@ THE SOFTWARE.
<!-- Notification area --> <!-- Notification area -->
<div id="notificationArea"> <div id="notificationArea">
<div ng-repeat="wrapper in notifications" ng-class="wrapper.notification.className" class="notification"> <div ng-repeat="wrapper in notifications">
<!-- Notification title --> <guac-notification
<div ng-show="wrapper.notification.title" class="title-bar"> class-name="wrapper.notification.className"
<div class="title">{{wrapper.notification.title | translate}}</div> title="wrapper.notification.title"
</div> text="wrapper.notification.text"
progress="wrapper.notification.progress"
<!-- Notification text --> actions="wrapper.notification.actions"
<p ng-show="wrapper.notification.text" class="text">{{wrapper.notification.text}}</p> countdown-text="wrapper.notification.countdown.text"
countdown="wrapper.notification.countdown.remaining"
<div ng-show="wrapper.notification.progress" class="progress"> default-callback="wrapper.notification.countdown.callback"/>
{{wrapper.notification.progress}}
</div> <div>
<div ng-show="wrapper.notification.actions && wrapper.notification.actions.length" class="buttons">
<button ng-repeat="action in wrapper.notification.actions" ng-click="action.callback()">{{action.name | translate}}</button>
</div>
</div>
</div> </div>
<script type="text/javascript" src="guacamole.min.js"></script> <script type="text/javascript" src="guacamole.min.js"></script>

View File

@@ -267,7 +267,8 @@
} }
}, },
"action" : { "action" : {
"reconnect" : "Reconnect" "reconnect" : "Reconnect",
"reconnectCountdown" : "Reconnecting in {REMAINING} {REMAINING, plural, one{second} other{seconds}}..."
}, },
"fileTransfer" : { "fileTransfer" : {
"title" : "File Transfer", "title" : "File Transfer",