]>
git.proxmox.com Git - proxmox-widget-toolkit.git/blob - src/window/Edit.js
1 Ext
.define('Proxmox.window.Edit', {
2 extend
: 'Ext.window.Window',
3 alias
: 'widget.proxmoxWindowEdit',
5 // autoLoad trigger a load() after component creation
7 // set extra options like params for the load request
8 autoLoadOptions
: undefined,
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
: {},
16 // use this to automatically generate a title like `Create: <subject>`
19 // set isCreate to true if you want a Create button (instead OK and RESET)
22 // set to true if you want an Add button (instead of Create)
25 // set to true if you want a Remove button (instead of Create)
28 // set to false, if you don't want the reset button present
32 submitText
: undefined,
34 // custom options for the submit api call
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
,
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
,
47 // needed for finding the reference to submitbutton
48 // because we do not have a controller
49 referenceHolder
: true,
50 defaultButton
: 'submitbutton',
52 // finds the first form field
53 defaultFocus
: 'field:focusable[disabled=false][hidden=false]',
57 showTaskViewer
: false,
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
,
63 // gets called when the api call is finished, right at the beginning
64 // function(success, response, options)
65 apiCallDone
: Ext
.emptyFn
,
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,
72 constructor: function(conf
) {
74 // make copies in order to prevent subclasses from accidentally writing
75 // to objects that are shared with other edit window subclasses
76 me
.extraRequestParams
= Object
.assign({}, me
.extraRequestParams
);
77 me
.submitOptions
= Object
.assign({}, me
.submitOptions
);
78 me
.callParent(arguments
);
84 let form
= me
.formPanel
.getForm();
85 return form
.isValid();
88 getValues: function(dirtyOnly
) {
92 Ext
.apply(values
, me
.extraRequestParams
);
94 let form
= me
.formPanel
.getForm();
96 form
.getFields().each(function(field
) {
97 if (!field
.up('inputpanel') && (!dirtyOnly
|| field
.isDirty())) {
98 Proxmox
.Utils
.assemble_field_data(values
, field
.getSubmitData());
102 Ext
.Array
.each(me
.query('inputpanel'), function(panel
) {
103 Proxmox
.Utils
.assemble_field_data(values
, panel
.getValues(dirtyOnly
));
109 setValues: function(values
) {
112 let form
= me
.formPanel
.getForm();
113 let formfields
= form
.getFields();
115 Ext
.iterate(values
, function(id
, val
) {
116 let fields
= formfields
.filterBy((f
) =>
117 (f
.id
=== id
|| f
.name
=== id
|| f
.dataIndex
=== id
) && !f
.up('inputpanel'),
119 fields
.each((field
) => {
121 if (form
.trackResetOnLoad
) {
122 field
.resetOriginalValue();
127 Ext
.Array
.each(me
.query('inputpanel'), function(panel
) {
128 panel
.setValues(values
);
132 setSubmitText: function(text
) {
133 this.lookup('submitbutton').setText(text
);
139 let form
= me
.formPanel
.getForm();
141 let values
= me
.getValues();
142 Ext
.Object
.each(values
, function(name
, val
) {
143 if (Object
.prototype.hasOwnProperty
.call(values
, name
)) {
144 if (Ext
.isArray(val
) && !val
.length
) {
151 values
.digest
= me
.digest
;
154 if (me
.backgroundDelay
) {
155 values
.background_delay
= me
.backgroundDelay
;
158 let url
= Ext
.isFunction(me
.submitUrl
)
159 ? me
.submitUrl(me
.url
, values
)
160 : me
.submitUrl
|| me
.url
;
161 if (me
.method
=== 'DELETE') {
162 url
= url
+ "?" + Ext
.Object
.toQueryString(values
);
166 let requestOptions
= Ext
.apply({
169 method
: me
.method
|| (me
.backgroundDelay
? 'POST' : 'PUT'),
171 failure: function(response
, options
) {
172 me
.apiCallDone(false, response
, options
);
174 if (response
.result
&& response
.result
.errors
) {
175 form
.markInvalid(response
.result
.errors
);
177 Ext
.Msg
.alert(gettext('Error'), response
.htmlStatus
);
179 success: function(response
, options
) {
181 (me
.backgroundDelay
|| me
.showProgress
|| me
.showTaskViewer
) &&
182 response
.result
.data
;
184 me
.apiCallDone(true, response
, options
);
186 if (hasProgressBar
) {
187 // only hide to allow delaying our close event until task is done
190 let upid
= response
.result
.data
;
191 let viewerClass
= me
.showTaskViewer
? 'Viewer' : 'Progress';
192 Ext
.create('Proxmox.window.Task' + viewerClass
, {
195 taskDone
: me
.taskDone
,
197 destroy: function() {
206 }, me
.submitOptions
?? {});
207 Proxmox
.Utils
.API2Request(requestOptions
);
210 load: function(options
) {
213 let form
= me
.formPanel
.getForm();
215 options
= options
|| {};
217 let newopts
= Ext
.apply({
221 if (Object
.keys(me
.extraRequestParams
).length
> 0) {
222 let params
= newopts
.params
|| {};
223 Ext
.applyIf(params
, me
.extraRequestParams
);
224 newopts
.params
= params
;
227 let url
= Ext
.isFunction(me
.loadUrl
)
228 ? me
.loadUrl(me
.url
, me
.initialConfig
)
229 : me
.loadUrl
|| me
.url
;
231 let createWrapper = function(successFn
) {
235 success: function(response
, opts
) {
237 me
.digest
= response
.result
?.digest
|| response
.result
?.data
?.digest
;
239 successFn(response
, opts
);
241 me
.setValues(response
.result
.data
);
243 // hack: fix ExtJS bug
244 Ext
.Array
.each(me
.query('radiofield'), f
=> f
.resetOriginalValue());
246 failure: function(response
, opts
) {
247 Ext
.Msg
.alert(gettext('Error'), response
.htmlStatus
, function() {
254 createWrapper(options
.success
);
256 Proxmox
.Utils
.API2Request(newopts
);
259 initComponent: function() {
263 !me
.submitUrl
|| !me
.loadUrl
|| me
.submitUrl
=== Ext
.identityFn
||
264 me
.loadUrl
=== Ext
.identityFn
267 throw "neither 'url' nor both, submitUrl and loadUrl specified";
270 throw "deprecated parameter, use isCreate";
273 let items
= Ext
.isArray(me
.items
) ? me
.items
: [me
.items
];
275 me
.items
= undefined;
277 me
.formPanel
= Ext
.create('Ext.form.Panel', {
278 url
: me
.url
, // FIXME: not in 'form' class, safe to remove??
279 method
: me
.method
|| 'PUT',
280 trackResetOnLoad
: true,
281 bodyPadding
: me
.bodyPadding
!== undefined ? me
.bodyPadding
: 10,
283 defaults
: Ext
.apply({}, me
.defaults
, {
286 fieldDefaults
: Ext
.apply({}, me
.fieldDefaults
, {
293 let inputPanel
= me
.formPanel
.down('inputpanel');
295 let form
= me
.formPanel
.getForm();
300 submitText
= me
.submitText
;
301 } else if (me
.isAdd
) {
302 submitText
= gettext('Add');
303 } else if (me
.isRemove
) {
304 submitText
= gettext('Remove');
306 submitText
= gettext('Create');
309 submitText
= me
.submitText
|| gettext('OK');
312 let submitBtn
= Ext
.create('Ext.Button', {
313 reference
: 'submitbutton',
315 disabled
: !me
.isCreate
,
316 handler: function() {
321 let resetTool
= Ext
.create('Ext.panel.Tool', {
322 glyph
: 'xf0e2@FontAwesome', // fa-undo
323 tooltip
: gettext('Reset form data'),
324 callback
: () => form
.reset(),
326 paddingRight
: '2px', // just slightly more room to breathe
331 let set_button_status = function() {
332 let valid
= form
.isValid();
333 let dirty
= form
.isDirty();
334 submitBtn
.setDisabled(!valid
|| !(dirty
|| me
.isCreate
));
335 resetTool
.setDisabled(!dirty
);
338 form
.on('dirtychange', set_button_status
);
339 form
.on('validitychange', set_button_status
);
342 if (me
.fieldDefaults
&& me
.fieldDefaults
.labelWidth
) {
343 colwidth
+= me
.fieldDefaults
.labelWidth
- 100;
346 let twoColumn
= inputPanel
&& (inputPanel
.column1
|| inputPanel
.column2
);
348 if (me
.subject
&& !me
.title
) {
349 me
.title
= Proxmox
.Utils
.dialog_title(me
.subject
, me
.isCreate
, me
.isAdd
);
352 me
.buttons
= [submitBtn
];
354 if (!me
.isCreate
&& me
.showReset
) {
355 me
.tools
= [resetTool
];
358 if (inputPanel
&& inputPanel
.hasAdvanced
) {
359 let sp
= Ext
.state
.Manager
.getProvider();
360 let advchecked
= sp
.get('proxmox-advanced-cb');
361 inputPanel
.setAdvancedVisible(advchecked
);
363 xtype
: 'proxmoxcheckbox',
364 itemId
: 'advancedcb',
365 boxLabelAlign
: 'before',
366 boxLabel
: gettext('Advanced'),
367 stateId
: 'proxmox-advanced-cb',
370 change: function(cb
, val
) {
371 inputPanel
.setAdvancedVisible(val
);
372 sp
.set('proxmox-advanced-cb', val
);
378 let onlineHelp
= me
.onlineHelp
;
379 if (!onlineHelp
&& inputPanel
&& inputPanel
.onlineHelp
) {
380 onlineHelp
= inputPanel
.onlineHelp
;
384 let helpButton
= Ext
.create('Proxmox.button.Help');
385 me
.buttons
.unshift(helpButton
, '->');
386 Ext
.GlobalEvents
.fireEvent('proxmoxShowHelp', onlineHelp
);
391 width
: twoColumn
? colwidth
*2 : colwidth
,
393 items
: [me
.formPanel
],
399 if (inputPanel
?.hasAdvanced
) {
400 let advancedItems
= inputPanel
.down('#advancedContainer').query('field');
401 advancedItems
.forEach(function(field
) {
402 me
.mon(field
, 'validitychange', (f
, valid
) => {
404 f
.up('inputpanel').setAdvancedVisible(true);
410 // always mark invalid fields
411 me
.on('afterlayout', function() {
412 // on touch devices, the isValid function
413 // triggers a layout, which triggers an isValid
415 // to prevent this we disable the layouting here
416 // and enable it afterwards
417 me
.suspendLayout
= true;
419 me
.suspendLayout
= false;
423 me
.load(me
.autoLoadOptions
);