htop/ScreenTabsPanel.c

375 lines
11 KiB
C

/*
htop - ScreenTabsPanel.c
(C) 2023 htop dev team
Released under the GNU GPLv2+, see the COPYING file
in the source distribution for its full text.
*/
#include "config.h" // IWYU pragma: keep
#include "ScreenTabsPanel.h"
#include <assert.h>
#include <ctype.h>
#include <stdbool.h>
#include <stdlib.h>
#include <string.h>
#include "CRT.h"
#include "FunctionBar.h"
#include "Hashtable.h"
#include "Macros.h"
#include "ProvideCurses.h"
#include "Settings.h"
#include "XUtils.h"
static HandlerResult ScreenNamesPanel_eventHandlerNormal(Panel* super, int ch);
ObjectClass ScreenTabListItem_class = {
.extends = Class(ListItem),
.display = ListItem_display,
.delete = ListItem_delete,
.compare = ListItem_compare
};
static void ScreenNamesPanel_fill(ScreenNamesPanel* this, DynamicScreen* ds) {
const Settings* settings = this->settings;
Panel* super = (Panel*) this;
Panel_prune(super);
for (unsigned int i = 0; i < settings->nScreens; i++) {
const ScreenSettings* ss = settings->screens[i];
if (ds == NULL) {
if (ss->dynamic != NULL)
continue;
/* built-in (processes, not dynamic) - e.g. Main or I/O */
} else {
if (ss->dynamic == NULL)
continue;
if (!String_eq(ds->name, ss->dynamic))
continue;
/* matching dynamic screen found, add it into the Panel */
}
Panel_add(super, (Object*) ListItem_new(ss->heading, i));
}
this->ds = ds;
}
static void ScreenTabsPanel_delete(Object* object) {
Panel* super = (Panel*) object;
ScreenTabsPanel* this = (ScreenTabsPanel*) object;
Panel_done(super);
free(this);
}
static HandlerResult ScreenTabsPanel_eventHandler(Panel* super, int ch) {
ScreenTabsPanel* const this = (ScreenTabsPanel* const) super;
HandlerResult result = IGNORED;
int selected = Panel_getSelectedIndex(super);
switch (ch) {
case EVENT_SET_SELECTED:
result = HANDLED;
break;
case KEY_F(5):
case KEY_CTRL('N'):
/* pass onto the Names panel for creating new screen */
return ScreenNamesPanel_eventHandlerNormal(&this->names->super, ch);
case KEY_UP:
case KEY_DOWN:
case KEY_NPAGE:
case KEY_PPAGE:
case KEY_HOME:
case KEY_END: {
int previous = selected;
Panel_onKey(super, ch);
selected = Panel_getSelectedIndex(super);
if (previous != selected)
result = HANDLED;
break;
}
default:
if (ch < 255 && isalpha(ch))
result = Panel_selectByTyping(super, ch);
if (result == BREAK_LOOP)
result = IGNORED;
break;
}
if (result == HANDLED) {
ScreenTabListItem* focus = (ScreenTabListItem*) Panel_getSelected(super);
if (focus) {
ScreenNamesPanel_fill(this->names, focus->ds);
}
}
return result;
}
PanelClass ScreenTabsPanel_class = {
.super = {
.extends = Class(Panel),
.delete = ScreenTabsPanel_delete,
},
.eventHandler = ScreenTabsPanel_eventHandler
};
static ScreenTabListItem* ScreenTabListItem_new(const char* value, DynamicScreen* ds) {
ScreenTabListItem* this = AllocThis(ScreenTabListItem);
ListItem_init((ListItem*)this, value, 0);
this->ds = ds;
return this;
}
static void addDynamicScreen(ATTR_UNUSED ht_key_t key, void* value, void* userdata) {
DynamicScreen* screen = (DynamicScreen*) value;
Panel* super = (Panel*) userdata;
const char* name = screen->heading ? screen->heading : screen->name;
Panel_add(super, (Object*) ScreenTabListItem_new(name, screen));
}
static const char* const ScreenTabsFunctions[] = {" ", " ", " ", " ", "New ", " ", " ", " ", " ", "Done ", NULL};
ScreenTabsPanel* ScreenTabsPanel_new(Settings* settings) {
ScreenTabsPanel* this = AllocThis(ScreenTabsPanel);
Panel* super = (Panel*) this;
FunctionBar* fuBar = FunctionBar_new(ScreenTabsFunctions, NULL, NULL);
Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
this->settings = settings;
this->names = ScreenNamesPanel_new(settings);
super->cursorOn = false;
this->cursor = 0;
Panel_setHeader(super, "Screen tabs");
assert(settings->dynamicScreens != NULL);
Panel_add(super, (Object*) ScreenTabListItem_new("Processes", NULL));
Hashtable_foreach(settings->dynamicScreens, addDynamicScreen, super);
return this;
}
// -------------
ObjectClass ScreenNameListItem_class = {
.extends = Class(ListItem),
.display = ListItem_display,
.delete = ListItem_delete,
.compare = ListItem_compare
};
ScreenNameListItem* ScreenNameListItem_new(const char* value, ScreenSettings* ss) {
ScreenNameListItem* this = AllocThis(ScreenNameListItem);
ListItem_init((ListItem*)this, value, 0);
this->ss = ss;
return this;
}
static const char* const ScreenNamesFunctions[] = {" ", " ", " ", " ", "New ", " ", " ", " ", " ", "Done ", NULL};
static void ScreenNamesPanel_delete(Object* object) {
Panel* super = (Panel*) object;
ScreenNamesPanel* this = (ScreenNamesPanel*) object;
/* do not delete screen settings still in use */
int n = Panel_size(super);
for (int i = 0; i < n; i++) {
ScreenNameListItem* item = (ScreenNameListItem*) Panel_get(super, i);
item->ss = NULL;
}
/* during renaming the ListItem's value points to our static buffer */
if (this->renamingItem)
this->renamingItem->value = this->saved;
Panel_done(super);
free(this);
}
static void renameScreenSettings(ScreenNamesPanel* this, const ListItem* item) {
const ScreenNameListItem* nameItem = (const ScreenNameListItem*) item;
ScreenSettings* ss = nameItem->ss;
free_and_xStrdup(&ss->heading, item->value);
Settings* settings = this->settings;
settings->changed = true;
settings->lastUpdate++;
}
static HandlerResult ScreenNamesPanel_eventHandlerRenaming(Panel* super, int ch) {
ScreenNamesPanel* const this = (ScreenNamesPanel*) super;
if (ch >= 32 && ch < 127 && ch != '=') {
if (this->cursor < SCREEN_NAME_LEN - 1) {
this->buffer[this->cursor] = (char)ch;
this->cursor++;
super->selectedLen = strlen(this->buffer);
Panel_setCursorToSelection(super);
}
return HANDLED;
}
switch (ch) {
case 127:
case KEY_BACKSPACE:
if (this->cursor > 0) {
this->cursor--;
this->buffer[this->cursor] = '\0';
super->selectedLen = strlen(this->buffer);
Panel_setCursorToSelection(super);
}
break;
case '\n':
case '\r':
case KEY_ENTER: {
ListItem* item = (ListItem*) Panel_getSelected(super);
if (!item)
break;
assert(item == this->renamingItem);
free(this->saved);
item->value = xStrdup(this->buffer);
this->renamingItem = NULL;
super->cursorOn = false;
Panel_setSelectionColor(super, PANEL_SELECTION_FOCUS);
renameScreenSettings(this, item);
break;
}
case 27: { // Esc
ListItem* item = (ListItem*) Panel_getSelected(super);
if (!item)
break;
assert(item == this->renamingItem);
item->value = this->saved;
this->renamingItem = NULL;
super->cursorOn = false;
Panel_setSelectionColor(super, PANEL_SELECTION_FOCUS);
break;
}
}
return HANDLED;
}
static void startRenaming(Panel* super) {
ScreenNamesPanel* const this = (ScreenNamesPanel*) super;
ListItem* item = (ListItem*) Panel_getSelected(super);
if (item == NULL)
return;
this->renamingItem = item;
super->cursorOn = true;
char* name = item->value;
this->saved = name;
strncpy(this->buffer, name, SCREEN_NAME_LEN);
this->buffer[SCREEN_NAME_LEN] = '\0';
this->cursor = strlen(this->buffer);
item->value = this->buffer;
Panel_setSelectionColor(super, PANEL_EDIT);
super->selectedLen = strlen(this->buffer);
Panel_setCursorToSelection(super);
}
static void addNewScreen(Panel* super, DynamicScreen* ds) {
ScreenNamesPanel* const this = (ScreenNamesPanel*) super;
const char* name = "New";
ScreenSettings* ss = (ds != NULL) ? Settings_newDynamicScreen(this->settings, name, ds, NULL) : Settings_newScreen(this->settings, &(const ScreenDefaults) { .name = name, .columns = "PID Command", .sortKey = "PID" });
ScreenNameListItem* item = ScreenNameListItem_new(name, ss);
int idx = Panel_getSelectedIndex(super);
Panel_insert(super, idx + 1, (Object*) item);
Panel_setSelected(super, idx + 1);
}
static HandlerResult ScreenNamesPanel_eventHandlerNormal(Panel* super, int ch) {
ScreenNamesPanel* const this = (ScreenNamesPanel*) super;
ScreenNameListItem* oldFocus = (ScreenNameListItem*) Panel_getSelected(super);
HandlerResult result = IGNORED;
switch (ch) {
case '\n':
case '\r':
case KEY_ENTER:
case KEY_MOUSE:
case KEY_RECLICK:
Panel_setSelectionColor(super, PANEL_SELECTION_FOCUS);
result = HANDLED;
break;
case EVENT_SET_SELECTED:
result = HANDLED;
break;
case KEY_NPAGE:
case KEY_PPAGE:
case KEY_HOME:
case KEY_END:
Panel_onKey(super, ch);
break;
case KEY_F(5):
case KEY_CTRL('N'):
addNewScreen(super, this->ds);
startRenaming(super);
result = HANDLED;
break;
default:
if (ch < 255 && isalpha(ch))
result = Panel_selectByTyping(super, ch);
if (result == BREAK_LOOP)
result = IGNORED;
break;
}
ScreenNameListItem* newFocus = (ScreenNameListItem*) Panel_getSelected(super);
if (newFocus && oldFocus != newFocus)
result = HANDLED;
return result;
}
static HandlerResult ScreenNamesPanel_eventHandler(Panel* super, int ch) {
ScreenNamesPanel* const this = (ScreenNamesPanel*) super;
if (!this->renamingItem)
return ScreenNamesPanel_eventHandlerNormal(super, ch);
return ScreenNamesPanel_eventHandlerRenaming(super, ch);
}
PanelClass ScreenNamesPanel_class = {
.super = {
.extends = Class(Panel),
.delete = ScreenNamesPanel_delete
},
.eventHandler = ScreenNamesPanel_eventHandler
};
ScreenNamesPanel* ScreenNamesPanel_new(Settings* settings) {
ScreenNamesPanel* this = AllocThis(ScreenNamesPanel);
Panel* super = (Panel*) this;
FunctionBar* fuBar = FunctionBar_new(ScreenNamesFunctions, NULL, NULL);
Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
this->settings = settings;
this->renamingItem = NULL;
memset(this->buffer, 0, sizeof(this->buffer));
this->ds = NULL;
this->saved = NULL;
this->cursor = 0;
super->cursorOn = false;
Panel_setHeader(super, "Screens");
for (unsigned int i = 0; i < settings->nScreens; i++) {
ScreenSettings* ss = settings->screens[i];
/* initially show only for Processes tabs (selected) */
if (ss->dynamic)
continue;
Panel_add(super, (Object*) ScreenNameListItem_new(ss->heading, ss));
}
return this;
}