123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165 |
- /*
- 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 "config.h" // IWYU pragma: keep
- #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 "RowField.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) {
- 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)
- Panel_moveSelectedUp(super);
- result = HANDLED;
- break;
- case KEY_DOWN:
- if (!this->moving)
- break;
- /* else fallthrough */
- case KEY_F(8):
- case ']':
- case '+':
- if (selected < size - 1)
- Panel_moveSelectedDown(super);
- result = HANDLED;
- break;
- case KEY_F(9):
- case KEY_DC:
- if (size > 1 && selected < size)
- 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 {
- /* heading preferred here but name is always available */
- name = column->heading ? column->heading : column->name;
- }
- }
- 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 RowField* 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;
- }
|