diff options
Diffstat (limited to 'src/client/util/SharingManager.tsx')
| -rw-r--r-- | src/client/util/SharingManager.tsx | 118 | 
1 files changed, 78 insertions, 40 deletions
| diff --git a/src/client/util/SharingManager.tsx b/src/client/util/SharingManager.tsx index 2aea73528..08dfb5066 100644 --- a/src/client/util/SharingManager.tsx +++ b/src/client/util/SharingManager.tsx @@ -7,7 +7,7 @@ import Select from "react-select";  import * as RequestPromise from "request-promise";  import { AclAddonly, AclAdmin, AclEdit, AclPrivate, AclReadonly, AclSym, DataSym, Doc, DocListCast, DocListCastAsync, Opt } from "../../fields/Doc";  import { List } from "../../fields/List"; -import { Cast, StrCast } from "../../fields/Types"; +import { Cast, NumCast, StrCast } from "../../fields/Types";  import { distributeAcls, GetEffectiveAcl, normalizeEmail, SharingPermissions, TraceMobx } from "../../fields/util";  import { Utils } from "../../Utils";  import { DocServer } from "../DocServer"; @@ -78,7 +78,16 @@ export class SharingManager extends React.Component<{}> {      @observable private showGroupOptions: boolean = false; // // whether to show groups as options when sharing (in the react-select component)      private populating: boolean = false; // whether the list of users is populating or not      @observable private layoutDocAcls: boolean = false; // whether the layout doc or data doc's acls are to be used -    @observable private myDocAcls: boolean = false; +    @observable private myDocAcls: boolean = false; // whether the My Docs checkbox is selected or not + +    // maps acl symbols to SharingPermissions +    private AclMap = new Map<symbol, string>([ +        [AclPrivate, SharingPermissions.None], +        [AclReadonly, SharingPermissions.View], +        [AclAddonly, SharingPermissions.Add], +        [AclEdit, SharingPermissions.Edit], +        [AclAdmin, SharingPermissions.Admin] +    ]);      // private get linkVisible() {      //     return this.sharingDoc ? this.sharingDoc[PublicKey] !== SharingPermissions.None : false; @@ -156,6 +165,33 @@ export class SharingManager extends React.Component<{}> {      }      /** +     * Shares the document with a user. +     */ +    setInternalSharing = (recipient: ValidatedUser, permission: string, targetDoc?: Doc) => { +        const { user, sharingDoc } = recipient; +        const target = targetDoc || this.targetDoc!; +        const acl = `acl-${normalizeEmail(user.email)}`; +        const myAcl = `acl-${Doc.CurrentUserEmailNormalized}`; + +        const docs = SelectionManager.Views().length < 2 ? [target] : SelectionManager.Views().map(docView => docView.props.Document); +        docs.forEach(doc => { +            doc.author === Doc.CurrentUserEmail && !doc[myAcl] && distributeAcls(myAcl, SharingPermissions.Admin, doc); + +            if (permission === SharingPermissions.None) { +                if (doc[acl] && doc[acl] !== SharingPermissions.None) doc.numUsersShared = NumCast(doc.numUsersShared, 1) - 1; +            } +            else { +                if (!doc[acl] || doc[acl] === SharingPermissions.None) doc.numUsersShared = NumCast(doc.numUsersShared, 0) + 1; +            } + +            distributeAcls(acl, permission as SharingPermissions, doc); + +            if (permission !== SharingPermissions.None) Doc.AddDocToList(sharingDoc, storage, doc); +            else GetEffectiveAcl(doc, user.email) === AclPrivate && Doc.RemoveDocFromList(sharingDoc, storage, (doc.aliasOf as Doc || doc)); +        }); +    } + +    /**       * Sets the permission on the target for the group.       * @param group        * @param permission  @@ -170,6 +206,14 @@ export class SharingManager extends React.Component<{}> {          docs.forEach(doc => {              doc.author === Doc.CurrentUserEmail && !doc[`acl-${Doc.CurrentUserEmailNormalized}`] && distributeAcls(`acl-${Doc.CurrentUserEmailNormalized}`, SharingPermissions.Admin, doc); + +            if (permission === SharingPermissions.None) { +                if (doc[acl] && doc[acl] !== SharingPermissions.None) doc.numGroupsShared = NumCast(doc.numGroupsShared, 1) - 1; +            } +            else { +                if (!doc[acl] || doc[acl] === SharingPermissions.None) doc.numGroupsShared = NumCast(doc.numGroupsShared, 0) + 1; +            } +              distributeAcls(acl, permission as SharingPermissions, doc);              if (group instanceof Doc) { @@ -265,21 +309,21 @@ export class SharingManager extends React.Component<{}> {      /**       * Shares the document with a user.       */ -    setInternalSharing = (recipient: ValidatedUser, permission: string, targetDoc?: Doc) => { -        const { user, sharingDoc } = recipient; -        const target = targetDoc || this.targetDoc!; -        const acl = `acl-${normalizeEmail(user.email)}`; -        const myAcl = `acl-${Doc.CurrentUserEmailNormalized}`; - -        const docs = SelectionManager.Views().length < 2 ? [target] : SelectionManager.Views().map(docView => docView.props.Document); -        docs.forEach(doc => { -            doc.author === Doc.CurrentUserEmail && !doc[myAcl] && distributeAcls(myAcl, SharingPermissions.Admin, doc); -            distributeAcls(acl, permission as SharingPermissions, doc); - -            if (permission !== SharingPermissions.None) Doc.AddDocToList(sharingDoc, storage, doc); -            else GetEffectiveAcl(doc, user.email) === AclPrivate && Doc.RemoveDocFromList(sharingDoc, storage, (doc.aliasOf as Doc || doc)); -        }); -    } +    // setInternalSharing = (recipient: ValidatedUser, permission: string, targetDoc?: Doc) => { +    //     const { user, sharingDoc } = recipient; +    //     const target = targetDoc || this.targetDoc!; +    //     const acl = `acl-${normalizeEmail(user.email)}`; +    //     const myAcl = `acl-${Doc.CurrentUserEmailNormalized}`; + +    //     const docs = SelectionManager.Views().length < 2 ? [target] : SelectionManager.Views().map(docView => docView.props.Document); +    //     docs.forEach(doc => { +    //         doc.author === Doc.CurrentUserEmail && !doc[myAcl] && distributeAcls(myAcl, SharingPermissions.Admin, doc); +    //         distributeAcls(acl, permission as SharingPermissions, doc); + +    //         if (permission !== SharingPermissions.None) Doc.AddDocToList(sharingDoc, storage, doc); +    //         else GetEffectiveAcl(doc, user.email) === AclPrivate && Doc.RemoveDocFromList(sharingDoc, storage, (doc.aliasOf as Doc || doc)); +    //     }); +    // }      // private setExternalSharing = (permission: string) => { @@ -313,11 +357,11 @@ export class SharingManager extends React.Component<{}> {          if (!uniform) dropdownValues.unshift("-multiple-");          if (override) dropdownValues.unshift("None");          return dropdownValues.filter(permission => permission !== SharingPermissions.View).map(permission => -            ( -                <option key={permission} value={permission}> -                    {permission === SharingPermissions.Add ? "Can Augment" : permission} -                </option> -            ) +        ( +            <option key={permission} value={permission}> +                {permission === SharingPermissions.Add ? "Can Augment" : permission} +            </option> +        )          );      } @@ -397,20 +441,12 @@ export class SharingManager extends React.Component<{}> {      }      distributeOverCollection = (targetDoc?: Doc) => { -        const AclMap = new Map<symbol, string>([ -            [AclPrivate, SharingPermissions.None], -            [AclReadonly, SharingPermissions.View], -            [AclAddonly, SharingPermissions.Add], -            [AclEdit, SharingPermissions.Edit], -            [AclAdmin, SharingPermissions.Admin] -        ]); -          const target = targetDoc || this.targetDoc!;          const docs = SelectionManager.Views().length < 2 ? [target] : SelectionManager.Views().map(docView => docView.props.Document);          docs.forEach(doc => {              for (const [key, value] of Object.entries(doc[AclSym])) { -                distributeAcls(key, AclMap.get(value)! as SharingPermissions, target); +                distributeAcls(key, this.AclMap.get(value)! as SharingPermissions, target);              }          });      } @@ -471,7 +507,8 @@ export class SharingManager extends React.Component<{}> {          const targetDoc = docs[0];          // tslint:disable-next-line: no-unnecessary-callback-wrapper -        const admin = this.myDocAcls ? Boolean(docs.length) : docs.map(doc => GetEffectiveAcl(doc)).every(acl => acl === AclAdmin); // if the user has admin access to all selected docs +        const effectiveAcls = docs.map(doc => GetEffectiveAcl(doc)); +        const admin = this.myDocAcls ? Boolean(docs.length) : effectiveAcls.every(acl => acl === AclAdmin);          // users in common between all docs          const commonKeys = intersection(...docs.map(doc => this.layoutDocAcls ? doc?.[AclSym] && Object.keys(doc[AclSym]) : doc?.[DataSym]?.[AclSym] && Object.keys(doc[DataSym][AclSym]))); @@ -535,7 +572,7 @@ export class SharingManager extends React.Component<{}> {                          <span className={"padding"}>Me</span>                          <div className="edit-actions">                              <div className={"permissions-dropdown"}> -                                {targetDoc?.[`acl-${Doc.CurrentUserEmailNormalized}`]} +                                {effectiveAcls.every(acl => acl === effectiveAcls[0]) ? this.AclMap.get(effectiveAcls[0])! : "-multiple-"}                              </div>                          </div>                      </div> @@ -545,7 +582,7 @@ export class SharingManager extends React.Component<{}> {          // the list of groups shared with          const groupListMap: (Doc | { title: string })[] = groups.filter(({ title }) => docs.length > 1 ? commonKeys.includes(`acl-${normalizeEmail(StrCast(title))}`) : true); -        groupListMap.unshift({ title: "Public" }, { title: "Override" }); +        groupListMap.unshift({ title: "Public" });//, { title: "Override" });          const groupListContents = groupListMap.map(group => {              const groupKey = `acl-${StrCast(group.title)}`;              const uniform = docs.every(doc => this.layoutDocAcls ? doc?.[AclSym]?.[groupKey] === docs[0]?.[AclSym]?.[groupKey] : doc?.[DataSym]?.[AclSym]?.[groupKey] === docs[0]?.[DataSym]?.[AclSym]?.[groupKey]); @@ -597,9 +634,10 @@ export class SharingManager extends React.Component<{}> {                      {<div className="share-container">                          <div className="share-setup">                              <Select -                                className={"user-search"} -                                placeholder={"Enter user or group name..."} +                                className="user-search" +                                placeholder="Enter user or group name..."                                  isMulti +                                isSearchable                                  closeMenuOnSelect={false}                                  options={options}                                  onChange={this.handleUsersChange} @@ -623,16 +661,16 @@ export class SharingManager extends React.Component<{}> {                          </div>                          <div className="acl-container"> -                            <div className="myDocs-acls"> +                            {/* <div className="myDocs-acls">                                  <input type="checkbox" onChange={action(() => this.myDocAcls = !this.myDocAcls)} checked={this.myDocAcls} /> <label>My Docs</label> -                            </div> +                            </div> */}                              {Doc.UserDoc().noviceMode ? (null) :                                  <div className="layoutDoc-acls">                                      <input type="checkbox" onChange={action(() => this.layoutDocAcls = !this.layoutDocAcls)} checked={this.layoutDocAcls} /> <label>Layout</label>                                  </div>} -                            <button className="distribute-button" onClick={() => this.distributeOverCollection()}> +                            {/* <button className="distribute-button" onClick={() => this.distributeOverCollection()}>                                  Distribute -                        </button> +                        </button> */}                          </div>                      </div>                      } | 
