feat(core): add track events for editor header (#7661)

close AF-1054
This commit is contained in:
JimmFly
2024-08-07 05:52:40 +00:00
parent b5e543c406
commit 74025fc85e
12 changed files with 125 additions and 19 deletions

View File

@@ -1,6 +1,8 @@
import { useEnableCloud } from '@affine/core/hooks/affine/use-enable-cloud';
import { track } from '@affine/core/mixpanel';
import type { Doc } from '@blocksuite/store';
import type { Workspace } from '@toeverything/infra';
import { type Workspace } from '@toeverything/infra';
import { useCallback } from 'react';
import { ShareMenu } from './share-menu';
@@ -11,6 +13,11 @@ type SharePageModalProps = {
export const SharePageButton = ({ workspace, page }: SharePageModalProps) => {
const confirmEnableCloud = useEnableCloud();
const handleOpenShareModal = useCallback((open: boolean) => {
if (open) {
track.$.sharePanel.$.open();
}
}, []);
return (
<ShareMenu
@@ -21,6 +28,7 @@ export const SharePageButton = ({ workspace, page }: SharePageModalProps) => {
openPageId: page.id,
})
}
onOpenShareModal={handleOpenShareModal}
/>
);
};

View File

@@ -21,6 +21,7 @@ export interface ShareMenuProps extends PropsWithChildren {
workspaceMetadata: WorkspaceMetadata;
currentPage: Doc;
onEnableAffineCloud: () => void;
onOpenShareModal?: (open: boolean) => void;
}
export const ShareMenuContent = (props: ShareMenuProps) => {
@@ -73,6 +74,7 @@ const LocalShareMenu = (props: ShareMenuProps) => {
}}
rootOptions={{
modal: false,
onOpenChange: props.onOpenShareModal,
}}
>
<div data-testid="local-share-menu-button">
@@ -92,6 +94,7 @@ const CloudShareMenu = (props: ShareMenuProps) => {
}}
rootOptions={{
modal: false,
onOpenChange: props.onOpenShareModal,
}}
>
<div data-testid="cloud-share-menu-button">

View File

