import { nextTick } from 'vue'; import { setHTMLFixture, resetHTMLFixture } from 'helpers/fixtures'; import { setTestTimeout } from 'helpers/timeout'; import waitForPromises from 'helpers/wait_for_promises'; import { waitForText } from 'helpers/wait_for_text'; import { createCommitId } from 'test_helpers/factories/commit_id'; import { useOverclockTimers } from 'test_helpers/utils/overclock_timers'; import * as ideHelper from './helpers/ide_helper'; import startWebIDE from './helpers/start'; describe('WebIDE', () => { useOverclockTimers(); let vm; let container; beforeEach(() => { // For some reason these tests were timing out in CI. // We will investigate in https://gitlab.com/gitlab-org/gitlab/-/issues/298714 setTestTimeout(20000); setHTMLFixture('<div class="webide-container"></div>'); container = document.querySelector('.webide-container'); }); afterEach(() => { vm.$destroy(); vm = null; resetHTMLFixture(); }); it('user commits changes', async () => { vm = startWebIDE(container); await ideHelper.createFile('foo/bar/test.txt', 'Lorem ipsum dolar sit'); await ideHelper.deleteFile('foo/bar/.gitkeep'); await ideHelper.commit(); const commitId = createCommitId(1); const commitShortId = commitId.slice(0, 8); await waitForText('All changes are committed'); await waitForText(commitShortId); expect(mockServer.db.branches.findBy({ name: 'master' }).commit).toMatchObject({ short_id: commitShortId, id: commitId, message: 'Update foo/bar/test.txt\nDeleted foo/bar/.gitkeep', __actions: [ { action: 'create', content: 'Lorem ipsum dolar sit\n', encoding: 'text', file_path: 'foo/bar/test.txt', last_commit_id: '', }, { action: 'delete', encoding: 'text', file_path: 'foo/bar/.gitkeep', }, ], }); }); it('user commits changes to new branch', async () => { vm = startWebIDE(container); expect(window.location.pathname).toBe('/-/ide/project/gitlab-test/lorem-ipsum/tree/master/-/'); await ideHelper.updateFile('README.md', 'Lorem dolar si amit\n'); await ideHelper.commit({ newBranch: true, newMR: false, newBranchName: 'test-hello-world' }); await waitForText('All changes are committed'); // Wait for IDE to load new commit await waitForText('10000000', document.querySelector('.ide-status-bar')); // It's important that the new branch is now in the route expect(window.location.pathname).toBe( '/-/ide/project/gitlab-test/lorem-ipsum/blob/test-hello-world/-/README.md', ); }); it('user adds file that starts with +', async () => { vm = startWebIDE(container); await ideHelper.createFile('+test', 'Hello world!'); await ideHelper.openFile('+test'); // Wait for monaco things await waitForPromises(); // Assert that +test is the only open tab const tabs = Array.from(document.querySelectorAll('.multi-file-tab')); expect(tabs.map((x) => x.textContent.trim())).toEqual(['+test']); }); describe('editor info', () => { let statusBar; let editor; beforeEach(async () => { vm = startWebIDE(container); await ideHelper.openFile('README.md'); editor = await ideHelper.waitForMonacoEditor(); statusBar = ideHelper.getStatusBar(); }); it('shows line position and type', () => { expect(statusBar).toHaveText('1:1'); expect(statusBar).toHaveText('markdown'); }); it('persists viewer', async () => { const markdownPreview = 'test preview_markdown result'; mockServer.post('/:namespace/:project/preview_markdown', () => ({ body: markdownPreview, })); await ideHelper.openFile('README.md'); ideHelper.clickPreviewMarkdown(); const el = await waitForText(markdownPreview); expect(el).toHaveText(markdownPreview); // Need to wait for monaco editor to load so it doesn't through errors on dispose await ideHelper.openFile('.gitignore'); await ideHelper.waitForEditorModelChange(editor); await ideHelper.openFile('README.md'); await ideHelper.waitForEditorModelChange(editor); expect(el).toHaveText(markdownPreview); }); describe('when editor position changes', () => { beforeEach(async () => { editor.setPosition({ lineNumber: 4, column: 10 }); await nextTick(); }); it('shows new line position', () => { expect(statusBar).not.toHaveText('1:1'); expect(statusBar).toHaveText('4:10'); }); it('updates after rename', async () => { await ideHelper.renameFile('README.md', 'READMEZ.txt'); await ideHelper.waitForEditorModelChange(editor); await nextTick(); expect(statusBar).toHaveText('1:1'); expect(statusBar).toHaveText('plaintext'); }); it('persists position after opening then rename', async () => { await ideHelper.openFile('files/js/application.js'); await ideHelper.waitForEditorModelChange(editor); await ideHelper.renameFile('README.md', 'READING_RAINBOW.md'); await ideHelper.openFile('READING_RAINBOW.md'); await ideHelper.waitForEditorModelChange(editor); expect(statusBar).toHaveText('4:10'); expect(statusBar).toHaveText('markdown'); }); it('persists position after closing', async () => { await ideHelper.closeFile('README.md'); await ideHelper.openFile('README.md'); await ideHelper.waitForMonacoEditor(); await nextTick(); expect(statusBar).toHaveText('4:10'); expect(statusBar).toHaveText('markdown'); }); }); }); });