ed4466d840
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
869 lines
20 KiB
JavaScript
869 lines
20 KiB
JavaScript
// for Toolkit.js
|
|
function gettext(val) { return val; };
|
|
|
|
Ext.onReady(function() {
|
|
const COLORS = {
|
|
'keep-last': 'orange',
|
|
'keep-hourly': 'purple',
|
|
'keep-daily': 'yellow',
|
|
'keep-weekly': 'green',
|
|
'keep-monthly': 'blue',
|
|
'keep-yearly': 'red',
|
|
'all zero': 'white',
|
|
};
|
|
const TEXT_COLORS = {
|
|
'keep-last': 'black',
|
|
'keep-hourly': 'white',
|
|
'keep-daily': 'black',
|
|
'keep-weekly': 'white',
|
|
'keep-monthly': 'white',
|
|
'keep-yearly': 'white',
|
|
'all zero': 'black',
|
|
};
|
|
|
|
Ext.define('PBS.prunesimulator.Documentation', {
|
|
extend: 'Ext.Panel',
|
|
alias: 'widget.prunesimulatorDocumentation',
|
|
|
|
html: '<iframe style="width:100%;height:100%;border:0px;" src="./documentation.html"/>',
|
|
});
|
|
|
|
Ext.define('PBS.prunesimulator.CalendarEvent', {
|
|
extend: 'Ext.form.field.ComboBox',
|
|
alias: 'widget.prunesimulatorCalendarEvent',
|
|
|
|
editable: true,
|
|
|
|
valueField: 'value',
|
|
queryMode: 'local',
|
|
|
|
store: {
|
|
field: ['value', 'text'],
|
|
data: [
|
|
{ value: '0/2:00', text: "Every two hours" },
|
|
{ value: '0/6:00', text: "Every six hours" },
|
|
{ value: '2,22:30', text: "At 02:30 and 22:30" },
|
|
{ value: '00:00', text: "At 00:00" },
|
|
{ value: '08..17:00/30', text: "From 08:00 to 17:30 every 30 minutes" },
|
|
{ value: 'HOUR:MINUTE', text: "Custom schedule" },
|
|
],
|
|
},
|
|
|
|
tpl: [
|
|
'<ul class="x-list-plain"><tpl for=".">',
|
|
'<li role="option" class="x-boundlist-item">{text}</li>',
|
|
'</tpl></ul>',
|
|
],
|
|
|
|
displayTpl: [
|
|
'<tpl for=".">',
|
|
'{value}',
|
|
'</tpl>',
|
|
],
|
|
});
|
|
|
|
Ext.define('PBS.prunesimulator.DayOfWeekSelector', {
|
|
extend: 'Ext.form.field.ComboBox',
|
|
alias: 'widget.prunesimulatorDayOfWeekSelector',
|
|
|
|
editable: false,
|
|
|
|
displayField: 'text',
|
|
valueField: 'value',
|
|
queryMode: 'local',
|
|
|
|
store: {
|
|
field: ['value', 'text'],
|
|
data: [
|
|
{ value: 'mon', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[1]) },
|
|
{ value: 'tue', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[2]) },
|
|
{ value: 'wed', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[3]) },
|
|
{ value: 'thu', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[4]) },
|
|
{ value: 'fri', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[5]) },
|
|
{ value: 'sat', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[6]) },
|
|
{ value: 'sun', text: Ext.util.Format.htmlDecode(Ext.Date.dayNames[0]) },
|
|
],
|
|
},
|
|
});
|
|
|
|
Ext.define('pbs-prune-list', {
|
|
extend: 'Ext.data.Model',
|
|
fields: [
|
|
{
|
|
name: 'backuptime',
|
|
type: 'date',
|
|
dateFormat: 'timestamp',
|
|
},
|
|
{
|
|
name: 'mark',
|
|
type: 'string',
|
|
},
|
|
{
|
|
name: 'keepName',
|
|
type: 'string',
|
|
},
|
|
],
|
|
});
|
|
|
|
Ext.define('PBS.prunesimulator.PruneList', {
|
|
extend: 'Ext.panel.Panel',
|
|
alias: 'widget.prunesimulatorPruneList',
|
|
|
|
viewModel: {},
|
|
|
|
items: [{
|
|
xtype: 'grid',
|
|
bind: {
|
|
store: '{store}',
|
|
},
|
|
border: false,
|
|
columns: [
|
|
{
|
|
header: 'Backup Time',
|
|
dataIndex: 'backuptime',
|
|
renderer: function(value, metaData, { data }) {
|
|
let text = Ext.Date.format(value, 'Y-m-d H:i:s');
|
|
if (data.mark !== 'keep') {
|
|
return `<div style="text-decoration: line-through;">${text}</div>`;
|
|
}
|
|
if (me.useColors) {
|
|
let bgColor = COLORS[data.keepName];
|
|
let textColor = TEXT_COLORS[data.keepName];
|
|
return `<div style="background-color: ${bgColor};color: ${textColor};">${text}</div>`;
|
|
} else {
|
|
return text;
|
|
}
|
|
},
|
|
flex: 1,
|
|
sortable: false,
|
|
},
|
|
{
|
|
header: 'Keep (reason)',
|
|
dataIndex: 'mark',
|
|
renderer: function(value, metaData, { data }) {
|
|
if (data.mark !== 'keep') {
|
|
return value;
|
|
}
|
|
if (data.keepCount) {
|
|
return `keep (${data.keepName}: ${data.keepCount})`;
|
|
} else {
|
|
return `keep (${data.keepName})`;
|
|
}
|
|
},
|
|
width: 200,
|
|
sortable: false,
|
|
},
|
|
],
|
|
}],
|
|
|
|
initComponent: function() {
|
|
let me = this;
|
|
|
|
if (!me.store) {
|
|
throw "no store specified";
|
|
}
|
|
me.callParent();
|
|
me.getViewModel().set('store', me.store);
|
|
},
|
|
});
|
|
|
|
Ext.define('PBS.prunesimulator.WeekTable', {
|
|
extend: 'Ext.panel.Panel',
|
|
alias: 'widget.prunesimulatorWeekTable',
|
|
|
|
reload: function() {
|
|
let me = this;
|
|
let backups = me.store.data.items;
|
|
|
|
let html = '<table class="cal">';
|
|
|
|
let now = new Date(me.up().getViewModel().get('now'));
|
|
let skip = 7 - parseInt(Ext.Date.format(now, 'N'), 10);
|
|
let tableStartDate = Ext.Date.add(now, Ext.Date.DAY, skip);
|
|
|
|
let bIndex = 0;
|
|
|
|
for (let i = 0; bIndex < backups.length; i++) {
|
|
html += '<tr>';
|
|
|
|
for (let j = 0; j < 7; j++) {
|
|
let date = Ext.Date.subtract(tableStartDate, Ext.Date.DAY, j + 7 * i);
|
|
let currentDay = Ext.Date.format(date, 'd/m/Y');
|
|
|
|
let dayOfWeekCls = Ext.Date.format(date, 'D').toLowerCase();
|
|
let firstOfMonthCls = Ext.Date.format(date, 'd') === '01'
|
|
? 'first-of-month'
|
|
: '';
|
|
html += `<td class="cal-day ${dayOfWeekCls} ${firstOfMonthCls}">`;
|
|
|
|
const isBackupOnDay = function(backup, day) {
|
|
return backup && Ext.Date.format(backup.data.backuptime, 'd/m/Y') === day;
|
|
};
|
|
|
|
let backup = backups[bIndex];
|
|
|
|
html += '<table><tr>';
|
|
html += `<th class="cal-day-date">${Ext.Date.format(date, 'D, d M Y')}</th>`;
|
|
|
|
while (isBackupOnDay(backup, currentDay)) {
|
|
html += '<tr><td>';
|
|
|
|
let text = Ext.Date.format(backup.data.backuptime, 'H:i');
|
|
if (backup.data.mark === 'remove') {
|
|
html += `<span class="strikethrough">${text}</span>`;
|
|
} else {
|
|
if (backup.data.keepCount) {
|
|
text += ` (${backup.data.keepName} ${backup.data.keepCount})`;
|
|
} else {
|
|
text += ` (${backup.data.keepName})`;
|
|
}
|
|
if (me.useColors) {
|
|
let bgColor = COLORS[backup.data.keepName];
|
|
let textColor = TEXT_COLORS[backup.data.keepName];
|
|
html += `<span style="background-color: ${bgColor}; color: ${textColor};">${text}</span>`;
|
|
} else {
|
|
html += `<span class="black">${text}</span>`;
|
|
}
|
|
}
|
|
html += '</td></tr>';
|
|
backup = backups[++bIndex];
|
|
}
|
|
html += '</table>';
|
|
html += '</div>';
|
|
html += '</td>';
|
|
}
|
|
|
|
html += '</tr>';
|
|
}
|
|
|
|
me.setHtml(html);
|
|
},
|
|
|
|
initComponent: function() {
|
|
let me = this;
|
|
|
|
if (!me.store) {
|
|
throw "no store specified";
|
|
}
|
|
|
|
let reload = function() {
|
|
me.reload();
|
|
};
|
|
|
|
me.store.on("datachanged", reload);
|
|
|
|
me.callParent();
|
|
|
|
me.reload();
|
|
},
|
|
});
|
|
|
|
Ext.define('PBS.PruneSimulatorKeepInput', {
|
|
extend: 'Ext.form.field.Number',
|
|
alias: 'widget.prunesimulatorKeepInput',
|
|
|
|
allowBlank: true,
|
|
fieldGroup: 'keep',
|
|
minValue: 1,
|
|
|
|
listeners: {
|
|
afterrender: function(field) {
|
|
this.triggers.clear.setVisible(field.value !== null);
|
|
},
|
|
change: function(field, newValue, oldValue) {
|
|
this.triggers.clear.setVisible(newValue !== null);
|
|
},
|
|
},
|
|
triggers: {
|
|
clear: {
|
|
cls: 'clear-trigger',
|
|
weight: -1,
|
|
handler: function() {
|
|
this.triggers.clear.setVisible(false);
|
|
this.setValue(null);
|
|
},
|
|
},
|
|
},
|
|
});
|
|
|
|
Ext.define('PBS.PruneSimulatorPanel', {
|
|
extend: 'Ext.panel.Panel',
|
|
alias: 'widget.prunesimulatorPanel',
|
|
|
|
viewModel: {
|
|
data: {
|
|
now: new Date(),
|
|
},
|
|
},
|
|
|
|
getValues: function() {
|
|
let me = this;
|
|
|
|
let values = {};
|
|
|
|
Ext.Array.each(me.query('[isFormField]'), function(field) {
|
|
let data = field.getSubmitData();
|
|
Ext.Object.each(data, function(name, val) {
|
|
values[name] = val;
|
|
});
|
|
});
|
|
|
|
return values;
|
|
},
|
|
|
|
controller: {
|
|
xclass: 'Ext.app.ViewController',
|
|
|
|
init: function(view) {
|
|
this.reloadFull(); // initial load
|
|
this.switchColor(true);
|
|
},
|
|
|
|
control: {
|
|
'field[fieldGroup=keep]': { change: 'reloadPrune' },
|
|
},
|
|
|
|
reloadFull: function() {
|
|
let me = this;
|
|
let view = me.getView();
|
|
|
|
let params = view.getValues();
|
|
|
|
let [hourSpec, minuteSpec] = params['schedule-time'].split(':');
|
|
|
|
if (!hourSpec || !minuteSpec) {
|
|
Ext.Msg.alert('Error', 'Invalid schedule');
|
|
return;
|
|
}
|
|
|
|
let matchTimeSpec = function(timeSpec, rangeMin, rangeMax) {
|
|
let specValues = timeSpec.split(',');
|
|
let matches = {};
|
|
|
|
let assertValid = function(value) {
|
|
let num = Number(value);
|
|
if (isNaN(num)) {
|
|
throw value + " is not an integer";
|
|
} else if (value < rangeMin || value > rangeMax) {
|
|
throw "number '" + value + "' is not in the range '" + rangeMin + ".." + rangeMax + "'";
|
|
}
|
|
return num;
|
|
};
|
|
|
|
specValues.forEach(function(value) {
|
|
if (value.includes('..')) {
|
|
let [start, end] = value.split('..');
|
|
start = assertValid(start);
|
|
end = assertValid(end);
|
|
if (start > end) {
|
|
throw "interval start is bigger then interval end '" + start + " > " + end + "'";
|
|
}
|
|
for (let i = start; i <= end; i++) {
|
|
matches[i] = 1;
|
|
}
|
|
} else if (value.includes('/')) {
|
|
let [start, step] = value.split('/');
|
|
start = assertValid(start);
|
|
step = assertValid(step);
|
|
for (let i = start; i <= rangeMax; i += step) {
|
|
matches[i] = 1;
|
|
}
|
|
} else if (value === '*') {
|
|
for (let i = rangeMin; i <= rangeMax; i++) {
|
|
matches[i] = 1;
|
|
}
|
|
} else {
|
|
value = assertValid(value);
|
|
matches[value] = 1;
|
|
}
|
|
});
|
|
|
|
return Object.keys(matches);
|
|
};
|
|
|
|
let hours, minutes;
|
|
try {
|
|
hours = matchTimeSpec(hourSpec, 0, 23);
|
|
minutes = matchTimeSpec(minuteSpec, 0, 59);
|
|
} catch (err) {
|
|
Ext.Msg.alert('Error', err);
|
|
return;
|
|
}
|
|
let formEl = view.down('form')?.el;
|
|
formEl?.mask(gettext('Please wait...'), 'x-mask-loading');
|
|
|
|
setTimeout(() => { // run re-calculation async afterwards to allow masking
|
|
let backups = me.populateFromSchedule(
|
|
params['schedule-weekdays'],
|
|
hours,
|
|
minutes,
|
|
params.numberOfWeeks,
|
|
);
|
|
|
|
me.pruneSelect(backups, params);
|
|
|
|
view.pruneStore.setData(backups);
|
|
|
|
formEl?.unmask();
|
|
}, 1);
|
|
},
|
|
|
|
reloadPrune: function() {
|
|
let me = this;
|
|
let view = me.getView();
|
|
|
|
let params = view.getValues();
|
|
|
|
let backups = [];
|
|
view.pruneStore.getData().items.forEach(function(item) {
|
|
backups.push({
|
|
backuptime: item.data.backuptime,
|
|
});
|
|
});
|
|
|
|
me.pruneSelect(backups, params);
|
|
|
|
view.pruneStore.setData(backups);
|
|
},
|
|
|
|
// backups are sorted descending by date
|
|
populateFromSchedule: function(weekdays, hours, minutes, weekCount) {
|
|
const me = this;
|
|
|
|
let weekdayFlags = ['sun', 'mon', 'tue', 'wed', 'thu', 'fri', 'sat']
|
|
.map(v => weekdays.includes(v));
|
|
|
|
const vmDate = me.getViewModel().get('now');
|
|
let todaysDate = new Date(vmDate);
|
|
|
|
let timesOnSingleDay = [];
|
|
|
|
hours.forEach(function(hour) {
|
|
minutes.forEach(function(minute) {
|
|
todaysDate.setHours(hour);
|
|
todaysDate.setMinutes(minute);
|
|
timesOnSingleDay.push(todaysDate.getTime());
|
|
});
|
|
});
|
|
|
|
// sort recent times first, backups array below is ordered now -> past
|
|
timesOnSingleDay.sort((a, b) => b - a);
|
|
|
|
let backups = [];
|
|
|
|
for (let i = 0; i < 7 * weekCount; i++) {
|
|
let daysDate = Ext.Date.subtract(todaysDate, Ext.Date.DAY, i);
|
|
let weekday = parseInt(Ext.Date.format(daysDate, 'w'), 10);
|
|
if (weekdayFlags[weekday]) {
|
|
timesOnSingleDay.forEach(function(time) {
|
|
const backuptime = Ext.Date.subtract(new Date(time), Ext.Date.DAY, i);
|
|
if (backuptime <= vmDate) {
|
|
backups.push({ backuptime: backuptime });
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
return backups;
|
|
},
|
|
|
|
pruneMark: function(backups, keepCount, keepName, idFunc) {
|
|
if (!keepCount) {
|
|
return;
|
|
}
|
|
|
|
let alreadyIncluded = {};
|
|
let newlyIncluded = {};
|
|
let newlyIncludedCount = 0;
|
|
|
|
let finished = false;
|
|
|
|
backups.forEach(function(backup) {
|
|
let mark = backup.mark;
|
|
if (mark && mark === 'keep') {
|
|
let id = idFunc(backup);
|
|
alreadyIncluded[id] = true;
|
|
}
|
|
});
|
|
|
|
backups.forEach(function(backup) {
|
|
let mark = backup.mark;
|
|
let id = idFunc(backup);
|
|
|
|
if (finished || alreadyIncluded[id] || mark) {
|
|
return;
|
|
}
|
|
|
|
if (!newlyIncluded[id]) {
|
|
if (newlyIncludedCount >= keepCount) {
|
|
finished = true;
|
|
return;
|
|
}
|
|
newlyIncluded[id] = true;
|
|
newlyIncludedCount++;
|
|
backup.mark = 'keep';
|
|
backup.keepName = keepName;
|
|
backup.keepCount = newlyIncludedCount;
|
|
} else {
|
|
backup.mark = 'remove';
|
|
}
|
|
});
|
|
},
|
|
|
|
// backups need to be sorted descending by date
|
|
pruneSelect: function(backups, keepParams) {
|
|
let me = this;
|
|
|
|
if (Number(keepParams['keep-last']) +
|
|
Number(keepParams['keep-hourly']) +
|
|
Number(keepParams['keep-daily']) +
|
|
Number(keepParams['keep-weekly']) +
|
|
Number(keepParams['keep-monthly']) +
|
|
Number(keepParams['keep-yearly']) === 0) {
|
|
backups.forEach(function(backup) {
|
|
backup.mark = 'keep';
|
|
backup.keepName = 'keep-all';
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
me.pruneMark(backups, keepParams['keep-last'], 'keep-last', function(backup) {
|
|
return backup.backuptime;
|
|
});
|
|
me.pruneMark(backups, keepParams['keep-hourly'], 'keep-hourly', function(backup) {
|
|
return Ext.Date.format(backup.backuptime, 'H/d/m/Y');
|
|
});
|
|
me.pruneMark(backups, keepParams['keep-daily'], 'keep-daily', function(backup) {
|
|
return Ext.Date.format(backup.backuptime, 'd/m/Y');
|
|
});
|
|
me.pruneMark(backups, keepParams['keep-weekly'], 'keep-weekly', function(backup) {
|
|
// ISO-8601 week and week-based year
|
|
return Ext.Date.format(backup.backuptime, 'W/o');
|
|
});
|
|
me.pruneMark(backups, keepParams['keep-monthly'], 'keep-monthly', function(backup) {
|
|
return Ext.Date.format(backup.backuptime, 'm/Y');
|
|
});
|
|
me.pruneMark(backups, keepParams['keep-yearly'], 'keep-yearly', function(backup) {
|
|
return Ext.Date.format(backup.backuptime, 'Y');
|
|
});
|
|
|
|
backups.forEach(function(backup) {
|
|
backup.mark = backup.mark || 'remove';
|
|
});
|
|
},
|
|
|
|
toggleColors: function(checkbox, checked) {
|
|
this.switchColor(checked);
|
|
},
|
|
|
|
switchColor: function(useColors) {
|
|
let me = this;
|
|
let view = me.getView();
|
|
|
|
const getStyle = name =>
|
|
`background-color: ${COLORS[name]}; color: ${TEXT_COLORS[name]};`;
|
|
|
|
for (const field of view.query('[isFormField]')) {
|
|
if (field.fieldGroup !== 'keep') {
|
|
continue;
|
|
}
|
|
if (useColors) {
|
|
field.setFieldStyle(getStyle(field.name));
|
|
} else {
|
|
field.setFieldStyle('background-color: white; color: #444;');
|
|
}
|
|
}
|
|
|
|
me.lookup('weekTable').useColors = useColors;
|
|
me.lookup('pruneList').useColors = useColors;
|
|
|
|
me.reloadPrune();
|
|
},
|
|
},
|
|
|
|
keepItems: [
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-last',
|
|
fieldLabel: 'keep-last',
|
|
value: 4,
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-hourly',
|
|
fieldLabel: 'keep-hourly',
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-daily',
|
|
fieldLabel: 'keep-daily',
|
|
value: 5,
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-weekly',
|
|
fieldLabel: 'keep-weekly',
|
|
value: 2,
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-monthly',
|
|
fieldLabel: 'keep-monthly',
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorKeepInput',
|
|
name: 'keep-yearly',
|
|
fieldLabel: 'keep-yearly',
|
|
},
|
|
],
|
|
|
|
initComponent: function() {
|
|
var me = this;
|
|
const vm = me.getViewModel();
|
|
|
|
me.pruneStore = Ext.create('Ext.data.Store', {
|
|
model: 'pbs-prune-list',
|
|
sorters: { property: 'backuptime', direction: 'DESC' },
|
|
});
|
|
|
|
me.items = [
|
|
{
|
|
xtype: 'panel',
|
|
layout: {
|
|
type: 'hbox',
|
|
align: 'stretch',
|
|
},
|
|
border: false,
|
|
items: [
|
|
{
|
|
title: 'View Options',
|
|
layout: 'anchor',
|
|
flex: 1,
|
|
border: false,
|
|
bodyPadding: 10,
|
|
items: [
|
|
{
|
|
xtype: 'checkbox',
|
|
name: 'showCalendar',
|
|
reference: 'showCalendar',
|
|
fieldLabel: 'Show Calendar:',
|
|
checked: true,
|
|
},
|
|
{
|
|
xtype: 'checkbox',
|
|
name: 'showColors',
|
|
reference: 'showColors',
|
|
fieldLabel: 'Show Colors:',
|
|
checked: true,
|
|
handler: 'toggleColors',
|
|
},
|
|
],
|
|
},
|
|
{ xtype: "panel", width: 1, border: 1 },
|
|
{
|
|
xtype: 'form',
|
|
layout: 'hbox',
|
|
flex: 2,
|
|
border: false,
|
|
title: 'Backup Job Simulation',
|
|
dockedItems: [{
|
|
xtype: 'button',
|
|
text: 'Update Simulation',
|
|
handler: 'reloadFull',
|
|
formBind: true,
|
|
dock: 'bottom',
|
|
margin: '1 15',
|
|
}],
|
|
bodyPadding: 3,
|
|
items: [
|
|
{
|
|
xtype: 'fieldset',
|
|
title: 'Backup Job',
|
|
layout: 'anchor',
|
|
flex: 4,
|
|
height: 110,
|
|
defaults: {
|
|
labelWidth: 90,
|
|
padding: '0 0 0 10',
|
|
width: '95%',
|
|
minWidth: 150,
|
|
},
|
|
items: [
|
|
{
|
|
xtype: 'prunesimulatorDayOfWeekSelector',
|
|
name: 'schedule-weekdays',
|
|
fieldLabel: 'Day of week',
|
|
value: ['mon', 'tue', 'wed', 'thu', 'fri', 'sat', 'sun'],
|
|
allowBlank: false,
|
|
multiSelect: true,
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorCalendarEvent',
|
|
name: 'schedule-time',
|
|
allowBlank: false,
|
|
value: '0/6:00',
|
|
fieldLabel: 'Schedule',
|
|
},
|
|
],
|
|
},
|
|
{
|
|
xtype: 'fieldset',
|
|
title: 'Simulation Time Range',
|
|
layout: 'anchor',
|
|
flex: 3,
|
|
height: 110,
|
|
defaults: {
|
|
labelWidth: 70,
|
|
width: 220,
|
|
padding: '0 0 0 10',
|
|
width: '95%',
|
|
minWidth: 150,
|
|
},
|
|
items: [
|
|
{
|
|
xtype: 'datefield',
|
|
name: 'currentDate',
|
|
fieldLabel: 'End Date',
|
|
allowBlank: false,
|
|
format: 'Y-m-d',
|
|
value: vm.get('now'),
|
|
listeners: {
|
|
change: function(self, newDate) {
|
|
if (!self.isValid()) {
|
|
return;
|
|
}
|
|
let date = me.getViewModel().get('now');
|
|
date.setFullYear(
|
|
newDate.getFullYear(),
|
|
newDate.getMonth(),
|
|
newDate.getDate(),
|
|
);
|
|
},
|
|
},
|
|
},
|
|
{
|
|
xtype: 'timefield',
|
|
name: 'currentTime',
|
|
reference: 'currentTime',
|
|
fieldLabel: 'End Time',
|
|
allowBlank: false,
|
|
format: 'H:i',
|
|
// cant bind value because ExtJS sets the year to 2008 to
|
|
// protect against DST issues and date picker zeroes hour/minute
|
|
value: vm.get('now'),
|
|
listeners: {
|
|
change: function(self, time) {
|
|
if (!self.isValid()) {
|
|
return;
|
|
}
|
|
let date = me.getViewModel().get('now');
|
|
date.setHours(time.getHours());
|
|
date.setMinutes(time.getMinutes());
|
|
},
|
|
},
|
|
},
|
|
{
|
|
xtype: 'fieldcontainer',
|
|
fieldLabel: 'Duration',
|
|
layout: 'hbox',
|
|
items: [{
|
|
xtype: 'numberfield',
|
|
name: 'numberOfWeeks',
|
|
hideLabel: true,
|
|
allowBlank: false,
|
|
minValue: 1,
|
|
value: 15,
|
|
maxValue: 260, // five years
|
|
flex: 1,
|
|
}, {
|
|
xtype: 'displayfield',
|
|
value: 'Weeks',
|
|
submitValue: false,
|
|
hideLabel: true,
|
|
padding: '0 0 0 5',
|
|
width: 40,
|
|
}],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
],
|
|
},
|
|
{
|
|
xtype: 'panel',
|
|
layout: {
|
|
type: 'hbox',
|
|
align: 'stretch',
|
|
},
|
|
flex: 1,
|
|
border: false,
|
|
items: [
|
|
{
|
|
layout: 'anchor',
|
|
title: 'Prune Options',
|
|
border: false,
|
|
bodyPadding: 10,
|
|
scrollable: true,
|
|
items: me.keepItems,
|
|
flex: 1,
|
|
},
|
|
{ xtype: "panel", width: 1, border: 1 },
|
|
{
|
|
layout: 'fit',
|
|
title: 'Backups',
|
|
border: false,
|
|
xtype: 'prunesimulatorPruneList',
|
|
store: me.pruneStore,
|
|
reference: 'pruneList',
|
|
flex: 2,
|
|
},
|
|
],
|
|
},
|
|
{
|
|
layout: 'anchor',
|
|
title: 'Calendar',
|
|
autoScroll: true,
|
|
flex: 2,
|
|
xtype: 'prunesimulatorWeekTable',
|
|
reference: 'weekTable',
|
|
store: me.pruneStore,
|
|
bind: {
|
|
hidden: '{!showCalendar.checked}',
|
|
},
|
|
},
|
|
];
|
|
|
|
me.callParent();
|
|
},
|
|
});
|
|
|
|
Ext.create('Ext.container.Viewport', {
|
|
layout: 'border',
|
|
renderTo: Ext.getBody(),
|
|
items: [
|
|
{
|
|
xtype: 'prunesimulatorPanel',
|
|
title: 'Proxmox Backup Server - Prune Simulator',
|
|
region: 'west',
|
|
layout: {
|
|
type: 'vbox',
|
|
align: 'stretch',
|
|
pack: 'start',
|
|
},
|
|
flex: 3,
|
|
maxWidth: 1090,
|
|
},
|
|
{
|
|
xtype: 'prunesimulatorDocumentation',
|
|
title: 'Usage',
|
|
border: false,
|
|
flex: 2,
|
|
region: 'center',
|
|
},
|
|
],
|
|
});
|
|
});
|
|
|