webapp.pdf-exporter-admin.js.style-package-weights.js Maven / Gradle / Ivy
SbbCommon.init({
extension: 'pdf-exporter',
scope: SbbCommon.getValueById('scope')
});
const StylePackageWeights = {
ID_PREFIX: 'input.weight.',
STYLE_INPUT: '.weight-input',
STYLE_ITEM: '.sortable-item',
init: function (listId) {
this.name = 'StylePackageWeights';
this.sortableList = document.getElementById(listId);
this.sortable = new Sortable(this.sortableList, {
animation: 150,
filter: '.static,.weight-input',
preventOnFilter: false,
onEnd: (evt) => {
this.afterPositionChange(evt.oldDraggableIndex, evt.newDraggableIndex);
}
});
},
loadPackageList: function () {
SbbCommon.callAsync({
method: 'GET',
url: `/polarion/${SbbCommon.extension}/rest/internal/settings/style-package/weights?scope=${SbbCommon.scope}`,
contentType: 'application/json',
onOk: (responseText) => {
this.setData(responseText);
},
onError: () => SbbCommon.setLoadingErrorNotificationVisible(true)
});
},
saveWeights: function () {
const result = [];
this.sortableList.querySelectorAll(this.STYLE_ITEM).forEach(item => {
if (item.classList.contains('static')) {
return;
}
const input = item.querySelector('input');
const name = input.id.replace(this.ID_PREFIX, '');
const weight = parseFloat(input.value);
const scope = SbbCommon.scope;
result.push({
name: name, scope: scope, weight: weight
});
});
SbbCommon.callAsync({
method: 'POST',
url: `/polarion/${SbbCommon.extension}/rest/internal/settings/style-package/weights`,
contentType: 'application/json',
body: JSON.stringify(result),
onOk: () => {
SbbCommon.showSaveSuccessAlert();
},
onError: () => SbbCommon.showSaveErrorAlert()
});
},
setData: function (jsonString) {
this.data = JSON.parse(jsonString);
const weightStyleName = this.STYLE_INPUT.replace('.', '');
this.sortableList.innerHTML = '';
this.data.forEach((item) => {
const li = document.createElement("li");
li.classList.add(this.STYLE_ITEM.replace('.', ''));
const globalScopedItem = item.scope === "" && SbbCommon.scope !== "";
if (globalScopedItem) {
li.classList.add("static"); // Add static class when scope is empty
}
const globalSuffix = globalScopedItem ? " (global)" : "";
li.innerHTML = `${item.name}${globalSuffix} `;
this.sortableList.appendChild(li);
});
this.sortableList.querySelectorAll(this.STYLE_INPUT).forEach(input => {
input.addEventListener('keydown', (event) => this.handleManualChange(event)); // Listen for 'Enter'
input.addEventListener('blur', (event) => this.handleManualChange(event)); // Listen for 'blur' (focus change)
});
this.sortList();
},
sortList: function () {
const listItems = Array.from(this.sortableList.querySelectorAll(this.STYLE_ITEM));
// Sort the items
listItems.sort((a, b) => {
let inputB = b.querySelector(this.STYLE_INPUT);
let inputA = a.querySelector(this.STYLE_INPUT);
const weightDifference = parseFloat(inputB.value) - parseFloat(inputA.value);
return weightDifference === 0 ? inputA.id.localeCompare(inputB.id) : weightDifference;
});
this.sortableList.innerHTML = '';
listItems.forEach(item => this.sortableList.appendChild(item));
},
handleManualChange: function (event) {
if (event.type === 'blur') {
StylePackageUtils.adjustWeight(event.target);
const modifiedSettingName = event.target.id.replace(this.ID_PREFIX, '');
const previousPos = this.getPositionByName(modifiedSettingName);
const items = Array.from(this.sortableList.querySelectorAll(this.STYLE_ITEM));
items.sort((a, b) => {
let inputB = b.querySelector(this.STYLE_INPUT);
let inputA = a.querySelector(this.STYLE_INPUT);
const weightDifference = parseFloat(inputB.value) - parseFloat(inputA.value);
return weightDifference === 0 ? inputA.id.replace(this.ID_PREFIX, '').localeCompare(b.id.replace(this.ID_PREFIX, '')) : weightDifference;
});
const newPos = this.getPositionByName(modifiedSettingName, items);
this.moveItem(previousPos, newPos);
}
},
moveItem: function (fromPosition, toPosition) {
if (fromPosition === toPosition) {
this.afterPositionChange(fromPosition, toPosition);
return;
}
const items = Array.from(this.sortableList.children);
const fromItem = items[fromPosition];
const toItem = items[toPosition];
const fromItemRect = fromItem.getBoundingClientRect();
const toItemRect = toItem.getBoundingClientRect();
const distance = fromItemRect.top - toItemRect.top;
// Temporarily apply transition for smooth movement
fromItem.style.transition = 'transform 0.5s ease';
fromItem.style.transform = `translateY(${-distance}px)`;
const self = this;
fromItem.addEventListener('transitionend', function handleTransitionEnd() {
fromItem.style.transition = '';
fromItem.style.transform = '';
self.sortableList.insertBefore(fromItem, fromPosition < toPosition ? toItem.nextSibling : toItem);
fromItem.removeEventListener('transitionend', handleTransitionEnd);
self.afterPositionChange(fromPosition, toPosition);
});
},
afterPositionChange: function (fromPosition, toPosition) {
if (fromPosition !== toPosition) {
this.updateWeightForNewPosition(toPosition);
}
// We have to sort list again even if its position hasn't changed.
// For example:
// 1. We have sequence like "ccc 45, aaa 42, bbb 42, ddd 42"
// 2. Manually change ccc value to 42
// 3. Its position initially wasn't changed, but according to our rules when multiple items
// have the same weight they must be sorted alphabetically
// 4. Sort the resulting list to get proper result: "aaa 42, bbb 42, ccc 42, ddd 42"
this.sortList();
},
updateWeightForNewPosition: function (newPosition) {
const items = this.sortableList.querySelectorAll(this.STYLE_ITEM);
const changedItemName = items[newPosition].querySelector(this.STYLE_INPUT).id.replace(this.ID_PREFIX, '');
// we are going to try to keep initial weight
const initialWeight = this.data.find(i => i.name === changedItemName).weight;
let newValue;
if (newPosition === 0) {
const nextItemValue = parseFloat(items[newPosition + 1].querySelector(this.STYLE_INPUT).value);
newValue = initialWeight > nextItemValue ? initialWeight : nextItemValue + 1;
} else if (newPosition === items.length - 1) {
const prevItemValue = parseFloat(items[items.length - 2].querySelector(this.STYLE_INPUT).value);
newValue = initialWeight < prevItemValue ? initialWeight : prevItemValue - 1;
} else {
const prevItemValue = parseFloat(items[newPosition - 1].querySelector(this.STYLE_INPUT).value);
const nextItemValue = parseFloat(items[newPosition + 1].querySelector(this.STYLE_INPUT).value);
newValue = initialWeight > nextItemValue && initialWeight < prevItemValue ? initialWeight :
parseFloat((prevItemValue + (nextItemValue - prevItemValue) / 2).toFixed(1));
}
items[newPosition].querySelector(this.STYLE_INPUT).value = Math.max(0, Math.min(100, newValue));
},
getPositionByName: function (name, dynamicItems) {
const items = dynamicItems === undefined ? this.sortableList.querySelectorAll(this.STYLE_ITEM) : dynamicItems;
const id = this.ID_PREFIX + name;
for (let i = 0; i < items.length; i++) {
if (items[i].querySelector(this.STYLE_INPUT).id === id) {
return i;
}
}
throw new Error(`Unknown name "${name}"`);
}
}
StylePackageWeights.init('sortable-list');
StylePackageWeights.loadPackageList();
© 2015 - 2025 Weber Informatics LLC | Privacy Policy