MetersPanel.c 6.71 KB
Newer Older
Hisham Muhammad's avatar
Hisham Muhammad committed
1
2
3
4
5
6
/*
htop - MetersPanel.c
(C) 2004-2011 Hisham H. Muhammad
Released under the GNU GPL, see the COPYING file
in the source distribution for its full text.
*/
Hisham Muhammad's avatar
Hisham Muhammad committed
7

8
#include "MetersPanel.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
9

Hisham Muhammad's avatar
Hisham Muhammad committed
10
#include <stdlib.h>
Hisham Muhammad's avatar
Hisham Muhammad committed
11
#include <assert.h>
12
#include "CRT.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
13
14

/*{
Hisham Muhammad's avatar
Hisham Muhammad committed
15
16
17
#include "Panel.h"
#include "Settings.h"
#include "ScreenManager.h"
Hisham Muhammad's avatar
Hisham Muhammad committed
18

19
20
21
typedef struct MetersPanel_ MetersPanel;

struct MetersPanel_ {
22
   Panel super;
Hisham Muhammad's avatar
Hisham Muhammad committed
23
24

   Settings* settings;
25
   Vector* meters;
Hisham Muhammad's avatar
Hisham Muhammad committed
26
   ScreenManager* scr;
27
28
   MetersPanel* leftNeighbor;
   MetersPanel* rightNeighbor;
Hisham Muhammad's avatar
Hisham Muhammad committed
29
   bool moving;
30
};
Hisham Muhammad's avatar
Hisham Muhammad committed
31
32
33

}*/

34
35
// Note: In code the meters are known to have bar/text/graph "Modes", but in UI
// we call them "Styles".
36
37
38
39
40
41
42
43
44
45
46
static const char* MetersFunctions[] = {"Style ", "Move  ", "                                       ", "Delete", "Done  ", NULL};
static const char* MetersKeys[] = {"Space", "Enter", "  ", "Del", "F10"};
static int MetersEvents[] = {' ', 13, ERR, KEY_DC, KEY_F(10)};

// We avoid UTF-8 arrows ← → here as they might display full-width on Chinese
// terminals, breaking our aligning.
// In <http://unicode.org/reports/tr11/>, arrows (U+2019..U+2199) are
// considered "Ambiguous characters".
static const char* MetersMovingFunctions[] = {"Style ", "Lock  ", "Up    ", "Down  ", "Left  ", "Right ", "       ", "Delete", "Done  ", NULL};
static const char* MetersMovingKeys[] = {"Space", "Enter", "Up", "Dn", "<-", "->", "  ", "Del", "F10"};
static int MetersMovingEvents[] = {' ', 13, KEY_UP, KEY_DOWN, KEY_LEFT, KEY_RIGHT, ERR, KEY_DC, KEY_F(10)};
47
48
static FunctionBar* Meters_movingBar = NULL;

49
static void MetersPanel_delete(Object* object) {
50
51
52
   Panel* super = (Panel*) object;
   MetersPanel* this = (MetersPanel*) object;
   Panel_done(super);
Hisham Muhammad's avatar
Hisham Muhammad committed
53
54
55
   free(this);
}

56
57
58
void MetersPanel_setMoving(MetersPanel* this, bool moving) {
   Panel* super = (Panel*) this;
   this->moving = moving;
59
60
61
62
   ListItem* selected = (ListItem*)Panel_getSelected(super);
   if (selected) {
      selected->moving = moving;
   }
63
   if (!moving) {
64
      Panel_setSelectionColor(super, CRT_colors[PANEL_SELECTION_FOCUS]);
65
66
      Panel_setDefaultBar(super);
   } else {
67
      Panel_setSelectionColor(super, CRT_colors[PANEL_SELECTION_FOLLOW]);
68
69
      super->currentBar = Meters_movingBar;
   }
Hisham's avatar
Hisham committed
70
   FunctionBar_draw(this->super.currentBar, NULL);
71
72
}

73
74
75
76
77
static inline bool moveToNeighbor(MetersPanel* this, MetersPanel* neighbor, int selected) {
   Panel* super = (Panel*) this;
   if (this->moving) {
      if (neighbor) {
         if (selected < Vector_size(this->meters)) {
78
            MetersPanel_setMoving(this, false);
79

80
81
82
            Meter* meter = (Meter*) Vector_take(this->meters, selected);
            Panel_remove(super, selected);
            Vector_insert(neighbor->meters, selected, meter);
83
            Panel_insert(&(neighbor->super), selected, (Object*) Meter_toListItem(meter, false));
84
85
            Panel_setSelected(&(neighbor->super), selected);

86
            MetersPanel_setMoving(neighbor, true);
87
88
89
90
91
92
93
            return true;
         }
      }
   }
   return false;
}

