]> git.proxmox.com Git - pve-manager.git/blame - www/manager6/window/Clone.js
ui: fix column behavior with browser scaling
[pve-manager.git] / www / manager6 / window / Clone.js
CommitLineData
f5226cd9
DM
1Ext.define('PVE.window.Clone', {
2 extend: 'Ext.window.Window',
3
4 resizable: false,
5
6 isTemplate: false,
7
ff79a116
DM
8 onlineHelp: 'qm_copy_and_clone',
9
26693bb9
EK
10 controller: {
11 xclass: 'Ext.app.ViewController',
12 control: {
13 'panel[reference=cloneform]': {
14 validitychange: 'disableSubmit'
15 }
16 },
17 disableSubmit: function(form) {
18 this.lookupReference('submitBtn').setDisabled(!form.isValid());
19 }
20 },
21
4e2b6cc8
EK
22 statics: {
23 // display a snapshot selector only if needed
0718aefc 24 wrap: function(nodename, vmid, isTemplate, guestType) {
e7ade592 25 Proxmox.Utils.API2Request({
0718aefc 26 url: '/nodes/' + nodename + '/' + guestType + '/' + vmid +'/snapshot',
4e2b6cc8
EK
27 failure: function(response, opts) {
28 Ext.Msg.alert('Error', response.htmlStatus);
29 },
30 success: function(response, opts) {
31 var snapshotList = response.result.data;
32 var hasSnapshots = snapshotList.length === 1 &&
33 snapshotList[0].name === 'current' ? false : true;
34
35 Ext.create('PVE.window.Clone', {
36 nodename: nodename,
0718aefc 37 guestType: guestType,
4e2b6cc8
EK
38 vmid: vmid,
39 isTemplate: isTemplate,
40 hasSnapshots: hasSnapshots
41 }).show();
42 }
43 });
44 }
45 },
2f3c5c38 46
f5226cd9
DM
47 create_clone: function(values) {
48 var me = this;
49
406ac585 50 var params = { newid: values.newvmid };
f5226cd9 51
406ac585
EK
52 if (values.snapname && values.snapname !== 'current') {
53 params.snapname = values.snapname;
54 }
f5226cd9
DM
55
56 if (values.pool) {
57 params.pool = values.pool;
58 }
59
60 if (values.name) {
0718aefc
DC
61 if (me.guestType === 'lxc') {
62 params.hostname = values.name;
63 } else {
64 params.name = values.name;
65 }
f5226cd9
DM
66 }
67
68 if (values.target) {
69 params.target = values.target;
70 }
71
72 if (values.clonemode === 'copy') {
73 params.full = 1;
1490ff3d
DC
74 if (values.hdstorage) {
75 params.storage = values.hdstorage;
0718aefc 76 if (values.diskformat && me.guestType !== 'lxc') {
f5226cd9
DM
77 params.format = values.diskformat;
78 }
79 }
80 }
81
e7ade592 82 Proxmox.Utils.API2Request({
f5226cd9 83 params: params,
0718aefc 84 url: '/nodes/' + me.nodename + '/' + me.guestType + '/' + me.vmid + '/clone',
f5226cd9
DM
85 waitMsgTarget: me,
86 method: 'POST',
87 failure: function(response, opts) {
88 Ext.Msg.alert('Error', response.htmlStatus);
89 },
90 success: function(response, options) {
91 me.close();
92 }
93 });
94
95 },
96
44494b4c 97 // disable the Storage selector when clone mode is linked clone
f5226cd9
DM
98 updateVisibility: function() {
99 var me = this;
406ac585 100 var clonemode = me.lookupReference('clonemodesel').getValue();
1490ff3d
DC
101 var disksel = me.lookup('diskselector');
102 disksel.setDisabled(clonemode === 'clone');
f5226cd9
DM
103 },
104
26693bb9 105 // add to the list of valid nodes each node where
44494b4c 106 // all the VM disks are available
f5226cd9
DM
107 verifyFeature: function() {
108 var me = this;
406ac585
EK
109
110 var snapname = me.lookupReference('snapshotsel').getValue();
111 var clonemode = me.lookupReference('clonemodesel').getValue();
f5226cd9
DM
112
113 var params = { feature: clonemode };
114 if (snapname !== 'current') {
115 params.snapname = snapname;
116 }
117
e7ade592 118 Proxmox.Utils.API2Request({
f5226cd9 119 waitMsgTarget: me,
0718aefc 120 url: '/nodes/' + me.nodename + '/' + me.guestType + '/' + me.vmid + '/feature',
f5226cd9
DM
121 params: params,
122 method: 'GET',
123 failure: function(response, opts) {
0057ea5c 124 me.lookupReference('submitBtn').setDisabled(true);
f5226cd9
DM
125 Ext.Msg.alert('Error', response.htmlStatus);
126 },
127 success: function(response, options) {
406ac585 128 var res = response.result.data;
f5226cd9 129
406ac585
EK
130 me.lookupReference('targetsel').allowedNodes = res.nodes;
131 me.lookupReference('targetsel').validate();
f5226cd9
DM
132 }
133 });
134 },
135
136 initComponent : function() {
f5226cd9
DM
137 var me = this;
138
139 if (!me.nodename) {
140 throw "no node name specified";
141 }
142
143 if (!me.vmid) {
144 throw "no VM ID specified";
145 }
146
147 if (!me.snapname) {
148 me.snapname = 'current';
149 }
150
0718aefc
DC
151 if (!me.guestType) {
152 throw "no Guest Type specified";
153 }
154
155 var titletext = me.guestType === 'lxc' ? 'CT' : 'VM';
156 if (me.isTemplate) {
157 titletext += ' Template';
158 }
406ac585
EK
159 me.title = "Clone " + titletext + " " + me.vmid;
160
f5226cd9
DM
161 var col1 = [];
162 var col2 = [];
163
406ac585
EK
164 col1.push({
165 xtype: 'pveNodeSelector',
f5226cd9 166 name: 'target',
406ac585 167 reference: 'targetsel',
f5226cd9
DM
168 fieldLabel: gettext('Target node'),
169 selectCurNode: true,
170 allowBlank: false,
406ac585
EK
171 onlineValidator: true,
172 listeners: {
173 change: function(f, value) {
1490ff3d 174 me.lookupReference('hdstorage').setTargetNode(value);
406ac585
EK
175 }
176 }
f5226cd9
DM
177 });
178
f5226cd9
DM
179 var modelist = [['copy', gettext('Full Clone')]];
180 if (me.isTemplate) {
181 modelist.push(['clone', gettext('Linked Clone')]);
182 }
183
406ac585
EK
184 col1.push({
185 xtype: 'pveGuestIDSelector',
186 name: 'newvmid',
0718aefc 187 guestType: me.guestType,
406ac585
EK
188 value: '',
189 loadNextFreeID: true,
190 validateExists: false
191 },
192 {
193 xtype: 'textfield',
194 name: 'name',
195 allowBlank: true,
0718aefc 196 fieldLabel: me.guestType === 'lxc' ? gettext('Hostname') : gettext('Name')
406ac585
EK
197 },
198 {
199 xtype: 'pvePoolSelector',
200 fieldLabel: gettext('Resource Pool'),
201 name: 'pool',
202 value: '',
203 allowBlank: true
204 }
205 );
f5226cd9 206
406ac585 207 col2.push({
09cacce7 208 xtype: 'proxmoxKVComboBox',
406ac585
EK
209 fieldLabel: gettext('Mode'),
210 name: 'clonemode',
211 reference: 'clonemodesel',
212 allowBlank: false,
213 hidden: !me.isTemplate,
214 value: me.isTemplate ? 'clone' : 'copy',
dbbe181c
TL
215 comboItems: modelist,
216 listeners: {
217 change: function(t, value) {
218 me.updateVisibility();
219 me.verifyFeature();
220 }
221 }
406ac585
EK
222 },
223 {
224 xtype: 'PVE.form.SnapshotSelector',
f5226cd9 225 name: 'snapname',
406ac585 226 reference: 'snapshotsel',
f5226cd9 227 fieldLabel: gettext('Snapshot'),
406ac585 228 nodename: me.nodename,
0718aefc 229 guestType: me.guestType,
406ac585 230 vmid: me.vmid,
2f3c5c38 231 hidden: me.isTemplate || !me.hasSnapshots ? true : false,
406ac585 232 disabled: false,
f5226cd9
DM
233 allowBlank: false,
234 value : me.snapname,
235 listeners: {
236 change: function(f, value) {
237 me.verifyFeature();
238 }
239 }
406ac585
EK
240 },
241 {
1490ff3d
DC
242 xtype: 'pveDiskStorageSelector',
243 reference: 'diskselector',
406ac585 244 nodename: me.nodename,
1490ff3d
DC
245 autoSelect: false,
246 hideSize: true,
247 hideSelection: true,
248 storageLabel: gettext('Target Storage'),
406ac585 249 allowBlank: true,
0718aefc 250 storageContent: me.guestType === 'qemu' ? 'images' : 'rootdir',
f4f9e662 251 emptyText: gettext('Same as source'),
1490ff3d 252 disabled: me.isTemplate ? true : false // because default mode is clone for templates
f5226cd9
DM
253 });
254
406ac585 255 var formPanel = Ext.create('Ext.form.Panel', {
f5226cd9 256 bodyPadding: 10,
26693bb9 257 reference: 'cloneform',
f5226cd9 258 border: false,
d27a44a6 259 layout: 'hbox',
f5226cd9 260 defaultType: 'container',
f5226cd9
DM
261 fieldDefaults: {
262 labelWidth: 100,
263 anchor: '100%'
264 },
265 items: [
266 {
d27a44a6 267 flex: 1,
f5226cd9
DM
268 padding: '0 10 0 0',
269 layout: 'anchor',
270 items: col1
271 },
272 {
d27a44a6 273 flex: 1,
f5226cd9
DM
274 padding: '0 0 0 10',
275 layout: 'anchor',
276 items: col2
277 }
278 ]
279 });
280
f5226cd9
DM
281 Ext.apply(me, {
282 modal: true,
283 width: 600,
284 height: 250,
285 border: false,
286 layout: 'fit',
406ac585 287 buttons: [ {
672a6270 288 xtype: 'proxmoxHelpButton',
406ac585
EK
289 listenToGlobalEvent: false,
290 hidden: false,
291 onlineHelp: me.onlineHelp
292 },
293 '->',
294 {
295 reference: 'submitBtn',
296 text: gettext('Clone'),
297 disabled: true,
298 handler: function() {
299 var cloneForm = me.lookupReference('cloneform');
300 if (cloneForm.isValid()) {
301 me.create_clone(cloneForm.getValues());
302 }
303 }
304 } ],
305 items: [ formPanel ]
f5226cd9
DM
306 });
307
308 me.callParent();
309
310 me.verifyFeature();
311 }
312});