pve-manager/www/manager6/window/BulkAction.js
Thomas Lamprecht a75bace1f4 ui: bulk shutdown: expose new timeout and force-stop to user
Signed-off-by: Thomas Lamprecht <t.lamprecht@proxmox.com>
2023-01-14 17:41:04 +01:00

191 lines
4.0 KiB
JavaScript

Ext.define('PVE.window.BulkAction', {
extend: 'Ext.window.Window',
resizable: true,
width: 800,
modal: true,
layout: {
type: 'fit',
},
border: false,
// the action to set, currently there are: `startall`, `migrateall`, `stopall`
action: undefined,
submit: function(params) {
let me = this;
Proxmox.Utils.API2Request({
params: params,
url: `/nodes/${me.nodename}/${me.action}`,
waitMsgTarget: me,
method: 'POST',
failure: response => Ext.Msg.alert('Error', response.htmlStatus),
success: function({ result }, options) {
Ext.create('Proxmox.window.TaskViewer', {
autoShow: true,
upid: result.data,
listeners: {
destroy: () => me.close(),
},
});
me.hide();
},
});
},
initComponent: function() {
let me = this;
if (!me.nodename) {
throw "no node name specified";
}
if (!me.action) {
throw "no action specified";
}
if (!me.btnText) {
throw "no button text specified";
}
if (!me.title) {
throw "no title specified";
}
let items = [];
if (me.action === 'migrateall') {
items.push(
{
xtype: 'pveNodeSelector',
name: 'target',
disallowedNodes: [me.nodename],
fieldLabel: gettext('Target node'),
allowBlank: false,
onlineValidator: true,
},
{
xtype: 'proxmoxintegerfield',
name: 'maxworkers',
minValue: 1,
maxValue: 100,
value: 1,
fieldLabel: gettext('Parallel jobs'),
allowBlank: false,
},
{
xtype: 'fieldcontainer',
fieldLabel: gettext('Allow local disk migration'),
layout: 'hbox',
items: [{
xtype: 'proxmoxcheckbox',
name: 'with-local-disks',
checked: true,
uncheckedValue: 0,
listeners: {
change: (cb, val) => me.down('#localdiskwarning').setVisible(val),
},
},
{
itemId: 'localdiskwarning',
xtype: 'displayfield',
flex: 1,
padding: '0 0 0 10',
userCls: 'pmx-hint',
value: 'Note: Migration with local disks might take long.',
}],
},
{
itemId: 'lxcwarning',
xtype: 'displayfield',
userCls: 'pmx-hint',
value: 'Warning: Running CTs will be migrated in Restart Mode.',
hidden: true, // only visible if running container chosen
},
);
} else if (me.action === 'startall') {
items.push({
xtype: 'hiddenfield',
name: 'force',
value: 1,
});
} else if (me.action === 'stopall') {
items.push(
{
xtype: 'proxmoxcheckbox',
name: 'force-stop',
fieldLabel: gettext('Force Stop'),
boxLabel: gettext('Force stop guest if shutdown times out.'),
checked: true,
uncheckedValue: 0,
},
{
xtype: 'proxmoxintegerfield',
name: 'timeout',
fieldLabel: gettext('Timeout (s)'),
emptyText: '180',
minValue: 0,
maxValue: 7200,
allowBlank: true,
},
);
}
items.push({
xtype: 'vmselector',
itemId: 'vms',
name: 'vms',
flex: 1,
height: 300,
selectAll: true,
allowBlank: false,
nodename: me.nodename,
action: me.action,
listeners: {
selectionchange: function(vmselector, records) {
if (me.action === 'migrateall') {
let showWarning = records.some(
item => item.data.type === 'lxc' && item.data.status === 'running',
);
me.down('#lxcwarning').setVisible(showWarning);
}
},
},
});
me.formPanel = Ext.create('Ext.form.Panel', {
bodyPadding: 10,
border: false,
layout: {
type: 'vbox',
align: 'stretch',
},
fieldDefaults: {
labelWidth: me.action === 'migrateall' ? 300 : 120,
anchor: '100%',
},
items: items,
});
let form = me.formPanel.getForm();
let submitBtn = Ext.create('Ext.Button', {
text: me.btnText,
handler: function() {
form.isValid();
me.submit(form.getValues());
},
});
Ext.apply(me, {
items: [me.formPanel],
buttons: [submitBtn],
});
me.callParent();
form.on('validitychange', function() {
let valid = form.isValid();
submitBtn.setDisabled(!valid);
});
form.isValid();
},
});