PageRenderTime 79ms CodeModel.GetById 48ms RepoModel.GetById 1ms app.codeStats 0ms

/tags/TMUX_1_2/window-choose.c

https://github.com/jordansissel/tmux
C | 457 lines | 377 code | 62 blank | 18 comment | 78 complexity | 59e8e419cc04dc98cce008b7f223175d MD5 | raw file
  1. /* $Id: window-choose.c,v 1.29 2010-02-02 23:55:21 tcunha Exp $ */
  2. /*
  3. * Copyright (c) 2009 Nicholas Marriott <nicm@users.sourceforge.net>
  4. *
  5. * Permission to use, copy, modify, and distribute this software for any
  6. * purpose with or without fee is hereby granted, provided that the above
  7. * copyright notice and this permission notice appear in all copies.
  8. *
  9. * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
  10. * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
  11. * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
  12. * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
  13. * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
  14. * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
  15. * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
  16. */
  17. #include <sys/types.h>
  18. #include <string.h>
  19. #include "tmux.h"
  20. struct screen *window_choose_init(struct window_pane *);
  21. void window_choose_free(struct window_pane *);
  22. void window_choose_resize(struct window_pane *, u_int, u_int);
  23. void window_choose_key(struct window_pane *, struct client *, int);
  24. void window_choose_mouse(
  25. struct window_pane *, struct client *, struct mouse_event *);
  26. void window_choose_redraw_screen(struct window_pane *);
  27. void window_choose_write_line(
  28. struct window_pane *, struct screen_write_ctx *, u_int);
  29. void window_choose_scroll_up(struct window_pane *);
  30. void window_choose_scroll_down(struct window_pane *);
  31. const struct window_mode window_choose_mode = {
  32. window_choose_init,
  33. window_choose_free,
  34. window_choose_resize,
  35. window_choose_key,
  36. window_choose_mouse,
  37. NULL,
  38. };
  39. struct window_choose_mode_item {
  40. char *name;
  41. int idx;
  42. };
  43. struct window_choose_mode_data {
  44. struct screen screen;
  45. struct mode_key_data mdata;
  46. ARRAY_DECL(, struct window_choose_mode_item) list;
  47. u_int top;
  48. u_int selected;
  49. void (*callbackfn)(void *, int);
  50. void (*freefn)(void *);
  51. void *data;
  52. };
  53. int window_choose_key_index(struct window_choose_mode_data *, u_int);
  54. int window_choose_index_key(struct window_choose_mode_data *, int);
  55. void
  56. window_choose_vadd(struct window_pane *wp, int idx, const char *fmt, va_list ap)
  57. {
  58. struct window_choose_mode_data *data = wp->modedata;
  59. struct window_choose_mode_item *item;
  60. ARRAY_EXPAND(&data->list, 1);
  61. item = &ARRAY_LAST(&data->list);
  62. xvasprintf(&item->name, fmt, ap);
  63. item->idx = idx;
  64. }
  65. void printflike3
  66. window_choose_add(struct window_pane *wp, int idx, const char *fmt, ...)
  67. {
  68. va_list ap;
  69. va_start(ap, fmt);
  70. window_choose_vadd(wp, idx, fmt, ap);
  71. va_end(ap);
  72. }
  73. void
  74. window_choose_ready(struct window_pane *wp, u_int cur,
  75. void (*callbackfn)(void *, int), void (*freefn)(void *), void *cdata)
  76. {
  77. struct window_choose_mode_data *data = wp->modedata;
  78. struct screen *s = &data->screen;
  79. data->selected = cur;
  80. if (data->selected > screen_size_y(s) - 1)
  81. data->top = ARRAY_LENGTH(&data->list) - screen_size_y(s);
  82. data->callbackfn = callbackfn;
  83. data->freefn = freefn;
  84. data->data = cdata;
  85. window_choose_redraw_screen(wp);
  86. }
  87. struct screen *
  88. window_choose_init(struct window_pane *wp)
  89. {
  90. struct window_choose_mode_data *data;
  91. struct screen *s;
  92. int keys;
  93. wp->modedata = data = xmalloc(sizeof *data);
  94. data->callbackfn = NULL;
  95. data->freefn = NULL;
  96. data->data = NULL;
  97. ARRAY_INIT(&data->list);
  98. data->top = 0;
  99. s = &data->screen;
  100. screen_init(s, screen_size_x(&wp->base), screen_size_y(&wp->base), 0);
  101. s->mode &= ~MODE_CURSOR;
  102. if (options_get_number(&wp->window->options, "mode-mouse"))
  103. s->mode |= MODE_MOUSE;
  104. keys = options_get_number(&wp->window->options, "mode-keys");
  105. if (keys == MODEKEY_EMACS)
  106. mode_key_init(&data->mdata, &mode_key_tree_emacs_choice);
  107. else
  108. mode_key_init(&data->mdata, &mode_key_tree_vi_choice);
  109. return (s);
  110. }
  111. void
  112. window_choose_free(struct window_pane *wp)
  113. {
  114. struct window_choose_mode_data *data = wp->modedata;
  115. u_int i;
  116. if (data->freefn != NULL && data->data != NULL)
  117. data->freefn(data->data);
  118. for (i = 0; i < ARRAY_LENGTH(&data->list); i++)
  119. xfree(ARRAY_ITEM(&data->list, i).name);
  120. ARRAY_FREE(&data->list);
  121. screen_free(&data->screen);
  122. xfree(data);
  123. }
  124. void
  125. window_choose_resize(struct window_pane *wp, u_int sx, u_int sy)
  126. {
  127. struct window_choose_mode_data *data = wp->modedata;
  128. struct screen *s = &data->screen;
  129. data->top = 0;
  130. if (data->selected > sy - 1)
  131. data->top = data->selected - (sy - 1);
  132. screen_resize(s, sx, sy);
  133. window_choose_redraw_screen(wp);
  134. }
  135. /* ARGSUSED */
  136. void
  137. window_choose_key(struct window_pane *wp, unused struct client *c, int key)
  138. {
  139. struct window_choose_mode_data *data = wp->modedata;
  140. struct screen *s = &data->screen;
  141. struct screen_write_ctx ctx;
  142. struct window_choose_mode_item *item;
  143. u_int items;
  144. int idx;
  145. items = ARRAY_LENGTH(&data->list);
  146. switch (mode_key_lookup(&data->mdata, key)) {
  147. case MODEKEYCHOICE_CANCEL:
  148. data->callbackfn(data->data, -1);
  149. window_pane_reset_mode(wp);
  150. break;
  151. case MODEKEYCHOICE_CHOOSE:
  152. item = &ARRAY_ITEM(&data->list, data->selected);
  153. data->callbackfn(data->data, item->idx);
  154. window_pane_reset_mode(wp);
  155. break;
  156. case MODEKEYCHOICE_UP:
  157. if (items == 0)
  158. break;
  159. if (data->selected == 0) {
  160. data->selected = items - 1;
  161. if (data->selected > screen_size_y(s) - 1)
  162. data->top = items - screen_size_y(s);
  163. window_choose_redraw_screen(wp);
  164. break;
  165. }
  166. data->selected--;
  167. if (data->selected < data->top)
  168. window_choose_scroll_up(wp);
  169. else {
  170. screen_write_start(&ctx, wp, NULL);
  171. window_choose_write_line(
  172. wp, &ctx, data->selected - data->top);
  173. window_choose_write_line(
  174. wp, &ctx, data->selected + 1 - data->top);
  175. screen_write_stop(&ctx);
  176. }
  177. break;
  178. case MODEKEYCHOICE_DOWN:
  179. if (items == 0)
  180. break;
  181. if (data->selected == items - 1) {
  182. data->selected = 0;
  183. data->top = 0;
  184. window_choose_redraw_screen(wp);
  185. break;
  186. }
  187. data->selected++;
  188. if (data->selected < data->top + screen_size_y(s)) {
  189. screen_write_start(&ctx, wp, NULL);
  190. window_choose_write_line(
  191. wp, &ctx, data->selected - data->top);
  192. window_choose_write_line(
  193. wp, &ctx, data->selected - 1 - data->top);
  194. screen_write_stop(&ctx);
  195. } else
  196. window_choose_scroll_down(wp);
  197. break;
  198. case MODEKEYCHOICE_SCROLLUP:
  199. if (items == 0 || data->top == 0)
  200. break;
  201. if (data->selected == data->top + screen_size_y(s) - 1) {
  202. data->selected--;
  203. window_choose_scroll_up(wp);
  204. screen_write_start(&ctx, wp, NULL);
  205. window_choose_write_line(
  206. wp, &ctx, screen_size_y(s) - 1);
  207. screen_write_stop(&ctx);
  208. } else
  209. window_choose_scroll_up(wp);
  210. break;
  211. case MODEKEYCHOICE_SCROLLDOWN:
  212. if (items == 0 ||
  213. data->top + screen_size_y(&data->screen) >= items)
  214. break;
  215. if (data->selected == data->top) {
  216. data->selected++;
  217. window_choose_scroll_down(wp);
  218. screen_write_start(&ctx, wp, NULL);
  219. window_choose_write_line(wp, &ctx, 0);
  220. screen_write_stop(&ctx);
  221. } else
  222. window_choose_scroll_down(wp);
  223. break;
  224. case MODEKEYCHOICE_PAGEUP:
  225. if (data->selected < screen_size_y(s)) {
  226. data->selected = 0;
  227. data->top = 0;
  228. } else {
  229. data->selected -= screen_size_y(s);
  230. if (data->top < screen_size_y(s))
  231. data->top = 0;
  232. else
  233. data->top -= screen_size_y(s);
  234. }
  235. window_choose_redraw_screen(wp);
  236. break;
  237. case MODEKEYCHOICE_PAGEDOWN:
  238. data->selected += screen_size_y(s);
  239. if (data->selected > items - 1)
  240. data->selected = items - 1;
  241. data->top += screen_size_y(s);
  242. if (screen_size_y(s) < items) {
  243. if (data->top + screen_size_y(s) > items)
  244. data->top = items - screen_size_y(s);
  245. } else
  246. data->top = 0;
  247. if (data->selected < data->top)
  248. data->top = data->selected;
  249. window_choose_redraw_screen(wp);
  250. break;
  251. default:
  252. idx = window_choose_index_key(data, key);
  253. if (idx < 0 || (u_int) idx >= ARRAY_LENGTH(&data->list))
  254. break;
  255. data->selected = idx;
  256. item = &ARRAY_ITEM(&data->list, data->selected);
  257. data->callbackfn(data->data, item->idx);
  258. window_pane_reset_mode(wp);
  259. break;
  260. }
  261. }
  262. /* ARGSUSED */
  263. void
  264. window_choose_mouse(
  265. struct window_pane *wp, unused struct client *c, struct mouse_event *m)
  266. {
  267. struct window_choose_mode_data *data = wp->modedata;
  268. struct screen *s = &data->screen;
  269. struct window_choose_mode_item *item;
  270. u_int idx;
  271. if ((m->b & 3) == 3)
  272. return;
  273. if (m->x >= screen_size_x(s))
  274. return;
  275. if (m->y >= screen_size_y(s))
  276. return;
  277. idx = data->top + m->y;
  278. if (idx >= ARRAY_LENGTH(&data->list))
  279. return;
  280. data->selected = idx;
  281. item = &ARRAY_ITEM(&data->list, data->selected);
  282. data->callbackfn(data->data, item->idx);
  283. window_pane_reset_mode(wp);
  284. }
  285. void
  286. window_choose_write_line(
  287. struct window_pane *wp, struct screen_write_ctx *ctx, u_int py)
  288. {
  289. struct window_choose_mode_data *data = wp->modedata;
  290. struct window_choose_mode_item *item;
  291. struct options *oo = &wp->window->options;
  292. struct screen *s = &data->screen;
  293. struct grid_cell gc;
  294. int utf8flag, key;
  295. if (data->callbackfn == NULL)
  296. fatalx("called before callback assigned");
  297. utf8flag = options_get_number(&wp->window->options, "utf8");
  298. memcpy(&gc, &grid_default_cell, sizeof gc);
  299. if (data->selected == data->top + py) {
  300. colour_set_fg(&gc, options_get_number(oo, "mode-fg"));
  301. colour_set_bg(&gc, options_get_number(oo, "mode-bg"));
  302. gc.attr |= options_get_number(oo, "mode-attr");
  303. }
  304. screen_write_cursormove(ctx, 0, py);
  305. if (data->top + py < ARRAY_LENGTH(&data->list)) {
  306. item = &ARRAY_ITEM(&data->list, data->top + py);
  307. key = window_choose_key_index(data, data->top + py);
  308. if (key != -1) {
  309. screen_write_nputs(ctx, screen_size_x(s) - 1,
  310. &gc, utf8flag, "(%c) %s", key, item->name);
  311. } else {
  312. screen_write_nputs(ctx, screen_size_x(s) - 1,
  313. &gc, utf8flag, " %s", item->name);
  314. }
  315. }
  316. while (s->cx < screen_size_x(s))
  317. screen_write_putc(ctx, &gc, ' ');
  318. }
  319. int
  320. window_choose_key_index(struct window_choose_mode_data *data, u_int idx)
  321. {
  322. static const char keys[] = "0123456789abcdefghijklmnopqrstuvwxyz";
  323. const char *ptr;
  324. int mkey;
  325. for (ptr = keys; *ptr != '\0'; ptr++) {
  326. mkey = mode_key_lookup(&data->mdata, *ptr);
  327. if (mkey != MODEKEY_NONE && mkey != MODEKEY_OTHER)
  328. continue;
  329. if (idx-- == 0)
  330. return (*ptr);
  331. }
  332. return (-1);
  333. }
  334. int
  335. window_choose_index_key(struct window_choose_mode_data *data, int key)
  336. {
  337. static const char keys[] = "0123456789abcdefghijklmnopqrstuvwxyz";
  338. const char *ptr;
  339. int mkey;
  340. u_int idx = 0;
  341. for (ptr = keys; *ptr != '\0'; ptr++) {
  342. mkey = mode_key_lookup(&data->mdata, *ptr);
  343. if (mkey != MODEKEY_NONE && mkey != MODEKEY_OTHER)
  344. continue;
  345. if (key == *ptr)
  346. return (idx);
  347. idx++;
  348. }
  349. return (-1);
  350. }
  351. void
  352. window_choose_redraw_screen(struct window_pane *wp)
  353. {
  354. struct window_choose_mode_data *data = wp->modedata;
  355. struct screen *s = &data->screen;
  356. struct screen_write_ctx ctx;
  357. u_int i;
  358. screen_write_start(&ctx, wp, NULL);
  359. for (i = 0; i < screen_size_y(s); i++)
  360. window_choose_write_line(wp, &ctx, i);
  361. screen_write_stop(&ctx);
  362. }
  363. void
  364. window_choose_scroll_up(struct window_pane *wp)
  365. {
  366. struct window_choose_mode_data *data = wp->modedata;
  367. struct screen_write_ctx ctx;
  368. if (data->top == 0)
  369. return;
  370. data->top--;
  371. screen_write_start(&ctx, wp, NULL);
  372. screen_write_cursormove(&ctx, 0, 0);
  373. screen_write_insertline(&ctx, 1);
  374. window_choose_write_line(wp, &ctx, 0);
  375. if (screen_size_y(&data->screen) > 1)
  376. window_choose_write_line(wp, &ctx, 1);
  377. screen_write_stop(&ctx);
  378. }
  379. void
  380. window_choose_scroll_down(struct window_pane *wp)
  381. {
  382. struct window_choose_mode_data *data = wp->modedata;
  383. struct screen *s = &data->screen;
  384. struct screen_write_ctx ctx;
  385. if (data->top >= ARRAY_LENGTH(&data->list))
  386. return;
  387. data->top++;
  388. screen_write_start(&ctx, wp, NULL);
  389. screen_write_cursormove(&ctx, 0, 0);
  390. screen_write_deleteline(&ctx, 1);
  391. window_choose_write_line(wp, &ctx, screen_size_y(s) - 1);
  392. if (screen_size_y(&data->screen) > 1)
  393. window_choose_write_line(wp, &ctx, screen_size_y(s) - 2);
  394. screen_write_stop(&ctx);
  395. }