mirror of
https://git.proxmox.com/git/proxmox-widget-toolkit
synced 2025-05-29 06:46:34 +00:00

use the view, viewmodel, controller style code, which makes the logic much more understandable this adds two features to it, namely a viewBuffer range, which controls the distance to the end of the data on which the panel loads the next data and an optional failCallback, which will be used for the cephlog this also fixes #2151 Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
261 lines
5.8 KiB
JavaScript
261 lines
5.8 KiB
JavaScript
/*
|
|
* Display log entries in a panel with scrollbar
|
|
* The log entries are automatically refreshed via a background task,
|
|
* with newest entries comming at the bottom
|
|
*/
|
|
Ext.define('Proxmox.panel.LogView', {
|
|
extend: 'Ext.panel.Panel',
|
|
xtype: 'proxmoxLogView',
|
|
|
|
pageSize: 500,
|
|
viewBuffer: 50,
|
|
lineHeight: 16,
|
|
|
|
scrollToEnd: true,
|
|
|
|
// callback for load failure, used for ceph
|
|
failCallback: undefined,
|
|
|
|
controller: {
|
|
xclass: 'Ext.app.ViewController',
|
|
|
|
updateParams: function() {
|
|
var me = this;
|
|
var viewModel = me.getViewModel();
|
|
var since = viewModel.get('since');
|
|
var until = viewModel.get('until');
|
|
if (viewModel.get('hide_timespan')) {
|
|
return;
|
|
}
|
|
|
|
if (since > until) {
|
|
Ext.Msg.alert('Error', 'Since date must be less equal than Until date.');
|
|
return;
|
|
}
|
|
|
|
viewModel.set('params.since', Ext.Date.format(since, 'Y-m-d'));
|
|
viewModel.set('params.until', Ext.Date.format(until, 'Y-m-d') + ' 23:59:59');
|
|
me.getView().loadTask.delay(200);
|
|
},
|
|
|
|
scrollPosBottom: function() {
|
|
var view = this.getView();
|
|
var pos = view.getScrollY();
|
|
var maxPos = view.getScrollable().getMaxPosition().y;
|
|
return maxPos - pos;
|
|
},
|
|
|
|
updateView: function(text, first, total) {
|
|
var me = this;
|
|
var view = me.getView();
|
|
var viewModel = me.getViewModel();
|
|
var content = me.lookup('content');
|
|
var data = viewModel.get('data');
|
|
|
|
if (first === data.first && total === data.total && text.length === data.textlen) {
|
|
return; // same content, skip setting and scrolling
|
|
}
|
|
viewModel.set('data', {
|
|
first: first,
|
|
total: total,
|
|
textlen: text.length
|
|
});
|
|
|
|
var scrollPos = me.scrollPosBottom();
|
|
|
|
content.update(text);
|
|
|
|
if (view.scrollToEnd && scrollPos <= 0) {
|
|
// we use setTimeout to work around scroll handling on touchscreens
|
|
setTimeout(function() { view.scrollTo(0, Infinity); }, 10);
|
|
}
|
|
},
|
|
|
|
doLoad: function() {
|
|
var me = this;
|
|
var view = me.getView();
|
|
var viewModel = me.getViewModel();
|
|
Proxmox.Utils.API2Request({
|
|
url: me.getView().url,
|
|
params: viewModel.get('params'),
|
|
method: 'GET',
|
|
success: function(response) {
|
|
Proxmox.Utils.setErrorMask(me, false);
|
|
var total = response.result.total;
|
|
var lines = new Array();
|
|
var first = Infinity;
|
|
|
|
Ext.Array.each(response.result.data, function(line) {
|
|
if (first > line.n) {
|
|
first = line.n;
|
|
}
|
|
lines[line.n - 1] = Ext.htmlEncode(line.t);
|
|
});
|
|
|
|
lines.length = total;
|
|
me.updateView(lines.join('<br>'), first - 1, total);
|
|
},
|
|
failure: function(response) {
|
|
if (view.failCallback) {
|
|
view.failCallback(response);
|
|
} else {
|
|
var msg = response.htmlStatus;
|
|
Proxmox.Utils.setErrorMask(me, msg);
|
|
}
|
|
}
|
|
});
|
|
},
|
|
|
|
onScroll: function(x, y) {
|
|
var me = this;
|
|
var view = me.getView();
|
|
var viewModel = me.getViewModel();
|
|
|
|
var lineHeight = view.lineHeight;
|
|
var line = view.getScrollY()/lineHeight;
|
|
var start = viewModel.get('params.start');
|
|
var limit = viewModel.get('params.limit');
|
|
var viewLines = view.getHeight()/lineHeight;
|
|
|
|
var viewStart = Math.max(parseInt(line - 1 - view.viewBuffer, 10), 0);
|
|
var viewEnd = parseInt(line + viewLines + 1 + view.viewBuffer, 10);
|
|
|
|
if (viewStart < start || viewEnd > (start+limit)) {
|
|
viewModel.set('params.start',
|
|
Math.max(parseInt(line - limit/2 + 10, 10), 0));
|
|
view.loadTask.delay(200);
|
|
}
|
|
},
|
|
|
|
init: function(view) {
|
|
var me = this;
|
|
|
|
if (!view.url) {
|
|
throw "no url specified";
|
|
}
|
|
|
|
var viewModel = this.getViewModel();
|
|
var since = new Date();
|
|
since.setDate(since.getDate() - 3);
|
|
viewModel.set('until', new Date());
|
|
viewModel.set('since', since);
|
|
viewModel.set('params.limit', view.pageSize);
|
|
viewModel.set('hide_timespan', !view.log_select_timespan);
|
|
me.lookup('content').setStyle('line-height', view.lineHeight + 'px');
|
|
|
|
view.loadTask = new Ext.util.DelayedTask(me.doLoad, me);
|
|
|
|
me.updateParams();
|
|
view.task = Ext.TaskManager.start({
|
|
run: function() {
|
|
if (!view.isVisible() || !view.scrollToEnd) {
|
|
return;
|
|
}
|
|
|
|
if (me.scrollPosBottom() <= 1) {
|
|
view.loadTask.delay(200);
|
|
}
|
|
},
|
|
interval: 1000
|
|
});
|
|
}
|
|
},
|
|
|
|
onDestroy: function() {
|
|
var me = this;
|
|
me.loadTask.cancel();
|
|
Ext.TaskManager.stop(me.task);
|
|
},
|
|
|
|
// for user to initiate a load from outside
|
|
requestUpdate: function() {
|
|
var me = this;
|
|
me.loadTask.delay(200);
|
|
},
|
|
|
|
viewModel: {
|
|
data: {
|
|
until: null,
|
|
since: null,
|
|
hide_timespan: false,
|
|
data: {
|
|
start: 0,
|
|
total: 0,
|
|
textlen: 0
|
|
},
|
|
params: {
|
|
start: 0,
|
|
limit: 500,
|
|
}
|
|
}
|
|
},
|
|
|
|
layout: 'auto',
|
|
bodyPadding: 5,
|
|
scrollable: {
|
|
x: 'auto',
|
|
y: 'auto',
|
|
listeners: {
|
|
// we have to have this here, since we cannot listen to events
|
|
// of the scroller in the viewcontroller (extjs bug?), nor does
|
|
// the panel have a 'scroll' event'
|
|
scroll: {
|
|
fn: function(scroller, x, y) {
|
|
var controller = this.component.getController();
|
|
if (controller) { // on destroy, controller can be gone
|
|
controller.onScroll(x,y);
|
|
}
|
|
},
|
|
buffer: 200
|
|
},
|
|
}
|
|
},
|
|
|
|
tbar: {
|
|
bind: {
|
|
hidden: '{hide_timespan}'
|
|
},
|
|
items: [
|
|
'->',
|
|
'Since: ',
|
|
{
|
|
xtype: 'datefield',
|
|
name: 'since_date',
|
|
reference: 'since',
|
|
format: 'Y-m-d',
|
|
bind: {
|
|
value: '{since}',
|
|
maxValue: '{until}'
|
|
}
|
|
},
|
|
'Until: ',
|
|
{
|
|
xtype: 'datefield',
|
|
name: 'until_date',
|
|
reference: 'until',
|
|
format: 'Y-m-d',
|
|
bind: {
|
|
value: '{until}',
|
|
minValue: '{since}'
|
|
}
|
|
},
|
|
{
|
|
xtype: 'button',
|
|
text: 'Update',
|
|
handler: 'updateParams'
|
|
}
|
|
],
|
|
},
|
|
|
|
items: [
|
|
{
|
|
xtype: 'box',
|
|
reference: 'content',
|
|
style: {
|
|
font: 'normal 11px tahoma, arial, verdana, sans-serif',
|
|
'white-space': 'pre'
|
|
},
|
|
}
|
|
]
|
|
});
|