htop/ScreenManager.c

400 lines
12 KiB
C
Raw Normal View History

2006-03-04 18:16:49 +00:00
/*
2011-12-26 21:35:57 +00:00
htop - ScreenManager.c
2011-05-26 16:35:07 +00:00
(C) 2004-2011 Hisham H. Muhammad
Released under the GNU GPLv2+, see the COPYING file
2006-03-04 18:16:49 +00:00
in the source distribution for its full text.
*/
2021-07-04 14:50:41 +00:00
#include "config.h" // IWYU pragma: keep
2006-03-04 18:16:49 +00:00
#include "ScreenManager.h"
2011-12-26 21:35:57 +00:00
#include <assert.h>
#include <stdbool.h>
#include <stdlib.h>
#include <sys/time.h>
#include "CRT.h"
#include "FunctionBar.h"
#include "Macros.h"
2020-10-05 13:14:54 +00:00
#include "Object.h"
#include "Platform.h"
2020-10-05 13:14:54 +00:00
#include "ProcessList.h"
#include "ProvideCurses.h"
#include "XUtils.h"
2006-03-04 18:16:49 +00:00
ScreenManager* ScreenManager_new(Header* header, const Settings* settings, const State* state, bool owner) {
2006-03-04 18:16:49 +00:00
ScreenManager* this;
2016-02-02 14:53:02 +00:00
this = xMalloc(sizeof(ScreenManager));
this->x1 = 0;
this->y1 = 0;
this->x2 = 0;
this->y2 = -1;
this->panels = Vector_new(Class(Panel), owner, DEFAULT_SIZE);
this->panelCount = 0;
this->header = header;
this->settings = settings;
2020-10-05 13:14:54 +00:00
this->state = state;
this->allowFocusChange = true;
2006-03-04 18:16:49 +00:00
return this;
}
void ScreenManager_delete(ScreenManager* this) {
Vector_delete(this->panels);
2006-03-04 18:16:49 +00:00
free(this);
}
inline int ScreenManager_size(const ScreenManager* this) {
return this->panelCount;
2006-03-04 18:16:49 +00:00
}
2015-03-23 18:26:56 +00:00
void ScreenManager_add(ScreenManager* this, Panel* item, int size) {
ScreenManager_insert(this, item, size, Vector_size(this->panels));
}
void ScreenManager_insert(ScreenManager* this, Panel* item, int size, int idx) {
int lastX = 0;
if (idx > 0) {
const Panel* last = (const Panel*) Vector_get(this->panels, idx - 1);
lastX = last->x + last->w + 1;
}
int height = LINES - this->y1 - (this->header ? this->header->height : 0) + this->y2;
if (size <= 0) {
size = COLS - this->x1 + this->x2 - lastX;
2006-03-04 18:16:49 +00:00
}
Panel_resize(item, size, height);
Panel_move(item, lastX, this->y1 + (this->header ? this->header->height : 0));
if (idx < this->panelCount) {
for (int i = idx + 1; i <= this->panelCount; i++) {
Panel* p = (Panel*) Vector_get(this->panels, i);
Panel_move(p, p->x + size, p->y);
}
}
Vector_insert(this->panels, idx, item);
2006-03-04 18:16:49 +00:00
item->needsRedraw = true;
this->panelCount++;
2006-03-04 18:16:49 +00:00
}
2010-02-25 01:43:18 +00:00
Panel* ScreenManager_remove(ScreenManager* this, int idx) {
assert(this->panelCount > idx);
int w = ((Panel*) Vector_get(this->panels, idx))->w;
Panel* panel = (Panel*) Vector_remove(this->panels, idx);
this->panelCount--;
if (idx < this->panelCount) {
for (int i = idx; i < this->panelCount; i++) {
Panel* p = (Panel*) Vector_get(this->panels, i);
Panel_move(p, p->x - w, p->y);
}
}
2006-05-30 14:00:18 +00:00
return panel;
2006-03-04 18:16:49 +00:00
}
void ScreenManager_resize(ScreenManager* this) {
int y1_header = this->y1 + (this->header ? this->header->height : 0);
int panels = this->panelCount;
int lastX = 0;
for (int i = 0; i < panels - 1; i++) {
Panel* panel = (Panel*) Vector_get(this->panels, i);
Panel_resize(panel, panel->w, LINES - y1_header + this->y2);
Panel_move(panel, lastX, y1_header);
lastX = panel->x + panel->w + 1;
2006-03-04 18:16:49 +00:00
}
Panel* panel = (Panel*) Vector_get(this->panels, panels - 1);
Panel_resize(panel, COLS - this->x1 + this->x2 - lastX, LINES - y1_header + this->y2);
Panel_move(panel, lastX, y1_header);
2006-03-04 18:16:49 +00:00
}
static void checkRecalculation(ScreenManager* this, double* oldTime, int* sortTimeout, bool* redraw, bool* rescan, bool* timedOut, bool *force_redraw) {
2015-03-22 05:50:40 +00:00
ProcessList* pl = this->header->pl;
Platform_gettime_realtime(&pl->realtime, &pl->realtimeMs);
double newTime = ((double)pl->realtime.tv_sec * 10) + ((double)pl->realtime.tv_usec / 100000);
2020-10-31 01:56:16 +00:00
2015-03-22 05:50:40 +00:00
*timedOut = (newTime - *oldTime > this->settings->delay);
2020-11-01 00:09:51 +00:00
*rescan |= *timedOut;
if (newTime < *oldTime) {
*rescan = true; // clock was adjusted?
}
if (*rescan) {
2015-03-22 05:50:40 +00:00
*oldTime = newTime;
int oldUidDigits = Process_uidDigits;
if (!this->state->pauseProcessUpdate && (*sortTimeout == 0 || this->settings->ss->treeView)) {
pl->needsSort = true;
2015-03-22 05:50:40 +00:00
*sortTimeout = 1;
}
// scan processes first - some header values are calculated there
ProcessList_scan(pl, this->state->pauseProcessUpdate);
// always update header, especially to avoid gaps in graph meters
Header_updateData(this->header);
// force redraw if the number of UID digits was changed
if (Process_uidDigits != oldUidDigits) {
*force_redraw = true;
}
2015-03-22 05:50:40 +00:00
*redraw = true;
}
if (*redraw) {
ProcessList_rebuildPanel(pl);
Header_draw(this->header);
}
*rescan = false;
}
static inline bool drawTab(int* y, int* x, int l, const char* name, bool cur) {
attrset(CRT_colors[cur ? SCREENS_CUR_BORDER : SCREENS_OTH_BORDER]);
mvaddch(*y, *x, '[');
(*x)++;
if (*x >= l)
return false;
int nameLen = strlen(name);
int n = MINIMUM(l - *x, nameLen);
attrset(CRT_colors[cur ? SCREENS_CUR_TEXT : SCREENS_OTH_TEXT]);
mvaddnstr(*y, *x, name, n);
*x += n;
if (*x >= l)
return false;
attrset(CRT_colors[cur ? SCREENS_CUR_BORDER : SCREENS_OTH_BORDER]);
mvaddch(*y, *x, ']');
*x += 2;
if (*x >= l)
return false;
return true;
}
static void ScreenManager_drawScreenTabs(ScreenManager* this) {
ScreenSettings** screens = this->settings->screens;
int cur = this->settings->ssIndex;
int l = COLS;
Panel* panel = (Panel*) Vector_get(this->panels, 0);
int y = panel->y - 1;
int x = 2;
if (this->name) {
drawTab(&y, &x, l, this->name, true);
return;
}
for (int s = 0; screens[s]; s++) {
bool ok = drawTab(&y, &x, l, screens[s]->name, s == cur);
if (!ok) {
break;
}
}
attrset(CRT_colors[RESET_COLOR]);
}
static void ScreenManager_drawPanels(ScreenManager* this, int focus, bool force_redraw) {
if (this->settings->screenTabs) {
ScreenManager_drawScreenTabs(this);
}
const int nPanels = this->panelCount;
2015-03-22 05:50:40 +00:00
for (int i = 0; i < nPanels; i++) {
Panel* panel = (Panel*) Vector_get(this->panels, i);
Panel_draw(panel,
force_redraw,
i == focus,
panel != (Panel*)this->state->mainPanel || !this->state->hideProcessSelection,
State_hideFunctionBar(this->state));
mvvline(panel->y, panel->x + panel->w, ' ', panel->h + (State_hideFunctionBar(this->state) ? 1 : 0));
2015-03-22 05:50:40 +00:00
}
}
void ScreenManager_run(ScreenManager* this, Panel** lastFocus, int* lastKey, const char* name) {
2006-03-04 18:16:49 +00:00
bool quit = false;
int focus = 0;
2019-07-12 19:41:09 +00:00
Panel* panelFocus = (Panel*) Vector_get(this->panels, focus);
double oldTime = 0.0;
int ch = ERR;
int closeTimeout = 0;
2015-03-22 05:50:40 +00:00
bool timedOut = true;
bool redraw = true;
bool force_redraw = true;
2015-03-22 05:50:40 +00:00
bool rescan = false;
int sortTimeout = 0;
int resetSortTimeout = 5;
this->name = name;
2006-03-04 18:16:49 +00:00
while (!quit) {
if (this->header) {
checkRecalculation(this, &oldTime, &sortTimeout, &redraw, &rescan, &timedOut, &force_redraw);
}
2019-07-12 19:41:09 +00:00
if (redraw || force_redraw) {
ScreenManager_drawPanels(this, focus, force_redraw);
force_redraw = false;
2006-03-04 18:16:49 +00:00
}
int prevCh = ch;
ch = Panel_getCh(panelFocus);
2015-03-25 02:12:43 +00:00
HandlerResult result = IGNORED;
#ifdef HAVE_GETMOUSE
2019-07-12 19:41:09 +00:00
if (ch == KEY_MOUSE && this->settings->enableMouse) {
2015-03-25 02:12:43 +00:00
ch = ERR;
2006-03-04 18:16:49 +00:00
MEVENT mevent;
int ok = getmouse(&mevent);
if (ok == OK) {
if (mevent.bstate & BUTTON1_RELEASED) {
if (mevent.y == LINES - 1) {
ch = FunctionBar_synthesizeEvent(panelFocus->currentBar, mevent.x);
} else {
for (int i = 0; i < this->panelCount; i++) {
Panel* panel = (Panel*) Vector_get(this->panels, i);
2020-10-31 22:28:02 +00:00
if (mevent.x >= panel->x && mevent.x <= panel->x + panel->w) {
if (mevent.y == panel->y) {
ch = EVENT_HEADER_CLICK(mevent.x - panel->x);
break;
} else if (this->settings->screenTabs && mevent.y == panel->y - 1) {
ch = EVENT_SCREEN_TAB_CLICK(mevent.x);
break;
2020-10-31 22:28:02 +00:00
} else if (mevent.y > panel->y && mevent.y <= panel->y + panel->h) {
ch = KEY_MOUSE;
if (panel == panelFocus || this->allowFocusChange) {
focus = i;
panelFocus = panel;
const Object* oldSelection = Panel_getSelected(panel);
Panel_setSelected(panel, mevent.y - panel->y + panel->scrollV - 1);
if (Panel_getSelected(panel) == oldSelection) {
ch = KEY_RECLICK;
}
}
break;
2015-03-25 02:12:43 +00:00
}
}
2006-03-04 18:16:49 +00:00
}
}
#if NCURSES_MOUSE_VERSION > 1
} else if (mevent.bstate & BUTTON4_PRESSED) {
ch = KEY_WHEELUP;
} else if (mevent.bstate & BUTTON5_PRESSED) {
ch = KEY_WHEELDOWN;
#endif
2006-03-04 18:16:49 +00:00
}
}
}
#endif
if (ch == ERR) {
if (sortTimeout > 0)
sortTimeout--;
2015-03-22 05:50:40 +00:00
if (prevCh == ch && !timedOut) {
closeTimeout++;
if (closeTimeout == 100) {
break;
}
2020-11-01 00:09:51 +00:00
} else {
closeTimeout = 0;
2020-11-01 00:09:51 +00:00
}
2015-03-22 05:50:40 +00:00
redraw = false;
2006-03-04 18:16:49 +00:00
continue;
}
switch (ch) {
case KEY_ALT('H'): ch = KEY_LEFT; break;
case KEY_ALT('J'): ch = KEY_DOWN; break;
case KEY_ALT('K'): ch = KEY_UP; break;
case KEY_ALT('L'): ch = KEY_RIGHT; break;
}
2015-03-22 05:50:40 +00:00
redraw = true;
2015-03-25 02:12:43 +00:00
if (Panel_eventHandlerFn(panelFocus)) {
result = Panel_eventHandler(panelFocus, ch);
}
if (result & SYNTH_KEY) {
ch = result >> 16;
}
if (result & REFRESH) {
2015-03-25 02:12:43 +00:00
sortTimeout = 0;
}
if (result & REDRAW) {
force_redraw = true;
}
if (result & RESIZE) {
ScreenManager_resize(this);
force_redraw = true;
}
2015-03-25 02:12:43 +00:00
if (result & RESCAN) {
rescan = true;
sortTimeout = 0;
}
if (result & HANDLED) {
continue;
} else if (result & BREAK_LOOP) {
quit = true;
continue;
}
2019-07-12 19:41:09 +00:00
switch (ch) {
2006-03-04 18:16:49 +00:00
case KEY_RESIZE:
{
ScreenManager_resize(this);
2006-03-04 18:16:49 +00:00
continue;
}
case KEY_LEFT:
case KEY_CTRL('B'):
if (this->panelCount < 2) {
goto defaultHandler;
}
2020-11-01 00:09:51 +00:00
if (!this->allowFocusChange) {
break;
2020-11-01 00:09:51 +00:00
}
tryLeft:
if (focus > 0) {
2006-03-04 18:16:49 +00:00
focus--;
2020-11-01 00:09:51 +00:00
}
panelFocus = (Panel*) Vector_get(this->panels, focus);
2020-11-01 00:09:51 +00:00
if (Panel_size(panelFocus) == 0 && focus > 0) {
2006-03-04 18:16:49 +00:00
goto tryLeft;
2020-11-01 00:09:51 +00:00
}
2006-03-04 18:16:49 +00:00
break;
case KEY_RIGHT:
case KEY_CTRL('F'):
2006-03-04 18:16:49 +00:00
case 9:
if (this->panelCount < 2) {
goto defaultHandler;
}
2020-11-01 00:09:51 +00:00
if (!this->allowFocusChange) {
break;
2020-11-01 00:09:51 +00:00
}
tryRight:
if (focus < this->panelCount - 1) {
2006-03-04 18:16:49 +00:00
focus++;
2020-11-01 00:09:51 +00:00
}
panelFocus = (Panel*) Vector_get(this->panels, focus);
2020-11-01 00:09:51 +00:00
if (Panel_size(panelFocus) == 0 && focus < this->panelCount - 1) {
2006-03-04 18:16:49 +00:00
goto tryRight;
2020-11-01 00:09:51 +00:00
}
2006-03-04 18:16:49 +00:00
break;
case 27:
2020-11-21 15:59:38 +00:00
case 'q':
case KEY_F(10):
2006-03-04 18:16:49 +00:00
quit = true;
continue;
default:
2020-11-01 00:09:51 +00:00
defaultHandler:
sortTimeout = resetSortTimeout;
2006-05-30 14:00:18 +00:00
Panel_onKey(panelFocus, ch);
2006-03-04 18:16:49 +00:00
break;
}
}
2020-11-01 00:09:51 +00:00
if (lastFocus) {
*lastFocus = panelFocus;
2020-11-01 00:09:51 +00:00
}
if (lastKey) {
*lastKey = ch;
2020-11-01 00:09:51 +00:00
}
2006-03-04 18:16:49 +00:00
}