Files
drmaterac.pl/iadmin/themes/new-theme/js/components/grid/extension/position-extension.js
2025-03-21 20:24:43 +01:00

205 lines
5.7 KiB
JavaScript

/**
* Copyright since 2007 PrestaShop SA and Contributors
* PrestaShop is an International Registered Trademark & Property of PrestaShop SA
*
* NOTICE OF LICENSE
*
* This source file is subject to the Open Software License (OSL 3.0)
* that is bundled with this package in the file LICENSE.md.
* It is also available through the world-wide-web at this URL:
* https://opensource.org/licenses/OSL-3.0
* If you did not receive a copy of the license and are unable to
* obtain it through the world-wide-web, please send an email
* to license@prestashop.com so we can send you a copy immediately.
*
* DISCLAIMER
*
* Do not edit or add to this file if you wish to upgrade PrestaShop to newer
* versions in the future. If you wish to customize PrestaShop for your
* needs please refer to https://devdocs.prestashop.com/ for more information.
*
* @author PrestaShop SA and Contributors <contact@prestashop.com>
* @copyright Since 2007 PrestaShop SA and Contributors
* @license https://opensource.org/licenses/OSL-3.0 Open Software License (OSL 3.0)
*/
import 'tablednd/dist/jquery.tablednd.min';
const {$} = window;
/**
* Class PositionExtension extends Grid with reorderable positions
*/
export default class PositionExtension {
constructor() {
return {
extend: (grid) => this.extend(grid),
};
}
/**
* Extend grid
*
* @param {Grid} grid
*/
extend(grid) {
this.grid = grid;
this.addIdsToGridTableRows();
grid.getContainer().find('.js-grid-table').tableDnD({
onDragClass: 'position-row-while-drag',
dragHandle: '.js-drag-handle',
onDrop: (table, row) => this.handlePositionChange(row),
});
grid.getContainer().find('.js-drag-handle').hover(
function () {
$(this).closest('tr').addClass('hover');
},
function () {
$(this).closest('tr').removeClass('hover');
},
);
}
/**
* When position is changed handle update
*
* @param {HTMLElement} row
*
* @private
*/
handlePositionChange(row) {
const $rowPositionContainer = $(row).find(`.js-${this.grid.getId()}-position:first`);
const updateUrl = $rowPositionContainer.data('update-url');
const method = $rowPositionContainer.data('update-method');
const positions = this.getRowsPositions();
const params = {positions};
this.updatePosition(updateUrl, params, method);
}
/**
* Returns the current table positions
* @returns {Array}
* @private
*/
getRowsPositions() {
const tableData = JSON.parse($.tableDnD.jsonize());
const rowsData = tableData[`${this.grid.getId()}_grid_table`];
const completeRowsData = [];
let trData;
// retrieve dragAndDropOffset offset to have all needed data
// for positions mapping evolution over time
for (let i = 0; i < rowsData.length; i += 1) {
trData = this.grid.getContainer()
.find(`#${rowsData[i]}`);
completeRowsData.push({
rowMarker: rowsData[i],
offset: trData.data('dragAndDropOffset'),
});
}
return this.computeMappingBetweenOldAndNewPositions(completeRowsData);
}
/**
* Add ID's to Grid table rows to make tableDnD.onDrop() function work.
*
* @private
*/
addIdsToGridTableRows() {
let counter = 0;
this.grid.getContainer()
.find(`.js-grid-table .js-${this.grid.getId()}-position`)
.each((index, positionWrapper) => {
const $positionWrapper = $(positionWrapper);
const rowId = $positionWrapper.data('id');
const position = $positionWrapper.data('position');
const id = `row_${rowId}_${position}`;
$positionWrapper.closest('tr').attr('id', id);
$positionWrapper.closest('td').addClass('js-drag-handle');
$positionWrapper.closest('tr').data('dragAndDropOffset', counter);
counter += 1;
});
}
/**
* Process rows positions update
*
* @param {String} url
* @param {Object} params
* @param {String} method
*
* @private
*/
updatePosition(url, params, method) {
const isGetOrPostMethod = ['GET', 'POST'].includes(method);
const $form = $('<form>', {
action: url,
method: isGetOrPostMethod ? method : 'POST',
}).appendTo('body');
const positionsNb = params.positions.length;
let position;
for (let i = 0; i < positionsNb; i += 1) {
position = params.positions[i];
$form.append(
$('<input>', {
type: 'hidden',
name: `positions[${i}][rowId]`,
value: position.rowId,
}),
$('<input>', {
type: 'hidden',
name: `positions[${i}][oldPosition]`,
value: position.oldPosition,
}),
$('<input>', {
type: 'hidden',
name: `positions[${i}][newPosition]`,
value: position.newPosition,
}),
);
}
// This _method param is used by Symfony to simulate DELETE and PUT methods
if (!isGetOrPostMethod) {
$form.append($('<input>', {
type: 'hidden',
name: '_method',
value: method,
}));
}
$form.submit();
}
/**
* Rows have been reordered. This function
* finds, for each row ID: the old position, the new position
*
* @returns {Array}
* @private
*/
computeMappingBetweenOldAndNewPositions(rowsData) {
const regex = /^row_(\d+)_(\d+)$/;
const mapping = Array(rowsData.length).fill().map(Object);
for (let i = 0; i < rowsData.length; i += 1) {
const [, rowId, oldPosition] = regex.exec(rowsData[i].rowMarker);
mapping[i].rowId = rowId;
mapping[i].oldPosition = parseInt(oldPosition, 10);
// This row will have as a new position the old position of the current one
mapping[rowsData[i].offset].newPosition = mapping[i].oldPosition;
}
return mapping;
}
}