2023-10-13 07:32:10 -07:00
|
|
|
import { test, expect } from './util/test';
|
|
|
|
import MainScreen from './models/MainScreen';
|
2023-10-31 08:05:28 -07:00
|
|
|
import { _electron as electron } from '@playwright/test';
|
|
|
|
import { writeFile } from 'fs-extra';
|
|
|
|
import { join } from 'path';
|
2023-11-12 07:06:32 -08:00
|
|
|
import createStartupArgs from './util/createStartupArgs';
|
|
|
|
import firstNonDevToolsWindow from './util/firstNonDevToolsWindow';
|
2023-12-06 11:17:16 -08:00
|
|
|
import setFilePickerResponse from './util/setFilePickerResponse';
|
2024-02-02 09:58:27 -08:00
|
|
|
import setMessageBoxResponse from './util/setMessageBoxResponse';
|
2024-08-02 06:47:43 -07:00
|
|
|
import getImageSourceSize from './util/getImageSourceSize';
|
2023-10-13 07:32:10 -07:00
|
|
|
|
|
|
|
|
|
|
|
test.describe('main', () => {
|
|
|
|
test('app should launch', async ({ mainWindow }) => {
|
|
|
|
// A window should open with the correct title
|
|
|
|
expect(await mainWindow.title()).toMatch(/^Joplin/);
|
|
|
|
|
|
|
|
const mainPage = new MainScreen(mainWindow);
|
|
|
|
await mainPage.waitFor();
|
|
|
|
});
|
|
|
|
|
|
|
|
test('should be able to create and edit a new note', async ({ mainWindow }) => {
|
|
|
|
const mainScreen = new MainScreen(mainWindow);
|
2023-12-06 11:17:16 -08:00
|
|
|
const editor = await mainScreen.createNewNote('Test note');
|
2023-10-13 07:32:10 -07:00
|
|
|
|
|
|
|
// Note list should contain the new note
|
2024-08-31 08:05:01 -07:00
|
|
|
await expect(mainScreen.noteList.getNoteItemByTitle('Test note')).toBeVisible();
|
2023-10-13 07:32:10 -07:00
|
|
|
|
|
|
|
// Focus the editor
|
|
|
|
await editor.codeMirrorEditor.click();
|
|
|
|
|
|
|
|
// Type some text
|
|
|
|
await mainWindow.keyboard.type('# Test note!');
|
|
|
|
await mainWindow.keyboard.press('Enter');
|
|
|
|
await mainWindow.keyboard.press('Enter');
|
|
|
|
await mainWindow.keyboard.type('New note content!');
|
|
|
|
|
|
|
|
// Should render
|
2024-09-12 09:54:10 -07:00
|
|
|
const viewerFrame = editor.getNoteViewerFrameLocator();
|
2023-10-13 07:32:10 -07:00
|
|
|
await expect(viewerFrame.locator('h1')).toHaveText('Test note!');
|
|
|
|
});
|
|
|
|
|
2023-12-24 06:50:22 -08:00
|
|
|
test('mermaid and KaTeX should render', async ({ mainWindow }) => {
|
|
|
|
const mainScreen = new MainScreen(mainWindow);
|
|
|
|
const editor = await mainScreen.createNewNote('🚧 Test 🚧');
|
|
|
|
|
|
|
|
const testCommitId = 'bf59b2';
|
2024-02-02 09:58:27 -08:00
|
|
|
await editor.focusCodeMirrorEditor();
|
2023-12-24 06:50:22 -08:00
|
|
|
const noteText = [
|
|
|
|
'```mermaid',
|
|
|
|
'gitGraph',
|
|
|
|
' commit id: "973193"',
|
|
|
|
` commit id: "${testCommitId}"`,
|
|
|
|
' branch dev',
|
|
|
|
' checkout dev',
|
|
|
|
' commit id: "ceea77"',
|
|
|
|
'```',
|
|
|
|
'',
|
|
|
|
'',
|
|
|
|
'KaTeX:',
|
|
|
|
'$$ \\int_0^1 \\cos(2x + 3) $$',
|
|
|
|
'',
|
|
|
|
'Sum: $\\sum_{x=0}^{100} \\tan x$',
|
|
|
|
];
|
2024-08-27 10:05:48 -07:00
|
|
|
let firstLine = true;
|
2023-12-24 06:50:22 -08:00
|
|
|
for (const line of noteText) {
|
|
|
|
if (line) {
|
2024-08-27 10:05:48 -07:00
|
|
|
if (!firstLine) {
|
|
|
|
// Remove any auto-indentation, but avoid pressing shift-tab at
|
|
|
|
// the beginning of the editor.
|
|
|
|
await mainWindow.keyboard.press('Shift+Tab');
|
|
|
|
}
|
|
|
|
|
2023-12-24 06:50:22 -08:00
|
|
|
await mainWindow.keyboard.type(line);
|
|
|
|
}
|
|
|
|
await mainWindow.keyboard.press('Enter');
|
2024-08-27 10:05:48 -07:00
|
|
|
firstLine = false;
|
2023-12-24 06:50:22 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Should render mermaid
|
2024-09-12 09:54:10 -07:00
|
|
|
const viewerFrame = editor.getNoteViewerFrameLocator();
|
2023-12-24 06:50:22 -08:00
|
|
|
await expect(
|
|
|
|
viewerFrame.locator('pre.mermaid text', { hasText: testCommitId }),
|
|
|
|
).toBeVisible();
|
|
|
|
|
|
|
|
// Should render KaTeX (block)
|
2023-12-29 09:58:07 -08:00
|
|
|
// toBeAttached: To be added to the DOM.
|
|
|
|
await expect(viewerFrame.locator('.joplin-editable > .katex-display').first()).toBeAttached();
|
2023-12-24 06:50:22 -08:00
|
|
|
await expect(
|
|
|
|
viewerFrame.locator(
|
|
|
|
'.katex-display *', { hasText: 'cos' },
|
|
|
|
).last(),
|
|
|
|
).toBeVisible();
|
|
|
|
|
|
|
|
// Should render KaTeX (inline)
|
2023-12-29 09:58:07 -08:00
|
|
|
await expect(viewerFrame.locator('.joplin-editable > .katex').first()).toBeAttached();
|
2023-12-24 06:50:22 -08:00
|
|
|
});
|
|
|
|
|
2024-02-02 09:58:27 -08:00
|
|
|
test('should correctly resize large images', async ({ electronApp, mainWindow }) => {
|
|
|
|
const mainScreen = new MainScreen(mainWindow);
|
|
|
|
await mainScreen.createNewNote('Image resize test (part 1)');
|
|
|
|
const editor = mainScreen.noteEditor;
|
|
|
|
|
|
|
|
await editor.focusCodeMirrorEditor();
|
|
|
|
|
|
|
|
const filename = 'large-jpg-image-with-exif-rotation.jpg';
|
|
|
|
await setFilePickerResponse(electronApp, [join(__dirname, 'resources', filename)]);
|
|
|
|
|
|
|
|
// Should be possible to cancel attaching for large images
|
|
|
|
await setMessageBoxResponse(electronApp, /^Cancel/i);
|
|
|
|
await editor.attachFileButton.click();
|
|
|
|
await expect(editor.codeMirrorEditor).toHaveText('', { useInnerText: true });
|
|
|
|
|
|
|
|
// Clicking "No" should not resize
|
|
|
|
await setMessageBoxResponse(electronApp, /^No/i);
|
|
|
|
await editor.attachFileButton.click();
|
|
|
|
|
2024-09-12 09:54:10 -07:00
|
|
|
const viewerFrame = editor.getNoteViewerFrameLocator();
|
2024-08-02 06:47:43 -07:00
|
|
|
const renderedImage = viewerFrame.getByAltText(filename);
|
|
|
|
|
|
|
|
const fullSize = await getImageSourceSize(renderedImage);
|
2024-02-02 09:58:27 -08:00
|
|
|
|
|
|
|
// To make it easier to find the image (one image per note), we switch to a new, empty note.
|
|
|
|
await mainScreen.createNewNote('Image resize test (part 2)');
|
|
|
|
await editor.focusCodeMirrorEditor();
|
|
|
|
|
|
|
|
// Clicking "Yes" should resize
|
|
|
|
await setMessageBoxResponse(electronApp, /^Yes/i);
|
|
|
|
await editor.attachFileButton.click();
|
|
|
|
|
2024-08-02 06:47:43 -07:00
|
|
|
const resizedSize = await getImageSourceSize(renderedImage);
|
2024-02-02 09:58:27 -08:00
|
|
|
expect(resizedSize[0]).toBeLessThan(fullSize[0]);
|
|
|
|
expect(resizedSize[1]).toBeLessThan(fullSize[1]);
|
|
|
|
|
|
|
|
// Should keep aspect ratio (regression test for #9597)
|
|
|
|
expect(fullSize[0] / resizedSize[0]).toBeCloseTo(fullSize[1] / resizedSize[1]);
|
|
|
|
});
|
|
|
|
|
2024-10-15 08:38:33 -07:00
|
|
|
for (const target of ['', '_blank']) {
|
|
|
|
test(`clicking on an external link with target=${JSON.stringify(target)} should try to launch a browser`, async ({ electronApp, mainWindow }) => {
|
|
|
|
const mainScreen = new MainScreen(mainWindow);
|
|
|
|
await mainScreen.waitFor();
|
|
|
|
|
|
|
|
// Mock openExternal
|
|
|
|
const nextExternalUrlPromise = electronApp.evaluate(({ shell }) => {
|
|
|
|
return new Promise<string>(resolve => {
|
|
|
|
const openExternal = async (url: string) => {
|
|
|
|
resolve(url);
|
|
|
|
};
|
|
|
|
shell.openExternal = openExternal;
|
|
|
|
});
|
|
|
|
});
|
2023-10-22 03:52:06 -07:00
|
|
|
|
2024-10-15 08:38:33 -07:00
|
|
|
// Create a test link
|
|
|
|
const testLinkTitle = 'This is a test link!';
|
|
|
|
const linkHref = 'https://joplinapp.org/';
|
|
|
|
|
|
|
|
await mainWindow.evaluate(({ testLinkTitle, linkHref, target }) => {
|
|
|
|
const testLink = document.createElement('a');
|
|
|
|
testLink.textContent = testLinkTitle;
|
|
|
|
testLink.onclick = () => {
|
|
|
|
// We need to navigate by setting location.href -- clicking on a link
|
|
|
|
// directly within the main window (i.e. not in a PDF viewer) doesn't
|
|
|
|
// navigate.
|
|
|
|
location.href = linkHref;
|
2023-10-22 03:52:06 -07:00
|
|
|
};
|
2024-10-15 08:38:33 -07:00
|
|
|
testLink.href = '#';
|
|
|
|
|
|
|
|
// Display on top of everything
|
|
|
|
testLink.style.zIndex = '99999';
|
|
|
|
testLink.style.position = 'fixed';
|
|
|
|
testLink.style.top = '0';
|
|
|
|
testLink.style.left = '0';
|
|
|
|
if (target) {
|
|
|
|
testLink.target = target;
|
|
|
|
}
|
2023-10-22 03:52:06 -07:00
|
|
|
|
2024-10-15 08:38:33 -07:00
|
|
|
document.body.appendChild(testLink);
|
|
|
|
}, { testLinkTitle, linkHref, target });
|
|
|
|
|
|
|
|
const testLink = mainWindow.getByText(testLinkTitle);
|
|
|
|
await expect(testLink).toBeVisible();
|
|
|
|
await testLink.click({ noWaitAfter: true });
|
|
|
|
|
|
|
|
expect(await nextExternalUrlPromise).toBe(linkHref);
|
|
|
|
});
|
|
|
|
}
|
2023-10-31 08:05:28 -07:00
|
|
|
|
|
|
|
test('should start in safe mode if profile-dir/force-safe-mode-on-next-start exists', async ({ profileDirectory }) => {
|
|
|
|
await writeFile(join(profileDirectory, 'force-safe-mode-on-next-start'), 'true', 'utf8');
|
|
|
|
|
|
|
|
// We need to write to the force-safe-mode file before opening the Electron app.
|
|
|
|
// Open the app ourselves:
|
2023-11-12 07:06:32 -08:00
|
|
|
const startupArgs = createStartupArgs(profileDirectory);
|
2023-10-31 08:05:28 -07:00
|
|
|
const electronApp = await electron.launch({ args: startupArgs });
|
2023-11-12 07:06:32 -08:00
|
|
|
const mainWindow = await firstNonDevToolsWindow(electronApp);
|
2023-10-31 08:05:28 -07:00
|
|
|
|
|
|
|
const safeModeDisableLink = mainWindow.getByText('Disable safe mode and restart');
|
|
|
|
await safeModeDisableLink.waitFor();
|
|
|
|
await expect(safeModeDisableLink).toBeInViewport();
|
|
|
|
|
|
|
|
await electronApp.close();
|
|
|
|
});
|
2023-10-13 07:32:10 -07:00
|
|
|
});
|
2023-10-31 08:05:28 -07:00
|
|
|
|