mirror of https://github.com/xzeldon/htop.git
179 lines
4.6 KiB
C
179 lines
4.6 KiB
C
/*
|
|
htop - ColumnsPanel.c
|
|
(C) 2004-2011 Hisham H. Muhammad
|
|
Released under the GNU GPLv2+, see the COPYING file
|
|
in the source distribution for its full text.
|
|
*/
|
|
|
|
#include "ColumnsPanel.h"
|
|
|
|
#include <assert.h>
|
|
#include <ctype.h>
|
|
#include <stdlib.h>
|
|
|
|
#include "CRT.h"
|
|
#include "DynamicColumn.h"
|
|
#include "FunctionBar.h"
|
|
#include "Hashtable.h"
|
|
#include "ListItem.h"
|
|
#include "Object.h"
|
|
#include "Process.h"
|
|
#include "ProvideCurses.h"
|
|
#include "XUtils.h"
|
|
|
|
|
|
static const char* const ColumnsFunctions[] = {" ", " ", " ", " ", " ", " ", "MoveUp", "MoveDn", "Remove", "Done ", NULL};
|
|
|
|
static void ColumnsPanel_delete(Object* object) {
|
|
Panel* super = (Panel*) object;
|
|
ColumnsPanel* this = (ColumnsPanel*) object;
|
|
Panel_done(super);
|
|
free(this);
|
|
}
|
|
|
|
static HandlerResult ColumnsPanel_eventHandler(Panel* super, int ch) {
|
|
ColumnsPanel* const this = (ColumnsPanel*) super;
|
|
|
|
int selected = Panel_getSelectedIndex(super);
|
|
HandlerResult result = IGNORED;
|
|
int size = Panel_size(super);
|
|
|
|
switch(ch) {
|
|
case 0x0a:
|
|
case 0x0d:
|
|
case KEY_ENTER:
|
|
case KEY_MOUSE:
|
|
case KEY_RECLICK:
|
|
{
|
|
if (selected < size - 1) {
|
|
this->moving = !(this->moving);
|
|
Panel_setSelectionColor(super, this->moving ? PANEL_SELECTION_FOLLOW : PANEL_SELECTION_FOCUS);
|
|
ListItem* selectedItem = (ListItem*) Panel_getSelected(super);
|
|
if (selectedItem)
|
|
selectedItem->moving = this->moving;
|
|
result = HANDLED;
|
|
}
|
|
break;
|
|
}
|
|
case KEY_UP:
|
|
{
|
|
if (!this->moving) {
|
|
break;
|
|
}
|
|
}
|
|
/* else fallthrough */
|
|
case KEY_F(7):
|
|
case '[':
|
|
case '-':
|
|
{
|
|
if (selected < size - 1)
|
|
Panel_moveSelectedUp(super);
|
|
result = HANDLED;
|
|
break;
|
|
}
|
|
case KEY_DOWN:
|
|
{
|
|
if (!this->moving) {
|
|
break;
|
|
}
|
|
}
|
|
/* else fallthrough */
|
|
case KEY_F(8):
|
|
case ']':
|
|
case '+':
|
|
{
|
|
if (selected < size - 2)
|
|
Panel_moveSelectedDown(super);
|
|
result = HANDLED;
|
|
break;
|
|
}
|
|
case KEY_F(9):
|
|
case KEY_DC:
|
|
{
|
|
if (selected < size - 1) {
|
|
Panel_remove(super, selected);
|
|
}
|
|
result = HANDLED;
|
|
break;
|
|
}
|
|
default:
|
|
{
|
|
if (0 < ch && ch < 255 && isgraph((unsigned char)ch))
|
|
result = Panel_selectByTyping(super, ch);
|
|
if (result == BREAK_LOOP)
|
|
result = IGNORED;
|
|
break;
|
|
}
|
|
}
|
|
if (result == HANDLED)
|
|
ColumnsPanel_update(super);
|
|
return result;
|
|
}
|
|
|
|
const PanelClass ColumnsPanel_class = {
|
|
.super = {
|
|
.extends = Class(Panel),
|
|
.delete = ColumnsPanel_delete
|
|
},
|
|
.eventHandler = ColumnsPanel_eventHandler
|
|
};
|
|
|
|
static void ColumnsPanel_add(Panel* super, unsigned int key, Hashtable* columns) {
|
|
const char* name;
|
|
if (key < LAST_PROCESSFIELD) {
|
|
name = Process_fields[key].name;
|
|
} else {
|
|
const DynamicColumn* column = Hashtable_get(columns, key);
|
|
assert(column);
|
|
if (!column) {
|
|
name = NULL;
|
|
} else {
|
|
name = column->caption ? column->caption : column->heading;
|
|
if (!name)
|
|
name = column->name; /* name is a mandatory field */
|
|
}
|
|
}
|
|
if (name == NULL)
|
|
name = "- ";
|
|
Panel_add(super, (Object*) ListItem_new(name, key));
|
|
}
|
|
|
|
void ColumnsPanel_fill(ColumnsPanel* this, ScreenSettings* ss, Hashtable* columns) {
|
|
Panel* super = (Panel*) this;
|
|
Panel_prune(super);
|
|
for (const ProcessField* fields = ss->fields; *fields; fields++)
|
|
ColumnsPanel_add(super, *fields, columns);
|
|
this->ss = ss;
|
|
}
|
|
|
|
ColumnsPanel* ColumnsPanel_new(ScreenSettings* ss, Hashtable* columns, bool* changed) {
|
|
ColumnsPanel* this = AllocThis(ColumnsPanel);
|
|
Panel* super = (Panel*) this;
|
|
FunctionBar* fuBar = FunctionBar_new(ColumnsFunctions, NULL, NULL);
|
|
Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
|
|
|
|
this->ss = ss;
|
|
this->changed = changed;
|
|
this->moving = false;
|
|
Panel_setHeader(super, "Active Columns");
|
|
|
|
ColumnsPanel_fill(this, ss, columns);
|
|
|
|
return this;
|
|
}
|
|
|
|
void ColumnsPanel_update(Panel* super) {
|
|
ColumnsPanel* this = (ColumnsPanel*) super;
|
|
int size = Panel_size(super);
|
|
*(this->changed) = true;
|
|
this->ss->fields = xRealloc(this->ss->fields, sizeof(ProcessField) * (size + 1));
|
|
this->ss->flags = 0;
|
|
for (int i = 0; i < size; i++) {
|
|
int key = ((ListItem*) Panel_get(super, i))->key;
|
|
this->ss->fields[i] = key;
|
|
if (key < LAST_PROCESSFIELD)
|
|
this->ss->flags |= Process_fields[key].flags;
|
|
}
|
|
this->ss->fields[size] = 0;
|
|
}
|