@@ -106,7 +106,7 @@ export const AffineSharePage = (props: ShareMenuProps) => {
await shareService.share.enableShare(
mode === 'edgeless' ? PublicPageMode.Edgeless : PublicPageMode.Page
);
track.$.header.share.createShareLink({
track.$.sharePanel.$.createShareLink({
mode,
});
notify.success({

View File

@@ -1,4 +1,5 @@
import { FavoriteTag } from '@affine/core/components/page-list';
import { track } from '@affine/core/mixpanel';
import { CompatibleFavoriteItemsAdapter } from '@affine/core/modules/properties';
import { toast } from '@affine/core/utils';
import { useI18n } from '@affine/i18n';
@@ -30,11 +31,16 @@ export const useFavorite = (pageId: string) => {
export const FavoriteButton = ({ pageId }: FavoriteButtonProps) => {
const { favorite, toggleFavorite } = useFavorite(pageId);
const handleFavorite = useCallback(() => {
track.$.header.actions.toggleFavorite();
toggleFavorite();
}, [toggleFavorite]);
return (
<FavoriteTag
data-testid="pin-button"
active={!!favorite}
onClick={toggleFavorite}
onClick={handleFavorite}
/>
);
};

View File

@@ -1,15 +1,20 @@
import { IconButton } from '@affine/component';
import { openInfoModalAtom } from '@affine/core/atoms';
import { track } from '@affine/core/mixpanel';
import { useI18n } from '@affine/i18n';
import { InformationIcon } from '@blocksuite/icons/rc';
import { useSetAtom } from 'jotai';
import { useCallback } from 'react';
export const InfoButton = () => {
const setOpenInfoModal = useSetAtom(openInfoModalAtom);
const t = useI18n();
const onOpenInfoModal = () => {
const onOpenInfoModal = useCallback(() => {
track.$.header.actions.openDocInfo();
setOpenInfoModal(true);
};
}, [setOpenInfoModal]);
return (
<IconButton
size="20"

View File

@@ -109,6 +109,7 @@ export const PageHeaderMenuButton = ({
const setOpenHistoryTipsModal = useSetAtom(openHistoryTipsModalAtom);
const openHistoryModal = useCallback(() => {
track.$.header.history.open();
if (workspace.flavour === WorkspaceFlavour.AFFINE_CLOUD) {
return setHistoryModalOpen(true);
}
@@ -116,9 +117,10 @@ export const PageHeaderMenuButton = ({
}, [setOpenHistoryTipsModal, workspace.flavour]);
const setOpenInfoModal = useSetAtom(openInfoModalAtom);
const openInfoModal = () => {
const openInfoModal = useCallback(() => {
track.$.header.pageInfo.open();
setOpenInfoModal(true);
};
}, [setOpenInfoModal]);
const handleOpenInNewTab = useCallback(() => {
workbench.openDoc(pageId, {
@@ -133,6 +135,7 @@ export const PageHeaderMenuButton = ({
}, [pageId, workbench]);
const handleOpenTrashModal = useCallback(() => {
track.$.header.docOptions.deleteDoc();
setTrashModal({
open: true,
pageIds: [pageId],
@@ -140,8 +143,14 @@ export const PageHeaderMenuButton = ({
});
}, [doc.meta$.value.title, pageId, setTrashModal]);
const handleRename = useCallback(() => {
rename?.();
track.$.header.docOptions.renameDoc();
}, [rename]);
const handleSwitchMode = useCallback(() => {
doc.toggleMode();
track.$.header.docOptions.switchPageMode();
toast(
currentMode === 'page'
? t['com.affine.toastMessage.edgelessMode']()
@@ -153,17 +162,24 @@ export const PageHeaderMenuButton = ({
transition: 'all 0.3s',
};
const handleMenuOpenChange = useCallback((open: boolean) => {
if (open) {
track.$.header.docOptions.open();
}
}, []);
const exportHandler = useExportPage(doc.blockSuiteDoc);
const handleDuplicate = useCallback(() => {
duplicate(pageId);
track.$.header.actions.createDoc({
track.$.header.docOptions.createDoc({
control: 'duplicate',
});
}, [duplicate, pageId]);
const onImportFile = useAsyncCallback(async () => {
const options = await importFile();
track.$.header.docOptions.import();
if (options.isWorkspaceFile) {
track.$.header.actions.createWorkspace({
control: 'import',
@@ -175,6 +191,17 @@ export const PageHeaderMenuButton = ({
}
}, [importFile]);
const handleShareMenuOpenChange = useCallback((open: boolean) => {
if (open) {
track.$.sharePanel.$.open();
}
}, []);
const handleToggleFavorite = useCallback(() => {
track.$.header.docOptions.toggleFavorite();
toggleFavorite();
}, [toggleFavorite]);
const showResponsiveMenu = hideShare;
const ResponsiveMenuItems = (
<>
@@ -204,6 +231,9 @@ export const PageHeaderMenuButton = ({
</MenuIcon>
),
}}
subOptions={{
onOpenChange: handleShareMenuOpenChange,
}}
>
{t['com.affine.share-menu.shareButton']()}
</MenuSub>
@@ -223,7 +253,7 @@ export const PageHeaderMenuButton = ({
</MenuIcon>
}
data-testid="editor-option-menu-rename"
onSelect={rename}
onSelect={handleRename}
style={menuItemStyle}
>
{t['Rename']()}
@@ -246,7 +276,7 @@ export const PageHeaderMenuButton = ({
</MenuItem>
<MenuItem
data-testid="editor-option-menu-favorite"
onSelect={toggleFavorite}
onSelect={handleToggleFavorite}
style={menuItemStyle}
preFix={
<MenuIcon>
@@ -389,6 +419,9 @@ export const PageHeaderMenuButton = ({
contentOptions={{
align: 'center',
}}
rootOptions={{
onOpenChange: handleMenuOpenChange,
}}
>
<HeaderDropDownButton />
</Menu>

View File

@@ -18,13 +18,14 @@ export interface BlockSuiteHeaderTitleProps {
isPublic?: boolean;
inputHandleRef?: InlineEditProps['handleRef'];
className?: string;
onEditSave?: () => void;
}
const inputAttrs = {
'data-testid': 'title-content',
} as HTMLAttributes<HTMLInputElement>;
export const BlocksuiteHeaderTitle = (props: BlockSuiteHeaderTitleProps) => {
const { docCollection, pageId, isPublic, inputHandleRef } = props;
const { docCollection, pageId, isPublic, inputHandleRef, onEditSave } = props;
const currentPage = docCollection.getDoc(pageId);
const pageMeta = useBlockSuiteDocMeta(docCollection).find(
meta => meta.id === currentPage?.id
@@ -34,9 +35,10 @@ export const BlocksuiteHeaderTitle = (props: BlockSuiteHeaderTitleProps) => {
const onChange = useCallback(
(v: string) => {
onEditSave?.();
setDocTitle(currentPage?.id || '', v);
},
[currentPage?.id, setDocTitle]
[currentPage?.id, onEditSave, setDocTitle]
);
return (

View File

@@ -1,4 +1,5 @@
import { MenuIcon, MenuItem, MenuSub } from '@affine/component';
import { track } from '@affine/core/mixpanel';
import { useI18n } from '@affine/i18n';
import {
ExportIcon,
@@ -8,7 +9,7 @@ import {
ExportToPngIcon,
} from '@blocksuite/icons/rc';
import type { ReactNode } from 'react';
import { useMemo } from 'react';
import { useCallback, useMemo } from 'react';
import { transitionStyle } from './index.css';
@@ -115,6 +116,12 @@ export const Export = ({ exportHandler, className, pageMode }: ExportProps) => {
pageMode={pageMode}
/>
);
const handleExportMenuOpenChange = useCallback((open: boolean) => {
if (open) {
track.$.header.docOptions.export();
}
}, []);
return (
<MenuSub
items={items}
@@ -127,6 +134,9 @@ export const Export = ({ exportHandler, className, pageMode }: ExportProps) => {
),
['data-testid' as string]: 'export-menu',
}}
subOptions={{
onOpenChange: handleExportMenuOpenChange,
}}
>
{t.Export()}
</MenuSub>

View File

@@ -26,7 +26,7 @@ async function exportHandler({ page, type }: ExportHandlerOptions) {
if (editorRoot) {
pageService = editorRoot.spec.getService<PageRootService>('affine:page');
}
track.$.header.share.export({
track.$.sharePanel.$.export({
type,
});
switch (type) {

View File

@@ -68,7 +68,7 @@ export const useSharingUrl = ({
.catch(err => {
console.error(err);
});
track.$.header.share.copyShareLink({
track.$.sharePanel.$.copyShareLink({
type: urlType === 'share' ? 'public' : 'private',
});
} else {

View File

@@ -35,7 +35,13 @@ type DocEvents =
| 'renameDoc'
| 'linkDoc'
| 'deleteDoc'
| 'switchPageMode';
| 'switchPageMode'
| 'openDocOptionsMenu'
| 'openDocInfo'
| 'renameDoc'
| 'deleteDoc'
| 'viewHistoryVersions'
| 'viewInfo';
type EditorEvents = 'bold' | 'italic' | 'underline' | 'strikeThrough';
// END SECTION
@@ -75,7 +81,11 @@ type OrganizeEvents =
// END SECTION
// SECTION: cloud events
type ShareEvents = 'createShareLink' | 'copyShareLink';
type ShareEvents =
| 'createShareLink'
| 'copyShareLink'
| 'openShareMenu'
| 'share';
type AuthEvents = 'signIn' | 'signUp' | 'oauth' | 'signOut';
type AccountEvents = 'uploadAvatar' | 'removeAvatar' | 'updateUserName';
type PaymentEvents =
@@ -127,6 +137,12 @@ const PageEvents = {
$: ['createWorkspace', 'checkout'],
auth: ['oauth', 'signIn', 'signUp'],
},
sharePanel: {
$: ['createShareLink', 'copyShareLink', 'export', 'open'],
},
docInfoPanel: {
$: ['open'],
},
settingsPanel: {
menu: ['openSettings'],
workspace: ['viewPlans'],
@@ -194,8 +210,26 @@ const PageEvents = {
aiAction: ['viewPlans'],
},
header: {
actions: ['createDoc', 'createWorkspace', 'switchPageMode'],
share: ['createShareLink', 'copyShareLink', 'export'],
actions: [
'createDoc',
'createWorkspace',
'switchPageMode',
'toggleFavorite',
'openDocInfo',
'renameDoc',
],
docOptions: [
'open',
'deleteDoc',
'renameDoc',
'switchPageMode',
'createDoc',
'import',
'toggleFavorite',
'export',
],
history: ['open'],
pageInfo: ['open'],
},
},
doc: {

View File

@@ -15,6 +15,7 @@ import { EditorModeSwitch } from '@affine/core/components/blocksuite/block-suite
import { useRegisterCopyLinkCommands } from '@affine/core/hooks/affine/use-register-copy-link-commands';
import { useDocCollectionPageTitle } from '@affine/core/hooks/use-block-suite-workspace-page-title';
import { useJournalInfoHelper } from '@affine/core/hooks/use-journal';
import { track } from '@affine/core/mixpanel';
import { ViewIcon, ViewTitle } from '@affine/core/modules/workbench';
import type { Doc } from '@blocksuite/store';
import {
@@ -129,6 +130,9 @@ export function NormalPageHeader({ page, workspace }: PageHeaderProps) {
const title = useDocCollectionPageTitle(workspace.docCollection, page?.id);
const doc = useService(DocService).doc;
const currentMode = useLiveData(doc.mode$);
const onEditSave = useCallback(() => {
track.$.header.actions.renameDoc();
}, []);
return (
<Header className={styles.header} ref={containerRef}>
@@ -142,6 +146,7 @@ export function NormalPageHeader({ page, workspace }: PageHeaderProps) {
inputHandleRef={titleInputHandleRef}
pageId={page?.id}
docCollection={workspace.docCollection}
onEditSave={onEditSave}
/>
<div className={styles.iconButtonContainer}>
{hideCollect ? null : (