blob: c32d22d671657152ee1a0e58d0486b1fc9e91b5c [file] [log] [blame] [raw]
/*
htop - ColumnsPanel.c
(C) 2004-2011 Hisham H. Muhammad
Released under the GNU GPL, see the COPYING file
in the source distribution for its full text.
*/
#include "config.h"
#include "ColumnsPanel.h"
#include "Platform.h"
#include "StringUtils.h"
#include "ListItem.h"
#include "CRT.h"
#include <assert.h>
#include <stdlib.h>
#include <ctype.h>
/*{
#include "Panel.h"
#include "Settings.h"
typedef struct ColumnsPanel_ {
Panel super;
Settings* settings;
bool moving;
} ColumnsPanel;
}*/
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, int repeat) {
ColumnsPanel *this = (ColumnsPanel *)super;
HandlerResult result = IGNORED;
int size = Panel_size(super);
#ifdef DISK_STATS
#define DISK_MODE (this->settings->disk_mode)
#else
#define DISK_MODE false
#endif
switch(ch) {
int selected;
case 0x0a:
case 0x0d:
case KEY_ENTER:
case KEY_MOUSE:
case KEY_RECLICK:
if (Panel_getSelectedIndex(super) < size - (DISK_MODE ? 0 : 1)) {
this->moving = !(this->moving);
Panel_setSelectionColor(super, CRT_colors[this->moving ? HTOP_PANEL_SELECTION_FOLLOW_COLOR : HTOP_PANEL_SELECTION_FOCUS_COLOR]);
((ListItem*)Panel_getSelected(super))->moving = this->moving;
result = HANDLED;
}
break;
case KEY_UP:
case KEY_PPAGE: // XXX
if (!this->moving) {
break;
}
/* else fallthrough */
case KEY_F(7):
case '[':
case '-':
while(repeat-- > 0 && Panel_getSelectedIndex(super) < size - (DISK_MODE ? 0 : 1)) {
Panel_moveSelectedUp(super);
}
result = HANDLED;
break;
case KEY_DOWN:
case KEY_NPAGE: // XXX
if (!this->moving) {
break;
}
/* else fallthrough */
case KEY_F(8):
case ']':
case '+':
while(repeat-- > 0 && Panel_getSelectedIndex(super) < size - (DISK_MODE ? 1 : 2)) {
Panel_moveSelectedDown(super);
}
result = HANDLED;
break;
case KEY_HOME:
if(!this->moving) break;
Panel_moveSelectedToTop(super);
result = HANDLED;
break;
case KEY_END:
if(!this->moving) break;
Panel_moveSelectedToBottom(super);
result = HANDLED;
break;
case KEY_F(9):
case KEY_DC:
#if 0
while(repeat-- > 0 && (selected = Panel_getSelectedIndex(super)) < size - (DISK_MODE ? 0 : 1)) {
Panel_remove(super, selected);
size--;
}
#else
// Don't repeat delete operation
selected = Panel_getSelectedIndex(super);
if(selected < size - (DISK_MODE ? 0 : 1)) {
Panel_remove(super, selected);
}
#endif
result = HANDLED;
break;
default:
if (ch < 255 && ch > 0 && isalpha(ch)) result = Panel_selectByTyping(super, ch);
if (result == BREAK_LOOP) result = IGNORED;
break;
}
#undef DISK_MODE
if (result == HANDLED) ColumnsPanel_update(super);
return result;
}
static void ColumnsPanel_onMouseSelect(Panel *this, int y) {
int old_selection = Panel_getSelectedIndex(this);
int new_selection = y - this->y + this->scrollV - 1;
if(old_selection == new_selection) return;
((ListItem *)Panel_get(this, old_selection))->moving = false;
Panel_setSelected(this, new_selection);
}
PanelClass ColumnsPanel_class = {
.super = {
.extends = Class(Panel),
.delete = ColumnsPanel_delete
},
.eventHandler = ColumnsPanel_eventHandler,
.onMouseSelect = ColumnsPanel_onMouseSelect
};
ColumnsPanel* ColumnsPanel_new(Settings* settings) {
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->settings = settings;
this->moving = false;
Panel_setHeader(super, "Active Columns");
#ifdef DISK_STATS
const unsigned int *field = settings->disk_mode ? settings->disk_fields : settings->fields;
const FieldData *field_data = settings->disk_mode ? Disk_fields : Process_fields;
#else
const unsigned int *field = settings->fields;
const FieldData *field_data = Process_fields;
#endif
while(*field) {
if (field_data[*field].name) {
Panel_add(super,
(Object *)ListItem_new(field_data[*field].name, HTOP_DEFAULT_COLOR, *field, settings));
}
field++;
}
return this;
}
int ColumnsPanel_fieldNameToIndex(const FieldData *field_data, unsigned int nfields, const char* name) {
for (unsigned int j = 1; j <= nfields; j++) {
if (String_eq(name, field_data[j].name)) {
return j;
}
}
return -1;
}
void ColumnsPanel_update(Panel* super) {
ColumnsPanel* this = (ColumnsPanel*) super;
int size = Panel_size(super);
const FieldData *field_data;
unsigned int **fields;
int *flags;
#ifdef DISK_STATS
if(this->settings->disk_mode) {
field_data = Disk_fields;
fields = &this->settings->disk_fields;
flags = &this->settings->disk_flags;
} else
#endif
{
field_data = Process_fields;
fields = &this->settings->fields;
flags = &this->settings->flags;
}
free(*fields);
*fields = xMalloc((size + 1) * sizeof(unsigned int));
*flags = 0;
for (int i = 0; i < size; i++) {
int key = ((ListItem*) Panel_get(super, i))->key;
(*fields)[i] = key;
*flags |= field_data[key].flags;
}
(*fields)[size] = 0;
this->settings->changed = true;
}