AvailableColumnsPanel.c 4.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141
  1. /*
  2. htop - AvailableColumnsPanel.c
  3. (C) 2004-2011 Hisham H. Muhammad
  4. Released under the GNU GPLv2+, see the COPYING file
  5. in the source distribution for its full text.
  6. */
  7. #include "config.h" // IWYU pragma: keep
  8. #include "AvailableColumnsPanel.h"
  9. #include <assert.h>
  10. #include <ctype.h>
  11. #include <stdbool.h>
  12. #include <stdlib.h>
  13. #include "ColumnsPanel.h"
  14. #include "DynamicColumn.h"
  15. #include "FunctionBar.h"
  16. #include "Hashtable.h"
  17. #include "ListItem.h"
  18. #include "Object.h"
  19. #include "Platform.h"
  20. #include "Process.h"
  21. #include "ProvideCurses.h"
  22. #include "RowField.h"
  23. #include "XUtils.h"
  24. static const char* const AvailableColumnsFunctions[] = {" ", " ", " ", " ", "Add ", " ", " ", " ", " ", "Done ", NULL};
  25. static void AvailableColumnsPanel_delete(Object* object) {
  26. AvailableColumnsPanel* this = (AvailableColumnsPanel*) object;
  27. Panel_done(&this->super);
  28. free(this);
  29. }
  30. static void AvailableColumnsPanel_insert(AvailableColumnsPanel* this, int at, int key) {
  31. const char* name;
  32. if (key >= ROW_DYNAMIC_FIELDS)
  33. name = DynamicColumn_name(key);
  34. else
  35. name = Process_fields[key].name;
  36. Panel_insert(this->columns, at, (Object*) ListItem_new(name, key));
  37. }
  38. static HandlerResult AvailableColumnsPanel_eventHandler(Panel* super, int ch) {
  39. AvailableColumnsPanel* this = (AvailableColumnsPanel*) super;
  40. HandlerResult result = IGNORED;
  41. switch (ch) {
  42. case 13:
  43. case KEY_ENTER:
  44. case KEY_F(5): {
  45. const ListItem* selected = (ListItem*) Panel_getSelected(super);
  46. if (!selected)
  47. break;
  48. int at = Panel_getSelectedIndex(this->columns);
  49. AvailableColumnsPanel_insert(this, at, selected->key);
  50. Panel_setSelected(this->columns, at + 1);
  51. ColumnsPanel_update(this->columns);
  52. result = HANDLED;
  53. break;
  54. }
  55. default:
  56. if (0 < ch && ch < 255 && isgraph((unsigned char)ch))
  57. result = Panel_selectByTyping(super, ch);
  58. break;
  59. }
  60. return result;
  61. }
  62. const PanelClass AvailableColumnsPanel_class = {
  63. .super = {
  64. .extends = Class(Panel),
  65. .delete = AvailableColumnsPanel_delete
  66. },
  67. .eventHandler = AvailableColumnsPanel_eventHandler
  68. };
  69. static void AvailableColumnsPanel_addDynamicColumn(ht_key_t key, void* value, void* data) {
  70. const DynamicColumn* column = (const DynamicColumn*) value;
  71. if (column->table) /* DynamicScreen, handled differently */
  72. return;
  73. AvailableColumnsPanel* this = (AvailableColumnsPanel*) data;
  74. const char* title = column->heading ? column->heading : column->name;
  75. const char* text = column->description ? column->description : column->caption;
  76. char description[256];
  77. if (text)
  78. xSnprintf(description, sizeof(description), "%s - %s", title, text);
  79. else
  80. xSnprintf(description, sizeof(description), "%s", title);
  81. Panel_add(&this->super, (Object*) ListItem_new(description, key));
  82. }
  83. // Handle DynamicColumns entries in the AvailableColumnsPanel
  84. static void AvailableColumnsPanel_addDynamicColumns(AvailableColumnsPanel* this, Hashtable* dynamicColumns) {
  85. assert(dynamicColumns);
  86. Hashtable_foreach(dynamicColumns, AvailableColumnsPanel_addDynamicColumn, this);
  87. }
  88. // Handle remaining Platform Meter entries in the AvailableColumnsPanel
  89. static void AvailableColumnsPanel_addPlatformColumns(AvailableColumnsPanel* this) {
  90. for (int i = 1; i < LAST_PROCESSFIELD; i++) {
  91. if (Process_fields[i].description) {
  92. char description[256];
  93. xSnprintf(description, sizeof(description), "%s - %s", Process_fields[i].name, Process_fields[i].description);
  94. Panel_add(&this->super, (Object*) ListItem_new(description, i));
  95. }
  96. }
  97. }
  98. // Handle DynamicColumns entries associated with DynamicScreens
  99. static void AvailableColumnsPanel_addDynamicScreens(AvailableColumnsPanel* this, const char* screen) {
  100. Platform_addDynamicScreenAvailableColumns(&this->super, screen);
  101. }
  102. void AvailableColumnsPanel_fill(AvailableColumnsPanel* this, const char* dynamicScreen, Hashtable* dynamicColumns) {
  103. Panel_prune(&this->super);
  104. if (dynamicScreen) {
  105. AvailableColumnsPanel_addDynamicScreens(this, dynamicScreen);
  106. } else {
  107. AvailableColumnsPanel_addPlatformColumns(this);
  108. AvailableColumnsPanel_addDynamicColumns(this, dynamicColumns);
  109. }
  110. }
  111. AvailableColumnsPanel* AvailableColumnsPanel_new(Panel* columns, Hashtable* dynamicColumns) {
  112. AvailableColumnsPanel* this = AllocThis(AvailableColumnsPanel);
  113. Panel* super = &this->super;
  114. FunctionBar* fuBar = FunctionBar_new(AvailableColumnsFunctions, NULL, NULL);
  115. Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
  116. Panel_setHeader(super, "Available Columns");
  117. this->columns = columns;
  118. AvailableColumnsPanel_fill(this, NULL, dynamicColumns);
  119. return this;
  120. }