4 * Copyright (c) 2007 Nicholas Marriott <nicm@users.sourceforge.net>
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>
25 int screen_redraw_cell_border1(struct window_pane
*, u_int
, u_int
);
26 int screen_redraw_cell_border(struct client
*, u_int
, u_int
);
27 int screen_redraw_check_cell(struct client
*, u_int
, u_int
,
28 struct window_pane
**);
29 int screen_redraw_check_active(u_int
, u_int
, int, struct window
*,
30 struct window_pane
*);
32 void screen_redraw_draw_borders(struct client
*, int, u_int
);
33 void screen_redraw_draw_panes(struct client
*, u_int
);
34 void screen_redraw_draw_status(struct client
*, u_int
);
35 void screen_redraw_draw_number(struct client
*, struct window_pane
*);
38 #define CELL_LEFTRIGHT 1
39 #define CELL_TOPBOTTOM 2
40 #define CELL_TOPLEFT 3
41 #define CELL_TOPRIGHT 4
42 #define CELL_BOTTOMLEFT 5
43 #define CELL_BOTTOMRIGHT 6
44 #define CELL_TOPJOIN 7
45 #define CELL_BOTTOMJOIN 8
46 #define CELL_LEFTJOIN 9
47 #define CELL_RIGHTJOIN 10
49 #define CELL_OUTSIDE 12
51 #define CELL_BORDERS " xqlkmjwvtun~"
53 /* Check if cell is on the border of a particular pane. */
55 screen_redraw_cell_border1(struct window_pane
*wp
, u_int px
, u_int py
)
58 if (px
>= wp
->xoff
&& px
< wp
->xoff
+ wp
->sx
&&
59 py
>= wp
->yoff
&& py
< wp
->yoff
+ wp
->sy
)
62 /* Left/right borders. */
63 if ((wp
->yoff
== 0 || py
>= wp
->yoff
- 1) && py
<= wp
->yoff
+ wp
->sy
) {
64 if (wp
->xoff
!= 0 && px
== wp
->xoff
- 1)
66 if (px
== wp
->xoff
+ wp
->sx
)
70 /* Top/bottom borders. */
71 if ((wp
->xoff
== 0 || px
>= wp
->xoff
- 1) && px
<= wp
->xoff
+ wp
->sx
) {
72 if (wp
->yoff
!= 0 && py
== wp
->yoff
- 1)
74 if (py
== wp
->yoff
+ wp
->sy
)
82 /* Check if a cell is on the pane border. */
84 screen_redraw_cell_border(struct client
*c
, u_int px
, u_int py
)
86 struct window
*w
= c
->session
->curw
->window
;
87 struct window_pane
*wp
;
90 /* Check all the panes. */
91 TAILQ_FOREACH(wp
, &w
->panes
, entry
) {
92 if (!window_pane_visible(wp
))
94 if ((retval
= screen_redraw_cell_border1(wp
, px
, py
)) != -1)
101 /* Check if cell inside a pane. */
103 screen_redraw_check_cell(struct client
*c
, u_int px
, u_int py
,
104 struct window_pane
**wpp
)
106 struct window
*w
= c
->session
->curw
->window
;
107 struct window_pane
*wp
;
110 if (px
> w
->sx
|| py
> w
->sy
)
111 return (CELL_OUTSIDE
);
113 TAILQ_FOREACH(wp
, &w
->panes
, entry
) {
114 if (!window_pane_visible(wp
))
118 /* If outside the pane and its border, skip it. */
119 if ((wp
->xoff
!= 0 && px
< wp
->xoff
- 1) ||
120 px
> wp
->xoff
+ wp
->sx
||
121 (wp
->yoff
!= 0 && py
< wp
->yoff
- 1) ||
122 py
> wp
->yoff
+ wp
->sy
)
125 /* If definitely inside, return so. */
126 if (!screen_redraw_cell_border(c
, px
, py
))
127 return (CELL_INSIDE
);
130 * Construct a bitmask of whether the cells to the left (bit
131 * 4), right, top, and bottom (bit 1) of this cell are borders.
134 if (px
== 0 || screen_redraw_cell_border(c
, px
- 1, py
))
136 if (px
<= w
->sx
&& screen_redraw_cell_border(c
, px
+ 1, py
))
138 if (py
== 0 || screen_redraw_cell_border(c
, px
, py
- 1))
140 if (py
<= w
->sy
&& screen_redraw_cell_border(c
, px
, py
+ 1))
144 * Figure out what kind of border this cell is. Only one bit
145 * set doesn't make sense (can't have a border cell with no
149 case 15: /* 1111, left right top bottom */
151 case 14: /* 1110, left right top */
152 return (CELL_BOTTOMJOIN
);
153 case 13: /* 1101, left right bottom */
154 return (CELL_TOPJOIN
);
155 case 12: /* 1100, left right */
156 return (CELL_TOPBOTTOM
);
157 case 11: /* 1011, left top bottom */
158 return (CELL_RIGHTJOIN
);
159 case 10: /* 1010, left top */
160 return (CELL_BOTTOMRIGHT
);
161 case 9: /* 1001, left bottom */
162 return (CELL_TOPRIGHT
);
163 case 7: /* 0111, right top bottom */
164 return (CELL_LEFTJOIN
);
165 case 6: /* 0110, right top */
166 return (CELL_BOTTOMLEFT
);
167 case 5: /* 0101, right bottom */
168 return (CELL_TOPLEFT
);
169 case 3: /* 0011, top bottom */
170 return (CELL_LEFTRIGHT
);
175 return (CELL_OUTSIDE
);
178 /* Check active pane indicator. */
180 screen_redraw_check_active(u_int px
, u_int py
, int type
, struct window
*w
,
181 struct window_pane
*wp
)
183 /* Is this off the active pane border? */
184 if (screen_redraw_cell_border1(w
->active
, px
, py
) != 1)
187 /* If there are more than two panes, that's enough. */
188 if (window_count_panes(w
) != 2)
191 /* Else if the cell is not a border cell, forget it. */
192 if (wp
== NULL
|| (type
== CELL_OUTSIDE
|| type
== CELL_INSIDE
))
195 /* Check if the pane covers the whole width. */
196 if (wp
->xoff
== 0 && wp
->sx
== w
->sx
) {
197 /* This can either be the top pane or the bottom pane. */
198 if (wp
->yoff
== 0) { /* top pane */
200 return (px
<= wp
->sx
/ 2);
201 return (px
> wp
->sx
/ 2);
206 /* Check if the pane covers the whole height. */
207 if (wp
->yoff
== 0 && wp
->sy
== w
->sy
) {
208 /* This can either be the left pane or the right pane. */
209 if (wp
->xoff
== 0) { /* left pane */
211 return (py
<= wp
->sy
/ 2);
212 return (py
> wp
->sy
/ 2);
220 /* Redraw entire screen. */
222 screen_redraw_screen(struct client
*c
, int draw_panes
, int draw_status
,
225 struct options
*oo
= &c
->session
->options
;
226 struct tty
*tty
= &c
->tty
;
230 /* Suspended clients should not be updated. */
231 if (c
->flags
& CLIENT_SUSPENDED
)
234 /* Get status line, er, status. */
235 spos
= options_get_number(oo
, "status-position");
236 if (c
->message_string
!= NULL
|| c
->prompt_string
!= NULL
)
239 status
= options_get_number(oo
, "status");
241 if (status
&& spos
== 0)
247 screen_redraw_draw_borders(c
, status
, top
);
249 screen_redraw_draw_panes(c
, top
);
251 screen_redraw_draw_status(c
, top
);
255 /* Draw a single pane. */
257 screen_redraw_pane(struct client
*c
, struct window_pane
*wp
)
261 if (!window_pane_visible(wp
))
265 if (status_at_line(c
) == 0)
268 for (i
= 0; i
< wp
->sy
; i
++)
269 tty_draw_line(&c
->tty
, wp
->screen
, i
, wp
->xoff
, yoff
);
273 /* Draw the borders. */
275 screen_redraw_draw_borders(struct client
*c
, int status
, u_int top
)
277 struct window
*w
= c
->session
->curw
->window
;
278 struct options
*oo
= &c
->session
->options
;
279 struct tty
*tty
= &c
->tty
;
280 struct window_pane
*wp
;
281 struct grid_cell active_gc
, other_gc
;
285 /* LSC: -Werror=maybe-uninitialized while compiling with -O3 */
287 #endif /* defined(__minix) */
288 style_apply(&other_gc
, oo
, "pane-border-style");
289 style_apply(&active_gc
, oo
, "pane-active-border-style");
290 active_gc
.attr
= other_gc
.attr
= GRID_ATTR_CHARSET
;
292 for (j
= 0; j
< tty
->sy
- status
; j
++) {
293 for (i
= 0; i
< tty
->sx
; i
++) {
294 type
= screen_redraw_check_cell(c
, i
, j
, &wp
);
295 if (type
== CELL_INSIDE
)
297 if (screen_redraw_check_active(i
, j
, type
, w
, wp
))
298 tty_attributes(tty
, &active_gc
);
300 tty_attributes(tty
, &other_gc
);
301 tty_cursor(tty
, i
, top
+ j
);
302 tty_putc(tty
, CELL_BORDERS
[type
]);
307 /* Draw the panes. */
309 screen_redraw_draw_panes(struct client
*c
, u_int top
)
311 struct window
*w
= c
->session
->curw
->window
;
312 struct tty
*tty
= &c
->tty
;
313 struct window_pane
*wp
;
317 TAILQ_FOREACH(wp
, &w
->panes
, entry
) {
318 if (!window_pane_visible(wp
))
321 for (i
= 0; i
< wp
->sy
; i
++)
322 tty_draw_line(tty
, s
, i
, wp
->xoff
, top
+ wp
->yoff
);
323 if (c
->flags
& CLIENT_IDENTIFY
)
324 screen_redraw_draw_number(c
, wp
);
328 /* Draw the status line. */
330 screen_redraw_draw_status(struct client
*c
, u_int top
)
332 struct tty
*tty
= &c
->tty
;
335 tty_draw_line(tty
, &c
->status
, 0, 0, 0);
337 tty_draw_line(tty
, &c
->status
, 0, 0, tty
->sy
- 1);
340 /* Draw number on a pane. */
342 screen_redraw_draw_number(struct client
*c
, struct window_pane
*wp
)
344 struct tty
*tty
= &c
->tty
;
345 struct session
*s
= c
->session
;
346 struct options
*oo
= &s
->options
;
347 struct window
*w
= wp
->window
;
349 u_int idx
, px
, py
, i
, j
, xoff
, yoff
;
350 int colour
, active_colour
;
354 if (window_pane_index(wp
, &idx
) != 0)
355 fatalx("index not found");
356 len
= xsnprintf(buf
, sizeof buf
, "%u", idx
);
360 colour
= options_get_number(oo
, "display-panes-colour");
361 active_colour
= options_get_number(oo
, "display-panes-active-colour");
363 px
= wp
->sx
/ 2; py
= wp
->sy
/ 2;
364 xoff
= wp
->xoff
; yoff
= wp
->yoff
;
366 if (wp
->sx
< len
* 6 || wp
->sy
< 5) {
367 tty_cursor(tty
, xoff
+ px
- len
/ 2, yoff
+ py
);
374 memcpy(&gc
, &grid_default_cell
, sizeof gc
);
376 colour_set_bg(&gc
, active_colour
);
378 colour_set_bg(&gc
, colour
);
379 tty_attributes(tty
, &gc
);
380 for (ptr
= buf
; *ptr
!= '\0'; ptr
++) {
381 if (*ptr
< '0' || *ptr
> '9')
385 for (j
= 0; j
< 5; j
++) {
386 for (i
= px
; i
< px
+ 5; i
++) {
387 tty_cursor(tty
, xoff
+ i
, yoff
+ py
+ j
);
388 if (clock_table
[idx
][j
][i
- px
])
395 len
= xsnprintf(buf
, sizeof buf
, "%ux%u", wp
->sx
, wp
->sy
);
396 if (wp
->sx
< len
|| wp
->sy
< 6)
398 tty_cursor(tty
, xoff
+ wp
->sx
- len
, yoff
);
401 memcpy(&gc
, &grid_default_cell
, sizeof gc
);
403 colour_set_fg(&gc
, active_colour
);
405 colour_set_fg(&gc
, colour
);
406 tty_attributes(tty
, &gc
);
409 tty_cursor(tty
, 0, 0);