diff --git a/public/index.css b/public/index.css index f326ddc0f1..e75c4b8dc8 100644 --- a/public/index.css +++ b/public/index.css @@ -44,3 +44,12 @@ label.pf-c-form__label { .keycloak-empty-state-card:hover { cursor: pointer; } + +td.pf-c-table__check > input { + vertical-align: baseline; +} + +.pf-c-table.tbody.pf-c-table__check > input { + margin-top: var(--pf-c-table__check--input--MarginTop); + vertical-align: baseline; +} diff --git a/src/user/JoinGroupDialog.tsx b/src/user/JoinGroupDialog.tsx index 6e47fe8962..3f29472676 100644 --- a/src/user/JoinGroupDialog.tsx +++ b/src/user/JoinGroupDialog.tsx @@ -23,7 +23,6 @@ import { useTranslation } from "react-i18next"; import { useFetch, useAdminClient } from "../context/auth/AdminClient"; import { AngleRightIcon, SearchIcon } from "@patternfly/react-icons"; import type GroupRepresentation from "keycloak-admin/lib/defs/groupRepresentation"; -import _ from "lodash"; import { useParams } from "react-router-dom"; export type JoinGroupDialogProps = { @@ -55,6 +54,7 @@ export const JoinGroupDialog = ({ const [groups, setGroups] = useState([]); const [filtered, setFiltered] = useState(); const [filter, setFilter] = useState(""); + const [joinedGroups, setJoinedGroups] = useState([]); const [groupId, setGroupId] = useState(); @@ -71,9 +71,9 @@ export const JoinGroupDialog = ({ const existingUserGroups = await adminClient.users.listGroups({ id, }); - + setJoinedGroups(existingUserGroups); return { - groups: _.differenceBy(allGroups, existingUserGroups, "id"), + groups: allGroups, }; } else return { @@ -95,6 +95,14 @@ export const JoinGroupDialog = ({ [groupId] ); + const isRowDisabled = (row?: GroupRepresentation) => { + return !!joinedGroups.find((group) => group.id === row?.id); + }; + + const hasSubgroups = (group: GroupRepresentation) => { + return group.subGroups!.length !== 0; + }; + return ( - { - setGroupId(value); - }} - aria-label={t("groups")} - isCompact - > + {(filtered || groups).map((group: Group) => ( { - if ((e.target as HTMLInputElement).type !== "checkbox") { - setGroupId(group.id); - } - }} + onClick={ + hasSubgroups(group) + ? (e) => { + if ((e.target as HTMLInputElement).type !== "checkbox") { + setGroupId(group.id); + } + } + : undefined + } > - + { group.checked = (e.target as HTMLInputElement).checked; let newSelectedRows: Group[]; @@ -232,9 +245,13 @@ export const JoinGroupDialog = ({ aria-label={t("groupName")} isPlainButtonAction > - + {hasSubgroups(group) ? ( + + ) : ( + "" + )} diff --git a/src/user/UserGroups.tsx b/src/user/UserGroups.tsx index b7570dd739..a26e22b58d 100644 --- a/src/user/UserGroups.tsx +++ b/src/user/UserGroups.tsx @@ -83,11 +83,14 @@ export const UserGroups = () => { return []; } - const joinedGroups = await adminClient.users.listGroups({ ...params, id }); + const joinedUserGroups = await adminClient.users.listGroups({ + ...params, + id, + }); const allCreatedGroups = await adminClient.groups.find(); - const getAllPaths = joinedGroups.reduce( + const getAllPaths = joinedUserGroups.reduce( (acc: string[], cur) => (cur.path && acc.push(cur.path), acc), [] ); @@ -157,7 +160,7 @@ export const UserGroups = () => { topLevelGroups.forEach((group) => subgroupArray.push(group.subGroups)); - const directMembership = joinedGroups.filter( + const directMembership = joinedUserGroups!.filter( (value) => !topLevelGroups.includes(value) ); @@ -168,11 +171,11 @@ export const UserGroups = () => { index === self.findIndex((t) => t.name === thing.name) ); - if (isDirectMembership) { - return alphabetize(directMembership); + if (!isDirectMembership) { + return alphabetize(filterDupesfromGroups); } - return alphabetize(filterDupesfromGroups); + return alphabetize(directMembership); }; useFetch( @@ -323,6 +326,7 @@ export const UserGroups = () => { id="kc-direct-membership-checkbox" onChange={() => setDirectMembership(!isDirectMembership)} isChecked={isDirectMembership} + className="direct-membership-check" /> {enabled && ( div > input { + height: 20px; + width: 20px; +} + +.data-list-check > input { + height: 20px; + width: 20px; +} + +td.pf-c-table__check > input { + height: 20px; + width: 20px; + vertical-align: text-top; +}