import React, { PureComponent, ReactElement } from 'react'; import { css, cx } from '@emotion/css'; import { Button, ConfirmButton, Field, HorizontalGroup, Icon, Modal, stylesFactory, Themeable, Tooltip, useStyles2, withTheme, } from '@grafana/ui'; import { GrafanaTheme, GrafanaTheme2 } from '@grafana/data'; import { AccessControlAction, Organization, OrgRole, UserOrg } from 'app/types'; import { OrgPicker, OrgSelectItem } from 'app/core/components/Select/OrgPicker'; import { OrgRolePicker } from './OrgRolePicker'; import { contextSrv } from 'app/core/core'; interface Props { orgs: UserOrg[]; isExternalUser?: boolean; onOrgRemove: (orgId: number) => void; onOrgRoleChange: (orgId: number, newRole: OrgRole) => void; onOrgAdd: (orgId: number, role: OrgRole) => void; } interface State { showAddOrgModal: boolean; } export class UserOrgs extends PureComponent { addToOrgButtonRef = React.createRef(); state = { showAddOrgModal: false, }; showOrgAddModal = () => { this.setState({ showAddOrgModal: true }); }; dismissOrgAddModal = () => { this.setState({ showAddOrgModal: false }, () => { this.addToOrgButtonRef.current?.focus(); }); }; render() { const { orgs, isExternalUser, onOrgRoleChange, onOrgRemove, onOrgAdd } = this.props; const { showAddOrgModal } = this.state; const addToOrgContainerClass = css` margin-top: 0.8rem; `; const canAddToOrg = contextSrv.hasPermission(AccessControlAction.OrgUsersAdd); return ( <>

Organizations

{orgs.map((org, index) => ( ))}
{canAddToOrg && ( )}
); } } const getOrgRowStyles = stylesFactory((theme: GrafanaTheme) => { return { removeButton: css` margin-right: 0.6rem; text-decoration: underline; color: ${theme.palette.blue95}; `, label: css` font-weight: 500; `, disabledTooltip: css` display: flex; `, tooltipItem: css` margin-left: 5px; `, tooltipItemLink: css` color: ${theme.palette.blue95}; `, }; }); interface OrgRowProps extends Themeable { org: UserOrg; isExternalUser?: boolean; onOrgRemove: (orgId: number) => void; onOrgRoleChange: (orgId: number, newRole: OrgRole) => void; } interface OrgRowState { currentRole: OrgRole; isChangingRole: boolean; } class UnThemedOrgRow extends PureComponent { state = { currentRole: this.props.org.role, isChangingRole: false, }; onOrgRemove = () => { const { org } = this.props; this.props.onOrgRemove(org.orgId); }; onChangeRoleClick = () => { const { org } = this.props; this.setState({ isChangingRole: true, currentRole: org.role }); }; onOrgRoleChange = (newRole: OrgRole) => { this.setState({ currentRole: newRole }); }; onOrgRoleSave = () => { this.props.onOrgRoleChange(this.props.org.orgId, this.state.currentRole); }; onCancelClick = () => { this.setState({ isChangingRole: false }); }; render() { const { org, isExternalUser, theme } = this.props; const { currentRole, isChangingRole } = this.state; const styles = getOrgRowStyles(theme); const labelClass = cx('width-16', styles.label); const canChangeRole = contextSrv.hasPermission(AccessControlAction.OrgUsersRoleUpdate); const canRemoveFromOrg = contextSrv.hasPermission(AccessControlAction.OrgUsersRemove); const inputId = `${org.name}-input`; return ( {isChangingRole ? ( ) : ( {org.role} )}
{canChangeRole && ( )}
{canRemoveFromOrg && ( Remove from organization )}
); } } const OrgRow = withTheme(UnThemedOrgRow); const getAddToOrgModalStyles = stylesFactory(() => ({ modal: css` width: 500px; `, buttonRow: css` text-align: center; `, modalContent: css` overflow: visible; `, })); interface AddToOrgModalProps { isOpen: boolean; onOrgAdd(orgId: number, role: string): void; onDismiss?(): void; } interface AddToOrgModalState { selectedOrg: Organization | null; role: OrgRole; } export class AddToOrgModal extends PureComponent { state: AddToOrgModalState = { selectedOrg: null, role: OrgRole.Admin, }; onOrgSelect = (org: OrgSelectItem) => { this.setState({ selectedOrg: org.value! }); }; onOrgRoleChange = (newRole: OrgRole) => { this.setState({ role: newRole, }); }; onAddUserToOrg = () => { const { selectedOrg, role } = this.state; this.props.onOrgAdd(selectedOrg!.id, role); }; onCancel = () => { if (this.props.onDismiss) { this.props.onDismiss(); } }; render() { const { isOpen } = this.props; const { role } = this.state; const styles = getAddToOrgModalStyles(); return ( ); } } interface ChangeOrgButtonProps { isExternalUser?: boolean; onChangeRoleClick: () => void; onCancelClick: () => void; onOrgRoleSave: () => void; } const getChangeOrgButtonTheme = (theme: GrafanaTheme2) => ({ disabledTooltip: css` display: flex; `, tooltipItemLink: css` color: ${theme.v1.palette.blue95}; `, }); export function ChangeOrgButton({ onChangeRoleClick, isExternalUser, onOrgRoleSave, onCancelClick, }: ChangeOrgButtonProps): ReactElement { const styles = useStyles2(getChangeOrgButtonTheme); return (
Change role {isExternalUser && ( This user's role is not editable because it is synchronized from your auth provider. Refer to the  Grafana authentication docs  for details.
} > )} ); }