From 7f9f92f17708f369dc3ffbe5058f5ac7c15d0879 Mon Sep 17 00:00:00 2001
From: Odei Maiz <33152403+odeimaiz@users.noreply.github.com>
Date: Wed, 11 Dec 2024 12:28:36 +0100
Subject: [PATCH] =?UTF-8?q?=F0=9F=8E=A8=20[Frontend]=20Enh:=20users=20are?=
=?UTF-8?q?=20identified=20by=20username=20(#6934)?=
MIME-Version: 1.0
Content-Type: text/plain; charset=UTF-8
Content-Transfer-Encoding: 8bit
---
.../client/source/class/osparc/auth/Data.js | 5 +--
.../source/class/osparc/auth/Manager.js | 4 +-
.../source/class/osparc/dashboard/CardBase.js | 2 +-
.../source/class/osparc/data/model/Group.js | 4 ++
.../source/class/osparc/data/model/User.js | 36 ++++++++++------
.../class/osparc/desktop/MainPageHandler.js | 2 +-
.../desktop/organizations/MembersList.js | 42 +++++++++----------
.../osparc/desktop/wallets/MembersList.js | 12 +++---
.../osparc/filter/CollaboratorToggleButton.js | 9 ++--
.../class/osparc/share/Collaborators.js | 18 +++-----
.../source/class/osparc/store/Groups.js | 22 +++++++---
.../client/source/class/osparc/store/Store.js | 12 ++++--
.../client/source/class/osparc/utils/Utils.js | 5 +++
13 files changed, 101 insertions(+), 72 deletions(-)
diff --git a/services/static-webserver/client/source/class/osparc/auth/Data.js b/services/static-webserver/client/source/class/osparc/auth/Data.js
index 4a55c04633d..306d3032558 100644
--- a/services/static-webserver/client/source/class/osparc/auth/Data.js
+++ b/services/static-webserver/client/source/class/osparc/auth/Data.js
@@ -82,12 +82,9 @@ qx.Class.define("osparc.auth.Data", {
event: "changeUsername",
},
- /**
- * Email of logged in user, otherwise null
- */
email: {
init: null,
- nullable: true,
+ nullable: true, // email of logged in user, otherwise null
check: "String"
},
diff --git a/services/static-webserver/client/source/class/osparc/auth/Manager.js b/services/static-webserver/client/source/class/osparc/auth/Manager.js
index 0cdda5c49b6..ca497e5eabb 100644
--- a/services/static-webserver/client/source/class/osparc/auth/Manager.js
+++ b/services/static-webserver/client/source/class/osparc/auth/Manager.js
@@ -241,8 +241,8 @@ qx.Class.define("osparc.auth.Manager", {
authData.set({
email: profile["login"],
username: profile["userName"],
- firstName: profile["first_name"] || profile["login"],
- lastName: profile["last_name"] || "",
+ firstName: profile["first_name"],
+ lastName: profile["last_name"],
expirationDate: "expirationDate" in profile ? new Date(profile["expirationDate"]) : null
});
},
diff --git a/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js b/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js
index 3bcd200c2ee..0d058644bce 100644
--- a/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js
+++ b/services/static-webserver/client/source/class/osparc/dashboard/CardBase.js
@@ -674,7 +674,7 @@ qx.Class.define("osparc.dashboard.CardBase", {
__showBlockedCardFromStatus: function(lockedStatus) {
const status = lockedStatus["status"];
const owner = lockedStatus["owner"];
- let toolTip = osparc.utils.Utils.firstsUp(owner["first_name"] || this.tr("A user"), owner["last_name"] || "");
+ let toolTip = osparc.utils.Utils.firstsUp(owner["first_name"] || this.tr("A user"), owner["last_name"] || ""); // it will be replaced by "userName"
let image = null;
switch (status) {
case "CLOSING":
diff --git a/services/static-webserver/client/source/class/osparc/data/model/Group.js b/services/static-webserver/client/source/class/osparc/data/model/Group.js
index 3afec8ca34b..e345265ba68 100644
--- a/services/static-webserver/client/source/class/osparc/data/model/Group.js
+++ b/services/static-webserver/client/source/class/osparc/data/model/Group.js
@@ -104,6 +104,10 @@ qx.Class.define("osparc.data.model.Group", {
return Object.values(this.getGroupMembers()).find(user => user.getUserId() === userId);
},
+ getGroupMemberByUsername: function(username) {
+ return Object.values(this.getGroupMembers()).find(user => user.getUsername() === username);
+ },
+
getGroupMemberByLogin: function(userEmail) {
return Object.values(this.getGroupMembers()).find(user => user.getEmail() === userEmail);
},
diff --git a/services/static-webserver/client/source/class/osparc/data/model/User.js b/services/static-webserver/client/source/class/osparc/data/model/User.js
index dc943c3202a..f0c8a5dabb1 100644
--- a/services/static-webserver/client/source/class/osparc/data/model/User.js
+++ b/services/static-webserver/client/source/class/osparc/data/model/User.js
@@ -28,22 +28,27 @@ qx.Class.define("osparc.data.model.User", {
construct: function(userData) {
this.base(arguments);
- let label = userData["login"];
+ let description = "";
if (userData["first_name"]) {
- label = qx.lang.String.firstUp(userData["first_name"]);
+ description = userData["first_name"];
if (userData["last_name"]) {
- label += " " + qx.lang.String.firstUp(userData["last_name"]);
+ description += " " + userData["last_name"];
}
+ description += " - ";
+ }
+ if (userData["login"]) {
+ description += userData["login"];
}
const thumbnail = osparc.utils.Avatar.emailToThumbnail(userData["login"]);
this.set({
- userId: userData["id"],
- groupId: userData["gid"],
- label: label,
- username: userData["username"] || "",
+ userId: parseInt(userData["id"]),
+ groupId: parseInt(userData["gid"]),
+ username: userData["userName"],
firstName: userData["first_name"],
lastName: userData["last_name"],
email: userData["login"],
+ label: userData["userName"],
+ description,
thumbnail,
});
},
@@ -70,24 +75,31 @@ qx.Class.define("osparc.data.model.User", {
event: "changeLabel",
},
- username: {
+ description: {
check: "String",
nullable: true,
init: null,
+ event: "changeDescription",
+ },
+
+ username: {
+ check: "String",
+ nullable: false,
+ init: null,
event: "changeUsername",
},
firstName: {
- init: "",
- nullable: true,
check: "String",
+ nullable: true,
+ init: "",
event: "changeFirstName"
},
lastName: {
- init: "",
- nullable: true,
check: "String",
+ nullable: true,
+ init: "",
event: "changeLastName"
},
diff --git a/services/static-webserver/client/source/class/osparc/desktop/MainPageHandler.js b/services/static-webserver/client/source/class/osparc/desktop/MainPageHandler.js
index 0872a1b627b..6e34be5d88c 100644
--- a/services/static-webserver/client/source/class/osparc/desktop/MainPageHandler.js
+++ b/services/static-webserver/client/source/class/osparc/desktop/MainPageHandler.js
@@ -94,7 +94,7 @@ qx.Class.define("osparc.desktop.MainPageHandler", {
lockedBy = studyData["state"]["locked"]["owner"];
}
if (locked && lockedBy["user_id"] !== osparc.auth.Data.getInstance().getUserId()) {
- const msg = `${studyAlias} ${qx.locale.Manager.tr("is already open by")} ${
+ const msg = `${studyAlias} ${qx.locale.Manager.tr("is already open by")} ${ // it will be replaced "userName"
"first_name" in lockedBy && lockedBy["first_name"] != null ?
lockedBy["first_name"] :
qx.locale.Manager.tr("another user.")
diff --git a/services/static-webserver/client/source/class/osparc/desktop/organizations/MembersList.js b/services/static-webserver/client/source/class/osparc/desktop/organizations/MembersList.js
index 0a3df0faa3b..acd68c25680 100644
--- a/services/static-webserver/client/source/class/osparc/desktop/organizations/MembersList.js
+++ b/services/static-webserver/client/source/class/osparc/desktop/organizations/MembersList.js
@@ -70,8 +70,8 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
if (sorted !== 0) {
return sorted;
}
- if (("email" in a) && ("email" in b)) {
- return a["email"].localeCompare(b["email"]);
+ if (("label" in a) && ("label" in b)) {
+ return a["label"].localeCompare(b["label"]);
}
return 0;
}
@@ -105,22 +105,17 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
alignY: "middle"
}));
- const userEmail = new qx.ui.form.TextField().set({
+ const newMemberUserName = new qx.ui.form.TextField().set({
required: true,
- placeholder: this.tr(" New Member's email")
+ placeholder: this.tr(" New Member's username")
});
- hBox.add(userEmail, {
+ hBox.add(newMemberUserName, {
flex: 1
});
- const validator = new qx.ui.form.validation.Manager();
- validator.add(userEmail, qx.util.Validate.email());
-
const addBtn = new qx.ui.form.Button(this.tr("Add"));
addBtn.addListener("execute", function() {
- if (validator.validate()) {
- this.__addMember(userEmail.getValue());
- }
+ this.__addMember(newMemberUserName.getValue());
}, this);
hBox.add(addBtn);
@@ -154,9 +149,9 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
ctrl.bindProperty("userId", "model", null, item, id);
ctrl.bindProperty("userId", "key", null, item, id);
ctrl.bindProperty("thumbnail", "thumbnail", null, item, id);
- ctrl.bindProperty("name", "title", null, item, id);
+ ctrl.bindProperty("label", "title", null, item, id);
+ ctrl.bindProperty("description", "subtitleMD", null, item, id);
ctrl.bindProperty("accessRights", "accessRights", null, item, id);
- ctrl.bindProperty("email", "subtitleMD", null, item, id);
ctrl.bindProperty("options", "options", null, item, id);
ctrl.bindProperty("showOptions", "showOptions", null, item, id);
},
@@ -217,7 +212,7 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
const canIDelete = organization.getAccessRights()["delete"];
const introText = canIWrite ?
- this.tr("You can add new members and promote or demote existing ones.") :
+ this.tr("You can add new members and promote or demote existing ones.
In order to add new members, type their username or email if this is public.") :
this.tr("You can't add new members to this Organization. Please contact an Administrator or Manager.");
this.__introLabel.setValue(introText);
@@ -225,15 +220,17 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
enabled: canIWrite
});
+ const myGroupId = osparc.auth.Data.getInstance().getGroupId();
const membersList = [];
const groupMembers = organization.getGroupMembers();
Object.values(groupMembers).forEach(groupMember => {
+ const gid = parseInt(groupMember.getGroupId());
const member = {};
- member["userId"] = groupMember.getUserId();
- member["groupId"] = groupMember.getGroupId();
+ member["userId"] = gid === myGroupId ? osparc.auth.Data.getInstance().getUserId() : groupMember.getUserId();
+ member["groupId"] = gid;
member["thumbnail"] = groupMember.getThumbnail();
- member["name"] = groupMember.getLabel();
- member["email"] = groupMember.getEmail();
+ member["label"] = groupMember.getLabel();
+ member["description"] = gid === myGroupId ? osparc.auth.Data.getInstance().getEmail() : groupMember.getDescription();
member["accessRights"] = groupMember.getAccessRights();
let options = [];
if (canIDelete) {
@@ -287,7 +284,6 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
}
// Let me go?
const openStudy = osparc.store.Store.getInstance().getCurrentStudy();
- const myGroupId = osparc.store.Groups.getInstance().getMyGroupId();
if (
openStudy === null &&
canIWrite &&
@@ -303,16 +299,18 @@ qx.Class.define("osparc.desktop.organizations.MembersList", {
membersList.forEach(member => membersModel.append(qx.data.marshal.Json.createModel(member)));
},
- __addMember: async function(orgMemberEmail) {
+ __addMember: async function(newMemberIdentifier) {
if (this.__currentOrg === null) {
return;
}
const orgId = this.__currentOrg.getGroupId();
const groupsStore = osparc.store.Groups.getInstance();
- groupsStore.postMember(orgId, orgMemberEmail)
+ const isEmail = osparc.utils.Utils.isEmail(newMemberIdentifier);
+ const request = isEmail ? groupsStore.addMember(orgId, null, newMemberIdentifier) : groupsStore.addMember(orgId, newMemberIdentifier);
+ request
.then(newMember => {
- const text = orgMemberEmail + this.tr(" successfully added");
+ const text = newMemberIdentifier + this.tr(" successfully added");
osparc.FlashMessenger.getInstance().logAs(text);
this.__reloadOrgMembers();
diff --git a/services/static-webserver/client/source/class/osparc/desktop/wallets/MembersList.js b/services/static-webserver/client/source/class/osparc/desktop/wallets/MembersList.js
index 4f5b1dd796a..dc27a0cfee3 100644
--- a/services/static-webserver/client/source/class/osparc/desktop/wallets/MembersList.js
+++ b/services/static-webserver/client/source/class/osparc/desktop/wallets/MembersList.js
@@ -70,8 +70,8 @@ qx.Class.define("osparc.desktop.wallets.MembersList", {
if (sorted !== 0) {
return sorted;
}
- if (("email" in a) && ("email" in b)) {
- return a["email"].localeCompare(b["email"]);
+ if (("label" in a) && ("label" in b)) {
+ return a["label"].localeCompare(b["label"]);
}
return 0;
}
@@ -170,9 +170,9 @@ qx.Class.define("osparc.desktop.wallets.MembersList", {
ctrl.bindProperty("userId", "key", null, item, id);
ctrl.bindProperty("groupId", "gid", null, item, id);
ctrl.bindProperty("thumbnail", "thumbnail", null, item, id);
- ctrl.bindProperty("name", "title", null, item, id);
+ ctrl.bindProperty("label", "title", null, item, id);
+ ctrl.bindProperty("description", "subtitleMD", null, item, id);
ctrl.bindProperty("accessRights", "accessRights", null, item, id);
- ctrl.bindProperty("email", "subtitleMD", null, item, id);
ctrl.bindProperty("options", "options", null, item, id);
ctrl.bindProperty("showOptions", "showOptions", null, item, id);
},
@@ -222,8 +222,8 @@ qx.Class.define("osparc.desktop.wallets.MembersList", {
collaborator["userId"] = gid === myGroupId ? osparc.auth.Data.getInstance().getUserId() : collab.getUserId();
collaborator["groupId"] = collab.getGroupId();
collaborator["thumbnail"] = collab.getThumbnail();
- collaborator["name"] = collab.getLabel();
- collaborator["email"] = gid === myGroupId ? osparc.auth.Data.getInstance().getEmail() : collab.getEmail();
+ collaborator["label"] = collab.getLabel();
+ collaborator["description"] = gid === myGroupId ? osparc.auth.Data.getInstance().getEmail() : collab.getDescription();
collaborator["accessRights"] = {
read: accessRights["read"],
write: accessRights["write"],
diff --git a/services/static-webserver/client/source/class/osparc/filter/CollaboratorToggleButton.js b/services/static-webserver/client/source/class/osparc/filter/CollaboratorToggleButton.js
index 23c061cebd3..0008c5d7c48 100644
--- a/services/static-webserver/client/source/class/osparc/filter/CollaboratorToggleButton.js
+++ b/services/static-webserver/client/source/class/osparc/filter/CollaboratorToggleButton.js
@@ -24,13 +24,16 @@ qx.Class.define("osparc.filter.CollaboratorToggleButton", {
});
let label = collaborator.getLabel();
- if ("getEmail" in collaborator) {
- // user
+ if ("getEmail" in collaborator && collaborator.getEmail()) {
label += ` (${collaborator.getEmail()})`;
- this.setToolTipText(collaborator.getEmail());
}
this.setLabel(label);
+ if (collaborator.getDescription()) {
+ const ttt = collaborator.getLabel() + "
" + collaborator.getDescription();
+ this.setToolTipText(ttt);
+ }
+
let iconPath = null;
switch (collaborator["collabType"]) {
case 0:
diff --git a/services/static-webserver/client/source/class/osparc/share/Collaborators.js b/services/static-webserver/client/source/class/osparc/share/Collaborators.js
index 426d201d2ab..b012119b025 100644
--- a/services/static-webserver/client/source/class/osparc/share/Collaborators.js
+++ b/services/static-webserver/client/source/class/osparc/share/Collaborators.js
@@ -293,10 +293,8 @@ qx.Class.define("osparc.share.Collaborators", {
ctrl.bindProperty("gid", "key", null, item, id);
ctrl.bindProperty("collabType", "collabType", null, item, id);
ctrl.bindProperty("thumbnail", "thumbnail", null, item, id);
- ctrl.bindProperty("name", "title", null, item, id); // user
- ctrl.bindProperty("label", "title", null, item, id); // organization
- ctrl.bindProperty("email", "subtitleMD", null, item, id); // user
- ctrl.bindProperty("description", "subtitle", null, item, id); // organization
+ ctrl.bindProperty("label", "title", null, item, id);
+ ctrl.bindProperty("description", "subtitleMD", null, item, id);
ctrl.bindProperty("resourceType", "resourceType", null, item, id); // Resource type
ctrl.bindProperty("accessRights", "accessRights", null, item, id);
ctrl.bindProperty("showOptions", "showOptions", null, item, id);
@@ -409,15 +407,11 @@ qx.Class.define("osparc.share.Collaborators", {
const collaborator = {
"gid": collab.getGroupId(),
"thumbnail": collab.getThumbnail(),
+ "label": collab.getLabel(),
+ "description": collab.getDescription(),
};
- if ("getUserId" in collab) {
- // user
- collaborator["name"] = collab.getLabel();
- collaborator["email"] = collab.getEmail();
- } else {
- // org/group
- collaborator["label"] = collab.getLabel();
- collaborator["description"] = collab.getDescription();
+ if (!("getUserId" in collab)) {
+ // orgnanization
if (everyoneGIds.includes(parseInt(gid))) {
collaborator["thumbnail"] = "@FontAwesome5Solid/globe/32";
} else if (!collaborator["thumbnail"]) {
diff --git a/services/static-webserver/client/source/class/osparc/store/Groups.js b/services/static-webserver/client/source/class/osparc/store/Groups.js
index 862f97d06b1..21e2d4b2a29 100644
--- a/services/static-webserver/client/source/class/osparc/store/Groups.js
+++ b/services/static-webserver/client/source/class/osparc/store/Groups.js
@@ -231,6 +231,7 @@ qx.Class.define("osparc.store.Groups", {
},
getOrganization: function(groupId) {
+ groupId = parseInt(groupId);
if (groupId && groupId in this.getOrganizations()) {
return this.getOrganizations()[groupId];
}
@@ -261,6 +262,14 @@ qx.Class.define("osparc.store.Groups", {
return null;
},
+ getGroupMemberByUsername: function(orgId, username) {
+ const org = this.getGroup(orgId);
+ if (org) {
+ return org.getGroupMemberByUsername(username);
+ }
+ return null;
+ },
+
getGroupMemberByLogin: function(orgId, userEmail) {
const org = this.getGroup(orgId);
if (org) {
@@ -330,16 +339,19 @@ qx.Class.define("osparc.store.Groups", {
// CRUD GROUP
// CRUD GROUP MEMBERS
- postMember: function(orgId, newMemberEmail) {
+ addMember: function(orgId, username, email = null) {
const gid = parseInt(orgId);
const params = {
url: {
"gid": gid
},
- data: {
- "email": newMemberEmail
- }
+ data: {},
};
+ if (email) {
+ params.data["email"] = email;
+ } else {
+ params.data["userName"] = username;
+ }
return osparc.data.Resources.fetch("organizationMembers", "post", params)
.then(() => {
// the backend doesn't return the user back,
@@ -347,7 +359,7 @@ qx.Class.define("osparc.store.Groups", {
return this.__fetchGroupMembers(gid);
})
.then(() => {
- const groupMember = this.getGroupMemberByLogin(gid, newMemberEmail);
+ const groupMember = email ? this.getGroupMemberByLogin(gid, email) : this.getGroupMemberByUsername(gid, username);
if (groupMember) {
return groupMember;
}
diff --git a/services/static-webserver/client/source/class/osparc/store/Store.js b/services/static-webserver/client/source/class/osparc/store/Store.js
index e560a643980..6b986a0a34d 100644
--- a/services/static-webserver/client/source/class/osparc/store/Store.js
+++ b/services/static-webserver/client/source/class/osparc/store/Store.js
@@ -55,6 +55,14 @@ qx.Class.define("osparc.store.Store", {
check: "Object",
init: {}
},
+ announcements: {
+ check: "Array",
+ init: []
+ },
+ maintenance: {
+ check: "Object",
+ init: {}
+ },
currentStudy: {
check: "osparc.data.model.Study",
init: null,
@@ -111,10 +119,6 @@ qx.Class.define("osparc.store.Store", {
init: [],
event: "changeIterations"
},
- maintenance: {
- check: "Object",
- init: {}
- },
templates: {
check: "Array",
init: []
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 075db07763e..233198affb8 100644
--- a/services/static-webserver/client/source/class/osparc/utils/Utils.js
+++ b/services/static-webserver/client/source/class/osparc/utils/Utils.js
@@ -107,6 +107,11 @@ qx.Class.define("osparc.utils.Utils", {
return newName;
},
+ isEmail: function(value) {
+ const reg = /^([A-Za-z0-9_\-.+])+@([A-Za-z0-9_\-.])+\.([A-Za-z]{2,})$/;
+ return reg.test(value);
+ },
+
replaceTokens: function(str, key, value) {
// `str` might be a a localized string, get the string first
str = str.toString ? str.toString() : str;