source: viewwin.c @ b6cf72f

release-1.10release-1.7release-1.8release-1.9
Last change on this file since b6cf72f was b6cf72f, checked in by David Benjamin <davidben@mit.edu>, 14 years ago
FINALLY! Add a popless:search command That took a while.
  • Property mode set to 100644
File size: 10.8 KB
RevLine 
[7d4fbcd]1#include <string.h>
2#include "owl.h"
3
4#define BOTTOM_OFFSET 1
[9116936]5#define EMPTY_INDICATOR "~"
[7d4fbcd]6
[60c1386]7static void owl_viewwin_redraw_content(owl_window *w, WINDOW *curswin, void *user_data);
8static void owl_viewwin_redraw_status(owl_window *w, WINDOW *curswin, void *user_data);
9static void owl_viewwin_layout(owl_viewwin *v);
[61c1f19]10static void owl_viewwin_set_window(owl_viewwin *v, owl_window *w);
[0b9e607]11
[9eb38bb]12/* Create a viewwin.  'win' is an already initialized owl_window that
13 * will be used by the viewwin
[b2b0773]14 */
[9eb38bb]15owl_viewwin *owl_viewwin_new_text(owl_window *win, const char *text)
[b2b0773]16{
[9eb38bb]17  owl_viewwin *v = owl_malloc(sizeof(owl_viewwin));
18  memset(v, 0, sizeof(*v));
[7d4fbcd]19  owl_fmtext_init_null(&(v->fmtext));
20  if (text) {
21    owl_fmtext_append_normal(&(v->fmtext), text);
[4083c49]22    if (text[0] != '\0' && text[strlen(text) - 1] != '\n') {
[7d4fbcd]23      owl_fmtext_append_normal(&(v->fmtext), "\n");
24    }
25    v->textlines=owl_fmtext_num_lines(&(v->fmtext));
26  }
27  v->topline=0;
28  v->rightshift=0;
[afbf668]29  v->onclose_hook = NULL;
[68f63a2]30
31  owl_viewwin_set_window(v, win);
[9eb38bb]32  return v;
[afbf668]33}
34
[9eb38bb]35/* Create a viewwin.  'win' is an already initialized owl_window that
36 * will be used by the viewwin
[b2b0773]37 */
[9eb38bb]38owl_viewwin *owl_viewwin_new_fmtext(owl_window *win, const owl_fmtext *fmtext)
[b2b0773]39{
[dd6af02]40  char *text;
[9eb38bb]41  owl_viewwin *v = owl_malloc(sizeof(owl_viewwin));
42  memset(v, 0, sizeof(*v));
[dd6af02]43
[7d4fbcd]44  owl_fmtext_copy(&(v->fmtext), fmtext);
[dd6af02]45  text = owl_fmtext_print_plain(fmtext);
46  if (text[0] != '\0' && text[strlen(text) - 1] != '\n') {
47      owl_fmtext_append_normal(&(v->fmtext), "\n");
48  }
[3e55268]49  owl_free(text);
[7d4fbcd]50  v->textlines=owl_fmtext_num_lines(&(v->fmtext));
51  v->topline=0;
52  v->rightshift=0;
[68f63a2]53
54  owl_viewwin_set_window(v, win);
[9eb38bb]55  return v;
[7d4fbcd]56}
57
[61c1f19]58static void owl_viewwin_set_window(owl_viewwin *v, owl_window *w)
[b2b0773]59{
[60c1386]60  v->window = g_object_ref(w);
61  v->content = owl_window_new(v->window);
62  v->status = owl_window_new(v->window);
[09ceee3]63  v->cmdwin = NULL;
[60c1386]64
65  v->sig_content_redraw_id =
66    g_signal_connect(v->content, "redraw", G_CALLBACK(owl_viewwin_redraw_content), v);
67  v->sig_status_redraw_id =
68    g_signal_connect(v->status, "redraw", G_CALLBACK(owl_viewwin_redraw_status), v);
69  v->sig_resize_id =
70    g_signal_connect_swapped(v->window, "resized", G_CALLBACK(owl_viewwin_layout), v);
71  owl_viewwin_layout(v);
72
73  owl_window_show(v->content);
74  owl_window_show(v->status);
[7d4fbcd]75}
76
[afbf668]77void owl_viewwin_set_onclose_hook(owl_viewwin *v, void (*onclose_hook) (owl_viewwin *vwin, void *data), void *onclose_hook_data) {
78  v->onclose_hook = onclose_hook;
79  v->onclose_hook_data = onclose_hook_data;
80}
81
[60c1386]82static void owl_viewwin_layout(owl_viewwin *v)
83{
84  int lines, cols;
85  owl_window_get_position(v->window, &lines, &cols, NULL, NULL);
86  owl_window_set_position(v->content, lines - BOTTOM_OFFSET, cols, 0, 0);
[09ceee3]87  /* Only one of these will be visible at a time: */
[60c1386]88  owl_window_set_position(v->status, BOTTOM_OFFSET, cols, lines - BOTTOM_OFFSET, 0);
[09ceee3]89  if (v->cmdwin)
90    owl_window_set_position(v->cmdwin, BOTTOM_OFFSET, cols, lines - BOTTOM_OFFSET, 0);
[60c1386]91}
92
[b2b0773]93/* regenerate text on the curses window. */
[60c1386]94static void owl_viewwin_redraw_content(owl_window *w, WINDOW *curswin, void *user_data)
[b2b0773]95{
[7d4fbcd]96  owl_fmtext fm1, fm2;
[68f63a2]97  owl_viewwin *v = user_data;
[c447d9c]98  int winlines, wincols;
[9116936]99  int y;
[c447d9c]100
101  owl_window_get_position(w, &winlines, &wincols, 0, 0);
[60c1386]102
[68f63a2]103  werase(curswin);
104  wmove(curswin, 0, 0);
[7d4fbcd]105
[af2ca19]106  owl_fmtext_init_null(&fm1);
107  owl_fmtext_init_null(&fm2);
[60c1386]108
109  owl_fmtext_truncate_lines(&(v->fmtext), v->topline, winlines, &fm1);
[c447d9c]110  owl_fmtext_truncate_cols(&fm1, v->rightshift, wincols-1+v->rightshift, &fm2);
[7d4fbcd]111
[449c682]112  owl_fmtext_curs_waddstr(&fm2, curswin);
[7d4fbcd]113
[9116936]114  /* Fill remaining lines with tildes. */
115  y = v->textlines - v->topline;
116  wmove(curswin, y, 0);
[60c1386]117  for (; y < winlines; y++) {
[9116936]118    waddstr(curswin, EMPTY_INDICATOR);
119    waddstr(curswin, "\n");
120  }
121
[60c1386]122  owl_fmtext_cleanup(&fm1);
123  owl_fmtext_cleanup(&fm2);
124}
125
126static void owl_viewwin_redraw_status(owl_window *w, WINDOW *curswin, void *user_data)
127{
128  owl_viewwin *v = user_data;
129  int winlines, wincols;
130
131  owl_window_get_position(v->content, &winlines, &wincols, 0, 0);
132
133  werase(curswin);
134  wmove(curswin, 0, 0);
[68f63a2]135  wattrset(curswin, A_REVERSE);
[60c1386]136  if (v->textlines - v->topline > winlines) {
[68f63a2]137    waddstr(curswin, "--More-- (Space to see more, 'q' to quit)");
[7d4fbcd]138  } else {
[68f63a2]139    waddstr(curswin, "--End-- (Press 'q' to quit)");
[7d4fbcd]140  }
[68f63a2]141  wattroff(curswin, A_REVERSE);
[7d4fbcd]142}
143
[b6cf72f]144char *owl_viewwin_command_search(owl_viewwin *v, int argc, const char *const *argv, const char *buff)
145{
146  int direction, mode;
147  const char *buffstart;
148
149  direction=OWL_DIRECTION_DOWNWARDS;
150  buffstart=skiptokens(buff, 1);
151  if (argc>1 && !strcmp(argv[1], "-r")) {
152    direction=OWL_DIRECTION_UPWARDS;
153    buffstart=skiptokens(buff, 2);
154  }
155
156  if (argc==1 || (argc==2 && !strcmp(argv[1], "-r"))) {
157    mode = OWL_SEARCH_CONTINUE;
158  } else {
159    owl_function_set_search(buffstart);
160    mode = OWL_SEARCH_MATCH_CURRENT;
161  }
162
163  if (!owl_viewwin_search(v, owl_global_get_search_re(&g), mode, direction))
164    owl_function_error("No more matches");
165  return NULL;
166}
167
[09ceee3]168char *owl_viewwin_start_command(owl_viewwin *v, int argc, const char *const *argv, const char *buff)
169{
170  owl_editwin *tw;
171  owl_context *ctx;
172
173  buff = skiptokens(buff, 1);
174
175  tw = owl_viewwin_set_typwin_active(v, owl_global_get_cmd_history(&g));
176  owl_editwin_set_locktext(tw, ":");
177
178  owl_editwin_insert_string(tw, buff);
179
180  ctx = owl_editcontext_new(OWL_CTX_EDITLINE, tw, "editline");
181  ctx->deactivate_cb = owl_viewwin_deactivate_editcontext;
182  ctx->cbdata = v;
183  owl_global_push_context_obj(&g, ctx);
184  owl_editwin_set_callback(tw, owl_callback_command);
185
186  return NULL;
187}
188
189void owl_viewwin_deactivate_editcontext(owl_context *ctx) {
190  owl_viewwin *v = ctx->cbdata;
191  owl_viewwin_set_typwin_inactive(v);
192}
193
194owl_editwin *owl_viewwin_set_typwin_active(owl_viewwin *v, owl_history *hist) {
195  int lines, cols;
196  if (v->cmdwin || v->cmdline)
197    return NULL;
198  /* Create the command line. */
199  v->cmdwin = owl_window_new(v->window);
200  owl_viewwin_layout(v);
201  owl_window_get_position(v->cmdwin, &lines, &cols, NULL, NULL);
202  v->cmdline = owl_editwin_new(v->cmdwin, lines, cols, OWL_EDITWIN_STYLE_ONELINE, hist);
203  /* Swap out the bottom window. */
204  owl_window_hide(v->status);
205  owl_window_show(v->cmdwin);
206  return v->cmdline;
207}
208
209void owl_viewwin_set_typwin_inactive(owl_viewwin *v) {
210  if (v->cmdwin) {
211    /* Swap out the bottom window. */
212    owl_window_hide(v->cmdwin);
213    owl_window_show(v->status);
214    /* Destroy the window itself. */
215    owl_window_unlink(v->cmdwin);
216    g_object_unref(v->cmdwin);
217    v->cmdwin = NULL;
218  }
219  if (v->cmdline) {
220    owl_editwin_unref(v->cmdline);
221    v->cmdline = NULL;
222  }
223}
224
[2640b63]225void owl_viewwin_append_text(owl_viewwin *v, const char *text) {
226    owl_fmtext_append_normal(&(v->fmtext), text);
227    v->textlines=owl_fmtext_num_lines(&(v->fmtext));
228    owl_viewwin_dirty(v);
229}
230
231/* Schedule a redraw of 'v'. Exported for hooking into the search
232   string; when we have some way of listening for changes, this can be
233   removed. */
234void owl_viewwin_dirty(owl_viewwin *v)
235{
[60c1386]236  owl_window_dirty(v->content);
237  owl_window_dirty(v->status);
[2640b63]238}
239
[d574d61]240void owl_viewwin_down(owl_viewwin *v, int amount) {
[c447d9c]241  int winlines;
[60c1386]242  owl_window_get_position(v->content, &winlines, 0, 0, 0);
[fa23002]243  /* Don't scroll past the bottom. */
[60c1386]244  amount = MIN(amount, v->textlines - (v->topline + winlines));
[fa23002]245  /* But if we're already past the bottom, don't back up either. */
246  if (amount > 0) {
247    v->topline += amount;
248    owl_viewwin_dirty(v);
[7d4fbcd]249  }
250}
251
[d574d61]252void owl_viewwin_up(owl_viewwin *v, int amount)
253{
254  v->topline -= amount;
255  if (v->topline<0) v->topline=0;
256  owl_viewwin_dirty(v);
257}
258
259void owl_viewwin_pagedown(owl_viewwin *v)
[b2b0773]260{
[c447d9c]261  int winlines;
[60c1386]262  owl_window_get_position(v->content, &winlines, 0, 0, 0);
263  owl_viewwin_down(v, winlines);
[d574d61]264}
265
266void owl_viewwin_linedown(owl_viewwin *v)
267{
268  owl_viewwin_down(v, 1);
[7d4fbcd]269}
270
[b2b0773]271void owl_viewwin_pageup(owl_viewwin *v)
272{
[c447d9c]273  int winlines;
[60c1386]274  owl_window_get_position(v->content, &winlines, 0, 0, 0);
275  owl_viewwin_up(v, winlines);
[7d4fbcd]276}
277
[b2b0773]278void owl_viewwin_lineup(owl_viewwin *v)
279{
[d574d61]280  owl_viewwin_up(v, 1);
[7d4fbcd]281}
282
[b2b0773]283void owl_viewwin_right(owl_viewwin *v, int n)
284{
[7d4fbcd]285  v->rightshift+=n;
[5b68c05]286  owl_viewwin_dirty(v);
[7d4fbcd]287}
288
[b2b0773]289void owl_viewwin_left(owl_viewwin *v, int n)
290{
[7d4fbcd]291  v->rightshift-=n;
292  if (v->rightshift<0) v->rightshift=0;
[5b68c05]293  owl_viewwin_dirty(v);
[7d4fbcd]294}
295
[b2b0773]296void owl_viewwin_top(owl_viewwin *v)
297{
[7d4fbcd]298  v->topline=0;
299  v->rightshift=0;
[5b68c05]300  owl_viewwin_dirty(v);
[7d4fbcd]301}
302
[b2b0773]303void owl_viewwin_bottom(owl_viewwin *v)
304{
[c447d9c]305  int winlines;
[60c1386]306  owl_window_get_position(v->content, &winlines, 0, 0, 0);
307  v->topline = v->textlines - winlines;
[5b68c05]308  owl_viewwin_dirty(v);
[7d4fbcd]309}
310
[6a35938]311/* This is a bit of a hack, because regexec doesn't have an 'n'
312 * version. */
313static int _re_memcompare(const owl_regex *re, const char *string, int start, int end)
314{
315  int ans;
316  char *tmp = g_strndup(string + start, end - start);
317  ans = owl_regex_compare(re, tmp, NULL, NULL);
318  g_free(tmp);
319  return !ans;
320}
321
[b8742ba]322/* Scroll in 'direction' to the next line containing 're' in 'v',
323 * starting from the current line. Returns 0 if no occurrence is
324 * found.
325 *
[9c1e61d4]326 * If mode is OWL_SEARCH_MATCH_CURRENT then stay on the current line
327 * if it matches.
[b8742ba]328 */
329int owl_viewwin_search(owl_viewwin *v, const owl_regex *re, int mode, int direction)
330{
331  int start, end, offset;
[6a35938]332  int lineend, linestart;
333  const char *buf, *linestartp;
[b8742ba]334  owl_fmtext_line_extents(&v->fmtext, v->topline, &start, &end);
335  if (direction == OWL_DIRECTION_DOWNWARDS) {
[9c1e61d4]336    offset = owl_fmtext_search(&v->fmtext, re,
337                               (mode == OWL_SEARCH_CONTINUE) ? end : start);
[b8742ba]338    if (offset < 0)
339      return 0;
340    v->topline = owl_fmtext_line_number(&v->fmtext, offset);
341    owl_viewwin_dirty(v);
342    return 1;
343  } else {
[6a35938]344    /* TODO: This is a hack. Really, we should have an owl_fmlines or
345     * something containing an array of owl_fmtext split into
346     * lines. Also, it cannot handle multi-line regex, if we ever care about
347     * them. */
348    buf = owl_fmtext_get_text(&v->fmtext);
[9c1e61d4]349    lineend = (mode == OWL_SEARCH_CONTINUE) ? start : end;
[6a35938]350    while (lineend > 0) {
351      linestartp = memrchr(buf, '\n', lineend - 1);
352      linestart = linestartp ? linestartp - buf + 1 : 0;
353      if (_re_memcompare(re, buf, linestart, lineend)) {
354        v->topline = owl_fmtext_line_number(&v->fmtext, linestart);
355        owl_viewwin_dirty(v);
356        return 1;
357      }
358      lineend = linestart;
359    }
360    return 0;
[b8742ba]361  }
362}
363
[9eb38bb]364void owl_viewwin_delete(owl_viewwin *v)
[b2b0773]365{
[afbf668]366  if (v->onclose_hook) {
367    v->onclose_hook(v, v->onclose_hook_data);
[8721756]368    v->onclose_hook = NULL;
369    v->onclose_hook_data = NULL;
[afbf668]370  }
[09ceee3]371  owl_viewwin_set_typwin_inactive(v);
[60c1386]372  /* TODO: This is far too tedious. owl_viewwin should own v->window
373   * and just unlink it in one go. Signals should also go away for
374   * free. */
375  g_signal_handler_disconnect(v->window, v->sig_resize_id);
376  g_signal_handler_disconnect(v->content, v->sig_content_redraw_id);
377  g_signal_handler_disconnect(v->status, v->sig_status_redraw_id);
378  owl_window_unlink(v->content);
379  owl_window_unlink(v->status);
380  g_object_unref(v->window);
381  g_object_unref(v->content);
382  g_object_unref(v->status);
[7ab0020]383  owl_fmtext_cleanup(&(v->fmtext));
[9eb38bb]384  owl_free(v);
[7d4fbcd]385}
Note: See TracBrowser for help on using the repository browser.