radare2/libr/core/panels.c

1554 lines
42 KiB
C
Raw Normal View History

/* Copyright radare2 2014-2018 - Author: pancake, vane11ope */
// pls move the typedefs into roons and rename it -> RConsPanel
#include <r_core.h>
#define LIMIT 256
#define PANEL_TITLE_SYMBOLS "Symbols"
#define PANEL_TITLE_STACK "Stack"
#define PANEL_TITLE_REGISTERS "Registers"
#define PANEL_TITLE_REGISTERREFS "RegisterRefs"
#define PANEL_TITLE_DISASSEMBLY "Disassembly"
#define PANEL_TITLE_NEWFILES "New files"
#define PANEL_TITLE_INFO "Info"
#define PANEL_TITLE_DATABASE "Database"
#define PANEL_TITLE_HEXDUMP "Hexdump"
#define PANEL_TITLE_FUNCTIONS "Functions"
#define PANEL_TITLE_COMMENTS "Comments"
#define PANEL_TITLE_ENTROPY "Entropy"
#define PANEL_TITLE_DRX "DRX"
#define PANEL_TITLE_SECTIONS "Sections"
#define PANEL_TITLE_STRINGS "Strings"
#define PANEL_TITLE_MAPS "Maps"
#define PANEL_TITLE_MODULES "Modules"
#define PANEL_TITLE_BACKTRACE "Backtrace"
#define PANEL_TITLE_BREAKPOINTS "Breakpoints"
#define PANEL_TITLE_IMPORTS "Imports"
#define PANEL_TITLE_CLIPBOARD "Clipboard"
#define PANEL_TITLE_FCNINFO "FcnInfo"
#define PANEL_TITLE_GRAPH "Graph"
#define PANEL_CMD_SYMBOLS "isq"
#define PANEL_CMD_STACK "px 256@r:SP"
#define PANEL_CMD_REGISTERS "dr="
#define PANEL_CMD_REGISTERREFS "drr"
#define PANEL_CMD_DISASSEMBLY "pd $r"
#define PANEL_CMD_GRAPH "agf"
#define PANEL_CONFIG_PAGE 10
static const int layoutMaxCount = 2;
2018-03-28 19:31:08 +09:00
enum {
LAYOUT_DEFAULT = 0,
LAYOUT_BALANCE = 1
};
static const char *menus[] = {
2015-08-31 11:53:03 +02:00
"File", "Edit", "View", "Tools", "Search", "Debug", "Analyze", "Help",
NULL
};
static const char *menus_File[] = {
2016-11-22 01:58:52 +01:00
"New", "Open", "Close", ".", "Sections", "Strings", "Symbols", "Imports", "Info", "Database", ".", "Quit",
NULL
};
static const char *menus_Edit[] = {
"Copy", "Paste", "Clipboard", "Write String", "Write Hex", "Write Value", "Assemble", "Fill", "io.cache",
NULL
};
static const char *menus_View[] = {
2015-08-31 05:43:38 +02:00
"Hexdump", "Disassembly", "Graph", "FcnInfo", "Functions", "Comments", "Entropy", "Colors",
NULL
};
static const char *menus_Tools[] = {
"Assembler", "Calculator", "R2 Shell", "System Shell",
NULL
};
2015-08-31 11:53:03 +02:00
static const char *menus_Search[] = {
"String", "ROP", "Code", "Hexpairs",
NULL
};
static const char *menus_Debug[] = {
2015-10-07 11:32:08 +02:00
"Registers", "RegisterRefs", "DRX", "Breakpoints",
2016-11-22 01:58:52 +01:00
"Watchpoints", "Maps", "Modules",
"Backtrace",
2016-11-22 01:58:52 +01:00
".",
"Continue", "Cont until.",
"Step", "Step Over",
NULL
};
static const char *menus_Analyze[] = {
"Function", "Program", "Calls", "References",
NULL
};
static const char *menus_Help[] = {
2016-11-22 01:58:52 +01:00
"Fortune", "Commands", "2048", "License", ".", "About",
NULL
};
static const char **menus_sub[] = {
menus_File,
menus_Edit,
menus_View,
menus_Tools,
2015-08-31 11:53:03 +02:00
menus_Search,
menus_Debug,
menus_Analyze,
menus_Help,
NULL
};
static void layoutMenu(RPanel *panel);
static void layoutDefault(RPanels *panels);
static void layoutBalance(RPanels *panels);
static void sortPanelForSplit(RPanels *panels);
static void splitPanelVertical(RCore *core);
static void splitPanelHorizontal(RCore *core);
static void panelPrint(RCore *core, RConsCanvas *can, RPanel *panel, int color);
static void addPanelFrame(RCore* core, RPanels* panels, const char *title, const char *cmd);
2018-06-13 18:12:18 +09:00
static bool checkFunc(RCore *core);
2018-05-11 18:41:22 +09:00
static void cursorLeft(RCore *core);
static void cursorRight(RCore *core);
static void delCurPanel(RPanels *panels);
static void doPanelsRefresh(RCore *core);
2018-05-11 18:41:22 +09:00
static bool handleCursorMode(RCore *core, const int key);
static void handleUpKey(RCore *core);
static void handleDownKey(RCore *core);
static void handleLeftKey(RCore *core);
static void handleRightKey(RCore *core);
static bool handleEnterKey(RCore *core);
2018-06-29 20:48:25 +09:00
static void handleTabKey(RCore *core, bool shift);
2018-05-11 18:41:22 +09:00
static int havePanel(RPanels *panels, const char *s);
static bool init(RCore *core, RPanels *panels, int w, int h);
static bool initPanels(RCore *core, RPanels *panels);
2018-05-11 18:41:22 +09:00
static void panelBreakpoint(RCore *core);
static void panelContinue(RCore *core);
2018-05-21 08:53:39 +09:00
static void panelPrompt(const char *prompt, char *buf, int len);
2018-05-11 18:41:22 +09:00
static void panelSingleStepIn(RCore *core);
static void panelSingleStepOver(RCore *core);
static void setRefreshAll(RPanels *panels);
static void setCursor(RCore *core, bool cur);
static void replaceCmd(RPanels* panels, char *title, char *cmd);
2018-05-11 18:41:22 +09:00
static void panelPrint(RCore *core, RConsCanvas *can, RPanel *panel, int color) {
if (!can || !panel|| !panel->refresh) {
return;
}
panel->refresh = false;
char *text;
char title[128];
2018-06-16 23:51:30 +09:00
int delta_x, delta_y, graph_pad = 0;
delta_x = panel->sx;
delta_y = panel->sy;
// clear the canvas first
2018-06-10 10:27:29 +09:00
r_cons_canvas_fill (can, panel->x, panel->y, panel->w, panel->h, ' ');
// for menu
RCons *cons = r_cons_singleton ();
if (panel->type == PANEL_TYPE_MENU) {
(void) r_cons_canvas_gotoxy (can, panel->x + 2, panel->y + 2);
text = r_str_ansi_crop (panel->title,
delta_x, delta_y, panel->w + 5, panel->h - delta_y);
if (text) {
r_cons_canvas_write (can, text);
free (text);
} else {
r_cons_canvas_write (can, panel->title);
}
} else {
if (color) {
const char *k = cons->pal.graph_box;
snprintf (title, sizeof (title) - 1,
"%s[x] %s"Color_RESET, k, panel->title);
} else {
snprintf (title, sizeof (title) - 1,
" %s ", panel->title);
}
if (r_cons_canvas_gotoxy (can, panel->x + 1, panel->y + 1)) {
r_cons_canvas_write (can, title); // delta_x
}
(void) r_cons_canvas_gotoxy (can, panel->x + 2, panel->y + 2);
char *cmdStr;
bool ce = core->print->cur_enabled;
if (!strcmp (panel->title, PANEL_TITLE_DISASSEMBLY)) {
core->offset = panel->addr;
r_core_seek (core, panel->addr, 1);
r_core_block_read (core);
core->print->cur_enabled = false;
cmdStr = r_core_cmd_str (core, panel->cmd);
} else if (!strcmp (panel->title, PANEL_TITLE_STACK)) {
const int delta = r_config_get_i (core->config, "stack.delta");
const char sign = (delta < 0)? '+': '-';
const int absdelta = R_ABS (delta);
cmdStr = r_core_cmd_strf (core, "%s%c%d", PANEL_CMD_STACK, sign, absdelta);
} else {
cmdStr = r_core_cmd_str (core, panel->cmd);
}
if (!strcmp (panel->title, PANEL_TITLE_GRAPH)) {
2018-06-16 23:51:30 +09:00
graph_pad = 1;
2018-06-21 10:52:06 +09:00
core->cons->event_data = core;
core->cons->event_resize = (RConsEvent) doPanelsRefresh;
2018-06-16 23:51:30 +09:00
}
if (delta_y < 0) {
delta_y = 0;
}
if (delta_x < 0) {
2018-06-16 23:51:30 +09:00
char *white = (char*)r_str_pad (' ', 128);
int idx = -delta_x;
if (idx >= sizeof (white)) {
idx = sizeof (white) - 1;
}
white[idx] = 0;
text = r_str_ansi_crop (cmdStr,
2018-06-16 23:51:30 +09:00
0, delta_y + graph_pad, panel->w + delta_x - 3, panel->h - 2 + delta_y);
char *newText = r_str_prefix_all (text, white);
if (newText) {
free (text);
text = newText;
}
} else {
text = r_str_ansi_crop (cmdStr,
2018-06-16 23:51:30 +09:00
delta_x, delta_y + graph_pad, panel->w + delta_x - 3, panel->h - 2 + delta_y);
}
if (text) {
r_cons_canvas_write (can, text);
free (text);
} else {
r_cons_canvas_write (can, panel->title);
}
free (cmdStr);
core->print->cur_enabled = ce;
}
if (color) {
r_cons_canvas_box (can, panel->x, panel->y, panel->w, panel->h, cons->pal.graph_box2);
} else {
r_cons_canvas_box (can, panel->x, panel->y, panel->w, panel->h, cons->pal.graph_box);
}
}
static void layoutMenu(RPanel *panel) {
panel->w = r_str_bounds (panel->title, &panel->h);
panel->h += 4;
}
R_API void r_core_panels_layout (RPanels *panels) {
panels->can->sx = 0;
panels->can->sy = 0;
layoutMenu (&panels->panel[panels->menu_pos]);
switch (panels->layout) {
case LAYOUT_DEFAULT:
layoutDefault (panels);
break;
case LAYOUT_BALANCE:
layoutBalance (panels);
break;
}
}
static void layoutDefault(RPanels *panels) {
int h, w = r_cons_get_size (&h);
int ph = (h - 1) / (panels->n_panels - 2);
int i;
int colpos = w - panels->columnWidth;
RPanel *panel = panels->panel;
panel[1].x = 0;
panel[1].y = 1;
if (panels->n_panels > 2) {
panel[1].w = colpos + 1;
} else {
panel[1].w = w;
}
panel[1].h = h - 1;
for (i = 2; i < panels->n_panels; i++) {
panel[i].x = colpos;
panel[i].y = 2 + (ph * (i - 2));
panel[i].w = w - colpos;
if (panel[i].w < 0) {
panel[i].w = 0;
}
if ((i + 1) == panels->n_panels) {
panel[i].h = h - panel[i].y;
} else {
panel[i].h = ph;
}
panel[i].y--;
panel[i].h++;
}
}
static void layoutBalance(RPanels *panels) {
int h, w = r_cons_get_size (&h);
int i, ii;
int panelNum = panels->n_panels - 1;
int leftCol = panelNum / 2;
int rightCol = panelNum - leftCol;
int pw = w / 2;
RPanel *panel = panels->panel;
for (i = 0; i < leftCol; i++) {
ii = i + 1;
panel[ii].x = 0;
panel[ii].y = 1 + i * (h / leftCol - 1);
panel[ii].w = pw + 2;
panel[ii].h = h / leftCol;
if (i == leftCol - 1) {
panel[ii].h = h - panel[ii].y;
} else {
panel[ii].h = h / leftCol;
}
}
for (i = 0; i < rightCol; i++) {
ii = i + 1 + leftCol;
panel[ii].x = pw + 1;
panel[ii].y = 1 + i * (h / rightCol - 1);
panel[ii].w = pw - 1;
if (i == rightCol - 1) {
panel[ii].h = h - panel[ii].y;
} else {
panel[ii].h = h / rightCol;
}
}
}
static void sortPanelForSplit(RPanels *panels) {
RPanel *panel = panels->panel;
const int n_panels = panels->n_panels;
const int curnode = panels->curnode;
int i;
RPanel tmpPanel = panel[n_panels - 1];
for (i = n_panels - 1; i >= curnode + 2; i--) {
panel[i] = panel[i - 1];
}
panel[curnode + 1] = tmpPanel;
}
static void splitPanelVertical(RCore *core) {
RPanels *panels = core->panels;
RPanel *panel = panels->panel;
const int curnode = panels->curnode;
const int owidth = panel[curnode].w;
addPanelFrame (core, panels, panel[curnode].title, panel[curnode].cmd);
sortPanelForSplit (panels);
panel[curnode].w = owidth / 2 + 1;
panel[curnode + 1].x = panel[curnode].x + panel[curnode].w - 1;
panel[curnode + 1].y = panel[curnode].y;
panel[curnode + 1].w = owidth - panel[curnode].w + 1;
panel[curnode + 1].h = panel[curnode].h;
setRefreshAll (panels);
}
static void splitPanelHorizontal(RCore *core) {
RPanels *panels = core->panels;
RPanel *panel = panels->panel;
const int curnode = panels->curnode;
const int oheight = panel[curnode].h;
addPanelFrame (core, panels, panel[curnode].title, panel[curnode].cmd);
sortPanelForSplit (panels);
panel[curnode].h = oheight / 2 + 1;
panel[curnode + 1].x = panel[curnode].x;
panel[curnode + 1].y = panel[curnode].y + panel[curnode].h - 1;
panel[curnode + 1].w = panel[curnode].w;
panel[curnode + 1].h = oheight - panel[curnode].h + 1;
setRefreshAll (panels);
}
2018-06-13 18:12:18 +09:00
R_API void r_core_panels_layout_refresh(RCore *core) {
r_core_panels_check_stackbase (core);
r_core_panels_refresh (core);
}
2018-05-11 18:41:22 +09:00
static void setCursor(RCore *core, bool cur) {
core->print->cur_enabled = cur;
if (cur) {
core->print->cur = core->panels->panel[core->panels->curnode].curpos;
} else {
core->panels->panel[core->panels->curnode].curpos = core->print->cur;
}
core->print->col = core->print->cur_enabled ? 1: 0;
}
2018-05-11 18:41:22 +09:00
static void cursorRight(RCore *core) {
if (!strcmp (core->panels->panel[core->panels->curnode].title, PANEL_TITLE_STACK) && core->print->cur >= 15) {
return;
}
core->print->cur++;
core->panels->panel[core->panels->curnode].addr++;
return;
}
2018-05-11 18:41:22 +09:00
static void cursorLeft(RCore *core) {
if (core->print->cur > 0) {
core->print->cur--;
core->panels->panel[core->panels->curnode].addr--;
}
return;
}
static void handleUpKey(RCore *core) {
RPanels *panels = core->panels;
r_cons_switchbuf (false);
panels->panel[panels->curnode].refresh = true;
if (panels->panel[panels->curnode].type == PANEL_TYPE_MENU) {
panels->menu_y--;
if (panels->menu_y < 0) {
panels->menu_y = 0;
}
} else {
if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_DISASSEMBLY)) {
core->offset = panels->panel[panels->curnode].addr;
r_core_cmd0 (core, "s-8");
panels->panel[panels->curnode].addr = core->offset;
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_STACK)) {
int width = r_config_get_i (core->config, "hex.cols");
if (width < 1) {
width = 16;
}
r_config_set_i (core->config, "stack.delta",
r_config_get_i (core->config, "stack.delta") + width);
panels->panel[panels->curnode].addr -= width;
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_REGISTERS)) {
if (core->print->cur_enabled) {
int cur = core->print->cur;
int cols = core->dbg->regcols;
cols = cols > 0 ? cols : 3;
cur -= cols;
if (cur >= 0) {
core->print->cur = cur;
}
}
2018-06-16 23:51:30 +09:00
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_GRAPH)) {
if (panels->panel[panels->curnode].sy > 0) {
panels->panel[panels->curnode].sy -= r_config_get_i (core->config, "graph.scroll");
}
} else {
if (panels->panel[panels->curnode].sy > 0) {
panels->panel[panels->curnode].sy--;
}
}
}
}
static void handleDownKey(RCore *core) {
RPanels *panels = core->panels;
r_cons_switchbuf (false);
panels->panel[panels->curnode].refresh = true;
if (panels->panel[panels->curnode].type == PANEL_TYPE_MENU) {
if (menus_sub[panels->menu_x][panels->menu_y]) {
panels->menu_y++;
}
} else {
if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_DISASSEMBLY)) {
core->offset = panels->panel[panels->curnode].addr;
int cols = core->print->cols;
RAnalFunction *f = NULL;
RAsmOp op;
f = r_anal_get_fcn_in (core->anal, core->offset, 0);
op.size = 1;
if (f && f->folded) {
cols = core->offset - f->addr + r_anal_fcn_size (f);
} else {
r_asm_set_pc (core->assembler, core->offset);
cols = r_asm_disassemble (core->assembler,
&op, core->block, 32);
}
if (cols < 1) {
cols = op.size > 1 ? op.size : 1;
}
r_core_seek (core, core->offset + cols, 1);
r_core_block_read (core);
panels->panel[panels->curnode].addr = core->offset;
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_STACK)) {
int width = r_config_get_i (core->config, "hex.cols");
if (width < 1) {
width = 16;
}
r_config_set_i (core->config, "stack.delta",
r_config_get_i (core->config, "stack.delta") - width);
panels->panel[panels->curnode].addr += width;
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_REGISTERS)) {
if (core->print->cur_enabled) {
const int cols = core->dbg->regcols;
core->print->cur += cols > 0 ? cols : 3;
}
2018-06-16 23:51:30 +09:00
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_GRAPH)) {
panels->panel[panels->curnode].sy += r_config_get_i (core->config, "graph.scroll");
} else {
panels->panel[panels->curnode].sy++;
}
}
}
static void handleLeftKey(RCore *core) {
RPanels *panels = core->panels;
r_cons_switchbuf (false);
panels->panel[panels->curnode].refresh = true;
if (core->print->cur_enabled) {
cursorLeft (core);
} else {
if (panels->curnode == panels->menu_pos) {
if (panels->menu_x) {
panels->menu_x--;
panels->menu_y = panels->menu_y? 1: 0;
}
2018-06-16 23:51:30 +09:00
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_GRAPH)) {
if (panels->panel[panels->curnode].sx > 0) {
panels->panel[panels->curnode].sx -= r_config_get_i (core->config, "graph.scroll");
}
} else {
if (panels->panel[panels->curnode].sx > 0) {
panels->panel[panels->curnode].sx--;
}
}
}
}
static void handleRightKey(RCore *core) {
RPanels *panels = core->panels;
r_cons_switchbuf (false);
panels->panel[panels->curnode].refresh = true;
if (core->print->cur_enabled) {
cursorRight (core);
} else {
if (panels->curnode == panels->menu_pos) {
if (menus[panels->menu_x + 1]) {
panels->menu_x++;
panels->menu_y = panels->menu_y ? 1: 0;
}
2018-06-16 23:51:30 +09:00
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_GRAPH)) {
panels->panel[panels->curnode].sx += r_config_get_i (core->config, "graph.scroll");
} else {
panels->panel[panels->curnode].sx++;
}
}
}
2018-05-11 18:41:22 +09:00
static bool handleCursorMode(RCore *core, const int key) {
const char *creg;
const RPanels *panels = core->panels;
char buf[128];
if (core->print->cur_enabled) {
switch (key) {
case 9: // TAB
case 'Z': // SHIFT-TAB
return true;
case 'Q':
case 'q':
setCursor (core, !core->print->cur_enabled);
panels->panel[panels->curnode].refresh = true;
return true;
case 'i':
if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_STACK)) {
// insert mode
const char *prompt = "insert hex: ";
panelPrompt (prompt, buf, sizeof (buf));
r_core_cmdf (core, "wx %s @ 0x%08" PFMT64x, buf, panels->panel[panels->curnode].addr);
2018-05-11 18:41:22 +09:00
panels->panel[panels->curnode].refresh = true;
} else if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_REGISTERS)) {
creg = core->dbg->creg;
if (creg) {
const char *prompt = "new-reg-value> ";
2018-05-21 08:53:39 +09:00
panelPrompt (prompt, buf, sizeof (buf));
r_core_cmdf (core, "dr %s = %s", creg, buf);
2018-05-11 18:41:22 +09:00
panels->panel[panels->curnode].refresh = true;
}
}
return true;
2018-05-11 18:41:22 +09:00
}
}
return false;
}
static void delCurPanel(RPanels *panels) {
2016-05-09 11:13:37 +02:00
int i;
if (panels->curnode > 0 && panels->n_panels > 3) {
for (i = panels->curnode; i < (panels->n_panels - 1); i++) {
panels->panel[i] = panels->panel[i + 1];
}
panels->panel[i].title = 0;
panels->n_panels--;
if (panels->curnode >= panels->n_panels) {
panels->curnode = panels->n_panels - 1;
}
}
}
static void replaceCmd(RPanels* panels, char *title, char *cmd) {
free (panels->panel[panels->curnode].title);
free (panels->panel[panels->curnode].cmd);
panels->panel[panels->curnode].title = strdup (title);
panels->panel[panels->curnode].cmd = r_str_newf (cmd);
setRefreshAll (panels);
}
2018-06-13 18:12:18 +09:00
static bool checkFunc(RCore *core) {
RAnalFunction *fun = r_anal_get_fcn_in (core->anal, core->offset, R_ANAL_FCN_TYPE_NULL);
if (!fun) {
r_cons_message ("Not in a function. Type 'df' to define it here");
return false;
}
if (r_list_empty (fun->bbs)) {
r_cons_message ("No basic blocks in this function. You may want to use 'afb+'.");
return false;
}
return true;
}
2018-05-11 18:41:22 +09:00
static void setRefreshAll(RPanels *panels) {
int i;
for (i = 0; i < panels->n_panels; i++) {
panels->panel[i].refresh = true;
}
}
static void addPanelFrame(RCore *core, RPanels* panels, const char *title, const char *cmd) {
RPanel* panel = panels->panel;
const int n_panels = panels->n_panels;
if (title) {
panel[n_panels].title = strdup (title);
panel[n_panels].cmd = r_str_newf (cmd);
} else {
panel[n_panels].title = r_core_cmd_str (core, cmd);
panel[n_panels].cmd = NULL;
}
panel[n_panels].type = PANEL_TYPE_FRAME;
panel[n_panels].refresh = true;
panel[n_panels].curpos = 0;
if (!strcmp (panel[n_panels].title, PANEL_TITLE_DISASSEMBLY)) {
panel[n_panels].addr = core->offset;
}
if (!strcmp (panel[n_panels].title, PANEL_TITLE_STACK)) {
const char *sp = r_reg_get_name (core->anal->reg, R_REG_NAME_SP);
const ut64 stackbase = r_reg_getv (core->anal->reg, sp);
panel[n_panels].baseAddr = stackbase;
panel[n_panels].addr = stackbase - r_config_get_i (core->config, "stack.delta");
}
panels->n_panels++;
panels->menu_y = 0;
}
static bool initPanels(RCore *core, RPanels *panels) {
panels->panel = NULL;
panels->panel = calloc (sizeof (RPanel), LIMIT);
if (!panels->panel) return false;
panels->n_panels = 0;
panels->panel[panels->n_panels].title = strdup ("");
panels->panel[panels->n_panels].type = PANEL_TYPE_MENU;
panels->panel[panels->n_panels].refresh = true;
panels->n_panels++;
addPanelFrame (core, panels, PANEL_TITLE_DISASSEMBLY, PANEL_CMD_DISASSEMBLY);
addPanelFrame (core, panels, PANEL_TITLE_SYMBOLS, PANEL_CMD_SYMBOLS);
addPanelFrame (core, panels, PANEL_TITLE_STACK, PANEL_CMD_STACK);
addPanelFrame (core, panels, PANEL_TITLE_REGISTERS, PANEL_CMD_REGISTERS);
addPanelFrame (core, panels, PANEL_TITLE_REGISTERREFS, PANEL_CMD_REGISTERREFS);
panels->curnode = 1;
return true;
}
// damn singletons.. there should be only one screen and therefor
// only one visual instance of the graph view. refactoring this
// into a struct makes the code to reference pointers unnecesarily
// we can look for a non-global solution here in the future if
// necessary
2018-06-13 18:12:18 +09:00
R_API void r_core_panels_refresh(RCore *core) {
RPanels *panels = core->panels;
RPanel *panel = panels->panel;
RConsCanvas *can = panels->can;
int menu_pos = panels->menu_pos;
int menu_x = panels->menu_x;
int menu_y = panels->menu_y;
2015-08-31 11:53:03 +02:00
char title[1024];
char str[1024];
int i, j, h, w = r_cons_get_size (&h);
const char *color = panels->curnode ? core->cons->pal.graph_box : core->cons->pal.graph_box2;
r_cons_gotoxy (0, 0);
if (!can) {
return;
}
if (panels->isResizing) {
panels->isResizing = false;
2018-06-16 15:25:00 +02:00
if (!r_cons_canvas_resize (can, w, h)) {
return;
}
2018-05-11 18:41:22 +09:00
setRefreshAll (panels);
}
2016-11-22 01:58:52 +01:00
#if 0
/* avoid flickering */
r_cons_canvas_clear (can);
r_cons_flush ();
2016-11-22 01:58:52 +01:00
#endif
if (panel) {
if (panel[panels->curnode].type == PANEL_TYPE_MENU) {
setRefreshAll (panels);
}
panel[menu_pos].x = (menu_y > 0) ? menu_x * 6 : w;
panel[menu_pos].y = 1;
free (panel[menu_pos].title);
panel[menu_pos].title = calloc (1, 1024); // r_str_newf ("%d", menu_y);
int maxsub = 0;
for (i = 0; menus_sub[i]; i++) {
maxsub = i;
}
if (menu_x < 0) {
panels->menu_x = 0;
}
if (menu_x > maxsub) {
panels->menu_x = maxsub;
}
if (menu_x >= 0 && menu_x <= maxsub && menus_sub[menu_x]) {
for (j = 0; menus_sub[menu_x][j]; j++) {
if (menu_y - 1 == j) {
strcat (panel[menu_pos].title, "> ");
} else {
strcat (panel[menu_pos].title, " ");
}
strcat (panel[menu_pos].title, menus_sub[menu_x][j]);
strcat (panel[menu_pos].title, " \n");
}
layoutMenu (&panel[menu_pos]);
}
for (i = 0; i < panels->n_panels; i++) {
if (i != panels->curnode) {
panelPrint (core, can, &panel[i], 0);
2015-08-31 11:53:03 +02:00
}
}
panelPrint (core, can, &panel[panels->curnode], 1);
}
(void) r_cons_canvas_gotoxy (can, -can->sx, -can->sy);
title[0] = 0;
if (panels->curnode == 0) {
strcpy (title, "> ");
2016-10-26 23:16:31 +02:00
}
for (i = 0; menus[i]; i++) {
if (menu_x == i) {
snprintf (str, sizeof (title) - 1, "%s[%s]"Color_RESET, color, menus[i]);
} else {
snprintf (str, sizeof (title) - 1, "%s %s "Color_RESET, color, menus[i]);
}
strcat (title, str);
}
if (panels->curnode == 0) {
r_cons_canvas_write (can, Color_BLUE);
r_cons_canvas_write (can, title);
r_cons_canvas_write (can, Color_RESET);
} else {
r_cons_canvas_write (can, Color_RESET);
r_cons_canvas_write (can, title);
}
snprintf (title, sizeof (title) - 1,
"[0x%08"PFMT64x "]", core->offset);
(void) r_cons_canvas_gotoxy (can, -can->sx + w - strlen (title), -can->sy);
r_cons_canvas_write (can, title);
r_cons_canvas_print (can);
r_cons_flush ();
}
static void doPanelsRefresh(RCore *core) {
if (!core->panels) {
return;
}
core->panels->isResizing = true;
2018-06-13 18:12:18 +09:00
r_core_panels_layout (core->panels);
r_core_panels_refresh (core);
}
static int havePanel(RPanels *panels, const char *s) {
2015-08-31 11:53:03 +02:00
int i;
if (!panels->panel || !panels->panel[0].title) {
2015-08-31 11:53:03 +02:00
return 0;
2016-10-26 23:16:31 +02:00
}
2015-08-31 11:53:03 +02:00
// add new panel for testing
for (i = 1; panels->panel[i].title; i++) {
if (!strcmp (panels->panel[i].title, s)) {
2015-08-31 11:53:03 +02:00
return 1;
}
}
return 0;
}
2018-05-11 18:41:22 +09:00
static void panelSingleStepIn(RCore *core) {
if (r_config_get_i (core->config, "cfg.debug")) {
if (core->print->cur_enabled) {
// dcu 0xaddr
r_core_cmdf (core, "dcu 0x%08"PFMT64x, core->offset + core->print->cur);
core->print->cur_enabled = 0;
} else {
r_core_cmd (core, "ds", 0);
r_core_cmd (core, ".dr*", 0);
}
} else {
r_core_cmd (core, "aes", 0);
r_core_cmd (core, ".ar*", 0);
}
if (!strcmp (core->panels->panel[core->panels->curnode].title, PANEL_TITLE_DISASSEMBLY)) {
core->panels->panel[core->panels->curnode].addr = core->offset;
}
}
2018-05-11 18:41:22 +09:00
static void panelSingleStepOver(RCore *core) {
bool io_cache = r_config_get_i (core->config, "io.cache");
r_config_set_i (core->config, "io.cache", false);
if (r_config_get_i (core->config, "cfg.debug")) {
r_core_cmd (core, "dso", 0);
r_core_cmd (core, ".dr*", 0);
} else {
r_core_cmd (core, "aeso", 0);
r_core_cmd (core, ".ar*", 0);
}
r_config_set_i (core->config, "io.cache", io_cache);
if (!strcmp (core->panels->panel[core->panels->curnode].title, PANEL_TITLE_DISASSEMBLY)) {
core->panels->panel[core->panels->curnode].addr = core->offset;
}
}
2018-05-11 18:41:22 +09:00
static void panelBreakpoint(RCore *core) {
r_core_cmd (core, "dbs $$", 0);
}
2018-05-11 18:41:22 +09:00
static void panelContinue(RCore *core) {
r_core_cmd (core, "dc", 0);
}
2018-06-13 18:12:18 +09:00
R_API void r_core_panels_check_stackbase(RCore *core) {
if (!core || !core->panels) {
return;
}
int i;
const char *sp = r_reg_get_name (core->anal->reg, R_REG_NAME_SP);
const ut64 stackbase = r_reg_getv (core->anal->reg, sp);
RPanels *panels = core->panels;
for (i = 1; i < panels->n_panels; i++) {
RPanel *panel = &panels->panel[i];
if (!strcmp (panel->title, PANEL_TITLE_STACK) && panel->baseAddr != stackbase) {
panel->baseAddr = stackbase;
panel->addr = stackbase - r_config_get_i (core->config, "stack.delta") + core->print->cur;
panel->refresh = true;
}
}
}
2018-05-21 08:53:39 +09:00
static void panelPrompt(const char *prompt, char *buf, int len) {
r_line_set_prompt (prompt);
*buf = 0;
2018-05-21 08:53:39 +09:00
r_cons_fgets (buf, len, 0, NULL);
}
static bool init (RCore *core, RPanels *panels, int w, int h) {
panels->panel = NULL;
panels->n_panels = 0;
panels->columnWidth = 80;
panels->layout = 0;
panels->menu_pos = 0;
panels->menu_x = 0;
panels->menu_y = 0;
panels->callgraph = 0;
panels->isResizing = false;
panels->can = r_cons_canvas_new (w, h);
2018-06-10 10:27:29 +09:00
r_cons_canvas_fill (panels->can, 0, 0, w, h, ' ');
if (!panels->can) {
eprintf ("Cannot create RCons.canvas context\n");
return false;
}
panels->can->linemode = r_config_get_i (core->config, "graph.linemode");
panels->can->color = r_config_get_i (core->config, "scr.color");
2016-05-09 11:13:37 +02:00
if (w < 140) {
panels->columnWidth = w / 3;
}
2018-02-21 19:49:54 +08:00
return true;
}
static bool handleEnterKey(RCore *core) {
RPanels *panels = core->panels;
if (panels->curnode == 0 && panels->menu_y) {
const char *action = menus_sub[panels->menu_x][panels->menu_y - 1];
if (strstr (action, "New")) {
addPanelFrame (core, panels, PANEL_TITLE_NEWFILES, "o");
} else if (strstr (action, "Open")) {
/* XXX doesnt autocompletes filenames */
r_cons_enable_mouse (false);
char *res = r_cons_input ("open file: ");
if (res) {
if (*res) {
r_core_cmdf (core, "o %s", res);
}
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "RegisterRefs")) {
addPanelFrame (core, panels, "drr", "drr");
} else if (strstr (action, "Registers")) {
addPanelFrame (core, panels, "dr=", "dr=");
} else if (strstr (action, "Info")) {
addPanelFrame (core, panels, PANEL_TITLE_INFO, "i");
} else if (strstr (action, "Database")) {
addPanelFrame (core, panels, PANEL_TITLE_DATABASE, "k ***");
} else if (strstr (action, "Registers")) {
if (!havePanel (panels, "Registers")) {
addPanelFrame (core, panels, PANEL_TITLE_REGISTERS, "dr=");
}
} else if (strstr (action, "About")) {
char *s = r_core_cmd_str (core, "?V");
r_cons_message (s);
free (s);
} else if (strstr (action, "Hexdump")) {
addPanelFrame (core, panels, PANEL_TITLE_HEXDUMP, "px 512");
} else if (strstr (action, "Disassembly")) {
addPanelFrame (core, panels, PANEL_TITLE_DISASSEMBLY, "pd 128");
} else if (strstr (action, "Functions")) {
addPanelFrame (core, panels, PANEL_TITLE_FUNCTIONS, "afl");
} else if (strstr (action, "Comments")) {
addPanelFrame (core, panels, PANEL_TITLE_COMMENTS, "CC");
} else if (strstr (action, "Entropy")) {
addPanelFrame (core, panels, PANEL_TITLE_ENTROPY, "p=e");
} else if (strstr (action, "Function")) {
r_core_cmdf (core, "af");
} else if (strstr (action, "DRX")) {
addPanelFrame (core, panels, PANEL_TITLE_DRX, "drx");
} else if (strstr (action, "Program")) {
r_core_cmdf (core, "aaa");
} else if (strstr (action, "Calls")) {
r_core_cmdf (core, "aac");
} else if (strstr (action, "ROP")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("rop grep: ");
if (res) {
r_core_cmdf (core, "\"/R %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "String")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("search string: ");
if (res) {
r_core_cmdf (core, "\"/ %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Hexpairs")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("search hexpairs: ");
if (res) {
r_core_cmdf (core, "\"/x %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Code")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("search code: ");
if (res) {
r_core_cmdf (core, "\"/c %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Copy")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("How many bytes? ");
if (res) {
r_core_cmdf (core, "\"y %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Write String")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("insert string: ");
if (res) {
r_core_cmdf (core, "\"w %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Write Value")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("insert number: ");
if (res) {
r_core_cmdf (core, "\"wv %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Write Hex")) {
r_cons_enable_mouse (false);
char *res = r_cons_input ("insert hexpairs: ");
if (res) {
r_core_cmdf (core, "\"wx %s\"", res);
free (res);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Calculator")) {
r_cons_enable_mouse (false);
for (;;) {
char *s = r_cons_input ("> ");
if (!s || !*s) {
free (s);
break;
}
r_core_cmdf (core, "? %s", s);
r_cons_flush ();
free (s);
}
r_cons_enable_mouse (true);
} else if (strstr (action, "Assemble")) {
r_core_visual_asm (core, core->offset);
} else if (strstr (action, "Sections")) {
addPanelFrame (core, panels, PANEL_TITLE_SECTIONS, "iSq");
} else if (strstr (action, "Close")) {
r_core_cmd0 (core, "o-*");
} else if (strstr (action, "Strings")) {
addPanelFrame (core, panels, PANEL_TITLE_STRINGS, "izq");
} else if (strstr (action, "Maps")) {
addPanelFrame (core, panels, PANEL_TITLE_MAPS, "dm");
} else if (strstr (action, "Modules")) {
addPanelFrame (core, panels, PANEL_TITLE_MODULES, "dmm");
} else if (strstr (action, "Backtrace")) {
addPanelFrame (core, panels, PANEL_TITLE_BACKTRACE, "dbt");
} else if (strstr (action, "Step")) {
r_core_cmd (core, "ds", 0);
r_cons_flush ();
} else if (strstr (action, "Step Over")) {
r_core_cmd (core, "dso", 0);
r_cons_flush ();
} else if (strstr (action, "Continue")) {
r_core_cmd (core, "dc", 0);
r_cons_flush ();
} else if (strstr (action, "Breakpoints")) {
addPanelFrame (core, panels, PANEL_TITLE_BREAKPOINTS, "db");
} else if (strstr (action, "Symbols")) {
addPanelFrame (core, panels, PANEL_TITLE_SYMBOLS, "isq");
} else if (strstr (action, "Imports")) {
addPanelFrame (core, panels, PANEL_TITLE_IMPORTS, "iiq");
} else if (strstr (action, "Paste")) {
r_core_cmd0 (core, "yy");
} else if (strstr (action, "Clipboard")) {
addPanelFrame (core, panels, PANEL_TITLE_CLIPBOARD, "yx");
} else if (strstr (action, "io.cache")) {
r_core_cmd0 (core, "e!io.cache");
} else if (strstr (action, "Fill")) {
r_cons_enable_mouse (false);
char *s = r_cons_input ("Fill with: ");
r_core_cmdf (core, "wow %s", s);
free (s);
r_cons_enable_mouse (true);
} else if (strstr (action, "References")) {
r_core_cmdf (core, "aar");
} else if (strstr (action, "FcnInfo")) {
addPanelFrame (core, panels, PANEL_TITLE_FCNINFO, "afi");
} else if (strstr (action, "Graph")) {
r_core_visual_graph (core, NULL, NULL, true);
// addPanelFrame ("Graph", "agf");
} else if (strstr (action, "System Shell")) {
r_cons_set_raw (0);
r_cons_flush ();
r_sys_cmd ("$SHELL");
} else if (strstr (action, "R2 Shell")) {
core->vmode = false;
r_core_visual_prompt_input (core);
core->vmode = true;
} else if (!strcmp (action, "2048")) {
r_cons_2048 (panels->can->color);
} else if (strstr (action, "License")) {
r_cons_message ("Copyright 2006-2016 - pancake - LGPL");
} else if (strstr (action, "Fortune")) {
char *s = r_core_cmd_str (core, "fo");
r_cons_message (s);
free (s);
} else if (strstr (action, "Commands")) {
r_core_cmd0 (core, "?;?@?;?$?;???");
r_cons_any_key (NULL);
} else if (strstr (action, "Colors")) {
r_core_cmd0 (core, "e!scr.color");
} else if (strstr (action, "Quit")) {
return false;
}
} else {
panels->curnode = panels->menu_pos;
panels->menu_y = 1;
}
return true;
}
2018-06-29 20:48:25 +09:00
static void handleTabKey(RCore *core, bool shift) {
RPanels *panels = core->panels;
r_cons_switchbuf (false);
panels->menu_y = 0;
panels->menu_x = -1;
panels->panel[panels->curnode].refresh = true;
if (!shift) {
if (panels->curnode >= panels->n_panels - 1) {
panels->curnode = 0;
panels->menu_x = 0;
} else {
panels->curnode++;
}
} else {
if (panels->curnode > 0) {
panels->curnode--;
} else {
panels->curnode = panels->n_panels - 1;
}
if (!panels->curnode) {
panels->menu_x = 0;
}
}
panels->panel[panels->curnode].refresh = true;
}
2018-06-13 18:12:18 +09:00
R_API RPanels *r_core_panels_new(RCore *core) {
int w, h;
RPanels *panels = R_NEW0 (RPanels);
if (!panels) {
return NULL;
}
w = r_cons_get_size (&h);
if (!init (core, panels, w, h)) {
2018-04-11 10:25:22 +02:00
free (panels);
return NULL;
}
return panels;
}
2018-06-13 18:12:18 +09:00
R_API void r_core_panels_free(RPanels *panels) {
int i;
r_cons_switchbuf (true);
if (panels) {
for (i = 0; i < panels->n_panels; i++) {
free (panels->panel[i].title);
free (panels->panel[i].cmd);
}
free (panels->panel);
if (panels->can) {
r_cons_canvas_free (panels->can);
}
free (panels);
}
}
R_API int r_core_visual_panels(RCore *core, RPanels *panels) {
int i, okey, key, wheel;
if (!panels) {
2018-06-13 18:12:18 +09:00
panels = r_core_panels_new (core);
if (!panels) {
2018-06-13 18:12:18 +09:00
r_core_panels_free (panels);
return false;
}
}
if (!initPanels (core, panels)) {
2018-06-13 18:12:18 +09:00
r_core_panels_free (panels);
2018-02-21 19:49:54 +08:00
return false;
}
r_cons_switchbuf (false);
int originCursor = core->print->cur;
core->print->cur = 0;
core->print->cur_enabled = false;
core->print->col = 0;
r_core_panels_layout (panels);
repeat:
core->panels = panels;
core->cons->event_data = core;
2018-05-11 18:41:22 +09:00
core->cons->event_resize = (RConsEvent) doPanelsRefresh;
2018-06-13 18:12:18 +09:00
r_core_panels_layout_refresh (core);
wheel = r_config_get_i (core->config, "scr.wheel");
2016-11-22 01:58:52 +01:00
if (wheel) {
2015-09-14 12:35:38 +02:00
r_cons_enable_mouse (true);
2016-11-22 01:58:52 +01:00
}
okey = r_cons_readchar ();
key = r_cons_arrow_to_hjkl (okey);
r_cons_switchbuf (true);
2018-05-11 18:41:22 +09:00
if (handleCursorMode (core, key)) {
goto repeat;
}
2018-05-11 18:41:22 +09:00
const char *cmd;
2018-06-13 18:12:18 +09:00
RConsCanvas *can = panels->can;
switch (key) {
case 'u':
r_core_cmd0 (core, "s-");
break;
case 'U':
r_core_cmd0 (core, "s+");
break;
case 'n':
{
r_cons_enable_mouse (false);
char *res = r_cons_input ("New panel with command: ");
if (res) {
if (*res) {
addPanelFrame (core, panels, res, res);
// do not refresh stuff
}
free (res);
}
r_cons_enable_mouse (true);
}
break;
case 'N':
{
r_cons_enable_mouse (false);
char *res = r_cons_input ("New panel with command: ");
if (res) {
if (*res) {
addPanelFrame (core, panels, NULL, res);
}
free (res);
}
r_cons_enable_mouse (true);
}
break;
case 'p':
r_core_cmd0 (core, "sp");
break;
case 'P':
r_core_cmd0 (core, "sn");
break;
case '.':
if (r_config_get_i (core->config, "cfg.debug")) {
2015-10-31 01:57:52 +01:00
r_core_cmd0 (core, "sr PC");
2017-03-24 10:46:13 +01:00
// r_core_seek (core, r_num_math (core->num, "entry0"), 1);
} else {
r_core_cmd0 (core, "s entry0; px");
}
setRefreshAll (panels);
break;
case ' ':
case '\r':
case '\n':
if (!handleEnterKey (core)) {
goto exit;
}
doPanelsRefresh (core);
break;
case '?':
r_cons_clear00 ();
r_cons_printf ("Visual Ascii Art Panels:\n"
" ! - run r2048 game\n"
" . - seek to PC or entrypoint\n"
" : - run r2 command in prompt\n"
" _ - start the hud input mode\n"
" | - split current panel vertically\n"
" - - split current panel horizontally\n"
" ? - show this help\n"
" X - close current panel\n"
" m - open menubar\n"
" V - view graph\n"
" b - browse symbols, flags, configurations, classes, ...\n"
" c - toggle cursor\n"
" C - toggle color\n"
" d - define in current address. Same as Vd\n"
" D - show disassembly in current frame\n"
" i - insert hex\n"
" M - open new custom frame\n"
" hl - toggle scr.color\n"
" HL - move vertical column split\n"
" jk - scroll/select menu\n"
" JK - select prev/next panels (same as TAB)\n"
" sS - step in / step over\n"
" uU - undo / redo seek\n"
" pP - seek to next or previous scr.nkey\n"
" nN - create new panel with given command\n"
" q - quit, back to visual mode\n"
);
r_cons_flush ();
r_cons_any_key (NULL);
break;
case 'b':
r_core_visual_browse (core);
break;
case 's':
if (r_config_get_i (core->config, "cfg.debug")) {
2017-03-24 10:46:13 +01:00
r_core_cmd0 (core, "ds;.dr*"); // ;sr PC");
} else {
2018-05-11 18:41:22 +09:00
panelSingleStepIn (core);
}
2018-05-11 18:41:22 +09:00
setRefreshAll (panels);
break;
case 'S':
panelSingleStepOver (core);
2018-05-11 18:41:22 +09:00
setRefreshAll (panels);
break;
case ':':
2015-09-14 12:35:38 +02:00
core->vmode = false;
r_core_visual_prompt_input (core);
2015-09-14 12:35:38 +02:00
core->vmode = true;
// FIX: Issue with visual mode instruction highlighter
// not updating after 'ds' or 'dcu' commands.
r_core_cmd0 (core, ".dr*");
setRefreshAll (panels);
break;
case 'c':
if (!strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_STACK) || !strcmp (panels->panel[panels->curnode].title, PANEL_TITLE_REGISTERS)) {
2018-05-11 18:41:22 +09:00
setCursor (core, !core->print->cur_enabled);
panels->panel[panels->curnode].refresh = true;
}
break;
case 'C':
2018-06-13 18:12:18 +09:00
can->color = !can->color;
2017-03-24 10:46:13 +01:00
// r_config_toggle (core->config, "scr.color");
// refresh graph
setRefreshAll (panels);
break;
case 'R':
2016-11-22 01:58:52 +01:00
if (r_config_get_i (core->config, "scr.randpal")) {
r_core_cmd0 (core, "ecr");
} else {
r_core_cmd0 (core, "ecn");
}
doPanelsRefresh (core);
break;
2018-05-10 10:51:35 +02:00
case 'A':
r_core_visual_asm (core, core->offset);
break;
case 'd':
2018-05-10 10:51:35 +02:00
r_core_visual_define (core, "");
2018-06-20 18:27:10 +09:00
setRefreshAll (panels);
2018-05-10 10:51:35 +02:00
break;
case 'D':
replaceCmd (panels, PANEL_TITLE_DISASSEMBLY, PANEL_CMD_DISASSEMBLY);
break;
case 'J':
for (i = 0; i < PANEL_CONFIG_PAGE; i++) {
handleDownKey (core);
}
break;
case 'K':
for (i = 0; i < PANEL_CONFIG_PAGE; i++) {
handleUpKey (core);
}
break;
case 'j':
handleDownKey (core);
break;
case 'k':
handleUpKey (core);
break;
2015-08-31 13:57:39 +02:00
case '_':
r_core_visual_hud (core);
break;
case 'X':
2018-05-11 18:41:22 +09:00
delCurPanel (panels);
r_core_panels_layout (core->panels);
2018-05-11 18:41:22 +09:00
setRefreshAll (panels);
break;
case 9: // TAB
handleTabKey (core, false);
break;
case 'Z': // SHIFT-TAB
handleTabKey (core, true);
break;
case 'M':
{
r_cons_enable_mouse (false);
char *name = r_cons_input ("Name: ");
char *cmd = r_cons_input ("Command: ");
if (name && *name && cmd && *cmd) {
addPanelFrame (core, panels, name, cmd);
}
free (name);
free (cmd);
r_cons_enable_mouse (true);
}
break;
case 'm':
if (panels->menu_x < 0) {
panels->menu_x = 0;
2018-06-13 18:12:18 +09:00
r_core_panels_refresh (core);
}
panels->curnode = panels->menu_pos;
panels->menu_y = 1;
break;
case 'H':
r_cons_switchbuf (false);
panels->isResizing = true;
if (panels->columnWidth + 4 < panels->can->w) {
panels->columnWidth += 4;
2018-06-20 10:13:58 +02:00
}
r_core_panels_layout (panels);
break;
case 'L':
r_cons_switchbuf (false);
panels->isResizing = true;
if (panels->columnWidth - 4 > 0) {
panels->columnWidth -= 4;
}
r_core_panels_layout (panels);
break;
case 'g':
if (checkFunc (core)) {
replaceCmd (panels, PANEL_TITLE_GRAPH, PANEL_CMD_GRAPH);
}
break;
case 'h':
handleLeftKey (core);
break;
case 'l':
handleRightKey (core);
break;
case 'V':
if (r_config_get_i (core->config, "graph.web")) {
r_core_cmd0 (core, "agv $$");
} else {
2018-06-13 18:12:18 +09:00
if (checkFunc (core)) {
int ocolor;
ocolor = r_config_get_i (core->config, "scr.color");
r_core_visual_graph (core, NULL, NULL, true);
r_config_set_i (core->config, "scr.color", ocolor);
setRefreshAll (panels);
}
}
break;
2016-11-22 01:58:52 +01:00
case ']':
r_config_set_i (core->config, "hex.cols", r_config_get_i (core->config, "hex.cols") + 1);
break;
case '[':
r_config_set_i (core->config, "hex.cols", r_config_get_i (core->config, "hex.cols") - 1);
break;
case 'w':
panels->layout++;
if (panels->layout >= layoutMaxCount) {
panels->layout = 0;
}
r_core_panels_layout (panels);
2018-05-11 18:41:22 +09:00
setRefreshAll (panels);
2016-11-22 01:58:52 +01:00
break;
case '|':
splitPanelVertical (core);
break;
case '-':
splitPanelHorizontal (core);
break;
case R_CONS_KEY_F1:
cmd = r_config_get (core->config, "key.f1");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F2:
cmd = r_config_get (core->config, "key.f2");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
} else {
2018-05-11 18:41:22 +09:00
panelBreakpoint (core);
}
break;
case R_CONS_KEY_F3:
cmd = r_config_get (core->config, "key.f3");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F4:
cmd = r_config_get (core->config, "key.f4");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F5:
cmd = r_config_get (core->config, "key.f5");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F6:
cmd = r_config_get (core->config, "key.f6");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F7:
cmd = r_config_get (core->config, "key.f7");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
} else {
2018-05-11 18:41:22 +09:00
panelSingleStepIn (core);
}
break;
case R_CONS_KEY_F8:
cmd = r_config_get (core->config, "key.f8");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
} else {
2018-05-11 18:41:22 +09:00
panelSingleStepOver (core);
}
break;
case R_CONS_KEY_F9:
cmd = r_config_get (core->config, "key.f9");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
} else {
2018-05-11 18:41:22 +09:00
panelContinue (core);
}
break;
case R_CONS_KEY_F10:
cmd = r_config_get (core->config, "key.f10");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F11:
cmd = r_config_get (core->config, "key.f11");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case R_CONS_KEY_F12:
cmd = r_config_get (core->config, "key.f12");
if (cmd && *cmd) {
key = r_core_cmd0 (core, cmd);
}
break;
case '!':
case 'q':
2015-03-23 14:45:31 +01:00
case -1: // EOF
if (panels->menu_y < 1) {
goto exit;
}
panels->menu_y = 0;
break;
2015-08-31 05:43:38 +02:00
#if 0
case 27: // ESC
if (r_cons_readchar () == 91) {
if (r_cons_readchar () == 90) {}
}
break;
2015-08-31 05:43:38 +02:00
#endif
default:
2017-03-24 10:46:13 +01:00
// eprintf ("Key %d\n", key);
// sleep (1);
break;
}
goto repeat;
exit:
core->print->cur = originCursor;
core->print->cur_enabled = false;
core->print->col = 0;
2018-06-13 18:12:18 +09:00
r_core_panels_free (panels);
core->panels = NULL;
2015-09-14 12:35:38 +02:00
return true;
}