]> git.proxmox.com Git - pve-manager.git/blame - www/manager6/lxc/Resources.js
ui: guest remove: modernize code and cleanup
[pve-manager.git] / www / manager6 / lxc / Resources.js
CommitLineData
c7ee0c11 1Ext.define('PVE.lxc.RessourceView', {
9fd8d73e 2 extend: 'Proxmox.grid.PendingObjectGrid',
c7ee0c11
DM
3 alias: ['widget.pveLxcRessourceView'],
4
ba93a9c6
DC
5 onlineHelp: 'pct_configuration',
6
c7ee0c11
DM
7 renderKey: function(key, metaData, rec, rowIndex, colIndex, store) {
8 var me = this;
14dd743b 9 var rowdef = me.rows[key] || {};
c7ee0c11
DM
10
11 metaData.tdAttr = "valign=middle";
c7ee0c11
DM
12 if (rowdef.tdCls) {
13 metaData.tdCls = rowdef.tdCls;
c7ee0c11
DM
14 }
15 return rowdef.header || key;
16 },
17
8058410f 18 initComponent: function() {
c7ee0c11 19 var me = this;
55ee6ba1 20 let confid;
c7ee0c11
DM
21
22 var nodename = me.pveSelNode.data.node;
2a4971d8 23 if (!nodename) {
c7ee0c11
DM
24 throw "no node name specified";
25 }
26
27 var vmid = me.pveSelNode.data.vmid;
28 if (!vmid) {
29 throw "no VM ID specified";
30 }
31
32 var caps = Ext.state.Manager.get('GuiCap');
d35b5b2a 33 var diskCap = caps.vms['VM.Config.Disk'];
c7ee0c11
DM
34
35 var mpeditor = caps.vms['VM.Config.Disk'] ? 'PVE.lxc.MountPointEdit' : undefined;
36
37 var rows = {
38 memory: {
39 header: gettext('Memory'),
40 editor: caps.vms['VM.Config.Memory'] ? 'PVE.lxc.MemoryEdit' : undefined,
c7ee0c11 41 defaultValue: 512,
a8ea1b68 42 tdCls: 'pmx-itype-icon-memory',
1a2fdf62 43 group: 1,
c7ee0c11 44 renderer: function(value) {
e7ade592 45 return Proxmox.Utils.format_size(value*1024*1024);
f6710aac 46 },
c7ee0c11
DM
47 },
48 swap: {
49 header: gettext('Swap'),
50 editor: caps.vms['VM.Config.Memory'] ? 'PVE.lxc.MemoryEdit' : undefined,
c7ee0c11
DM
51 defaultValue: 512,
52 tdCls: 'pve-itype-icon-swap',
1a2fdf62 53 group: 2,
c7ee0c11 54 renderer: function(value) {
e7ade592 55 return Proxmox.Utils.format_size(value*1024*1024);
f6710aac 56 },
c7ee0c11 57 },
92b5029f
DM
58 cores: {
59 header: gettext('Cores'),
92b5029f
DM
60 editor: caps.vms['VM.Config.CPU'] ? 'PVE.lxc.CPUEdit' : undefined,
61 defaultValue: '',
a8ea1b68 62 tdCls: 'pmx-itype-icon-processor',
1a2fdf62 63 group: 3,
92b5029f 64 renderer: function(value) {
3e4b78dc
DC
65 var cpulimit = me.getObjectValue('cpulimit');
66 var cpuunits = me.getObjectValue('cpuunits');
67 var res;
68 if (value) {
69 res = value;
70 } else {
71 res = gettext('unlimited');
72 }
73
74 if (cpulimit) {
75 res += ' [cpulimit=' + cpulimit + ']';
76 }
77
78 if (cpuunits) {
79 res += ' [cpuunits=' + cpuunits + ']';
80 }
81 return res;
f6710aac 82 },
92b5029f 83 },
c7ee0c11
DM
84 rootfs: {
85 header: gettext('Root Disk'),
e7ade592 86 defaultValue: Proxmox.Utils.noneText,
c7ee0c11 87 editor: mpeditor,
1a2fdf62 88 tdCls: 'pve-itype-icon-storage',
f6710aac 89 group: 4,
c15e9cd5 90 },
3e4b78dc 91 cpulimit: {
f6710aac 92 visible: false,
3e4b78dc
DC
93 },
94 cpuunits: {
f6710aac 95 visible: false,
3e4b78dc 96 },
c15e9cd5 97 unprivileged: {
f6710aac
TL
98 visible: false,
99 },
c7ee0c11
DM
100 };
101
14a845bc
DC
102 PVE.Utils.forEachMP(function(bus, i) {
103 confid = bus + i;
1a2fdf62
DL
104 var group = 5;
105 var header;
14a845bc
DC
106 if (bus === 'mp') {
107 header = gettext('Mount Point') + ' (' + confid + ')';
108 } else {
109 header = gettext('Unused Disk') + ' ' + i;
1a2fdf62 110 group += 1;
14a845bc 111 }
c7ee0c11 112 rows[confid] = {
1a2fdf62
DL
113 group: group,
114 order: i,
c7ee0c11
DM
115 tdCls: 'pve-itype-icon-storage',
116 editor: mpeditor,
f6710aac 117 header: header,
c7ee0c11 118 };
14a845bc 119 }, true);
c7ee0c11 120
c7ee0c11
DM
121 var baseurl = 'nodes/' + nodename + '/lxc/' + vmid + '/config';
122
14dd743b 123 me.selModel = Ext.create('Ext.selection.RowModel', {});
c7ee0c11
DM
124
125 var run_resize = function() {
14dd743b 126 var rec = me.selModel.getSelection()[0];
c7ee0c11
DM
127 if (!rec) {
128 return;
129 }
130
131 var win = Ext.create('PVE.window.MPResize', {
132 disk: rec.data.key,
133 nodename: nodename,
f6710aac 134 vmid: vmid,
c7ee0c11
DM
135 });
136
137 win.show();
c7ee0c11
DM
138 };
139
140 var run_remove = function(b, e, rec) {
e7ade592 141 Proxmox.Utils.API2Request({
c7ee0c11
DM
142 url: '/api2/extjs/' + baseurl,
143 waitMsgTarget: me,
144 method: 'PUT',
145 params: {
f6710aac 146 'delete': rec.data.key,
c7ee0c11 147 },
8058410f 148 failure: function(response, opts) {
c7ee0c11 149 Ext.Msg.alert('Error', response.htmlStatus);
f6710aac 150 },
c7ee0c11
DM
151 });
152 };
153
c7164db7
DC
154 var run_move = function(b, e, rec) {
155 if (!rec) {
156 return;
157 }
158
159 var win = Ext.create('PVE.window.HDMove', {
160 disk: rec.data.key,
161 nodename: nodename,
162 vmid: vmid,
f6710aac 163 type: 'lxc',
c7164db7
DC
164 });
165
166 win.show();
167
168 win.on('destroy', me.reload, me);
169 };
170
5720fafa 171 var edit_btn = new Proxmox.button.Button({
c7ee0c11 172 text: gettext('Edit'),
14dd743b 173 selModel: me.selModel,
c7ee0c11
DM
174 disabled: true,
175 enableFn: function(rec) {
176 if (!rec) {
177 return false;
178 }
179 var rowdef = rows[rec.data.key];
180 return !!rowdef.editor;
181 },
f6710aac 182 handler: function() { me.run_editor(); },
c7ee0c11
DM
183 });
184
5720fafa 185 var resize_btn = new Proxmox.button.Button({
c7ee0c11 186 text: gettext('Resize disk'),
14dd743b 187 selModel: me.selModel,
c7ee0c11 188 disabled: true,
f6710aac 189 handler: run_resize,
c7ee0c11
DM
190 });
191
5720fafa 192 var remove_btn = new Proxmox.button.Button({
c7ee0c11 193 text: gettext('Remove'),
986ccfe4
AL
194 defaultText: gettext('Remove'),
195 altText: gettext('Detach'),
14dd743b 196 selModel: me.selModel,
c7ee0c11
DM
197 disabled: true,
198 dangerous: true,
199 confirmMsg: function(rec) {
986ccfe4
AL
200 let warn = Ext.String.format(gettext('Are you sure you want to remove entry {0}'));
201 if (this.text === this.altText) {
202 warn = gettext('Are you sure you want to detach entry {0}');
203 }
cc80f765
TL
204 let rendered = me.renderKey(rec.data.key, {}, rec);
205 let msg = Ext.String.format(warn, `'${rendered}'`);
986ccfe4 206
c7ee0c11 207 if (rec.data.key.match(/^unused\d+$/)) {
16152937 208 msg += " " + gettext('This will permanently erase all data.');
c7ee0c11 209 }
c7ee0c11
DM
210 return msg;
211 },
f6710aac 212 handler: run_remove,
986ccfe4
AL
213 listeners: {
214 render: function(btn) {
215 // hack: calculate the max button width on first display to prevent the whole
216 // toolbar to move when we switch between the "Remove" and "Detach" labels
217 let def = btn.getSize().width;
218
219 btn.setText(btn.altText);
220 let alt = btn.getSize().width;
221
222 btn.setText(btn.defaultText);
223
224 let optimal = alt > def ? alt : def;
225 btn.setSize({ width: optimal });
226 },
227 },
c7ee0c11
DM
228 });
229
c7164db7
DC
230 var move_btn = new Proxmox.button.Button({
231 text: gettext('Move Volume'),
232 selModel: me.selModel,
233 disabled: true,
234 dangerous: true,
f6710aac 235 handler: run_move,
c7164db7
DC
236 });
237
417f904f 238 var revert_btn = new PVE.button.PendingRevert();
273b5ce3 239
c7ee0c11 240 var set_button_status = function() {
14dd743b 241 var rec = me.selModel.getSelection()[0];
c7ee0c11
DM
242
243 if (!rec) {
244 edit_btn.disable();
245 remove_btn.disable();
246 resize_btn.disable();
273b5ce3 247 revert_btn.disable();
c7ee0c11
DM
248 return;
249 }
250 var key = rec.data.key;
251 var value = rec.data.value;
252 var rowdef = rows[key];
253
399ffa76 254 var pending = rec.data.delete || me.hasPendingChanges(key);
91a47a76 255 var isDisk = rowdef.tdCls === 'pve-itype-icon-storage';
5ef877b9 256 var isUnusedDisk = key.match(/^unused\d+/);
986ccfe4 257 var isUsedDisk = isDisk && !isUnusedDisk;
c7ee0c11 258
399ffa76 259 var noedit = rec.data.delete || !rowdef.editor;
35a04562 260 if (!noedit && Proxmox.UserName !== 'root@pam' && key.match(/^mp\d+$/)) {
c7ee0c11
DM
261 var mp = PVE.Parser.parseLxcMountPoint(value);
262 if (mp.type !== 'volume') {
263 noedit = true;
264 }
265 }
266 edit_btn.setDisabled(noedit);
267
cef9b653 268 remove_btn.setDisabled(!isDisk || rec.data.key === 'rootfs' || !diskCap || pending);
5ef877b9 269 resize_btn.setDisabled(!isDisk || !diskCap || isUnusedDisk);
d35b5b2a 270 move_btn.setDisabled(!isDisk || !diskCap);
273b5ce3 271 revert_btn.setDisabled(!pending);
986ccfe4
AL
272
273 remove_btn.setText(isUsedDisk ? remove_btn.altText : remove_btn.defaultText);
c7ee0c11 274 };
2a4971d8 275
1a2fdf62
DL
276 var sorterFn = function(rec1, rec2) {
277 var v1 = rec1.data.key;
278 var v2 = rec2.data.key;
279 var g1 = rows[v1].group || 0;
280 var g2 = rows[v2].group || 0;
281 var order1 = rows[v1].order || 0;
282 var order2 = rows[v2].order || 0;
283
53e3ea84 284 if (g1 - g2 !== 0) {
1a2fdf62
DL
285 return g1 - g2;
286 }
287
53e3ea84 288 if (order1 - order2 !== 0) {
1a2fdf62
DL
289 return order1 - order2;
290 }
291
292 if (v1 > v2) {
293 return 1;
294 } else if (v1 < v2) {
295 return -1;
296 } else {
297 return 0;
298 }
93bd0d75 299 };
1a2fdf62 300
f7993618 301 Ext.apply(me, {
9fd8d73e 302 url: "/api2/json/nodes/" + nodename + "/lxc/" + vmid + "/pending",
14dd743b
TL
303 selModel: me.selModel,
304 interval: 2000,
c7ee0c11
DM
305 cwidth1: 170,
306 tbar: [
307 {
308 text: gettext('Add'),
309 menu: new Ext.menu.Menu({
310 items: [
311 {
312 text: gettext('Mount Point'),
313 iconCls: 'pve-itype-icon-storage',
314 disabled: !caps.vms['VM.Config.Disk'],
315 handler: function() {
316 var win = Ext.create('PVE.lxc.MountPointEdit', {
317 url: '/api2/extjs/' + baseurl,
c15e9cd5 318 unprivileged: me.getObjectValue('unprivileged'),
f6710aac 319 pveSelNode: me.pveSelNode,
c7ee0c11 320 });
69f36699 321 win.on('destroy', me.reload, me);
c7ee0c11 322 win.show();
f6710aac
TL
323 },
324 },
325 ],
326 }),
c7ee0c11
DM
327 },
328 edit_btn,
329 remove_btn,
c7164db7 330 resize_btn,
273b5ce3 331 move_btn,
f6710aac 332 revert_btn,
c7ee0c11
DM
333 ],
334 rows: rows,
1a2fdf62 335 sorterFn: sorterFn,
14dd743b
TL
336 editorConfig: {
337 pveSelNode: me.pveSelNode,
f6710aac 338 url: '/api2/extjs/' + baseurl,
14dd743b 339 },
c7ee0c11 340 listeners: {
14dd743b 341 itemdblclick: me.run_editor,
f6710aac
TL
342 selectionchange: set_button_status,
343 },
c7ee0c11
DM
344 });
345
346 me.callParent();
14dd743b
TL
347
348 me.on('activate', me.rstore.startUpdate);
349 me.on('destroy', me.rstore.stopUpdate);
350 me.on('deactivate', me.rstore.stopUpdate);
351
69f36699
DC
352 me.mon(me.getStore(), 'datachanged', function() {
353 set_button_status();
354 });
355
14dd743b 356 Ext.apply(me.editorConfig, { unprivileged: me.getObjectValue('unprivileged') });
f6710aac 357 },
c7ee0c11 358});