Files
AFFiNE-Mirror/tests/blocksuite/e2e/edgeless/pan.spec.ts

264 lines
6.6 KiB
TypeScript

import { expect, type Locator, type Page } from '@playwright/test';
import {
activeNoteInEdgeless,
addNote,
assertEdgelessTool,
locatorEdgelessToolButton,
multiTouchDown,
multiTouchMove,
multiTouchUp,
setEdgelessTool,
switchEditorMode,
} from '../utils/actions/edgeless.js';
import {
addBasicRectShapeElement,
dragBetweenCoords,
enterPlaygroundRoom,
initEmptyEdgelessState,
toggleEditorReadonly,
type,
waitForInlineEditorStateUpdated,
waitNextFrame,
} from '../utils/actions/index.js';
import {
assertEdgelessSelectedRect,
assertNotHasClass,
assertRichTexts,
} from '../utils/asserts.js';
import { test } from '../utils/playwright.js';
test('pan tool basic', async ({ page }) => {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
const start = { x: 100, y: 100 };
const end = { x: 200, y: 200 };
await addBasicRectShapeElement(page, start, end);
await setEdgelessTool(page, 'pan');
await dragBetweenCoords(
page,
{
x: start.x + 5,
y: start.y + 5,
},
{
x: start.x + 25,
y: start.y + 25,
}
);
await setEdgelessTool(page, 'default');
await page.mouse.click(start.x + 25, start.y + 25);
await assertEdgelessSelectedRect(page, [120, 120, 100, 100]);
});
test('pan tool shortcut', async ({ page }) => {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
const start = { x: 100, y: 100 };
const end = { x: 200, y: 200 };
await addBasicRectShapeElement(page, start, end);
await page.mouse.click(start.x + 5, start.y + 5);
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
await page.keyboard.down('Space');
await assertEdgelessTool(page, 'pan');
await dragBetweenCoords(
page,
{
x: start.x + 5,
y: start.y + 5,
},
{
x: start.x + 25,
y: start.y + 25,
}
);
await page.keyboard.up('Space');
await assertEdgelessSelectedRect(page, [120, 120, 100, 100]);
});
// FIXME(@doouding): Failed on CI
test.skip('pan tool with middle button', async ({ page }) => {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
const start = { x: 100, y: 100 };
const end = { x: 200, y: 200 };
await addBasicRectShapeElement(page, start, end);
await page.mouse.click(start.x + 5, start.y + 5);
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
await dragBetweenCoords(
page,
{
x: 400,
y: 400,
},
{
x: 420,
y: 420,
},
{
button: 'middle',
}
);
await assertEdgelessTool(page, 'default');
await assertEdgelessSelectedRect(page, [120, 120, 100, 100]);
});
test('pan tool shortcut should revert to the previous tool on keyup', async ({
page,
}) => {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
await page.mouse.click(100, 100);
await setEdgelessTool(page, 'brush');
{
await page.keyboard.down('Space');
await assertEdgelessTool(page, 'pan');
await page.keyboard.up('Space');
await assertEdgelessTool(page, 'brush');
}
});
test('pan tool shortcut does not affect other tools while using the tool', async ({
page,
}) => {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
// Test if while drawing shortcut does not switch to pan tool
await setEdgelessTool(page, 'brush');
await dragBetweenCoords(
page,
{ x: 100, y: 110 },
{ x: 200, y: 300 },
{
click: true,
beforeMouseUp: async () => {
await page.keyboard.down('Space');
await assertEdgelessTool(page, 'brush');
},
}
);
await setEdgelessTool(page, 'eraser');
await dragBetweenCoords(
page,
{ x: 100, y: 110 },
{ x: 200, y: 300 },
{
click: true,
beforeMouseUp: async () => {
await page.keyboard.down('Space');
await assertEdgelessTool(page, 'eraser');
},
}
);
// Maybe add other tools too
});
test('pan tool shortcut when user is editing', async ({ page }) => {
await enterPlaygroundRoom(page);
const ids = await initEmptyEdgelessState(page);
await switchEditorMode(page);
await setEdgelessTool(page, 'default');
await activeNoteInEdgeless(page, ids.noteId);
await waitForInlineEditorStateUpdated(page);
await type(page, 'hello');
await assertRichTexts(page, ['hello']);
await page.keyboard.down('Space');
const defaultButton = await locatorEdgelessToolButton(page, 'pan', false);
await assertNotHasClass(defaultButton, 'pan');
await waitNextFrame(page);
});
test.describe('pan tool in readonly mode', () => {
async function setupReadonlyEdgeless(page: Page) {
await enterPlaygroundRoom(page);
await initEmptyEdgelessState(page);
await switchEditorMode(page);
const noteId = await addNote(page, 'hello world', 100, 200);
await page.mouse.click(50, 100);
const edgelessNote = page.locator(
`affine-edgeless-note[data-block-id="${noteId}"]`
);
const originalBoundingBox = await edgelessNote.boundingBox();
expect(originalBoundingBox).not.toBeNull();
const { x: originalX, y: originalY } = originalBoundingBox!;
// Toggle readonly mode
await toggleEditorReadonly(page);
await page.waitForTimeout(100);
return { edgelessNote, originalX, originalY };
}
async function assertPanned(
edgelessNote: Locator,
originalX: number,
originalY: number
) {
const newBoundingBox = await edgelessNote.boundingBox();
expect(newBoundingBox).not.toBeNull();
const { x: newX, y: newY } = newBoundingBox!;
expect(newX).toBeGreaterThan(originalX);
expect(newY).toBeGreaterThan(originalY);
}
test('can be used by keyboard', async ({ page }) => {
const { edgelessNote, originalX, originalY } =
await setupReadonlyEdgeless(page);
await page.keyboard.down('Space');
await assertEdgelessTool(page, 'pan');
// Pan the viewport
await dragBetweenCoords(page, { x: 300, y: 300 }, { x: 400, y: 400 });
await assertPanned(edgelessNote, originalX, originalY);
});
test('can be used by multi-touch', async ({ page }) => {
const { edgelessNote, originalX, originalY } =
await setupReadonlyEdgeless(page);
// Pan the viewport using multi-touch
const from = [
{ x: 300, y: 300 },
{ x: 400, y: 300 },
];
const to = [
{ x: 350, y: 350 },
{ x: 450, y: 350 },
];
await multiTouchDown(page, from);
await multiTouchMove(page, from, to);
await multiTouchUp(page, to);
await assertPanned(edgelessNote, originalX, originalY);
});
});