Skip to content

Commit

Permalink
feat(core): improve invite link (#9111)
Browse files Browse the repository at this point in the history
  • Loading branch information
JimmFly authored Dec 12, 2024
1 parent cdb55a3 commit dd39d04
Show file tree
Hide file tree
Showing 5 changed files with 50 additions and 29 deletions.
Original file line number Diff line number Diff line change
@@ -1,5 +1,8 @@
import { emailRegex } from '@affine/component/auth-components';
import type { WorkspaceInviteLinkExpireTime } from '@affine/graphql';
import type {
InviteLink,
WorkspaceInviteLinkExpireTime,
} from '@affine/graphql';
import { useI18n } from '@affine/i18n';
import { useCallback, useEffect, useState } from 'react';

Expand All @@ -18,6 +21,7 @@ export interface InviteTeamMemberModalProps {
) => Promise<string>;
onRevokeInviteLink: () => Promise<boolean>;
importCSV: React.ReactNode;
invitationLink: InviteLink | null;
}

const parseEmailString = (emailString: string): string[] => {
Expand All @@ -36,6 +40,7 @@ export const InviteTeamMemberModal = ({
onGenerateInviteLink,
onRevokeInviteLink,
importCSV,
invitationLink,
}: InviteTeamMemberModalProps) => {
const t = useI18n();
const [inviteEmails, setInviteEmails] = useState('');
Expand Down Expand Up @@ -95,6 +100,7 @@ export const InviteTeamMemberModal = ({
childrenContentClassName={styles.contentStyle}
>
<ModalContent
invitationLink={invitationLink}
inviteEmail={inviteEmails}
setInviteEmail={setInviteEmails}
handleConfirm={handleConfirm}
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
import { WorkspaceInviteLinkExpireTime } from '@affine/graphql';
import {
type InviteLink,
WorkspaceInviteLinkExpireTime,
} from '@affine/graphql';
import { useI18n } from '@affine/i18n';
import { CloseIcon } from '@blocksuite/icons/rc';
import { cssVar } from '@toeverything/theme';
Expand Down Expand Up @@ -39,10 +42,12 @@ const getMenuItems = (t: ReturnType<typeof useI18n>) => [
];

export const LinkInvite = ({
invitationLink,
copyTextToClipboard,
generateInvitationLink,
revokeInvitationLink,
}: {
invitationLink: InviteLink | null;
generateInvitationLink: (
expireTime: WorkspaceInviteLinkExpireTime
) => Promise<string>;
Expand All @@ -53,7 +58,6 @@ export const LinkInvite = ({
const [selectedValue, setSelectedValue] = useState(
WorkspaceInviteLinkExpireTime.OneWeek
);
const [invitationLink, setInvitationLink] = useState('');
const menuItems = getMenuItems(t);
const items = useMemo(() => {
return menuItems.map(item => (
Expand All @@ -69,21 +73,20 @@ export const LinkInvite = ({
);

const onGenerate = useCallback(() => {
generateInvitationLink(selectedValue)
.then(link => {
setInvitationLink(link);
})
.catch(err => {
console.error('Failed to generate invitation link: ', err);
notify.error({
title: 'Failed to generate invitation link',
message: err.message,
});
generateInvitationLink(selectedValue).catch(err => {
console.error('Failed to generate invitation link: ', err);
notify.error({
title: 'Failed to generate invitation link',
message: err.message,
});
});
}, [generateInvitationLink, selectedValue]);

const onCopy = useCallback(() => {
copyTextToClipboard(invitationLink)
if (!invitationLink) {
return;
}
copyTextToClipboard(invitationLink.link)
.then(() =>
notify.success({
title: t['Copied link to clipboard'](),
Expand All @@ -99,17 +102,13 @@ export const LinkInvite = ({
}, [copyTextToClipboard, invitationLink, t]);

const onReset = useCallback(() => {
revokeInvitationLink()
.then(() => {
setInvitationLink('');
})
.catch(err => {
console.error('Failed to revoke invitation link: ', err);
notify.error({
title: 'Failed to revoke invitation link',
message: err.message,
});
revokeInvitationLink().catch(err => {
console.error('Failed to revoke invitation link: ', err);
notify.error({
title: 'Failed to revoke invitation link',
message: err.message,
});
});
}, [revokeInvitationLink]);

return (
Expand All @@ -136,7 +135,7 @@ export const LinkInvite = ({
<Input
value={
invitationLink
? invitationLink
? invitationLink.link
: 'https://your-app.com/invite/xxxxxxxx'
}
inputMode="none"
Expand Down
Original file line number Diff line number Diff line change
@@ -1,4 +1,7 @@
import type { WorkspaceInviteLinkExpireTime } from '@affine/graphql';
import type {
InviteLink,
WorkspaceInviteLinkExpireTime,
} from '@affine/graphql';
import { useI18n } from '@affine/i18n';
import { EmailIcon, LinkIcon } from '@blocksuite/icons/rc';

Expand All @@ -20,9 +23,11 @@ export const ModalContent = ({
onGenerateInviteLink,
onRevokeInviteLink,
importCSV,
invitationLink,
}: {
inviteEmail: string;
importCSV: React.ReactNode;
invitationLink: InviteLink | null;
setInviteEmail: (value: string) => void;
inviteMethod: InviteMethodType;
onInviteMethodChange: (value: InviteMethodType) => void;
Expand Down Expand Up @@ -78,6 +83,7 @@ export const ModalContent = ({
/>
) : (
<LinkInvite
invitationLink={invitationLink}
copyTextToClipboard={copyTextToClipboard}
generateInvitationLink={onGenerateInviteLink}
revokeInvitationLink={onRevokeInviteLink}
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,6 +13,7 @@ import {
WorkspacePermissionService,
} from '@affine/core/modules/permissions';
import { WorkspaceQuotaService } from '@affine/core/modules/quota';
import { WorkspaceShareSettingService } from '@affine/core/modules/share-setting';
import { copyTextToClipboard } from '@affine/core/utils/clipboard';
import { emailRegex } from '@affine/core/utils/email-regex';
import type { WorkspaceInviteLinkExpireTime } from '@affine/graphql';
Expand Down Expand Up @@ -49,6 +50,10 @@ export const CloudWorkspaceMembersPanel = ({
onChangeSettingState: (settingState: SettingState) => void;
isTeam?: boolean;
}) => {
const workspaceShareSettingService = useService(WorkspaceShareSettingService);
const inviteLink = useLiveData(
workspaceShareSettingService.sharePreview.inviteLink$
);
const serverService = useService(ServerService);
const hasPaymentFeature = useLiveData(
serverService.server.features$.map(f => f?.payment)
Expand Down Expand Up @@ -90,15 +95,17 @@ export const CloudWorkspaceMembersPanel = ({
async (expireTime: WorkspaceInviteLinkExpireTime) => {
const { link } =
await permissionService.permission.generateInviteLink(expireTime);
workspaceShareSettingService.sharePreview.revalidate();
return link;
},
[permissionService.permission]
[permissionService.permission, workspaceShareSettingService.sharePreview]
);

const onRevokeInviteLink = useCallback(async () => {
const success = await permissionService.permission.revokeInviteLink();
workspaceShareSettingService.sharePreview.revalidate();
return success;
}, [permissionService.permission]);
}, [permissionService.permission, workspaceShareSettingService.sharePreview]);

const onInviteBatchConfirm = useCallback<
InviteTeamMemberModalProps['onConfirm']
Expand Down Expand Up @@ -218,6 +225,7 @@ export const CloudWorkspaceMembersPanel = ({
onGenerateInviteLink={onGenerateInviteLink}
onRevokeInviteLink={onRevokeInviteLink}
importCSV={<ImportCSV onImport={onImportCSV} />}
invitationLink={inviteLink}
/>
)}
</>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,5 @@
import { DebugLogger } from '@affine/debug';
import type { GetWorkspaceConfigQuery } from '@affine/graphql';
import type { GetWorkspaceConfigQuery, InviteLink } from '@affine/graphql';
import type { WorkspaceService } from '@toeverything/infra';
import {
backoffRetry,
Expand All @@ -25,6 +25,7 @@ const logger = new DebugLogger('affine:workspace-permission');
export class WorkspaceShareSetting extends Entity {
enableAi$ = new LiveData<EnableAi | null>(null);
enableUrlPreview$ = new LiveData<EnableUrlPreview | null>(null);
inviteLink$ = new LiveData<InviteLink | null>(null);
isLoading$ = new LiveData(false);
error$ = new LiveData<any>(null);

Expand Down Expand Up @@ -56,6 +57,7 @@ export class WorkspaceShareSetting extends Entity {
if (value) {
this.enableAi$.next(value.enableAi);
this.enableUrlPreview$.next(value.enableUrlPreview);
this.inviteLink$.next(value.inviteLink);
}
return EMPTY;
}),
Expand Down

0 comments on commit dd39d04

Please sign in to comment.