chore: alert the user while deleting users who are team owners
SH Admin user management.
This commit is contained in:
@@ -168,7 +168,9 @@
|
|||||||
"remove_admin_from_users_failure": "Failed to remove admin status from selected users!!",
|
"remove_admin_from_users_failure": "Failed to remove admin status from selected users!!",
|
||||||
"remove_admin_from_users_success": "Admin status removed from selected users!!",
|
"remove_admin_from_users_success": "Admin status removed from selected users!!",
|
||||||
"remove_admin_to_delete_user": "Remove admin privilege to delete the user!!",
|
"remove_admin_to_delete_user": "Remove admin privilege to delete the user!!",
|
||||||
|
"remove_owner_to_delete_user": "Remove team ownership status to delete the user!!",
|
||||||
"remove_admin_for_deletion": "Remove admin status before attempting deletion!!",
|
"remove_admin_for_deletion": "Remove admin status before attempting deletion!!",
|
||||||
|
"remove_owner_for_deletion": "One or more users are team owners. Update ownership before deletion!!",
|
||||||
"remove_invitee_failure": "Removal of invitee failed!!",
|
"remove_invitee_failure": "Removal of invitee failed!!",
|
||||||
"remove_invitee_success": "Removal of invitee is successfull!!",
|
"remove_invitee_success": "Removal of invitee is successfull!!",
|
||||||
"remove_member_failure": "Member couldn't be removed!!",
|
"remove_member_failure": "Member couldn't be removed!!",
|
||||||
|
|||||||
@@ -17,3 +17,6 @@ export const ADMIN_CANNOT_BE_DELETED =
|
|||||||
// When trying to invite a user that is already invited
|
// When trying to invite a user that is already invited
|
||||||
export const USER_ALREADY_INVITED =
|
export const USER_ALREADY_INVITED =
|
||||||
'[GraphQL] admin/user_already_invited' as const;
|
'[GraphQL] admin/user_already_invited' as const;
|
||||||
|
|
||||||
|
// When attempting to delete a user who is an owner of a team
|
||||||
|
export const USER_IS_OWNER = 'user/is_owner' as const;
|
||||||
|
|||||||
@@ -73,7 +73,7 @@ import {
|
|||||||
RemoveUsersByAdminDocument,
|
RemoveUsersByAdminDocument,
|
||||||
UserInfoDocument,
|
UserInfoDocument,
|
||||||
} from '~/helpers/backend/graphql';
|
} from '~/helpers/backend/graphql';
|
||||||
import { ADMIN_CANNOT_BE_DELETED } from '~/helpers/errors';
|
import { ADMIN_CANNOT_BE_DELETED, USER_IS_OWNER } from '~/helpers/errors';
|
||||||
|
|
||||||
const t = useI18n();
|
const t = useI18n();
|
||||||
const toast = useToast();
|
const toast = useToast();
|
||||||
@@ -210,13 +210,35 @@ const deleteUserMutation = async (id: string | null) => {
|
|||||||
} else {
|
} else {
|
||||||
const deletedUsers = result.data?.removeUsersByAdmin || [];
|
const deletedUsers = result.data?.removeUsersByAdmin || [];
|
||||||
|
|
||||||
const isAdminError = deletedUsers.some(
|
const uniqueErrorMessages = new Set(
|
||||||
(user) => user.errorMessage === ADMIN_CANNOT_BE_DELETED
|
deletedUsers.map(({ errorMessage }) => errorMessage).filter(Boolean)
|
||||||
);
|
) as Set<string>;
|
||||||
|
|
||||||
isAdminError
|
if (uniqueErrorMessages.size > 0) {
|
||||||
? toast.error(t('state.delete_user_failed_only_one_admin'))
|
const errMsgMap = {
|
||||||
: toast.success(t('state.delete_user_success'));
|
[ADMIN_CANNOT_BE_DELETED]: t('state.remove_admin_to_delete_user'),
|
||||||
|
[USER_IS_OWNER]: t('state.remove_owner_to_delete_user'),
|
||||||
|
};
|
||||||
|
|
||||||
|
const errMsgMapKeys = Object.keys(errMsgMap);
|
||||||
|
|
||||||
|
uniqueErrorMessages.forEach((errorMessage) => {
|
||||||
|
if (errMsgMapKeys.includes(errorMessage)) {
|
||||||
|
toast.error(errMsgMap[errorMessage as keyof typeof errMsgMap]);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Fallback for the case where the error message is not in the compiled list
|
||||||
|
if (
|
||||||
|
Array.from(uniqueErrorMessages).some(
|
||||||
|
(key) => !((key as string) in errMsgMap)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
toast.error(t('state.delete_user_failure'));
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
toast.success(t('state.delete_user_success'));
|
||||||
|
}
|
||||||
}
|
}
|
||||||
confirmDeletion.value = false;
|
confirmDeletion.value = false;
|
||||||
deleteUserUID.value = null;
|
deleteUserUID.value = null;
|
||||||
|
|||||||
@@ -210,7 +210,7 @@
|
|||||||
<HoppSmartConfirmModal
|
<HoppSmartConfirmModal
|
||||||
:show="confirmUsersToAdmin"
|
:show="confirmUsersToAdmin"
|
||||||
:title="
|
:title="
|
||||||
AreMultipleUsersSelected
|
areMultipleUsersSelected
|
||||||
? t('state.confirm_users_to_admin')
|
? t('state.confirm_users_to_admin')
|
||||||
: t('state.confirm_user_to_admin')
|
: t('state.confirm_user_to_admin')
|
||||||
"
|
"
|
||||||
@@ -220,7 +220,7 @@
|
|||||||
<HoppSmartConfirmModal
|
<HoppSmartConfirmModal
|
||||||
:show="confirmAdminsToUsers"
|
:show="confirmAdminsToUsers"
|
||||||
:title="
|
:title="
|
||||||
AreMultipleUsersSelectedToAdmin
|
areMultipleUsersSelectedToAdmin
|
||||||
? t('state.confirm_admins_to_users')
|
? t('state.confirm_admins_to_users')
|
||||||
: t('state.confirm_admin_to_user')
|
: t('state.confirm_admin_to_user')
|
||||||
"
|
"
|
||||||
@@ -230,7 +230,7 @@
|
|||||||
<HoppSmartConfirmModal
|
<HoppSmartConfirmModal
|
||||||
:show="confirmUsersDeletion"
|
:show="confirmUsersDeletion"
|
||||||
:title="
|
:title="
|
||||||
AreMultipleUsersSelectedForDeletion
|
areMultipleUsersSelectedForDeletion
|
||||||
? t('state.confirm_users_deletion')
|
? t('state.confirm_users_deletion')
|
||||||
: t('state.confirm_user_deletion')
|
: t('state.confirm_user_deletion')
|
||||||
"
|
"
|
||||||
@@ -260,8 +260,8 @@ import {
|
|||||||
} from '~/helpers/backend/graphql';
|
} from '~/helpers/backend/graphql';
|
||||||
import {
|
import {
|
||||||
ADMIN_CANNOT_BE_DELETED,
|
ADMIN_CANNOT_BE_DELETED,
|
||||||
DELETE_USER_FAILED_ONLY_ONE_ADMIN,
|
|
||||||
USER_ALREADY_INVITED,
|
USER_ALREADY_INVITED,
|
||||||
|
USER_IS_OWNER,
|
||||||
} from '~/helpers/errors';
|
} from '~/helpers/errors';
|
||||||
import IconCheck from '~icons/lucide/check';
|
import IconCheck from '~icons/lucide/check';
|
||||||
import IconLeft from '~icons/lucide/chevron-left';
|
import IconLeft from '~icons/lucide/chevron-left';
|
||||||
@@ -462,7 +462,7 @@ const confirmUsersToAdmin = ref(false);
|
|||||||
const usersToAdminUID = ref<string | null>(null);
|
const usersToAdminUID = ref<string | null>(null);
|
||||||
const usersToAdmin = useMutation(MakeUsersAdminDocument);
|
const usersToAdmin = useMutation(MakeUsersAdminDocument);
|
||||||
|
|
||||||
const AreMultipleUsersSelected = computed(() => selectedRows.value.length > 1);
|
const areMultipleUsersSelected = computed(() => selectedRows.value.length > 1);
|
||||||
|
|
||||||
const confirmUserToAdmin = (id: string | null) => {
|
const confirmUserToAdmin = (id: string | null) => {
|
||||||
confirmUsersToAdmin.value = true;
|
confirmUsersToAdmin.value = true;
|
||||||
@@ -482,11 +482,15 @@ const makeUsersToAdmin = async (id: string | null) => {
|
|||||||
|
|
||||||
if (result.error) {
|
if (result.error) {
|
||||||
toast.error(
|
toast.error(
|
||||||
id ? t('state.admin_failure') : t('state.users_to_admin_failure')
|
areMultipleUsersSelected.value
|
||||||
|
? t('state.users_to_admin_failure')
|
||||||
|
: t('state.admin_failure')
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
toast.success(
|
toast.success(
|
||||||
id ? t('state.admin_success') : t('state.users_to_admin_success')
|
areMultipleUsersSelected.value
|
||||||
|
? t('state.users_to_admin_success')
|
||||||
|
: t('state.admin_success')
|
||||||
);
|
);
|
||||||
usersList.value = usersList.value.map((user) => ({
|
usersList.value = usersList.value.map((user) => ({
|
||||||
...user,
|
...user,
|
||||||
@@ -514,7 +518,7 @@ const resetConfirmAdminToUser = () => {
|
|||||||
adminsToUserUID.value = null;
|
adminsToUserUID.value = null;
|
||||||
};
|
};
|
||||||
|
|
||||||
const AreMultipleUsersSelectedToAdmin = computed(
|
const areMultipleUsersSelectedToAdmin = computed(
|
||||||
() => selectedRows.value.length > 1
|
() => selectedRows.value.length > 1
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -525,15 +529,15 @@ const makeAdminsToUsers = async (id: string | null) => {
|
|||||||
const result = await adminsToUser.executeMutation(variables);
|
const result = await adminsToUser.executeMutation(variables);
|
||||||
if (result.error) {
|
if (result.error) {
|
||||||
toast.error(
|
toast.error(
|
||||||
id
|
areMultipleUsersSelected.value
|
||||||
? t('state.remove_admin_failure')
|
? t('state.remove_admin_from_users_failure')
|
||||||
: t('state.remove_admin_from_users_failure')
|
: t('state.remove_admin_failure')
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
toast.success(
|
toast.success(
|
||||||
id
|
areMultipleUsersSelected.value
|
||||||
? t('state.remove_admin_success')
|
? t('state.remove_admin_from_users_success')
|
||||||
: t('state.remove_admin_from_users_success')
|
: t('state.remove_admin_success')
|
||||||
);
|
);
|
||||||
usersList.value = usersList.value.map((user) => ({
|
usersList.value = usersList.value.map((user) => ({
|
||||||
...user,
|
...user,
|
||||||
@@ -562,7 +566,7 @@ const resetConfirmUserDeletion = () => {
|
|||||||
deleteUserUID.value = null;
|
deleteUserUID.value = null;
|
||||||
};
|
};
|
||||||
|
|
||||||
const AreMultipleUsersSelectedForDeletion = computed(
|
const areMultipleUsersSelectedForDeletion = computed(
|
||||||
() => selectedRows.value.length > 1
|
() => selectedRows.value.length > 1
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -572,12 +576,9 @@ const deleteUsers = async (id: string | null) => {
|
|||||||
const result = await usersDeletion.executeMutation(variables);
|
const result = await usersDeletion.executeMutation(variables);
|
||||||
|
|
||||||
if (result.error) {
|
if (result.error) {
|
||||||
const errorMessage =
|
const errorMessage = areMultipleUsersSelected.value
|
||||||
result.error.message === DELETE_USER_FAILED_ONLY_ONE_ADMIN
|
? t('state.delete_users_failure')
|
||||||
? t('state.delete_user_failed_only_one_admin')
|
: t('state.delete_user_failure');
|
||||||
: id
|
|
||||||
? t('state.delete_user_failure')
|
|
||||||
: t('state.delete_users_failure');
|
|
||||||
toast.error(errorMessage);
|
toast.error(errorMessage);
|
||||||
} else {
|
} else {
|
||||||
const deletedUsers = result.data?.removeUsersByAdmin || [];
|
const deletedUsers = result.data?.removeUsersByAdmin || [];
|
||||||
@@ -585,32 +586,63 @@ const deleteUsers = async (id: string | null) => {
|
|||||||
.filter((user) => user.isDeleted)
|
.filter((user) => user.isDeleted)
|
||||||
.map((user) => user.userUID);
|
.map((user) => user.userUID);
|
||||||
|
|
||||||
const isAdminError = deletedUsers.some(
|
const uniqueErrorMessages = new Set(
|
||||||
(user) => user.errorMessage === ADMIN_CANNOT_BE_DELETED
|
deletedUsers.map(({ errorMessage }) => errorMessage).filter(Boolean)
|
||||||
);
|
) as Set<string>;
|
||||||
|
|
||||||
|
if (uniqueErrorMessages.size > 0) {
|
||||||
|
const errMsgMap = {
|
||||||
|
[ADMIN_CANNOT_BE_DELETED]: t('state.remove_admin_for_deletion'),
|
||||||
|
[USER_IS_OWNER]: t('state.remove_owner_for_deletion'),
|
||||||
|
};
|
||||||
|
|
||||||
|
const errMsgMapKeys = Object.keys(errMsgMap);
|
||||||
|
|
||||||
|
// Show toast messages with the count of users deleted only if multiple users are selected
|
||||||
|
if (areMultipleUsersSelected.value) {
|
||||||
|
toast.success(
|
||||||
|
t('state.delete_some_users_success', { count: deletedIDs.length })
|
||||||
|
);
|
||||||
|
toast.error(
|
||||||
|
t('state.delete_some_users_failure', {
|
||||||
|
count: deletedUsers.length - deletedIDs.length,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
uniqueErrorMessages.forEach((errorMessage) => {
|
||||||
|
if (errMsgMapKeys.includes(errorMessage)) {
|
||||||
|
toastTimeout = setTimeout(
|
||||||
|
() => {
|
||||||
|
toast.error(errMsgMap[errorMessage as keyof typeof errMsgMap]);
|
||||||
|
},
|
||||||
|
areMultipleUsersSelected.value ? 2000 : 0
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Fallback for the case where the error message is not in the compiled list
|
||||||
|
if (
|
||||||
|
Array.from(uniqueErrorMessages).some(
|
||||||
|
(key) => !((key as string) in errMsgMap)
|
||||||
|
)
|
||||||
|
) {
|
||||||
|
areMultipleUsersSelected.value
|
||||||
|
? t('state.delete_users_failure')
|
||||||
|
: t('state.delete_user_failure');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
toast.success(
|
||||||
|
areMultipleUsersSelected.value
|
||||||
|
? t('state.delete_users_success')
|
||||||
|
: t('state.delete_user_success')
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
usersList.value = usersList.value.filter(
|
usersList.value = usersList.value.filter(
|
||||||
(user) => !deletedIDs.includes(user.uid)
|
(user) => !deletedIDs.includes(user.uid)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (isAdminError) {
|
|
||||||
toast.success(
|
|
||||||
t('state.delete_some_users_success', { count: deletedIDs.length })
|
|
||||||
);
|
|
||||||
toast.error(
|
|
||||||
t('state.delete_some_users_failure', {
|
|
||||||
count: deletedUsers.length - deletedIDs.length,
|
|
||||||
})
|
|
||||||
);
|
|
||||||
toastTimeout = setTimeout(() => {
|
|
||||||
toast.error(t('state.remove_admin_for_deletion'));
|
|
||||||
}, 2000);
|
|
||||||
} else {
|
|
||||||
toast.success(
|
|
||||||
id ? t('state.delete_user_success') : t('state.delete_users_success')
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
selectedRows.value.splice(0, selectedRows.value.length);
|
selectedRows.value.splice(0, selectedRows.value.length);
|
||||||
}
|
}
|
||||||
confirmUsersDeletion.value = false;
|
confirmUsersDeletion.value = false;
|
||||||
|
|||||||
Reference in New Issue
Block a user