diff --git a/services/static-webserver/client/source/class/osparc/desktop/credits/CurrentUsage.js b/services/static-webserver/client/source/class/osparc/desktop/credits/CurrentUsage.js
index ad7360b40ea..243922015d1 100644
--- a/services/static-webserver/client/source/class/osparc/desktop/credits/CurrentUsage.js
+++ b/services/static-webserver/client/source/class/osparc/desktop/credits/CurrentUsage.js
@@ -37,57 +37,47 @@ qx.Class.define("osparc.desktop.credits.CurrentUsage", {
}
},
- statics: {
- POLLING_INTERVAL: 10000
- },
-
members: {
- __interval: null,
-
__currentStudyChanged: function(currentStudy) {
if (osparc.desktop.credits.Utils.areWalletsEnabled()) {
if (currentStudy) {
- this.__startRequesting();
+ const store = osparc.store.Store.getInstance();
+ const contextWallet = store.getContextWallet();
+ if (contextWallet) {
+ this.__fetchUsedCredits();
+ contextWallet.addListener("changeCreditsAvailable", () => this.__fetchUsedCredits());
+ }
} else {
- this.__stopRequesting();
+ this.setUsedCredits(null);
}
}
},
- __startRequesting: function() {
- this.setUsedCredits(0);
-
- this.__interval = setInterval(() => this.__fetchUsedCredits(), this.self().POLLING_INTERVAL);
- this.__fetchUsedCredits();
- },
-
- __stopRequesting: function() {
- this.setUsedCredits(null);
-
- if (this.__interval) {
- clearInterval(this.__interval);
- }
- },
-
__fetchUsedCredits: function() {
- const params = {
- url: {
- offset: 0,
- limit: 10
- }
- };
- osparc.data.Resources.fetch("resourceUsage", "getPage", params)
- .then(data => {
- const currentStudy = osparc.store.Store.getInstance().getCurrentStudy();
- const currentTasks = data.filter(d => (d.project_id === currentStudy.getUuid()) && d.service_run_status === "RUNNING");
- let cost = 0;
- currentTasks.forEach(currentTask => {
- if (currentTask["credit_cost"]) {
- cost += currentTask["credit_cost"];
- }
+ const store = osparc.store.Store.getInstance();
+ const currentStudy = store.getCurrentStudy();
+ const contextWallet = store.getContextWallet();
+ if (currentStudy && contextWallet) {
+ const walletId = contextWallet.getWalletId();
+ const params = {
+ url: {
+ walletId,
+ offset: 0,
+ limit: 10
+ }
+ };
+ osparc.data.Resources.fetch("resourceUsagePerWallet", "getPage", params)
+ .then(data => {
+ const currentTasks = data.filter(d => (d.project_id === currentStudy.getUuid()) && d.service_run_status === "RUNNING");
+ let cost = 0;
+ currentTasks.forEach(currentTask => {
+ if (currentTask["credit_cost"]) {
+ cost += currentTask["credit_cost"];
+ }
+ });
+ this.setUsedCredits(cost);
});
- this.setUsedCredits(cost);
- });
+ }
}
}
});
diff --git a/services/static-webserver/client/source/class/osparc/desktop/credits/Utils.js b/services/static-webserver/client/source/class/osparc/desktop/credits/Utils.js
index 693eb6e6c1a..bb184469035 100644
--- a/services/static-webserver/client/source/class/osparc/desktop/credits/Utils.js
+++ b/services/static-webserver/client/source/class/osparc/desktop/credits/Utils.js
@@ -43,7 +43,8 @@ qx.Class.define("osparc.desktop.credits.Utils", {
creditsToColor: function(credits, defaultColor = "text") {
const preferencesSettings = osparc.Preferences.getInstance();
let color = defaultColor;
- if (credits <= 0) {
+ const dangerZone = 25; // one hour consumption
+ if (credits <= dangerZone) {
color = "danger-red";
} else if (credits <= preferencesSettings.getCreditsWarningThreshold()) {
color = "warning-yellow";
@@ -60,7 +61,7 @@ qx.Class.define("osparc.desktop.credits.Utils", {
},
creditsToFixed: function(credits) {
- if (credits < 100) {
+ if (credits < 10) {
return (credits).toFixed(1);
}
return parseInt(credits);
diff --git a/services/static-webserver/client/source/class/osparc/navigation/CreditsMenuButton.js b/services/static-webserver/client/source/class/osparc/navigation/CreditsMenuButton.js
index 93c7de9730a..2a8b5a40b55 100644
--- a/services/static-webserver/client/source/class/osparc/navigation/CreditsMenuButton.js
+++ b/services/static-webserver/client/source/class/osparc/navigation/CreditsMenuButton.js
@@ -27,9 +27,19 @@ qx.Class.define("osparc.navigation.CreditsMenuButton", {
this.set({
font: "text-16",
- backgroundColor: "transparent"
+ padding: 1,
+ paddingLeft: 8,
+ paddingRight: 8,
+ marginTop: 4,
+ marginBottom: 4,
+ rich: true
});
+ this.getChildControl("label").set({
+ textAlign: "right"
+ });
+ this.getContentElement().setStyle("line-height", 1.2);
+
const preferencesSettings = osparc.Preferences.getInstance();
this.__computeVisibility();
preferencesSettings.addListener("changeWalletIndicatorVisibility", () => this.__computeVisibility());
@@ -60,7 +70,25 @@ qx.Class.define("osparc.navigation.CreditsMenuButton", {
osparc.utils.Utils.prettifyMenu(menu);
},
+ properties: {
+ currentUsage: {
+ check: "osparc.desktop.credits.CurrentUsage",
+ init: null,
+ nullable: true,
+ apply: "__applyCurrentUsage"
+ }
+ },
+
members: {
+ __applyCurrentUsage: function(currentUsage) {
+ if (currentUsage) {
+ currentUsage.addListener("changeUsedCredits", () => {
+ this.__updateCredits();
+ this.__animate();
+ });
+ }
+ },
+
__contextWalletChanged: function() {
const store = osparc.store.Store.getInstance();
const wallet = store.getContextWallet();
@@ -70,14 +98,37 @@ qx.Class.define("osparc.navigation.CreditsMenuButton", {
}
},
+ __animate: function() {
+ const label = this.getChildControl("label");
+ osparc.utils.Utils.animateUsage(label.getContentElement().getDomElement());
+ },
+
__updateCredits: function() {
const store = osparc.store.Store.getInstance();
const wallet = store.getContextWallet();
if (wallet) {
- const credits = wallet.getCreditsAvailable();
+ let text = "-";
+ const currentUsage = this.getCurrentUsage();
+ let used = null;
+ if (currentUsage) {
+ used = currentUsage.getUsedCredits();
+ }
+ const creditsLeft = wallet.getCreditsAvailable();
+ if (creditsLeft !== null) {
+ text = "CREDITS
";
+ let creditsLeftText = osparc.desktop.credits.Utils.creditsToFixed(creditsLeft);
+ if (used !== null) {
+ creditsLeftText += " / " + osparc.desktop.credits.Utils.creditsToFixed(used);
+ }
+ text += `${creditsLeftText}`;
+ this.set({
+ minWidth: used ? 90 : null,
+ width: used ? 90 : null
+ });
+ }
this.set({
- label: credits === null ? "-" : osparc.desktop.credits.Utils.creditsToFixed(credits) + this.tr(" credits"),
- textColor: osparc.desktop.credits.Utils.creditsToColor(credits, "text")
+ label: text,
+ textColor: osparc.desktop.credits.Utils.creditsToColor(creditsLeft, "text")
});
}
},
diff --git a/services/static-webserver/client/source/class/osparc/navigation/NavigationBar.js b/services/static-webserver/client/source/class/osparc/navigation/NavigationBar.js
index 3e12a54727d..3c62ba3bca3 100644
--- a/services/static-webserver/client/source/class/osparc/navigation/NavigationBar.js
+++ b/services/static-webserver/client/source/class/osparc/navigation/NavigationBar.js
@@ -132,7 +132,6 @@ qx.Class.define("osparc.navigation.NavigationBar", {
this.getChildControl("expiration-icon");
this.getChildControl("help");
if (osparc.desktop.credits.Utils.areWalletsEnabled()) {
- this.getChildControl("current-usage-indicator");
this.getChildControl("credits-menu-button");
}
this.getChildControl("log-in-button");
@@ -227,21 +226,15 @@ qx.Class.define("osparc.navigation.NavigationBar", {
this.getChildControl("center-items").add(control);
break;
}
- case "current-usage-indicator": {
+ case "credits-menu-button": {
const currentUsage = new osparc.desktop.credits.CurrentUsage();
- control = new osparc.desktop.credits.CurrentUsageIndicator(currentUsage).set({
- allowGrowY: false,
- alignY: "middle"
- });
- this.getChildControl("right-items").add(control);
- break;
- }
- case "credits-menu-button":
control = new osparc.navigation.CreditsMenuButton().set({
+ currentUsage,
maxHeight: this.self().HEIGHT
});
this.getChildControl("right-items").add(control);
break;
+ }
case "wallets-viewer":
control = new osparc.desktop.credits.WalletsMiniViewer().set({
maxHeight: this.self().HEIGHT
diff --git a/services/static-webserver/client/source/class/osparc/navigation/UserMenuButton.js b/services/static-webserver/client/source/class/osparc/navigation/UserMenuButton.js
index 0553a29f42d..27060fc80b4 100644
--- a/services/static-webserver/client/source/class/osparc/navigation/UserMenuButton.js
+++ b/services/static-webserver/client/source/class/osparc/navigation/UserMenuButton.js
@@ -48,6 +48,9 @@ qx.Class.define("osparc.navigation.UserMenuButton", {
this.__bindWalletToHalo();
store.addListener("changeContextWallet", () => this.__bindWalletToHalo());
+ const preferencesSettings = osparc.Preferences.getInstance();
+ preferencesSettings.addListener("changeCreditsWarningThreshold", () => this.__updateHalloCredits());
+
const userEmail = authData.getEmail() || "bizzy@itis.ethz.ch";
const icon = this.getChildControl("icon");
authData.bind("role", this, "icon", {
@@ -78,23 +81,28 @@ qx.Class.define("osparc.navigation.UserMenuButton", {
const store = osparc.store.Store.getInstance();
const contextWallet = store.getContextWallet();
if (contextWallet) {
- this.__updateHalloCredits(contextWallet.getCreditsAvailable());
- contextWallet.addListener("changeCreditsAvailable", e => this.__updateHalloCredits(e.getData()));
+ this.__updateHalloCredits();
+ contextWallet.addListener("changeCreditsAvailable", () => this.__updateHalloCredits());
}
},
- __updateHalloCredits: function(credits) {
- if (credits !== null) {
- const progress = osparc.desktop.credits.Utils.normalizeCredits(credits);
- const creditsColor = osparc.desktop.credits.Utils.creditsToColor(credits, "strong-main");
- const color1 = qx.theme.manager.Color.getInstance().resolve(creditsColor);
- const textColor = qx.theme.manager.Color.getInstance().resolve("text");
- const arr = qx.util.ColorUtil.stringToRgb(textColor);
- arr[3] = 0.5;
- const color2 = qx.util.ColorUtil.rgbToRgbString(arr);
- this.getContentElement().setStyles({
- "background": `radial-gradient(closest-side, white 79%, transparent 80% 100%), conic-gradient(${color1} ${progress}%, ${color2} 0)`
- });
+ __updateHalloCredits: function() {
+ const store = osparc.store.Store.getInstance();
+ const contextWallet = store.getContextWallet();
+ if (contextWallet) {
+ const credits = contextWallet.getCreditsAvailable();
+ if (credits !== null) {
+ const progress = osparc.desktop.credits.Utils.normalizeCredits(credits);
+ const creditsColor = osparc.desktop.credits.Utils.creditsToColor(credits, "strong-main");
+ const color1 = qx.theme.manager.Color.getInstance().resolve(creditsColor);
+ const textColor = qx.theme.manager.Color.getInstance().resolve("text");
+ const arr = qx.util.ColorUtil.stringToRgb(textColor);
+ arr[3] = 0.5;
+ const color2 = qx.util.ColorUtil.rgbToRgbString(arr);
+ this.getContentElement().setStyles({
+ "background": `radial-gradient(closest-side, white 79%, transparent 80% 100%), conic-gradient(${color1} ${progress}%, ${color2} 0)`
+ });
+ }
}
},
diff --git a/services/static-webserver/client/source/class/osparc/notification/RibbonNotifications.js b/services/static-webserver/client/source/class/osparc/notification/RibbonNotifications.js
index 2565793f42d..1cbe3b5f7ea 100644
--- a/services/static-webserver/client/source/class/osparc/notification/RibbonNotifications.js
+++ b/services/static-webserver/client/source/class/osparc/notification/RibbonNotifications.js
@@ -61,7 +61,7 @@ qx.Class.define("osparc.notification.RibbonNotifications", {
*/
__createNotificationUI: function(notification) {
const notificationLayout = new qx.ui.container.Composite(new qx.ui.layout.HBox(5, "center")).set({
- backgroundColor: notification.getType() === "announcement" ? "strong-main" : "warning-yellow-s4l",
+ backgroundColor: notification.getType() === "announcement" ? "strong-main" : "warning-yellow",
allowGrowX: true
});
diff --git a/services/static-webserver/client/source/class/osparc/theme/mixin/Color.js b/services/static-webserver/client/source/class/osparc/theme/mixin/Color.js
index f9aead7fab1..7605fa71f95 100644
--- a/services/static-webserver/client/source/class/osparc/theme/mixin/Color.js
+++ b/services/static-webserver/client/source/class/osparc/theme/mixin/Color.js
@@ -4,8 +4,7 @@ qx.Theme.define("osparc.theme.mixin.Color", {
"activitytree-background-memory": "#358475",
"ready-green": "#58A6FF", // It is not really green because of reasons
- "warning-yellow": "#FFFF00",
- "warning-yellow-s4l": "#f8db1f",
+ "warning-yellow": "#f8db1f",
"busy-orange": "#FFA500",
"failed-red": "#FF2D2D",
"danger-red": osparc.theme.colorProvider.ColorProvider.getColor("color.scales.danger", 40),
diff --git a/services/static-webserver/client/source/class/osparc/ui/message/Loading.js b/services/static-webserver/client/source/class/osparc/ui/message/Loading.js
index 52cb6104c7f..7e3cf10c688 100644
--- a/services/static-webserver/client/source/class/osparc/ui/message/Loading.js
+++ b/services/static-webserver/client/source/class/osparc/ui/message/Loading.js
@@ -131,7 +131,7 @@ qx.Class.define("osparc.ui.message.Loading", {
padding: 15,
gap: 20,
icon: "@FontAwesome5Solid/exclamation-triangle/20",
- backgroundColor: "warning-yellow-s4l",
+ backgroundColor: "warning-yellow",
textColor: "black",
alignX: "center"
});
diff --git a/services/static-webserver/client/source/class/osparc/utils/Utils.js b/services/static-webserver/client/source/class/osparc/utils/Utils.js
index 711c8ebeea1..c1f09370581 100644
--- a/services/static-webserver/client/source/class/osparc/utils/Utils.js
+++ b/services/static-webserver/client/source/class/osparc/utils/Utils.js
@@ -106,6 +106,25 @@ qx.Class.define("osparc.utils.Utils", {
return defaultFont;
},
+ animateUsage: function(domElement) {
+ const desc = {
+ duration: 500,
+ timing: "ease-out",
+ keyFrames: {
+ 0: {
+ "opacity": 1
+ },
+ 70: {
+ "opacity": 0.8
+ },
+ 100: {
+ "opacity": 1
+ }
+ }
+ };
+ qx.bom.element.Animation.animate(domElement, desc);
+ },
+
prettifyMenu: function(menu) {
menu.set({
font: "text-14",