Hisham Muhammad's avatar
Hisham Muhammad committed
94
static HandlerResult MetersPanel_eventHandler(Panel* super, int ch) {
95
   MetersPanel* this = (MetersPanel*) super;
Hisham Muhammad's avatar
Hisham Muhammad committed
96
   
97
   int selected = Panel_getSelectedIndex(super);
Hisham Muhammad's avatar
Hisham Muhammad committed
98
   HandlerResult result = IGNORED;
99
   bool sideMove = false;
Hisham Muhammad's avatar
Hisham Muhammad committed
100
101
102
103
104

   switch(ch) {
      case 0x0a:
      case 0x0d:
      case KEY_ENTER:
Hisham Muhammad's avatar
Hisham Muhammad committed
105
      {
106
107
         if (!Vector_size(this->meters))
            break;
108
         MetersPanel_setMoving(this, !(this->moving));
Hisham Muhammad's avatar
Hisham Muhammad committed
109
110
111
112
         result = HANDLED;
         break;
      }
      case ' ':
Hisham Muhammad's avatar
Hisham Muhammad committed
113
114
115
      case KEY_F(4):
      case 't':
      {
116
117
         if (!Vector_size(this->meters))
            break;
118
         Meter* meter = (Meter*) Vector_get(this->meters, selected);
119
120
         int mode = meter->mode + 1;
         if (mode == LAST_METERMODE) mode = 1;
Hisham Muhammad's avatar
Hisham Muhammad committed
121
         Meter_setMode(meter, mode);
122
         Panel_set(super, selected, (Object*) Meter_toListItem(meter, this->moving));
Hisham Muhammad's avatar
Hisham Muhammad committed
123
124
125
         result = HANDLED;
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
126
127
128
129
130
131
132
      case KEY_UP:
      {
         if (!this->moving) {
            break;
         }
         /* else fallthrough */
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
133
134
135
136
      case KEY_F(7):
      case '[':
      case '-':
      {
137
         Vector_moveUp(this->meters, selected);
138
         Panel_moveSelectedUp(super);
Hisham Muhammad's avatar
Hisham Muhammad committed
139
140
141
         result = HANDLED;
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
142
143
144
145
146
147
148
      case KEY_DOWN:
      {
         if (!this->moving) {
            break;
         }
         /* else fallthrough */
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
149
150
151
152
      case KEY_F(8):
      case ']':
      case '+':
      {
153
         Vector_moveDown(this->meters, selected);
154
         Panel_moveSelectedDown(super);
Hisham Muhammad's avatar
Hisham Muhammad committed
155
156
157
         result = HANDLED;
         break;
      }
158
159
160
      case KEY_RIGHT:
      {
         sideMove = moveToNeighbor(this, this->rightNeighbor, selected);
161
162
163
164
165
166
         if (this->moving && !sideMove) {
            // lock user here until it exits positioning-mode
            result = HANDLED;
         }
         // if user is free, don't set HANDLED;
         // let ScreenManager handle focus.
167
168
169
170
171
         break;
      }
      case KEY_LEFT:
      {
         sideMove = moveToNeighbor(this, this->leftNeighbor, selected);
172
173
174
         if (this->moving && !sideMove) {
            result = HANDLED;
         }
175
176
         break;
      }
Hisham Muhammad's avatar
Hisham Muhammad committed
177
178
179
      case KEY_F(9):
      case KEY_DC:
      {
180
181
         if (!Vector_size(this->meters))
            break;
182
183
         if (selected < Vector_size(this->meters)) {
            Vector_remove(this->meters, selected);
184
            Panel_remove(super, selected);
Hisham Muhammad's avatar
Hisham Muhammad committed
185
         }
Hisham Muhammad's avatar
Hisham Muhammad committed
186
         MetersPanel_setMoving(this, false);
Hisham Muhammad's avatar
Hisham Muhammad committed
187
188
189
190
         result = HANDLED;
         break;
      }
   }
191
   if (result == HANDLED || sideMove) {
Hisham Muhammad's avatar
Hisham Muhammad committed
192
      Header* header = (Header*) this->scr->header;
Hisham Muhammad's avatar
Hisham Muhammad committed
193
      this->settings->changed = true;
Hisham Muhammad's avatar
Hisham Muhammad committed
194
195
196
197
198
199
      Header_calculateHeight(header);
      Header_draw(header);
      ScreenManager_resize(this->scr, this->scr->x1, header->height, this->scr->x2, this->scr->y2);
   }
   return result;
}
200

201
202
203
204
205
206
207
208
PanelClass MetersPanel_class = {
   .super = {
      .extends = Class(Panel),
      .delete = MetersPanel_delete
   },
   .eventHandler = MetersPanel_eventHandler
};

Hisham Muhammad's avatar
Hisham Muhammad committed
209
MetersPanel* MetersPanel_new(Settings* settings, const char* header, Vector* meters, ScreenManager* scr) {
210
   MetersPanel* this = AllocThis(MetersPanel);
211
   Panel* super = (Panel*) this;
212
   FunctionBar* fuBar = FunctionBar_new(MetersFunctions, MetersKeys, MetersEvents);
213
214
215
   if (!Meters_movingBar) {
      Meters_movingBar = FunctionBar_new(MetersMovingFunctions, MetersMovingKeys, MetersMovingEvents);
   }
216
   Panel_init(super, 1, 1, 1, 1, Class(ListItem), true, fuBar);
217
218
219
220

   this->settings = settings;
   this->meters = meters;
   this->scr = scr;
Hisham Muhammad's avatar
Hisham Muhammad committed
221
   this->moving = false;
222
223
   this->rightNeighbor = NULL;
   this->leftNeighbor = NULL;
224
225
226
   Panel_setHeader(super, header);
   for (int i = 0; i < Vector_size(meters); i++) {
      Meter* meter = (Meter*) Vector_get(meters, i);
227
      Panel_add(super, (Object*) Meter_toListItem(meter, false));
228
229
230
   }
   return this;
}