Unit tests for StreamBlock

pull/6931/head
Karl Hobley 2021-01-22 14:23:48 +00:00 zatwierdzone przez Matt Westcott
rodzic 889689f754
commit f0635165b3
2 zmienionych plików z 353 dodań i 15 usunięć

Wyświetl plik

@ -100,6 +100,148 @@ exports[`telepath: wagtail.blocks.ListBlock it renders correctly 1`] = `
</div>" </div>"
`; `;
exports[`telepath: wagtail.blocks.StreamBlock it renders correctly 1`] = `
"<div class=\\"c-sf-container \\">
<input type=\\"hidden\\" name=\\"the-prefix-count\\" data-streamfield-stream-count=\\"\\" value=\\"2\\">
<div data-streamfield-stream-container=\\"\\"><div>
<button data-streamblock-menu-open=\\"\\" type=\\"button\\" title=\\"Add\\" class=\\"c-sf-add-button c-sf-add-button--visible\\">
<i aria-hidden=\\"true\\">+</i>
</button>
<div data-streamblock-menu-outer=\\"\\" style=\\"display: none;\\" aria-hidden=\\"true\\">
<div data-streamblock-menu-inner=\\"\\" class=\\"c-sf-add-panel\\"><div class=\\"c-sf-add-panel__grid\\"><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_a\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-placeholder\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_b\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-pilcrow\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button></div></div>
</div>
</div><div aria-hidden=\\"false\\">
<input type=\\"hidden\\" data-streamblock-deleted=\\"\\" name=\\"the-prefix-0-deleted\\" value=\\"\\">
<input type=\\"hidden\\" data-streamblock-index=\\"\\" name=\\"the-prefix-0-order\\" value=\\"0\\">
<input type=\\"hidden\\" name=\\"the-prefix-0-type\\" value=\\"test_block_a\\">
<input type=\\"hidden\\" name=\\"the-prefix-0-id\\" value=\\"1\\">
<div>
<div class=\\"c-sf-container__block-container\\">
<div class=\\"c-sf-block\\">
<div class=\\"c-sf-block__header\\">
<span class=\\"c-sf-block__header__icon\\">
<i class=\\"icon icon-placeholder\\"></i>
</span>
<h3 class=\\"c-sf-block__header__title\\"></h3>
<div class=\\"c-sf-block__actions\\">
<span class=\\"c-sf-block__type\\">Test Block B</span>
<button type=\\"button\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Move up' %}\\">
<i class=\\"icon icon-arrow-up\\" aria-hidden=\\"true\\"></i>
</button>
<button type=\\"button\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Move down' %}\\">
<i class=\\"icon icon-arrow-down\\" aria-hidden=\\"true\\"></i>
</button>
<button type=\\"button\\" data-streamblock-delete-button=\\"\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Delete' %}\\">
<i class=\\"icon icon-bin\\" aria-hidden=\\"true\\"></i>
</button>
</div>
</div>
<div class=\\"c-sf-block__content\\" aria-hidden=\\"false\\">
<div class=\\"c-sf-block__content-inner\\">
<div class=\\"field char_field widget-text_input fieldname-test_charblock\\">
<div class=\\"field-content\\">
<div class=\\"input\\">
<p name=\\"the-prefix-0-value\\" id=\\"the-prefix-0-value\\">Block A widget</p>
<span></span>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div><div>
<button data-streamblock-menu-open=\\"\\" type=\\"button\\" title=\\"Add\\" class=\\"c-sf-add-button c-sf-add-button--visible\\">
<i aria-hidden=\\"true\\">+</i>
</button>
<div data-streamblock-menu-outer=\\"\\" style=\\"display: none;\\" aria-hidden=\\"true\\">
<div data-streamblock-menu-inner=\\"\\" class=\\"c-sf-add-panel\\"><div class=\\"c-sf-add-panel__grid\\"><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_a\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-placeholder\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_b\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-pilcrow\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button></div></div>
</div>
</div><div aria-hidden=\\"false\\">
<input type=\\"hidden\\" data-streamblock-deleted=\\"\\" name=\\"the-prefix-1-deleted\\" value=\\"\\">
<input type=\\"hidden\\" data-streamblock-index=\\"\\" name=\\"the-prefix-1-order\\" value=\\"1\\">
<input type=\\"hidden\\" name=\\"the-prefix-1-type\\" value=\\"test_block_b\\">
<input type=\\"hidden\\" name=\\"the-prefix-1-id\\" value=\\"2\\">
<div>
<div class=\\"c-sf-container__block-container\\">
<div class=\\"c-sf-block\\">
<div class=\\"c-sf-block__header\\">
<span class=\\"c-sf-block__header__icon\\">
<i class=\\"icon icon-pilcrow\\"></i>
</span>
<h3 class=\\"c-sf-block__header__title\\"></h3>
<div class=\\"c-sf-block__actions\\">
<span class=\\"c-sf-block__type\\">Test Block B</span>
<button type=\\"button\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Move up' %}\\">
<i class=\\"icon icon-arrow-up\\" aria-hidden=\\"true\\"></i>
</button>
<button type=\\"button\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Move down' %}\\">
<i class=\\"icon icon-arrow-down\\" aria-hidden=\\"true\\"></i>
</button>
<button type=\\"button\\" data-streamblock-delete-button=\\"\\" class=\\"c-sf-block__actions__single\\" title=\\"{% trans 'Delete' %}\\">
<i class=\\"icon icon-bin\\" aria-hidden=\\"true\\"></i>
</button>
</div>
</div>
<div class=\\"c-sf-block__content\\" aria-hidden=\\"false\\">
<div class=\\"c-sf-block__content-inner\\">
<div class=\\"field char_field widget-admin_auto_height_text_input fieldname-test_textblock\\">
<div class=\\"field-content\\">
<div class=\\"input\\">
<p name=\\"the-prefix-1-value\\" id=\\"the-prefix-1-value\\">Block B widget</p>
<span></span>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div><div>
<button data-streamblock-menu-open=\\"\\" type=\\"button\\" title=\\"Add\\" class=\\"c-sf-add-button c-sf-add-button--visible\\">
<i aria-hidden=\\"true\\">+</i>
</button>
<div data-streamblock-menu-outer=\\"\\" style=\\"display: none;\\" aria-hidden=\\"true\\">
<div data-streamblock-menu-inner=\\"\\" class=\\"c-sf-add-panel\\"><div class=\\"c-sf-add-panel__grid\\"><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_a\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-placeholder\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button><button type=\\"button\\" class=\\"c-sf-button action-add-block-test_block_b\\">
<span class=\\"c-sf-button__icon\\">
<i class=\\"icon icon-pilcrow\\"></i>
</span>
<span class=\\"c-sf-button__label\\">Test Block B</span>
</button></div></div>
</div>
</div></div>
</div>"
`;
exports[`telepath: wagtail.blocks.StructBlock it renders correctly 1`] = ` exports[`telepath: wagtail.blocks.StructBlock it renders correctly 1`] = `
"<div class=\\"struct-block\\"> "<div class=\\"struct-block\\">
<div class=\\"field\\"> <div class=\\"field\\">

Wyświetl plik

@ -24,8 +24,8 @@ class DummyWidget {
$(placeholder).replaceWith(`<p name="${name}" id="${id}">${widgetName}</p>`); $(placeholder).replaceWith(`<p name="${name}" id="${id}">${widgetName}</p>`);
return { return {
setState(state) { setState(widgetName, state); }, setState(state) { setState(widgetName, state); },
getState() { getState(widgetName); return `state: ${widgetName}`; }, getState() { getState(widgetName); return `state: ${widgetName} - ${name}`; },
getValue() { getValue(widgetName); return `value: ${widgetName}`; }, getValue() { getValue(widgetName); return `value: ${widgetName} - ${name}`; },
focus() { focus(widgetName); }, focus() { focus(widgetName); },
}; };
} }
@ -79,13 +79,13 @@ describe('telepath: wagtail.blocks.FieldBlock', () => {
test('getValue() calls widget getValue()', () => { test('getValue() calls widget getValue()', () => {
const value = boundField.getValue(); const value = boundField.getValue();
expect(getValue.mock.calls.length).toBe(1); expect(getValue.mock.calls.length).toBe(1);
expect(value).toEqual('value: The widget'); expect(value).toEqual('value: The widget - the-prefix');
}); });
test('getState() calls widget getState()', () => { test('getState() calls widget getState()', () => {
const state = boundField.getState(); const state = boundField.getState();
expect(getState.mock.calls.length).toBe(1); expect(getState.mock.calls.length).toBe(1);
expect(state).toEqual('state: The widget'); expect(state).toEqual('state: The widget - the-prefix');
}); });
test('setState() calls widget setState()', () => { test('setState() calls widget setState()', () => {
@ -181,8 +181,8 @@ describe('telepath: wagtail.blocks.StructBlock', () => {
const value = boundField.getValue(); const value = boundField.getValue();
expect(getValue.mock.calls.length).toBe(2); expect(getValue.mock.calls.length).toBe(2);
expect(value).toEqual({ expect(value).toEqual({
heading_text: 'value: Heading widget', heading_text: 'value: Heading widget - the-prefix-heading_text',
size: 'value: Size widget' size: 'value: Size widget - the-prefix-size'
}); });
}); });
@ -190,8 +190,8 @@ describe('telepath: wagtail.blocks.StructBlock', () => {
const state = boundField.getState(); const state = boundField.getState();
expect(getState.mock.calls.length).toBe(2); expect(getState.mock.calls.length).toBe(2);
expect(state).toEqual({ expect(state).toEqual({
heading_text: 'state: Heading widget', heading_text: 'state: Heading widget - the-prefix-heading_text',
size: 'state: Size widget' size: 'state: Size widget - the-prefix-size'
}); });
}); });
@ -281,8 +281,8 @@ describe('telepath: wagtail.blocks.ListBlock', () => {
const value = boundField.getValue(); const value = boundField.getValue();
expect(getValue.mock.calls.length).toBe(2); expect(getValue.mock.calls.length).toBe(2);
expect(value).toEqual([ expect(value).toEqual([
'value: The widget', 'value: The widget - the-prefix-0-value',
'value: The widget' 'value: The widget - the-prefix-1-value'
]); ]);
}); });
@ -290,8 +290,8 @@ describe('telepath: wagtail.blocks.ListBlock', () => {
const state = boundField.getState(); const state = boundField.getState();
expect(getState.mock.calls.length).toBe(2); expect(getState.mock.calls.length).toBe(2);
expect(state).toEqual([ expect(state).toEqual([
'state: The widget', 'state: The widget - the-prefix-0-value',
'state: The widget' 'state: The widget - the-prefix-1-value'
]); ]);
}); });
@ -330,9 +330,9 @@ describe('telepath: wagtail.blocks.ListBlock', () => {
const state = boundField.getState(); const state = boundField.getState();
expect(getState.mock.calls.length).toBe(3); expect(getState.mock.calls.length).toBe(3);
expect(state).toEqual([ expect(state).toEqual([
'state: The widget', 'state: The widget - the-prefix-0-value',
'state: The widget', 'state: The widget - the-prefix-1-value',
'state: The widget' 'state: The widget - the-prefix-2-value'
]); ]);
}); });
@ -342,3 +342,199 @@ describe('telepath: wagtail.blocks.ListBlock', () => {
expect(focus.mock.calls[0][0]).toBe('The widget'); expect(focus.mock.calls[0][0]).toBe('The widget');
}); });
}); });
describe('telepath: wagtail.blocks.StreamBlock', () => {
let boundField;
beforeEach(() => {
// Create mocks for callbacks
constructor = jest.fn();
setState = jest.fn();
getState = jest.fn();
getValue = jest.fn();
focus = jest.fn();
// Create a placeholder to render the block
document.body.innerHTML = '<div id="placeholder"></div>';
// Unpack and render
const fieldDef = window.telepath.unpack({
_type: 'wagtail.blocks.StreamBlock',
_args: ['', [['', [{
_type: 'wagtail.blocks.FieldBlock',
_args: ['test_block_a', {
_type: 'wagtail.widgets.DummyWidget',
_args: ['Block A widget']
}, {
label: 'Test Block B',
required: true,
icon: 'placeholder',
classname: 'field char_field widget-text_input fieldname-test_charblock'
}]
}, {
_type: 'wagtail.blocks.FieldBlock',
_args: ['test_block_b', {
_type: 'wagtail.widgets.DummyWidget',
_args: ['Block B widget']
}, {
label: 'Test Block B',
required: true,
icon: 'pilcrow',
classname: 'field char_field widget-admin_auto_height_text_input fieldname-test_textblock'
}]
}
]]], {
test_block_a: 'Block A options',
test_block_b: 'Block B options',
}, {
label: '',
required: true,
icon: 'placeholder',
classname: null,
helpText: null,
maxNum: null,
minNum: null,
blockCounts: {}
}]
});
boundField = fieldDef.render($('#placeholder'), 'the-prefix', [
{
id: '1',
type: 'test_block_a',
value: 'First value'
},
{
id: '2',
type: 'test_block_b',
value: 'Second value'
},
]);
});
test('it renders correctly', () => {
expect(document.body.innerHTML).toMatchSnapshot();
});
test('Widget constructors are called with correct parameters', () => {
expect(constructor.mock.calls.length).toBe(2);
expect(constructor.mock.calls[0][0]).toBe('Block A widget');
expect(constructor.mock.calls[0][1]).toEqual({
name: 'the-prefix-0-value',
id: 'the-prefix-0-value',
initialState: 'First value',
});
expect(constructor.mock.calls[1][0]).toBe('Block B widget');
expect(constructor.mock.calls[1][1]).toEqual({
name: 'the-prefix-1-value',
id: 'the-prefix-1-value',
initialState: 'Second value',
});
});
test('getValue() calls getValue() on widget for both values', () => {
const value = boundField.getValue();
expect(getValue.mock.calls.length).toBe(2);
expect(value).toEqual([
{
id: '1',
type: 'test_block_a',
value: 'value: Block A widget - the-prefix-0-value'
},
{
id: '2',
type: 'test_block_b',
value: 'value: Block B widget - the-prefix-1-value'
},
]);
});
test('getState() calls getState() on all widgets', () => {
const state = boundField.getState();
expect(getState.mock.calls.length).toBe(2);
expect(state).toEqual([
{
id: '1',
type: 'test_block_a',
value: 'state: Block A widget - the-prefix-0-value'
},
{
id: '2',
type: 'test_block_b',
value: 'state: Block B widget - the-prefix-1-value'
},
]);
});
test('setState() creates new widgets', () => {
boundField.setState([
{
id: '1',
type: 'test_block_a',
value: 'Changed first value'
},
{
id: '3',
type: 'test_block_b',
value: 'Third value'
},
{
id: '2',
type: 'test_block_b',
value: 'Changed second value'
},
]);
// Includes the two initial calls, plus the three new ones
expect(constructor.mock.calls.length).toBe(5);
expect(constructor.mock.calls[2][0]).toBe('Block A widget');
expect(constructor.mock.calls[2][1]).toEqual({
name: 'the-prefix-0-value',
id: 'the-prefix-0-value',
initialState: 'Changed first value',
});
expect(constructor.mock.calls[3][0]).toBe('Block B widget');
expect(constructor.mock.calls[3][1]).toEqual({
name: 'the-prefix-1-value',
id: 'the-prefix-1-value',
initialState: 'Third value',
});
expect(constructor.mock.calls[4][0]).toBe('Block B widget');
expect(constructor.mock.calls[4][1]).toEqual({
name: 'the-prefix-2-value',
id: 'the-prefix-2-value',
initialState: 'Changed second value',
});
// Let's get the state now to make sure the initial widgets are gone
const state = boundField.getState();
expect(getState.mock.calls.length).toBe(3);
expect(state).toEqual([
{
id: '1',
type: 'test_block_a',
value: 'state: Block A widget - the-prefix-0-value'
},
{
id: '3',
type: 'test_block_b',
value: 'state: Block B widget - the-prefix-1-value'
},
{
id: '2',
type: 'test_block_b',
value: 'state: Block B widget - the-prefix-2-value'
},
]);
});
test('focus() calls focus() on first widget', () => {
boundField.focus();
expect(focus.mock.calls.length).toBe(1);
expect(focus.mock.calls[0][0]).toBe('Block A widget');
});
});