From eae8a5f146feceeff09b4a0eed999891dc357dce Mon Sep 17 00:00:00 2001 From: James Lucas Date: Thu, 4 Jul 2024 12:51:48 +1000 Subject: [PATCH 1/2] fix: syncFieldWithNewRow incorrectly replacing row values in field className attribute --- src/js/form-builder.js | 20 +++++++++++++++++--- 1 file changed, 17 insertions(+), 3 deletions(-) diff --git a/src/js/form-builder.js b/src/js/form-builder.js index b44a5c0b1..355e7212f 100644 --- a/src/js/form-builder.js +++ b/src/js/form-builder.js @@ -2139,13 +2139,27 @@ function FormBuilder(opts, element, $) { }) } + /** + * Updates the field's className to include the current wrapping row, removing the previous row if defined + * @param fieldID + */ function syncFieldWithNewRow(fieldID) { if (fieldID) { const inputClassElement = $(`#className-${fieldID.replace('-cont', '')}`) - if (inputClassElement.val()) { - const oldRow = h.getRowClass(inputClassElement.val()) + const currentClassName = inputClassElement.val().trim() + if (currentClassName) { + let currentClasses = currentClassName.split(' ') + const oldRow = h.getRowClass(currentClassName) const wrapperRow = h.getRowClass(inputClassElement.closest(rowWrapperClassSelector).attr('class')) - inputClassElement.val(inputClassElement.val().replace(oldRow, wrapperRow)) + if (oldRow !== wrapperRow) { + if (oldRow) { + currentClasses = currentClasses.filter(function(obj) { + return obj !== oldRow + }) + } + currentClasses.push(wrapperRow) + inputClassElement.val(currentClasses.join(' ')) + } checkRowCleanup() } } From 670688add0a8f11b92e7d3ee90b3659c8121b55e Mon Sep 17 00:00:00 2001 From: James Lucas Date: Fri, 5 Jul 2024 13:02:43 +1000 Subject: [PATCH 2/2] fix: move syncFieldWithNewRow to Helpers to make the feature directly testable --- src/js/form-builder.js | 33 ++++------------------------ src/js/helpers.js | 28 ++++++++++++++++++++++++ tests/bootstrap.test.js | 48 +++++++++++++++++++++++++++++++++++++++++ 3 files changed, 80 insertions(+), 29 deletions(-) diff --git a/src/js/form-builder.js b/src/js/form-builder.js index 355e7212f..0ba5c87b8 100644 --- a/src/js/form-builder.js +++ b/src/js/form-builder.js @@ -1397,7 +1397,7 @@ function FormBuilder(opts, element, $) { colWrapper.appendTo(rowWrapperNode) setupSortableRowWrapper(rowWrapperNode) - syncFieldWithNewRow(colWrapper.attr('id')) + h.syncFieldWithNewRow(colWrapper[0], colWrapper.closest(rowWrapperClassSelector)[0]) } } @@ -1443,10 +1443,11 @@ function FormBuilder(opts, element, $) { stop: (event, ui) => { $stage.removeClass('__preventColButtons') $stage.children(tmpRowPlaceholderClassSelector).removeClass('hoverDropStyleInverse') + checkRowCleanup() autoSizeRowColumns(ui.item.closest(rowWrapperClassSelector), true) }, update: (event, ui) => { - syncFieldWithNewRow(ui.item.attr('id')) + h.syncFieldWithNewRow(ui.item, $(ui.item).closest(rowWrapperClassSelector)[0]) }, }) @@ -1934,7 +1935,7 @@ function FormBuilder(opts, element, $) { setupSortableRowWrapper(rowWrapper) ResetAllInvisibleRowPlaceholders() - syncFieldWithNewRow($clone.attr('id')) + h.syncFieldWithNewRow($clone[0], $clone.closest(rowWrapperClassSelector)[0]) } // Delete field @@ -2139,32 +2140,6 @@ function FormBuilder(opts, element, $) { }) } - /** - * Updates the field's className to include the current wrapping row, removing the previous row if defined - * @param fieldID - */ - function syncFieldWithNewRow(fieldID) { - if (fieldID) { - const inputClassElement = $(`#className-${fieldID.replace('-cont', '')}`) - const currentClassName = inputClassElement.val().trim() - if (currentClassName) { - let currentClasses = currentClassName.split(' ') - const oldRow = h.getRowClass(currentClassName) - const wrapperRow = h.getRowClass(inputClassElement.closest(rowWrapperClassSelector).attr('class')) - if (oldRow !== wrapperRow) { - if (oldRow) { - currentClasses = currentClasses.filter(function(obj) { - return obj !== oldRow - }) - } - currentClasses.push(wrapperRow) - inputClassElement.val(currentClasses.join(' ')) - } - checkRowCleanup() - } - } - } - //When mouse moves away a certain distance, cancel grid mode $(document).mousemove(e => { if ( diff --git a/src/js/helpers.js b/src/js/helpers.js index 5228c23fd..e4972825d 100644 --- a/src/js/helpers.js +++ b/src/js/helpers.js @@ -1462,4 +1462,32 @@ export default class Helpers { inputClassElement.val(this.changeBootstrapClass(inputClassElement.val(), newValue)) } } + + /** + * Updates the field's className to include the current wrapping row, removing the previous row if defined + * @param {HTMLElement} field + * @param {HTMLElement} wrapperRow + */ + syncFieldWithNewRow(field, wrapperRow) { + if (field) { + const inputClassElement = $(field).find('.fld-className') + const currentClassName = inputClassElement.val()?.trim() + if (currentClassName) { + let currentClasses = currentClassName.split(' ') + const oldRow = this.getRowClass(currentClassName) + const newRow = this.getRowClass(wrapperRow?.className ?? '') + if (oldRow !== newRow) { + if (oldRow) { + currentClasses = currentClasses.filter(function(obj) { + return obj !== oldRow + }) + } + if (newRow) { + currentClasses.push(newRow) + } + inputClassElement.val(currentClasses.join(' ')) + } + } + } + } } diff --git a/tests/bootstrap.test.js b/tests/bootstrap.test.js index 5bf2a580c..2ffe57323 100644 --- a/tests/bootstrap.test.js +++ b/tests/bootstrap.test.js @@ -70,4 +70,52 @@ describe('Test Boostrap Helper functions', () => { expect(helper.changeBootstrapClass('col-md-5 row-1 random-class row col', 7)).toBe('col-md-7 row-1 random-class row col') }) + test('syncFieldWithNewRow swaps existing with new', () => { + const field = $('
') + const input = $('', {class: 'fld-className', value: 'prefix-class form-control row-1 col-md-12 suffix-class'}) + field.append(input) + const newRow = $('
', {class: 'row-2'}) + + expect(input.val()).toEqual('prefix-class form-control row-1 col-md-12 suffix-class') + helper.syncFieldWithNewRow(field[0], newRow[0]) + + expect(input.val()).toEqual('prefix-class form-control col-md-12 suffix-class row-2') + }) + + test('syncFieldWithNewRow unsets row class when no new row', () => { + const field = $('
') + const input = $('', {class: 'fld-className', value: 'prefix-class form-control row-1 col-md-12 suffix-class'}) + field.append(input) + const newRow = $('
', {}) + + expect(input.val()).toEqual('prefix-class form-control row-1 col-md-12 suffix-class') + helper.syncFieldWithNewRow(field[0], newRow[0]) + + expect(input.val()).toEqual('prefix-class form-control col-md-12 suffix-class') + }) + + test('syncFieldWithNewRow sets new row when no previous row', () => { + const field = $('
') + const input = $('', {class: 'fld-className', value: 'prefix-class form-control col-md-12 suffix-class'}) + field.append(input) + const newRow = $('
', {class: 'row-2'}) + + expect(input.val()).toEqual('prefix-class form-control col-md-12 suffix-class') + helper.syncFieldWithNewRow(field[0], newRow[0]) + + expect(input.val()).toEqual('prefix-class form-control col-md-12 suffix-class row-2') + }) + + test('syncFieldWithNewRow no change if same row id', () => { + const field = $('
') + const input = $('', {class: 'fld-className', value: 'prefix-class form-control row-2 col-md-12 suffix-class'}) + field.append(input) + const newRow = $('
', {class: 'row-2'}) + + expect(input.val()).toEqual('prefix-class form-control row-2 col-md-12 suffix-class') + helper.syncFieldWithNewRow(field[0], newRow[0]) + + expect(input.val()).toEqual('prefix-class form-control row-2 col-md-12 suffix-class') + }) + }) \ No newline at end of file