Commit 9b440585 authored by Himanshu Kapoor's avatar Himanshu Kapoor Committed by Enrique Alcántara

Show warning for markdown structure changes

Show a warning in content editor when on editing a document, the
markdown style of the original document might not match the
resulting document.

Changelog: added
parent 75e4516c
...@@ -3,7 +3,7 @@ import { GlLoadingIcon } from '@gitlab/ui'; ...@@ -3,7 +3,7 @@ import { GlLoadingIcon } from '@gitlab/ui';
import { EditorContent as TiptapEditorContent } from '@tiptap/vue-2'; import { EditorContent as TiptapEditorContent } from '@tiptap/vue-2';
import { LOADING_CONTENT_EVENT, LOADING_SUCCESS_EVENT, LOADING_ERROR_EVENT } from '../constants'; import { LOADING_CONTENT_EVENT, LOADING_SUCCESS_EVENT, LOADING_ERROR_EVENT } from '../constants';
import { createContentEditor } from '../services/create_content_editor'; import { createContentEditor } from '../services/create_content_editor';
import ContentEditorError from './content_editor_error.vue'; import ContentEditorAlert from './content_editor_alert.vue';
import ContentEditorProvider from './content_editor_provider.vue'; import ContentEditorProvider from './content_editor_provider.vue';
import EditorStateObserver from './editor_state_observer.vue'; import EditorStateObserver from './editor_state_observer.vue';
import FormattingBubbleMenu from './formatting_bubble_menu.vue'; import FormattingBubbleMenu from './formatting_bubble_menu.vue';
...@@ -12,7 +12,7 @@ import TopToolbar from './top_toolbar.vue'; ...@@ -12,7 +12,7 @@ import TopToolbar from './top_toolbar.vue';
export default { export default {
components: { components: {
GlLoadingIcon, GlLoadingIcon,
ContentEditorError, ContentEditorAlert,
ContentEditorProvider, ContentEditorProvider,
TiptapEditorContent, TiptapEditorContent,
TopToolbar, TopToolbar,
...@@ -92,7 +92,7 @@ export default { ...@@ -92,7 +92,7 @@ export default {
<content-editor-provider :content-editor="contentEditor"> <content-editor-provider :content-editor="contentEditor">
<div> <div>
<editor-state-observer @docUpdate="notifyChange" @focus="focus" @blur="blur" /> <editor-state-observer @docUpdate="notifyChange" @focus="focus" @blur="blur" />
<content-editor-error /> <content-editor-alert />
<div <div
data-testid="content-editor" data-testid="content-editor"
data-qa-selector="content_editor_container" data-qa-selector="content_editor_container"
......
...@@ -9,23 +9,25 @@ export default { ...@@ -9,23 +9,25 @@ export default {
}, },
data() { data() {
return { return {
error: null, message: null,
variant: 'danger',
}; };
}, },
methods: { methods: {
displayError({ error }) { displayAlert({ message, variant }) {
this.error = error; this.message = message;
this.variant = variant;
}, },
dismissError() { dismissAlert() {
this.error = null; this.message = null;
}, },
}, },
}; };
</script> </script>
<template> <template>
<editor-state-observer @error="displayError"> <editor-state-observer @alert="displayAlert">
<gl-alert v-if="error" class="gl-mb-6" variant="danger" @dismiss="dismissError"> <gl-alert v-if="message" class="gl-mb-6" :variant="variant" @dismiss="dismissAlert">
{{ error }} {{ message }}
</gl-alert> </gl-alert>
</editor-state-observer> </editor-state-observer>
</template> </template>
...@@ -7,7 +7,7 @@ export const tiptapToComponentMap = { ...@@ -7,7 +7,7 @@ export const tiptapToComponentMap = {
transaction: 'transaction', transaction: 'transaction',
focus: 'focus', focus: 'focus',
blur: 'blur', blur: 'blur',
error: 'error', alert: 'alert',
}; };
const getComponentEventName = (tiptapEventName) => tiptapToComponentMap[tiptapEventName]; const getComponentEventName = (tiptapEventName) => tiptapToComponentMap[tiptapEventName];
......
...@@ -26,8 +26,8 @@ export default { ...@@ -26,8 +26,8 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
getPos: { node: {
type: Function, type: Object,
required: true, required: true,
}, },
}, },
...@@ -61,7 +61,17 @@ export default { ...@@ -61,7 +61,17 @@ export default {
const { state } = this.editor; const { state } = this.editor;
const { $cursor } = state.selection; const { $cursor } = state.selection;
this.displayActionsDropdown = $cursor?.pos - $cursor?.parentOffset - 1 === this.getPos(); if (!$cursor) return;
this.displayActionsDropdown = false;
for (let level = 0; level < $cursor.depth; level += 1) {
if ($cursor.node(level) === this.node) {
this.displayActionsDropdown = true;
break;
}
}
if (this.displayActionsDropdown) { if (this.displayActionsDropdown) {
this.selectedRect = getSelectedRect(state); this.selectedRect = getSelectedRect(state);
} }
...@@ -99,7 +109,11 @@ export default { ...@@ -99,7 +109,11 @@ export default {
:as="cellType" :as="cellType"
@click="hideDropdown" @click="hideDropdown"
> >
<span v-if="displayActionsDropdown" class="gl-absolute gl-right-0 gl-top-0"> <span
v-if="displayActionsDropdown"
contenteditable="false"
class="gl-absolute gl-right-0 gl-top-0"
>
<gl-dropdown <gl-dropdown
ref="dropdown" ref="dropdown"
dropup dropup
......
...@@ -11,8 +11,8 @@ export default { ...@@ -11,8 +11,8 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
getPos: { node: {
type: Function, type: Object,
required: true, required: true,
}, },
}, },
......
...@@ -11,8 +11,8 @@ export default { ...@@ -11,8 +11,8 @@ export default {
type: Object, type: Object,
required: true, required: true,
}, },
getPos: { node: {
type: Function, type: Object,
required: true, required: true,
}, },
}, },
......
export { Table as default } from '@tiptap/extension-table'; import { Table } from '@tiptap/extension-table';
import { debounce } from 'lodash';
import { __ } from '~/locale';
import { getMarkdownSource } from '../services/markdown_sourcemap';
import { shouldRenderHTMLTable } from '../services/serialization_helpers';
let alertShown = false;
const onUpdate = debounce((editor) => {
if (alertShown) return;
editor.state.doc.descendants((node) => {
if (node.type.name === 'table' && node.attrs.isMarkdown && shouldRenderHTMLTable(node)) {
editor.emit('alert', {
message: __(
'The content editor may change the markdown formatting style of the document, which may not match your original markdown style.',
),
variant: 'warning',
});
alertShown = true;
return false;
}
return true;
});
}, 1000);
export default Table.extend({
addAttributes() {
return {
isMarkdown: {
default: null,
parseHTML: (element) => Boolean(getMarkdownSource(element)),
},
};
},
onUpdate({ editor }) {
onUpdate(editor);
},
});
import { TableCell } from '@tiptap/extension-table-cell'; import { TableCell } from '@tiptap/extension-table-cell';
import { VueNodeViewRenderer } from '@tiptap/vue-2'; import { VueNodeViewRenderer } from '@tiptap/vue-2';
import TableCellBodyWrapper from '../components/wrappers/table_cell_body.vue'; import TableCellBodyWrapper from '../components/wrappers/table_cell_body.vue';
import { isBlockTablesFeatureEnabled } from '../services/feature_flags';
export default TableCell.extend({ export default TableCell.extend({
content: isBlockTablesFeatureEnabled() ? 'block+' : 'inline*', content: 'block+',
addNodeView() { addNodeView() {
return VueNodeViewRenderer(TableCellBodyWrapper); return VueNodeViewRenderer(TableCellBodyWrapper);
......
import { TableHeader } from '@tiptap/extension-table-header'; import { TableHeader } from '@tiptap/extension-table-header';
import { VueNodeViewRenderer } from '@tiptap/vue-2'; import { VueNodeViewRenderer } from '@tiptap/vue-2';
import TableCellHeaderWrapper from '../components/wrappers/table_cell_header.vue'; import TableCellHeaderWrapper from '../components/wrappers/table_cell_header.vue';
import { isBlockTablesFeatureEnabled } from '../services/feature_flags';
export default TableHeader.extend({ export default TableHeader.extend({
content: isBlockTablesFeatureEnabled() ? 'block+' : 'inline*', content: 'block+',
addNodeView() { addNodeView() {
return VueNodeViewRenderer(TableCellHeaderWrapper); return VueNodeViewRenderer(TableCellHeaderWrapper);
}, },
......
export function isBlockTablesFeatureEnabled() {
return gon.features?.contentEditorBlockTables;
}
import { uniq } from 'lodash'; import { uniq } from 'lodash';
import { isBlockTablesFeatureEnabled } from './feature_flags';
const defaultAttrs = { const defaultAttrs = {
td: { colspan: 1, rowspan: 1, colwidth: null }, td: { colspan: 1, rowspan: 1, colwidth: null },
...@@ -75,7 +74,7 @@ function getChildren(node) { ...@@ -75,7 +74,7 @@ function getChildren(node) {
return children; return children;
} }
function shouldRenderHTMLTable(table) { export function shouldRenderHTMLTable(table) {
const { rows, cells } = getRowsAndCells(table); const { rows, cells } = getRowsAndCells(table);
const cellChildCount = Math.max(...cells.map((cell) => cell.childCount)); const cellChildCount = Math.max(...cells.map((cell) => cell.childCount));
...@@ -282,11 +281,6 @@ export function renderOrderedList(state, node) { ...@@ -282,11 +281,6 @@ export function renderOrderedList(state, node) {
} }
export function renderTableCell(state, node) { export function renderTableCell(state, node) {
if (!isBlockTablesFeatureEnabled()) {
state.renderInline(node);
return;
}
if (!isInBlockTable(node) || containsParagraphWithOnlyText(node)) { if (!isInBlockTable(node) || containsParagraphWithOnlyText(node)) {
state.renderInline(node.child(0)); state.renderInline(node.child(0));
} else { } else {
...@@ -303,9 +297,7 @@ export function renderTableRow(state, node) { ...@@ -303,9 +297,7 @@ export function renderTableRow(state, node) {
} }
export function renderTable(state, node) { export function renderTable(state, node) {
if (isBlockTablesFeatureEnabled()) { setIsInBlockTable(node, shouldRenderHTMLTable(node));
setIsInBlockTable(node, shouldRenderHTMLTable(node));
}
if (isInBlockTable(node)) renderTagOpen(state, 'table'); if (isInBlockTable(node)) renderTagOpen(state, 'table');
...@@ -317,9 +309,7 @@ export function renderTable(state, node) { ...@@ -317,9 +309,7 @@ export function renderTable(state, node) {
state.closeBlock(node); state.closeBlock(node);
state.flushClose(); state.flushClose();
if (isBlockTablesFeatureEnabled()) { unsetIsInBlockTable(node);
unsetIsInBlockTable(node);
}
} }
export function renderHardBreak(state, node, parent, index) { export function renderHardBreak(state, node, parent, index) {
......
...@@ -72,8 +72,9 @@ const uploadImage = async ({ editor, file, uploadsPath, renderMarkdown }) => { ...@@ -72,8 +72,9 @@ const uploadImage = async ({ editor, file, uploadsPath, renderMarkdown }) => {
); );
} catch (e) { } catch (e) {
editor.commands.deleteRange({ from: position, to: position + 1 }); editor.commands.deleteRange({ from: position, to: position + 1 });
editor.emit('error', { editor.emit('alert', {
error: __('An error occurred while uploading the image. Please try again.'), message: __('An error occurred while uploading the image. Please try again.'),
variant: 'danger',
}); });
} }
}; };
...@@ -102,8 +103,9 @@ const uploadAttachment = async ({ editor, file, uploadsPath, renderMarkdown }) = ...@@ -102,8 +103,9 @@ const uploadAttachment = async ({ editor, file, uploadsPath, renderMarkdown }) =
); );
} catch (e) { } catch (e) {
editor.commands.deleteRange({ from, to: from + 1 }); editor.commands.deleteRange({ from, to: from + 1 });
editor.emit('error', { editor.emit('alert', {
error: __('An error occurred while uploading the file. Please try again.'), message: __('An error occurred while uploading the file. Please try again.'),
variant: 'danger',
}); });
} }
}; };
......
...@@ -7,9 +7,5 @@ class Projects::WikisController < Projects::ApplicationController ...@@ -7,9 +7,5 @@ class Projects::WikisController < Projects::ApplicationController
alias_method :container, :project alias_method :container, :project
before_action do
push_frontend_feature_flag(:content_editor_block_tables, @project, default_enabled: :yaml)
end
feature_category :wiki feature_category :wiki
end end
---
name: content_editor_block_tables
introduced_by_url: https://gitlab.com/gitlab-org/gitlab/-/merge_requests/66187
rollout_issue_url: https://gitlab.com/gitlab-org/gitlab/-/issues/338937
milestone: '14.3'
type: development
group: group::editor
default_enabled: false
...@@ -34369,6 +34369,9 @@ msgstr "" ...@@ -34369,6 +34369,9 @@ msgstr ""
msgid "The contact does not belong to the same group as the issue" msgid "The contact does not belong to the same group as the issue"
msgstr "" msgstr ""
msgid "The content editor may change the markdown formatting style of the document, which may not match your original markdown style."
msgstr ""
msgid "The content of this page is not encoded in UTF-8. Edits can only be made via the Git repository." msgid "The content of this page is not encoded in UTF-8. Edits can only be made via the Git repository."
msgstr "" msgstr ""
......
import { GlAlert } from '@gitlab/ui'; import { GlAlert } from '@gitlab/ui';
import { nextTick } from 'vue'; import { nextTick } from 'vue';
import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
import ContentEditorError from '~/content_editor/components/content_editor_error.vue'; import ContentEditorAlert from '~/content_editor/components/content_editor_alert.vue';
import EditorStateObserver from '~/content_editor/components/editor_state_observer.vue'; import EditorStateObserver from '~/content_editor/components/editor_state_observer.vue';
import { createTestEditor, emitEditorEvent } from '../test_utils'; import { createTestEditor, emitEditorEvent } from '../test_utils';
describe('content_editor/components/content_editor_error', () => { describe('content_editor/components/content_editor_alert', () => {
let wrapper; let wrapper;
let tiptapEditor; let tiptapEditor;
...@@ -14,7 +14,7 @@ describe('content_editor/components/content_editor_error', () => { ...@@ -14,7 +14,7 @@ describe('content_editor/components/content_editor_error', () => {
const createWrapper = async () => { const createWrapper = async () => {
tiptapEditor = createTestEditor(); tiptapEditor = createTestEditor();
wrapper = shallowMountExtended(ContentEditorError, { wrapper = shallowMountExtended(ContentEditorAlert, {
provide: { provide: {
tiptapEditor, tiptapEditor,
}, },
...@@ -28,22 +28,28 @@ describe('content_editor/components/content_editor_error', () => { ...@@ -28,22 +28,28 @@ describe('content_editor/components/content_editor_error', () => {
wrapper.destroy(); wrapper.destroy();
}); });
it('renders error when content editor emits an error event', async () => { it.each`
const error = 'error message'; variant | message
${'danger'} | ${'An error occurred'}
${'warning'} | ${'A warning'}
`(
'renders error when content editor emits an error event for variant: $variant',
async ({ message, variant }) => {
createWrapper();
createWrapper(); await emitEditorEvent({ tiptapEditor, event: 'alert', params: { message, variant } });
await emitEditorEvent({ tiptapEditor, event: 'error', params: { error } });
expect(findErrorAlert().text()).toBe(error); expect(findErrorAlert().text()).toBe(message);
}); expect(findErrorAlert().attributes().variant).toBe(variant);
},
);
it('allows dismissing the error', async () => { it('allows dismissing the error', async () => {
const error = 'error message'; const message = 'error message';
createWrapper(); createWrapper();
await emitEditorEvent({ tiptapEditor, event: 'error', params: { error } }); await emitEditorEvent({ tiptapEditor, event: 'alert', params: { message } });
findErrorAlert().vm.$emit('dismiss'); findErrorAlert().vm.$emit('dismiss');
......
...@@ -3,7 +3,7 @@ import { EditorContent } from '@tiptap/vue-2'; ...@@ -3,7 +3,7 @@ import { EditorContent } from '@tiptap/vue-2';
import { nextTick } from 'vue'; import { nextTick } from 'vue';
import { shallowMountExtended } from 'helpers/vue_test_utils_helper'; import { shallowMountExtended } from 'helpers/vue_test_utils_helper';
import ContentEditor from '~/content_editor/components/content_editor.vue'; import ContentEditor from '~/content_editor/components/content_editor.vue';
import ContentEditorError from '~/content_editor/components/content_editor_error.vue'; import ContentEditorAlert from '~/content_editor/components/content_editor_alert.vue';
import ContentEditorProvider from '~/content_editor/components/content_editor_provider.vue'; import ContentEditorProvider from '~/content_editor/components/content_editor_provider.vue';
import EditorStateObserver from '~/content_editor/components/editor_state_observer.vue'; import EditorStateObserver from '~/content_editor/components/editor_state_observer.vue';
import FormattingBubbleMenu from '~/content_editor/components/formatting_bubble_menu.vue'; import FormattingBubbleMenu from '~/content_editor/components/formatting_bubble_menu.vue';
...@@ -111,10 +111,10 @@ describe('ContentEditor', () => { ...@@ -111,10 +111,10 @@ describe('ContentEditor', () => {
]); ]);
}); });
it('renders content_editor_error component', () => { it('renders content_editor_alert component', () => {
createWrapper(); createWrapper();
expect(wrapper.findComponent(ContentEditorError).exists()).toBe(true); expect(wrapper.findComponent(ContentEditorAlert).exists()).toBe(true);
}); });
describe('when loading content', () => { describe('when loading content', () => {
......
...@@ -11,13 +11,13 @@ jest.mock('prosemirror-tables'); ...@@ -11,13 +11,13 @@ jest.mock('prosemirror-tables');
describe('content/components/wrappers/table_cell_base', () => { describe('content/components/wrappers/table_cell_base', () => {
let wrapper; let wrapper;
let editor; let editor;
let getPos; let node;
const createWrapper = async (propsData = { cellType: 'td' }) => { const createWrapper = async (propsData = { cellType: 'td' }) => {
wrapper = shallowMountExtended(TableCellBaseWrapper, { wrapper = shallowMountExtended(TableCellBaseWrapper, {
propsData: { propsData: {
editor, editor,
getPos, node,
...propsData, ...propsData,
}, },
}); });
...@@ -36,7 +36,7 @@ describe('content/components/wrappers/table_cell_base', () => { ...@@ -36,7 +36,7 @@ describe('content/components/wrappers/table_cell_base', () => {
const setCurrentPositionInCell = () => { const setCurrentPositionInCell = () => {
const { $cursor } = editor.state.selection; const { $cursor } = editor.state.selection;
getPos.mockReturnValue($cursor.pos - $cursor.parentOffset - 1); jest.spyOn($cursor, 'node').mockReturnValue(node);
}; };
const mockDropdownHide = () => { const mockDropdownHide = () => {
/* /*
...@@ -48,7 +48,7 @@ describe('content/components/wrappers/table_cell_base', () => { ...@@ -48,7 +48,7 @@ describe('content/components/wrappers/table_cell_base', () => {
}; };
beforeEach(() => { beforeEach(() => {
getPos = jest.fn(); node = {};
editor = createTestEditor({}); editor = createTestEditor({});
}); });
......
...@@ -6,19 +6,19 @@ import { createTestEditor } from '../../test_utils'; ...@@ -6,19 +6,19 @@ import { createTestEditor } from '../../test_utils';
describe('content/components/wrappers/table_cell_body', () => { describe('content/components/wrappers/table_cell_body', () => {
let wrapper; let wrapper;
let editor; let editor;
let getPos; let node;
const createWrapper = async () => { const createWrapper = async () => {
wrapper = shallowMount(TableCellBodyWrapper, { wrapper = shallowMount(TableCellBodyWrapper, {
propsData: { propsData: {
editor, editor,
getPos, node,
}, },
}); });
}; };
beforeEach(() => { beforeEach(() => {
getPos = jest.fn(); node = {};
editor = createTestEditor({}); editor = createTestEditor({});
}); });
...@@ -30,7 +30,7 @@ describe('content/components/wrappers/table_cell_body', () => { ...@@ -30,7 +30,7 @@ describe('content/components/wrappers/table_cell_body', () => {
createWrapper(); createWrapper();
expect(wrapper.findComponent(TableCellBaseWrapper).props()).toEqual({ expect(wrapper.findComponent(TableCellBaseWrapper).props()).toEqual({
editor, editor,
getPos, node,
cellType: 'td', cellType: 'td',
}); });
}); });
......
...@@ -6,19 +6,19 @@ import { createTestEditor } from '../../test_utils'; ...@@ -6,19 +6,19 @@ import { createTestEditor } from '../../test_utils';
describe('content/components/wrappers/table_cell_header', () => { describe('content/components/wrappers/table_cell_header', () => {
let wrapper; let wrapper;
let editor; let editor;
let getPos; let node;
const createWrapper = async () => { const createWrapper = async () => {
wrapper = shallowMount(TableCellHeaderWrapper, { wrapper = shallowMount(TableCellHeaderWrapper, {
propsData: { propsData: {
editor, editor,
getPos, node,
}, },
}); });
}; };
beforeEach(() => { beforeEach(() => {
getPos = jest.fn(); node = {};
editor = createTestEditor({}); editor = createTestEditor({});
}); });
...@@ -30,7 +30,7 @@ describe('content/components/wrappers/table_cell_header', () => { ...@@ -30,7 +30,7 @@ describe('content/components/wrappers/table_cell_header', () => {
createWrapper(); createWrapper();
expect(wrapper.findComponent(TableCellBaseWrapper).props()).toEqual({ expect(wrapper.findComponent(TableCellBaseWrapper).props()).toEqual({
editor, editor,
getPos, node,
cellType: 'th', cellType: 'th',
}); });
}); });
......
...@@ -157,11 +157,11 @@ describe('content_editor/extensions/attachment', () => { ...@@ -157,11 +157,11 @@ describe('content_editor/extensions/attachment', () => {
}); });
}); });
it('emits an error event that includes an error message', (done) => { it('emits an alert event that includes an error message', (done) => {
tiptapEditor.commands.uploadAttachment({ file: imageFile }); tiptapEditor.commands.uploadAttachment({ file: imageFile });
tiptapEditor.on('error', ({ error }) => { tiptapEditor.on('alert', ({ message }) => {
expect(error).toBe('An error occurred while uploading the image. Please try again.'); expect(message).toBe('An error occurred while uploading the image. Please try again.');
done(); done();
}); });
}); });
...@@ -233,11 +233,11 @@ describe('content_editor/extensions/attachment', () => { ...@@ -233,11 +233,11 @@ describe('content_editor/extensions/attachment', () => {
}); });
}); });
it('emits an error event that includes an error message', (done) => { it('emits an alert event that includes an error message', (done) => {
tiptapEditor.commands.uploadAttachment({ file: attachmentFile }); tiptapEditor.commands.uploadAttachment({ file: attachmentFile });
tiptapEditor.on('error', ({ error }) => { tiptapEditor.on('alert', ({ message }) => {
expect(error).toBe('An error occurred while uploading the file. Please try again.'); expect(message).toBe('An error occurred while uploading the file. Please try again.');
done(); done();
}); });
}); });
......
import Bold from '~/content_editor/extensions/bold';
import BulletList from '~/content_editor/extensions/bullet_list';
import ListItem from '~/content_editor/extensions/list_item';
import Table from '~/content_editor/extensions/table';
import TableCell from '~/content_editor/extensions/table_cell';
import TableRow from '~/content_editor/extensions/table_row';
import TableHeader from '~/content_editor/extensions/table_header';
import { createTestEditor, createDocBuilder } from '../test_utils';
describe('content_editor/extensions/table', () => {
let tiptapEditor;
let doc;
let p;
let table;
let tableHeader;
let tableCell;
let tableRow;
let initialDoc;
let mockAlert;
beforeEach(() => {
tiptapEditor = createTestEditor({
extensions: [Table, TableCell, TableRow, TableHeader, BulletList, Bold, ListItem],
});
({
builders: { doc, p, table, tableCell, tableHeader, tableRow },
} = createDocBuilder({
tiptapEditor,
names: {
bold: { markType: Bold.name },
table: { nodeType: Table.name },
tableHeader: { nodeType: TableHeader.name },
tableCell: { nodeType: TableCell.name },
tableRow: { nodeType: TableRow.name },
bulletList: { nodeType: BulletList.name },
listItem: { nodeType: ListItem.name },
},
}));
initialDoc = doc(
table(
{ isMarkdown: true },
tableRow(tableHeader(p('This is')), tableHeader(p('a table'))),
tableRow(tableCell(p('this is')), tableCell(p('the first row'))),
),
);
mockAlert = jest.fn();
});
it('triggers a warning (just once) if the table is markdown, but the changes in the document will render an HTML table instead', () => {
tiptapEditor.commands.setContent(initialDoc.toJSON());
tiptapEditor.on('alert', mockAlert);
tiptapEditor.commands.setTextSelection({ from: 20, to: 22 });
tiptapEditor.commands.toggleBulletList();
jest.advanceTimersByTime(1001);
expect(mockAlert).toHaveBeenCalled();
mockAlert.mockReset();
tiptapEditor.commands.setTextSelection({ from: 4, to: 6 });
tiptapEditor.commands.toggleBulletList();
jest.advanceTimersByTime(1001);
expect(mockAlert).not.toHaveBeenCalled();
});
it('does not trigger a warning if the table is markdown, and the changes in the document can generate a markdown table', () => {
tiptapEditor.commands.setContent(initialDoc.toJSON());
tiptapEditor.on('alert', mockAlert);
tiptapEditor.commands.setTextSelection({ from: 20, to: 22 });
tiptapEditor.commands.toggleBold();
jest.advanceTimersByTime(1001);
expect(mockAlert).not.toHaveBeenCalled();
});
it('does not trigger any warnings if the table is not markdown', () => {
initialDoc = doc(
table(
tableRow(tableHeader(p('This is')), tableHeader(p('a table'))),
tableRow(tableCell(p('this is')), tableCell(p('the first row'))),
),
);
tiptapEditor.commands.setContent(initialDoc.toJSON());
tiptapEditor.on('alert', mockAlert);
tiptapEditor.commands.setTextSelection({ from: 20, to: 22 });
tiptapEditor.commands.toggleBulletList();
jest.advanceTimersByTime(1001);
expect(mockAlert).not.toHaveBeenCalled();
});
});
...@@ -34,10 +34,6 @@ import { createTestEditor, createDocBuilder } from '../test_utils'; ...@@ -34,10 +34,6 @@ import { createTestEditor, createDocBuilder } from '../test_utils';
jest.mock('~/emoji'); jest.mock('~/emoji');
jest.mock('~/content_editor/services/feature_flags', () => ({
isBlockTablesFeatureEnabled: jest.fn().mockReturnValue(true),
}));
const tiptapEditor = createTestEditor({ const tiptapEditor = createTestEditor({
extensions: [ extensions: [
Blockquote, Blockquote,
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment