]> git.proxmox.com Git - proxmox-widget-toolkit.git/blame - src/window/Edit.js
edit window: add optional custom submit options
[proxmox-widget-toolkit.git] / src / window / Edit.js
CommitLineData
06694509
DM
1Ext.define('Proxmox.window.Edit', {
2 extend: 'Ext.window.Window',
3 alias: 'widget.proxmoxWindowEdit',
1b07a95a 4
a33ba257
DM
5 // autoLoad trigger a load() after component creation
6 autoLoad: false,
ca204f31
TL
7 // set extra options like params for the load request
8 autoLoadOptions: undefined,
a33ba257 9
9244bb72
TL
10 // to submit extra params on load and submit, useful, e.g., if not all ID
11 // parameters are included in the URL
12 extraRequestParams: {},
13
06694509
DM
14 resizable: false,
15
aca2ab3c 16 // use this to automatically generate a title like `Create: <subject>`
06694509
DM
17 subject: undefined,
18
1289d326 19 // set isCreate to true if you want a Create button (instead OK and RESET)
b33f451f 20 isCreate: false,
06694509
DM
21
22 // set to true if you want an Add button (instead of Create)
23 isAdd: false,
24
aca2ab3c 25 // set to true if you want a Remove button (instead of Create)
06694509
DM
26 isRemove: false,
27
95bba12d
AL
28 // set to false, if you don't want the reset button present
29 showReset: true,
30
ffea05ec
DC
31 // custom submitText
32 submitText: undefined,
33
03e44f5b
DC
34 // custom options for the submit api call
35 submitOptions: {},
36
06694509
DM
37 backgroundDelay: 0,
38
4d07ee6c
TL
39 // string or function, called as (url, values) - useful if the ID of the
40 // new object is part of the URL, or that URL differs from GET/PUT URL
41 submitUrl: Ext.identityFn,
42
8b121714
TL
43 // string or function, called as (url, initialConfig) - mostly for
44 // consistency with submitUrl existing. If both are set `url` gets optional
45 loadUrl: Ext.identityFn,
46
a7dcbdeb
DM
47 // needed for finding the reference to submitbutton
48 // because we do not have a controller
49 referenceHolder: true,
50 defaultButton: 'submitbutton',
51
52 // finds the first form field
4f3b2a93 53 defaultFocus: 'field:focusable[disabled=false][hidden=false]',
a7dcbdeb 54
06694509
DM
55 showProgress: false,
56
8d8dbfc5
TL
57 showTaskViewer: false,
58
fde8e8bb
TL
59 // gets called if we have a progress bar or taskview and it detected that
60 // the task finished. function(success)
61 taskDone: Ext.emptyFn,
62
a498f279
DC
63 // gets called when the api call is finished, right at the beginning
64 // function(success, response, options)
65 apiCallDone: Ext.emptyFn,
66
42a9df8b
DC
67 // assign a reference from docs, to add a help button docked to the
68 // bottom of the window. If undefined we magically fall back to the
69 // onlineHelp of our first item, if set.
70 onlineHelp: undefined,
71
06694509 72 isValid: function() {
4b5c6b3f 73 let me = this;
06694509 74
4b5c6b3f 75 let form = me.formPanel.getForm();
06694509
DM
76 return form.isValid();
77 },
78
79 getValues: function(dirtyOnly) {
4b5c6b3f 80 let me = this;
06694509 81
1289d326 82 let values = {};
9244bb72
TL
83 Ext.apply(values, me.extraRequestParams);
84
4b5c6b3f 85 let form = me.formPanel.getForm();
06694509 86
1289d326
TL
87 form.getFields().each(function(field) {
88 if (!field.up('inputpanel') && (!dirtyOnly || field.isDirty())) {
89 Proxmox.Utils.assemble_field_data(values, field.getSubmitData());
90 }
91 });
06694509
DM
92
93 Ext.Array.each(me.query('inputpanel'), function(panel) {
94 Proxmox.Utils.assemble_field_data(values, panel.getValues(dirtyOnly));
95 });
96
97 return values;
98 },
99
100 setValues: function(values) {
4b5c6b3f 101 let me = this;
06694509 102
4b5c6b3f 103 let form = me.formPanel.getForm();
b24601af
TL
104 let formfields = form.getFields();
105
106 Ext.iterate(values, function(id, val) {
107 let fields = formfields.filterBy((f) =>
01031528 108 (f.id === id || f.name === id || f.dataIndex === id) && !f.up('inputpanel'),
b24601af
TL
109 );
110 fields.each((field) => {
111 field.setValue(val);
68689d73
TL
112 if (form.trackResetOnLoad) {
113 field.resetOriginalValue();
114 }
b24601af 115 });
06694509 116 });
1b07a95a 117
06694509
DM
118 Ext.Array.each(me.query('inputpanel'), function(panel) {
119 panel.setValues(values);
120 });
121 },
122
4e78d719
TL
123 setSubmitText: function(text) {
124 this.lookup('submitbutton').setText(text);
125 },
126
06694509 127 submit: function() {
4b5c6b3f 128 let me = this;
06694509 129
4b5c6b3f 130 let form = me.formPanel.getForm();
06694509 131
4b5c6b3f 132 let values = me.getValues();
06694509 133 Ext.Object.each(values, function(name, val) {
4b5c6b3f
TL
134 if (Object.prototype.hasOwnProperty.call(values, name)) {
135 if (Ext.isArray(val) && !val.length) {
06694509
DM
136 values[name] = '';
137 }
138 }
139 });
140
141 if (me.digest) {
142 values.digest = me.digest;
143 }
144
145 if (me.backgroundDelay) {
146 values.background_delay = me.backgroundDelay;
147 }
148
4d07ee6c
TL
149 let url = Ext.isFunction(me.submitUrl)
150 ? me.submitUrl(me.url, values)
151 : me.submitUrl || me.url;
06694509
DM
152 if (me.method === 'DELETE') {
153 url = url + "?" + Ext.Object.toQueryString(values);
154 values = undefined;
155 }
156
03e44f5b 157 let requestOptions = Ext.apply({
06694509
DM
158 url: url,
159 waitMsgTarget: me,
160 method: me.method || (me.backgroundDelay ? 'POST' : 'PUT'),
161 params: values,
162 failure: function(response, options) {
a498f279
DC
163 me.apiCallDone(false, response, options);
164
06694509
DM
165 if (response.result && response.result.errors) {
166 form.markInvalid(response.result.errors);
167 }
168 Ext.Msg.alert(gettext('Error'), response.htmlStatus);
169 },
170 success: function(response, options) {
4b5c6b3f
TL
171 let hasProgressBar =
172 (me.backgroundDelay || me.showProgress || me.showTaskViewer) &&
173 response.result.data;
06694509 174
a498f279
DC
175 me.apiCallDone(true, response, options);
176
06694509 177 if (hasProgressBar) {
267c5365 178 // only hide to allow delaying our close event until task is done
06694509
DM
179 me.hide();
180
4b5c6b3f
TL
181 let upid = response.result.data;
182 let viewerClass = me.showTaskViewer ? 'Viewer' : 'Progress';
183 Ext.create('Proxmox.window.Task' + viewerClass, {
184 autoShow: true,
06694509 185 upid: upid,
fde8e8bb 186 taskDone: me.taskDone,
06694509 187 listeners: {
4b5c6b3f 188 destroy: function() {
06694509 189 me.close();
4b5c6b3f
TL
190 },
191 },
06694509 192 });
06694509
DM
193 } else {
194 me.close();
195 }
4b5c6b3f 196 },
03e44f5b
DC
197 }, me.submitOptions ?? {});
198 Proxmox.Utils.API2Request(requestOptions);
06694509
DM
199 },
200
201 load: function(options) {
4b5c6b3f 202 let me = this;
06694509 203
4b5c6b3f 204 let form = me.formPanel.getForm();
06694509
DM
205
206 options = options || {};
207
4b5c6b3f
TL
208 let newopts = Ext.apply({
209 waitMsgTarget: me,
06694509
DM
210 }, options);
211
9244bb72
TL
212 if (Object.keys(me.extraRequestParams).length > 0) {
213 let params = newopts.params || {};
214 Ext.applyIf(params, me.extraRequestParams);
215 newopts.params = params;
216 }
217
8b121714
TL
218 let url = Ext.isFunction(me.loadUrl)
219 ? me.loadUrl(me.url, me.initialConfig)
220 : me.loadUrl || me.url;
221
4b5c6b3f 222 let createWrapper = function(successFn) {
06694509 223 Ext.apply(newopts, {
8b121714 224 url: url,
06694509
DM
225 method: 'GET',
226 success: function(response, opts) {
227 form.clearInvalid();
4f5e2bd1 228 me.digest = response.result?.digest || response.result?.data?.digest;
06694509
DM
229 if (successFn) {
230 successFn(response, opts);
231 } else {
232 me.setValues(response.result.data);
233 }
234 // hack: fix ExtJS bug
3637defc 235 Ext.Array.each(me.query('radiofield'), f => f.resetOriginalValue());
06694509
DM
236 },
237 failure: function(response, opts) {
238 Ext.Msg.alert(gettext('Error'), response.htmlStatus, function() {
239 me.close();
240 });
4b5c6b3f 241 },
06694509
DM
242 });
243 };
244
245 createWrapper(options.success);
246
247 Proxmox.Utils.API2Request(newopts);
248 },
249
4b5c6b3f
TL
250 initComponent: function() {
251 let me = this;
06694509 252
8969bdad
TL
253 if (!me.url && (
254 !me.submitUrl || !me.loadUrl || me.submitUrl === Ext.identityFn ||
255 me.loadUrl === Ext.identityFn
256 )
257 ) {
258 throw "neither 'url' nor both, submitUrl and loadUrl specified";
06694509 259 }
3637defc
TL
260 if (me.create) {
261 throw "deprecated parameter, use isCreate";
262 }
b33f451f 263
4b5c6b3f 264 let items = Ext.isArray(me.items) ? me.items : [me.items];
06694509
DM
265
266 me.items = undefined;
267
268 me.formPanel = Ext.create('Ext.form.Panel', {
8969bdad 269 url: me.url, // FIXME: not in 'form' class, safe to remove??
06694509
DM
270 method: me.method || 'PUT',
271 trackResetOnLoad: true,
271171f8 272 bodyPadding: me.bodyPadding !== undefined ? me.bodyPadding : 10,
06694509 273 border: false,
31a50251 274 defaults: Ext.apply({}, me.defaults, {
4b5c6b3f 275 border: false,
31a50251 276 }),
06694509
DM
277 fieldDefaults: Ext.apply({}, me.fieldDefaults, {
278 labelWidth: 100,
4b5c6b3f 279 anchor: '100%',
06694509 280 }),
4b5c6b3f 281 items: items,
06694509
DM
282 });
283
4b5c6b3f 284 let inputPanel = me.formPanel.down('inputpanel');
1b07a95a 285
4b5c6b3f 286 let form = me.formPanel.getForm();
06694509 287
4b5c6b3f 288 let submitText;
b33f451f 289 if (me.isCreate) {
ffea05ec
DC
290 if (me.submitText) {
291 submitText = me.submitText;
292 } else if (me.isAdd) {
06694509
DM
293 submitText = gettext('Add');
294 } else if (me.isRemove) {
295 submitText = gettext('Remove');
296 } else {
297 submitText = gettext('Create');
298 }
299 } else {
ffea05ec 300 submitText = me.submitText || gettext('OK');
06694509
DM
301 }
302
4b5c6b3f 303 let submitBtn = Ext.create('Ext.Button', {
a7dcbdeb 304 reference: 'submitbutton',
06694509 305 text: submitText,
b33f451f 306 disabled: !me.isCreate,
06694509
DM
307 handler: function() {
308 me.submit();
4b5c6b3f 309 },
06694509
DM
310 });
311
4b5c6b3f 312 let resetBtn = Ext.create('Ext.Button', {
06694509
DM
313 text: 'Reset',
314 disabled: true,
4b5c6b3f 315 handler: function() {
06694509 316 form.reset();
4b5c6b3f 317 },
06694509
DM
318 });
319
4b5c6b3f
TL
320 let set_button_status = function() {
321 let valid = form.isValid();
322 let dirty = form.isDirty();
b33f451f 323 submitBtn.setDisabled(!valid || !(dirty || me.isCreate));
06694509
DM
324 resetBtn.setDisabled(!dirty);
325 };
326
327 form.on('dirtychange', set_button_status);
328 form.on('validitychange', set_button_status);
329
4b5c6b3f 330 let colwidth = 300;
06694509
DM
331 if (me.fieldDefaults && me.fieldDefaults.labelWidth) {
332 colwidth += me.fieldDefaults.labelWidth - 100;
333 }
06694509 334
4b5c6b3f 335 let twoColumn = inputPanel && (inputPanel.column1 || inputPanel.column2);
06694509
DM
336
337 if (me.subject && !me.title) {
b33f451f 338 me.title = Proxmox.Utils.dialog_title(me.subject, me.isCreate, me.isAdd);
06694509
DM
339 }
340
95bba12d 341 if (me.isCreate || !me.showReset) {
4b5c6b3f 342 me.buttons = [submitBtn];
06694509 343 } else {
4b5c6b3f 344 me.buttons = [submitBtn, resetBtn];
06694509
DM
345 }
346
880df5d5 347 if (inputPanel && inputPanel.hasAdvanced) {
4b5c6b3f
TL
348 let sp = Ext.state.Manager.getProvider();
349 let advchecked = sp.get('proxmox-advanced-cb');
880df5d5 350 inputPanel.setAdvancedVisible(advchecked);
17c580c2
TL
351 me.buttons.unshift({
352 xtype: 'proxmoxcheckbox',
353 itemId: 'advancedcb',
354 boxLabelAlign: 'before',
355 boxLabel: gettext('Advanced'),
356 stateId: 'proxmox-advanced-cb',
357 value: advchecked,
358 listeners: {
359 change: function(cb, val) {
360 inputPanel.setAdvancedVisible(val);
361 sp.set('proxmox-advanced-cb', val);
362 },
363 },
364 });
880df5d5
DC
365 }
366
4b5c6b3f 367 let onlineHelp = me.onlineHelp;
c3457485
DM
368 if (!onlineHelp && inputPanel && inputPanel.onlineHelp) {
369 onlineHelp = inputPanel.onlineHelp;
370 }
371
372 if (onlineHelp) {
4b5c6b3f 373 let helpButton = Ext.create('Proxmox.button.Help');
06694509 374 me.buttons.unshift(helpButton, '->');
c3457485 375 Ext.GlobalEvents.fireEvent('proxmoxShowHelp', onlineHelp);
06694509
DM
376 }
377
378 Ext.applyIf(me, {
379 modal: true,
380 width: twoColumn ? colwidth*2 : colwidth,
381 border: false,
4b5c6b3f 382 items: [me.formPanel],
06694509
DM
383 });
384
385 me.callParent();
386
b2471e89
DC
387
388 if (inputPanel?.hasAdvanced) {
389 let advancedItems = inputPanel.down('#advancedContainer').query('field');
390 advancedItems.forEach(function(field) {
391 me.mon(field, 'validitychange', (f, valid) => {
392 if (!valid) {
393 f.up('inputpanel').setAdvancedVisible(true);
394 }
395 });
396 });
397 }
398
06694509
DM
399 // always mark invalid fields
400 me.on('afterlayout', function() {
401 // on touch devices, the isValid function
402 // triggers a layout, which triggers an isValid
403 // and so on
404 // to prevent this we disable the layouting here
405 // and enable it afterwards
406 me.suspendLayout = true;
407 me.isValid();
408 me.suspendLayout = false;
409 });
a33ba257
DM
410
411 if (me.autoLoad) {
ca204f31 412 me.load(me.autoLoadOptions);
a33ba257 413 }
4b5c6b3f 414 },
06694509 415});