Merge pull request #590 from jenny-s51/rowsandchecks
Users: update checkbox styling and add disabled checkboxes/row functionality to modal
This commit is contained in:
commit
bf3ed710b4
4 changed files with 80 additions and 25 deletions
|
@ -44,3 +44,12 @@ label.pf-c-form__label {
|
||||||
.keycloak-empty-state-card:hover {
|
.keycloak-empty-state-card:hover {
|
||||||
cursor: pointer;
|
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;
|
||||||
|
}
|
||||||
|
|
|
@ -23,7 +23,6 @@ import { useTranslation } from "react-i18next";
|
||||||
import { useFetch, useAdminClient } from "../context/auth/AdminClient";
|
import { useFetch, useAdminClient } from "../context/auth/AdminClient";
|
||||||
import { AngleRightIcon, SearchIcon } from "@patternfly/react-icons";
|
import { AngleRightIcon, SearchIcon } from "@patternfly/react-icons";
|
||||||
import type GroupRepresentation from "keycloak-admin/lib/defs/groupRepresentation";
|
import type GroupRepresentation from "keycloak-admin/lib/defs/groupRepresentation";
|
||||||
import _ from "lodash";
|
|
||||||
import { useParams } from "react-router-dom";
|
import { useParams } from "react-router-dom";
|
||||||
|
|
||||||
export type JoinGroupDialogProps = {
|
export type JoinGroupDialogProps = {
|
||||||
|
@ -55,6 +54,7 @@ export const JoinGroupDialog = ({
|
||||||
const [groups, setGroups] = useState<Group[]>([]);
|
const [groups, setGroups] = useState<Group[]>([]);
|
||||||
const [filtered, setFiltered] = useState<GroupRepresentation[]>();
|
const [filtered, setFiltered] = useState<GroupRepresentation[]>();
|
||||||
const [filter, setFilter] = useState("");
|
const [filter, setFilter] = useState("");
|
||||||
|
const [joinedGroups, setJoinedGroups] = useState<GroupRepresentation[]>([]);
|
||||||
|
|
||||||
const [groupId, setGroupId] = useState<string>();
|
const [groupId, setGroupId] = useState<string>();
|
||||||
|
|
||||||
|
@ -71,9 +71,9 @@ export const JoinGroupDialog = ({
|
||||||
const existingUserGroups = await adminClient.users.listGroups({
|
const existingUserGroups = await adminClient.users.listGroups({
|
||||||
id,
|
id,
|
||||||
});
|
});
|
||||||
|
setJoinedGroups(existingUserGroups);
|
||||||
return {
|
return {
|
||||||
groups: _.differenceBy(allGroups, existingUserGroups, "id"),
|
groups: allGroups,
|
||||||
};
|
};
|
||||||
} else
|
} else
|
||||||
return {
|
return {
|
||||||
|
@ -95,6 +95,14 @@ export const JoinGroupDialog = ({
|
||||||
[groupId]
|
[groupId]
|
||||||
);
|
);
|
||||||
|
|
||||||
|
const isRowDisabled = (row?: GroupRepresentation) => {
|
||||||
|
return !!joinedGroups.find((group) => group.id === row?.id);
|
||||||
|
};
|
||||||
|
|
||||||
|
const hasSubgroups = (group: GroupRepresentation) => {
|
||||||
|
return group.subGroups!.length !== 0;
|
||||||
|
};
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Modal
|
<Modal
|
||||||
variant={ModalVariant.small}
|
variant={ModalVariant.small}
|
||||||
|
@ -181,28 +189,33 @@ export const JoinGroupDialog = ({
|
||||||
</ToolbarItem>
|
</ToolbarItem>
|
||||||
</ToolbarContent>
|
</ToolbarContent>
|
||||||
</Toolbar>
|
</Toolbar>
|
||||||
<DataList
|
<DataList aria-label={t("groups")} isCompact>
|
||||||
onSelectDataListItem={(value) => {
|
|
||||||
setGroupId(value);
|
|
||||||
}}
|
|
||||||
aria-label={t("groups")}
|
|
||||||
isCompact
|
|
||||||
>
|
|
||||||
{(filtered || groups).map((group: Group) => (
|
{(filtered || groups).map((group: Group) => (
|
||||||
<DataListItem
|
<DataListItem
|
||||||
aria-labelledby={group.name}
|
aria-labelledby={group.name}
|
||||||
key={group.id}
|
key={group.id}
|
||||||
id={group.id}
|
id={group.id}
|
||||||
onClick={(e) => {
|
onClick={
|
||||||
if ((e.target as HTMLInputElement).type !== "checkbox") {
|
hasSubgroups(group)
|
||||||
setGroupId(group.id);
|
? (e) => {
|
||||||
}
|
if ((e.target as HTMLInputElement).type !== "checkbox") {
|
||||||
}}
|
setGroupId(group.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
: undefined
|
||||||
|
}
|
||||||
>
|
>
|
||||||
<DataListItemRow data-testid={group.name}>
|
<DataListItemRow
|
||||||
|
className={`join-group-dialog-row-${
|
||||||
|
isRowDisabled(group) ? "m-disabled" : ""
|
||||||
|
}`}
|
||||||
|
data-testid={group.name}
|
||||||
|
>
|
||||||
<DataListCheck
|
<DataListCheck
|
||||||
|
className="join-group-modal-check"
|
||||||
data-testid={`${group.name}-check`}
|
data-testid={`${group.name}-check`}
|
||||||
isChecked={group.checked}
|
isChecked={group.checked}
|
||||||
|
isDisabled={isRowDisabled(group)}
|
||||||
onChange={(checked, e) => {
|
onChange={(checked, e) => {
|
||||||
group.checked = (e.target as HTMLInputElement).checked;
|
group.checked = (e.target as HTMLInputElement).checked;
|
||||||
let newSelectedRows: Group[];
|
let newSelectedRows: Group[];
|
||||||
|
@ -232,9 +245,13 @@ export const JoinGroupDialog = ({
|
||||||
aria-label={t("groupName")}
|
aria-label={t("groupName")}
|
||||||
isPlainButtonAction
|
isPlainButtonAction
|
||||||
>
|
>
|
||||||
<Button isDisabled variant="link">
|
{hasSubgroups(group) ? (
|
||||||
<AngleRightIcon />
|
<Button isDisabled variant="link">
|
||||||
</Button>
|
<AngleRightIcon />
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
""
|
||||||
|
)}
|
||||||
</DataListAction>
|
</DataListAction>
|
||||||
</DataListItemRow>
|
</DataListItemRow>
|
||||||
</DataListItem>
|
</DataListItem>
|
||||||
|
|
|
@ -83,11 +83,14 @@ export const UserGroups = () => {
|
||||||
return [];
|
return [];
|
||||||
}
|
}
|
||||||
|
|
||||||
const joinedGroups = await adminClient.users.listGroups({ ...params, id });
|
const joinedUserGroups = await adminClient.users.listGroups({
|
||||||
|
...params,
|
||||||
|
id,
|
||||||
|
});
|
||||||
|
|
||||||
const allCreatedGroups = await adminClient.groups.find();
|
const allCreatedGroups = await adminClient.groups.find();
|
||||||
|
|
||||||
const getAllPaths = joinedGroups.reduce(
|
const getAllPaths = joinedUserGroups.reduce(
|
||||||
(acc: string[], cur) => (cur.path && acc.push(cur.path), acc),
|
(acc: string[], cur) => (cur.path && acc.push(cur.path), acc),
|
||||||
[]
|
[]
|
||||||
);
|
);
|
||||||
|
@ -157,7 +160,7 @@ export const UserGroups = () => {
|
||||||
|
|
||||||
topLevelGroups.forEach((group) => subgroupArray.push(group.subGroups));
|
topLevelGroups.forEach((group) => subgroupArray.push(group.subGroups));
|
||||||
|
|
||||||
const directMembership = joinedGroups.filter(
|
const directMembership = joinedUserGroups!.filter(
|
||||||
(value) => !topLevelGroups.includes(value)
|
(value) => !topLevelGroups.includes(value)
|
||||||
);
|
);
|
||||||
|
|
||||||
|
@ -168,11 +171,11 @@ export const UserGroups = () => {
|
||||||
index === self.findIndex((t) => t.name === thing.name)
|
index === self.findIndex((t) => t.name === thing.name)
|
||||||
);
|
);
|
||||||
|
|
||||||
if (isDirectMembership) {
|
if (!isDirectMembership) {
|
||||||
return alphabetize(directMembership);
|
return alphabetize(filterDupesfromGroups);
|
||||||
}
|
}
|
||||||
|
|
||||||
return alphabetize(filterDupesfromGroups);
|
return alphabetize(directMembership);
|
||||||
};
|
};
|
||||||
|
|
||||||
useFetch(
|
useFetch(
|
||||||
|
@ -323,6 +326,7 @@ export const UserGroups = () => {
|
||||||
id="kc-direct-membership-checkbox"
|
id="kc-direct-membership-checkbox"
|
||||||
onChange={() => setDirectMembership(!isDirectMembership)}
|
onChange={() => setDirectMembership(!isDirectMembership)}
|
||||||
isChecked={isDirectMembership}
|
isChecked={isDirectMembership}
|
||||||
|
className="direct-membership-check"
|
||||||
/>
|
/>
|
||||||
{enabled && (
|
{enabled && (
|
||||||
<Popover
|
<Popover
|
||||||
|
|
|
@ -20,3 +20,28 @@ button#kc-join-groups-button {
|
||||||
margin-right: var(--pf-global--spacer--md);
|
margin-right: var(--pf-global--spacer--md);
|
||||||
padding: var(--pf-global--spacer--xs);
|
padding: var(--pf-global--spacer--xs);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
.join-group-dialog-row-m-disabled {
|
||||||
|
pointer-events: none;
|
||||||
|
color: var(--pf-global--disabled-color--200);
|
||||||
|
}
|
||||||
|
|
||||||
|
input#kc-direct-membership-checkbox {
|
||||||
|
width: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.join-group-modal-check > 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;
|
||||||
|
}
|
||||||
|
|
Loading…
Reference in a new issue