mirror of
https://github.com/toeverything/AFFiNE.git
synced 2026-02-12 12:28:42 +00:00
449 lines
12 KiB
TypeScript
449 lines
12 KiB
TypeScript
import { expect, type Page } from '@playwright/test';
|
|
|
|
import {
|
|
createShapeElement,
|
|
edgelessCommonSetup,
|
|
getFirstContainerId,
|
|
getSelectedBound,
|
|
getSortedIds,
|
|
initThreeOverlapFilledShapes,
|
|
initThreeOverlapNotes,
|
|
Shape,
|
|
shiftClickView,
|
|
switchEditorMode,
|
|
triggerComponentToolbarAction,
|
|
zoomResetByKeyboard,
|
|
} from '../utils/actions/edgeless.js';
|
|
import {
|
|
captureHistory,
|
|
clickView,
|
|
enterPlaygroundRoom,
|
|
initEmptyEdgelessState,
|
|
redoByKeyboard,
|
|
undoByKeyboard,
|
|
waitNextFrame,
|
|
} from '../utils/actions/index.js';
|
|
import {
|
|
assertEdgelessSelectedRect,
|
|
assertSelectedBound,
|
|
} from '../utils/asserts.js';
|
|
import {
|
|
DEFAULT_NOTE_HEIGHT,
|
|
DEFAULT_NOTE_WIDTH,
|
|
} from '../utils/bs-alternative.js';
|
|
import { test } from '../utils/playwright.js';
|
|
|
|
test.describe('reordering', () => {
|
|
test.describe('group index', () => {
|
|
let sortedIds: string[];
|
|
|
|
async function init(page: Page) {
|
|
await edgelessCommonSetup(page);
|
|
await createShapeElement(page, [0, 0], [100, 100], Shape.Square);
|
|
await createShapeElement(page, [100, 0], [200, 100], Shape.Square);
|
|
await createShapeElement(page, [200, 0], [300, 100], Shape.Square);
|
|
await createShapeElement(page, [300, 0], [400, 100], Shape.Square);
|
|
sortedIds = await getSortedIds(page);
|
|
}
|
|
|
|
test('group', async ({ page }) => {
|
|
await init(page);
|
|
await clickView(page, [50, 50]);
|
|
await shiftClickView(page, [150, 50]);
|
|
await triggerComponentToolbarAction(page, 'addGroup');
|
|
const groupId = await getFirstContainerId(page);
|
|
const currentSortedIds = await getSortedIds(page);
|
|
|
|
expect(currentSortedIds).toEqual([
|
|
...sortedIds.slice(2),
|
|
groupId,
|
|
...sortedIds.slice(0, 2),
|
|
]);
|
|
});
|
|
|
|
test('release from group', async ({ page }) => {
|
|
await init(page);
|
|
await clickView(page, [50, 50]);
|
|
await shiftClickView(page, [150, 50]);
|
|
await triggerComponentToolbarAction(page, 'addGroup');
|
|
const groupId = await getFirstContainerId(page);
|
|
await clickView(page, [50, 50]);
|
|
await triggerComponentToolbarAction(page, 'releaseFromGroup');
|
|
const currentSortedIds = await getSortedIds(page);
|
|
const releasedShapeId = sortedIds[0];
|
|
|
|
expect(currentSortedIds).toEqual([
|
|
...sortedIds.slice(2),
|
|
groupId,
|
|
sortedIds[1],
|
|
releasedShapeId,
|
|
]);
|
|
});
|
|
|
|
test('ungroup', async ({ page }) => {
|
|
await init(page);
|
|
await clickView(page, [50, 50]);
|
|
await shiftClickView(page, [150, 50]);
|
|
await triggerComponentToolbarAction(page, 'addGroup');
|
|
await triggerComponentToolbarAction(page, 'ungroup');
|
|
const currentSortedIds = await getSortedIds(page);
|
|
const ungroupedIds = [sortedIds[0], sortedIds[1]];
|
|
|
|
expect(currentSortedIds).toEqual([
|
|
...sortedIds.filter(id => !ungroupedIds.includes(id)),
|
|
...ungroupedIds,
|
|
]);
|
|
});
|
|
});
|
|
|
|
test.describe('reordering shapes', () => {
|
|
async function init(page: Page) {
|
|
await enterPlaygroundRoom(page);
|
|
await initEmptyEdgelessState(page);
|
|
await switchEditorMode(page);
|
|
await initThreeOverlapFilledShapes(page);
|
|
await page.mouse.click(0, 0);
|
|
}
|
|
|
|
test('bring to front', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [160, 160, 100, 100]);
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect1
|
|
await page.mouse.click(150, 150);
|
|
await assertEdgelessSelectedRect(page, [130, 130, 100, 100]);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(110, 130);
|
|
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
|
|
|
|
// bring rect0 to front
|
|
await triggerComponentToolbarAction(page, 'bringToFront');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
|
|
});
|
|
|
|
test('bring forward', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(120, 120);
|
|
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
|
|
|
|
// bring rect0 forward
|
|
await triggerComponentToolbarAction(page, 'bringForward');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(150, 150);
|
|
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
|
|
});
|
|
|
|
test('send backward', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [160, 160, 100, 100]);
|
|
|
|
// bring rect2 backward
|
|
await triggerComponentToolbarAction(page, 'sendBackward');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect1
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [130, 130, 100, 100]);
|
|
});
|
|
|
|
test('send to back', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [160, 160, 100, 100]);
|
|
|
|
// bring rect2 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect1
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [130, 130, 100, 100]);
|
|
|
|
// send rect1 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [100, 100, 100, 100]);
|
|
});
|
|
|
|
test('undo and redo', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [160, 160, 100, 100]);
|
|
|
|
// send rect2 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect1
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [130, 130, 100, 100]);
|
|
|
|
// undo
|
|
await undoByKeyboard(page);
|
|
|
|
// clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [160, 160, 100, 100]);
|
|
|
|
// redo
|
|
await redoByKeyboard(page);
|
|
|
|
// clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect2
|
|
await page.mouse.click(180, 180);
|
|
await assertEdgelessSelectedRect(page, [130, 130, 100, 100]);
|
|
});
|
|
});
|
|
|
|
test.describe('reordering notes', () => {
|
|
async function init(page: Page) {
|
|
await enterPlaygroundRoom(page);
|
|
await initEmptyEdgelessState(page);
|
|
await switchEditorMode(page);
|
|
await zoomResetByKeyboard(page);
|
|
await initThreeOverlapNotes(page);
|
|
await waitNextFrame(page);
|
|
await page.mouse.click(0, 0);
|
|
}
|
|
|
|
test('bring to front', async ({ page }) => {
|
|
await edgelessCommonSetup(page);
|
|
await zoomResetByKeyboard(page);
|
|
await initThreeOverlapNotes(page, 130, 190);
|
|
await waitNextFrame(page);
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 100);
|
|
// should be note2
|
|
await page.mouse.click(180, 200);
|
|
const bound = await getSelectedBound(page);
|
|
|
|
await assertSelectedBound(page, bound);
|
|
|
|
await clickView(page, [bound[0] - 15, bound[1] + 10]);
|
|
bound[0] -= 30;
|
|
await assertSelectedBound(page, bound);
|
|
|
|
await clickView(page, [bound[0] - 15, bound[1] + 10]);
|
|
bound[0] -= 30;
|
|
await assertSelectedBound(page, bound);
|
|
|
|
// bring note0 to front
|
|
await triggerComponentToolbarAction(page, 'bringToFront');
|
|
// clear
|
|
await page.mouse.click(100, 50);
|
|
// should be note0
|
|
await clickView(page, [bound[0] + 40, bound[1] + 10]);
|
|
await assertSelectedBound(page, bound);
|
|
});
|
|
|
|
test('bring forward', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note0
|
|
await page.mouse.click(120, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
100,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// bring note0 forward
|
|
await triggerComponentToolbarAction(page, 'bringForward');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be rect0
|
|
await page.mouse.click(150, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
100,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
});
|
|
|
|
test('send backward', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note2
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
160,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// bring note2 backward
|
|
await triggerComponentToolbarAction(page, 'sendBackward');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note1
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
130,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
});
|
|
|
|
test('send to back', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note2
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
160,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// bring note2 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note1
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
130,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// send note1 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note0
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
100,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
});
|
|
|
|
test('undo and redo', async ({ page }) => {
|
|
await init(page);
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note2
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
160,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
await captureHistory(page);
|
|
|
|
// bring note2 to back
|
|
await triggerComponentToolbarAction(page, 'sendToBack');
|
|
|
|
// click outside to clear selection
|
|
await page.mouse.click(50, 50);
|
|
|
|
// should be note1
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
130,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// undo
|
|
await undoByKeyboard(page);
|
|
// clear selection
|
|
await page.mouse.click(50, 50);
|
|
// should be note2
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
160,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
|
|
// redo
|
|
await redoByKeyboard(page);
|
|
// clear selection
|
|
await page.mouse.click(50, 50);
|
|
// should be note1
|
|
await page.mouse.click(180, 140);
|
|
await assertEdgelessSelectedRect(page, [
|
|
130,
|
|
100,
|
|
DEFAULT_NOTE_WIDTH,
|
|
DEFAULT_NOTE_HEIGHT,
|
|
]);
|
|
});
|
|
});
|
|
});
|