2006-03-04 21:16:49 +03:00
|
|
|
/*
|
2006-05-30 17:47:28 +04:00
|
|
|
htop - Panel.c
|
2011-05-26 20:35:07 +04:00
|
|
|
(C) 2004-2011 Hisham H. Muhammad
|
2006-03-04 21:16:49 +03:00
|
|
|
Released under the GNU GPL, see the COPYING file
|
|
|
|
in the source distribution for its full text.
|
|
|
|
*/
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
#include "Panel.h"
|
2011-12-27 01:35:57 +04:00
|
|
|
|
2006-03-04 21:16:49 +03:00
|
|
|
#include "CRT.h"
|
|
|
|
#include "RichString.h"
|
2006-07-11 10:13:32 +04:00
|
|
|
#include "ListItem.h"
|
2015-08-19 19:43:20 +03:00
|
|
|
#include "StringUtils.h"
|
2006-03-04 21:16:49 +03:00
|
|
|
|
|
|
|
#include <math.h>
|
|
|
|
#include <stdbool.h>
|
2011-12-27 01:35:57 +04:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <ctype.h>
|
|
|
|
#include <string.h>
|
2006-03-04 21:16:49 +03:00
|
|
|
#include <assert.h>
|
2011-12-27 01:35:57 +04:00
|
|
|
|
2006-03-04 21:16:49 +03:00
|
|
|
//#link curses
|
|
|
|
|
|
|
|
/*{
|
2011-12-27 01:35:57 +04:00
|
|
|
#include "Object.h"
|
|
|
|
#include "Vector.h"
|
2015-03-23 21:26:56 +03:00
|
|
|
#include "FunctionBar.h"
|
2006-03-04 21:16:49 +03:00
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
typedef struct Panel_ Panel;
|
2006-03-04 21:16:49 +03:00
|
|
|
|
|
|
|
typedef enum HandlerResult_ {
|
2015-01-23 08:08:21 +03:00
|
|
|
HANDLED = 0x01,
|
|
|
|
IGNORED = 0x02,
|
|
|
|
BREAK_LOOP = 0x04,
|
2015-03-22 08:50:40 +03:00
|
|
|
REDRAW = 0x08,
|
|
|
|
RESCAN = 0x10,
|
2015-02-04 00:32:07 +03:00
|
|
|
SYNTH_KEY = 0x20,
|
2006-03-04 21:16:49 +03:00
|
|
|
} HandlerResult;
|
|
|
|
|
2015-03-25 05:12:43 +03:00
|
|
|
#define EVENT_SET_SELECTED -1
|
|
|
|
|
|
|
|
#define EVENT_HEADER_CLICK(x_) (-10000 + x_)
|
|
|
|
#define EVENT_IS_HEADER_CLICK(ev_) (ev_ >= -10000 && ev_ <= -9000)
|
|
|
|
#define EVENT_HEADER_CLICK_GET_X(ev_) (ev_ + 10000)
|
2008-03-05 09:54:30 +03:00
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
typedef HandlerResult(*Panel_EventHandler)(Panel*, int);
|
2006-03-04 21:16:49 +03:00
|
|
|
|
2012-12-05 19:12:20 +04:00
|
|
|
typedef struct PanelClass_ {
|
|
|
|
const ObjectClass super;
|
|
|
|
const Panel_EventHandler eventHandler;
|
|
|
|
} PanelClass;
|
|
|
|
|
|
|
|
#define As_Panel(this_) ((PanelClass*)((this_)->super.klass))
|
|
|
|
#define Panel_eventHandlerFn(this_) As_Panel(this_)->eventHandler
|
|
|
|
#define Panel_eventHandler(this_, ev_) As_Panel(this_)->eventHandler((Panel*)(this_), ev_)
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
struct Panel_ {
|
2006-03-04 21:16:49 +03:00
|
|
|
Object super;
|
|
|
|
int x, y, w, h;
|
|
|
|
WINDOW* window;
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector* items;
|
2006-03-04 21:16:49 +03:00
|
|
|
int selected;
|
|
|
|
int oldSelected;
|
2016-06-15 18:45:23 +03:00
|
|
|
int selectedLen;
|
2015-01-22 04:27:31 +03:00
|
|
|
void* eventHandlerState;
|
2014-02-27 23:35:22 +04:00
|
|
|
int scrollV;
|
|
|
|
short scrollH;
|
2006-03-04 21:16:49 +03:00
|
|
|
bool needsRedraw;
|
2015-03-23 21:26:56 +03:00
|
|
|
FunctionBar* currentBar;
|
|
|
|
FunctionBar* defaultBar;
|
2006-03-04 21:16:49 +03:00
|
|
|
RichString header;
|
2015-04-09 21:17:20 +03:00
|
|
|
int selectionColor;
|
2006-03-04 21:16:49 +03:00
|
|
|
};
|
|
|
|
|
2015-03-23 23:04:53 +03:00
|
|
|
#define Panel_setDefaultBar(this_) do{ (this_)->currentBar = (this_)->defaultBar; }while(0)
|
|
|
|
|
2006-03-04 21:16:49 +03:00
|
|
|
}*/
|
|
|
|
|
|
|
|
#ifndef MIN
|
|
|
|
#define MIN(a,b) ((a)<(b)?(a):(b))
|
|
|
|
#endif
|
|
|
|
#ifndef MAX
|
|
|
|
#define MAX(a,b) ((a)>(b)?(a):(b))
|
|
|
|
#endif
|
|
|
|
|
2016-06-15 18:45:23 +03:00
|
|
|
#define KEY_CTRL(l) ((l)-'A'+1)
|
2006-03-04 21:16:49 +03:00
|
|
|
|
2012-12-05 19:12:20 +04:00
|
|
|
PanelClass Panel_class = {
|
|
|
|
.super = {
|
|
|
|
.extends = Class(Object),
|
|
|
|
.delete = Panel_delete
|
|
|
|
},
|
2015-03-25 05:12:43 +03:00
|
|
|
.eventHandler = Panel_selectByTyping,
|
2012-12-05 19:12:20 +04:00
|
|
|
};
|
|
|
|
|
2015-03-23 21:26:56 +03:00
|
|
|
Panel* Panel_new(int x, int y, int w, int h, bool owner, ObjectClass* type, FunctionBar* fuBar) {
|
2006-05-30 17:47:28 +04:00
|
|
|
Panel* this;
|
2016-02-02 17:53:02 +03:00
|
|
|
this = xMalloc(sizeof(Panel));
|
2012-12-05 19:12:20 +04:00
|
|
|
Object_setClass(this, Class(Panel));
|
2015-03-23 21:26:56 +03:00
|
|
|
Panel_init(this, x, y, w, h, type, owner, fuBar);
|
2006-03-04 21:16:49 +03:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_delete(Object* cast) {
|
|
|
|
Panel* this = (Panel*)cast;
|
|
|
|
Panel_done(this);
|
2006-03-04 21:16:49 +03:00
|
|
|
free(this);
|
|
|
|
}
|
|
|
|
|
2015-03-23 21:26:56 +03:00
|
|
|
void Panel_init(Panel* this, int x, int y, int w, int h, ObjectClass* type, bool owner, FunctionBar* fuBar) {
|
2006-03-04 21:16:49 +03:00
|
|
|
this->x = x;
|
|
|
|
this->y = y;
|
|
|
|
this->w = w;
|
|
|
|
this->h = h;
|
2015-01-22 04:27:31 +03:00
|
|
|
this->eventHandlerState = NULL;
|
2012-12-05 19:12:20 +04:00
|
|
|
this->items = Vector_new(type, owner, DEFAULT_SIZE);
|
2006-03-04 21:16:49 +03:00
|
|
|
this->scrollV = 0;
|
|
|
|
this->scrollH = 0;
|
|
|
|
this->selected = 0;
|
|
|
|
this->oldSelected = 0;
|
|
|
|
this->needsRedraw = true;
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_beginAllocated(this->header);
|
2015-03-23 21:26:56 +03:00
|
|
|
this->defaultBar = fuBar;
|
|
|
|
this->currentBar = fuBar;
|
2015-04-09 21:17:20 +03:00
|
|
|
this->selectionColor = CRT_colors[PANEL_SELECTION_FOCUS];
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_done(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
2015-01-22 04:27:31 +03:00
|
|
|
free(this->eventHandlerState);
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_delete(this->items);
|
2015-03-23 21:26:56 +03:00
|
|
|
FunctionBar_delete(this->defaultBar);
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_end(this->header);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2015-04-09 21:17:20 +03:00
|
|
|
void Panel_setSelectionColor(Panel* this, int color) {
|
|
|
|
this->selectionColor = color;
|
|
|
|
}
|
|
|
|
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString* Panel_getHeader(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
|
|
|
this->needsRedraw = true;
|
2010-11-22 15:40:20 +03:00
|
|
|
return &(this->header);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2010-02-25 04:43:18 +03:00
|
|
|
inline void Panel_setHeader(Panel* this, const char* header) {
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_write(&(this->header), CRT_colors[PANEL_HEADER_FOCUS], header);
|
|
|
|
this->needsRedraw = true;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_move(Panel* this, int x, int y) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
|
|
|
this->x = x;
|
|
|
|
this->y = y;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_resize(Panel* this, int w, int h) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2010-11-22 15:40:20 +03:00
|
|
|
if (RichString_sizeVal(this->header) > 0)
|
2006-03-04 21:16:49 +03:00
|
|
|
h--;
|
|
|
|
this->w = w;
|
|
|
|
this->h = h;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_prune(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_prune(this->items);
|
2006-03-04 21:16:49 +03:00
|
|
|
this->scrollV = 0;
|
|
|
|
this->selected = 0;
|
|
|
|
this->oldSelected = 0;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_add(Panel* this, Object* o) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_add(this->items, o);
|
2006-03-04 21:16:49 +03:00
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_insert(Panel* this, int i, Object* o) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_insert(this->items, i, o);
|
2006-03-04 21:16:49 +03:00
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_set(Panel* this, int i, Object* o) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_set(this->items, i, o);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
Object* Panel_get(Panel* this, int i) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
return Vector_get(this->items, i);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
Object* Panel_remove(Panel* this, int i) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
|
|
|
this->needsRedraw = true;
|
2006-05-30 17:45:40 +04:00
|
|
|
Object* removed = Vector_remove(this->items, i);
|
|
|
|
if (this->selected > 0 && this->selected >= Vector_size(this->items))
|
2006-03-04 21:16:49 +03:00
|
|
|
this->selected--;
|
|
|
|
return removed;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
Object* Panel_getSelected(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
2012-03-05 15:18:27 +04:00
|
|
|
if (Vector_size(this->items) > 0)
|
|
|
|
return Vector_get(this->items, this->selected);
|
|
|
|
else
|
|
|
|
return NULL;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_moveSelectedUp(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_moveUp(this->items, this->selected);
|
2006-03-04 21:16:49 +03:00
|
|
|
if (this->selected > 0)
|
|
|
|
this->selected--;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_moveSelectedDown(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
Vector_moveDown(this->items, this->selected);
|
|
|
|
if (this->selected + 1 < Vector_size(this->items))
|
2006-03-04 21:16:49 +03:00
|
|
|
this->selected++;
|
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
int Panel_getSelectedIndex(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
|
|
|
return this->selected;
|
|
|
|
}
|
|
|
|
|
2009-06-02 08:51:23 +04:00
|
|
|
int Panel_size(Panel* this) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2006-05-30 17:45:40 +04:00
|
|
|
return Vector_size(this->items);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_setSelected(Panel* this, int selected) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2015-08-28 01:14:25 +03:00
|
|
|
int size = Vector_size(this->items);
|
|
|
|
if (selected >= size) {
|
2015-11-02 18:33:22 +03:00
|
|
|
selected = size - 1;
|
2015-08-28 01:14:25 +03:00
|
|
|
}
|
2014-05-04 01:08:46 +04:00
|
|
|
if (selected < 0)
|
|
|
|
selected = 0;
|
2006-03-04 21:16:49 +03:00
|
|
|
this->selected = selected;
|
2012-12-05 19:12:20 +04:00
|
|
|
if (Panel_eventHandlerFn(this)) {
|
2015-03-25 05:12:43 +03:00
|
|
|
Panel_eventHandler(this, EVENT_SET_SELECTED);
|
2008-03-05 09:54:30 +03:00
|
|
|
}
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
|
2006-05-30 17:47:28 +04:00
|
|
|
void Panel_draw(Panel* this, bool focus) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
|
|
|
|
2015-01-22 04:27:31 +03:00
|
|
|
int size = Vector_size(this->items);
|
2006-03-04 21:16:49 +03:00
|
|
|
int scrollH = this->scrollH;
|
2015-01-22 04:27:31 +03:00
|
|
|
int y = this->y;
|
|
|
|
int x = this->x;
|
|
|
|
int h = this->h;
|
2006-03-04 21:16:49 +03:00
|
|
|
|
2010-11-22 15:40:20 +03:00
|
|
|
int headerLen = RichString_sizeVal(this->header);
|
|
|
|
if (headerLen > 0) {
|
2006-03-04 21:16:49 +03:00
|
|
|
int attr = focus
|
|
|
|
? CRT_colors[PANEL_HEADER_FOCUS]
|
|
|
|
: CRT_colors[PANEL_HEADER_UNFOCUS];
|
|
|
|
attrset(attr);
|
|
|
|
mvhline(y, x, ' ', this->w);
|
2010-11-22 15:40:20 +03:00
|
|
|
if (scrollH < headerLen) {
|
2008-03-09 05:33:23 +03:00
|
|
|
RichString_printoffnVal(this->header, y, x, scrollH,
|
2010-11-22 15:40:20 +03:00
|
|
|
MIN(headerLen - scrollH, this->w));
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
attrset(CRT_colors[RESET_COLOR]);
|
|
|
|
y++;
|
|
|
|
}
|
2015-01-22 04:27:31 +03:00
|
|
|
|
|
|
|
// ensure scroll area is on screen
|
|
|
|
if (this->scrollV < 0) {
|
|
|
|
this->scrollV = 0;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
} else if (this->scrollV >= size) {
|
|
|
|
this->scrollV = MAX(size - 1, 0);
|
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
// ensure selection is on screen
|
|
|
|
if (this->selected < this->scrollV) {
|
|
|
|
this->scrollV = this->selected;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
} else if (this->selected >= this->scrollV + h) {
|
|
|
|
this->scrollV = this->selected - h + 1;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
int first = this->scrollV;
|
|
|
|
int upTo = MIN(first + h, size);
|
|
|
|
|
2015-04-09 21:17:20 +03:00
|
|
|
int selectionColor = focus
|
|
|
|
? this->selectionColor
|
|
|
|
: CRT_colors[PANEL_SELECTION_UNFOCUS];
|
2006-03-04 21:16:49 +03:00
|
|
|
|
|
|
|
if (this->needsRedraw) {
|
2015-01-22 04:27:31 +03:00
|
|
|
int line = 0;
|
|
|
|
for(int i = first; line < h && i < upTo; i++) {
|
2006-05-30 17:45:40 +04:00
|
|
|
Object* itemObj = Vector_get(this->items, i);
|
2013-02-26 20:50:21 +04:00
|
|
|
assert(itemObj); if(!itemObj) continue;
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_begin(item);
|
2012-12-05 19:12:20 +04:00
|
|
|
Object_display(itemObj, &item);
|
2010-11-22 15:40:20 +03:00
|
|
|
int itemLen = RichString_sizeVal(item);
|
|
|
|
int amt = MIN(itemLen - scrollH, this->w);
|
2014-02-27 23:35:22 +04:00
|
|
|
bool selected = (i == this->selected);
|
|
|
|
if (selected) {
|
2015-04-09 21:17:20 +03:00
|
|
|
attrset(selectionColor);
|
|
|
|
RichString_setAttr(&item, selectionColor);
|
2016-06-15 18:45:23 +03:00
|
|
|
this->selectedLen = itemLen;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
2015-01-22 04:27:31 +03:00
|
|
|
mvhline(y + line, x, ' ', this->w);
|
2014-02-27 23:35:22 +04:00
|
|
|
if (amt > 0)
|
2015-01-22 04:27:31 +03:00
|
|
|
RichString_printoffnVal(item, y + line, x, scrollH, amt);
|
2014-02-27 23:35:22 +04:00
|
|
|
if (selected)
|
|
|
|
attrset(CRT_colors[RESET_COLOR]);
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_end(item);
|
2015-01-22 04:27:31 +03:00
|
|
|
line++;
|
|
|
|
}
|
|
|
|
while (line < h) {
|
|
|
|
mvhline(y + line, x, ' ', this->w);
|
|
|
|
line++;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
this->needsRedraw = false;
|
|
|
|
|
|
|
|
} else {
|
2006-05-30 17:45:40 +04:00
|
|
|
Object* oldObj = Vector_get(this->items, this->oldSelected);
|
2013-02-26 20:50:21 +04:00
|
|
|
assert(oldObj);
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_begin(old);
|
2012-12-05 19:12:20 +04:00
|
|
|
Object_display(oldObj, &old);
|
2010-11-22 15:40:20 +03:00
|
|
|
int oldLen = RichString_sizeVal(old);
|
2006-05-30 17:45:40 +04:00
|
|
|
Object* newObj = Vector_get(this->items, this->selected);
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_begin(new);
|
2012-12-05 19:12:20 +04:00
|
|
|
Object_display(newObj, &new);
|
2010-11-22 15:40:20 +03:00
|
|
|
int newLen = RichString_sizeVal(new);
|
2016-06-15 18:45:23 +03:00
|
|
|
this->selectedLen = newLen;
|
2015-01-22 04:27:31 +03:00
|
|
|
mvhline(y+ this->oldSelected - first, x+0, ' ', this->w);
|
2010-11-22 15:40:20 +03:00
|
|
|
if (scrollH < oldLen)
|
2015-01-22 04:27:31 +03:00
|
|
|
RichString_printoffnVal(old, y+this->oldSelected - first, x,
|
2014-02-27 23:35:22 +04:00
|
|
|
scrollH, MIN(oldLen - scrollH, this->w));
|
2015-04-09 21:17:20 +03:00
|
|
|
attrset(selectionColor);
|
2015-01-22 04:27:31 +03:00
|
|
|
mvhline(y+this->selected - first, x+0, ' ', this->w);
|
2015-04-09 21:17:20 +03:00
|
|
|
RichString_setAttr(&new, selectionColor);
|
2010-11-22 15:40:20 +03:00
|
|
|
if (scrollH < newLen)
|
2015-01-22 04:27:31 +03:00
|
|
|
RichString_printoffnVal(new, y+this->selected - first, x,
|
2014-02-27 23:35:22 +04:00
|
|
|
scrollH, MIN(newLen - scrollH, this->w));
|
2006-03-04 21:16:49 +03:00
|
|
|
attrset(CRT_colors[RESET_COLOR]);
|
2010-11-22 15:40:20 +03:00
|
|
|
RichString_end(new);
|
|
|
|
RichString_end(old);
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
|
|
|
this->oldSelected = this->selected;
|
|
|
|
move(0, 0);
|
|
|
|
}
|
|
|
|
|
2009-06-02 08:51:23 +04:00
|
|
|
bool Panel_onKey(Panel* this, int key) {
|
2006-03-04 21:16:49 +03:00
|
|
|
assert (this != NULL);
|
2015-01-22 04:27:31 +03:00
|
|
|
|
|
|
|
int size = Vector_size(this->items);
|
2006-03-04 21:16:49 +03:00
|
|
|
switch (key) {
|
|
|
|
case KEY_DOWN:
|
2016-06-15 18:45:23 +03:00
|
|
|
case KEY_CTRL('N'):
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected++;
|
|
|
|
break;
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_UP:
|
2016-06-15 18:45:23 +03:00
|
|
|
case KEY_CTRL('P'):
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected--;
|
|
|
|
break;
|
2009-06-02 08:51:23 +04:00
|
|
|
#ifdef KEY_C_DOWN
|
|
|
|
case KEY_C_DOWN:
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected++;
|
|
|
|
break;
|
2009-06-02 08:51:23 +04:00
|
|
|
#endif
|
|
|
|
#ifdef KEY_C_UP
|
|
|
|
case KEY_C_UP:
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected--;
|
|
|
|
break;
|
2009-06-02 08:51:23 +04:00
|
|
|
#endif
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_LEFT:
|
2016-06-15 18:45:23 +03:00
|
|
|
case KEY_CTRL('B'):
|
2006-03-04 21:16:49 +03:00
|
|
|
if (this->scrollH > 0) {
|
2016-06-15 18:45:23 +03:00
|
|
|
this->scrollH -= MAX(CRT_scrollHAmount, 0);
|
2006-03-04 21:16:49 +03:00
|
|
|
this->needsRedraw = true;
|
|
|
|
}
|
2015-01-22 04:27:31 +03:00
|
|
|
break;
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_RIGHT:
|
2016-06-15 18:45:23 +03:00
|
|
|
case KEY_CTRL('F'):
|
2014-02-27 23:35:22 +04:00
|
|
|
this->scrollH += CRT_scrollHAmount;
|
2006-03-04 21:16:49 +03:00
|
|
|
this->needsRedraw = true;
|
2015-01-22 04:27:31 +03:00
|
|
|
break;
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_PPAGE:
|
2009-06-02 08:51:23 +04:00
|
|
|
this->selected -= (this->h - 1);
|
2016-08-07 02:49:37 +03:00
|
|
|
this->scrollV = MAX(0, this->scrollV - this->h + 1);
|
2009-06-02 08:51:23 +04:00
|
|
|
this->needsRedraw = true;
|
2015-01-22 04:27:31 +03:00
|
|
|
break;
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_NPAGE:
|
2009-06-02 08:51:23 +04:00
|
|
|
this->selected += (this->h - 1);
|
2016-08-07 02:49:37 +03:00
|
|
|
this->scrollV = MAX(0, MIN(Vector_size(this->items) - this->h,
|
|
|
|
this->scrollV + this->h - 1));
|
2009-06-02 08:51:23 +04:00
|
|
|
this->needsRedraw = true;
|
2015-01-22 04:27:31 +03:00
|
|
|
break;
|
2015-08-20 00:55:24 +03:00
|
|
|
case KEY_WHEELUP:
|
|
|
|
this->selected -= CRT_scrollWheelVAmount;
|
|
|
|
this->scrollV -= CRT_scrollWheelVAmount;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
break;
|
|
|
|
case KEY_WHEELDOWN:
|
|
|
|
{
|
|
|
|
this->selected += CRT_scrollWheelVAmount;
|
|
|
|
this->scrollV += CRT_scrollWheelVAmount;
|
|
|
|
if (this->scrollV > Vector_size(this->items) - this->h) {
|
|
|
|
this->scrollV = Vector_size(this->items) - this->h;
|
|
|
|
}
|
|
|
|
this->needsRedraw = true;
|
|
|
|
break;
|
|
|
|
}
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_HOME:
|
|
|
|
this->selected = 0;
|
2015-01-22 04:27:31 +03:00
|
|
|
break;
|
2006-03-04 21:16:49 +03:00
|
|
|
case KEY_END:
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected = size - 1;
|
|
|
|
break;
|
2016-06-15 18:45:23 +03:00
|
|
|
case KEY_CTRL('A'):
|
|
|
|
case '^':
|
|
|
|
this->scrollH = 0;
|
2016-06-20 00:05:18 +03:00
|
|
|
this->needsRedraw = true;
|
2016-06-15 18:45:23 +03:00
|
|
|
break;
|
|
|
|
case KEY_CTRL('E'):
|
|
|
|
case '$':
|
|
|
|
this->scrollH = MAX(this->selectedLen - this->w, 0);
|
2016-06-20 00:05:18 +03:00
|
|
|
this->needsRedraw = true;
|
2016-06-15 18:45:23 +03:00
|
|
|
break;
|
2015-01-22 04:27:31 +03:00
|
|
|
default:
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// ensure selection within bounds
|
2016-02-18 22:45:04 +03:00
|
|
|
if (this->selected < 0 || size == 0) {
|
2015-01-22 04:27:31 +03:00
|
|
|
this->selected = 0;
|
|
|
|
this->needsRedraw = true;
|
|
|
|
} else if (this->selected >= size) {
|
|
|
|
this->selected = size - 1;
|
|
|
|
this->needsRedraw = true;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
2015-01-22 04:27:31 +03:00
|
|
|
return true;
|
2006-03-04 21:16:49 +03:00
|
|
|
}
|
2011-11-18 10:08:56 +04:00
|
|
|
|
|
|
|
|
|
|
|
HandlerResult Panel_selectByTyping(Panel* this, int ch) {
|
|
|
|
int size = Panel_size(this);
|
2015-01-22 04:27:31 +03:00
|
|
|
if (!this->eventHandlerState)
|
2016-02-02 17:53:02 +03:00
|
|
|
this->eventHandlerState = xCalloc(100, sizeof(char));
|
2015-01-22 04:27:31 +03:00
|
|
|
char* buffer = this->eventHandlerState;
|
2011-11-18 10:08:56 +04:00
|
|
|
|
2018-02-05 12:59:20 +03:00
|
|
|
if (ch > 0 && ch < 255 && isalnum(ch)) {
|
2015-01-22 04:27:31 +03:00
|
|
|
int len = strlen(buffer);
|
2011-11-18 10:08:56 +04:00
|
|
|
if (len < 99) {
|
2015-01-22 04:27:31 +03:00
|
|
|
buffer[len] = ch;
|
|
|
|
buffer[len+1] = '\0';
|
2011-11-18 10:08:56 +04:00
|
|
|
}
|
|
|
|
for (int try = 0; try < 2; try++) {
|
2015-01-22 04:27:31 +03:00
|
|
|
len = strlen(buffer);
|
2011-11-18 10:08:56 +04:00
|
|
|
for (int i = 0; i < size; i++) {
|
|
|
|
char* cur = ((ListItem*) Panel_get(this, i))->value;
|
|
|
|
while (*cur == ' ') cur++;
|
2015-01-22 04:27:31 +03:00
|
|
|
if (strncasecmp(cur, buffer, len) == 0) {
|
2011-11-18 10:08:56 +04:00
|
|
|
Panel_setSelected(this, i);
|
|
|
|
return HANDLED;
|
|
|
|
}
|
|
|
|
}
|
2015-01-22 04:27:31 +03:00
|
|
|
// if current word did not match,
|
|
|
|
// retry considering the character the start of a new word.
|
|
|
|
buffer[0] = ch;
|
|
|
|
buffer[1] = '\0';
|
2011-11-18 10:08:56 +04:00
|
|
|
}
|
|
|
|
return HANDLED;
|
|
|
|
} else if (ch != ERR) {
|
2015-01-22 04:27:31 +03:00
|
|
|
buffer[0] = '\0';
|
2011-11-18 10:08:56 +04:00
|
|
|
}
|
|
|
|
if (ch == 13) {
|
|
|
|
return BREAK_LOOP;
|
|
|
|
}
|
|
|
|
return IGNORED;
|
|
|
|
}
|