import { FontAwesomeIcon } from '@fortawesome/react-fontawesome'; import { action, computed, observable } from 'mobx'; import { observer } from 'mobx-react'; import * as React from 'react'; import Select from 'react-select'; import * as RequestPromise from 'request-promise'; import { Doc, DocListCast, DocListCastAsync, Opt } from '../../fields/Doc'; import { StrCast, Cast } from '../../fields/Types'; import { Utils } from '../../Utils'; import { MainViewModal } from '../views/MainViewModal'; import { TaskCompletionBox } from '../views/nodes/TaskCompletedBox'; import './GroupManager.scss'; import { GroupMemberView } from './GroupMemberView'; import { SharingManager, User } from './SharingManager'; import { listSpec } from '../../fields/Schema'; import { DateField } from '../../fields/DateField'; /** * Interface for options for the react-select component */ export interface UserOptions { label: string; value: string; } @observer export class GroupManager extends React.Component<{}> { static Instance: GroupManager; @observable isOpen: boolean = false; // whether the GroupManager is to be displayed or not. @observable private users: string[] = []; // list of users populated from the database. @observable private selectedUsers: UserOptions[] | null = null; // list of users selected in the "Select users" dropdown. @observable currentGroup: Opt; // the currently selected group. @observable private createGroupModalOpen: boolean = false; private inputRef: React.RefObject = React.createRef(); // the ref for the input box. private createGroupButtonRef: React.RefObject = React.createRef(); // the ref for the group creation button @observable private buttonColour: '#979797' | 'black' = '#979797'; @observable private groupSort: 'ascending' | 'descending' | 'none' = 'none'; constructor(props: Readonly<{}>) { super(props); GroupManager.Instance = this; } componentDidMount() { this.populateUsers(); } /** * Fetches the list of users stored on the database. */ populateUsers = async () => { const userList = await RequestPromise.get(Utils.prepend('/getUsers')); const raw = JSON.parse(userList) as User[]; raw.map(action(user => !this.users.some(umail => umail === user.email) && this.users.push(user.email))); }; /** * @returns the options to be rendered in the dropdown menu to add users and create a group. */ @computed get options() { return this.users.map(user => ({ label: user, value: user })); } /** * Makes the GroupManager visible. */ @action open = () => { // SelectionManager.DeselectAll(); this.isOpen = true; this.populateUsers(); }; /** * Hides the GroupManager. */ @action close = () => { this.isOpen = false; this.currentGroup = undefined; this.selectedUsers = null; // this.users = []; this.createGroupModalOpen = false; TaskCompletionBox.taskCompleted = false; }; /** * @returns the database of groups. */ @computed get GroupManagerDoc(): Doc | undefined { return Doc.UserDoc().globalGroupDatabase as Doc; } /** * @returns a list of all group documents. */ @computed get allGroups(): Doc[] { return DocListCast(this.GroupManagerDoc?.data); } /** * @returns the members of the admin group. */ @computed get adminGroupMembers(): string[] { return this.getGroup('Admin') ? JSON.parse(StrCast(this.getGroup('Admin')!.members)) : ''; } /** * @returns a group document based on the group name. * @param groupName */ getGroup(groupName: string): Doc | undefined { return this.allGroups.find(group => group.title === groupName); } /** * Returns an array of the list of members of a given group. */ getGroupMembers(group: string | Doc): string[] { if (group instanceof Doc) return JSON.parse(StrCast(group.members)) as string[]; return JSON.parse(StrCast(this.getGroup(group)!.members)) as string[]; } /** * @returns a boolean indicating whether the current user has access to edit group documents. * @param groupDoc */ hasEditAccess(groupDoc: Doc): boolean { if (!groupDoc) return false; const accessList: string[] = JSON.parse(StrCast(groupDoc.owners)); return accessList.includes(Doc.CurrentUserEmail) || this.adminGroupMembers?.includes(Doc.CurrentUserEmail); } /** * Helper method that sets up the group document. * @param groupName * @param memberEmails */ createGroupDoc(groupName: string, memberEmails: string[] = []) { const name = groupName.toLowerCase() === 'admin' ? 'Admin' : groupName; const groupDoc = new Doc('GROUP:' + name, true); groupDoc.title = name; groupDoc.owners = JSON.stringify([Doc.CurrentUserEmail]); groupDoc.members = JSON.stringify(memberEmails); this.addGroup(groupDoc); } /** * Helper method that adds a group document to the database of group documents and @returns whether it was successfully added or not. * @param groupDoc */ addGroup(groupDoc: Doc): boolean { if (this.GroupManagerDoc) { Doc.AddDocToList(this.GroupManagerDoc, 'data', groupDoc); this.GroupManagerDoc['data-lastModified'] = new DateField(); return true; } return false; } /** * Deletes a group from the database of group documents and @returns whether the group was deleted or not. * @param group */ @action deleteGroup(group: Doc): boolean { if (group) { if (this.GroupManagerDoc && this.hasEditAccess(group)) { Doc.RemoveDocFromList(this.GroupManagerDoc, 'data', group); SharingManager.Instance.removeGroup(group); const members = JSON.parse(StrCast(group.members)); if (members.includes(Doc.CurrentUserEmail)) { const index = DocListCast(this.GroupManagerDoc.data).findIndex(grp => grp === group); index !== -1 && Cast(this.GroupManagerDoc.data, listSpec(Doc), [])?.splice(index, 1); } this.GroupManagerDoc['data-lastModified'] = new DateField(); if (group === this.currentGroup) { this.currentGroup = undefined; } return true; } } return false; } /** * Adds a member to a group. * @param groupDoc * @param email */ addMemberToGroup(groupDoc: Doc, email: string) { if (this.hasEditAccess(groupDoc)) { const memberList = JSON.parse(StrCast(groupDoc.members)); !memberList.includes(email) && memberList.push(email); groupDoc.members = JSON.stringify(memberList); SharingManager.Instance.shareWithAddedMember(groupDoc, email); this.GroupManagerDoc && (this.GroupManagerDoc['data-lastModified'] = new DateField()); } } /** * Removes a member from the group. * @param groupDoc * @param email */ removeMemberFromGroup(groupDoc: Doc, email: string) { if (this.hasEditAccess(groupDoc)) { const memberList = JSON.parse(StrCast(groupDoc.members)); const index = memberList.indexOf(email); if (index !== -1) { const user = memberList.splice(index, 1)[0]; groupDoc.members = JSON.stringify(memberList); SharingManager.Instance.removeMember(groupDoc, email); this.GroupManagerDoc && (this.GroupManagerDoc['data-lastModified'] = new DateField()); } } } /** * Handles changes in the users selected in the "Select users" dropdown. * @param selectedOptions */ @action handleChange = (selectedOptions: any) => { this.selectedUsers = selectedOptions as UserOptions[]; }; /** * Creates the group when the enter key has been pressed (when in the input). * @param e */ handleKeyDown = (e: React.KeyboardEvent) => { e.key === 'Enter' && this.createGroup(); }; /** * Handles the input of required fields in the setup of a group and resets the relevant variables. */ @action createGroup = () => { const { value } = this.inputRef.current!; if (!value) { alert('Please enter a group name'); return; } if (['admin', 'public', 'override'].includes(value.toLowerCase())) { if (value.toLowerCase() !== 'admin' || (value.toLowerCase() === 'admin' && this.getGroup('Admin'))) { alert(`You cannot override the ${value.charAt(0).toUpperCase() + value.slice(1)} group`); return; } } if (this.getGroup(value)) { alert('Please select a unique group name'); return; } this.createGroupDoc( value, this.selectedUsers?.map(user => user.value) ); this.selectedUsers = null; this.inputRef.current!.value = ''; this.buttonColour = '#979797'; const { left, width, top } = this.createGroupButtonRef.current!.getBoundingClientRect(); TaskCompletionBox.popupX = left - 2 * width; TaskCompletionBox.popupY = top; TaskCompletionBox.textDisplayed = 'Group created!'; TaskCompletionBox.taskCompleted = true; setTimeout( action(() => (TaskCompletionBox.taskCompleted = false)), 2000 ); }; /** * @returns the MainViewModal which allows the user to create groups. */ private get groupCreationModal() { const contents = (

New Group

{ this.createGroupModalOpen = false; TaskCompletionBox.taskCompleted = false; })}>
(this.buttonColour = this.inputRef.current?.value ? 'black' : '#979797'))} />