From 08e5b8bd93628b26d3593d7c324669518ede4398 Mon Sep 17 00:00:00 2001 From: "anna.shakhova" Date: Thu, 26 Dec 2024 15:42:35 +0100 Subject: [PATCH] CollectionAsync: add tests for async render cases --- .../collection/m_collection_widget.async.ts | 2 +- .../listParts/commonTests.js | 48 +++++++ .../treeView.async.tests.js | 122 ++++++++++++++++++ 3 files changed, 171 insertions(+), 1 deletion(-) create mode 100644 packages/devextreme/testing/tests/DevExpress.ui.widgets/treeView.async.tests.js diff --git a/packages/devextreme/js/__internal/ui/collection/m_collection_widget.async.ts b/packages/devextreme/js/__internal/ui/collection/m_collection_widget.async.ts index 20f337f88f5b..e4c3dc9f1365 100644 --- a/packages/devextreme/js/__internal/ui/collection/m_collection_widget.async.ts +++ b/packages/devextreme/js/__internal/ui/collection/m_collection_widget.async.ts @@ -46,7 +46,7 @@ const AsyncCollectionWidget = CollectionWidgetEdit.inherit({ when.apply(this, asyncTemplateItems).done(() => { this._postProcessRenderItems(...args); - d.resolve().then(() => { + d.resolve().done(() => { this._asyncTemplateItemsMap = {}; }); }); diff --git a/packages/devextreme/testing/tests/DevExpress.ui.widgets/listParts/commonTests.js b/packages/devextreme/testing/tests/DevExpress.ui.widgets/listParts/commonTests.js index 73d02c4d7201..6485b75d712b 100644 --- a/packages/devextreme/testing/tests/DevExpress.ui.widgets/listParts/commonTests.js +++ b/packages/devextreme/testing/tests/DevExpress.ui.widgets/listParts/commonTests.js @@ -3683,6 +3683,54 @@ QUnit.module('regressions', moduleSetup, () => { assert.equal(count, 1); }); + + QUnit.test('Selection: item selected correctly on async render with tree structure (T1269855)', function(assert) { + this.clock.restore(); + const done = assert.async(); + + const data = [ + { id: 1, name: 'Item 1_1', group: 'group_1' }, + { id: 2, name: 'Item 1_2', group: 'group_1' }, + { id: 3, name: 'Item 1_3', group: 'group_1' }, + { id: 4, name: 'Item 2_1', group: 'group_2' }, + ]; + + const dataSource = new DataSource({ + store: new ArrayStore({ data, key: 'id' }), + group: 'group', + }); + + const instance = new List($('#list'), { + dataSource, + grouped: true, + templatesRenderAsynchronously: true, + integrationOptions: { + templates: { + 'item': { + render: function({ model, container, onRendered }) { + setTimeout(function() { + const $item = $(`
${model.name}
`); + $item.appendTo(container); + + onRendered(); + }, 100); + } + }, + } + }, + selectionMode: 'single', + selectedItemKeys: [data[0].id], + }); + + instance.option('_onItemsRendered', () => { + const listElement = instance.element(); + const $firstGroup = $(listElement).find(`.${LIST_GROUP_CLASS}`).eq(0); + const $firstItemInFirstGroup = $firstGroup.find(`.${LIST_ITEM_CLASS}`).eq(0); + + assert.ok($firstItemInFirstGroup.hasClass('dx-list-item-selected'), 'First item in first group should be selected'); + done(); + }); + }); }); QUnit.module('widget sizing render', {}, () => { diff --git a/packages/devextreme/testing/tests/DevExpress.ui.widgets/treeView.async.tests.js b/packages/devextreme/testing/tests/DevExpress.ui.widgets/treeView.async.tests.js new file mode 100644 index 000000000000..9cba4f0d050b --- /dev/null +++ b/packages/devextreme/testing/tests/DevExpress.ui.widgets/treeView.async.tests.js @@ -0,0 +1,122 @@ +import $ from 'jquery'; +import TreeView from 'ui/tree_view'; + +import 'generic_light.css!'; + +const { testStart } = QUnit; + +testStart(function() { + const markup = '
'; + + $('#qunit-fixture').html(markup); +}); + +const asyncTemplateRenderTimeout = 50; + +const CHECKBOX_CLASS = 'dx-checkbox'; +const CHECKBOX_CHECKED_CLASS = 'dx-checkbox-checked'; +const CHECKBOX_INDETERMINATE_CLASS = 'dx-checkbox-indeterminate'; +const TREEVIEW_ROOT_NODE_CLASS = 'dx-treeview-root-node'; + +QUnit.module('Async render', () => { + ['normal', 'selectAll'].forEach((showCheckBoxesMode) => { + QUnit.test(`TreeView checkboxed should be correctly rendered in async mode. checkboxMode: ${showCheckBoxesMode} (T1269855)`, function(assert) { + const done = assert.async(); + + const data = [ + { + id: 1, + text: 'Item 1', + expanded: true, + selected: true, + items: [ + { + id: 12, text: 'Nested Item 2', expanded: true, items: [ + { id: 121, text: 'Third level item 1' }, + { id: 122, text: 'Third level item 2' } + ] + } + ] + }, + { + id: 2, + text: 'Item 2', + expanded: true, + items: [ + { + id: 22, text: 'Nested Item 2', expanded: true, items: [ + { id: 221, text: 'Third level item 1' }, + { id: 222, text: 'Third level item 2', selected: true } + ] + } + ] + }, + { + id: 3, + text: 'Item 3', + expanded: true, + items: [ + { + id: 33, text: 'Nested Item 3', expanded: true, items: [ + { id: 331, text: 'Third level item 1' }, + { id: 332, text: 'Third level item 2' } + ] + } + ] + } + ]; + + const instance = new TreeView($('#treeView'), { + items: data, + showCheckBoxesMode, + templatesRenderAsynchronously: true, + itemTemplate: 'myTemplate', + integrationOptions: { + templates: { + myTemplate: { + render({ model, container, onRendered }) { + setTimeout(() => { + const $item = $(`
${model.text}
`); + $item.appendTo(container); + + onRendered(); + }); + } + } + } + }, + }); + + setTimeout(() => { + const element = instance.itemsContainer(); + const $treeRootNodes = $(element).find(`.${TREEVIEW_ROOT_NODE_CLASS}`); + + const $firstRootNode = $treeRootNodes.eq(0); + const $firstGroupCheckboxes = $firstRootNode.find(`.${CHECKBOX_CLASS}`); + + assert.ok($firstGroupCheckboxes.eq(0).hasClass(CHECKBOX_CHECKED_CLASS), 'First group root checkbox has selected class'); + assert.ok($firstGroupCheckboxes.eq(1).hasClass(CHECKBOX_CHECKED_CLASS), 'First group nested node checkbox has selected class'); + assert.ok($firstGroupCheckboxes.eq(2).hasClass(CHECKBOX_CHECKED_CLASS), 'First group leaf node 1 checkbox has selected class'); + assert.ok($firstGroupCheckboxes.eq(3).hasClass(CHECKBOX_CHECKED_CLASS), 'First group leaf node 2 checkbox has selected class'); + + const $secondRootNode = $treeRootNodes.eq(1); + const $secondGroupCheckboxes = $secondRootNode.find(`.${CHECKBOX_CLASS}`); + + assert.ok($secondGroupCheckboxes.eq(0).hasClass(CHECKBOX_INDETERMINATE_CLASS), 'Second group root checkbox has indeterminate class'); + assert.ok($secondGroupCheckboxes.eq(1).hasClass(CHECKBOX_INDETERMINATE_CLASS), 'Second group nested node checkbox has indeterminate class'); + assert.notOk($secondGroupCheckboxes.eq(2).hasClass(CHECKBOX_CHECKED_CLASS), 'Second group leaf node 1 checkbox has not selected class'); + assert.ok($secondGroupCheckboxes.eq(3).hasClass(CHECKBOX_CHECKED_CLASS), 'Second group leaf node 2 checkbox has selected class'); + + const $thirdRootNode = $treeRootNodes.eq(2); + const $thirdGroupCheckboxes = $thirdRootNode.find(`.${CHECKBOX_CLASS}`); + + assert.notOk($thirdGroupCheckboxes.eq(0).hasClass(CHECKBOX_CHECKED_CLASS), 'Third group root checkbox has not selected class'); + assert.notOk($thirdGroupCheckboxes.eq(1).hasClass(CHECKBOX_CHECKED_CLASS), 'Third group nested node checkbox has not selected class'); + assert.notOk($thirdGroupCheckboxes.eq(2).hasClass(CHECKBOX_CHECKED_CLASS), 'Third group leaf node 1 checkbox has not selected class'); + assert.notOk($thirdGroupCheckboxes.eq(3).hasClass(CHECKBOX_CHECKED_CLASS), 'Third group leaf node 2 checkbox has not selected class'); + + done(); + }, asyncTemplateRenderTimeout); + }); + }); +});