4 * Copyright (c) 2007 Nicholas Marriott <nicholas.marriott@gmail.com>
6 * Permission to use, copy, modify, and distribute this software for any
7 * purpose with or without fee is hereby granted, provided that the above
8 * copyright notice and this permission notice appear in all copies.
10 * THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES
11 * WITH REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF
12 * MERCHANTABILITY AND FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR
13 * ANY SPECIAL, DIRECT, INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES
14 * WHATSOEVER RESULTING FROM LOSS OF MIND, USE, DATA OR PROFITS, WHETHER
15 * IN AN ACTION OF CONTRACT, NEGLIGENCE OR OTHER TORTIOUS ACTION, ARISING
16 * OUT OF OR IN CONNECTION WITH THE USE OR PERFORMANCE OF THIS SOFTWARE.
19 #include <sys/types.h>
29 struct window_copy_mode_data
;
31 static const char *window_copy_key_table(struct window_mode_entry
*);
32 static void window_copy_command(struct window_mode_entry
*, struct client
*,
33 struct session
*, struct winlink
*, struct args
*,
34 struct mouse_event
*);
35 static struct screen
*window_copy_init(struct window_mode_entry
*,
36 struct cmd_find_state
*, struct args
*);
37 static struct screen
*window_copy_view_init(struct window_mode_entry
*,
38 struct cmd_find_state
*, struct args
*);
39 static void window_copy_free(struct window_mode_entry
*);
40 static void window_copy_resize(struct window_mode_entry
*, u_int
, u_int
);
41 static void window_copy_formats(struct window_mode_entry
*,
42 struct format_tree
*);
43 static void window_copy_pageup1(struct window_mode_entry
*, int);
44 static int window_copy_pagedown(struct window_mode_entry
*, int, int);
45 static void window_copy_next_paragraph(struct window_mode_entry
*);
46 static void window_copy_previous_paragraph(struct window_mode_entry
*);
47 static void window_copy_redraw_selection(struct window_mode_entry
*, u_int
);
48 static void window_copy_redraw_lines(struct window_mode_entry
*, u_int
,
50 static void window_copy_redraw_screen(struct window_mode_entry
*);
51 static void window_copy_write_line(struct window_mode_entry
*,
52 struct screen_write_ctx
*, u_int
);
53 static void window_copy_write_lines(struct window_mode_entry
*,
54 struct screen_write_ctx
*, u_int
, u_int
);
55 static char *window_copy_match_at_cursor(struct window_copy_mode_data
*);
56 static void window_copy_scroll_to(struct window_mode_entry
*, u_int
, u_int
,
58 static int window_copy_search_compare(struct grid
*, u_int
, u_int
,
59 struct grid
*, u_int
, int);
60 static int window_copy_search_lr(struct grid
*, struct grid
*, u_int
*,
61 u_int
, u_int
, u_int
, int);
62 static int window_copy_search_rl(struct grid
*, struct grid
*, u_int
*,
63 u_int
, u_int
, u_int
, int);
64 static int window_copy_last_regex(struct grid
*, u_int
, u_int
, u_int
,
65 u_int
, u_int
*, u_int
*, const char *, const regex_t
*,
67 static int window_copy_search_mark_at(struct window_copy_mode_data
*,
68 u_int
, u_int
, u_int
*);
69 static char *window_copy_stringify(struct grid
*, u_int
, u_int
, u_int
,
71 static void window_copy_cstrtocellpos(struct grid
*, u_int
, u_int
*,
72 u_int
*, const char *);
73 static int window_copy_search_marks(struct window_mode_entry
*,
74 struct screen
*, int, int);
75 static void window_copy_clear_marks(struct window_mode_entry
*);
76 static int window_copy_is_lowercase(const char *);
77 static void window_copy_search_back_overlap(struct grid
*, regex_t
*,
78 u_int
*, u_int
*, u_int
*, u_int
);
79 static int window_copy_search_jump(struct window_mode_entry
*,
80 struct grid
*, struct grid
*, u_int
, u_int
, u_int
, int, int,
82 static int window_copy_search(struct window_mode_entry
*, int, int);
83 static int window_copy_search_up(struct window_mode_entry
*, int);
84 static int window_copy_search_down(struct window_mode_entry
*, int);
85 static void window_copy_goto_line(struct window_mode_entry
*, const char *);
86 static void window_copy_update_cursor(struct window_mode_entry
*, u_int
,
88 static void window_copy_start_selection(struct window_mode_entry
*);
89 static int window_copy_adjust_selection(struct window_mode_entry
*,
91 static int window_copy_set_selection(struct window_mode_entry
*, int, int);
92 static int window_copy_update_selection(struct window_mode_entry
*, int,
94 static void window_copy_synchronize_cursor(struct window_mode_entry
*, int);
95 static void *window_copy_get_selection(struct window_mode_entry
*, size_t *);
96 static void window_copy_copy_buffer(struct window_mode_entry
*,
97 const char *, void *, size_t);
98 static void window_copy_pipe(struct window_mode_entry
*,
99 struct session
*, const char *);
100 static void window_copy_copy_pipe(struct window_mode_entry
*,
101 struct session
*, const char *, const char *);
102 static void window_copy_copy_selection(struct window_mode_entry
*,
104 static void window_copy_append_selection(struct window_mode_entry
*);
105 static void window_copy_clear_selection(struct window_mode_entry
*);
106 static void window_copy_copy_line(struct window_mode_entry
*, char **,
107 size_t *, u_int
, u_int
, u_int
);
108 static int window_copy_in_set(struct window_mode_entry
*, u_int
, u_int
,
110 static u_int
window_copy_find_length(struct window_mode_entry
*, u_int
);
111 static void window_copy_cursor_start_of_line(struct window_mode_entry
*);
112 static void window_copy_cursor_back_to_indentation(
113 struct window_mode_entry
*);
114 static void window_copy_cursor_end_of_line(struct window_mode_entry
*);
115 static void window_copy_other_end(struct window_mode_entry
*);
116 static void window_copy_cursor_left(struct window_mode_entry
*);
117 static void window_copy_cursor_right(struct window_mode_entry
*, int);
118 static void window_copy_cursor_up(struct window_mode_entry
*, int);
119 static void window_copy_cursor_down(struct window_mode_entry
*, int);
120 static void window_copy_cursor_jump(struct window_mode_entry
*);
121 static void window_copy_cursor_jump_back(struct window_mode_entry
*);
122 static void window_copy_cursor_jump_to(struct window_mode_entry
*);
123 static void window_copy_cursor_jump_to_back(struct window_mode_entry
*);
124 static void window_copy_cursor_next_word(struct window_mode_entry
*,
126 static void window_copy_cursor_next_word_end_pos(struct window_mode_entry
*,
127 const char *, u_int
*, u_int
*);
128 static void window_copy_cursor_next_word_end(struct window_mode_entry
*,
130 static void window_copy_cursor_previous_word_pos(struct window_mode_entry
*,
131 const char *, u_int
*, u_int
*);
132 static void window_copy_cursor_previous_word(struct window_mode_entry
*,
134 static void window_copy_cursor_prompt(struct window_mode_entry
*, int,
136 static void window_copy_scroll_up(struct window_mode_entry
*, u_int
);
137 static void window_copy_scroll_down(struct window_mode_entry
*, u_int
);
138 static void window_copy_rectangle_set(struct window_mode_entry
*, int);
139 static void window_copy_move_mouse(struct mouse_event
*);
140 static void window_copy_drag_update(struct client
*, struct mouse_event
*);
141 static void window_copy_drag_release(struct client
*, struct mouse_event
*);
142 static void window_copy_jump_to_mark(struct window_mode_entry
*);
143 static void window_copy_acquire_cursor_up(struct window_mode_entry
*,
144 u_int
, u_int
, u_int
, u_int
, u_int
);
145 static void window_copy_acquire_cursor_down(struct window_mode_entry
*,
146 u_int
, u_int
, u_int
, u_int
, u_int
, u_int
, int);
148 const struct window_mode window_copy_mode
= {
151 .init
= window_copy_init
,
152 .free
= window_copy_free
,
153 .resize
= window_copy_resize
,
154 .key_table
= window_copy_key_table
,
155 .command
= window_copy_command
,
156 .formats
= window_copy_formats
,
159 const struct window_mode window_view_mode
= {
162 .init
= window_copy_view_init
,
163 .free
= window_copy_free
,
164 .resize
= window_copy_resize
,
165 .key_table
= window_copy_key_table
,
166 .command
= window_copy_command
,
167 .formats
= window_copy_formats
,
172 WINDOW_COPY_SEARCHUP
,
173 WINDOW_COPY_SEARCHDOWN
,
174 WINDOW_COPY_JUMPFORWARD
,
175 WINDOW_COPY_JUMPBACKWARD
,
176 WINDOW_COPY_JUMPTOFORWARD
,
177 WINDOW_COPY_JUMPTOBACKWARD
,
181 WINDOW_COPY_REL_POS_ABOVE
,
182 WINDOW_COPY_REL_POS_ON_SCREEN
,
183 WINDOW_COPY_REL_POS_BELOW
,
186 enum window_copy_cmd_action
{
187 WINDOW_COPY_CMD_NOTHING
,
188 WINDOW_COPY_CMD_REDRAW
,
189 WINDOW_COPY_CMD_CANCEL
,
192 enum window_copy_cmd_clear
{
193 WINDOW_COPY_CMD_CLEAR_ALWAYS
,
194 WINDOW_COPY_CMD_CLEAR_NEVER
,
195 WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
198 struct window_copy_cmd_state
{
199 struct window_mode_entry
*wme
;
201 struct mouse_event
*m
;
209 * Copy mode's visible screen (the "screen" field) is filled from one of two
210 * sources: the original contents of the pane (used when we actually enter via
211 * the "copy-mode" command, to copy the contents of the current pane), or else
212 * a series of lines containing the output from an output-writing tmux command
213 * (such as any of the "show-*" or "list-*" commands).
215 * In either case, the full content of the copy-mode grid is pointed at by the
216 * "backing" field, and is copied into "screen" as needed (that is, when
217 * scrolling occurs). When copy-mode is backed by a pane, backing points
218 * directly at that pane's screen structure (&wp->base); when backed by a list
219 * of output-lines from a command, it points at a newly-allocated screen
220 * structure (which is deallocated when the mode ends).
222 struct window_copy_mode_data
{
223 struct screen screen
;
225 struct screen
*backing
;
226 int backing_written
; /* backing display started */
227 struct screen
*writing
;
228 struct input_ctx
*ictx
;
230 int viewmode
; /* view mode entered */
232 u_int oy
; /* number of lines scrolled up */
234 u_int selx
; /* beginning of selection */
237 u_int endselx
; /* end of selection */
241 CURSORDRAG_NONE
, /* selection is independent of cursor */
242 CURSORDRAG_ENDSEL
, /* end is synchronized with cursor */
243 CURSORDRAG_SEL
, /* start is synchronized with cursor */
251 } lineflag
; /* line selection mode */
252 int rectflag
; /* in rectangle copy mode? */
253 int scroll_exit
; /* exit on scroll to end? */
254 int hide_position
; /* hide position marker */
257 SEL_CHAR
, /* select one char at a time */
258 SEL_WORD
, /* select one word at a time */
259 SEL_LINE
, /* select one line at a time */
262 const char *separators
; /* word separators */
264 u_int dx
; /* drag start position */
267 u_int selrx
; /* selection reset positions */
275 u_int lastcx
; /* position in last line w/ content */
276 u_int lastsx
; /* size of last line w/ content */
278 u_int mx
; /* mark position */
295 int timeout
; /* search has timed out */
296 #define WINDOW_COPY_SEARCH_TIMEOUT 10000
297 #define WINDOW_COPY_SEARCH_ALL_TIMEOUT 200
300 struct utf8_data
*jumpchar
;
302 struct event dragtimer
;
303 #define WINDOW_COPY_DRAG_REPEAT_TIME 50000
307 window_copy_scroll_timer(__unused
int fd
, __unused
short events
, void *arg
)
309 struct window_mode_entry
*wme
= arg
;
310 struct window_pane
*wp
= wme
->wp
;
311 struct window_copy_mode_data
*data
= wme
->data
;
312 struct timeval tv
= {
313 .tv_usec
= WINDOW_COPY_DRAG_REPEAT_TIME
316 evtimer_del(&data
->dragtimer
);
318 if (TAILQ_FIRST(&wp
->modes
) != wme
)
322 evtimer_add(&data
->dragtimer
, &tv
);
323 window_copy_cursor_up(wme
, 1);
324 } else if (data
->cy
== screen_size_y(&data
->screen
) - 1) {
325 evtimer_add(&data
->dragtimer
, &tv
);
326 window_copy_cursor_down(wme
, 1);
330 static struct screen
*
331 window_copy_clone_screen(struct screen
*src
, struct screen
*hint
, u_int
*cx
,
335 const struct grid_line
*gl
;
339 dst
= xcalloc(1, sizeof *dst
);
341 sy
= screen_hsize(src
) + screen_size_y(src
);
343 while (sy
> screen_hsize(src
)) {
344 gl
= grid_peek_line(src
->grid
, sy
- 1);
345 if (gl
->cellused
!= 0)
350 log_debug("%s: target screen is %ux%u, source %ux%u", __func__
,
351 screen_size_x(src
), sy
, screen_size_x(hint
),
352 screen_hsize(src
) + screen_size_y(src
));
353 screen_init(dst
, screen_size_x(src
), sy
, screen_hlimit(src
));
356 * Ensure history is on for the backing grid so lines are not deleted
359 dst
->grid
->flags
|= GRID_HISTORY
;
360 grid_duplicate_lines(dst
->grid
, 0, src
->grid
, 0, sy
);
362 dst
->grid
->sy
= sy
- screen_hsize(src
);
363 dst
->grid
->hsize
= screen_hsize(src
);
364 dst
->grid
->hscrolled
= src
->grid
->hscrolled
;
365 if (src
->cy
> dst
->grid
->sy
- 1) {
367 dst
->cy
= dst
->grid
->sy
- 1;
373 if (cx
!= NULL
&& cy
!= NULL
) {
375 *cy
= screen_hsize(dst
) + dst
->cy
;
376 reflow
= (screen_size_x(hint
) != screen_size_x(dst
));
381 grid_wrap_position(dst
->grid
, *cx
, *cy
, &wx
, &wy
);
382 screen_resize_cursor(dst
, screen_size_x(hint
), screen_size_y(hint
), 1,
385 grid_unwrap_position(dst
->grid
, cx
, cy
, wx
, wy
);
390 static struct window_copy_mode_data
*
391 window_copy_common_init(struct window_mode_entry
*wme
)
393 struct window_pane
*wp
= wme
->wp
;
394 struct window_copy_mode_data
*data
;
395 struct screen
*base
= &wp
->base
;
397 wme
->data
= data
= xcalloc(1, sizeof *data
);
399 data
->cursordrag
= CURSORDRAG_NONE
;
400 data
->lineflag
= LINE_SEL_NONE
;
401 data
->selflag
= SEL_CHAR
;
403 if (wp
->searchstr
!= NULL
) {
404 data
->searchtype
= WINDOW_COPY_SEARCHUP
;
405 data
->searchregex
= wp
->searchregex
;
406 data
->searchstr
= xstrdup(wp
->searchstr
);
408 data
->searchtype
= WINDOW_COPY_OFF
;
409 data
->searchregex
= 0;
410 data
->searchstr
= NULL
;
412 data
->searchx
= data
->searchy
= data
->searcho
= -1;
415 data
->jumptype
= WINDOW_COPY_OFF
;
416 data
->jumpchar
= NULL
;
418 screen_init(&data
->screen
, screen_size_x(base
), screen_size_y(base
), 0);
419 data
->modekeys
= options_get_number(wp
->window
->options
, "mode-keys");
421 evtimer_set(&data
->dragtimer
, window_copy_scroll_timer
, wme
);
426 static struct screen
*
427 window_copy_init(struct window_mode_entry
*wme
,
428 __unused
struct cmd_find_state
*fs
, struct args
*args
)
430 struct window_pane
*wp
= wme
->swp
;
431 struct window_copy_mode_data
*data
;
432 struct screen
*base
= &wp
->base
;
433 struct screen_write_ctx ctx
;
436 data
= window_copy_common_init(wme
);
437 data
->backing
= window_copy_clone_screen(base
, &data
->screen
, &cx
, &cy
,
438 wme
->swp
!= wme
->wp
);
441 if (cy
< screen_hsize(data
->backing
)) {
443 data
->oy
= screen_hsize(data
->backing
) - cy
;
445 data
->cy
= cy
- screen_hsize(data
->backing
);
449 data
->scroll_exit
= args_has(args
, 'e');
450 data
->hide_position
= args_has(args
, 'H');
452 data
->screen
.cx
= data
->cx
;
453 data
->screen
.cy
= data
->cy
;
455 data
->my
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
458 screen_write_start(&ctx
, &data
->screen
);
459 for (i
= 0; i
< screen_size_y(&data
->screen
); i
++)
460 window_copy_write_line(wme
, &ctx
, i
);
461 screen_write_cursormove(&ctx
, data
->cx
, data
->cy
, 0);
462 screen_write_stop(&ctx
);
464 return (&data
->screen
);
467 static struct screen
*
468 window_copy_view_init(struct window_mode_entry
*wme
,
469 __unused
struct cmd_find_state
*fs
, __unused
struct args
*args
)
471 struct window_pane
*wp
= wme
->wp
;
472 struct window_copy_mode_data
*data
;
473 struct screen
*base
= &wp
->base
;
474 u_int sx
= screen_size_x(base
);
476 data
= window_copy_common_init(wme
);
479 data
->backing
= xmalloc(sizeof *data
->backing
);
480 screen_init(data
->backing
, sx
, screen_size_y(base
), UINT_MAX
);
481 data
->writing
= xmalloc(sizeof *data
->writing
);
482 screen_init(data
->writing
, sx
, screen_size_y(base
), 0);
483 data
->ictx
= input_init(NULL
, NULL
, NULL
);
485 data
->my
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
488 return (&data
->screen
);
492 window_copy_free(struct window_mode_entry
*wme
)
494 struct window_copy_mode_data
*data
= wme
->data
;
496 evtimer_del(&data
->dragtimer
);
498 free(data
->searchmark
);
499 free(data
->searchstr
);
500 free(data
->jumpchar
);
502 if (data
->writing
!= NULL
) {
503 screen_free(data
->writing
);
506 if (data
->ictx
!= NULL
)
507 input_free(data
->ictx
);
508 screen_free(data
->backing
);
511 screen_free(&data
->screen
);
516 window_copy_add(struct window_pane
*wp
, int parse
, const char *fmt
, ...)
521 window_copy_vadd(wp
, parse
, fmt
, ap
);
526 window_copy_init_ctx_cb(__unused
struct screen_write_ctx
*ctx
,
527 struct tty_ctx
*ttyctx
)
529 memcpy(&ttyctx
->defaults
, &grid_default_cell
, sizeof ttyctx
->defaults
);
530 ttyctx
->palette
= NULL
;
531 ttyctx
->redraw_cb
= NULL
;
532 ttyctx
->set_client_cb
= NULL
;
537 window_copy_vadd(struct window_pane
*wp
, int parse
, const char *fmt
, va_list ap
)
539 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
540 struct window_copy_mode_data
*data
= wme
->data
;
541 struct screen
*backing
= data
->backing
;
542 struct screen
*writing
= data
->writing
;
543 struct screen_write_ctx writing_ctx
, backing_ctx
, ctx
;
545 u_int old_hsize
, old_cy
;
546 u_int sx
= screen_size_x(backing
);
550 vasprintf(&text
, fmt
, ap
);
551 screen_write_start(&writing_ctx
, writing
);
552 screen_write_reset(&writing_ctx
);
553 input_parse_screen(data
->ictx
, writing
, window_copy_init_ctx_cb
,
554 data
, text
, strlen(text
));
558 old_hsize
= screen_hsize(data
->backing
);
559 screen_write_start(&backing_ctx
, backing
);
560 if (data
->backing_written
) {
562 * On the second or later line, do a CRLF before writing
563 * (so it's on a new line).
565 screen_write_carriagereturn(&backing_ctx
);
566 screen_write_linefeed(&backing_ctx
, 0, 8);
568 data
->backing_written
= 1;
569 old_cy
= backing
->cy
;
571 screen_write_fast_copy(&backing_ctx
, writing
, 0, 0, sx
, 1);
573 memcpy(&gc
, &grid_default_cell
, sizeof gc
);
574 screen_write_vnputs(&backing_ctx
, 0, &gc
, fmt
, ap
);
576 screen_write_stop(&backing_ctx
);
578 data
->oy
+= screen_hsize(data
->backing
) - old_hsize
;
580 screen_write_start_pane(&ctx
, wp
, &data
->screen
);
583 * If the history has changed, draw the top line.
584 * (If there's any history at all, it has changed.)
586 if (screen_hsize(data
->backing
))
587 window_copy_redraw_lines(wme
, 0, 1);
589 /* Write the new lines. */
590 window_copy_redraw_lines(wme
, old_cy
, backing
->cy
- old_cy
+ 1);
592 screen_write_stop(&ctx
);
596 window_copy_pageup(struct window_pane
*wp
, int half_page
)
598 window_copy_pageup1(TAILQ_FIRST(&wp
->modes
), half_page
);
602 window_copy_pageup1(struct window_mode_entry
*wme
, int half_page
)
604 struct window_copy_mode_data
*data
= wme
->data
;
605 struct screen
*s
= &data
->screen
;
606 u_int n
, ox
, oy
, px
, py
;
608 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
609 ox
= window_copy_find_length(wme
, oy
);
611 if (data
->cx
!= ox
) {
612 data
->lastcx
= data
->cx
;
615 data
->cx
= data
->lastcx
;
618 if (screen_size_y(s
) > 2) {
620 n
= screen_size_y(s
) / 2;
622 n
= screen_size_y(s
) - 2;
625 if (data
->oy
+ n
> screen_hsize(data
->backing
)) {
626 data
->oy
= screen_hsize(data
->backing
);
634 if (data
->screen
.sel
== NULL
|| !data
->rectflag
) {
635 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
636 px
= window_copy_find_length(wme
, py
);
637 if ((data
->cx
>= data
->lastsx
&& data
->cx
!= px
) ||
639 window_copy_cursor_end_of_line(wme
);
642 if (data
->searchmark
!= NULL
&& !data
->timeout
)
643 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
644 window_copy_update_selection(wme
, 1, 0);
645 window_copy_redraw_screen(wme
);
649 window_copy_pagedown(struct window_mode_entry
*wme
, int half_page
,
652 struct window_copy_mode_data
*data
= wme
->data
;
653 struct screen
*s
= &data
->screen
;
654 u_int n
, ox
, oy
, px
, py
;
656 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
657 ox
= window_copy_find_length(wme
, oy
);
659 if (data
->cx
!= ox
) {
660 data
->lastcx
= data
->cx
;
663 data
->cx
= data
->lastcx
;
666 if (screen_size_y(s
) > 2) {
668 n
= screen_size_y(s
) / 2;
670 n
= screen_size_y(s
) - 2;
675 if (data
->cy
+ (n
- data
->oy
) >= screen_size_y(data
->backing
))
676 data
->cy
= screen_size_y(data
->backing
) - 1;
678 data
->cy
+= n
- data
->oy
;
682 if (data
->screen
.sel
== NULL
|| !data
->rectflag
) {
683 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
684 px
= window_copy_find_length(wme
, py
);
685 if ((data
->cx
>= data
->lastsx
&& data
->cx
!= px
) ||
687 window_copy_cursor_end_of_line(wme
);
690 if (scroll_exit
&& data
->oy
== 0)
692 if (data
->searchmark
!= NULL
&& !data
->timeout
)
693 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
694 window_copy_update_selection(wme
, 1, 0);
695 window_copy_redraw_screen(wme
);
700 window_copy_previous_paragraph(struct window_mode_entry
*wme
)
702 struct window_copy_mode_data
*data
= wme
->data
;
705 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
707 while (oy
> 0 && window_copy_find_length(wme
, oy
) == 0)
710 while (oy
> 0 && window_copy_find_length(wme
, oy
) > 0)
713 window_copy_scroll_to(wme
, 0, oy
, 0);
717 window_copy_next_paragraph(struct window_mode_entry
*wme
)
719 struct window_copy_mode_data
*data
= wme
->data
;
720 struct screen
*s
= &data
->screen
;
723 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
724 maxy
= screen_hsize(data
->backing
) + screen_size_y(s
) - 1;
726 while (oy
< maxy
&& window_copy_find_length(wme
, oy
) == 0)
729 while (oy
< maxy
&& window_copy_find_length(wme
, oy
) > 0)
732 ox
= window_copy_find_length(wme
, oy
);
733 window_copy_scroll_to(wme
, ox
, oy
, 0);
737 window_copy_get_word(struct window_pane
*wp
, u_int x
, u_int y
)
739 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
740 struct window_copy_mode_data
*data
= wme
->data
;
741 struct grid
*gd
= data
->screen
.grid
;
743 return (format_grid_word(gd
, x
, gd
->hsize
+ y
));
747 window_copy_get_line(struct window_pane
*wp
, u_int y
)
749 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
750 struct window_copy_mode_data
*data
= wme
->data
;
751 struct grid
*gd
= data
->screen
.grid
;
753 return (format_grid_line(gd
, gd
->hsize
+ y
));
757 window_copy_cursor_word_cb(struct format_tree
*ft
)
759 struct window_pane
*wp
= format_get_pane(ft
);
760 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
761 struct window_copy_mode_data
*data
= wme
->data
;
763 return (window_copy_get_word(wp
, data
->cx
, data
->cy
));
767 window_copy_cursor_line_cb(struct format_tree
*ft
)
769 struct window_pane
*wp
= format_get_pane(ft
);
770 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
771 struct window_copy_mode_data
*data
= wme
->data
;
773 return (window_copy_get_line(wp
, data
->cy
));
777 window_copy_search_match_cb(struct format_tree
*ft
)
779 struct window_pane
*wp
= format_get_pane(ft
);
780 struct window_mode_entry
*wme
= TAILQ_FIRST(&wp
->modes
);
781 struct window_copy_mode_data
*data
= wme
->data
;
783 return (window_copy_match_at_cursor(data
));
787 window_copy_formats(struct window_mode_entry
*wme
, struct format_tree
*ft
)
789 struct window_copy_mode_data
*data
= wme
->data
;
791 format_add(ft
, "scroll_position", "%d", data
->oy
);
792 format_add(ft
, "rectangle_toggle", "%d", data
->rectflag
);
794 format_add(ft
, "copy_cursor_x", "%d", data
->cx
);
795 format_add(ft
, "copy_cursor_y", "%d", data
->cy
);
797 format_add(ft
, "selection_present", "%d", data
->screen
.sel
!= NULL
);
798 if (data
->screen
.sel
!= NULL
) {
799 format_add(ft
, "selection_start_x", "%d", data
->selx
);
800 format_add(ft
, "selection_start_y", "%d", data
->sely
);
801 format_add(ft
, "selection_end_x", "%d", data
->endselx
);
802 format_add(ft
, "selection_end_y", "%d", data
->endsely
);
803 format_add(ft
, "selection_active", "%d",
804 data
->cursordrag
!= CURSORDRAG_NONE
);
806 format_add(ft
, "selection_active", "%d", 0);
808 format_add(ft
, "search_present", "%d", data
->searchmark
!= NULL
);
809 format_add_cb(ft
, "search_match", window_copy_search_match_cb
);
811 format_add_cb(ft
, "copy_cursor_word", window_copy_cursor_word_cb
);
812 format_add_cb(ft
, "copy_cursor_line", window_copy_cursor_line_cb
);
816 window_copy_size_changed(struct window_mode_entry
*wme
)
818 struct window_copy_mode_data
*data
= wme
->data
;
819 struct screen
*s
= &data
->screen
;
820 struct screen_write_ctx ctx
;
821 int search
= (data
->searchmark
!= NULL
);
823 window_copy_clear_selection(wme
);
824 window_copy_clear_marks(wme
);
826 screen_write_start(&ctx
, s
);
827 window_copy_write_lines(wme
, &ctx
, 0, screen_size_y(s
));
828 screen_write_stop(&ctx
);
830 if (search
&& !data
->timeout
)
831 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 0);
832 data
->searchx
= data
->cx
;
833 data
->searchy
= data
->cy
;
834 data
->searcho
= data
->oy
;
838 window_copy_resize(struct window_mode_entry
*wme
, u_int sx
, u_int sy
)
840 struct window_copy_mode_data
*data
= wme
->data
;
841 struct screen
*s
= &data
->screen
;
842 struct grid
*gd
= data
->backing
->grid
;
843 u_int cx
, cy
, wx
, wy
;
846 screen_resize(s
, sx
, sy
, 0);
848 cy
= gd
->hsize
+ data
->cy
- data
->oy
;
849 reflow
= (gd
->sx
!= sx
);
851 grid_wrap_position(gd
, cx
, cy
, &wx
, &wy
);
852 screen_resize_cursor(data
->backing
, sx
, sy
, 1, 0, 0);
854 grid_unwrap_position(gd
, &cx
, &cy
, wx
, wy
);
857 if (cy
< gd
->hsize
) {
859 data
->oy
= gd
->hsize
- cy
;
861 data
->cy
= cy
- gd
->hsize
;
865 window_copy_size_changed(wme
);
866 window_copy_redraw_screen(wme
);
870 window_copy_key_table(struct window_mode_entry
*wme
)
872 struct window_pane
*wp
= wme
->wp
;
874 if (options_get_number(wp
->window
->options
, "mode-keys") == MODEKEY_VI
)
875 return ("copy-mode-vi");
876 return ("copy-mode");
880 window_copy_expand_search_string(struct window_copy_cmd_state
*cs
)
882 struct window_mode_entry
*wme
= cs
->wme
;
883 struct window_copy_mode_data
*data
= wme
->data
;
884 const char *ss
= args_string(cs
->args
, 1);
887 if (ss
== NULL
|| *ss
== '\0')
890 if (args_has(cs
->args
, 'F')) {
891 expanded
= format_single(NULL
, ss
, NULL
, NULL
, NULL
, wme
->wp
);
892 if (*expanded
== '\0') {
896 free(data
->searchstr
);
897 data
->searchstr
= expanded
;
899 free(data
->searchstr
);
900 data
->searchstr
= xstrdup(ss
);
905 static enum window_copy_cmd_action
906 window_copy_cmd_append_selection(struct window_copy_cmd_state
*cs
)
908 struct window_mode_entry
*wme
= cs
->wme
;
909 struct session
*s
= cs
->s
;
912 window_copy_append_selection(wme
);
913 window_copy_clear_selection(wme
);
914 return (WINDOW_COPY_CMD_REDRAW
);
917 static enum window_copy_cmd_action
918 window_copy_cmd_append_selection_and_cancel(struct window_copy_cmd_state
*cs
)
920 struct window_mode_entry
*wme
= cs
->wme
;
921 struct session
*s
= cs
->s
;
924 window_copy_append_selection(wme
);
925 window_copy_clear_selection(wme
);
926 return (WINDOW_COPY_CMD_CANCEL
);
929 static enum window_copy_cmd_action
930 window_copy_cmd_back_to_indentation(struct window_copy_cmd_state
*cs
)
932 struct window_mode_entry
*wme
= cs
->wme
;
934 window_copy_cursor_back_to_indentation(wme
);
935 return (WINDOW_COPY_CMD_NOTHING
);
938 static enum window_copy_cmd_action
939 window_copy_cmd_begin_selection(struct window_copy_cmd_state
*cs
)
941 struct window_mode_entry
*wme
= cs
->wme
;
942 struct client
*c
= cs
->c
;
943 struct mouse_event
*m
= cs
->m
;
944 struct window_copy_mode_data
*data
= wme
->data
;
947 window_copy_start_drag(c
, m
);
948 return (WINDOW_COPY_CMD_NOTHING
);
951 data
->lineflag
= LINE_SEL_NONE
;
952 data
->selflag
= SEL_CHAR
;
953 window_copy_start_selection(wme
);
954 return (WINDOW_COPY_CMD_REDRAW
);
957 static enum window_copy_cmd_action
958 window_copy_cmd_stop_selection(struct window_copy_cmd_state
*cs
)
960 struct window_mode_entry
*wme
= cs
->wme
;
961 struct window_copy_mode_data
*data
= wme
->data
;
963 data
->cursordrag
= CURSORDRAG_NONE
;
964 data
->lineflag
= LINE_SEL_NONE
;
965 data
->selflag
= SEL_CHAR
;
966 return (WINDOW_COPY_CMD_NOTHING
);
969 static enum window_copy_cmd_action
970 window_copy_cmd_bottom_line(struct window_copy_cmd_state
*cs
)
972 struct window_mode_entry
*wme
= cs
->wme
;
973 struct window_copy_mode_data
*data
= wme
->data
;
976 data
->cy
= screen_size_y(&data
->screen
) - 1;
978 window_copy_update_selection(wme
, 1, 0);
979 return (WINDOW_COPY_CMD_REDRAW
);
982 static enum window_copy_cmd_action
983 window_copy_cmd_cancel(__unused
struct window_copy_cmd_state
*cs
)
985 return (WINDOW_COPY_CMD_CANCEL
);
988 static enum window_copy_cmd_action
989 window_copy_cmd_clear_selection(struct window_copy_cmd_state
*cs
)
991 struct window_mode_entry
*wme
= cs
->wme
;
993 window_copy_clear_selection(wme
);
994 return (WINDOW_COPY_CMD_REDRAW
);
997 static enum window_copy_cmd_action
998 window_copy_do_copy_end_of_line(struct window_copy_cmd_state
*cs
, int pipe
,
1001 struct window_mode_entry
*wme
= cs
->wme
;
1002 struct client
*c
= cs
->c
;
1003 struct session
*s
= cs
->s
;
1004 struct winlink
*wl
= cs
->wl
;
1005 struct window_pane
*wp
= wme
->wp
;
1006 u_int count
= args_count(cs
->args
);
1007 u_int np
= wme
->prefix
, ocx
, ocy
, ooy
;
1008 struct window_copy_mode_data
*data
= wme
->data
;
1009 char *prefix
= NULL
, *command
= NULL
;
1010 const char *arg1
= args_string(cs
->args
, 1);
1011 const char *arg2
= args_string(cs
->args
, 2);
1015 prefix
= format_single(NULL
, arg2
, c
, s
, wl
, wp
);
1016 if (s
!= NULL
&& count
> 1 && *arg1
!= '\0')
1017 command
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
1020 prefix
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
1027 window_copy_start_selection(wme
);
1028 for (; np
> 1; np
--)
1029 window_copy_cursor_down(wme
, 0);
1030 window_copy_cursor_end_of_line(wme
);
1034 window_copy_copy_pipe(wme
, s
, prefix
, command
);
1036 window_copy_copy_selection(wme
, prefix
);
1041 return (WINDOW_COPY_CMD_CANCEL
);
1044 window_copy_clear_selection(wme
);
1052 return (WINDOW_COPY_CMD_REDRAW
);
1055 static enum window_copy_cmd_action
1056 window_copy_cmd_copy_end_of_line(struct window_copy_cmd_state
*cs
)
1058 return (window_copy_do_copy_end_of_line(cs
, 0, 0));
1061 static enum window_copy_cmd_action
1062 window_copy_cmd_copy_end_of_line_and_cancel(struct window_copy_cmd_state
*cs
)
1064 return (window_copy_do_copy_end_of_line(cs
, 0, 1));
1067 static enum window_copy_cmd_action
1068 window_copy_cmd_copy_pipe_end_of_line(struct window_copy_cmd_state
*cs
)
1070 return (window_copy_do_copy_end_of_line(cs
, 1, 0));
1073 static enum window_copy_cmd_action
1074 window_copy_cmd_copy_pipe_end_of_line_and_cancel(
1075 struct window_copy_cmd_state
*cs
)
1077 return (window_copy_do_copy_end_of_line(cs
, 1, 1));
1080 static enum window_copy_cmd_action
1081 window_copy_do_copy_line(struct window_copy_cmd_state
*cs
, int pipe
, int cancel
)
1083 struct window_mode_entry
*wme
= cs
->wme
;
1084 struct client
*c
= cs
->c
;
1085 struct session
*s
= cs
->s
;
1086 struct winlink
*wl
= cs
->wl
;
1087 struct window_pane
*wp
= wme
->wp
;
1088 struct window_copy_mode_data
*data
= wme
->data
;
1089 u_int count
= args_count(cs
->args
);
1090 u_int np
= wme
->prefix
, ocx
, ocy
, ooy
;
1091 char *prefix
= NULL
, *command
= NULL
;
1092 const char *arg1
= args_string(cs
->args
, 1);
1093 const char *arg2
= args_string(cs
->args
, 2);
1097 prefix
= format_single(NULL
, arg2
, c
, s
, wl
, wp
);
1098 if (s
!= NULL
&& count
> 1 && *arg1
!= '\0')
1099 command
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
1102 prefix
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
1109 data
->selflag
= SEL_CHAR
;
1110 window_copy_cursor_start_of_line(wme
);
1111 window_copy_start_selection(wme
);
1112 for (; np
> 1; np
--)
1113 window_copy_cursor_down(wme
, 0);
1114 window_copy_cursor_end_of_line(wme
);
1118 window_copy_copy_pipe(wme
, s
, prefix
, command
);
1120 window_copy_copy_selection(wme
, prefix
);
1125 return (WINDOW_COPY_CMD_CANCEL
);
1128 window_copy_clear_selection(wme
);
1136 return (WINDOW_COPY_CMD_REDRAW
);
1139 static enum window_copy_cmd_action
1140 window_copy_cmd_copy_line(struct window_copy_cmd_state
*cs
)
1142 return (window_copy_do_copy_line(cs
, 0, 0));
1145 static enum window_copy_cmd_action
1146 window_copy_cmd_copy_line_and_cancel(struct window_copy_cmd_state
*cs
)
1148 return (window_copy_do_copy_line(cs
, 0, 1));
1151 static enum window_copy_cmd_action
1152 window_copy_cmd_copy_pipe_line(struct window_copy_cmd_state
*cs
)
1154 return (window_copy_do_copy_line(cs
, 1, 0));
1157 static enum window_copy_cmd_action
1158 window_copy_cmd_copy_pipe_line_and_cancel(struct window_copy_cmd_state
*cs
)
1160 return (window_copy_do_copy_line(cs
, 1, 1));
1163 static enum window_copy_cmd_action
1164 window_copy_cmd_copy_selection_no_clear(struct window_copy_cmd_state
*cs
)
1166 struct window_mode_entry
*wme
= cs
->wme
;
1167 struct client
*c
= cs
->c
;
1168 struct session
*s
= cs
->s
;
1169 struct winlink
*wl
= cs
->wl
;
1170 struct window_pane
*wp
= wme
->wp
;
1171 char *prefix
= NULL
;
1172 const char *arg1
= args_string(cs
->args
, 1);
1175 prefix
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
1178 window_copy_copy_selection(wme
, prefix
);
1181 return (WINDOW_COPY_CMD_NOTHING
);
1184 static enum window_copy_cmd_action
1185 window_copy_cmd_copy_selection(struct window_copy_cmd_state
*cs
)
1187 struct window_mode_entry
*wme
= cs
->wme
;
1189 window_copy_cmd_copy_selection_no_clear(cs
);
1190 window_copy_clear_selection(wme
);
1191 return (WINDOW_COPY_CMD_REDRAW
);
1194 static enum window_copy_cmd_action
1195 window_copy_cmd_copy_selection_and_cancel(struct window_copy_cmd_state
*cs
)
1197 struct window_mode_entry
*wme
= cs
->wme
;
1199 window_copy_cmd_copy_selection_no_clear(cs
);
1200 window_copy_clear_selection(wme
);
1201 return (WINDOW_COPY_CMD_CANCEL
);
1204 static enum window_copy_cmd_action
1205 window_copy_cmd_cursor_down(struct window_copy_cmd_state
*cs
)
1207 struct window_mode_entry
*wme
= cs
->wme
;
1208 u_int np
= wme
->prefix
;
1210 for (; np
!= 0; np
--)
1211 window_copy_cursor_down(wme
, 0);
1212 return (WINDOW_COPY_CMD_NOTHING
);
1215 static enum window_copy_cmd_action
1216 window_copy_cmd_cursor_down_and_cancel(struct window_copy_cmd_state
*cs
)
1218 struct window_mode_entry
*wme
= cs
->wme
;
1219 struct window_copy_mode_data
*data
= wme
->data
;
1220 u_int np
= wme
->prefix
, cy
;
1223 for (; np
!= 0; np
--)
1224 window_copy_cursor_down(wme
, 0);
1225 if (cy
== data
->cy
&& data
->oy
== 0)
1226 return (WINDOW_COPY_CMD_CANCEL
);
1227 return (WINDOW_COPY_CMD_NOTHING
);
1230 static enum window_copy_cmd_action
1231 window_copy_cmd_cursor_left(struct window_copy_cmd_state
*cs
)
1233 struct window_mode_entry
*wme
= cs
->wme
;
1234 u_int np
= wme
->prefix
;
1236 for (; np
!= 0; np
--)
1237 window_copy_cursor_left(wme
);
1238 return (WINDOW_COPY_CMD_NOTHING
);
1241 static enum window_copy_cmd_action
1242 window_copy_cmd_cursor_right(struct window_copy_cmd_state
*cs
)
1244 struct window_mode_entry
*wme
= cs
->wme
;
1245 struct window_copy_mode_data
*data
= wme
->data
;
1246 u_int np
= wme
->prefix
;
1248 for (; np
!= 0; np
--) {
1249 window_copy_cursor_right(wme
, data
->screen
.sel
!= NULL
&&
1252 return (WINDOW_COPY_CMD_NOTHING
);
1255 /* Scroll line containing the cursor to the given position. */
1256 static enum window_copy_cmd_action
1257 window_copy_cmd_scroll_to(struct window_copy_cmd_state
*cs
, u_int to
)
1259 struct window_mode_entry
*wme
= cs
->wme
;
1260 struct window_copy_mode_data
*data
= wme
->data
;
1262 int scroll_up
; /* >0 up, <0 down */
1264 scroll_up
= data
->cy
- to
;
1265 delta
= abs(scroll_up
);
1266 oy
= screen_hsize(data
->backing
) - data
->oy
;
1269 * oy is the maximum scroll down amount, while data->oy is the maximum
1272 if (scroll_up
> 0 && data
->oy
>= delta
) {
1273 window_copy_scroll_up(wme
, delta
);
1275 } else if (scroll_up
< 0 && oy
>= delta
) {
1276 window_copy_scroll_down(wme
, delta
);
1280 window_copy_update_selection(wme
, 0, 0);
1281 return (WINDOW_COPY_CMD_REDRAW
);
1284 /* Scroll line containing the cursor to the bottom. */
1285 static enum window_copy_cmd_action
1286 window_copy_cmd_scroll_bottom(struct window_copy_cmd_state
*cs
)
1288 struct window_copy_mode_data
*data
= cs
->wme
->data
;
1291 bottom
= screen_size_y(&data
->screen
) - 1;
1292 return (window_copy_cmd_scroll_to(cs
, bottom
));
1295 /* Scroll line containing the cursor to the middle. */
1296 static enum window_copy_cmd_action
1297 window_copy_cmd_scroll_middle(struct window_copy_cmd_state
*cs
)
1299 struct window_copy_mode_data
*data
= cs
->wme
->data
;
1302 mid_value
= (screen_size_y(&data
->screen
) - 1) / 2;
1303 return (window_copy_cmd_scroll_to(cs
, mid_value
));
1306 /* Scroll line containing the cursor to the top. */
1307 static enum window_copy_cmd_action
1308 window_copy_cmd_scroll_top(struct window_copy_cmd_state
*cs
)
1310 return (window_copy_cmd_scroll_to(cs
, 0));
1313 static enum window_copy_cmd_action
1314 window_copy_cmd_cursor_up(struct window_copy_cmd_state
*cs
)
1316 struct window_mode_entry
*wme
= cs
->wme
;
1317 u_int np
= wme
->prefix
;
1319 for (; np
!= 0; np
--)
1320 window_copy_cursor_up(wme
, 0);
1321 return (WINDOW_COPY_CMD_NOTHING
);
1324 static enum window_copy_cmd_action
1325 window_copy_cmd_end_of_line(struct window_copy_cmd_state
*cs
)
1327 struct window_mode_entry
*wme
= cs
->wme
;
1329 window_copy_cursor_end_of_line(wme
);
1330 return (WINDOW_COPY_CMD_NOTHING
);
1333 static enum window_copy_cmd_action
1334 window_copy_cmd_halfpage_down(struct window_copy_cmd_state
*cs
)
1336 struct window_mode_entry
*wme
= cs
->wme
;
1337 struct window_copy_mode_data
*data
= wme
->data
;
1338 u_int np
= wme
->prefix
;
1340 for (; np
!= 0; np
--) {
1341 if (window_copy_pagedown(wme
, 1, data
->scroll_exit
))
1342 return (WINDOW_COPY_CMD_CANCEL
);
1344 return (WINDOW_COPY_CMD_NOTHING
);
1347 static enum window_copy_cmd_action
1348 window_copy_cmd_halfpage_down_and_cancel(struct window_copy_cmd_state
*cs
)
1351 struct window_mode_entry
*wme
= cs
->wme
;
1352 u_int np
= wme
->prefix
;
1354 for (; np
!= 0; np
--) {
1355 if (window_copy_pagedown(wme
, 1, 1))
1356 return (WINDOW_COPY_CMD_CANCEL
);
1358 return (WINDOW_COPY_CMD_NOTHING
);
1361 static enum window_copy_cmd_action
1362 window_copy_cmd_halfpage_up(struct window_copy_cmd_state
*cs
)
1364 struct window_mode_entry
*wme
= cs
->wme
;
1365 u_int np
= wme
->prefix
;
1367 for (; np
!= 0; np
--)
1368 window_copy_pageup1(wme
, 1);
1369 return (WINDOW_COPY_CMD_NOTHING
);
1372 static enum window_copy_cmd_action
1373 window_copy_cmd_toggle_position(struct window_copy_cmd_state
*cs
)
1375 struct window_mode_entry
*wme
= cs
->wme
;
1376 struct window_copy_mode_data
*data
= wme
->data
;
1378 data
->hide_position
= !data
->hide_position
;
1379 return (WINDOW_COPY_CMD_REDRAW
);
1382 static enum window_copy_cmd_action
1383 window_copy_cmd_history_bottom(struct window_copy_cmd_state
*cs
)
1385 struct window_mode_entry
*wme
= cs
->wme
;
1386 struct window_copy_mode_data
*data
= wme
->data
;
1387 struct screen
*s
= data
->backing
;
1390 oy
= screen_hsize(s
) + data
->cy
- data
->oy
;
1391 if (data
->lineflag
== LINE_SEL_RIGHT_LEFT
&& oy
== data
->endsely
)
1392 window_copy_other_end(wme
);
1394 data
->cy
= screen_size_y(&data
->screen
) - 1;
1395 data
->cx
= window_copy_find_length(wme
, screen_hsize(s
) + data
->cy
);
1398 if (data
->searchmark
!= NULL
&& !data
->timeout
)
1399 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
1400 window_copy_update_selection(wme
, 1, 0);
1401 return (WINDOW_COPY_CMD_REDRAW
);
1404 static enum window_copy_cmd_action
1405 window_copy_cmd_history_top(struct window_copy_cmd_state
*cs
)
1407 struct window_mode_entry
*wme
= cs
->wme
;
1408 struct window_copy_mode_data
*data
= wme
->data
;
1411 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
1412 if (data
->lineflag
== LINE_SEL_LEFT_RIGHT
&& oy
== data
->sely
)
1413 window_copy_other_end(wme
);
1417 data
->oy
= screen_hsize(data
->backing
);
1419 if (data
->searchmark
!= NULL
&& !data
->timeout
)
1420 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
1421 window_copy_update_selection(wme
, 1, 0);
1422 return (WINDOW_COPY_CMD_REDRAW
);
1425 static enum window_copy_cmd_action
1426 window_copy_cmd_jump_again(struct window_copy_cmd_state
*cs
)
1428 struct window_mode_entry
*wme
= cs
->wme
;
1429 struct window_copy_mode_data
*data
= wme
->data
;
1430 u_int np
= wme
->prefix
;
1432 switch (data
->jumptype
) {
1433 case WINDOW_COPY_JUMPFORWARD
:
1434 for (; np
!= 0; np
--)
1435 window_copy_cursor_jump(wme
);
1437 case WINDOW_COPY_JUMPBACKWARD
:
1438 for (; np
!= 0; np
--)
1439 window_copy_cursor_jump_back(wme
);
1441 case WINDOW_COPY_JUMPTOFORWARD
:
1442 for (; np
!= 0; np
--)
1443 window_copy_cursor_jump_to(wme
);
1445 case WINDOW_COPY_JUMPTOBACKWARD
:
1446 for (; np
!= 0; np
--)
1447 window_copy_cursor_jump_to_back(wme
);
1450 return (WINDOW_COPY_CMD_NOTHING
);
1453 static enum window_copy_cmd_action
1454 window_copy_cmd_jump_reverse(struct window_copy_cmd_state
*cs
)
1456 struct window_mode_entry
*wme
= cs
->wme
;
1457 struct window_copy_mode_data
*data
= wme
->data
;
1458 u_int np
= wme
->prefix
;
1460 switch (data
->jumptype
) {
1461 case WINDOW_COPY_JUMPFORWARD
:
1462 for (; np
!= 0; np
--)
1463 window_copy_cursor_jump_back(wme
);
1465 case WINDOW_COPY_JUMPBACKWARD
:
1466 for (; np
!= 0; np
--)
1467 window_copy_cursor_jump(wme
);
1469 case WINDOW_COPY_JUMPTOFORWARD
:
1470 for (; np
!= 0; np
--)
1471 window_copy_cursor_jump_to_back(wme
);
1473 case WINDOW_COPY_JUMPTOBACKWARD
:
1474 for (; np
!= 0; np
--)
1475 window_copy_cursor_jump_to(wme
);
1478 return (WINDOW_COPY_CMD_NOTHING
);
1481 static enum window_copy_cmd_action
1482 window_copy_cmd_middle_line(struct window_copy_cmd_state
*cs
)
1484 struct window_mode_entry
*wme
= cs
->wme
;
1485 struct window_copy_mode_data
*data
= wme
->data
;
1488 data
->cy
= (screen_size_y(&data
->screen
) - 1) / 2;
1490 window_copy_update_selection(wme
, 1, 0);
1491 return (WINDOW_COPY_CMD_REDRAW
);
1494 static enum window_copy_cmd_action
1495 window_copy_cmd_previous_matching_bracket(struct window_copy_cmd_state
*cs
)
1497 struct window_mode_entry
*wme
= cs
->wme
;
1498 u_int np
= wme
->prefix
;
1499 struct window_copy_mode_data
*data
= wme
->data
;
1500 struct screen
*s
= data
->backing
;
1501 char open
[] = "{[(", close
[] = "}])";
1502 char tried
, found
, start
, *cp
;
1503 u_int px
, py
, xx
, n
;
1504 struct grid_cell gc
;
1507 for (; np
!= 0; np
--) {
1508 /* Get cursor position and line length. */
1510 py
= screen_hsize(s
) + data
->cy
- data
->oy
;
1511 xx
= window_copy_find_length(wme
, py
);
1516 * Get the current character. If not on a bracket, try the
1517 * previous. If still not, then behave like previous-word.
1521 grid_get_cell(s
->grid
, px
, py
, &gc
);
1522 if (gc
.data
.size
!= 1 || (gc
.flags
& GRID_FLAG_PADDING
))
1525 found
= *gc
.data
.data
;
1526 cp
= strchr(close
, found
);
1529 if (data
->modekeys
== MODEKEY_EMACS
) {
1530 if (!tried
&& px
> 0) {
1535 window_copy_cursor_previous_word(wme
, close
, 1);
1539 start
= open
[cp
- close
];
1541 /* Walk backward until the matching bracket is reached. */
1552 xx
= window_copy_find_length(wme
, py
);
1553 } while (xx
== 0 && py
> 0);
1554 if (xx
== 0 && py
== 0) {
1562 grid_get_cell(s
->grid
, px
, py
, &gc
);
1563 if (gc
.data
.size
== 1 &&
1564 (~gc
.flags
& GRID_FLAG_PADDING
)) {
1565 if (*gc
.data
.data
== found
)
1567 else if (*gc
.data
.data
== start
)
1572 /* Move the cursor to the found location if any. */
1574 window_copy_scroll_to(wme
, px
, py
, 0);
1577 return (WINDOW_COPY_CMD_NOTHING
);
1580 static enum window_copy_cmd_action
1581 window_copy_cmd_next_matching_bracket(struct window_copy_cmd_state
*cs
)
1583 struct window_mode_entry
*wme
= cs
->wme
;
1584 u_int np
= wme
->prefix
;
1585 struct window_copy_mode_data
*data
= wme
->data
;
1586 struct screen
*s
= data
->backing
;
1587 char open
[] = "{[(", close
[] = "}])";
1588 char tried
, found
, end
, *cp
;
1589 u_int px
, py
, xx
, yy
, sx
, sy
, n
;
1590 struct grid_cell gc
;
1592 struct grid_line
*gl
;
1594 for (; np
!= 0; np
--) {
1595 /* Get cursor position and line length. */
1597 py
= screen_hsize(s
) + data
->cy
- data
->oy
;
1598 xx
= window_copy_find_length(wme
, py
);
1599 yy
= screen_hsize(s
) + screen_size_y(s
) - 1;
1604 * Get the current character. If not on a bracket, try the
1605 * next. If still not, then behave like next-word.
1609 grid_get_cell(s
->grid
, px
, py
, &gc
);
1610 if (gc
.data
.size
!= 1 || (gc
.flags
& GRID_FLAG_PADDING
))
1613 found
= *gc
.data
.data
;
1616 * In vi mode, attempt to move to previous bracket if a
1617 * closing bracket is found first. If this fails,
1618 * return to the original cursor position.
1620 cp
= strchr(close
, found
);
1621 if (cp
!= NULL
&& data
->modekeys
== MODEKEY_VI
) {
1623 sy
= screen_hsize(s
) + data
->cy
- data
->oy
;
1625 window_copy_scroll_to(wme
, px
, py
, 0);
1626 window_copy_cmd_previous_matching_bracket(cs
);
1629 py
= screen_hsize(s
) + data
->cy
- data
->oy
;
1630 grid_get_cell(s
->grid
, px
, py
, &gc
);
1631 if (gc
.data
.size
== 1 &&
1632 (~gc
.flags
& GRID_FLAG_PADDING
) &&
1633 strchr(close
, *gc
.data
.data
) != NULL
)
1634 window_copy_scroll_to(wme
, sx
, sy
, 0);
1638 cp
= strchr(open
, found
);
1641 if (data
->modekeys
== MODEKEY_EMACS
) {
1642 if (!tried
&& px
<= xx
) {
1647 window_copy_cursor_next_word_end(wme
, open
, 0);
1650 /* For vi, continue searching for bracket until EOL. */
1654 gl
= grid_get_line(s
->grid
, py
);
1655 if (~gl
->flags
& GRID_LINE_WRAPPED
)
1657 if (gl
->cellsize
> s
->grid
->sx
)
1661 xx
= window_copy_find_length(wme
, py
);
1666 end
= close
[cp
- open
];
1668 /* Walk forward until the matching bracket is reached. */
1679 xx
= window_copy_find_length(wme
, py
);
1683 grid_get_cell(s
->grid
, px
, py
, &gc
);
1684 if (gc
.data
.size
== 1 &&
1685 (~gc
.flags
& GRID_FLAG_PADDING
)) {
1686 if (*gc
.data
.data
== found
)
1688 else if (*gc
.data
.data
== end
)
1693 /* Move the cursor to the found location if any. */
1695 window_copy_scroll_to(wme
, px
, py
, 0);
1698 return (WINDOW_COPY_CMD_NOTHING
);
1701 static enum window_copy_cmd_action
1702 window_copy_cmd_next_paragraph(struct window_copy_cmd_state
*cs
)
1704 struct window_mode_entry
*wme
= cs
->wme
;
1705 u_int np
= wme
->prefix
;
1707 for (; np
!= 0; np
--)
1708 window_copy_next_paragraph(wme
);
1709 return (WINDOW_COPY_CMD_NOTHING
);
1712 static enum window_copy_cmd_action
1713 window_copy_cmd_next_space(struct window_copy_cmd_state
*cs
)
1715 struct window_mode_entry
*wme
= cs
->wme
;
1716 u_int np
= wme
->prefix
;
1718 for (; np
!= 0; np
--)
1719 window_copy_cursor_next_word(wme
, "");
1720 return (WINDOW_COPY_CMD_NOTHING
);
1723 static enum window_copy_cmd_action
1724 window_copy_cmd_next_space_end(struct window_copy_cmd_state
*cs
)
1726 struct window_mode_entry
*wme
= cs
->wme
;
1727 u_int np
= wme
->prefix
;
1729 for (; np
!= 0; np
--)
1730 window_copy_cursor_next_word_end(wme
, "", 0);
1731 return (WINDOW_COPY_CMD_NOTHING
);
1734 static enum window_copy_cmd_action
1735 window_copy_cmd_next_word(struct window_copy_cmd_state
*cs
)
1737 struct window_mode_entry
*wme
= cs
->wme
;
1738 u_int np
= wme
->prefix
;
1739 const char *separators
;
1741 separators
= options_get_string(cs
->s
->options
, "word-separators");
1743 for (; np
!= 0; np
--)
1744 window_copy_cursor_next_word(wme
, separators
);
1745 return (WINDOW_COPY_CMD_NOTHING
);
1748 static enum window_copy_cmd_action
1749 window_copy_cmd_next_word_end(struct window_copy_cmd_state
*cs
)
1751 struct window_mode_entry
*wme
= cs
->wme
;
1752 u_int np
= wme
->prefix
;
1753 const char *separators
;
1755 separators
= options_get_string(cs
->s
->options
, "word-separators");
1757 for (; np
!= 0; np
--)
1758 window_copy_cursor_next_word_end(wme
, separators
, 0);
1759 return (WINDOW_COPY_CMD_NOTHING
);
1762 static enum window_copy_cmd_action
1763 window_copy_cmd_other_end(struct window_copy_cmd_state
*cs
)
1765 struct window_mode_entry
*wme
= cs
->wme
;
1766 u_int np
= wme
->prefix
;
1767 struct window_copy_mode_data
*data
= wme
->data
;
1769 data
->selflag
= SEL_CHAR
;
1771 window_copy_other_end(wme
);
1772 return (WINDOW_COPY_CMD_NOTHING
);
1775 static enum window_copy_cmd_action
1776 window_copy_cmd_page_down(struct window_copy_cmd_state
*cs
)
1778 struct window_mode_entry
*wme
= cs
->wme
;
1779 struct window_copy_mode_data
*data
= wme
->data
;
1780 u_int np
= wme
->prefix
;
1782 for (; np
!= 0; np
--) {
1783 if (window_copy_pagedown(wme
, 0, data
->scroll_exit
))
1784 return (WINDOW_COPY_CMD_CANCEL
);
1786 return (WINDOW_COPY_CMD_NOTHING
);
1789 static enum window_copy_cmd_action
1790 window_copy_cmd_page_down_and_cancel(struct window_copy_cmd_state
*cs
)
1792 struct window_mode_entry
*wme
= cs
->wme
;
1793 u_int np
= wme
->prefix
;
1795 for (; np
!= 0; np
--) {
1796 if (window_copy_pagedown(wme
, 0, 1))
1797 return (WINDOW_COPY_CMD_CANCEL
);
1799 return (WINDOW_COPY_CMD_NOTHING
);
1802 static enum window_copy_cmd_action
1803 window_copy_cmd_page_up(struct window_copy_cmd_state
*cs
)
1805 struct window_mode_entry
*wme
= cs
->wme
;
1806 u_int np
= wme
->prefix
;
1808 for (; np
!= 0; np
--)
1809 window_copy_pageup1(wme
, 0);
1810 return (WINDOW_COPY_CMD_NOTHING
);
1813 static enum window_copy_cmd_action
1814 window_copy_cmd_previous_paragraph(struct window_copy_cmd_state
*cs
)
1816 struct window_mode_entry
*wme
= cs
->wme
;
1817 u_int np
= wme
->prefix
;
1819 for (; np
!= 0; np
--)
1820 window_copy_previous_paragraph(wme
);
1821 return (WINDOW_COPY_CMD_NOTHING
);
1824 static enum window_copy_cmd_action
1825 window_copy_cmd_previous_space(struct window_copy_cmd_state
*cs
)
1827 struct window_mode_entry
*wme
= cs
->wme
;
1828 u_int np
= wme
->prefix
;
1830 for (; np
!= 0; np
--)
1831 window_copy_cursor_previous_word(wme
, "", 1);
1832 return (WINDOW_COPY_CMD_NOTHING
);
1835 static enum window_copy_cmd_action
1836 window_copy_cmd_previous_word(struct window_copy_cmd_state
*cs
)
1838 struct window_mode_entry
*wme
= cs
->wme
;
1839 u_int np
= wme
->prefix
;
1840 const char *separators
;
1842 separators
= options_get_string(cs
->s
->options
, "word-separators");
1844 for (; np
!= 0; np
--)
1845 window_copy_cursor_previous_word(wme
, separators
, 1);
1846 return (WINDOW_COPY_CMD_NOTHING
);
1849 static enum window_copy_cmd_action
1850 window_copy_cmd_rectangle_on(struct window_copy_cmd_state
*cs
)
1852 struct window_mode_entry
*wme
= cs
->wme
;
1853 struct window_copy_mode_data
*data
= wme
->data
;
1855 data
->lineflag
= LINE_SEL_NONE
;
1856 window_copy_rectangle_set(wme
, 1);
1858 return (WINDOW_COPY_CMD_NOTHING
);
1861 static enum window_copy_cmd_action
1862 window_copy_cmd_rectangle_off(struct window_copy_cmd_state
*cs
)
1864 struct window_mode_entry
*wme
= cs
->wme
;
1865 struct window_copy_mode_data
*data
= wme
->data
;
1867 data
->lineflag
= LINE_SEL_NONE
;
1868 window_copy_rectangle_set(wme
, 0);
1870 return (WINDOW_COPY_CMD_NOTHING
);
1873 static enum window_copy_cmd_action
1874 window_copy_cmd_rectangle_toggle(struct window_copy_cmd_state
*cs
)
1876 struct window_mode_entry
*wme
= cs
->wme
;
1877 struct window_copy_mode_data
*data
= wme
->data
;
1879 data
->lineflag
= LINE_SEL_NONE
;
1880 window_copy_rectangle_set(wme
, !data
->rectflag
);
1882 return (WINDOW_COPY_CMD_NOTHING
);
1885 static enum window_copy_cmd_action
1886 window_copy_cmd_scroll_down(struct window_copy_cmd_state
*cs
)
1888 struct window_mode_entry
*wme
= cs
->wme
;
1889 struct window_copy_mode_data
*data
= wme
->data
;
1890 u_int np
= wme
->prefix
;
1892 for (; np
!= 0; np
--)
1893 window_copy_cursor_down(wme
, 1);
1894 if (data
->scroll_exit
&& data
->oy
== 0)
1895 return (WINDOW_COPY_CMD_CANCEL
);
1896 return (WINDOW_COPY_CMD_NOTHING
);
1899 static enum window_copy_cmd_action
1900 window_copy_cmd_scroll_down_and_cancel(struct window_copy_cmd_state
*cs
)
1902 struct window_mode_entry
*wme
= cs
->wme
;
1903 struct window_copy_mode_data
*data
= wme
->data
;
1904 u_int np
= wme
->prefix
;
1906 for (; np
!= 0; np
--)
1907 window_copy_cursor_down(wme
, 1);
1909 return (WINDOW_COPY_CMD_CANCEL
);
1910 return (WINDOW_COPY_CMD_NOTHING
);
1913 static enum window_copy_cmd_action
1914 window_copy_cmd_scroll_up(struct window_copy_cmd_state
*cs
)
1916 struct window_mode_entry
*wme
= cs
->wme
;
1917 u_int np
= wme
->prefix
;
1919 for (; np
!= 0; np
--)
1920 window_copy_cursor_up(wme
, 1);
1921 return (WINDOW_COPY_CMD_NOTHING
);
1924 static enum window_copy_cmd_action
1925 window_copy_cmd_search_again(struct window_copy_cmd_state
*cs
)
1927 struct window_mode_entry
*wme
= cs
->wme
;
1928 struct window_copy_mode_data
*data
= wme
->data
;
1929 u_int np
= wme
->prefix
;
1931 if (data
->searchtype
== WINDOW_COPY_SEARCHUP
) {
1932 for (; np
!= 0; np
--)
1933 window_copy_search_up(wme
, data
->searchregex
);
1934 } else if (data
->searchtype
== WINDOW_COPY_SEARCHDOWN
) {
1935 for (; np
!= 0; np
--)
1936 window_copy_search_down(wme
, data
->searchregex
);
1938 return (WINDOW_COPY_CMD_NOTHING
);
1941 static enum window_copy_cmd_action
1942 window_copy_cmd_search_reverse(struct window_copy_cmd_state
*cs
)
1944 struct window_mode_entry
*wme
= cs
->wme
;
1945 struct window_copy_mode_data
*data
= wme
->data
;
1946 u_int np
= wme
->prefix
;
1948 if (data
->searchtype
== WINDOW_COPY_SEARCHUP
) {
1949 for (; np
!= 0; np
--)
1950 window_copy_search_down(wme
, data
->searchregex
);
1951 } else if (data
->searchtype
== WINDOW_COPY_SEARCHDOWN
) {
1952 for (; np
!= 0; np
--)
1953 window_copy_search_up(wme
, data
->searchregex
);
1955 return (WINDOW_COPY_CMD_NOTHING
);
1958 static enum window_copy_cmd_action
1959 window_copy_cmd_select_line(struct window_copy_cmd_state
*cs
)
1961 struct window_mode_entry
*wme
= cs
->wme
;
1962 struct window_copy_mode_data
*data
= wme
->data
;
1963 u_int np
= wme
->prefix
;
1965 data
->lineflag
= LINE_SEL_LEFT_RIGHT
;
1967 data
->selflag
= SEL_LINE
;
1968 data
->dx
= data
->cx
;
1969 data
->dy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
1971 window_copy_cursor_start_of_line(wme
);
1972 data
->selrx
= data
->cx
;
1973 data
->selry
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
1974 data
->endselry
= data
->selry
;
1975 window_copy_start_selection(wme
);
1976 window_copy_cursor_end_of_line(wme
);
1977 data
->endselry
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
1978 data
->endselrx
= window_copy_find_length(wme
, data
->endselry
);
1979 for (; np
> 1; np
--) {
1980 window_copy_cursor_down(wme
, 0);
1981 window_copy_cursor_end_of_line(wme
);
1984 return (WINDOW_COPY_CMD_REDRAW
);
1987 static enum window_copy_cmd_action
1988 window_copy_cmd_select_word(struct window_copy_cmd_state
*cs
)
1990 struct window_mode_entry
*wme
= cs
->wme
;
1991 struct options
*session_options
= cs
->s
->options
;
1992 struct window_copy_mode_data
*data
= wme
->data
;
1993 u_int px
, py
, nextx
, nexty
;
1995 data
->lineflag
= LINE_SEL_LEFT_RIGHT
;
1997 data
->selflag
= SEL_WORD
;
1998 data
->dx
= data
->cx
;
1999 data
->dy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
2001 data
->separators
= options_get_string(session_options
,
2003 window_copy_cursor_previous_word(wme
, data
->separators
, 0);
2005 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
2008 window_copy_start_selection(wme
);
2010 /* Handle single character words. */
2013 if (grid_get_line(data
->backing
->grid
, nexty
)->flags
&
2014 GRID_LINE_WRAPPED
&& nextx
> screen_size_x(data
->backing
) - 1) {
2018 if (px
>= window_copy_find_length(wme
, py
) ||
2019 !window_copy_in_set(wme
, nextx
, nexty
, WHITESPACE
))
2020 window_copy_cursor_next_word_end(wme
, data
->separators
, 1);
2022 window_copy_update_cursor(wme
, px
, data
->cy
);
2023 if (window_copy_update_selection(wme
, 1, 1))
2024 window_copy_redraw_lines(wme
, data
->cy
, 1);
2026 data
->endselrx
= data
->cx
;
2027 data
->endselry
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
2028 if (data
->dy
> data
->endselry
) {
2029 data
->dy
= data
->endselry
;
2030 data
->dx
= data
->endselrx
;
2031 } else if (data
->dx
> data
->endselrx
)
2032 data
->dx
= data
->endselrx
;
2034 return (WINDOW_COPY_CMD_REDRAW
);
2037 static enum window_copy_cmd_action
2038 window_copy_cmd_set_mark(struct window_copy_cmd_state
*cs
)
2040 struct window_copy_mode_data
*data
= cs
->wme
->data
;
2042 data
->mx
= data
->cx
;
2043 data
->my
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
2045 return (WINDOW_COPY_CMD_REDRAW
);
2048 static enum window_copy_cmd_action
2049 window_copy_cmd_start_of_line(struct window_copy_cmd_state
*cs
)
2051 struct window_mode_entry
*wme
= cs
->wme
;
2053 window_copy_cursor_start_of_line(wme
);
2054 return (WINDOW_COPY_CMD_NOTHING
);
2057 static enum window_copy_cmd_action
2058 window_copy_cmd_top_line(struct window_copy_cmd_state
*cs
)
2060 struct window_mode_entry
*wme
= cs
->wme
;
2061 struct window_copy_mode_data
*data
= wme
->data
;
2066 window_copy_update_selection(wme
, 1, 0);
2067 return (WINDOW_COPY_CMD_REDRAW
);
2070 static enum window_copy_cmd_action
2071 window_copy_cmd_copy_pipe_no_clear(struct window_copy_cmd_state
*cs
)
2073 struct window_mode_entry
*wme
= cs
->wme
;
2074 struct client
*c
= cs
->c
;
2075 struct session
*s
= cs
->s
;
2076 struct winlink
*wl
= cs
->wl
;
2077 struct window_pane
*wp
= wme
->wp
;
2078 char *command
= NULL
, *prefix
= NULL
;
2079 const char *arg1
= args_string(cs
->args
, 1);
2080 const char *arg2
= args_string(cs
->args
, 2);
2083 prefix
= format_single(NULL
, arg2
, c
, s
, wl
, wp
);
2085 if (s
!= NULL
&& arg1
!= NULL
&& *arg1
!= '\0')
2086 command
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
2087 window_copy_copy_pipe(wme
, s
, prefix
, command
);
2091 return (WINDOW_COPY_CMD_NOTHING
);
2094 static enum window_copy_cmd_action
2095 window_copy_cmd_copy_pipe(struct window_copy_cmd_state
*cs
)
2097 struct window_mode_entry
*wme
= cs
->wme
;
2099 window_copy_cmd_copy_pipe_no_clear(cs
);
2100 window_copy_clear_selection(wme
);
2101 return (WINDOW_COPY_CMD_REDRAW
);
2104 static enum window_copy_cmd_action
2105 window_copy_cmd_copy_pipe_and_cancel(struct window_copy_cmd_state
*cs
)
2107 struct window_mode_entry
*wme
= cs
->wme
;
2109 window_copy_cmd_copy_pipe_no_clear(cs
);
2110 window_copy_clear_selection(wme
);
2111 return (WINDOW_COPY_CMD_CANCEL
);
2114 static enum window_copy_cmd_action
2115 window_copy_cmd_pipe_no_clear(struct window_copy_cmd_state
*cs
)
2117 struct window_mode_entry
*wme
= cs
->wme
;
2118 struct client
*c
= cs
->c
;
2119 struct session
*s
= cs
->s
;
2120 struct winlink
*wl
= cs
->wl
;
2121 struct window_pane
*wp
= wme
->wp
;
2122 char *command
= NULL
;
2123 const char *arg1
= args_string(cs
->args
, 1);
2125 if (s
!= NULL
&& arg1
!= NULL
&& *arg1
!= '\0')
2126 command
= format_single(NULL
, arg1
, c
, s
, wl
, wp
);
2127 window_copy_pipe(wme
, s
, command
);
2130 return (WINDOW_COPY_CMD_NOTHING
);
2133 static enum window_copy_cmd_action
2134 window_copy_cmd_pipe(struct window_copy_cmd_state
*cs
)
2136 struct window_mode_entry
*wme
= cs
->wme
;
2138 window_copy_cmd_pipe_no_clear(cs
);
2139 window_copy_clear_selection(wme
);
2140 return (WINDOW_COPY_CMD_REDRAW
);
2143 static enum window_copy_cmd_action
2144 window_copy_cmd_pipe_and_cancel(struct window_copy_cmd_state
*cs
)
2146 struct window_mode_entry
*wme
= cs
->wme
;
2148 window_copy_cmd_pipe_no_clear(cs
);
2149 window_copy_clear_selection(wme
);
2150 return (WINDOW_COPY_CMD_CANCEL
);
2153 static enum window_copy_cmd_action
2154 window_copy_cmd_goto_line(struct window_copy_cmd_state
*cs
)
2156 struct window_mode_entry
*wme
= cs
->wme
;
2157 const char *arg1
= args_string(cs
->args
, 1);
2160 window_copy_goto_line(wme
, arg1
);
2161 return (WINDOW_COPY_CMD_NOTHING
);
2164 static enum window_copy_cmd_action
2165 window_copy_cmd_jump_backward(struct window_copy_cmd_state
*cs
)
2167 struct window_mode_entry
*wme
= cs
->wme
;
2168 struct window_copy_mode_data
*data
= wme
->data
;
2169 u_int np
= wme
->prefix
;
2170 const char *arg1
= args_string(cs
->args
, 1);
2172 if (*arg1
!= '\0') {
2173 data
->jumptype
= WINDOW_COPY_JUMPBACKWARD
;
2174 free(data
->jumpchar
);
2175 data
->jumpchar
= utf8_fromcstr(arg1
);
2176 for (; np
!= 0; np
--)
2177 window_copy_cursor_jump_back(wme
);
2179 return (WINDOW_COPY_CMD_NOTHING
);
2182 static enum window_copy_cmd_action
2183 window_copy_cmd_jump_forward(struct window_copy_cmd_state
*cs
)
2185 struct window_mode_entry
*wme
= cs
->wme
;
2186 struct window_copy_mode_data
*data
= wme
->data
;
2187 u_int np
= wme
->prefix
;
2188 const char *arg1
= args_string(cs
->args
, 1);
2190 if (*arg1
!= '\0') {
2191 data
->jumptype
= WINDOW_COPY_JUMPFORWARD
;
2192 free(data
->jumpchar
);
2193 data
->jumpchar
= utf8_fromcstr(arg1
);
2194 for (; np
!= 0; np
--)
2195 window_copy_cursor_jump(wme
);
2197 return (WINDOW_COPY_CMD_NOTHING
);
2200 static enum window_copy_cmd_action
2201 window_copy_cmd_jump_to_backward(struct window_copy_cmd_state
*cs
)
2203 struct window_mode_entry
*wme
= cs
->wme
;
2204 struct window_copy_mode_data
*data
= wme
->data
;
2205 u_int np
= wme
->prefix
;
2206 const char *arg1
= args_string(cs
->args
, 1);
2208 if (*arg1
!= '\0') {
2209 data
->jumptype
= WINDOW_COPY_JUMPTOBACKWARD
;
2210 free(data
->jumpchar
);
2211 data
->jumpchar
= utf8_fromcstr(arg1
);
2212 for (; np
!= 0; np
--)
2213 window_copy_cursor_jump_to_back(wme
);
2215 return (WINDOW_COPY_CMD_NOTHING
);
2218 static enum window_copy_cmd_action
2219 window_copy_cmd_jump_to_forward(struct window_copy_cmd_state
*cs
)
2221 struct window_mode_entry
*wme
= cs
->wme
;
2222 struct window_copy_mode_data
*data
= wme
->data
;
2223 u_int np
= wme
->prefix
;
2224 const char *arg1
= args_string(cs
->args
, 1);
2226 if (*arg1
!= '\0') {
2227 data
->jumptype
= WINDOW_COPY_JUMPTOFORWARD
;
2228 free(data
->jumpchar
);
2229 data
->jumpchar
= utf8_fromcstr(arg1
);
2230 for (; np
!= 0; np
--)
2231 window_copy_cursor_jump_to(wme
);
2233 return (WINDOW_COPY_CMD_NOTHING
);
2236 static enum window_copy_cmd_action
2237 window_copy_cmd_jump_to_mark(struct window_copy_cmd_state
*cs
)
2239 struct window_mode_entry
*wme
= cs
->wme
;
2241 window_copy_jump_to_mark(wme
);
2242 return (WINDOW_COPY_CMD_NOTHING
);
2245 static enum window_copy_cmd_action
2246 window_copy_cmd_next_prompt(struct window_copy_cmd_state
*cs
)
2248 struct window_mode_entry
*wme
= cs
->wme
;
2249 const char *arg1
= args_string(cs
->args
, 1);
2251 window_copy_cursor_prompt(wme
, 1, arg1
);
2252 return (WINDOW_COPY_CMD_NOTHING
);
2255 static enum window_copy_cmd_action
2256 window_copy_cmd_previous_prompt(struct window_copy_cmd_state
*cs
)
2258 struct window_mode_entry
*wme
= cs
->wme
;
2259 const char *arg1
= args_string(cs
->args
, 1);
2261 window_copy_cursor_prompt(wme
, 0, arg1
);
2262 return (WINDOW_COPY_CMD_NOTHING
);
2265 static enum window_copy_cmd_action
2266 window_copy_cmd_search_backward(struct window_copy_cmd_state
*cs
)
2268 struct window_mode_entry
*wme
= cs
->wme
;
2269 struct window_copy_mode_data
*data
= wme
->data
;
2270 u_int np
= wme
->prefix
;
2272 if (!window_copy_expand_search_string(cs
))
2273 return (WINDOW_COPY_CMD_NOTHING
);
2275 if (data
->searchstr
!= NULL
) {
2276 data
->searchtype
= WINDOW_COPY_SEARCHUP
;
2277 data
->searchregex
= 1;
2279 for (; np
!= 0; np
--)
2280 window_copy_search_up(wme
, 1);
2282 return (WINDOW_COPY_CMD_NOTHING
);
2285 static enum window_copy_cmd_action
2286 window_copy_cmd_search_backward_text(struct window_copy_cmd_state
*cs
)
2288 struct window_mode_entry
*wme
= cs
->wme
;
2289 struct window_copy_mode_data
*data
= wme
->data
;
2290 u_int np
= wme
->prefix
;
2292 if (!window_copy_expand_search_string(cs
))
2293 return (WINDOW_COPY_CMD_NOTHING
);
2295 if (data
->searchstr
!= NULL
) {
2296 data
->searchtype
= WINDOW_COPY_SEARCHUP
;
2297 data
->searchregex
= 0;
2299 for (; np
!= 0; np
--)
2300 window_copy_search_up(wme
, 0);
2302 return (WINDOW_COPY_CMD_NOTHING
);
2305 static enum window_copy_cmd_action
2306 window_copy_cmd_search_forward(struct window_copy_cmd_state
*cs
)
2308 struct window_mode_entry
*wme
= cs
->wme
;
2309 struct window_copy_mode_data
*data
= wme
->data
;
2310 u_int np
= wme
->prefix
;
2312 if (!window_copy_expand_search_string(cs
))
2313 return (WINDOW_COPY_CMD_NOTHING
);
2315 if (data
->searchstr
!= NULL
) {
2316 data
->searchtype
= WINDOW_COPY_SEARCHDOWN
;
2317 data
->searchregex
= 1;
2319 for (; np
!= 0; np
--)
2320 window_copy_search_down(wme
, 1);
2322 return (WINDOW_COPY_CMD_NOTHING
);
2325 static enum window_copy_cmd_action
2326 window_copy_cmd_search_forward_text(struct window_copy_cmd_state
*cs
)
2328 struct window_mode_entry
*wme
= cs
->wme
;
2329 struct window_copy_mode_data
*data
= wme
->data
;
2330 u_int np
= wme
->prefix
;
2332 if (!window_copy_expand_search_string(cs
))
2333 return (WINDOW_COPY_CMD_NOTHING
);
2335 if (data
->searchstr
!= NULL
) {
2336 data
->searchtype
= WINDOW_COPY_SEARCHDOWN
;
2337 data
->searchregex
= 0;
2339 for (; np
!= 0; np
--)
2340 window_copy_search_down(wme
, 0);
2342 return (WINDOW_COPY_CMD_NOTHING
);
2345 static enum window_copy_cmd_action
2346 window_copy_cmd_search_backward_incremental(struct window_copy_cmd_state
*cs
)
2348 struct window_mode_entry
*wme
= cs
->wme
;
2349 struct window_copy_mode_data
*data
= wme
->data
;
2350 const char *arg1
= args_string(cs
->args
, 1);
2351 const char *ss
= data
->searchstr
;
2353 enum window_copy_cmd_action action
= WINDOW_COPY_CMD_NOTHING
;
2357 log_debug("%s: %s", __func__
, arg1
);
2360 if (data
->searchx
== -1 || data
->searchy
== -1) {
2361 data
->searchx
= data
->cx
;
2362 data
->searchy
= data
->cy
;
2363 data
->searcho
= data
->oy
;
2364 } else if (ss
!= NULL
&& strcmp(arg1
, ss
) != 0) {
2365 data
->cx
= data
->searchx
;
2366 data
->cy
= data
->searchy
;
2367 data
->oy
= data
->searcho
;
2368 action
= WINDOW_COPY_CMD_REDRAW
;
2370 if (*arg1
== '\0') {
2371 window_copy_clear_marks(wme
);
2372 return (WINDOW_COPY_CMD_REDRAW
);
2377 data
->searchtype
= WINDOW_COPY_SEARCHUP
;
2378 data
->searchregex
= 0;
2379 free(data
->searchstr
);
2380 data
->searchstr
= xstrdup(arg1
);
2381 if (!window_copy_search_up(wme
, 0)) {
2382 window_copy_clear_marks(wme
);
2383 return (WINDOW_COPY_CMD_REDRAW
);
2387 data
->searchtype
= WINDOW_COPY_SEARCHDOWN
;
2388 data
->searchregex
= 0;
2389 free(data
->searchstr
);
2390 data
->searchstr
= xstrdup(arg1
);
2391 if (!window_copy_search_down(wme
, 0)) {
2392 window_copy_clear_marks(wme
);
2393 return (WINDOW_COPY_CMD_REDRAW
);
2400 static enum window_copy_cmd_action
2401 window_copy_cmd_search_forward_incremental(struct window_copy_cmd_state
*cs
)
2403 struct window_mode_entry
*wme
= cs
->wme
;
2404 struct window_copy_mode_data
*data
= wme
->data
;
2405 const char *arg1
= args_string(cs
->args
, 1);
2406 const char *ss
= data
->searchstr
;
2408 enum window_copy_cmd_action action
= WINDOW_COPY_CMD_NOTHING
;
2412 log_debug("%s: %s", __func__
, arg1
);
2415 if (data
->searchx
== -1 || data
->searchy
== -1) {
2416 data
->searchx
= data
->cx
;
2417 data
->searchy
= data
->cy
;
2418 data
->searcho
= data
->oy
;
2419 } else if (ss
!= NULL
&& strcmp(arg1
, ss
) != 0) {
2420 data
->cx
= data
->searchx
;
2421 data
->cy
= data
->searchy
;
2422 data
->oy
= data
->searcho
;
2423 action
= WINDOW_COPY_CMD_REDRAW
;
2425 if (*arg1
== '\0') {
2426 window_copy_clear_marks(wme
);
2427 return (WINDOW_COPY_CMD_REDRAW
);
2432 data
->searchtype
= WINDOW_COPY_SEARCHDOWN
;
2433 data
->searchregex
= 0;
2434 free(data
->searchstr
);
2435 data
->searchstr
= xstrdup(arg1
);
2436 if (!window_copy_search_down(wme
, 0)) {
2437 window_copy_clear_marks(wme
);
2438 return (WINDOW_COPY_CMD_REDRAW
);
2442 data
->searchtype
= WINDOW_COPY_SEARCHUP
;
2443 data
->searchregex
= 0;
2444 free(data
->searchstr
);
2445 data
->searchstr
= xstrdup(arg1
);
2446 if (!window_copy_search_up(wme
, 0)) {
2447 window_copy_clear_marks(wme
);
2448 return (WINDOW_COPY_CMD_REDRAW
);
2454 static enum window_copy_cmd_action
2455 window_copy_cmd_refresh_from_pane(struct window_copy_cmd_state
*cs
)
2457 struct window_mode_entry
*wme
= cs
->wme
;
2458 struct window_pane
*wp
= wme
->swp
;
2459 struct window_copy_mode_data
*data
= wme
->data
;
2462 return (WINDOW_COPY_CMD_NOTHING
);
2464 screen_free(data
->backing
);
2465 free(data
->backing
);
2466 data
->backing
= window_copy_clone_screen(&wp
->base
, &data
->screen
, NULL
, NULL
, wme
->swp
!= wme
->wp
);
2468 window_copy_size_changed(wme
);
2469 return (WINDOW_COPY_CMD_REDRAW
);
2472 static const struct {
2473 const char *command
;
2476 enum window_copy_cmd_clear clear
;
2477 enum window_copy_cmd_action (*f
)(struct window_copy_cmd_state
*);
2478 } window_copy_cmd_table
[] = {
2479 { .command
= "append-selection",
2482 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2483 .f
= window_copy_cmd_append_selection
2485 { .command
= "append-selection-and-cancel",
2488 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2489 .f
= window_copy_cmd_append_selection_and_cancel
2491 { .command
= "back-to-indentation",
2494 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2495 .f
= window_copy_cmd_back_to_indentation
2497 { .command
= "begin-selection",
2500 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2501 .f
= window_copy_cmd_begin_selection
2503 { .command
= "bottom-line",
2506 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2507 .f
= window_copy_cmd_bottom_line
2509 { .command
= "cancel",
2512 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2513 .f
= window_copy_cmd_cancel
2515 { .command
= "clear-selection",
2518 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2519 .f
= window_copy_cmd_clear_selection
2521 { .command
= "copy-end-of-line",
2524 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2525 .f
= window_copy_cmd_copy_end_of_line
2527 { .command
= "copy-end-of-line-and-cancel",
2530 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2531 .f
= window_copy_cmd_copy_end_of_line_and_cancel
2533 { .command
= "copy-pipe-end-of-line",
2536 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2537 .f
= window_copy_cmd_copy_pipe_end_of_line
2539 { .command
= "copy-pipe-end-of-line-and-cancel",
2542 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2543 .f
= window_copy_cmd_copy_pipe_end_of_line_and_cancel
2545 { .command
= "copy-line",
2548 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2549 .f
= window_copy_cmd_copy_line
2551 { .command
= "copy-line-and-cancel",
2554 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2555 .f
= window_copy_cmd_copy_line_and_cancel
2557 { .command
= "copy-pipe-line",
2560 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2561 .f
= window_copy_cmd_copy_pipe_line
2563 { .command
= "copy-pipe-line-and-cancel",
2566 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2567 .f
= window_copy_cmd_copy_pipe_line_and_cancel
2569 { .command
= "copy-pipe-no-clear",
2572 .clear
= WINDOW_COPY_CMD_CLEAR_NEVER
,
2573 .f
= window_copy_cmd_copy_pipe_no_clear
2575 { .command
= "copy-pipe",
2578 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2579 .f
= window_copy_cmd_copy_pipe
2581 { .command
= "copy-pipe-and-cancel",
2584 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2585 .f
= window_copy_cmd_copy_pipe_and_cancel
2587 { .command
= "copy-selection-no-clear",
2590 .clear
= WINDOW_COPY_CMD_CLEAR_NEVER
,
2591 .f
= window_copy_cmd_copy_selection_no_clear
2593 { .command
= "copy-selection",
2596 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2597 .f
= window_copy_cmd_copy_selection
2599 { .command
= "copy-selection-and-cancel",
2602 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2603 .f
= window_copy_cmd_copy_selection_and_cancel
2605 { .command
= "cursor-down",
2608 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2609 .f
= window_copy_cmd_cursor_down
2611 { .command
= "cursor-down-and-cancel",
2614 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2615 .f
= window_copy_cmd_cursor_down_and_cancel
2617 { .command
= "cursor-left",
2620 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2621 .f
= window_copy_cmd_cursor_left
2623 { .command
= "cursor-right",
2626 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2627 .f
= window_copy_cmd_cursor_right
2629 { .command
= "cursor-up",
2632 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2633 .f
= window_copy_cmd_cursor_up
2635 { .command
= "end-of-line",
2638 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2639 .f
= window_copy_cmd_end_of_line
2641 { .command
= "goto-line",
2644 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2645 .f
= window_copy_cmd_goto_line
2647 { .command
= "halfpage-down",
2650 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2651 .f
= window_copy_cmd_halfpage_down
2653 { .command
= "halfpage-down-and-cancel",
2656 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2657 .f
= window_copy_cmd_halfpage_down_and_cancel
2659 { .command
= "halfpage-up",
2662 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2663 .f
= window_copy_cmd_halfpage_up
2665 { .command
= "history-bottom",
2668 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2669 .f
= window_copy_cmd_history_bottom
2671 { .command
= "history-top",
2674 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2675 .f
= window_copy_cmd_history_top
2677 { .command
= "jump-again",
2680 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2681 .f
= window_copy_cmd_jump_again
2683 { .command
= "jump-backward",
2686 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2687 .f
= window_copy_cmd_jump_backward
2689 { .command
= "jump-forward",
2692 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2693 .f
= window_copy_cmd_jump_forward
2695 { .command
= "jump-reverse",
2698 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2699 .f
= window_copy_cmd_jump_reverse
2701 { .command
= "jump-to-backward",
2704 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2705 .f
= window_copy_cmd_jump_to_backward
2707 { .command
= "jump-to-forward",
2710 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2711 .f
= window_copy_cmd_jump_to_forward
2713 { .command
= "jump-to-mark",
2716 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2717 .f
= window_copy_cmd_jump_to_mark
2719 { .command
= "next-prompt",
2722 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2723 .f
= window_copy_cmd_next_prompt
2725 { .command
= "previous-prompt",
2728 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2729 .f
= window_copy_cmd_previous_prompt
2731 { .command
= "middle-line",
2734 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2735 .f
= window_copy_cmd_middle_line
2737 { .command
= "next-matching-bracket",
2740 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2741 .f
= window_copy_cmd_next_matching_bracket
2743 { .command
= "next-paragraph",
2746 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2747 .f
= window_copy_cmd_next_paragraph
2749 { .command
= "next-space",
2752 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2753 .f
= window_copy_cmd_next_space
2755 { .command
= "next-space-end",
2758 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2759 .f
= window_copy_cmd_next_space_end
2761 { .command
= "next-word",
2764 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2765 .f
= window_copy_cmd_next_word
2767 { .command
= "next-word-end",
2770 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2771 .f
= window_copy_cmd_next_word_end
2773 { .command
= "other-end",
2776 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2777 .f
= window_copy_cmd_other_end
2779 { .command
= "page-down",
2782 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2783 .f
= window_copy_cmd_page_down
2785 { .command
= "page-down-and-cancel",
2788 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2789 .f
= window_copy_cmd_page_down_and_cancel
2791 { .command
= "page-up",
2794 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2795 .f
= window_copy_cmd_page_up
2797 { .command
= "pipe-no-clear",
2800 .clear
= WINDOW_COPY_CMD_CLEAR_NEVER
,
2801 .f
= window_copy_cmd_pipe_no_clear
2803 { .command
= "pipe",
2806 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2807 .f
= window_copy_cmd_pipe
2809 { .command
= "pipe-and-cancel",
2812 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2813 .f
= window_copy_cmd_pipe_and_cancel
2815 { .command
= "previous-matching-bracket",
2818 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2819 .f
= window_copy_cmd_previous_matching_bracket
2821 { .command
= "previous-paragraph",
2824 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2825 .f
= window_copy_cmd_previous_paragraph
2827 { .command
= "previous-space",
2830 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2831 .f
= window_copy_cmd_previous_space
2833 { .command
= "previous-word",
2836 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2837 .f
= window_copy_cmd_previous_word
2839 { .command
= "rectangle-on",
2842 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2843 .f
= window_copy_cmd_rectangle_on
2845 { .command
= "rectangle-off",
2848 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2849 .f
= window_copy_cmd_rectangle_off
2851 { .command
= "rectangle-toggle",
2854 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2855 .f
= window_copy_cmd_rectangle_toggle
2857 { .command
= "refresh-from-pane",
2860 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2861 .f
= window_copy_cmd_refresh_from_pane
2863 { .command
= "scroll-bottom",
2866 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2867 .f
= window_copy_cmd_scroll_bottom
2869 { .command
= "scroll-down",
2872 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2873 .f
= window_copy_cmd_scroll_down
2875 { .command
= "scroll-down-and-cancel",
2878 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2879 .f
= window_copy_cmd_scroll_down_and_cancel
2881 { .command
= "scroll-middle",
2884 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2885 .f
= window_copy_cmd_scroll_middle
2887 { .command
= "scroll-top",
2890 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2891 .f
= window_copy_cmd_scroll_top
2893 { .command
= "scroll-up",
2896 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2897 .f
= window_copy_cmd_scroll_up
2899 { .command
= "search-again",
2902 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2903 .f
= window_copy_cmd_search_again
2905 { .command
= "search-backward",
2908 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2909 .f
= window_copy_cmd_search_backward
2911 { .command
= "search-backward-text",
2914 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2915 .f
= window_copy_cmd_search_backward_text
2917 { .command
= "search-backward-incremental",
2920 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2921 .f
= window_copy_cmd_search_backward_incremental
2923 { .command
= "search-forward",
2926 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2927 .f
= window_copy_cmd_search_forward
2929 { .command
= "search-forward-text",
2932 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2933 .f
= window_copy_cmd_search_forward_text
2935 { .command
= "search-forward-incremental",
2938 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2939 .f
= window_copy_cmd_search_forward_incremental
2941 { .command
= "search-reverse",
2944 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2945 .f
= window_copy_cmd_search_reverse
2947 { .command
= "select-line",
2950 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2951 .f
= window_copy_cmd_select_line
2953 { .command
= "select-word",
2956 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2957 .f
= window_copy_cmd_select_word
2959 { .command
= "set-mark",
2962 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2963 .f
= window_copy_cmd_set_mark
2965 { .command
= "start-of-line",
2968 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2969 .f
= window_copy_cmd_start_of_line
2971 { .command
= "stop-selection",
2974 .clear
= WINDOW_COPY_CMD_CLEAR_ALWAYS
,
2975 .f
= window_copy_cmd_stop_selection
2977 { .command
= "toggle-position",
2980 .clear
= WINDOW_COPY_CMD_CLEAR_NEVER
,
2981 .f
= window_copy_cmd_toggle_position
2983 { .command
= "top-line",
2986 .clear
= WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
,
2987 .f
= window_copy_cmd_top_line
2992 window_copy_command(struct window_mode_entry
*wme
, struct client
*c
,
2993 struct session
*s
, struct winlink
*wl
, struct args
*args
,
2994 struct mouse_event
*m
)
2996 struct window_copy_mode_data
*data
= wme
->data
;
2997 struct window_copy_cmd_state cs
;
2998 enum window_copy_cmd_action action
;
2999 enum window_copy_cmd_clear clear
= WINDOW_COPY_CMD_CLEAR_NEVER
;
3000 const char *command
;
3001 u_int i
, count
= args_count(args
);
3006 command
= args_string(args
, 0);
3008 if (m
!= NULL
&& m
->valid
&& !MOUSE_WHEEL(m
->b
))
3009 window_copy_move_mouse(m
);
3019 action
= WINDOW_COPY_CMD_NOTHING
;
3020 for (i
= 0; i
< nitems(window_copy_cmd_table
); i
++) {
3021 if (strcmp(window_copy_cmd_table
[i
].command
, command
) == 0) {
3022 if (count
- 1 < window_copy_cmd_table
[i
].minargs
||
3023 count
- 1 > window_copy_cmd_table
[i
].maxargs
)
3025 clear
= window_copy_cmd_table
[i
].clear
;
3026 action
= window_copy_cmd_table
[i
].f(&cs
);
3031 if (strncmp(command
, "search-", 7) != 0 && data
->searchmark
!= NULL
) {
3032 keys
= options_get_number(wme
->wp
->window
->options
, "mode-keys");
3033 if (clear
== WINDOW_COPY_CMD_CLEAR_EMACS_ONLY
&&
3035 clear
= WINDOW_COPY_CMD_CLEAR_NEVER
;
3036 if (clear
!= WINDOW_COPY_CMD_CLEAR_NEVER
) {
3037 window_copy_clear_marks(wme
);
3038 data
->searchx
= data
->searchy
= -1;
3040 if (action
== WINDOW_COPY_CMD_NOTHING
)
3041 action
= WINDOW_COPY_CMD_REDRAW
;
3045 if (action
== WINDOW_COPY_CMD_CANCEL
)
3046 window_pane_reset_mode(wme
->wp
);
3047 else if (action
== WINDOW_COPY_CMD_REDRAW
)
3048 window_copy_redraw_screen(wme
);
3052 window_copy_scroll_to(struct window_mode_entry
*wme
, u_int px
, u_int py
,
3055 struct window_copy_mode_data
*data
= wme
->data
;
3056 struct grid
*gd
= data
->backing
->grid
;
3061 if (py
>= gd
->hsize
- data
->oy
&& py
< gd
->hsize
- data
->oy
+ gd
->sy
)
3062 data
->cy
= py
- (gd
->hsize
- data
->oy
);
3068 } else if (py
> gd
->hsize
+ gd
->sy
- gap
) {
3070 data
->cy
= py
- gd
->hsize
;
3072 offset
= py
+ gap
- gd
->sy
;
3073 data
->cy
= py
- offset
;
3075 data
->oy
= gd
->hsize
- offset
;
3078 if (!no_redraw
&& data
->searchmark
!= NULL
&& !data
->timeout
)
3079 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
3080 window_copy_update_selection(wme
, 1, 0);
3082 window_copy_redraw_screen(wme
);
3086 window_copy_search_compare(struct grid
*gd
, u_int px
, u_int py
,
3087 struct grid
*sgd
, u_int spx
, int cis
)
3089 struct grid_cell gc
, sgc
;
3090 const struct utf8_data
*ud
, *sud
;
3092 grid_get_cell(gd
, px
, py
, &gc
);
3094 grid_get_cell(sgd
, spx
, 0, &sgc
);
3097 if (ud
->size
!= sud
->size
|| ud
->width
!= sud
->width
)
3100 if (cis
&& ud
->size
== 1)
3101 return (tolower(ud
->data
[0]) == sud
->data
[0]);
3103 return (memcmp(ud
->data
, sud
->data
, ud
->size
) == 0);
3107 window_copy_search_lr(struct grid
*gd
, struct grid
*sgd
, u_int
*ppx
, u_int py
,
3108 u_int first
, u_int last
, int cis
)
3110 u_int ax
, bx
, px
, pywrap
, endline
;
3112 struct grid_line
*gl
;
3114 endline
= gd
->hsize
+ gd
->sy
- 1;
3115 for (ax
= first
; ax
< last
; ax
++) {
3116 for (bx
= 0; bx
< sgd
->sx
; bx
++) {
3120 while (px
>= gd
->sx
&& pywrap
< endline
) {
3121 gl
= grid_get_line(gd
, pywrap
);
3122 if (~gl
->flags
& GRID_LINE_WRAPPED
)
3127 /* We have run off the end of the grid. */
3130 matched
= window_copy_search_compare(gd
, px
, pywrap
,
3135 if (bx
== sgd
->sx
) {
3144 window_copy_search_rl(struct grid
*gd
,
3145 struct grid
*sgd
, u_int
*ppx
, u_int py
, u_int first
, u_int last
, int cis
)
3147 u_int ax
, bx
, px
, pywrap
, endline
;
3149 struct grid_line
*gl
;
3151 endline
= gd
->hsize
+ gd
->sy
- 1;
3152 for (ax
= last
; ax
> first
; ax
--) {
3153 for (bx
= 0; bx
< sgd
->sx
; bx
++) {
3157 while (px
>= gd
->sx
&& pywrap
< endline
) {
3158 gl
= grid_get_line(gd
, pywrap
);
3159 if (~gl
->flags
& GRID_LINE_WRAPPED
)
3164 /* We have run off the end of the grid. */
3167 matched
= window_copy_search_compare(gd
, px
, pywrap
,
3172 if (bx
== sgd
->sx
) {
3181 window_copy_search_lr_regex(struct grid
*gd
, u_int
*ppx
, u_int
*psx
, u_int py
,
3182 u_int first
, u_int last
, regex_t
*reg
)
3185 u_int endline
, foundx
, foundy
, len
, pywrap
, size
= 1;
3187 regmatch_t regmatch
;
3188 struct grid_line
*gl
;
3191 * This can happen during search if the last match was the last
3192 * character on a line.
3197 /* Set flags for regex search. */
3199 eflags
|= REG_NOTBOL
;
3201 /* Need to look at the entire string. */
3202 buf
= xmalloc(size
);
3204 buf
= window_copy_stringify(gd
, py
, first
, gd
->sx
, buf
, &size
);
3205 len
= gd
->sx
- first
;
3206 endline
= gd
->hsize
+ gd
->sy
- 1;
3208 while (buf
!= NULL
&& pywrap
<= endline
) {
3209 gl
= grid_get_line(gd
, pywrap
);
3210 if (~gl
->flags
& GRID_LINE_WRAPPED
)
3213 buf
= window_copy_stringify(gd
, pywrap
, 0, gd
->sx
, buf
, &size
);
3217 if (regexec(reg
, buf
, 1, ®match
, eflags
) == 0 &&
3218 regmatch
.rm_so
!= regmatch
.rm_eo
) {
3221 window_copy_cstrtocellpos(gd
, len
, &foundx
, &foundy
,
3222 buf
+ regmatch
.rm_so
);
3223 if (foundy
== py
&& foundx
< last
) {
3225 len
-= foundx
- first
;
3226 window_copy_cstrtocellpos(gd
, len
, &foundx
, &foundy
,
3227 buf
+ regmatch
.rm_eo
);
3229 while (foundy
> py
) {
3246 window_copy_search_rl_regex(struct grid
*gd
, u_int
*ppx
, u_int
*psx
, u_int py
,
3247 u_int first
, u_int last
, regex_t
*reg
)
3250 u_int endline
, len
, pywrap
, size
= 1;
3252 struct grid_line
*gl
;
3254 /* Set flags for regex search. */
3256 eflags
|= REG_NOTBOL
;
3258 /* Need to look at the entire string. */
3259 buf
= xmalloc(size
);
3261 buf
= window_copy_stringify(gd
, py
, first
, gd
->sx
, buf
, &size
);
3262 len
= gd
->sx
- first
;
3263 endline
= gd
->hsize
+ gd
->sy
- 1;
3265 while (buf
!= NULL
&& (pywrap
<= endline
)) {
3266 gl
= grid_get_line(gd
, pywrap
);
3267 if (~gl
->flags
& GRID_LINE_WRAPPED
)
3270 buf
= window_copy_stringify(gd
, pywrap
, 0, gd
->sx
, buf
, &size
);
3274 if (window_copy_last_regex(gd
, py
, first
, last
, len
, ppx
, psx
, buf
,
3288 window_copy_cellstring(const struct grid_line
*gl
, u_int px
, size_t *size
,
3291 static struct utf8_data ud
;
3292 struct grid_cell_entry
*gce
;
3295 if (px
>= gl
->cellsize
) {
3301 gce
= &gl
->celldata
[px
];
3302 if (gce
->flags
& GRID_FLAG_PADDING
) {
3307 if (~gce
->flags
& GRID_FLAG_EXTENDED
) {
3310 return (&gce
->data
.data
);
3313 utf8_to_data(gl
->extddata
[gce
->offset
].data
, &ud
);
3322 copy
= xmalloc(ud
.size
);
3323 memcpy(copy
, ud
.data
, ud
.size
);
3327 /* Find last match in given range. */
3329 window_copy_last_regex(struct grid
*gd
, u_int py
, u_int first
, u_int last
,
3330 u_int len
, u_int
*ppx
, u_int
*psx
, const char *buf
, const regex_t
*preg
,
3333 u_int foundx
, foundy
, oldx
, px
= 0, savepx
, savesx
= 0;
3334 regmatch_t regmatch
;
3339 while (regexec(preg
, buf
+ px
, 1, ®match
, eflags
) == 0) {
3340 if (regmatch
.rm_so
== regmatch
.rm_eo
)
3342 window_copy_cstrtocellpos(gd
, len
, &foundx
, &foundy
,
3343 buf
+ px
+ regmatch
.rm_so
);
3344 if (foundy
> py
|| foundx
>= last
)
3346 len
-= foundx
- oldx
;
3348 window_copy_cstrtocellpos(gd
, len
, &foundx
, &foundy
,
3349 buf
+ px
+ regmatch
.rm_eo
);
3350 if (foundy
> py
|| foundx
>= last
) {
3353 while (foundy
> py
) {
3360 savesx
= foundx
- savepx
;
3364 px
+= regmatch
.rm_eo
;
3378 /* Stringify line and append to input buffer. Caller frees. */
3380 window_copy_stringify(struct grid
*gd
, u_int py
, u_int first
, u_int last
,
3381 char *buf
, u_int
*size
)
3383 u_int ax
, bx
, newsize
= *size
;
3384 const struct grid_line
*gl
;
3386 size_t bufsize
= 1024, dlen
;
3389 while (bufsize
< newsize
)
3391 buf
= xrealloc(buf
, bufsize
);
3393 gl
= grid_peek_line(gd
, py
);
3395 for (ax
= first
; ax
< last
; ax
++) {
3396 d
= window_copy_cellstring(gl
, ax
, &dlen
, &allocated
);
3398 while (bufsize
< newsize
) {
3400 buf
= xrealloc(buf
, bufsize
);
3405 memcpy(buf
+ bx
, d
, dlen
);
3411 buf
[newsize
- 1] = '\0';
3417 /* Map start of C string containing UTF-8 data to grid cell position. */
3419 window_copy_cstrtocellpos(struct grid
*gd
, u_int ncells
, u_int
*ppx
, u_int
*ppy
,
3422 u_int cell
, ccell
, px
, pywrap
, pos
, len
;
3424 const struct grid_line
*gl
;
3433 /* Populate the array of cell data. */
3434 cells
= xreallocarray(NULL
, ncells
, sizeof cells
[0]);
3438 gl
= grid_peek_line(gd
, pywrap
);
3439 while (cell
< ncells
) {
3440 cells
[cell
].d
= window_copy_cellstring(gl
, px
,
3441 &cells
[cell
].dlen
, &cells
[cell
].allocated
);
3447 gl
= grid_peek_line(gd
, pywrap
);
3451 /* Locate starting cell. */
3454 while (cell
< ncells
) {
3458 while (ccell
< ncells
) {
3459 if (str
[pos
] == '\0') {
3464 dlen
= cells
[ccell
].dlen
;
3466 if (str
[pos
] != *d
) {
3472 if (dlen
> len
- pos
)
3474 if (memcmp(str
+ pos
, d
, dlen
) != 0) {
3487 /* If not found this will be one past the end. */
3490 while (px
>= gd
->sx
) {
3498 /* Free cell data. */
3499 for (cell
= 0; cell
< ncells
; cell
++) {
3500 if (cells
[cell
].allocated
)
3501 free((void *)cells
[cell
].d
);
3507 window_copy_move_left(struct screen
*s
, u_int
*fx
, u_int
*fy
, int wrapflag
)
3509 if (*fx
== 0) { /* left */
3510 if (*fy
== 0) { /* top */
3512 *fx
= screen_size_x(s
) - 1;
3513 *fy
= screen_hsize(s
) + screen_size_y(s
) - 1;
3517 *fx
= screen_size_x(s
) - 1;
3524 window_copy_move_right(struct screen
*s
, u_int
*fx
, u_int
*fy
, int wrapflag
)
3526 if (*fx
== screen_size_x(s
) - 1) { /* right */
3527 if (*fy
== screen_hsize(s
) + screen_size_y(s
) - 1) { /* bottom */
3541 window_copy_is_lowercase(const char *ptr
)
3543 while (*ptr
!= '\0') {
3544 if (*ptr
!= tolower((u_char
)*ptr
))
3552 * Handle backward wrapped regex searches with overlapping matches. In this case
3553 * find the longest overlapping match from previous wrapped lines.
3556 window_copy_search_back_overlap(struct grid
*gd
, regex_t
*preg
, u_int
*ppx
,
3557 u_int
*psx
, u_int
*ppy
, u_int endline
)
3559 u_int endx
, endy
, oldendx
, oldendy
, px
, py
, sx
;
3562 oldendx
= *ppx
+ *psx
;
3564 while (oldendx
> gd
->sx
- 1) {
3572 while (found
&& px
== 0 && py
- 1 > endline
&&
3573 grid_get_line(gd
, py
- 2)->flags
& GRID_LINE_WRAPPED
&&
3574 endx
== oldendx
&& endy
== oldendy
) {
3576 found
= window_copy_search_rl_regex(gd
, &px
, &sx
, py
- 1, 0,
3581 while (endx
> gd
->sx
- 1) {
3585 if (endx
== oldendx
&& endy
== oldendy
) {
3594 * Search for text stored in sgd starting from position fx,fy up to endline. If
3595 * found, jump to it. If cis then ignore case. The direction is 0 for searching
3596 * up, down otherwise. If wrap then go to begin/end of grid and try again if
3600 window_copy_search_jump(struct window_mode_entry
*wme
, struct grid
*gd
,
3601 struct grid
*sgd
, u_int fx
, u_int fy
, u_int endline
, int cis
, int wrap
,
3602 int direction
, int regex
)
3604 u_int i
, px
, sx
, ssize
= 1;
3605 int found
= 0, cflags
= REG_EXTENDED
;
3610 sbuf
= xmalloc(ssize
);
3612 sbuf
= window_copy_stringify(sgd
, 0, 0, sgd
->sx
, sbuf
, &ssize
);
3614 cflags
|= REG_ICASE
;
3615 if (regcomp(®
, sbuf
, cflags
) != 0) {
3623 for (i
= fy
; i
<= endline
; i
++) {
3625 found
= window_copy_search_lr_regex(gd
,
3626 &px
, &sx
, i
, fx
, gd
->sx
, ®
);
3628 found
= window_copy_search_lr(gd
, sgd
,
3629 &px
, i
, fx
, gd
->sx
, cis
);
3636 for (i
= fy
+ 1; endline
< i
; i
--) {
3638 found
= window_copy_search_rl_regex(gd
,
3639 &px
, &sx
, i
- 1, 0, fx
+ 1, ®
);
3641 window_copy_search_back_overlap(gd
,
3642 ®
, &px
, &sx
, &i
, endline
);
3645 found
= window_copy_search_rl(gd
, sgd
,
3646 &px
, i
- 1, 0, fx
+ 1, cis
);
3659 window_copy_scroll_to(wme
, px
, i
, 1);
3663 return (window_copy_search_jump(wme
, gd
, sgd
,
3664 direction
? 0 : gd
->sx
- 1,
3665 direction
? 0 : gd
->hsize
+ gd
->sy
- 1, fy
, cis
, 0,
3672 window_copy_move_after_search_mark(struct window_copy_mode_data
*data
,
3673 u_int
*fx
, u_int
*fy
, int wrapflag
)
3675 struct screen
*s
= data
->backing
;
3678 if (window_copy_search_mark_at(data
, *fx
, *fy
, &start
) == 0 &&
3679 data
->searchmark
[start
] != 0) {
3680 while (window_copy_search_mark_at(data
, *fx
, *fy
, &at
) == 0) {
3681 if (data
->searchmark
[at
] != data
->searchmark
[start
])
3683 /* Stop if not wrapping and at the end of the grid. */
3685 *fx
== screen_size_x(s
) - 1 &&
3686 *fy
== screen_hsize(s
) + screen_size_y(s
) - 1)
3689 window_copy_move_right(s
, fx
, fy
, wrapflag
);
3695 * Search in for text searchstr. If direction is 0 then search up, otherwise
3699 window_copy_search(struct window_mode_entry
*wme
, int direction
, int regex
)
3701 struct window_pane
*wp
= wme
->wp
;
3702 struct window_copy_mode_data
*data
= wme
->data
;
3703 struct screen
*s
= data
->backing
, ss
;
3704 struct screen_write_ctx ctx
;
3705 struct grid
*gd
= s
->grid
;
3706 const char *str
= data
->searchstr
;
3707 u_int at
, endline
, fx
, fy
, start
;
3708 int cis
, found
, keys
, visible_only
;
3711 if (regex
&& str
[strcspn(str
, "^$*+()?[].\\")] == '\0')
3714 data
->searchdirection
= direction
;
3719 if (data
->searchall
|| wp
->searchstr
== NULL
||
3720 wp
->searchregex
!= regex
) {
3722 data
->searchall
= 0;
3724 visible_only
= (strcmp(wp
->searchstr
, str
) == 0);
3725 if (visible_only
== 0 && data
->searchmark
!= NULL
)
3726 window_copy_clear_marks(wme
);
3727 free(wp
->searchstr
);
3728 wp
->searchstr
= xstrdup(str
);
3729 wp
->searchregex
= regex
;
3732 fy
= screen_hsize(data
->backing
) - data
->oy
+ data
->cy
;
3734 screen_init(&ss
, screen_write_strlen("%s", str
), 1, 0);
3735 screen_write_start(&ctx
, &ss
);
3736 screen_write_nputs(&ctx
, -1, &grid_default_cell
, "%s", str
);
3737 screen_write_stop(&ctx
);
3739 wrapflag
= options_get_number(wp
->window
->options
, "wrap-search");
3740 cis
= window_copy_is_lowercase(str
);
3742 keys
= options_get_number(wp
->window
->options
, "mode-keys");
3746 * Behave according to mode-keys. If it is emacs, search forward
3747 * leaves the cursor after the match. If it is vi, the cursor
3748 * remains at the beginning of the match, regardless of
3749 * direction, which means that we need to start the next search
3750 * after the term the cursor is currently on when searching
3753 if (keys
== MODEKEY_VI
) {
3754 if (data
->searchmark
!= NULL
)
3755 window_copy_move_after_search_mark(data
, &fx
,
3759 * When there are no search marks, start the
3760 * search after the current cursor position.
3762 window_copy_move_right(s
, &fx
, &fy
, wrapflag
);
3765 endline
= gd
->hsize
+ gd
->sy
- 1;
3767 window_copy_move_left(s
, &fx
, &fy
, wrapflag
);
3771 found
= window_copy_search_jump(wme
, gd
, ss
.grid
, fx
, fy
, endline
, cis
,
3772 wrapflag
, direction
, regex
);
3774 window_copy_search_marks(wme
, &ss
, regex
, visible_only
);
3776 fy
= screen_hsize(data
->backing
) - data
->oy
+ data
->cy
;
3779 * When searching forward, if the cursor is not at the beginning
3780 * of the mark, search again.
3783 window_copy_search_mark_at(data
, fx
, fy
, &at
) == 0 &&
3785 data
->searchmark
!= NULL
&&
3786 data
->searchmark
[at
] == data
->searchmark
[at
- 1]) {
3787 window_copy_move_after_search_mark(data
, &fx
, &fy
,
3789 window_copy_search_jump(wme
, gd
, ss
.grid
, fx
,
3790 fy
, endline
, cis
, wrapflag
, direction
,
3793 fy
= screen_hsize(data
->backing
) - data
->oy
+ data
->cy
;
3798 * When in Emacs mode, position the cursor just after
3801 if (keys
== MODEKEY_EMACS
) {
3802 window_copy_move_after_search_mark(data
, &fx
,
3805 data
->cy
= fy
- screen_hsize(data
->backing
) +
3810 * When searching backward, position the cursor at the
3811 * beginning of the mark.
3813 if (window_copy_search_mark_at(data
, fx
, fy
,
3815 while (window_copy_search_mark_at(data
, fx
, fy
,
3817 data
->searchmark
!= NULL
&&
3818 data
->searchmark
[at
] ==
3819 data
->searchmark
[start
]) {
3822 screen_hsize(data
->backing
) +
3827 window_copy_move_left(s
, &fx
, &fy
, 0);
3832 window_copy_redraw_screen(wme
);
3839 window_copy_visible_lines(struct window_copy_mode_data
*data
, u_int
*start
,
3842 struct grid
*gd
= data
->backing
->grid
;
3843 const struct grid_line
*gl
;
3845 for (*start
= gd
->hsize
- data
->oy
; *start
> 0; (*start
)--) {
3846 gl
= grid_peek_line(gd
, (*start
) - 1);
3847 if (~gl
->flags
& GRID_LINE_WRAPPED
)
3850 *end
= gd
->hsize
- data
->oy
+ gd
->sy
;
3854 window_copy_search_mark_at(struct window_copy_mode_data
*data
, u_int px
,
3855 u_int py
, u_int
*at
)
3857 struct screen
*s
= data
->backing
;
3858 struct grid
*gd
= s
->grid
;
3860 if (py
< gd
->hsize
- data
->oy
)
3862 if (py
> gd
->hsize
- data
->oy
+ gd
->sy
- 1)
3864 *at
= ((py
- (gd
->hsize
- data
->oy
)) * gd
->sx
) + px
;
3869 window_copy_search_marks(struct window_mode_entry
*wme
, struct screen
*ssp
,
3870 int regex
, int visible_only
)
3872 struct window_copy_mode_data
*data
= wme
->data
;
3873 struct screen
*s
= data
->backing
, ss
;
3874 struct screen_write_ctx ctx
;
3875 struct grid
*gd
= s
->grid
;
3876 int found
, cis
, stopped
= 0;
3877 int cflags
= REG_EXTENDED
;
3878 u_int px
, py
, i
, b
, nfound
= 0, width
;
3879 u_int ssize
= 1, start
, end
;
3882 uint64_t stop
= 0, tstart
, t
;
3885 width
= screen_write_strlen("%s", data
->searchstr
);
3886 screen_init(&ss
, width
, 1, 0);
3887 screen_write_start(&ctx
, &ss
);
3888 screen_write_nputs(&ctx
, -1, &grid_default_cell
, "%s",
3890 screen_write_stop(&ctx
);
3893 width
= screen_size_x(ssp
);
3895 cis
= window_copy_is_lowercase(data
->searchstr
);
3898 sbuf
= xmalloc(ssize
);
3900 sbuf
= window_copy_stringify(ssp
->grid
, 0, 0, ssp
->grid
->sx
,
3903 cflags
|= REG_ICASE
;
3904 if (regcomp(®
, sbuf
, cflags
) != 0) {
3910 tstart
= get_timer();
3913 window_copy_visible_lines(data
, &start
, &end
);
3916 end
= gd
->hsize
+ gd
->sy
;
3917 stop
= get_timer() + WINDOW_COPY_SEARCH_ALL_TIMEOUT
;
3921 free(data
->searchmark
);
3922 data
->searchmark
= xcalloc(gd
->sx
, gd
->sy
);
3923 data
->searchgen
= 1;
3925 for (py
= start
; py
< end
; py
++) {
3929 found
= window_copy_search_lr_regex(gd
,
3930 &px
, &width
, py
, px
, gd
->sx
, ®
);
3934 found
= window_copy_search_lr(gd
, ssp
->grid
,
3935 &px
, py
, px
, gd
->sx
, cis
);
3941 if (window_copy_search_mark_at(data
, px
, py
, &b
) == 0) {
3942 if (b
+ width
> gd
->sx
* gd
->sy
)
3943 width
= (gd
->sx
* gd
->sy
) - b
;
3944 for (i
= b
; i
< b
+ width
; i
++) {
3945 if (data
->searchmark
[i
] != 0)
3947 data
->searchmark
[i
] = data
->searchgen
;
3949 if (data
->searchgen
== UCHAR_MAX
)
3950 data
->searchgen
= 1;
3958 if (t
- tstart
> WINDOW_COPY_SEARCH_TIMEOUT
) {
3962 if (stop
!= 0 && t
> stop
) {
3967 if (data
->timeout
) {
3968 window_copy_clear_marks(wme
);
3972 if (stopped
&& stop
!= 0) {
3973 /* Try again but just the visible context. */
3974 window_copy_visible_lines(data
, &start
, &end
);
3979 if (!visible_only
) {
3982 data
->searchcount
= 1000;
3983 else if (nfound
> 100)
3984 data
->searchcount
= 100;
3985 else if (nfound
> 10)
3986 data
->searchcount
= 10;
3988 data
->searchcount
= -1;
3989 data
->searchmore
= 1;
3991 data
->searchcount
= nfound
;
3992 data
->searchmore
= 0;
4005 window_copy_clear_marks(struct window_mode_entry
*wme
)
4007 struct window_copy_mode_data
*data
= wme
->data
;
4009 free(data
->searchmark
);
4010 data
->searchmark
= NULL
;
4014 window_copy_search_up(struct window_mode_entry
*wme
, int regex
)
4016 return (window_copy_search(wme
, 0, regex
));
4020 window_copy_search_down(struct window_mode_entry
*wme
, int regex
)
4022 return (window_copy_search(wme
, 1, regex
));
4026 window_copy_goto_line(struct window_mode_entry
*wme
, const char *linestr
)
4028 struct window_copy_mode_data
*data
= wme
->data
;
4032 lineno
= strtonum(linestr
, -1, INT_MAX
, &errstr
);
4035 if (lineno
< 0 || (u_int
)lineno
> screen_hsize(data
->backing
))
4036 lineno
= screen_hsize(data
->backing
);
4039 window_copy_update_selection(wme
, 1, 0);
4040 window_copy_redraw_screen(wme
);
4044 window_copy_match_start_end(struct window_copy_mode_data
*data
, u_int at
,
4045 u_int
*start
, u_int
*end
)
4047 struct grid
*gd
= data
->backing
->grid
;
4048 u_int last
= (gd
->sy
* gd
->sx
) - 1;
4049 u_char mark
= data
->searchmark
[at
];
4052 while (*start
!= 0 && data
->searchmark
[*start
] == mark
)
4054 if (data
->searchmark
[*start
] != mark
)
4056 while (*end
!= last
&& data
->searchmark
[*end
] == mark
)
4058 if (data
->searchmark
[*end
] != mark
)
4063 window_copy_match_at_cursor(struct window_copy_mode_data
*data
)
4065 struct grid
*gd
= data
->backing
->grid
;
4066 struct grid_cell gc
;
4067 u_int at
, start
, end
, cy
, px
, py
;
4068 u_int sx
= screen_size_x(data
->backing
);
4072 if (data
->searchmark
== NULL
)
4075 cy
= screen_hsize(data
->backing
) - data
->oy
+ data
->cy
;
4076 if (window_copy_search_mark_at(data
, data
->cx
, cy
, &at
) != 0)
4078 if (data
->searchmark
[at
] == 0) {
4079 /* Allow one position after the match. */
4080 if (at
== 0 || data
->searchmark
[--at
] == 0)
4083 window_copy_match_start_end(data
, at
, &start
, &end
);
4086 * Cells will not be set in the marked array unless they are valid text
4087 * and wrapping will be taken care of, so we can just copy.
4089 for (at
= start
; at
<= end
; at
++) {
4091 px
= at
- (py
* sx
);
4093 grid_get_cell(gd
, px
, gd
->hsize
+ py
- data
->oy
, &gc
);
4094 buf
= xrealloc(buf
, len
+ gc
.data
.size
+ 1);
4095 memcpy(buf
+ len
, gc
.data
.data
, gc
.data
.size
);
4096 len
+= gc
.data
.size
;
4104 window_copy_update_style(struct window_mode_entry
*wme
, u_int fx
, u_int fy
,
4105 struct grid_cell
*gc
, const struct grid_cell
*mgc
,
4106 const struct grid_cell
*cgc
, const struct grid_cell
*mkgc
)
4108 struct window_pane
*wp
= wme
->wp
;
4109 struct window_copy_mode_data
*data
= wme
->data
;
4110 u_int mark
, start
, end
, cy
, cursor
, current
;
4111 int inv
= 0, found
= 0;
4114 if (data
->showmark
&& fy
== data
->my
) {
4115 gc
->attr
= mkgc
->attr
;
4128 if (data
->searchmark
== NULL
)
4131 if (window_copy_search_mark_at(data
, fx
, fy
, ¤t
) != 0)
4133 mark
= data
->searchmark
[current
];
4137 cy
= screen_hsize(data
->backing
) - data
->oy
+ data
->cy
;
4138 if (window_copy_search_mark_at(data
, data
->cx
, cy
, &cursor
) == 0) {
4139 keys
= options_get_number(wp
->window
->options
, "mode-keys");
4141 keys
== MODEKEY_EMACS
&&
4142 data
->searchdirection
) {
4143 if (data
->searchmark
[cursor
- 1] == mark
) {
4147 } else if (data
->searchmark
[cursor
] == mark
)
4150 window_copy_match_start_end(data
, cursor
, &start
, &end
);
4151 if (current
>= start
&& current
<= end
) {
4152 gc
->attr
= cgc
->attr
;
4166 gc
->attr
= mgc
->attr
;
4178 window_copy_write_one(struct window_mode_entry
*wme
,
4179 struct screen_write_ctx
*ctx
, u_int py
, u_int fy
, u_int nx
,
4180 const struct grid_cell
*mgc
, const struct grid_cell
*cgc
,
4181 const struct grid_cell
*mkgc
)
4183 struct window_copy_mode_data
*data
= wme
->data
;
4184 struct grid
*gd
= data
->backing
->grid
;
4185 struct grid_cell gc
;
4188 screen_write_cursormove(ctx
, 0, py
, 0);
4189 for (fx
= 0; fx
< nx
; fx
++) {
4190 grid_get_cell(gd
, fx
, fy
, &gc
);
4191 if (fx
+ gc
.data
.width
<= nx
) {
4192 window_copy_update_style(wme
, fx
, fy
, &gc
, mgc
, cgc
,
4194 screen_write_cell(ctx
, &gc
);
4200 window_copy_write_line(struct window_mode_entry
*wme
,
4201 struct screen_write_ctx
*ctx
, u_int py
)
4203 struct window_pane
*wp
= wme
->wp
;
4204 struct window_copy_mode_data
*data
= wme
->data
;
4205 struct screen
*s
= &data
->screen
;
4206 struct options
*oo
= wp
->window
->options
;
4207 struct grid_line
*gl
;
4208 struct grid_cell gc
, mgc
, cgc
, mkgc
;
4209 char hdr
[512], tmp
[256], *t
;
4211 u_int hsize
= screen_hsize(data
->backing
);
4213 style_apply(&gc
, oo
, "mode-style", NULL
);
4214 gc
.flags
|= GRID_FLAG_NOPALETTE
;
4215 style_apply(&mgc
, oo
, "copy-mode-match-style", NULL
);
4216 mgc
.flags
|= GRID_FLAG_NOPALETTE
;
4217 style_apply(&cgc
, oo
, "copy-mode-current-match-style", NULL
);
4218 cgc
.flags
|= GRID_FLAG_NOPALETTE
;
4219 style_apply(&mkgc
, oo
, "copy-mode-mark-style", NULL
);
4220 mkgc
.flags
|= GRID_FLAG_NOPALETTE
;
4222 if (py
== 0 && s
->rupper
< s
->rlower
&& !data
->hide_position
) {
4223 gl
= grid_get_line(data
->backing
->grid
, hsize
- data
->oy
);
4225 xsnprintf(tmp
, sizeof tmp
, "[%u/%u]", data
->oy
, hsize
);
4227 t
= format_pretty_time(gl
->time
, 1);
4228 xsnprintf(tmp
, sizeof tmp
, "%s [%u/%u]", t
, data
->oy
,
4233 if (data
->searchmark
== NULL
) {
4234 if (data
->timeout
) {
4235 size
= xsnprintf(hdr
, sizeof hdr
,
4236 "(timed out) %s", tmp
);
4238 size
= xsnprintf(hdr
, sizeof hdr
, "%s", tmp
);
4240 if (data
->searchcount
== -1)
4241 size
= xsnprintf(hdr
, sizeof hdr
, "%s", tmp
);
4243 size
= xsnprintf(hdr
, sizeof hdr
,
4244 "(%d%s results) %s", data
->searchcount
,
4245 data
->searchmore
? "+" : "", tmp
);
4248 if (size
> screen_size_x(s
))
4249 size
= screen_size_x(s
);
4250 screen_write_cursormove(ctx
, screen_size_x(s
) - size
, 0, 0);
4251 screen_write_puts(ctx
, &gc
, "%s", hdr
);
4255 if (size
< screen_size_x(s
)) {
4256 window_copy_write_one(wme
, ctx
, py
, hsize
- data
->oy
+ py
,
4257 screen_size_x(s
) - size
, &mgc
, &cgc
, &mkgc
);
4260 if (py
== data
->cy
&& data
->cx
== screen_size_x(s
)) {
4261 screen_write_cursormove(ctx
, screen_size_x(s
) - 1, py
, 0);
4262 screen_write_putc(ctx
, &grid_default_cell
, '$');
4267 window_copy_write_lines(struct window_mode_entry
*wme
,
4268 struct screen_write_ctx
*ctx
, u_int py
, u_int ny
)
4272 for (yy
= py
; yy
< py
+ ny
; yy
++)
4273 window_copy_write_line(wme
, ctx
, py
);
4277 window_copy_redraw_selection(struct window_mode_entry
*wme
, u_int old_y
)
4279 struct window_copy_mode_data
*data
= wme
->data
;
4280 struct grid
*gd
= data
->backing
->grid
;
4281 u_int new_y
, start
, end
;
4284 if (old_y
<= new_y
) {
4293 * In word selection mode the first word on the line below the cursor
4294 * might be selected, so add this line to the redraw area.
4296 if (data
->selflag
== SEL_WORD
) {
4297 /* Last grid line in data coordinates. */
4298 if (end
< gd
->sy
+ data
->oy
- 1)
4301 window_copy_redraw_lines(wme
, start
, end
- start
+ 1);
4305 window_copy_redraw_lines(struct window_mode_entry
*wme
, u_int py
, u_int ny
)
4307 struct window_pane
*wp
= wme
->wp
;
4308 struct window_copy_mode_data
*data
= wme
->data
;
4309 struct screen_write_ctx ctx
;
4312 screen_write_start_pane(&ctx
, wp
, NULL
);
4313 for (i
= py
; i
< py
+ ny
; i
++)
4314 window_copy_write_line(wme
, &ctx
, i
);
4315 screen_write_cursormove(&ctx
, data
->cx
, data
->cy
, 0);
4316 screen_write_stop(&ctx
);
4320 window_copy_redraw_screen(struct window_mode_entry
*wme
)
4322 struct window_copy_mode_data
*data
= wme
->data
;
4324 window_copy_redraw_lines(wme
, 0, screen_size_y(&data
->screen
));
4328 window_copy_synchronize_cursor_end(struct window_mode_entry
*wme
, int begin
,
4331 struct window_copy_mode_data
*data
= wme
->data
;
4335 yy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
4336 switch (data
->selflag
) {
4341 if (data
->dy
> yy
|| (data
->dy
== yy
&& data
->dx
> xx
)) {
4342 /* Right to left selection. */
4343 window_copy_cursor_previous_word_pos(wme
,
4344 data
->separators
, &xx
, &yy
);
4347 /* Reset the end. */
4348 data
->endselx
= data
->endselrx
;
4349 data
->endsely
= data
->endselry
;
4351 /* Left to right selection. */
4352 if (xx
>= window_copy_find_length(wme
, yy
) ||
4353 !window_copy_in_set(wme
, xx
+ 1, yy
, WHITESPACE
)) {
4354 window_copy_cursor_next_word_end_pos(wme
,
4355 data
->separators
, &xx
, &yy
);
4358 /* Reset the start. */
4359 data
->selx
= data
->selrx
;
4360 data
->sely
= data
->selry
;
4367 if (data
->dy
> yy
) {
4368 /* Right to left selection. */
4372 /* Reset the end. */
4373 data
->endselx
= data
->endselrx
;
4374 data
->endsely
= data
->endselry
;
4376 /* Left to right selection. */
4377 if (yy
< data
->endselry
)
4378 yy
= data
->endselry
;
4379 xx
= window_copy_find_length(wme
, yy
);
4381 /* Reset the start. */
4382 data
->selx
= data
->selrx
;
4383 data
->sely
= data
->selry
;
4399 window_copy_synchronize_cursor(struct window_mode_entry
*wme
, int no_reset
)
4401 struct window_copy_mode_data
*data
= wme
->data
;
4403 switch (data
->cursordrag
) {
4404 case CURSORDRAG_ENDSEL
:
4405 window_copy_synchronize_cursor_end(wme
, 0, no_reset
);
4407 case CURSORDRAG_SEL
:
4408 window_copy_synchronize_cursor_end(wme
, 1, no_reset
);
4410 case CURSORDRAG_NONE
:
4416 window_copy_update_cursor(struct window_mode_entry
*wme
, u_int cx
, u_int cy
)
4418 struct window_pane
*wp
= wme
->wp
;
4419 struct window_copy_mode_data
*data
= wme
->data
;
4420 struct screen
*s
= &data
->screen
;
4421 struct screen_write_ctx ctx
;
4422 u_int old_cx
, old_cy
;
4424 old_cx
= data
->cx
; old_cy
= data
->cy
;
4425 data
->cx
= cx
; data
->cy
= cy
;
4426 if (old_cx
== screen_size_x(s
))
4427 window_copy_redraw_lines(wme
, old_cy
, 1);
4428 if (data
->cx
== screen_size_x(s
))
4429 window_copy_redraw_lines(wme
, data
->cy
, 1);
4431 screen_write_start_pane(&ctx
, wp
, NULL
);
4432 screen_write_cursormove(&ctx
, data
->cx
, data
->cy
, 0);
4433 screen_write_stop(&ctx
);
4438 window_copy_start_selection(struct window_mode_entry
*wme
)
4440 struct window_copy_mode_data
*data
= wme
->data
;
4442 data
->selx
= data
->cx
;
4443 data
->sely
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
4445 data
->endselx
= data
->selx
;
4446 data
->endsely
= data
->sely
;
4448 data
->cursordrag
= CURSORDRAG_ENDSEL
;
4450 window_copy_set_selection(wme
, 1, 0);
4454 window_copy_adjust_selection(struct window_mode_entry
*wme
, u_int
*selx
,
4457 struct window_copy_mode_data
*data
= wme
->data
;
4458 struct screen
*s
= &data
->screen
;
4465 ty
= screen_hsize(data
->backing
) - data
->oy
;
4467 relpos
= WINDOW_COPY_REL_POS_ABOVE
;
4468 if (!data
->rectflag
)
4471 } else if (sy
> ty
+ screen_size_y(s
) - 1) {
4472 relpos
= WINDOW_COPY_REL_POS_BELOW
;
4473 if (!data
->rectflag
)
4474 sx
= screen_size_x(s
) - 1;
4475 sy
= screen_size_y(s
) - 1;
4477 relpos
= WINDOW_COPY_REL_POS_ON_SCREEN
;
4487 window_copy_update_selection(struct window_mode_entry
*wme
, int may_redraw
,
4490 struct window_copy_mode_data
*data
= wme
->data
;
4491 struct screen
*s
= &data
->screen
;
4493 if (s
->sel
== NULL
&& data
->lineflag
== LINE_SEL_NONE
)
4495 return (window_copy_set_selection(wme
, may_redraw
, no_reset
));
4499 window_copy_set_selection(struct window_mode_entry
*wme
, int may_redraw
,
4502 struct window_pane
*wp
= wme
->wp
;
4503 struct window_copy_mode_data
*data
= wme
->data
;
4504 struct screen
*s
= &data
->screen
;
4505 struct options
*oo
= wp
->window
->options
;
4506 struct grid_cell gc
;
4507 u_int sx
, sy
, cy
, endsx
, endsy
;
4508 int startrelpos
, endrelpos
;
4510 window_copy_synchronize_cursor(wme
, no_reset
);
4512 /* Adjust the selection. */
4515 startrelpos
= window_copy_adjust_selection(wme
, &sx
, &sy
);
4517 /* Adjust the end of selection. */
4518 endsx
= data
->endselx
;
4519 endsy
= data
->endsely
;
4520 endrelpos
= window_copy_adjust_selection(wme
, &endsx
, &endsy
);
4522 /* Selection is outside of the current screen */
4523 if (startrelpos
== endrelpos
&&
4524 startrelpos
!= WINDOW_COPY_REL_POS_ON_SCREEN
) {
4525 screen_hide_selection(s
);
4529 /* Set colours and selection. */
4530 style_apply(&gc
, oo
, "mode-style", NULL
);
4531 gc
.flags
|= GRID_FLAG_NOPALETTE
;
4532 screen_set_selection(s
, sx
, sy
, endsx
, endsy
, data
->rectflag
,
4533 data
->modekeys
, &gc
);
4535 if (data
->rectflag
&& may_redraw
) {
4537 * Can't rely on the caller to redraw the right lines for
4538 * rectangle selection - find the highest line and the number
4539 * of lines, and redraw just past that in both directions
4542 if (data
->cursordrag
== CURSORDRAG_ENDSEL
) {
4544 window_copy_redraw_lines(wme
, sy
, cy
- sy
+ 1);
4546 window_copy_redraw_lines(wme
, cy
, sy
- cy
+ 1);
4549 window_copy_redraw_lines(wme
, endsy
,
4552 window_copy_redraw_lines(wme
, cy
,
4562 window_copy_get_selection(struct window_mode_entry
*wme
, size_t *len
)
4564 struct window_pane
*wp
= wme
->wp
;
4565 struct window_copy_mode_data
*data
= wme
->data
;
4566 struct screen
*s
= &data
->screen
;
4569 u_int i
, xx
, yy
, sx
, sy
, ex
, ey
, ey_last
;
4570 u_int firstsx
, lastex
, restex
, restsx
, selx
;
4573 if (data
->screen
.sel
== NULL
&& data
->lineflag
== LINE_SEL_NONE
) {
4574 buf
= window_copy_match_at_cursor(data
);
4588 * The selection extends from selx,sely to (adjusted) cx,cy on
4592 /* Find start and end. */
4595 if (yy
< data
->sely
|| (yy
== data
->sely
&& xx
< data
->selx
)) {
4597 ex
= data
->selx
; ey
= data
->sely
;
4599 sx
= data
->selx
; sy
= data
->sely
;
4603 /* Trim ex to end of line. */
4604 ey_last
= window_copy_find_length(wme
, ey
);
4609 * Deal with rectangle-copy if necessary; four situations: start of
4610 * first line (firstsx), end of last line (lastex), start (restsx) and
4611 * end (restex) of all other lines.
4613 xx
= screen_size_x(s
);
4616 * Behave according to mode-keys. If it is emacs, copy like emacs,
4617 * keeping the top-left-most character, and dropping the
4618 * bottom-right-most, regardless of copy direction. If it is vi, also
4619 * keep bottom-right-most character.
4621 keys
= options_get_number(wp
->window
->options
, "mode-keys");
4622 if (data
->rectflag
) {
4624 * Need to ignore the column with the cursor in it, which for
4625 * rectangular copy means knowing which side the cursor is on.
4627 if (data
->cursordrag
== CURSORDRAG_ENDSEL
)
4630 selx
= data
->endselx
;
4631 if (selx
< data
->cx
) {
4632 /* Selection start is on the left. */
4633 if (keys
== MODEKEY_EMACS
) {
4638 lastex
= data
->cx
+ 1;
4639 restex
= data
->cx
+ 1;
4644 /* Cursor is on the left. */
4651 if (keys
== MODEKEY_EMACS
)
4660 /* Copy the lines. */
4661 for (i
= sy
; i
<= ey
; i
++) {
4662 window_copy_copy_line(wme
, &buf
, &off
, i
,
4663 (i
== sy
? firstsx
: restsx
),
4664 (i
== ey
? lastex
: restex
));
4667 /* Don't bother if no data. */
4673 /* Remove final \n (unless at end in vi mode). */
4674 if (keys
== MODEKEY_EMACS
|| lastex
<= ey_last
) {
4675 if (~grid_get_line(data
->backing
->grid
, ey
)->flags
&
4676 GRID_LINE_WRAPPED
|| lastex
!= ey_last
)
4684 window_copy_copy_buffer(struct window_mode_entry
*wme
, const char *prefix
,
4685 void *buf
, size_t len
)
4687 struct window_pane
*wp
= wme
->wp
;
4688 struct screen_write_ctx ctx
;
4690 if (options_get_number(global_options
, "set-clipboard") != 0) {
4691 screen_write_start_pane(&ctx
, wp
, NULL
);
4692 screen_write_setselection(&ctx
, "", buf
, len
);
4693 screen_write_stop(&ctx
);
4694 notify_pane("pane-set-clipboard", wp
);
4697 paste_add(prefix
, buf
, len
);
4701 window_copy_pipe_run(struct window_mode_entry
*wme
, struct session
*s
,
4702 const char *cmd
, size_t *len
)
4707 buf
= window_copy_get_selection(wme
, len
);
4708 if (cmd
== NULL
|| *cmd
== '\0')
4709 cmd
= options_get_string(global_options
, "copy-command");
4710 if (cmd
!= NULL
&& *cmd
!= '\0') {
4711 job
= job_run(cmd
, 0, NULL
, NULL
, s
, NULL
, NULL
, NULL
, NULL
,
4712 NULL
, JOB_NOWAIT
, -1, -1);
4713 bufferevent_write(job_get_event(job
), buf
, *len
);
4719 window_copy_pipe(struct window_mode_entry
*wme
, struct session
*s
,
4724 window_copy_pipe_run(wme
, s
, cmd
, &len
);
4728 window_copy_copy_pipe(struct window_mode_entry
*wme
, struct session
*s
,
4729 const char *prefix
, const char *cmd
)
4734 buf
= window_copy_pipe_run(wme
, s
, cmd
, &len
);
4736 window_copy_copy_buffer(wme
, prefix
, buf
, len
);
4740 window_copy_copy_selection(struct window_mode_entry
*wme
, const char *prefix
)
4745 buf
= window_copy_get_selection(wme
, &len
);
4747 window_copy_copy_buffer(wme
, prefix
, buf
, len
);
4751 window_copy_append_selection(struct window_mode_entry
*wme
)
4753 struct window_pane
*wp
= wme
->wp
;
4755 struct paste_buffer
*pb
;
4756 const char *bufdata
, *bufname
= NULL
;
4757 size_t len
, bufsize
;
4758 struct screen_write_ctx ctx
;
4760 buf
= window_copy_get_selection(wme
, &len
);
4764 if (options_get_number(global_options
, "set-clipboard") != 0) {
4765 screen_write_start_pane(&ctx
, wp
, NULL
);
4766 screen_write_setselection(&ctx
, "", buf
, len
);
4767 screen_write_stop(&ctx
);
4768 notify_pane("pane-set-clipboard", wp
);
4771 pb
= paste_get_top(&bufname
);
4773 bufdata
= paste_buffer_data(pb
, &bufsize
);
4774 buf
= xrealloc(buf
, len
+ bufsize
);
4775 memmove(buf
+ bufsize
, buf
, len
);
4776 memcpy(buf
, bufdata
, bufsize
);
4779 if (paste_set(buf
, len
, bufname
, NULL
) != 0)
4784 window_copy_copy_line(struct window_mode_entry
*wme
, char **buf
, size_t *off
,
4785 u_int sy
, u_int sx
, u_int ex
)
4787 struct window_copy_mode_data
*data
= wme
->data
;
4788 struct grid
*gd
= data
->backing
->grid
;
4789 struct grid_cell gc
;
4790 struct grid_line
*gl
;
4791 struct utf8_data ud
;
4792 u_int i
, xx
, wrapped
= 0;
4799 * Work out if the line was wrapped at the screen edge and all of it is
4802 gl
= grid_get_line(gd
, sy
);
4803 if (gl
->flags
& GRID_LINE_WRAPPED
&& gl
->cellsize
<= gd
->sx
)
4806 /* If the line was wrapped, don't strip spaces (use the full length). */
4810 xx
= window_copy_find_length(wme
, sy
);
4817 for (i
= sx
; i
< ex
; i
++) {
4818 grid_get_cell(gd
, i
, sy
, &gc
);
4819 if (gc
.flags
& GRID_FLAG_PADDING
)
4821 utf8_copy(&ud
, &gc
.data
);
4822 if (ud
.size
== 1 && (gc
.attr
& GRID_ATTR_CHARSET
)) {
4823 s
= tty_acs_get(NULL
, ud
.data
[0]);
4824 if (s
!= NULL
&& strlen(s
) <= sizeof ud
.data
) {
4825 ud
.size
= strlen(s
);
4826 memcpy(ud
.data
, s
, ud
.size
);
4830 *buf
= xrealloc(*buf
, (*off
) + ud
.size
);
4831 memcpy(*buf
+ *off
, ud
.data
, ud
.size
);
4836 /* Only add a newline if the line wasn't wrapped. */
4837 if (!wrapped
|| ex
!= xx
) {
4838 *buf
= xrealloc(*buf
, (*off
) + 1);
4839 (*buf
)[(*off
)++] = '\n';
4844 window_copy_clear_selection(struct window_mode_entry
*wme
)
4846 struct window_copy_mode_data
*data
= wme
->data
;
4849 screen_clear_selection(&data
->screen
);
4851 data
->cursordrag
= CURSORDRAG_NONE
;
4852 data
->lineflag
= LINE_SEL_NONE
;
4853 data
->selflag
= SEL_CHAR
;
4855 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
4856 px
= window_copy_find_length(wme
, py
);
4858 window_copy_update_cursor(wme
, px
, data
->cy
);
4862 window_copy_in_set(struct window_mode_entry
*wme
, u_int px
, u_int py
,
4865 struct window_copy_mode_data
*data
= wme
->data
;
4866 struct grid_cell gc
;
4868 grid_get_cell(data
->backing
->grid
, px
, py
, &gc
);
4869 if (gc
.flags
& GRID_FLAG_PADDING
)
4871 return (utf8_cstrhas(set
, &gc
.data
));
4875 window_copy_find_length(struct window_mode_entry
*wme
, u_int py
)
4877 struct window_copy_mode_data
*data
= wme
->data
;
4879 return (grid_line_length(data
->backing
->grid
, py
));
4883 window_copy_cursor_start_of_line(struct window_mode_entry
*wme
)
4885 struct window_copy_mode_data
*data
= wme
->data
;
4886 struct screen
*back_s
= data
->backing
;
4887 struct grid_reader gr
;
4888 u_int px
, py
, oldy
, hsize
;
4891 hsize
= screen_hsize(back_s
);
4892 py
= hsize
+ data
->cy
- data
->oy
;
4895 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
4896 grid_reader_cursor_start_of_line(&gr
, 1);
4897 grid_reader_get_cursor(&gr
, &px
, &py
);
4898 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
, py
);
4902 window_copy_cursor_back_to_indentation(struct window_mode_entry
*wme
)
4904 struct window_copy_mode_data
*data
= wme
->data
;
4905 struct screen
*back_s
= data
->backing
;
4906 struct grid_reader gr
;
4907 u_int px
, py
, oldy
, hsize
;
4910 hsize
= screen_hsize(back_s
);
4911 py
= hsize
+ data
->cy
- data
->oy
;
4914 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
4915 grid_reader_cursor_back_to_indentation(&gr
);
4916 grid_reader_get_cursor(&gr
, &px
, &py
);
4917 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
, py
);
4921 window_copy_cursor_end_of_line(struct window_mode_entry
*wme
)
4923 struct window_copy_mode_data
*data
= wme
->data
;
4924 struct screen
*back_s
= data
->backing
;
4925 struct grid_reader gr
;
4926 u_int px
, py
, oldy
, hsize
;
4929 hsize
= screen_hsize(back_s
);
4930 py
= hsize
+ data
->cy
- data
->oy
;
4933 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
4934 if (data
->screen
.sel
!= NULL
&& data
->rectflag
)
4935 grid_reader_cursor_end_of_line(&gr
, 1, 1);
4937 grid_reader_cursor_end_of_line(&gr
, 1, 0);
4938 grid_reader_get_cursor(&gr
, &px
, &py
);
4939 window_copy_acquire_cursor_down(wme
, hsize
, screen_size_y(back_s
),
4940 data
->oy
, oldy
, px
, py
, 0);
4944 window_copy_other_end(struct window_mode_entry
*wme
)
4946 struct window_copy_mode_data
*data
= wme
->data
;
4947 struct screen
*s
= &data
->screen
;
4948 u_int selx
, sely
, cy
, yy
, hsize
;
4950 if (s
->sel
== NULL
&& data
->lineflag
== LINE_SEL_NONE
)
4953 if (data
->lineflag
== LINE_SEL_LEFT_RIGHT
)
4954 data
->lineflag
= LINE_SEL_RIGHT_LEFT
;
4955 else if (data
->lineflag
== LINE_SEL_RIGHT_LEFT
)
4956 data
->lineflag
= LINE_SEL_LEFT_RIGHT
;
4958 switch (data
->cursordrag
) {
4959 case CURSORDRAG_NONE
:
4960 case CURSORDRAG_SEL
:
4961 data
->cursordrag
= CURSORDRAG_ENDSEL
;
4963 case CURSORDRAG_ENDSEL
:
4964 data
->cursordrag
= CURSORDRAG_SEL
;
4968 selx
= data
->endselx
;
4969 sely
= data
->endsely
;
4970 if (data
->cursordrag
== CURSORDRAG_SEL
) {
4976 yy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
4980 hsize
= screen_hsize(data
->backing
);
4981 if (sely
< hsize
- data
->oy
) { /* above */
4982 data
->oy
= hsize
- sely
;
4984 } else if (sely
> hsize
- data
->oy
+ screen_size_y(s
)) { /* below */
4985 data
->oy
= hsize
- sely
+ screen_size_y(s
) - 1;
4986 data
->cy
= screen_size_y(s
) - 1;
4988 data
->cy
= cy
+ sely
- yy
;
4990 window_copy_update_selection(wme
, 1, 1);
4991 window_copy_redraw_screen(wme
);
4995 window_copy_cursor_left(struct window_mode_entry
*wme
)
4997 struct window_copy_mode_data
*data
= wme
->data
;
4998 struct screen
*back_s
= data
->backing
;
4999 struct grid_reader gr
;
5000 u_int px
, py
, oldy
, hsize
;
5003 hsize
= screen_hsize(back_s
);
5004 py
= hsize
+ data
->cy
- data
->oy
;
5007 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5008 grid_reader_cursor_left(&gr
, 1);
5009 grid_reader_get_cursor(&gr
, &px
, &py
);
5010 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
, py
);
5014 window_copy_cursor_right(struct window_mode_entry
*wme
, int all
)
5016 struct window_copy_mode_data
*data
= wme
->data
;
5017 struct screen
*back_s
= data
->backing
;
5018 struct grid_reader gr
;
5019 u_int px
, py
, oldy
, hsize
;
5022 hsize
= screen_hsize(back_s
);
5023 py
= hsize
+ data
->cy
- data
->oy
;
5026 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5027 grid_reader_cursor_right(&gr
, 1, all
);
5028 grid_reader_get_cursor(&gr
, &px
, &py
);
5029 window_copy_acquire_cursor_down(wme
, hsize
, screen_size_y(back_s
),
5030 data
->oy
, oldy
, px
, py
, 0);
5034 window_copy_cursor_up(struct window_mode_entry
*wme
, int scroll_only
)
5036 struct window_copy_mode_data
*data
= wme
->data
;
5037 struct screen
*s
= &data
->screen
;
5038 u_int ox
, oy
, px
, py
;
5041 norectsel
= data
->screen
.sel
== NULL
|| !data
->rectflag
;
5042 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5043 ox
= window_copy_find_length(wme
, oy
);
5044 if (norectsel
&& data
->cx
!= ox
) {
5045 data
->lastcx
= data
->cx
;
5049 if (data
->lineflag
== LINE_SEL_LEFT_RIGHT
&& oy
== data
->sely
)
5050 window_copy_other_end(wme
);
5052 if (scroll_only
|| data
->cy
== 0) {
5054 data
->cx
= data
->lastcx
;
5055 window_copy_scroll_down(wme
, 1);
5057 if (data
->cy
== screen_size_y(s
) - 1)
5058 window_copy_redraw_lines(wme
, data
->cy
, 1);
5060 window_copy_redraw_lines(wme
, data
->cy
, 2);
5064 window_copy_update_cursor(wme
, data
->lastcx
,
5067 window_copy_update_cursor(wme
, data
->cx
, data
->cy
- 1);
5068 if (window_copy_update_selection(wme
, 1, 0)) {
5069 if (data
->cy
== screen_size_y(s
) - 1)
5070 window_copy_redraw_lines(wme
, data
->cy
, 1);
5072 window_copy_redraw_lines(wme
, data
->cy
, 2);
5077 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5078 px
= window_copy_find_length(wme
, py
);
5079 if ((data
->cx
>= data
->lastsx
&& data
->cx
!= px
) ||
5082 window_copy_update_cursor(wme
, px
, data
->cy
);
5083 if (window_copy_update_selection(wme
, 1, 0))
5084 window_copy_redraw_lines(wme
, data
->cy
, 1);
5088 if (data
->lineflag
== LINE_SEL_LEFT_RIGHT
)
5090 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5092 px
= screen_size_x(data
->backing
);
5094 px
= window_copy_find_length(wme
, py
);
5095 window_copy_update_cursor(wme
, px
, data
->cy
);
5096 if (window_copy_update_selection(wme
, 1, 0))
5097 window_copy_redraw_lines(wme
, data
->cy
, 1);
5099 else if (data
->lineflag
== LINE_SEL_RIGHT_LEFT
)
5101 window_copy_update_cursor(wme
, 0, data
->cy
);
5102 if (window_copy_update_selection(wme
, 1, 0))
5103 window_copy_redraw_lines(wme
, data
->cy
, 1);
5108 window_copy_cursor_down(struct window_mode_entry
*wme
, int scroll_only
)
5110 struct window_copy_mode_data
*data
= wme
->data
;
5111 struct screen
*s
= &data
->screen
;
5112 u_int ox
, oy
, px
, py
;
5115 norectsel
= data
->screen
.sel
== NULL
|| !data
->rectflag
;
5116 oy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5117 ox
= window_copy_find_length(wme
, oy
);
5118 if (norectsel
&& data
->cx
!= ox
) {
5119 data
->lastcx
= data
->cx
;
5123 if (data
->lineflag
== LINE_SEL_RIGHT_LEFT
&& oy
== data
->endsely
)
5124 window_copy_other_end(wme
);
5126 if (scroll_only
|| data
->cy
== screen_size_y(s
) - 1) {
5128 data
->cx
= data
->lastcx
;
5129 window_copy_scroll_up(wme
, 1);
5130 if (scroll_only
&& data
->cy
> 0)
5131 window_copy_redraw_lines(wme
, data
->cy
- 1, 2);
5134 window_copy_update_cursor(wme
, data
->lastcx
,
5137 window_copy_update_cursor(wme
, data
->cx
, data
->cy
+ 1);
5138 if (window_copy_update_selection(wme
, 1, 0))
5139 window_copy_redraw_lines(wme
, data
->cy
- 1, 2);
5143 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5144 px
= window_copy_find_length(wme
, py
);
5145 if ((data
->cx
>= data
->lastsx
&& data
->cx
!= px
) ||
5148 window_copy_update_cursor(wme
, px
, data
->cy
);
5149 if (window_copy_update_selection(wme
, 1, 0))
5150 window_copy_redraw_lines(wme
, data
->cy
, 1);
5154 if (data
->lineflag
== LINE_SEL_LEFT_RIGHT
)
5156 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5158 px
= screen_size_x(data
->backing
);
5160 px
= window_copy_find_length(wme
, py
);
5161 window_copy_update_cursor(wme
, px
, data
->cy
);
5162 if (window_copy_update_selection(wme
, 1, 0))
5163 window_copy_redraw_lines(wme
, data
->cy
, 1);
5165 else if (data
->lineflag
== LINE_SEL_RIGHT_LEFT
)
5167 window_copy_update_cursor(wme
, 0, data
->cy
);
5168 if (window_copy_update_selection(wme
, 1, 0))
5169 window_copy_redraw_lines(wme
, data
->cy
, 1);
5174 window_copy_cursor_jump(struct window_mode_entry
*wme
)
5176 struct window_copy_mode_data
*data
= wme
->data
;
5177 struct screen
*back_s
= data
->backing
;
5178 struct grid_reader gr
;
5179 u_int px
, py
, oldy
, hsize
;
5182 hsize
= screen_hsize(back_s
);
5183 py
= hsize
+ data
->cy
- data
->oy
;
5186 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5187 if (grid_reader_cursor_jump(&gr
, data
->jumpchar
)) {
5188 grid_reader_get_cursor(&gr
, &px
, &py
);
5189 window_copy_acquire_cursor_down(wme
, hsize
,
5190 screen_size_y(back_s
), data
->oy
, oldy
, px
, py
, 0);
5195 window_copy_cursor_jump_back(struct window_mode_entry
*wme
)
5197 struct window_copy_mode_data
*data
= wme
->data
;
5198 struct screen
*back_s
= data
->backing
;
5199 struct grid_reader gr
;
5200 u_int px
, py
, oldy
, hsize
;
5203 hsize
= screen_hsize(back_s
);
5204 py
= hsize
+ data
->cy
- data
->oy
;
5207 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5208 grid_reader_cursor_left(&gr
, 0);
5209 if (grid_reader_cursor_jump_back(&gr
, data
->jumpchar
)) {
5210 grid_reader_get_cursor(&gr
, &px
, &py
);
5211 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
,
5217 window_copy_cursor_jump_to(struct window_mode_entry
*wme
)
5219 struct window_copy_mode_data
*data
= wme
->data
;
5220 struct screen
*back_s
= data
->backing
;
5221 struct grid_reader gr
;
5222 u_int px
, py
, oldy
, hsize
;
5225 hsize
= screen_hsize(back_s
);
5226 py
= hsize
+ data
->cy
- data
->oy
;
5229 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5230 if (grid_reader_cursor_jump(&gr
, data
->jumpchar
)) {
5231 grid_reader_cursor_left(&gr
, 1);
5232 grid_reader_get_cursor(&gr
, &px
, &py
);
5233 window_copy_acquire_cursor_down(wme
, hsize
,
5234 screen_size_y(back_s
), data
->oy
, oldy
, px
, py
, 0);
5239 window_copy_cursor_jump_to_back(struct window_mode_entry
*wme
)
5241 struct window_copy_mode_data
*data
= wme
->data
;
5242 struct screen
*back_s
= data
->backing
;
5243 struct grid_reader gr
;
5244 u_int px
, py
, oldy
, hsize
;
5247 hsize
= screen_hsize(back_s
);
5248 py
= hsize
+ data
->cy
- data
->oy
;
5251 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5252 grid_reader_cursor_left(&gr
, 0);
5253 grid_reader_cursor_left(&gr
, 0);
5254 if (grid_reader_cursor_jump_back(&gr
, data
->jumpchar
)) {
5255 grid_reader_cursor_right(&gr
, 1, 0);
5256 grid_reader_get_cursor(&gr
, &px
, &py
);
5257 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
,
5263 window_copy_cursor_next_word(struct window_mode_entry
*wme
,
5264 const char *separators
)
5266 struct window_copy_mode_data
*data
= wme
->data
;
5267 struct screen
*back_s
= data
->backing
;
5268 struct grid_reader gr
;
5269 u_int px
, py
, oldy
, hsize
;
5272 hsize
= screen_hsize(back_s
);
5273 py
= hsize
+ data
->cy
- data
->oy
;
5276 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5277 grid_reader_cursor_next_word(&gr
, separators
);
5278 grid_reader_get_cursor(&gr
, &px
, &py
);
5279 window_copy_acquire_cursor_down(wme
, hsize
, screen_size_y(back_s
),
5280 data
->oy
, oldy
, px
, py
, 0);
5283 /* Compute the next place where a word ends. */
5285 window_copy_cursor_next_word_end_pos(struct window_mode_entry
*wme
,
5286 const char *separators
, u_int
*ppx
, u_int
*ppy
)
5288 struct window_pane
*wp
= wme
->wp
;
5289 struct window_copy_mode_data
*data
= wme
->data
;
5290 struct options
*oo
= wp
->window
->options
;
5291 struct screen
*back_s
= data
->backing
;
5292 struct grid_reader gr
;
5293 u_int px
, py
, hsize
;
5296 hsize
= screen_hsize(back_s
);
5297 py
= hsize
+ data
->cy
- data
->oy
;
5299 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5300 if (options_get_number(oo
, "mode-keys") == MODEKEY_VI
) {
5301 if (!grid_reader_in_set(&gr
, WHITESPACE
))
5302 grid_reader_cursor_right(&gr
, 0, 0);
5303 grid_reader_cursor_next_word_end(&gr
, separators
);
5304 grid_reader_cursor_left(&gr
, 1);
5306 grid_reader_cursor_next_word_end(&gr
, separators
);
5307 grid_reader_get_cursor(&gr
, &px
, &py
);
5312 /* Move to the next place where a word ends. */
5314 window_copy_cursor_next_word_end(struct window_mode_entry
*wme
,
5315 const char *separators
, int no_reset
)
5317 struct window_pane
*wp
= wme
->wp
;
5318 struct window_copy_mode_data
*data
= wme
->data
;
5319 struct options
*oo
= wp
->window
->options
;
5320 struct screen
*back_s
= data
->backing
;
5321 struct grid_reader gr
;
5322 u_int px
, py
, oldy
, hsize
;
5325 hsize
= screen_hsize(back_s
);
5326 py
= hsize
+ data
->cy
- data
->oy
;
5329 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5330 if (options_get_number(oo
, "mode-keys") == MODEKEY_VI
) {
5331 if (!grid_reader_in_set(&gr
, WHITESPACE
))
5332 grid_reader_cursor_right(&gr
, 0, 0);
5333 grid_reader_cursor_next_word_end(&gr
, separators
);
5334 grid_reader_cursor_left(&gr
, 1);
5336 grid_reader_cursor_next_word_end(&gr
, separators
);
5337 grid_reader_get_cursor(&gr
, &px
, &py
);
5338 window_copy_acquire_cursor_down(wme
, hsize
, screen_size_y(back_s
),
5339 data
->oy
, oldy
, px
, py
, no_reset
);
5342 /* Compute the previous place where a word begins. */
5344 window_copy_cursor_previous_word_pos(struct window_mode_entry
*wme
,
5345 const char *separators
, u_int
*ppx
, u_int
*ppy
)
5347 struct window_copy_mode_data
*data
= wme
->data
;
5348 struct screen
*back_s
= data
->backing
;
5349 struct grid_reader gr
;
5350 u_int px
, py
, hsize
;
5353 hsize
= screen_hsize(back_s
);
5354 py
= hsize
+ data
->cy
- data
->oy
;
5356 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5357 grid_reader_cursor_previous_word(&gr
, separators
, /* already= */ 0,
5358 /* stop_at_eol= */ 1);
5359 grid_reader_get_cursor(&gr
, &px
, &py
);
5364 /* Move to the previous place where a word begins. */
5366 window_copy_cursor_previous_word(struct window_mode_entry
*wme
,
5367 const char *separators
, int already
)
5369 struct window_copy_mode_data
*data
= wme
->data
;
5370 struct window
*w
= wme
->wp
->window
;
5371 struct screen
*back_s
= data
->backing
;
5372 struct grid_reader gr
;
5373 u_int px
, py
, oldy
, hsize
;
5376 if (options_get_number(w
->options
, "mode-keys") == MODEKEY_EMACS
)
5382 hsize
= screen_hsize(back_s
);
5383 py
= hsize
+ data
->cy
- data
->oy
;
5386 grid_reader_start(&gr
, back_s
->grid
, px
, py
);
5387 grid_reader_cursor_previous_word(&gr
, separators
, already
, stop_at_eol
);
5388 grid_reader_get_cursor(&gr
, &px
, &py
);
5389 window_copy_acquire_cursor_up(wme
, hsize
, data
->oy
, oldy
, px
, py
);
5393 window_copy_cursor_prompt(struct window_mode_entry
*wme
, int direction
,
5396 struct window_copy_mode_data
*data
= wme
->data
;
5397 struct screen
*s
= data
->backing
;
5398 struct grid
*gd
= s
->grid
;
5400 u_int line
= gd
->hsize
- data
->oy
+ data
->cy
;
5403 if (args
!= NULL
&& strcmp(args
, "-o") == 0)
5404 line_flag
= GRID_LINE_START_OUTPUT
;
5406 line_flag
= GRID_LINE_START_PROMPT
;
5408 if (direction
== 0) { /* up */
5413 end_line
= gd
->hsize
+ gd
->sy
- 1;
5416 if (line
== end_line
)
5419 if (line
== end_line
)
5423 if (grid_get_line(gd
, line
)->flags
& line_flag
)
5428 if (line
> gd
->hsize
) {
5429 data
->cy
= line
- gd
->hsize
;
5433 data
->oy
= gd
->hsize
- line
;
5436 window_copy_update_selection(wme
, 1, 0);
5437 window_copy_redraw_screen(wme
);
5441 window_copy_scroll_up(struct window_mode_entry
*wme
, u_int ny
)
5443 struct window_pane
*wp
= wme
->wp
;
5444 struct window_copy_mode_data
*data
= wme
->data
;
5445 struct screen
*s
= &data
->screen
;
5446 struct screen_write_ctx ctx
;
5454 if (data
->searchmark
!= NULL
&& !data
->timeout
)
5455 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
5456 window_copy_update_selection(wme
, 0, 0);
5458 screen_write_start_pane(&ctx
, wp
, NULL
);
5459 screen_write_cursormove(&ctx
, 0, 0, 0);
5460 screen_write_deleteline(&ctx
, ny
, 8);
5461 window_copy_write_lines(wme
, &ctx
, screen_size_y(s
) - ny
, ny
);
5462 window_copy_write_line(wme
, &ctx
, 0);
5463 if (screen_size_y(s
) > 1)
5464 window_copy_write_line(wme
, &ctx
, 1);
5465 if (screen_size_y(s
) > 3)
5466 window_copy_write_line(wme
, &ctx
, screen_size_y(s
) - 2);
5467 if (s
->sel
!= NULL
&& screen_size_y(s
) > ny
)
5468 window_copy_write_line(wme
, &ctx
, screen_size_y(s
) - ny
- 1);
5469 screen_write_cursormove(&ctx
, data
->cx
, data
->cy
, 0);
5470 screen_write_stop(&ctx
);
5474 window_copy_scroll_down(struct window_mode_entry
*wme
, u_int ny
)
5476 struct window_pane
*wp
= wme
->wp
;
5477 struct window_copy_mode_data
*data
= wme
->data
;
5478 struct screen
*s
= &data
->screen
;
5479 struct screen_write_ctx ctx
;
5481 if (ny
> screen_hsize(data
->backing
))
5484 if (data
->oy
> screen_hsize(data
->backing
) - ny
)
5485 ny
= screen_hsize(data
->backing
) - data
->oy
;
5490 if (data
->searchmark
!= NULL
&& !data
->timeout
)
5491 window_copy_search_marks(wme
, NULL
, data
->searchregex
, 1);
5492 window_copy_update_selection(wme
, 0, 0);
5494 screen_write_start_pane(&ctx
, wp
, NULL
);
5495 screen_write_cursormove(&ctx
, 0, 0, 0);
5496 screen_write_insertline(&ctx
, ny
, 8);
5497 window_copy_write_lines(wme
, &ctx
, 0, ny
);
5498 if (s
->sel
!= NULL
&& screen_size_y(s
) > ny
)
5499 window_copy_write_line(wme
, &ctx
, ny
);
5500 else if (ny
== 1) /* nuke position */
5501 window_copy_write_line(wme
, &ctx
, 1);
5502 screen_write_cursormove(&ctx
, data
->cx
, data
->cy
, 0);
5503 screen_write_stop(&ctx
);
5507 window_copy_rectangle_set(struct window_mode_entry
*wme
, int rectflag
)
5509 struct window_copy_mode_data
*data
= wme
->data
;
5512 data
->rectflag
= rectflag
;
5514 py
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5515 px
= window_copy_find_length(wme
, py
);
5517 window_copy_update_cursor(wme
, px
, data
->cy
);
5519 window_copy_update_selection(wme
, 1, 0);
5520 window_copy_redraw_screen(wme
);
5524 window_copy_move_mouse(struct mouse_event
*m
)
5526 struct window_pane
*wp
;
5527 struct window_mode_entry
*wme
;
5530 wp
= cmd_mouse_pane(m
, NULL
, NULL
);
5533 wme
= TAILQ_FIRST(&wp
->modes
);
5536 if (wme
->mode
!= &window_copy_mode
&& wme
->mode
!= &window_view_mode
)
5539 if (cmd_mouse_at(wp
, m
, &x
, &y
, 0) != 0)
5542 window_copy_update_cursor(wme
, x
, y
);
5546 window_copy_start_drag(struct client
*c
, struct mouse_event
*m
)
5548 struct window_pane
*wp
;
5549 struct window_mode_entry
*wme
;
5550 struct window_copy_mode_data
*data
;
5556 wp
= cmd_mouse_pane(m
, NULL
, NULL
);
5559 wme
= TAILQ_FIRST(&wp
->modes
);
5562 if (wme
->mode
!= &window_copy_mode
&& wme
->mode
!= &window_view_mode
)
5565 if (cmd_mouse_at(wp
, m
, &x
, &y
, 1) != 0)
5568 c
->tty
.mouse_drag_update
= window_copy_drag_update
;
5569 c
->tty
.mouse_drag_release
= window_copy_drag_release
;
5572 yg
= screen_hsize(data
->backing
) + y
- data
->oy
;
5573 if (x
< data
->selrx
|| x
> data
->endselrx
|| yg
!= data
->selry
)
5574 data
->selflag
= SEL_CHAR
;
5575 switch (data
->selflag
) {
5577 if (data
->separators
!= NULL
) {
5578 window_copy_update_cursor(wme
, x
, y
);
5579 window_copy_cursor_previous_word_pos(wme
,
5580 data
->separators
, &x
, &y
);
5581 y
-= screen_hsize(data
->backing
) - data
->oy
;
5583 window_copy_update_cursor(wme
, x
, y
);
5586 window_copy_update_cursor(wme
, 0, y
);
5589 window_copy_update_cursor(wme
, x
, y
);
5590 window_copy_start_selection(wme
);
5594 window_copy_redraw_screen(wme
);
5595 window_copy_drag_update(c
, m
);
5599 window_copy_drag_update(struct client
*c
, struct mouse_event
*m
)
5601 struct window_pane
*wp
;
5602 struct window_mode_entry
*wme
;
5603 struct window_copy_mode_data
*data
;
5604 u_int x
, y
, old_cx
, old_cy
;
5605 struct timeval tv
= {
5606 .tv_usec
= WINDOW_COPY_DRAG_REPEAT_TIME
5612 wp
= cmd_mouse_pane(m
, NULL
, NULL
);
5615 wme
= TAILQ_FIRST(&wp
->modes
);
5618 if (wme
->mode
!= &window_copy_mode
&& wme
->mode
!= &window_view_mode
)
5622 evtimer_del(&data
->dragtimer
);
5624 if (cmd_mouse_at(wp
, m
, &x
, &y
, 0) != 0)
5629 window_copy_update_cursor(wme
, x
, y
);
5630 if (window_copy_update_selection(wme
, 1, 0))
5631 window_copy_redraw_selection(wme
, old_cy
);
5632 if (old_cy
!= data
->cy
|| old_cx
== data
->cx
) {
5634 evtimer_add(&data
->dragtimer
, &tv
);
5635 window_copy_cursor_up(wme
, 1);
5636 } else if (y
== screen_size_y(&data
->screen
) - 1) {
5637 evtimer_add(&data
->dragtimer
, &tv
);
5638 window_copy_cursor_down(wme
, 1);
5644 window_copy_drag_release(struct client
*c
, struct mouse_event
*m
)
5646 struct window_pane
*wp
;
5647 struct window_mode_entry
*wme
;
5648 struct window_copy_mode_data
*data
;
5653 wp
= cmd_mouse_pane(m
, NULL
, NULL
);
5656 wme
= TAILQ_FIRST(&wp
->modes
);
5659 if (wme
->mode
!= &window_copy_mode
&& wme
->mode
!= &window_view_mode
)
5663 evtimer_del(&data
->dragtimer
);
5667 window_copy_jump_to_mark(struct window_mode_entry
*wme
)
5669 struct window_copy_mode_data
*data
= wme
->data
;
5673 tmy
= screen_hsize(data
->backing
) + data
->cy
- data
->oy
;
5674 data
->cx
= data
->mx
;
5675 if (data
->my
< screen_hsize(data
->backing
)) {
5677 data
->oy
= screen_hsize(data
->backing
) - data
->my
;
5679 data
->cy
= data
->my
- screen_hsize(data
->backing
);
5685 window_copy_update_selection(wme
, 0, 0);
5686 window_copy_redraw_screen(wme
);
5689 /* Scroll up if the cursor went off the visible screen. */
5691 window_copy_acquire_cursor_up(struct window_mode_entry
*wme
, u_int hsize
,
5692 u_int oy
, u_int oldy
, u_int px
, u_int py
)
5694 u_int cy
, yy
, ny
, nd
;
5707 window_copy_cursor_up(wme
, 1);
5710 window_copy_update_cursor(wme
, px
, cy
);
5711 if (window_copy_update_selection(wme
, 1, 0))
5712 window_copy_redraw_lines(wme
, cy
, nd
);
5715 /* Scroll down if the cursor went off the visible screen. */
5717 window_copy_acquire_cursor_down(struct window_mode_entry
*wme
, u_int hsize
,
5718 u_int sy
, u_int oy
, u_int oldy
, u_int px
, u_int py
, int no_reset
)
5720 u_int cy
, yy
, ny
, nd
;
5722 cy
= py
- hsize
+ oy
;
5733 window_copy_cursor_down(wme
, 1);
5737 window_copy_update_cursor(wme
, px
, yy
);
5739 window_copy_update_cursor(wme
, px
, cy
);
5740 if (window_copy_update_selection(wme
, 1, no_reset
))
5741 window_copy_redraw_lines(wme
, oldy
, nd
);