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
,
144 url
: "/api2/extjs/cluster/options",
145 onlineHelp
: 'ha_manager_crs',
147 xtype
: 'proxmoxKVComboBox',
149 fieldLabel
: gettext('HA Scheduling'),
151 value
: '__default__',
153 ['__default__', Proxmox
.Utils
.defaultText
+ ' (basic)'],
154 ['basic', 'Basic (Resource Count)'],
155 ['static', 'Static Load'],
157 defaultValue
: '__default__',
160 me
.add_inputpanel_row('u2f', gettext('U2F Settings'), {
161 renderer
: v
=> !v
? Proxmox
.Utils
.NoneText
: PVE
.Parser
.printPropertyString(v
),
163 url
: "/api2/extjs/cluster/options",
164 onlineHelp
: 'pveum_configure_u2f',
168 fieldLabel
: gettext('U2F AppID URL'),
169 emptyText
: gettext('Defaults to origin'),
173 submitEmptyText
: false,
177 fieldLabel
: gettext('U2F Origin'),
178 emptyText
: gettext('Defaults to requesting host URI'),
182 submitEmptyText
: false,
187 html
: `<span class='pmx-hint'>${gettext('Note:')}</span> `
188 + Ext
.String
.format(gettext('{0} is deprecated, use {1}'), 'U2F', 'WebAuthn'),
191 xtype
: 'displayfield',
193 value
: gettext('NOTE: Changing an AppID breaks existing U2F registrations!'),
196 me
.add_inputpanel_row('webauthn', gettext('WebAuthn Settings'), {
197 renderer
: v
=> !v
? Proxmox
.Utils
.NoneText
: PVE
.Parser
.printPropertyString(v
),
199 url
: "/api2/extjs/cluster/options",
200 onlineHelp
: 'pveum_configure_webauthn',
203 fieldLabel
: gettext('Name'),
204 name
: 'rp', // NOTE: relying party consists of name and id, this is the name
209 fieldLabel
: gettext('Origin'),
210 emptyText
: Ext
.String
.format(gettext("Domain Lockdown (e.g., {0})"), document
.location
.origin
),
220 dirtychange
: (f
, isDirty
) =>
221 f
.up('panel').down('box[id=idChangeWarning]').setHidden(!f
.originalValue
|| !isDirty
),
234 text
: gettext('Auto-fill'),
235 iconCls
: 'fa fa-fw fa-pencil-square-o',
236 handler: function(button
, ev
) {
237 let panel
= this.up('panel');
238 let fqdn
= document
.location
.hostname
;
240 panel
.down('field[name=rp]').setValue(fqdn
);
242 let idField
= panel
.down('field[name=id]');
243 let currentID
= idField
.getValue();
244 if (!currentID
|| currentID
.length
=== 0) {
245 idField
.setValue(fqdn
);
254 html
: `<span class='pmx-hint'>${gettext('Note:')}</span> `
255 + gettext('WebAuthn requires using a trusted certificate.'),
259 id
: 'idChangeWarning',
262 html
: '<i class="fa fa-exclamation-triangle warning"></i> '
263 + gettext('Changing the ID breaks existing WebAuthn TFA entries.'),
266 me
.add_inputpanel_row('bwlimit', gettext('Bandwidth Limits'), {
267 renderer
: me
.render_bwlimits
,
269 url
: "/api2/extjs/cluster/options",
270 parseBeforeSet
: true,
273 xtype
: 'pveBandwidthField',
275 fieldLabel
: gettext('Default'),
276 emptyText
: gettext('none'),
280 xtype
: 'pveBandwidthField',
282 fieldLabel
: gettext('Backup Restore'),
283 emptyText
: gettext('default'),
287 xtype
: 'pveBandwidthField',
289 fieldLabel
: gettext('Migration'),
290 emptyText
: gettext('default'),
294 xtype
: 'pveBandwidthField',
296 fieldLabel
: gettext('Clone'),
297 emptyText
: gettext('default'),
301 xtype
: 'pveBandwidthField',
303 fieldLabel
: gettext('Disk Move'),
304 emptyText
: gettext('default'),
308 me
.add_integer_row('max_workers', gettext('Maximal Workers/bulk-action'), {
312 maxValue
: 64, // arbitrary but generous limit as limits are good
314 me
.add_inputpanel_row('next-id', gettext('Next Free VMID Range'), {
315 renderer
: PVE
.Utils
.render_as_property_string
,
316 url
: "/api2/extjs/cluster/options",
318 xtype
: 'proxmoxintegerfield',
320 fieldLabel
: gettext('Lower'),
323 maxValue
: 1000 * 1000 * 1000 - 1,
326 xtype
: 'proxmoxintegerfield',
328 fieldLabel
: gettext('Upper'),
329 emptyText
: '1.000.000',
331 maxValue
: 1000 * 1000 * 1000 - 1,
335 me
.rows
['tag-style'] = {
337 renderer
: (value
) => {
338 if (value
=== undefined) {
339 return gettext('No Overrides');
341 let colors
= PVE
.Utils
.parseTagOverrides(value
?.['color-map']);
342 let shape
= value
.shape
;
343 let shapeText
= PVE
.Utils
.tagTreeStyles
[shape
?? '__default__'];
344 let txt
= Ext
.String
.format(gettext("Tree Shape: {0}"), shapeText
);
345 let orderText
= PVE
.Utils
.tagOrderOptions
[value
.ordering
?? '__default__'];
346 txt
+= `, ${Ext.String.format(gettext("Ordering: {0}"), orderText)}`;
347 if (Object
.keys(colors
).length
> 0) {
350 for (const tag
of Object
.keys(colors
)) {
351 txt
+= Proxmox
.Utils
.getTagElement(tag
, colors
);
355 header
: gettext('Tag Style Override'),
357 xtype
: 'proxmoxWindowEdit',
359 subject
: gettext('Tag Color Override'),
360 onlineHelp
: 'datacenter_configuration_file',
364 url
: '/api2/extjs/cluster/options',
368 setValues: function(values
) {
369 if (values
=== undefined) {
372 values
= values
?.['tag-style'] ?? {};
373 values
.shape
= values
.shape
|| '__default__';
374 values
.colors
= values
['color-map'];
375 return Proxmox
.panel
.InputPanel
.prototype.setValues
.call(this, values
);
377 onGetValues: function(values
) {
380 style
['color-map'] = values
.colors
;
382 if (values
.shape
&& values
.shape
!== '__default__') {
383 style
.shape
= values
.shape
;
385 if (values
.ordering
) {
386 style
.ordering
= values
.ordering
;
388 let value
= PVE
.Parser
.printPropertyString(style
);
391 'delete': 'tag-style',
402 xtype
: 'proxmoxComboGrid',
403 fieldLabel
: gettext('Tree Shape'),
405 displayField
: 'display',
410 header
: gettext('Option'),
411 dataIndex
: 'display',
415 header
: gettext('Preview'),
417 renderer: function(value
) {
418 let cls
= value
?? '__default__';
419 if (value
=== '__default__') {
422 let tags
= PVE
.Utils
.renderTags('preview');
423 return `<div class="proxmox-tags-${cls}">${tags}</div>`;
430 data
: Object
.entries(PVE
.Utils
.tagTreeStyles
).map(v
=> ({
436 defaultValue
: '__default__',
441 xtype
: 'proxmoxKVComboBox',
442 fieldLabel
: gettext('Ordering'),
443 comboItems
: Object
.entries(PVE
.Utils
.tagOrderOptions
),
444 defaultValue
: '__default__',
445 value
: '__default__',
449 xtype
: 'displayfield',
450 fieldLabel
: gettext('Color Overrides'),
454 xtype
: 'pveTagColorGrid',
464 me
.rows
['user-tag-access'] = {
466 renderer
: (value
) => {
467 if (value
=== undefined) {
468 return Ext
.String
.format(gettext('Mode: {0}'), 'free');
470 let mode
= value
?.['user-allow'] ?? 'free';
471 let list
= value
?.['user-allow-list']?.join(',') ?? '';
472 let modeTxt
= Ext
.String
.format(gettext('Mode: {0}'), mode
);
473 let overrides
= PVE
.Utils
.tagOverrides
;
474 let tags
= PVE
.Utils
.renderTags(list
, overrides
);
475 let listTxt
= tags
!== '' ? `, ${gettext('Pre-defiend:')} ${tags}` : '';
476 return `${modeTxt}${listTxt}`;
478 header
: gettext('User Tag Access'),
480 xtype
: 'pveUserTagAccessEdit',
484 me
.rows
['registered-tags'] = {
486 renderer
: (value
) => {
487 if (value
=== undefined) {
488 return gettext('No Registered Tags');
490 let overrides
= PVE
.Utils
.tagOverrides
;
491 return PVE
.Utils
.renderTags(value
.join(','), overrides
);
493 header
: gettext('Registered Tags'),
495 xtype
: 'pveRegisteredTagEdit',
499 me
.selModel
= Ext
.create('Ext.selection.RowModel', {});
503 text
: gettext('Edit'),
504 xtype
: 'proxmoxButton',
506 handler: function() { me
.run_editor(); },
507 selModel
: me
.selModel
,
509 url
: "/api2/json/cluster/options",
511 url
: "/api2/extjs/cluster/options",
516 itemdblclick
: me
.run_editor
,
522 // set the new value for the default console
523 me
.mon(me
.rstore
, 'load', function(store
, records
, success
) {
528 var rec
= store
.getById('console');
529 PVE
.UIOptions
.console
= rec
.data
.value
;
530 if (rec
.data
.value
=== '__default__') {
531 delete PVE
.UIOptions
.console
;
534 PVE
.UIOptions
['tag-style'] = store
.getById('tag-style')?.data
?.value
;
535 PVE
.Utils
.updateTagSettings(PVE
.UIOptions
['tag-style']);
538 me
.on('activate', me
.rstore
.startUpdate
);
539 me
.on('destroy', me
.rstore
.stopUpdate
);
540 me
.on('deactivate', me
.rstore
.stopUpdate
);