1 Ext
.define('PVE.dc.OptionView', {
2 extend
: 'Proxmox.grid.ObjectGrid',
3 alias
: ['widget.pveDcOptionView'],
5 onlineHelp
: 'datacenter_configuration_file',
8 userCls
: 'proxmox-tags-full',
10 add_inputpanel_row: function(name
, text
, opts
) {
14 me
.rows
= me
.rows
|| {};
16 let canEdit
= !Object
.prototype.hasOwnProperty
.call(opts
, 'caps') || opts
.caps
;
19 defaultValue
: opts
.defaultValue
,
21 renderer
: opts
.renderer
,
23 xtype
: 'proxmoxWindowEdit',
24 width
: opts
.width
|| 350,
26 onlineHelp
: opts
.onlineHelp
,
28 labelWidth
: opts
.labelWidth
|| 100,
30 setValues: function(values
) {
31 var edit_value
= values
[name
];
33 if (opts
.parseBeforeSet
) {
34 edit_value
= PVE
.Parser
.parsePropertyString(edit_value
);
37 Ext
.Array
.each(this.query('inputpanel'), function(panel
) {
38 panel
.setValues(edit_value
);
44 onGetValues: function(values
) {
45 if (values
=== undefined || Object
.keys(values
).length
=== 0) {
46 return { 'delete': name
};
49 ret_val
[name
] = PVE
.Parser
.printPropertyString(values
);
58 render_bwlimits: function(value
) {
60 return gettext("None");
63 let parsed
= PVE
.Parser
.parsePropertyString(value
);
64 return Object
.entries(parsed
)
65 .map(([k
, v
]) => k
+ ": " + Proxmox
.Utils
.format_size(v
* 1024) + "/s")
69 initComponent: function() {
72 me
.add_combobox_row('keyboard', gettext('Keyboard Layout'), {
73 renderer
: PVE
.Utils
.render_kvm_language
,
74 comboItems
: Object
.entries(PVE
.Utils
.kvm_keymaps
),
75 defaultValue
: '__default__',
78 me
.add_text_row('http_proxy', gettext('HTTP proxy'), {
79 defaultValue
: Proxmox
.Utils
.noneText
,
83 me
.add_combobox_row('console', gettext('Console Viewer'), {
84 renderer
: PVE
.Utils
.render_console_viewer
,
85 comboItems
: Object
.entries(PVE
.Utils
.console_map
),
86 defaultValue
: '__default__',
89 me
.add_text_row('email_from', gettext('Email from address'), {
92 defaultValue
: 'root@$hostname',
94 me
.add_text_row('mac_prefix', gettext('MAC address prefix'), {
97 defaultValue
: Proxmox
.Utils
.noneText
,
99 me
.add_inputpanel_row('migration', gettext('Migration Settings'), {
100 renderer
: PVE
.Utils
.render_as_property_string
,
102 url
: "/api2/extjs/cluster/options",
105 xtype
: 'displayfield',
107 fieldLabel
: gettext('Type'),
111 xtype
: 'proxmoxNetworkSelector',
113 fieldLabel
: gettext('Network'),
115 emptyText
: Proxmox
.Utils
.defaultText
,
120 me
.add_inputpanel_row('ha', gettext('HA Settings'), {
121 renderer
: PVE
.Utils
.render_dc_ha_opts
,
123 url
: "/api2/extjs/cluster/options",
124 onlineHelp
: 'ha_manager_shutdown_policy',
126 xtype
: 'proxmoxKVComboBox',
127 name
: 'shutdown_policy',
128 fieldLabel
: gettext('Shutdown Policy'),
130 value
: '__default__',
132 ['__default__', Proxmox
.Utils
.defaultText
+ ' (conditional)'],
133 ['freeze', 'freeze'],
134 ['failover', 'failover'],
135 ['migrate', 'migrate'],
136 ['conditional', 'conditional'],
138 defaultValue
: '__default__',
141 me
.add_inputpanel_row('crs', gettext('Cluster Resource Scheduling'), {
142 renderer
: PVE
.Utils
.render_as_property_string
,
145 url
: "/api2/extjs/cluster/options",
146 onlineHelp
: 'ha_manager_crs',
148 xtype
: 'proxmoxKVComboBox',
150 fieldLabel
: gettext('HA Scheduling'),
152 value
: '__default__',
154 ['__default__', Proxmox
.Utils
.defaultText
+ ' (basic)'],
155 ['basic', 'Basic (Resource Count)'],
156 ['static', 'Static Load'],
158 defaultValue
: '__default__',
160 xtype
: 'proxmoxcheckbox',
161 name
: 'ha-rebalance-on-start',
162 fieldLabel
: gettext('Rebalance on Start'),
163 boxLabel
: gettext('Use CRS to select the least loaded node when starting an HA service'),
167 me
.add_inputpanel_row('u2f', gettext('U2F Settings'), {
168 renderer
: v
=> !v
? Proxmox
.Utils
.NoneText
: PVE
.Parser
.printPropertyString(v
),
170 url
: "/api2/extjs/cluster/options",
171 onlineHelp
: 'pveum_configure_u2f',
175 fieldLabel
: gettext('U2F AppID URL'),
176 emptyText
: gettext('Defaults to origin'),
180 submitEmptyText
: false,
184 fieldLabel
: gettext('U2F Origin'),
185 emptyText
: gettext('Defaults to requesting host URI'),
189 submitEmptyText
: false,
194 html
: `<span class='pmx-hint'>${gettext('Note:')}</span> `
195 + Ext
.String
.format(gettext('{0} is deprecated, use {1}'), 'U2F', 'WebAuthn'),
198 xtype
: 'displayfield',
200 value
: gettext('NOTE: Changing an AppID breaks existing U2F registrations!'),
203 me
.add_inputpanel_row('webauthn', gettext('WebAuthn Settings'), {
204 renderer
: v
=> !v
? Proxmox
.Utils
.NoneText
: PVE
.Parser
.printPropertyString(v
),
206 url
: "/api2/extjs/cluster/options",
207 onlineHelp
: 'pveum_configure_webauthn',
210 fieldLabel
: gettext('Name'),
211 name
: 'rp', // NOTE: relying party consists of name and id, this is the name
216 fieldLabel
: gettext('Origin'),
217 emptyText
: Ext
.String
.format(gettext("Domain Lockdown (e.g., {0})"), document
.location
.origin
),
227 dirtychange
: (f
, isDirty
) =>
228 f
.up('panel').down('box[id=idChangeWarning]').setHidden(!f
.originalValue
|| !isDirty
),
241 text
: gettext('Auto-fill'),
242 iconCls
: 'fa fa-fw fa-pencil-square-o',
243 handler: function(button
, ev
) {
244 let panel
= this.up('panel');
245 let fqdn
= document
.location
.hostname
;
247 panel
.down('field[name=rp]').setValue(fqdn
);
249 let idField
= panel
.down('field[name=id]');
250 let currentID
= idField
.getValue();
251 if (!currentID
|| currentID
.length
=== 0) {
252 idField
.setValue(fqdn
);
261 html
: `<span class='pmx-hint'>${gettext('Note:')}</span> `
262 + gettext('WebAuthn requires using a trusted certificate.'),
266 id
: 'idChangeWarning',
269 html
: '<i class="fa fa-exclamation-triangle warning"></i> '
270 + gettext('Changing the ID breaks existing WebAuthn TFA entries.'),
273 me
.add_inputpanel_row('bwlimit', gettext('Bandwidth Limits'), {
274 renderer
: me
.render_bwlimits
,
276 url
: "/api2/extjs/cluster/options",
277 parseBeforeSet
: true,
280 xtype
: 'pveBandwidthField',
282 fieldLabel
: gettext('Default'),
283 emptyText
: gettext('none'),
287 xtype
: 'pveBandwidthField',
289 fieldLabel
: gettext('Backup Restore'),
290 emptyText
: gettext('default'),
294 xtype
: 'pveBandwidthField',
296 fieldLabel
: gettext('Migration'),
297 emptyText
: gettext('default'),
301 xtype
: 'pveBandwidthField',
303 fieldLabel
: gettext('Clone'),
304 emptyText
: gettext('default'),
308 xtype
: 'pveBandwidthField',
310 fieldLabel
: gettext('Disk Move'),
311 emptyText
: gettext('default'),
315 me
.add_integer_row('max_workers', gettext('Maximal Workers/bulk-action'), {
319 maxValue
: 64, // arbitrary but generous limit as limits are good
321 me
.add_inputpanel_row('next-id', gettext('Next Free VMID Range'), {
322 renderer
: PVE
.Utils
.render_as_property_string
,
323 url
: "/api2/extjs/cluster/options",
325 xtype
: 'proxmoxintegerfield',
327 fieldLabel
: gettext('Lower'),
330 maxValue
: 1000 * 1000 * 1000 - 1,
333 xtype
: 'proxmoxintegerfield',
335 fieldLabel
: gettext('Upper'),
336 emptyText
: '1.000.000',
338 maxValue
: 1000 * 1000 * 1000 - 1,
342 me
.rows
['tag-style'] = {
344 renderer
: (value
) => {
345 if (value
=== undefined) {
346 return gettext('No Overrides');
348 let colors
= PVE
.UIOptions
.parseTagOverrides(value
?.['color-map']);
349 let shape
= value
.shape
;
350 let shapeText
= PVE
.UIOptions
.tagTreeStyles
[shape
?? '__default__'];
351 let txt
= Ext
.String
.format(gettext("Tree Shape: {0}"), shapeText
);
352 let orderText
= PVE
.UIOptions
.tagOrderOptions
[value
.ordering
?? '__default__'];
353 txt
+= `, ${Ext.String.format(gettext("Ordering: {0}"), orderText)}`;
354 if (value
['case-sensitive']) {
355 txt
+= `, ${gettext('Case-Sensitive')}`;
357 if (Object
.keys(colors
).length
> 0) {
358 txt
+= `, ${gettext('Color Overrides')}: `;
359 for (const tag
of Object
.keys(colors
)) {
360 txt
+= Proxmox
.Utils
.getTagElement(tag
, colors
);
365 header
: gettext('Tag Style Override'),
367 xtype
: 'proxmoxWindowEdit',
369 subject
: gettext('Tag Color Override'),
370 onlineHelp
: 'datacenter_configuration_file',
374 url
: '/api2/extjs/cluster/options',
378 setValues: function(values
) {
379 if (values
=== undefined) {
382 values
= values
?.['tag-style'] ?? {};
383 values
.shape
= values
.shape
|| '__default__';
384 values
.colors
= values
['color-map'];
385 return Proxmox
.panel
.InputPanel
.prototype.setValues
.call(this, values
);
387 onGetValues: function(values
) {
390 style
['color-map'] = values
.colors
;
392 if (values
.shape
&& values
.shape
!== '__default__') {
393 style
.shape
= values
.shape
;
395 if (values
.ordering
) {
396 style
.ordering
= values
.ordering
;
398 if (values
['case-sensitive']) {
399 style
['case-sensitive'] = 1;
401 let value
= PVE
.Parser
.printPropertyString(style
);
404 'delete': 'tag-style',
415 xtype
: 'proxmoxComboGrid',
416 fieldLabel
: gettext('Tree Shape'),
418 displayField
: 'display',
423 header
: gettext('Option'),
424 dataIndex
: 'display',
428 header
: gettext('Preview'),
430 renderer: function(value
) {
431 let cls
= value
?? '__default__';
432 if (value
=== '__default__') {
435 let tags
= PVE
.Utils
.renderTags('preview');
436 return `<div class="proxmox-tags-${cls}">${tags}</div>`;
443 data
: Object
.entries(PVE
.UIOptions
.tagTreeStyles
).map(v
=> ({
449 defaultValue
: '__default__',
454 xtype
: 'proxmoxKVComboBox',
455 fieldLabel
: gettext('Ordering'),
456 comboItems
: Object
.entries(PVE
.UIOptions
.tagOrderOptions
),
457 defaultValue
: '__default__',
458 value
: '__default__',
462 name
: 'case-sensitive',
463 xtype
: 'proxmoxcheckbox',
464 fieldLabel
: gettext('Case-Sensitive'),
465 boxLabel
: gettext('Applies to new edits'),
469 xtype
: 'displayfield',
470 fieldLabel
: gettext('Color Overrides'),
474 xtype
: 'pveTagColorGrid',
484 me
.rows
['user-tag-access'] = {
486 renderer
: (value
) => {
487 if (value
=== undefined) {
488 return Ext
.String
.format(gettext('Mode: {0}'), 'free');
490 let mode
= value
?.['user-allow'] ?? 'free';
491 let list
= value
?.['user-allow-list']?.join(',') ?? '';
492 let modeTxt
= Ext
.String
.format(gettext('Mode: {0}'), mode
);
493 let overrides
= PVE
.UIOptions
.tagOverrides
;
494 let tags
= PVE
.Utils
.renderTags(list
, overrides
);
495 let listTxt
= tags
!== '' ? `, ${gettext('Pre-defined:')} ${tags}` : '';
496 return `${modeTxt}${listTxt}`;
498 header
: gettext('User Tag Access'),
500 xtype
: 'pveUserTagAccessEdit',
504 me
.rows
['registered-tags'] = {
506 renderer
: (value
) => {
507 if (value
=== undefined) {
508 return gettext('No Registered Tags');
510 let overrides
= PVE
.UIOptions
.tagOverrides
;
511 return PVE
.Utils
.renderTags(value
.join(','), overrides
);
513 header
: gettext('Registered Tags'),
515 xtype
: 'pveRegisteredTagEdit',
519 me
.selModel
= Ext
.create('Ext.selection.RowModel', {});
523 text
: gettext('Edit'),
524 xtype
: 'proxmoxButton',
526 handler: function() { me
.run_editor(); },
527 selModel
: me
.selModel
,
529 url
: "/api2/json/cluster/options",
531 url
: "/api2/extjs/cluster/options",
536 itemdblclick
: me
.run_editor
,
542 // set the new value for the default console
543 me
.mon(me
.rstore
, 'load', function(store
, records
, success
) {
548 var rec
= store
.getById('console');
549 PVE
.UIOptions
.options
.console
= rec
.data
.value
;
550 if (rec
.data
.value
=== '__default__') {
551 delete PVE
.UIOptions
.options
.console
;
554 PVE
.UIOptions
.options
['tag-style'] = store
.getById('tag-style')?.data
?.value
;
555 PVE
.UIOptions
.updateTagSettings(PVE
.UIOptions
.options
['tag-style']);
556 PVE
.UIOptions
.fireUIConfigChanged();
559 me
.on('activate', me
.rstore
.startUpdate
);
560 me
.on('destroy', me
.rstore
.stopUpdate
);
561 me
.on('deactivate', me
.rstore
.stopUpdate
);