1 /* Window creation, deletion and examination for GNU Emacs.
2 Does not include redisplay.
3 Copyright (C) 1985,86,87,93,94,95,96,97,1998,2000, 2001, 2002
4 Free Software Foundation, Inc.
6 This file is part of GNU Emacs.
8 GNU Emacs is free software; you can redistribute it and/or modify
9 it under the terms of the GNU General Public License as published by
10 the Free Software Foundation; either version 2, or (at your option)
13 GNU Emacs is distributed in the hope that it will be useful,
14 but WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16 GNU General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with GNU Emacs; see the file COPYING. If not, write to
20 the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21 Boston, MA 02111-1307, USA. */
34 #include "dispextern.h"
35 #include "blockinput.h"
36 #include "intervals.h"
40 #endif /* HAVE_X_WINDOWS */
51 /* Values returned from coordinates_in_window. */
67 Lisp_Object Qwindowp
, Qwindow_live_p
, Qwindow_configuration_p
;
68 Lisp_Object Qwindow_size_fixed
;
69 extern Lisp_Object Qleft_margin
, Qright_margin
;
70 extern Lisp_Object Qheight
, Qwidth
;
72 static int displayed_window_lines
P_ ((struct window
*));
73 static struct window
*decode_window
P_ ((Lisp_Object
));
74 static Lisp_Object select_window_1
P_ ((Lisp_Object
, int));
75 static int count_windows
P_ ((struct window
*));
76 static int get_leaf_windows
P_ ((struct window
*, struct window
**, int));
77 static void window_scroll
P_ ((Lisp_Object
, int, int, int));
78 static void window_scroll_pixel_based
P_ ((Lisp_Object
, int, int, int));
79 static void window_scroll_line_based
P_ ((Lisp_Object
, int, int, int));
80 static int window_min_size_1
P_ ((struct window
*, int));
81 static int window_min_size
P_ ((struct window
*, int, int, int *));
82 static void size_window
P_ ((Lisp_Object
, int, int, int));
83 static int freeze_window_start
P_ ((struct window
*, void *));
84 static int window_fixed_size_p
P_ ((struct window
*, int, int));
85 static void enlarge_window
P_ ((Lisp_Object
, int, int, int));
86 static Lisp_Object window_list
P_ ((void));
87 static int add_window_to_list
P_ ((struct window
*, void *));
88 static int candidate_window_p
P_ ((Lisp_Object
, Lisp_Object
, Lisp_Object
,
90 static Lisp_Object next_window
P_ ((Lisp_Object
, Lisp_Object
,
92 static void decode_next_window_args
P_ ((Lisp_Object
*, Lisp_Object
*,
94 static int foreach_window_1
P_ ((struct window
*,
95 int (* fn
) (struct window
*, void *),
97 static Lisp_Object window_list_1
P_ ((Lisp_Object
, Lisp_Object
, Lisp_Object
));
99 /* The value of `window-size-fixed'. */
101 int window_size_fixed
;
103 /* This is the window in which the terminal's cursor should
104 be left when nothing is being done with it. This must
105 always be a leaf window, and its buffer is selected by
106 the top level editing loop at the end of each command.
108 This value is always the same as
109 FRAME_SELECTED_WINDOW (selected_frame). */
111 Lisp_Object selected_window
;
113 /* A list of all windows for use by next_window and Fwindow_list.
114 Functions creating or deleting windows should invalidate this cache
115 by setting it to nil. */
117 Lisp_Object Vwindow_list
;
119 /* The mini-buffer window of the selected frame.
120 Note that you cannot test for mini-bufferness of an arbitrary window
121 by comparing against this; but you can test for mini-bufferness of
122 the selected window. */
124 Lisp_Object minibuf_window
;
126 /* Non-nil means it is the window whose mode line should be
127 shown as the selected window when the minibuffer is selected. */
129 Lisp_Object minibuf_selected_window
;
131 /* Non-nil means it is the window for C-M-v to scroll
132 when the mini-buffer is selected. */
134 Lisp_Object Vminibuf_scroll_window
;
136 /* Non-nil means this is the buffer whose window C-M-v should scroll. */
138 Lisp_Object Vother_window_scroll_buffer
;
140 /* Non-nil means it's function to call to display temp buffers. */
142 Lisp_Object Vtemp_buffer_show_function
;
144 /* Non-zero means to use mode-line-inactive face in all windows but the
145 selected-window and the minibuffer-scroll-window when the
146 minibuffer is active. */
147 int mode_line_in_non_selected_windows
;
149 /* If a window gets smaller than either of these, it is removed. */
151 EMACS_INT window_min_height
;
152 EMACS_INT window_min_width
;
154 /* Nonzero implies Fdisplay_buffer should create windows. */
158 /* Nonzero implies make new frames for Fdisplay_buffer. */
162 /* Nonzero means reuse existing frames for displaying buffers. */
164 int display_buffer_reuse_frames
;
166 /* Non-nil means use this function instead of default */
168 Lisp_Object Vpop_up_frame_function
;
170 /* Function to call to handle Fdisplay_buffer. */
172 Lisp_Object Vdisplay_buffer_function
;
174 /* Non-nil means that Fdisplay_buffer should even the heights of windows. */
176 Lisp_Object Veven_window_heights
;
178 /* List of buffer *names* for buffers that should have their own frames. */
180 Lisp_Object Vspecial_display_buffer_names
;
182 /* List of regexps for buffer names that should have their own frames. */
184 Lisp_Object Vspecial_display_regexps
;
186 /* Function to pop up a special frame. */
188 Lisp_Object Vspecial_display_function
;
190 /* List of buffer *names* for buffers to appear in selected window. */
192 Lisp_Object Vsame_window_buffer_names
;
194 /* List of regexps for buffer names to appear in selected window. */
196 Lisp_Object Vsame_window_regexps
;
198 /* Hook run at end of temp_output_buffer_show. */
200 Lisp_Object Qtemp_buffer_show_hook
;
202 /* Fdisplay_buffer always splits the largest window
203 if that window is more than this high. */
205 EMACS_INT split_height_threshold
;
207 /* Number of lines of continuity in scrolling by screenfuls. */
209 EMACS_INT next_screen_context_lines
;
211 /* Incremented for each window created. */
213 static int sequence_number
;
215 /* Nonzero after init_window_once has finished. */
217 static int window_initialized
;
219 /* Hook to run when window config changes. */
221 Lisp_Object Qwindow_configuration_change_hook
;
222 Lisp_Object Vwindow_configuration_change_hook
;
224 /* Nonzero means scroll commands try to put point
225 at the same screen height as previously. */
227 Lisp_Object Vscroll_preserve_screen_position
;
229 #if 0 /* This isn't used anywhere. */
230 /* Nonzero means we can split a frame even if it is "unsplittable". */
231 static int inhibit_frame_unsplittable
;
234 extern EMACS_INT scroll_margin
;
236 extern Lisp_Object Qwindow_scroll_functions
, Vwindow_scroll_functions
;
238 DEFUN ("windowp", Fwindowp
, Swindowp
, 1, 1, 0,
239 doc
: /* Returns t if OBJECT is a window. */)
243 return WINDOWP (object
) ? Qt
: Qnil
;
246 DEFUN ("window-live-p", Fwindow_live_p
, Swindow_live_p
, 1, 1, 0,
247 doc
: /* Returns t if OBJECT is a window which is currently visible. */)
251 return WINDOW_LIVE_P (object
) ? Qt
: Qnil
;
258 register struct window
*p
;
260 p
= allocate_window ();
261 XSETFASTINT (p
->sequence_number
, ++sequence_number
);
262 XSETFASTINT (p
->left
, 0);
263 XSETFASTINT (p
->top
, 0);
264 XSETFASTINT (p
->height
, 0);
265 XSETFASTINT (p
->width
, 0);
266 XSETFASTINT (p
->hscroll
, 0);
267 XSETFASTINT (p
->min_hscroll
, 0);
268 p
->orig_top
= p
->orig_height
= Qnil
;
269 p
->start
= Fmake_marker ();
270 p
->pointm
= Fmake_marker ();
271 XSETFASTINT (p
->use_time
, 0);
273 p
->display_table
= Qnil
;
275 p
->pseudo_window_p
= 0;
276 bzero (&p
->cursor
, sizeof (p
->cursor
));
277 bzero (&p
->last_cursor
, sizeof (p
->last_cursor
));
278 bzero (&p
->phys_cursor
, sizeof (p
->phys_cursor
));
279 p
->desired_matrix
= p
->current_matrix
= 0;
280 p
->phys_cursor_type
= -1;
281 p
->phys_cursor_width
= -1;
282 p
->must_be_updated_p
= 0;
283 XSETFASTINT (p
->window_end_vpos
, 0);
284 XSETFASTINT (p
->window_end_pos
, 0);
285 p
->window_end_valid
= Qnil
;
288 XSETFASTINT (p
->last_point
, 0);
289 p
->frozen_window_start_p
= 0;
290 p
->height_fixed_p
= 0;
291 p
->last_cursor_off_p
= p
->cursor_off_p
= 0;
297 DEFUN ("selected-window", Fselected_window
, Sselected_window
, 0, 0, 0,
298 doc
: /* Return the window that the cursor now appears in and commands apply to. */)
301 return selected_window
;
304 DEFUN ("minibuffer-window", Fminibuffer_window
, Sminibuffer_window
, 0, 1, 0,
305 doc
: /* Return the window used now for minibuffers.
306 If the optional argument FRAME is specified, return the minibuffer window
307 used by that frame. */)
312 frame
= selected_frame
;
313 CHECK_LIVE_FRAME (frame
);
314 return FRAME_MINIBUF_WINDOW (XFRAME (frame
));
317 DEFUN ("window-minibuffer-p", Fwindow_minibuffer_p
, Swindow_minibuffer_p
, 0, 1, 0,
318 doc
: /* Returns non-nil if WINDOW is a minibuffer window. */)
322 struct window
*w
= decode_window (window
);
323 return MINI_WINDOW_P (w
) ? Qt
: Qnil
;
327 DEFUN ("pos-visible-in-window-p", Fpos_visible_in_window_p
,
328 Spos_visible_in_window_p
, 0, 3, 0,
329 doc
: /* Return t if position POS is currently on the frame in WINDOW.
330 Return nil if that position is scrolled vertically out of view.
331 If a character is only partially visible, nil is returned, unless the
332 optional argument PARTIALLY is non-nil.
333 POS defaults to point in WINDOW; WINDOW defaults to the selected window. */)
334 (pos
, window
, partially
)
335 Lisp_Object pos
, window
, partially
;
337 register struct window
*w
;
339 register struct buffer
*buf
;
341 Lisp_Object in_window
;
344 w
= decode_window (window
);
345 buf
= XBUFFER (w
->buffer
);
346 SET_TEXT_POS_FROM_MARKER (top
, w
->start
);
350 CHECK_NUMBER_COERCE_MARKER (pos
);
353 else if (w
== XWINDOW (selected_window
))
356 posint
= XMARKER (w
->pointm
)->charpos
;
358 /* If position is above window start, it's not visible. */
359 if (posint
< CHARPOS (top
))
361 else if (XFASTINT (w
->last_modified
) >= BUF_MODIFF (buf
)
362 && XFASTINT (w
->last_overlay_modified
) >= BUF_OVERLAY_MODIFF (buf
)
363 && posint
< BUF_Z (buf
) - XFASTINT (w
->window_end_pos
))
365 /* If frame is up-to-date, and POSINT is < window end pos, use
366 that info. This doesn't work for POSINT == end pos, because
367 the window end pos is actually the position _after_ the last
368 char in the window. */
369 if (NILP (partially
))
371 pos_visible_p (w
, posint
, &fully_p
, NILP (partially
));
372 in_window
= fully_p
? Qt
: Qnil
;
377 else if (posint
> BUF_ZV (buf
))
379 else if (CHARPOS (top
) < BUF_BEGV (buf
) || CHARPOS (top
) > BUF_ZV (buf
))
380 /* If window start is out of range, do something reasonable. */
384 if (pos_visible_p (w
, posint
, &fully_p
, NILP (partially
)))
385 in_window
= !NILP (partially
) || fully_p
? Qt
: Qnil
;
394 static struct window
*
395 decode_window (window
)
396 register Lisp_Object window
;
399 return XWINDOW (selected_window
);
401 CHECK_LIVE_WINDOW (window
);
402 return XWINDOW (window
);
405 DEFUN ("window-buffer", Fwindow_buffer
, Swindow_buffer
, 0, 1, 0,
406 doc
: /* Return the buffer that WINDOW is displaying. */)
410 return decode_window (window
)->buffer
;
413 DEFUN ("window-height", Fwindow_height
, Swindow_height
, 0, 1, 0,
414 doc
: /* Return the number of lines in WINDOW (including its mode line). */)
418 return decode_window (window
)->height
;
421 DEFUN ("window-width", Fwindow_width
, Swindow_width
, 0, 1, 0,
422 doc
: /* Return the number of display columns in WINDOW.
423 This is the width that is usable columns available for text in WINDOW.
424 If you want to find out how many columns WINDOW takes up,
425 use (let ((edges (window-edges))) (- (nth 2 edges) (nth 0 edges))). */)
429 return make_number (window_internal_width (decode_window (window
)));
432 DEFUN ("window-hscroll", Fwindow_hscroll
, Swindow_hscroll
, 0, 1, 0,
433 doc
: /* Return the number of columns by which WINDOW is scrolled from left margin. */)
437 return decode_window (window
)->hscroll
;
440 DEFUN ("set-window-hscroll", Fset_window_hscroll
, Sset_window_hscroll
, 2, 2, 0,
441 doc
: /* Set number of columns WINDOW is scrolled from left margin to NCOL.
442 NCOL should be zero or positive.
444 Note that if `automatic-hscrolling' is non-nil, you cannot scroll the
445 window so that the location of point becomes invisible. */)
447 Lisp_Object window
, ncol
;
449 struct window
*w
= decode_window (window
);
453 hscroll
= max (0, XINT (ncol
));
455 /* Prevent redisplay shortcuts when changing the hscroll. */
456 if (XINT (w
->hscroll
) != hscroll
)
457 XBUFFER (w
->buffer
)->prevent_redisplay_optimizations_p
= 1;
459 w
->hscroll
= make_number (hscroll
);
463 DEFUN ("window-redisplay-end-trigger", Fwindow_redisplay_end_trigger
,
464 Swindow_redisplay_end_trigger
, 0, 1, 0,
465 doc
: /* Return WINDOW's redisplay end trigger value.
466 See `set-window-redisplay-end-trigger' for more information. */)
470 return decode_window (window
)->redisplay_end_trigger
;
473 DEFUN ("set-window-redisplay-end-trigger", Fset_window_redisplay_end_trigger
,
474 Sset_window_redisplay_end_trigger
, 2, 2, 0,
475 doc
: /* Set WINDOW's redisplay end trigger value to VALUE.
476 VALUE should be a buffer position (typically a marker) or nil.
477 If it is a buffer position, then if redisplay in WINDOW reaches a position
478 beyond VALUE, the functions in `redisplay-end-trigger-functions' are called
479 with two arguments: WINDOW, and the end trigger value.
480 Afterwards the end-trigger value is reset to nil. */)
482 register Lisp_Object window
, value
;
484 register struct window
*w
;
486 w
= decode_window (window
);
487 w
->redisplay_end_trigger
= value
;
491 DEFUN ("window-edges", Fwindow_edges
, Swindow_edges
, 0, 1, 0,
492 doc
: /* Return a list of the edge coordinates of WINDOW.
493 \(LEFT TOP RIGHT BOTTOM), all relative to 0, 0 at top left corner of frame.
494 RIGHT is one more than the rightmost column used by WINDOW,
495 and BOTTOM is one more than the bottommost row used by WINDOW
496 and its mode-line. */)
500 register struct window
*w
= decode_window (window
);
502 return Fcons (w
->left
, Fcons (w
->top
,
503 Fcons (make_number (WINDOW_RIGHT_EDGE (w
)),
504 Fcons (make_number (XFASTINT (w
->top
)
505 + XFASTINT (w
->height
)),
509 /* Test if the character at column *X, row *Y is within window W.
510 If it is not, return 0;
511 if it is in the window's text area,
512 set *x and *y to its location relative to the upper left corner
515 if it is on the window's modeline, return 2;
516 if it is on the border between the window and its right sibling,
518 if it is on the window's top line, return 4;
519 if it is in left or right fringe of the window,
520 return 5 or 6, and convert *X and *Y to window-relative coordinates;
521 if it is in the marginal area to the left/right of the window,
522 return 7 or 8, and convert *X and *Y to window-relative coordinates.
524 X and Y are frame relative pixel coordinates. */
526 static enum window_part
527 coordinates_in_window (w
, x
, y
)
528 register struct window
*w
;
531 /* Let's make this a global enum later, instead of using numbers
533 struct frame
*f
= XFRAME (WINDOW_FRAME (w
));
534 int left_x
, right_x
, top_y
, bottom_y
;
535 enum window_part part
;
536 int ux
= CANON_X_UNIT (f
);
537 int x0
= XFASTINT (w
->left
) * ux
;
538 int x1
= x0
+ XFASTINT (w
->width
) * ux
;
539 /* The width of the area where the vertical line can be dragged.
540 (Between mode lines for instance. */
541 int grabbable_width
= ux
;
542 int lmargin_width
= 0, rmargin_width
= 0;
544 if (*x
< x0
|| *x
>= x1
)
547 /* In what's below, we subtract 1 when computing right_x because we
548 want the rightmost pixel, which is given by left_pixel+width-1. */
549 if (w
->pseudo_window_p
)
552 right_x
= XFASTINT (w
->width
) * CANON_X_UNIT (f
) - 1;
553 top_y
= WINDOW_DISPLAY_TOP_EDGE_PIXEL_Y (w
);
554 bottom_y
= WINDOW_DISPLAY_BOTTOM_EDGE_PIXEL_Y (w
);
558 left_x
= (WINDOW_DISPLAY_LEFT_EDGE_PIXEL_X (w
)
559 - FRAME_INTERNAL_BORDER_WIDTH_SAFE (f
));
560 right_x
= WINDOW_DISPLAY_RIGHT_EDGE_PIXEL_X (w
) - 1;
561 top_y
= (WINDOW_DISPLAY_TOP_EDGE_PIXEL_Y (w
)
562 - FRAME_INTERNAL_BORDER_WIDTH_SAFE (f
));
563 bottom_y
= WINDOW_DISPLAY_BOTTOM_EDGE_PIXEL_Y (w
);
566 /* On the mode line or header line? If it's near the start of
567 the mode or header line of window that's has a horizontal
568 sibling, say it's on the vertical line. That's to be able
569 to resize windows horizontally in case we're using toolkit
572 if (WINDOW_WANTS_MODELINE_P (w
)
573 && *y
>= bottom_y
- CURRENT_MODE_LINE_HEIGHT (w
)
576 /* We're somewhere on the mode line. We consider the place
577 between mode lines of horizontally adjacent mode lines
578 as the vertical border. If scroll bars on the left,
579 return the right window. */
582 if (FRAME_HAS_VERTICAL_SCROLL_BARS_ON_LEFT (f
))
584 if (abs (*x
- x0
) < grabbable_width
)
585 part
= ON_VERTICAL_BORDER
;
587 else if (!WINDOW_RIGHTMOST_P (w
) && abs (*x
- x1
) < grabbable_width
)
588 part
= ON_VERTICAL_BORDER
;
590 else if (WINDOW_WANTS_HEADER_LINE_P (w
)
591 && *y
< top_y
+ CURRENT_HEADER_LINE_HEIGHT (w
)
594 part
= ON_HEADER_LINE
;
596 if (FRAME_HAS_VERTICAL_SCROLL_BARS_ON_LEFT (f
))
598 if (abs (*x
- x0
) < grabbable_width
)
599 part
= ON_VERTICAL_BORDER
;
601 else if (!WINDOW_RIGHTMOST_P (w
) && abs (*x
- x1
) < grabbable_width
)
602 part
= ON_VERTICAL_BORDER
;
604 /* Outside anything interesting? */
608 - FRAME_LEFT_FRINGE_WIDTH (f
)
609 - FRAME_LEFT_SCROLL_BAR_WIDTH (f
) * ux
)
611 + FRAME_RIGHT_FRINGE_WIDTH (f
)
612 + FRAME_RIGHT_SCROLL_BAR_WIDTH (f
) * ux
))
616 else if (FRAME_WINDOW_P (f
))
618 if (!w
->pseudo_window_p
619 && !FRAME_HAS_VERTICAL_SCROLL_BARS (f
)
620 && !WINDOW_RIGHTMOST_P (w
)
621 && (abs (*x
- right_x
- FRAME_RIGHT_FRINGE_WIDTH (f
)) < grabbable_width
))
623 part
= ON_VERTICAL_BORDER
;
625 else if (*x
< left_x
|| *x
> right_x
)
627 /* Other lines than the mode line don't include fringes and
628 scroll bars on the left. */
630 /* Convert X and Y to window-relative pixel coordinates. */
633 part
= *x
< left_x
? ON_LEFT_FRINGE
: ON_RIGHT_FRINGE
;
637 lmargin_width
= window_box_width (w
, LEFT_MARGIN_AREA
);
638 rmargin_width
= window_box_width (w
, RIGHT_MARGIN_AREA
);
639 /* You can never be on a margin area if its width is zero. */
641 && *x
<= window_box_right (w
, LEFT_MARGIN_AREA
))
642 part
= ON_LEFT_MARGIN
;
643 else if (rmargin_width
644 && *x
>= window_box_left (w
, RIGHT_MARGIN_AREA
))
645 part
= ON_RIGHT_MARGIN
;
656 /* Need to say "*x > right_x" rather than >=, since on character
657 terminals, the vertical line's x coordinate is right_x. */
658 if (*x
< left_x
|| *x
> right_x
)
660 /* Other lines than the mode line don't include fringes and
661 scroll bars on the left. */
663 /* Convert X and Y to window-relative pixel coordinates. */
666 part
= *x
< left_x
? ON_LEFT_FRINGE
: ON_RIGHT_FRINGE
;
668 /* Here, too, "*x > right_x" is because of character terminals. */
669 else if (!w
->pseudo_window_p
670 && !WINDOW_RIGHTMOST_P (w
)
671 && *x
> right_x
- ux
)
673 /* On the border on the right side of the window? Assume that
674 this area begins at RIGHT_X minus a canonical char width. */
675 part
= ON_VERTICAL_BORDER
;
679 lmargin_width
= window_box_width (w
, LEFT_MARGIN_AREA
);
680 rmargin_width
= window_box_width (w
, RIGHT_MARGIN_AREA
);
681 /* You can never be on a margin area if its width is zero.
682 This is especially important for character terminals. */
684 && *x
<= window_box_right (w
, LEFT_MARGIN_AREA
))
685 part
= ON_LEFT_MARGIN
;
686 else if (rmargin_width
687 && *x
>= window_box_left (w
, RIGHT_MARGIN_AREA
))
688 part
= ON_RIGHT_MARGIN
;
692 /* Convert X and Y to window-relative pixel coordinates. */
703 DEFUN ("coordinates-in-window-p", Fcoordinates_in_window_p
,
704 Scoordinates_in_window_p
, 2, 2, 0,
705 doc
: /* Return non-nil if COORDINATES are in WINDOW.
706 COORDINATES is a cons of the form (X . Y), X and Y being distances
707 measured in characters from the upper-left corner of the frame.
708 \(0 . 0) denotes the character in the upper left corner of the
710 If COORDINATES are in the text portion of WINDOW,
711 the coordinates relative to the window are returned.
712 If they are in the mode line of WINDOW, `mode-line' is returned.
713 If they are in the top mode line of WINDOW, `header-line' is returned.
714 If they are in the left fringe of WINDOW, `left-fringe' is returned.
715 If they are in the right fringe of WINDOW, `right-fringe' is returned.
716 If they are on the border between WINDOW and its right sibling,
717 `vertical-line' is returned.
718 If they are in the windows's left or right marginal areas, `left-margin'\n\
719 or `right-margin' is returned. */)
720 (coordinates
, window
)
721 register Lisp_Object coordinates
, window
;
728 CHECK_LIVE_WINDOW (window
);
729 w
= XWINDOW (window
);
730 f
= XFRAME (w
->frame
);
731 CHECK_CONS (coordinates
);
732 lx
= Fcar (coordinates
);
733 ly
= Fcdr (coordinates
);
734 CHECK_NUMBER_OR_FLOAT (lx
);
735 CHECK_NUMBER_OR_FLOAT (ly
);
736 x
= PIXEL_X_FROM_CANON_X (f
, lx
);
737 y
= PIXEL_Y_FROM_CANON_Y (f
, ly
);
739 switch (coordinates_in_window (w
, &x
, &y
))
745 /* X and Y are now window relative pixel coordinates. Convert
746 them to canonical char units before returning them. */
747 return Fcons (CANON_X_FROM_PIXEL_X (f
, x
),
748 CANON_Y_FROM_PIXEL_Y (f
, y
));
753 case ON_VERTICAL_BORDER
:
754 return Qvertical_line
;
762 case ON_RIGHT_FRINGE
:
763 return Qright_fringe
;
768 case ON_RIGHT_MARGIN
:
769 return Qright_margin
;
777 /* Callback for foreach_window, used in window_from_coordinates.
778 Check if window W contains coordinates specified by USER_DATA which
779 is actually a pointer to a struct check_window_data CW.
781 Check if window W contains coordinates *CW->x and *CW->y. If it
782 does, return W in *CW->window, as Lisp_Object, and return in
783 *CW->part the part of the window under coordinates *X,*Y. Return
784 zero from this function to stop iterating over windows. */
786 struct check_window_data
793 check_window_containing (w
, user_data
)
797 struct check_window_data
*cw
= (struct check_window_data
*) user_data
;
798 enum window_part found
;
801 found
= coordinates_in_window (w
, cw
->x
, cw
->y
);
802 if (found
!= ON_NOTHING
)
804 *cw
->part
= found
- 1;
805 XSETWINDOW (*cw
->window
, w
);
813 /* Find the window containing frame-relative pixel position X/Y and
814 return it as a Lisp_Object. If X, Y is on the window's modeline,
815 set *PART to 1; if it is on the separating line between the window
816 and its right sibling, set it to 2; otherwise set it to 0. If
817 there is no window under X, Y return nil and leave *PART
818 unmodified. TOOL_BAR_P non-zero means detect tool-bar windows.
820 This function was previously implemented with a loop cycling over
821 windows with Fnext_window, and starting with the frame's selected
822 window. It turned out that this doesn't work with an
823 implementation of next_window using Vwindow_list, because
824 FRAME_SELECTED_WINDOW (F) is not always contained in the window
825 tree of F when this function is called asynchronously from
826 note_mouse_highlight. The original loop didn't terminate in this
830 window_from_coordinates (f
, x
, y
, part
, tool_bar_p
)
837 struct check_window_data cw
;
840 cw
.window
= &window
, cw
.x
= &x
, cw
.y
= &y
; cw
.part
= part
;
841 foreach_window (f
, check_window_containing
, &cw
);
843 /* If not found above, see if it's in the tool bar window, if a tool
847 && WINDOWP (f
->tool_bar_window
)
848 && XINT (XWINDOW (f
->tool_bar_window
)->height
) > 0
849 && (coordinates_in_window (XWINDOW (f
->tool_bar_window
), &x
, &y
)
853 window
= f
->tool_bar_window
;
859 DEFUN ("window-at", Fwindow_at
, Swindow_at
, 2, 3, 0,
860 doc
: /* Return window containing coordinates X and Y on FRAME.
861 If omitted, FRAME defaults to the currently selected frame.
862 The top left corner of the frame is considered to be row 0,
865 Lisp_Object x
, y
, frame
;
871 frame
= selected_frame
;
872 CHECK_LIVE_FRAME (frame
);
875 /* Check that arguments are integers or floats. */
876 CHECK_NUMBER_OR_FLOAT (x
);
877 CHECK_NUMBER_OR_FLOAT (y
);
879 return window_from_coordinates (f
,
880 PIXEL_X_FROM_CANON_X (f
, x
),
881 PIXEL_Y_FROM_CANON_Y (f
, y
),
885 DEFUN ("window-point", Fwindow_point
, Swindow_point
, 0, 1, 0,
886 doc
: /* Return current value of point in WINDOW.
887 For a nonselected window, this is the value point would have
888 if that window were selected.
890 Note that, when WINDOW is the selected window and its buffer
891 is also currently selected, the value returned is the same as (point).
892 It would be more strictly correct to return the `top-level' value
893 of point, outside of any save-excursion forms.
894 But that is hard to define. */)
898 register struct window
*w
= decode_window (window
);
900 if (w
== XWINDOW (selected_window
)
901 && current_buffer
== XBUFFER (w
->buffer
))
903 return Fmarker_position (w
->pointm
);
906 DEFUN ("window-start", Fwindow_start
, Swindow_start
, 0, 1, 0,
907 doc
: /* Return position at which display currently starts in WINDOW.
908 This is updated by redisplay or by calling `set-window-start'. */)
912 return Fmarker_position (decode_window (window
)->start
);
915 /* This is text temporarily removed from the doc string below.
917 This function returns nil if the position is not currently known.
918 That happens when redisplay is preempted and doesn't finish.
919 If in that case you want to compute where the end of the window would
920 have been if redisplay had finished, do this:
922 (goto-char (window-start window))
923 (vertical-motion (1- (window-height window)) window)
926 DEFUN ("window-end", Fwindow_end
, Swindow_end
, 0, 2, 0,
927 doc
: /* Return position at which display currently ends in WINDOW.
928 This is updated by redisplay, when it runs to completion.
929 Simply changing the buffer text or setting `window-start'
930 does not update this value.
931 If UPDATE is non-nil, compute the up-to-date position
932 if it isn't already recorded. */)
934 Lisp_Object window
, update
;
937 struct window
*w
= decode_window (window
);
943 #if 0 /* This change broke some things. We should make it later. */
944 /* If we don't know the end position, return nil.
945 The user can compute it with vertical-motion if he wants to.
946 It would be nicer to do it automatically,
947 but that's so slow that it would probably bother people. */
948 if (NILP (w
->window_end_valid
))
953 && ! (! NILP (w
->window_end_valid
)
954 && XFASTINT (w
->last_modified
) >= MODIFF
))
956 struct text_pos startp
;
958 struct buffer
*old_buffer
= NULL
, *b
= XBUFFER (buf
);
960 /* In case W->start is out of the range, use something
961 reasonable. This situation occurred when loading a file with
962 `-l' containing a call to `rmail' with subsequent other
963 commands. At the end, W->start happened to be BEG, while
964 rmail had already narrowed the buffer. */
965 if (XMARKER (w
->start
)->charpos
< BEGV
)
966 SET_TEXT_POS (startp
, BEGV
, BEGV_BYTE
);
967 else if (XMARKER (w
->start
)->charpos
> ZV
)
968 SET_TEXT_POS (startp
, ZV
, ZV_BYTE
);
970 SET_TEXT_POS_FROM_MARKER (startp
, w
->start
);
972 /* Cannot use Fvertical_motion because that function doesn't
973 cope with variable-height lines. */
974 if (b
!= current_buffer
)
976 old_buffer
= current_buffer
;
977 set_buffer_internal (b
);
980 start_display (&it
, w
, startp
);
981 move_it_vertically (&it
, window_box_height (w
));
982 if (it
.current_y
< it
.last_visible_y
)
983 move_it_past_eol (&it
);
984 value
= make_number (IT_CHARPOS (it
));
987 set_buffer_internal (old_buffer
);
990 XSETINT (value
, BUF_Z (XBUFFER (buf
)) - XFASTINT (w
->window_end_pos
));
995 DEFUN ("set-window-point", Fset_window_point
, Sset_window_point
, 2, 2, 0,
996 doc
: /* Make point value in WINDOW be at position POS in WINDOW's buffer. */)
998 Lisp_Object window
, pos
;
1000 register struct window
*w
= decode_window (window
);
1002 CHECK_NUMBER_COERCE_MARKER (pos
);
1003 if (w
== XWINDOW (selected_window
)
1004 && XBUFFER (w
->buffer
) == current_buffer
)
1007 set_marker_restricted (w
->pointm
, pos
, w
->buffer
);
1009 /* We have to make sure that redisplay updates the window to show
1010 the new value of point. */
1011 if (!EQ (window
, selected_window
))
1012 ++windows_or_buffers_changed
;
1017 DEFUN ("set-window-start", Fset_window_start
, Sset_window_start
, 2, 3, 0,
1018 doc
: /* Make display in WINDOW start at position POS in WINDOW's buffer.
1019 Optional third arg NOFORCE non-nil inhibits next redisplay
1020 from overriding motion of point in order to display at this exact start. */)
1021 (window
, pos
, noforce
)
1022 Lisp_Object window
, pos
, noforce
;
1024 register struct window
*w
= decode_window (window
);
1026 CHECK_NUMBER_COERCE_MARKER (pos
);
1027 set_marker_restricted (w
->start
, pos
, w
->buffer
);
1028 /* this is not right, but much easier than doing what is right. */
1029 w
->start_at_line_beg
= Qnil
;
1031 w
->force_start
= Qt
;
1032 w
->update_mode_line
= Qt
;
1033 XSETFASTINT (w
->last_modified
, 0);
1034 XSETFASTINT (w
->last_overlay_modified
, 0);
1035 if (!EQ (window
, selected_window
))
1036 windows_or_buffers_changed
++;
1041 DEFUN ("window-dedicated-p", Fwindow_dedicated_p
, Swindow_dedicated_p
,
1043 doc
: /* Return WINDOW's dedicated object, usually t or nil.
1044 See also `set-window-dedicated-p'. */)
1048 return decode_window (window
)->dedicated
;
1051 DEFUN ("set-window-dedicated-p", Fset_window_dedicated_p
,
1052 Sset_window_dedicated_p
, 2, 2, 0,
1053 doc
: /* Control whether WINDOW is dedicated to the buffer it displays.
1054 If it is dedicated, Emacs will not automatically change
1055 which buffer appears in it.
1056 The second argument is the new value for the dedication flag;
1057 non-nil means yes. */)
1059 Lisp_Object window
, arg
;
1061 register struct window
*w
= decode_window (window
);
1064 w
->dedicated
= Qnil
;
1068 return w
->dedicated
;
1071 DEFUN ("window-display-table", Fwindow_display_table
, Swindow_display_table
,
1073 doc
: /* Return the display-table that WINDOW is using. */)
1077 return decode_window (window
)->display_table
;
1080 /* Get the display table for use on window W. This is either W's
1081 display table or W's buffer's display table. Ignore the specified
1082 tables if they are not valid; if no valid table is specified,
1085 struct Lisp_Char_Table
*
1086 window_display_table (w
)
1089 struct Lisp_Char_Table
*dp
= NULL
;
1091 if (DISP_TABLE_P (w
->display_table
))
1092 dp
= XCHAR_TABLE (w
->display_table
);
1093 else if (BUFFERP (w
->buffer
))
1095 struct buffer
*b
= XBUFFER (w
->buffer
);
1097 if (DISP_TABLE_P (b
->display_table
))
1098 dp
= XCHAR_TABLE (b
->display_table
);
1099 else if (DISP_TABLE_P (Vstandard_display_table
))
1100 dp
= XCHAR_TABLE (Vstandard_display_table
);
1106 DEFUN ("set-window-display-table", Fset_window_display_table
, Sset_window_display_table
, 2, 2, 0,
1107 doc
: /* Set WINDOW's display-table to TABLE. */)
1109 register Lisp_Object window
, table
;
1111 register struct window
*w
;
1113 w
= decode_window (window
);
1114 w
->display_table
= table
;
1118 /* Record info on buffer window w is displaying
1119 when it is about to cease to display that buffer. */
1122 register struct window
*w
;
1129 if (b
!= XMARKER (w
->pointm
)->buffer
)
1133 if (w
== XWINDOW (selected_window
)
1134 || ! EQ (buf
, XWINDOW (selected_window
)->buffer
))
1135 /* Do this except when the selected window's buffer
1136 is being removed from some other window. */
1138 /* last_window_start records the start position that this buffer
1139 had in the last window to be disconnected from it.
1140 Now that this statement is unconditional,
1141 it is possible for the buffer to be displayed in the
1142 selected window, while last_window_start reflects another
1143 window which was recently showing the same buffer.
1144 Some people might say that might be a good thing. Let's see. */
1145 b
->last_window_start
= marker_position (w
->start
);
1147 /* Point in the selected window's buffer
1148 is actually stored in that buffer, and the window's pointm isn't used.
1149 So don't clobber point in that buffer. */
1150 if (! EQ (buf
, XWINDOW (selected_window
)->buffer
)
1151 /* This line helps to fix Horsley's testbug.el bug. */
1152 && !(WINDOWP (b
->last_selected_window
)
1153 && w
!= XWINDOW (b
->last_selected_window
)
1154 && EQ (buf
, XWINDOW (b
->last_selected_window
)->buffer
)))
1155 temp_set_point_both (b
,
1156 clip_to_bounds (BUF_BEGV (b
),
1157 XMARKER (w
->pointm
)->charpos
,
1159 clip_to_bounds (BUF_BEGV_BYTE (b
),
1160 marker_byte_position (w
->pointm
),
1163 if (WINDOWP (b
->last_selected_window
)
1164 && w
== XWINDOW (b
->last_selected_window
))
1165 b
->last_selected_window
= Qnil
;
1168 /* Put replacement into the window structure in place of old. */
1170 replace_window (old
, replacement
)
1171 Lisp_Object old
, replacement
;
1173 register Lisp_Object tem
;
1174 register struct window
*o
= XWINDOW (old
), *p
= XWINDOW (replacement
);
1176 /* If OLD is its frame's root_window, then replacement is the new
1177 root_window for that frame. */
1179 if (EQ (old
, FRAME_ROOT_WINDOW (XFRAME (o
->frame
))))
1180 FRAME_ROOT_WINDOW (XFRAME (o
->frame
)) = replacement
;
1184 p
->width
= o
->width
;
1185 p
->height
= o
->height
;
1186 p
->desired_matrix
= p
->current_matrix
= 0;
1188 bzero (&p
->cursor
, sizeof (p
->cursor
));
1189 bzero (&p
->last_cursor
, sizeof (p
->last_cursor
));
1190 bzero (&p
->phys_cursor
, sizeof (p
->phys_cursor
));
1191 p
->phys_cursor_type
= -1;
1192 p
->phys_cursor_width
= -1;
1193 p
->must_be_updated_p
= 0;
1194 p
->pseudo_window_p
= 0;
1195 XSETFASTINT (p
->window_end_vpos
, 0);
1196 XSETFASTINT (p
->window_end_pos
, 0);
1197 p
->window_end_valid
= Qnil
;
1198 p
->frozen_window_start_p
= 0;
1199 p
->orig_top
= p
->orig_height
= Qnil
;
1201 p
->next
= tem
= o
->next
;
1203 XWINDOW (tem
)->prev
= replacement
;
1205 p
->prev
= tem
= o
->prev
;
1207 XWINDOW (tem
)->next
= replacement
;
1209 p
->parent
= tem
= o
->parent
;
1212 if (EQ (XWINDOW (tem
)->vchild
, old
))
1213 XWINDOW (tem
)->vchild
= replacement
;
1214 if (EQ (XWINDOW (tem
)->hchild
, old
))
1215 XWINDOW (tem
)->hchild
= replacement
;
1218 /*** Here, if replacement is a vertical combination
1219 and so is its new parent, we should make replacement's
1220 children be children of that parent instead. ***/
1223 DEFUN ("delete-window", Fdelete_window
, Sdelete_window
, 0, 1, "",
1224 doc
: /* Remove WINDOW from the display. Default is selected window. */)
1226 register Lisp_Object window
;
1228 delete_window (window
);
1230 if (! NILP (Vwindow_configuration_change_hook
)
1231 && ! NILP (Vrun_hooks
))
1232 call1 (Vrun_hooks
, Qwindow_configuration_change_hook
);
1238 delete_window (window
)
1239 register Lisp_Object window
;
1241 register Lisp_Object tem
, parent
, sib
;
1242 register struct window
*p
;
1243 register struct window
*par
;
1246 /* Because this function is called by other C code on non-leaf
1247 windows, the CHECK_LIVE_WINDOW macro would choke inappropriately,
1248 so we can't decode_window here. */
1250 window
= selected_window
;
1252 CHECK_WINDOW (window
);
1253 p
= XWINDOW (window
);
1255 /* It's okay to delete an already-deleted window. */
1256 if (NILP (p
->buffer
)
1258 && NILP (p
->vchild
))
1263 error ("Attempt to delete minibuffer or sole ordinary window");
1264 par
= XWINDOW (parent
);
1266 windows_or_buffers_changed
++;
1267 Vwindow_list
= Qnil
;
1268 f
= XFRAME (WINDOW_FRAME (p
));
1269 FRAME_WINDOW_SIZES_CHANGED (f
) = 1;
1271 /* Are we trying to delete any frame's selected window? */
1273 Lisp_Object swindow
, pwindow
;
1275 /* See if the frame's selected window is either WINDOW
1276 or any subwindow of it, by finding all that window's parents
1277 and comparing each one with WINDOW. */
1278 swindow
= FRAME_SELECTED_WINDOW (f
);
1283 while (!NILP (pwindow
))
1285 if (EQ (window
, pwindow
))
1287 pwindow
= XWINDOW (pwindow
)->parent
;
1290 /* If the window being deleted is not a parent of SWINDOW,
1291 then SWINDOW is ok as the new selected window. */
1292 if (!EQ (window
, pwindow
))
1294 /* Otherwise, try another window for SWINDOW. */
1295 swindow
= Fnext_window (swindow
, Qlambda
, Qnil
);;
1297 /* If we get back to the frame's selected window,
1298 it means there was no acceptable alternative,
1299 so we cannot delete. */
1300 if (EQ (swindow
, FRAME_SELECTED_WINDOW (f
)))
1301 error ("Cannot delete window");
1304 /* If we need to change SWINDOW, do it. */
1305 if (! EQ (swindow
, FRAME_SELECTED_WINDOW (f
)))
1307 /* If we're about to delete the selected window on the
1308 selected frame, then we should use Fselect_window to select
1309 the new window. On the other hand, if we're about to
1310 delete the selected window on any other frame, we shouldn't do
1311 anything but set the frame's selected_window slot. */
1312 if (EQ (FRAME_SELECTED_WINDOW (f
), selected_window
))
1313 Fselect_window (swindow
);
1315 FRAME_SELECTED_WINDOW (f
) = swindow
;
1320 /* tem is null for dummy parent windows
1321 (which have inferiors but not any contents themselves) */
1325 unchain_marker (p
->pointm
);
1326 unchain_marker (p
->start
);
1329 /* Free window glyph matrices. It is sure that they are allocated
1330 again when ADJUST_GLYPHS is called. Block input so that expose
1331 events and other events that access glyph matrices are not
1332 processed while we are changing them. */
1334 free_window_matrices (XWINDOW (FRAME_ROOT_WINDOW (f
)));
1338 XWINDOW (tem
)->prev
= p
->prev
;
1342 XWINDOW (tem
)->next
= p
->next
;
1344 if (EQ (window
, par
->hchild
))
1345 par
->hchild
= p
->next
;
1346 if (EQ (window
, par
->vchild
))
1347 par
->vchild
= p
->next
;
1349 /* Find one of our siblings to give our space to. */
1353 /* If p gives its space to its next sibling, that sibling needs
1354 to have its top/left side pulled back to where p's is.
1355 set_window_{height,width} will re-position the sibling's
1358 XWINDOW (sib
)->top
= p
->top
;
1359 XWINDOW (sib
)->left
= p
->left
;
1362 /* Stretch that sibling. */
1363 if (!NILP (par
->vchild
))
1364 set_window_height (sib
,
1365 XFASTINT (XWINDOW (sib
)->height
) + XFASTINT (p
->height
),
1367 if (!NILP (par
->hchild
))
1368 set_window_width (sib
,
1369 XFASTINT (XWINDOW (sib
)->width
) + XFASTINT (p
->width
),
1372 /* If parent now has only one child,
1373 put the child into the parent's place. */
1377 if (NILP (XWINDOW (tem
)->next
))
1378 replace_window (parent
, tem
);
1380 /* Since we may be deleting combination windows, we must make sure that
1381 not only p but all its children have been marked as deleted. */
1382 if (! NILP (p
->hchild
))
1383 delete_all_subwindows (XWINDOW (p
->hchild
));
1384 else if (! NILP (p
->vchild
))
1385 delete_all_subwindows (XWINDOW (p
->vchild
));
1387 /* Mark this window as deleted. */
1388 p
->buffer
= p
->hchild
= p
->vchild
= Qnil
;
1390 /* Adjust glyph matrices. */
1397 /***********************************************************************
1399 ***********************************************************************/
1401 /* Add window W to *USER_DATA. USER_DATA is actually a Lisp_Object
1402 pointer. This is a callback function for foreach_window, used in
1403 function window_list. */
1406 add_window_to_list (w
, user_data
)
1410 Lisp_Object
*list
= (Lisp_Object
*) user_data
;
1412 XSETWINDOW (window
, w
);
1413 *list
= Fcons (window
, *list
);
1418 /* Return a list of all windows, for use by next_window. If
1419 Vwindow_list is a list, return that list. Otherwise, build a new
1420 list, cache it in Vwindow_list, and return that. */
1425 if (!CONSP (Vwindow_list
))
1429 Vwindow_list
= Qnil
;
1430 for (tail
= Vframe_list
; CONSP (tail
); tail
= XCDR (tail
))
1432 Lisp_Object args
[2];
1434 /* We are visiting windows in canonical order, and add
1435 new windows at the front of args[1], which means we
1436 have to reverse this list at the end. */
1438 foreach_window (XFRAME (XCAR (tail
)), add_window_to_list
, &args
[1]);
1439 args
[0] = Vwindow_list
;
1440 args
[1] = Fnreverse (args
[1]);
1441 Vwindow_list
= Fnconc (2, args
);
1445 return Vwindow_list
;
1449 /* Value is non-zero if WINDOW satisfies the constraints given by
1450 OWINDOW, MINIBUF and ALL_FRAMES.
1452 MINIBUF t means WINDOW may be minibuffer windows.
1453 `lambda' means WINDOW may not be a minibuffer window.
1454 a window means a specific minibuffer window
1456 ALL_FRAMES t means search all frames,
1457 nil means search just current frame,
1458 `visible' means search just visible frames,
1459 0 means search visible and iconified frames,
1460 a window means search the frame that window belongs to,
1461 a frame means consider windows on that frame, only. */
1464 candidate_window_p (window
, owindow
, minibuf
, all_frames
)
1465 Lisp_Object window
, owindow
, minibuf
, all_frames
;
1467 struct window
*w
= XWINDOW (window
);
1468 struct frame
*f
= XFRAME (w
->frame
);
1469 int candidate_p
= 1;
1471 if (!BUFFERP (w
->buffer
))
1473 else if (MINI_WINDOW_P (w
)
1474 && (EQ (minibuf
, Qlambda
)
1475 || (WINDOWP (minibuf
) && !EQ (minibuf
, window
))))
1477 /* If MINIBUF is `lambda' don't consider any mini-windows.
1478 If it is a window, consider only that one. */
1481 else if (EQ (all_frames
, Qt
))
1483 else if (NILP (all_frames
))
1485 xassert (WINDOWP (owindow
));
1486 candidate_p
= EQ (w
->frame
, XWINDOW (owindow
)->frame
);
1488 else if (EQ (all_frames
, Qvisible
))
1490 FRAME_SAMPLE_VISIBILITY (f
);
1491 candidate_p
= FRAME_VISIBLE_P (f
);
1493 else if (INTEGERP (all_frames
) && XINT (all_frames
) == 0)
1495 FRAME_SAMPLE_VISIBILITY (f
);
1496 candidate_p
= FRAME_VISIBLE_P (f
) || FRAME_ICONIFIED_P (f
);
1498 else if (WINDOWP (all_frames
))
1499 candidate_p
= (EQ (FRAME_MINIBUF_WINDOW (f
), all_frames
)
1500 || EQ (XWINDOW (all_frames
)->frame
, w
->frame
)
1501 || EQ (XWINDOW (all_frames
)->frame
, FRAME_FOCUS_FRAME (f
)));
1502 else if (FRAMEP (all_frames
))
1503 candidate_p
= EQ (all_frames
, w
->frame
);
1509 /* Decode arguments as allowed by Fnext_window, Fprevious_window, and
1510 Fwindow_list. See there for the meaning of WINDOW, MINIBUF, and
1514 decode_next_window_args (window
, minibuf
, all_frames
)
1515 Lisp_Object
*window
, *minibuf
, *all_frames
;
1518 *window
= selected_window
;
1520 CHECK_LIVE_WINDOW (*window
);
1522 /* MINIBUF nil may or may not include minibuffers. Decide if it
1524 if (NILP (*minibuf
))
1525 *minibuf
= minibuf_level
? minibuf_window
: Qlambda
;
1526 else if (!EQ (*minibuf
, Qt
))
1529 /* Now *MINIBUF can be t => count all minibuffer windows, `lambda'
1530 => count none of them, or a specific minibuffer window (the
1531 active one) to count. */
1533 /* ALL_FRAMES nil doesn't specify which frames to include. */
1534 if (NILP (*all_frames
))
1535 *all_frames
= (!EQ (*minibuf
, Qlambda
)
1536 ? FRAME_MINIBUF_WINDOW (XFRAME (XWINDOW (*window
)->frame
))
1538 else if (EQ (*all_frames
, Qvisible
))
1540 else if (XFASTINT (*all_frames
) == 0)
1542 else if (FRAMEP (*all_frames
))
1544 else if (!EQ (*all_frames
, Qt
))
1547 /* Now *ALL_FRAMES is t meaning search all frames, nil meaning
1548 search just current frame, `visible' meaning search just visible
1549 frames, 0 meaning search visible and iconified frames, or a
1550 window, meaning search the frame that window belongs to, or a
1551 frame, meaning consider windows on that frame, only. */
1555 /* Return the next or previous window of WINDOW in canonical ordering
1556 of windows. NEXT_P non-zero means return the next window. See the
1557 documentation string of next-window for the meaning of MINIBUF and
1561 next_window (window
, minibuf
, all_frames
, next_p
)
1562 Lisp_Object window
, minibuf
, all_frames
;
1565 decode_next_window_args (&window
, &minibuf
, &all_frames
);
1567 /* If ALL_FRAMES is a frame, and WINDOW isn't on that frame, just
1568 return the first window on the frame. */
1569 if (FRAMEP (all_frames
)
1570 && !EQ (all_frames
, XWINDOW (window
)->frame
))
1571 return Fframe_first_window (all_frames
);
1577 /* Find WINDOW in the list of all windows. */
1578 list
= Fmemq (window
, window_list ());
1580 /* Scan forward from WINDOW to the end of the window list. */
1582 for (list
= XCDR (list
); CONSP (list
); list
= XCDR (list
))
1583 if (candidate_window_p (XCAR (list
), window
, minibuf
, all_frames
))
1586 /* Scan from the start of the window list up to WINDOW. */
1588 for (list
= Vwindow_list
;
1589 CONSP (list
) && !EQ (XCAR (list
), window
);
1591 if (candidate_window_p (XCAR (list
), window
, minibuf
, all_frames
))
1595 window
= XCAR (list
);
1599 Lisp_Object candidate
, list
;
1601 /* Scan through the list of windows for candidates. If there are
1602 candidate windows in front of WINDOW, the last one of these
1603 is the one we want. If there are candidates following WINDOW
1604 in the list, again the last one of these is the one we want. */
1606 for (list
= window_list (); CONSP (list
); list
= XCDR (list
))
1608 if (EQ (XCAR (list
), window
))
1610 if (WINDOWP (candidate
))
1613 else if (candidate_window_p (XCAR (list
), window
, minibuf
,
1615 candidate
= XCAR (list
);
1618 if (WINDOWP (candidate
))
1626 DEFUN ("next-window", Fnext_window
, Snext_window
, 0, 3, 0,
1627 doc
: /* Return next window after WINDOW in canonical ordering of windows.
1628 If omitted, WINDOW defaults to the selected window.
1630 Optional second arg MINIBUF t means count the minibuffer window even
1631 if not active. MINIBUF nil or omitted means count the minibuffer iff
1632 it is active. MINIBUF neither t nor nil means not to count the
1633 minibuffer even if it is active.
1635 Several frames may share a single minibuffer; if the minibuffer
1636 counts, all windows on all frames that share that minibuffer count
1637 too. Therefore, `next-window' can be used to iterate through the
1638 set of windows even when the minibuffer is on another frame. If the
1639 minibuffer does not count, only windows from WINDOW's frame count.
1641 Optional third arg ALL-FRAMES t means include windows on all frames.
1642 ALL-FRAMES nil or omitted means cycle within the frames as specified
1643 above. ALL-FRAMES = `visible' means include windows on all visible frames.
1644 ALL-FRAMES = 0 means include windows on all visible and iconified frames.
1645 If ALL-FRAMES is a frame, restrict search to windows on that frame.
1646 Anything else means restrict to WINDOW's frame.
1648 If you use consistent values for MINIBUF and ALL-FRAMES, you can use
1649 `next-window' to iterate through the entire cycle of acceptable
1650 windows, eventually ending up back at the window you started with.
1651 `previous-window' traverses the same cycle, in the reverse order. */)
1652 (window
, minibuf
, all_frames
)
1653 Lisp_Object window
, minibuf
, all_frames
;
1655 return next_window (window
, minibuf
, all_frames
, 1);
1659 DEFUN ("previous-window", Fprevious_window
, Sprevious_window
, 0, 3, 0,
1660 doc
: /* Return the window preceding WINDOW in canonical ordering of windows.
1661 If omitted, WINDOW defaults to the selected window.
1663 Optional second arg MINIBUF t means count the minibuffer window even
1664 if not active. MINIBUF nil or omitted means count the minibuffer iff
1665 it is active. MINIBUF neither t nor nil means not to count the
1666 minibuffer even if it is active.
1668 Several frames may share a single minibuffer; if the minibuffer
1669 counts, all windows on all frames that share that minibuffer count
1670 too. Therefore, `previous-window' can be used to iterate through
1671 the set of windows even when the minibuffer is on another frame. If
1672 the minibuffer does not count, only windows from WINDOW's frame count
1674 Optional third arg ALL-FRAMES t means include windows on all frames.
1675 ALL-FRAMES nil or omitted means cycle within the frames as specified
1676 above. ALL-FRAMES = `visible' means include windows on all visible frames.
1677 ALL-FRAMES = 0 means include windows on all visible and iconified frames.
1678 If ALL-FRAMES is a frame, restrict search to windows on that frame.
1679 Anything else means restrict to WINDOW's frame.
1681 If you use consistent values for MINIBUF and ALL-FRAMES, you can use
1682 `previous-window' to iterate through the entire cycle of acceptable
1683 windows, eventually ending up back at the window you started with.
1684 `next-window' traverses the same cycle, in the reverse order. */)
1685 (window
, minibuf
, all_frames
)
1686 Lisp_Object window
, minibuf
, all_frames
;
1688 return next_window (window
, minibuf
, all_frames
, 0);
1692 DEFUN ("other-window", Fother_window
, Sother_window
, 1, 2, "p",
1693 doc
: /* Select the ARG'th different window on this frame.
1694 All windows on current frame are arranged in a cyclic order.
1695 This command selects the window ARG steps away in that order.
1696 A negative ARG moves in the opposite order. If the optional second
1697 argument ALL_FRAMES is non-nil, cycle through all frames. */)
1699 Lisp_Object arg
, all_frames
;
1705 window
= selected_window
;
1707 for (i
= XINT (arg
); i
> 0; --i
)
1708 window
= Fnext_window (window
, Qnil
, all_frames
);
1710 window
= Fprevious_window (window
, Qnil
, all_frames
);
1712 Fselect_window (window
);
1717 DEFUN ("window-list", Fwindow_list
, Swindow_list
, 0, 3, 0,
1718 doc
: /* Return a list of windows on FRAME, starting with WINDOW.
1719 FRAME nil or omitted means use the selected frame.
1720 WINDOW nil or omitted means use the selected window.
1721 MINIBUF t means include the minibuffer window, even if it isn't active.
1722 MINIBUF nil or omitted means include the minibuffer window only
1724 MINIBUF neither nil nor t means never include the minibuffer window. */)
1725 (frame
, minibuf
, window
)
1726 Lisp_Object frame
, minibuf
, window
;
1729 window
= selected_window
;
1731 frame
= selected_frame
;
1733 if (!EQ (frame
, XWINDOW (window
)->frame
))
1734 error ("Window is on a different frame");
1736 return window_list_1 (window
, minibuf
, frame
);
1740 /* Return a list of windows in canonical ordering. Arguments are like
1741 for `next-window'. */
1744 window_list_1 (window
, minibuf
, all_frames
)
1745 Lisp_Object window
, minibuf
, all_frames
;
1747 Lisp_Object tail
, list
;
1749 decode_next_window_args (&window
, &minibuf
, &all_frames
);
1752 for (tail
= window_list (); CONSP (tail
); tail
= XCDR (tail
))
1753 if (candidate_window_p (XCAR (tail
), window
, minibuf
, all_frames
))
1754 list
= Fcons (XCAR (tail
), list
);
1756 return Fnreverse (list
);
1761 /* Look at all windows, performing an operation specified by TYPE
1763 If FRAMES is Qt, look at all frames;
1764 Qnil, look at just the selected frame;
1765 Qvisible, look at visible frames;
1766 a frame, just look at windows on that frame.
1767 If MINI is non-zero, perform the operation on minibuffer windows too. */
1772 GET_BUFFER_WINDOW
, /* Arg is buffer */
1773 GET_LRU_WINDOW
, /* Arg is t for full-width windows only */
1774 DELETE_OTHER_WINDOWS
, /* Arg is window not to delete */
1775 DELETE_BUFFER_WINDOWS
, /* Arg is buffer */
1777 UNSHOW_BUFFER
, /* Arg is buffer */
1782 window_loop (type
, obj
, mini
, frames
)
1783 enum window_loop type
;
1784 Lisp_Object obj
, frames
;
1787 Lisp_Object window
, windows
, best_window
, frame_arg
;
1789 struct gcpro gcpro1
;
1791 /* If we're only looping through windows on a particular frame,
1792 frame points to that frame. If we're looping through windows
1793 on all frames, frame is 0. */
1794 if (FRAMEP (frames
))
1795 f
= XFRAME (frames
);
1796 else if (NILP (frames
))
1797 f
= SELECTED_FRAME ();
1802 frame_arg
= Qlambda
;
1803 else if (XFASTINT (frames
) == 0)
1805 else if (EQ (frames
, Qvisible
))
1810 /* frame_arg is Qlambda to stick to one frame,
1811 Qvisible to consider all visible frames,
1814 /* Pick a window to start with. */
1818 window
= FRAME_SELECTED_WINDOW (f
);
1820 window
= FRAME_SELECTED_WINDOW (SELECTED_FRAME ());
1822 windows
= window_list_1 (window
, mini
? Qt
: Qnil
, frame_arg
);
1826 for (; CONSP (windows
); windows
= CDR (windows
))
1830 window
= XCAR (windows
);
1831 w
= XWINDOW (window
);
1833 /* Note that we do not pay attention here to whether the frame
1834 is visible, since Fwindow_list skips non-visible frames if
1835 that is desired, under the control of frame_arg. */
1836 if (!MINI_WINDOW_P (w
)
1837 /* For UNSHOW_BUFFER, we must always consider all windows. */
1838 || type
== UNSHOW_BUFFER
1839 || (mini
&& minibuf_level
> 0))
1842 case GET_BUFFER_WINDOW
:
1843 if (EQ (w
->buffer
, obj
)
1844 /* Don't find any minibuffer window
1845 except the one that is currently in use. */
1846 && (MINI_WINDOW_P (w
)
1847 ? EQ (window
, minibuf_window
)
1850 if (NILP (best_window
))
1851 best_window
= window
;
1852 else if (EQ (window
, selected_window
))
1853 /* For compatibility with 20.x, prefer to return
1855 best_window
= window
;
1859 case GET_LRU_WINDOW
:
1860 /* t as arg means consider only full-width windows */
1861 if (!NILP (obj
) && !WINDOW_FULL_WIDTH_P (w
))
1863 /* Ignore dedicated windows and minibuffers. */
1864 if (MINI_WINDOW_P (w
) || !NILP (w
->dedicated
))
1866 if (NILP (best_window
)
1867 || (XFASTINT (XWINDOW (best_window
)->use_time
)
1868 > XFASTINT (w
->use_time
)))
1869 best_window
= window
;
1872 case DELETE_OTHER_WINDOWS
:
1873 if (!EQ (window
, obj
))
1874 Fdelete_window (window
);
1877 case DELETE_BUFFER_WINDOWS
:
1878 if (EQ (w
->buffer
, obj
))
1880 struct frame
*f
= XFRAME (WINDOW_FRAME (w
));
1882 /* If this window is dedicated, and in a frame of its own,
1884 if (EQ (window
, FRAME_ROOT_WINDOW (f
))
1885 && !NILP (w
->dedicated
)
1886 && other_visible_frames (f
))
1888 /* Skip the other windows on this frame.
1889 There might be one, the minibuffer! */
1890 while (CONSP (XCDR (windows
))
1891 && EQ (XWINDOW (XCAR (windows
))->frame
,
1892 XWINDOW (XCAR (XCDR (windows
)))->frame
))
1893 windows
= XCDR (windows
);
1895 /* Now we can safely delete the frame. */
1896 Fdelete_frame (w
->frame
, Qnil
);
1898 else if (NILP (w
->parent
))
1900 /* If we're deleting the buffer displayed in the
1901 only window on the frame, find a new buffer to
1904 buffer
= Fother_buffer (obj
, Qnil
, w
->frame
);
1905 Fset_window_buffer (window
, buffer
);
1906 if (EQ (window
, selected_window
))
1907 Fset_buffer (w
->buffer
);
1910 Fdelete_window (window
);
1914 case GET_LARGEST_WINDOW
:
1916 /* Ignore dedicated windows and minibuffers. */
1917 if (MINI_WINDOW_P (w
) || !NILP (w
->dedicated
))
1920 if (NILP (best_window
))
1921 best_window
= window
;
1924 struct window
*b
= XWINDOW (best_window
);
1925 if (XFASTINT (w
->height
) * XFASTINT (w
->width
)
1926 > XFASTINT (b
->height
) * XFASTINT (b
->width
))
1927 best_window
= window
;
1933 if (EQ (w
->buffer
, obj
))
1936 struct frame
*f
= XFRAME (w
->frame
);
1938 /* Find another buffer to show in this window. */
1939 buffer
= Fother_buffer (obj
, Qnil
, w
->frame
);
1941 /* If this window is dedicated, and in a frame of its own,
1943 if (EQ (window
, FRAME_ROOT_WINDOW (f
))
1944 && !NILP (w
->dedicated
)
1945 && other_visible_frames (f
))
1947 /* Skip the other windows on this frame.
1948 There might be one, the minibuffer! */
1949 while (CONSP (XCDR (windows
))
1950 && EQ (XWINDOW (XCAR (windows
))->frame
,
1951 XWINDOW (XCAR (XCDR (windows
)))->frame
))
1952 windows
= XCDR (windows
);
1954 /* Now we can safely delete the frame. */
1955 Fdelete_frame (w
->frame
, Qnil
);
1959 /* Otherwise show a different buffer in the window. */
1960 w
->dedicated
= Qnil
;
1961 Fset_window_buffer (window
, buffer
);
1962 if (EQ (window
, selected_window
))
1963 Fset_buffer (w
->buffer
);
1968 /* Check for a window that has a killed buffer. */
1969 case CHECK_ALL_WINDOWS
:
1970 if (! NILP (w
->buffer
)
1971 && NILP (XBUFFER (w
->buffer
)->name
))
1975 case WINDOW_LOOP_UNUSED
:
1984 /* Used for debugging. Abort if any window has a dead buffer. */
1987 check_all_windows ()
1989 window_loop (CHECK_ALL_WINDOWS
, Qnil
, 1, Qt
);
1992 DEFUN ("get-lru-window", Fget_lru_window
, Sget_lru_window
, 0, 1, 0,
1993 doc
: /* Return the window least recently selected or used for display.
1994 If optional argument FRAME is `visible', search all visible frames.
1995 If FRAME is 0, search all visible and iconified frames.
1996 If FRAME is t, search all frames.
1997 If FRAME is nil, search only the selected frame.
1998 If FRAME is a frame, search only that frame. */)
2002 register Lisp_Object w
;
2003 /* First try for a window that is full-width */
2004 w
= window_loop (GET_LRU_WINDOW
, Qt
, 0, frame
);
2005 if (!NILP (w
) && !EQ (w
, selected_window
))
2007 /* If none of them, try the rest */
2008 return window_loop (GET_LRU_WINDOW
, Qnil
, 0, frame
);
2011 DEFUN ("get-largest-window", Fget_largest_window
, Sget_largest_window
, 0, 1, 0,
2012 doc
: /* Return the largest window in area.
2013 If optional argument FRAME is `visible', search all visible frames.
2014 If FRAME is 0, search all visible and iconified frames.
2015 If FRAME is t, search all frames.
2016 If FRAME is nil, search only the selected frame.
2017 If FRAME is a frame, search only that frame. */)
2021 return window_loop (GET_LARGEST_WINDOW
, Qnil
, 0,
2025 DEFUN ("get-buffer-window", Fget_buffer_window
, Sget_buffer_window
, 1, 2, 0,
2026 doc
: /* Return a window currently displaying BUFFER, or nil if none.
2027 If optional argument FRAME is `visible', search all visible frames.
2028 If optional argument FRAME is 0, search all visible and iconified frames.
2029 If FRAME is t, search all frames.
2030 If FRAME is nil, search only the selected frame.
2031 If FRAME is a frame, search only that frame. */)
2033 Lisp_Object buffer
, frame
;
2035 buffer
= Fget_buffer (buffer
);
2036 if (BUFFERP (buffer
))
2037 return window_loop (GET_BUFFER_WINDOW
, buffer
, 1, frame
);
2042 DEFUN ("delete-other-windows", Fdelete_other_windows
, Sdelete_other_windows
,
2044 doc
: /* Make WINDOW (or the selected window) fill its frame.
2045 Only the frame WINDOW is on is affected.
2046 This function tries to reduce display jumps
2047 by keeping the text previously visible in WINDOW
2048 in the same place on the frame. Doing this depends on
2049 the value of (window-start WINDOW), so if calling this function
2050 in a program gives strange scrolling, make sure the window-start
2051 value is reasonable when this function is called. */)
2060 window
= selected_window
;
2062 CHECK_LIVE_WINDOW (window
);
2063 w
= XWINDOW (window
);
2065 startpos
= marker_position (w
->start
);
2066 top
= XFASTINT (w
->top
) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w
)));
2068 if (MINI_WINDOW_P (w
) && top
> 0)
2069 error ("Can't expand minibuffer to full frame");
2071 window_loop (DELETE_OTHER_WINDOWS
, window
, 0, WINDOW_FRAME (w
));
2073 /* Try to minimize scrolling, by setting the window start to the point
2074 will cause the text at the old window start to be at the same place
2075 on the frame. But don't try to do this if the window start is
2076 outside the visible portion (as might happen when the display is
2077 not current, due to typeahead). */
2078 new_top
= XFASTINT (w
->top
) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w
)));
2080 && startpos
>= BUF_BEGV (XBUFFER (w
->buffer
))
2081 && startpos
<= BUF_ZV (XBUFFER (w
->buffer
)))
2083 struct position pos
;
2084 struct buffer
*obuf
= current_buffer
;
2086 Fset_buffer (w
->buffer
);
2087 /* This computation used to temporarily move point, but that can
2088 have unwanted side effects due to text properties. */
2089 pos
= *vmotion (startpos
, -top
, w
);
2091 set_marker_both (w
->start
, w
->buffer
, pos
.bufpos
, pos
.bytepos
);
2092 w
->window_end_valid
= Qnil
;
2093 w
->start_at_line_beg
= ((pos
.bytepos
== BEGV_BYTE
2094 || FETCH_BYTE (pos
.bytepos
- 1) == '\n') ? Qt
2096 /* We need to do this, so that the window-scroll-functions
2098 w
->optional_new_start
= Qt
;
2100 set_buffer_internal (obuf
);
2106 DEFUN ("delete-windows-on", Fdelete_windows_on
, Sdelete_windows_on
,
2107 1, 2, "bDelete windows on (buffer): ",
2108 doc
: /* Delete all windows showing BUFFER.
2109 Optional second argument FRAME controls which frames are affected.
2110 If optional argument FRAME is `visible', search all visible frames.
2111 If FRAME is 0, search all visible and iconified frames.
2112 If FRAME is nil, search all frames.
2113 If FRAME is t, search only the selected frame.
2114 If FRAME is a frame, search only that frame. */)
2116 Lisp_Object buffer
, frame
;
2118 /* FRAME uses t and nil to mean the opposite of what window_loop
2122 else if (EQ (frame
, Qt
))
2127 buffer
= Fget_buffer (buffer
);
2128 CHECK_BUFFER (buffer
);
2129 window_loop (DELETE_BUFFER_WINDOWS
, buffer
, 0, frame
);
2135 DEFUN ("replace-buffer-in-windows", Freplace_buffer_in_windows
,
2136 Sreplace_buffer_in_windows
,
2137 1, 1, "bReplace buffer in windows: ",
2138 doc
: /* Replace BUFFER with some other buffer in all windows showing it. */)
2144 buffer
= Fget_buffer (buffer
);
2145 CHECK_BUFFER (buffer
);
2146 window_loop (UNSHOW_BUFFER
, buffer
, 0, Qt
);
2151 /* Replace BUFFER with some other buffer in all windows
2152 of all frames, even those on other keyboards. */
2155 replace_buffer_in_all_windows (buffer
)
2159 Lisp_Object tail
, frame
;
2161 /* A single call to window_loop won't do the job
2162 because it only considers frames on the current keyboard.
2163 So loop manually over frames, and handle each one. */
2164 FOR_EACH_FRAME (tail
, frame
)
2165 window_loop (UNSHOW_BUFFER
, buffer
, 1, frame
);
2167 window_loop (UNSHOW_BUFFER
, buffer
, 1, Qt
);
2171 /* Set the height of WINDOW and all its inferiors. */
2173 /* The smallest acceptable dimensions for a window. Anything smaller
2174 might crash Emacs. */
2176 #define MIN_SAFE_WINDOW_WIDTH (2)
2177 #define MIN_SAFE_WINDOW_HEIGHT (1)
2179 /* Make sure that window_min_height and window_min_width are
2180 not too small; if they are, set them to safe minima. */
2183 check_min_window_sizes ()
2185 /* Smaller values might permit a crash. */
2186 if (window_min_width
< MIN_SAFE_WINDOW_WIDTH
)
2187 window_min_width
= MIN_SAFE_WINDOW_WIDTH
;
2188 if (window_min_height
< MIN_SAFE_WINDOW_HEIGHT
)
2189 window_min_height
= MIN_SAFE_WINDOW_HEIGHT
;
2192 /* If *ROWS or *COLS are too small a size for FRAME, set them to the
2193 minimum allowable size. */
2196 check_frame_size (frame
, rows
, cols
)
2200 /* For height, we have to see:
2201 how many windows the frame has at minimum (one or two),
2202 and whether it has a menu bar or other special stuff at the top. */
2204 = ((FRAME_MINIBUF_ONLY_P (frame
) || ! FRAME_HAS_MINIBUF_P (frame
))
2205 ? MIN_SAFE_WINDOW_HEIGHT
2206 : 2 * MIN_SAFE_WINDOW_HEIGHT
);
2208 if (FRAME_TOP_MARGIN (frame
) > 0)
2209 min_height
+= FRAME_TOP_MARGIN (frame
);
2211 if (*rows
< min_height
)
2213 if (*cols
< MIN_SAFE_WINDOW_WIDTH
)
2214 *cols
= MIN_SAFE_WINDOW_WIDTH
;
2218 /* Value is non-zero if window W is fixed-size. WIDTH_P non-zero means
2219 check if W's width can be changed, otherwise check W's height.
2220 CHECK_SIBLINGS_P non-zero means check resizablity of WINDOW's
2221 siblings, too. If none of the siblings is resizable, WINDOW isn't
2225 window_fixed_size_p (w
, width_p
, check_siblings_p
)
2227 int width_p
, check_siblings_p
;
2232 if (!NILP (w
->hchild
))
2234 c
= XWINDOW (w
->hchild
);
2238 /* A horiz. combination is fixed-width if all of if its
2240 while (c
&& window_fixed_size_p (c
, width_p
, 0))
2241 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2242 fixed_p
= c
== NULL
;
2246 /* A horiz. combination is fixed-height if one of if its
2248 while (c
&& !window_fixed_size_p (c
, width_p
, 0))
2249 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2250 fixed_p
= c
!= NULL
;
2253 else if (!NILP (w
->vchild
))
2255 c
= XWINDOW (w
->vchild
);
2259 /* A vert. combination is fixed-width if one of if its
2261 while (c
&& !window_fixed_size_p (c
, width_p
, 0))
2262 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2263 fixed_p
= c
!= NULL
;
2267 /* A vert. combination is fixed-height if all of if its
2269 while (c
&& window_fixed_size_p (c
, width_p
, 0))
2270 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2271 fixed_p
= c
== NULL
;
2274 else if (BUFFERP (w
->buffer
))
2276 if (w
->height_fixed_p
&& !width_p
)
2280 struct buffer
*old
= current_buffer
;
2283 current_buffer
= XBUFFER (w
->buffer
);
2284 val
= find_symbol_value (Qwindow_size_fixed
);
2285 current_buffer
= old
;
2288 if (!EQ (val
, Qunbound
))
2290 fixed_p
= !NILP (val
);
2293 && ((EQ (val
, Qheight
) && width_p
)
2294 || (EQ (val
, Qwidth
) && !width_p
)))
2299 /* Can't tell if this one is resizable without looking at
2300 siblings. If all siblings are fixed-size this one is too. */
2301 if (!fixed_p
&& check_siblings_p
&& WINDOWP (w
->parent
))
2305 for (child
= w
->prev
; !NILP (child
); child
= XWINDOW (child
)->prev
)
2306 if (!window_fixed_size_p (XWINDOW (child
), width_p
, 0))
2310 for (child
= w
->next
; !NILP (child
); child
= XWINDOW (child
)->next
)
2311 if (!window_fixed_size_p (XWINDOW (child
), width_p
, 0))
2325 /* Return the minimum size of window W, not taking fixed-width windows
2326 into account. WIDTH_P non-zero means return the minimum width,
2327 otherwise return the minimum height. If W is a combination window,
2328 compute the minimum size from the minimum sizes of W's children. */
2331 window_min_size_1 (w
, width_p
)
2338 if (!NILP (w
->hchild
))
2340 c
= XWINDOW (w
->hchild
);
2345 /* The min width of a horizontal combination is
2346 the sum of the min widths of its children. */
2349 size
+= window_min_size_1 (c
, width_p
);
2350 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2355 /* The min height a horizontal combination equals
2356 the maximum of all min height of its children. */
2359 int min_size
= window_min_size_1 (c
, width_p
);
2360 size
= max (min_size
, size
);
2361 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2365 else if (!NILP (w
->vchild
))
2367 c
= XWINDOW (w
->vchild
);
2372 /* The min width of a vertical combination is
2373 the maximum of the min widths of its children. */
2376 int min_size
= window_min_size_1 (c
, width_p
);
2377 size
= max (min_size
, size
);
2378 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2383 /* The min height of a vertical combination equals
2384 the sum of the min height of its children. */
2387 size
+= window_min_size_1 (c
, width_p
);
2388 c
= WINDOWP (c
->next
) ? XWINDOW (c
->next
) : NULL
;
2395 size
= window_min_width
;
2398 if (MINI_WINDOW_P (w
)
2399 || (!WINDOW_WANTS_MODELINE_P (w
)
2400 && !WINDOW_WANTS_HEADER_LINE_P (w
)))
2403 size
= window_min_height
;
2411 /* Return the minimum size of window W, taking fixed-size windows into
2412 account. WIDTH_P non-zero means return the minimum width,
2413 otherwise return the minimum height. IGNORE_FIXED_P non-zero means
2414 ignore if W is fixed-size. Set *FIXED to 1 if W is fixed-size
2415 unless FIXED is null. */
2418 window_min_size (w
, width_p
, ignore_fixed_p
, fixed
)
2420 int width_p
, ignore_fixed_p
, *fixed
;
2427 fixed_p
= window_fixed_size_p (w
, width_p
, 1);
2433 size
= width_p
? XFASTINT (w
->width
) : XFASTINT (w
->height
);
2435 size
= window_min_size_1 (w
, width_p
);
2441 /* Set WINDOW's height or width to SIZE. WIDTH_P non-zero means set
2442 WINDOW's width. Resize WINDOW's children, if any, so that they
2443 keep their proportionate size relative to WINDOW. Propagate
2444 WINDOW's top or left edge position to children. Delete windows
2445 that become too small unless NODELETE_P is non-zero.
2447 If NODELETE_P is 2, that means we do delete windows that are
2448 too small, even if they were too small before! */
2451 size_window (window
, size
, width_p
, nodelete_p
)
2453 int size
, width_p
, nodelete_p
;
2455 struct window
*w
= XWINDOW (window
);
2457 Lisp_Object child
, *forward
, *sideward
;
2458 int old_size
, min_size
;
2460 if (nodelete_p
== 2)
2463 check_min_window_sizes ();
2464 size
= max (0, size
);
2466 /* If the window has been "too small" at one point,
2467 don't delete it for being "too small" in the future.
2468 Preserve it as long as that is at all possible. */
2471 old_size
= XINT (w
->width
);
2472 min_size
= window_min_width
;
2476 old_size
= XINT (w
->height
);
2477 min_size
= window_min_height
;
2480 if (old_size
< min_size
&& nodelete_p
!= 2)
2481 w
->too_small_ok
= Qt
;
2483 /* Maybe delete WINDOW if it's too small. */
2484 if (nodelete_p
!= 1 && !NILP (w
->parent
))
2486 if (!MINI_WINDOW_P (w
) && !NILP (w
->too_small_ok
))
2487 min_size
= width_p
? MIN_SAFE_WINDOW_WIDTH
: MIN_SAFE_WINDOW_HEIGHT
;
2489 min_size
= width_p
? window_min_width
: window_min_height
;
2491 if (size
< min_size
)
2493 delete_window (window
);
2498 /* Set redisplay hints. */
2499 w
->last_modified
= make_number (0);
2500 w
->last_overlay_modified
= make_number (0);
2501 windows_or_buffers_changed
++;
2502 FRAME_WINDOW_SIZES_CHANGED (XFRAME (w
->frame
)) = 1;
2506 sideward
= &w
->vchild
;
2507 forward
= &w
->hchild
;
2508 w
->width
= make_number (size
);
2512 sideward
= &w
->hchild
;
2513 forward
= &w
->vchild
;
2514 w
->height
= make_number (size
);
2515 w
->orig_height
= Qnil
;
2518 if (!NILP (*sideward
))
2520 for (child
= *sideward
; !NILP (child
); child
= c
->next
)
2522 c
= XWINDOW (child
);
2527 size_window (child
, size
, width_p
, nodelete_p
);
2530 else if (!NILP (*forward
))
2532 int fixed_size
, each
, extra
, n
;
2533 int resize_fixed_p
, nfixed
;
2534 int last_pos
, first_pos
, nchildren
, total
;
2536 /* Determine the fixed-size portion of the this window, and the
2537 number of child windows. */
2538 fixed_size
= nchildren
= nfixed
= total
= 0;
2539 for (child
= *forward
; !NILP (child
); child
= c
->next
, ++nchildren
)
2543 c
= XWINDOW (child
);
2544 child_size
= width_p
? XINT (c
->width
) : XINT (c
->height
);
2545 total
+= child_size
;
2547 if (window_fixed_size_p (c
, width_p
, 0))
2549 fixed_size
+= child_size
;
2554 /* If the new size is smaller than fixed_size, or if there
2555 aren't any resizable windows, allow resizing fixed-size
2557 resize_fixed_p
= nfixed
== nchildren
|| size
< fixed_size
;
2559 /* Compute how many lines/columns to add to each child. The
2560 value of extra takes care of rounding errors. */
2561 n
= resize_fixed_p
? nchildren
: nchildren
- nfixed
;
2562 each
= (size
- total
) / n
;
2563 extra
= (size
- total
) - n
* each
;
2565 /* Compute new children heights and edge positions. */
2566 first_pos
= width_p
? XINT (w
->left
) : XINT (w
->top
);
2567 last_pos
= first_pos
;
2568 for (child
= *forward
; !NILP (child
); child
= c
->next
)
2570 int new_size
, old_size
;
2572 c
= XWINDOW (child
);
2573 old_size
= width_p
? XFASTINT (c
->width
) : XFASTINT (c
->height
);
2574 new_size
= old_size
;
2576 /* The top or left edge position of this child equals the
2577 bottom or right edge of its predecessor. */
2579 c
->left
= make_number (last_pos
);
2581 c
->top
= make_number (last_pos
);
2583 /* If this child can be resized, do it. */
2584 if (resize_fixed_p
|| !window_fixed_size_p (c
, width_p
, 0))
2586 new_size
= old_size
+ each
+ extra
;
2590 /* Set new height. Note that size_window also propagates
2591 edge positions to children, so it's not a no-op if we
2592 didn't change the child's size. */
2593 size_window (child
, new_size
, width_p
, 1);
2595 /* Remember the bottom/right edge position of this child; it
2596 will be used to set the top/left edge of the next child. */
2597 last_pos
+= new_size
;
2600 /* We should have covered the parent exactly with child windows. */
2601 xassert (size
== last_pos
- first_pos
);
2603 /* Now delete any children that became too small. */
2605 for (child
= *forward
; !NILP (child
); child
= c
->next
)
2608 c
= XWINDOW (child
);
2609 child_size
= width_p
? XINT (c
->width
) : XINT (c
->height
);
2610 size_window (child
, child_size
, width_p
, 2);
2615 /* Set WINDOW's height to HEIGHT, and recursively change the height of
2616 WINDOW's children. NODELETE non-zero means don't delete windows
2617 that become too small in the process. (The caller should check
2618 later and do so if appropriate.) */
2621 set_window_height (window
, height
, nodelete
)
2626 size_window (window
, height
, 0, nodelete
);
2630 /* Set WINDOW's width to WIDTH, and recursively change the width of
2631 WINDOW's children. NODELETE non-zero means don't delete windows
2632 that become too small in the process. (The caller should check
2633 later and do so if appropriate.) */
2636 set_window_width (window
, width
, nodelete
)
2641 size_window (window
, width
, 1, nodelete
);
2645 int window_select_count
;
2648 Fset_window_buffer_unwind (obuf
)
2656 /* Make WINDOW display BUFFER as its contents. RUN_HOOKS_P non-zero
2657 means it's allowed to run hooks. See make_frame for a case where
2658 it's not allowed. */
2661 set_window_buffer (window
, buffer
, run_hooks_p
)
2662 Lisp_Object window
, buffer
;
2665 struct window
*w
= XWINDOW (window
);
2666 struct buffer
*b
= XBUFFER (buffer
);
2667 int count
= SPECPDL_INDEX ();
2671 if (EQ (window
, selected_window
))
2672 b
->last_selected_window
= window
;
2674 /* Update time stamps of buffer display. */
2675 if (INTEGERP (b
->display_count
))
2676 XSETINT (b
->display_count
, XINT (b
->display_count
) + 1);
2677 b
->display_time
= Fcurrent_time ();
2679 XSETFASTINT (w
->window_end_pos
, 0);
2680 XSETFASTINT (w
->window_end_vpos
, 0);
2681 bzero (&w
->last_cursor
, sizeof w
->last_cursor
);
2682 w
->window_end_valid
= Qnil
;
2683 w
->hscroll
= w
->min_hscroll
= make_number (0);
2685 set_marker_both (w
->pointm
, buffer
, BUF_PT (b
), BUF_PT_BYTE (b
));
2686 set_marker_restricted (w
->start
,
2687 make_number (b
->last_window_start
),
2689 w
->start_at_line_beg
= Qnil
;
2690 w
->force_start
= Qnil
;
2691 XSETFASTINT (w
->last_modified
, 0);
2692 XSETFASTINT (w
->last_overlay_modified
, 0);
2693 windows_or_buffers_changed
++;
2695 /* We must select BUFFER for running the window-scroll-functions.
2696 If WINDOW is selected, switch permanently.
2697 Otherwise, switch but go back to the ambient buffer afterward. */
2698 if (EQ (window
, selected_window
))
2699 Fset_buffer (buffer
);
2700 /* We can't check ! NILP (Vwindow_scroll_functions) here
2701 because that might itself be a local variable. */
2702 else if (window_initialized
)
2704 record_unwind_protect (Fset_window_buffer_unwind
, Fcurrent_buffer ());
2705 Fset_buffer (buffer
);
2708 /* Set left and right marginal area width from buffer. */
2709 Fset_window_margins (window
, b
->left_margin_width
, b
->right_margin_width
);
2713 if (! NILP (Vwindow_scroll_functions
))
2714 run_hook_with_args_2 (Qwindow_scroll_functions
, window
,
2715 Fmarker_position (w
->start
));
2717 if (! NILP (Vwindow_configuration_change_hook
)
2718 && ! NILP (Vrun_hooks
))
2719 call1 (Vrun_hooks
, Qwindow_configuration_change_hook
);
2722 unbind_to (count
, Qnil
);
2726 DEFUN ("set-window-buffer", Fset_window_buffer
, Sset_window_buffer
, 2, 2, 0,
2727 doc
: /* Make WINDOW display BUFFER as its contents.
2728 BUFFER can be a buffer or buffer name. */)
2730 register Lisp_Object window
, buffer
;
2732 register Lisp_Object tem
;
2733 register struct window
*w
= decode_window (window
);
2735 XSETWINDOW (window
, w
);
2736 buffer
= Fget_buffer (buffer
);
2737 CHECK_BUFFER (buffer
);
2739 if (NILP (XBUFFER (buffer
)->name
))
2740 error ("Attempt to display deleted buffer");
2744 error ("Window is deleted");
2745 else if (! EQ (tem
, Qt
)) /* w->buffer is t when the window
2746 is first being set up. */
2748 if (!NILP (w
->dedicated
) && !EQ (tem
, buffer
))
2749 error ("Window is dedicated to `%s'",
2750 SDATA (XBUFFER (tem
)->name
));
2755 set_window_buffer (window
, buffer
, 1);
2759 DEFUN ("select-window", Fselect_window
, Sselect_window
, 1, 1, 0,
2760 doc
: /* Select WINDOW. Most editing will apply to WINDOW's buffer.
2761 If WINDOW is not already selected, also make WINDOW's buffer current.
2762 If WINDOW's frame is the selected frame, also make WINDOW the frame's
2765 Note that the main editor command loop
2766 selects the buffer of the selected window before each command. */)
2768 register Lisp_Object window
;
2770 return select_window_1 (window
, 1);
2773 /* Note that selected_window can be nil
2774 when this is called from Fset_window_configuration. */
2777 select_window_1 (window
, recordflag
)
2778 register Lisp_Object window
;
2781 register struct window
*w
;
2782 register struct window
*ow
;
2785 CHECK_LIVE_WINDOW (window
);
2787 w
= XWINDOW (window
);
2788 w
->frozen_window_start_p
= 0;
2790 XSETFASTINT (w
->use_time
, ++window_select_count
);
2791 if (EQ (window
, selected_window
))
2794 if (!NILP (selected_window
))
2796 ow
= XWINDOW (selected_window
);
2797 if (! NILP (ow
->buffer
))
2798 set_marker_both (ow
->pointm
, ow
->buffer
,
2799 BUF_PT (XBUFFER (ow
->buffer
)),
2800 BUF_PT_BYTE (XBUFFER (ow
->buffer
)));
2803 selected_window
= window
;
2804 sf
= SELECTED_FRAME ();
2805 if (XFRAME (WINDOW_FRAME (w
)) == sf
)
2806 sf
->selected_window
= window
;
2809 record_buffer (w
->buffer
);
2810 Fset_buffer (w
->buffer
);
2812 XBUFFER (w
->buffer
)->last_selected_window
= window
;
2814 /* Go to the point recorded in the window.
2815 This is important when the buffer is in more
2816 than one window. It also matters when
2817 redisplay_window has altered point after scrolling,
2818 because it makes the change only in the window. */
2820 register int new_point
= marker_position (w
->pointm
);
2821 if (new_point
< BEGV
)
2823 else if (new_point
> ZV
)
2829 windows_or_buffers_changed
++;
2833 /* Deiconify the frame containing the window WINDOW,
2834 unless it is the selected frame;
2837 The reason for the exception for the selected frame
2838 is that it seems better not to change the selected frames visibility
2839 merely because of displaying a different buffer in it.
2840 The deiconification is useful when a buffer gets shown in
2841 another frame that you were not using lately. */
2844 display_buffer_1 (window
)
2847 Lisp_Object frame
= XWINDOW (window
)->frame
;
2848 FRAME_PTR f
= XFRAME (frame
);
2850 FRAME_SAMPLE_VISIBILITY (f
);
2852 if (!EQ (frame
, selected_frame
))
2854 if (FRAME_ICONIFIED_P (f
))
2855 Fmake_frame_visible (frame
);
2856 else if (FRAME_VISIBLE_P (f
))
2857 Fraise_frame (frame
);
2863 DEFUN ("special-display-p", Fspecial_display_p
, Sspecial_display_p
, 1, 1, 0,
2864 doc
: /* Returns non-nil if a buffer named BUFFER-NAME would be created specially.
2865 The value is actually t if the frame should be called with default frame
2866 parameters, and a list of frame parameters if they were specified.
2867 See `special-display-buffer-names', and `special-display-regexps'. */)
2869 Lisp_Object buffer_name
;
2873 CHECK_STRING (buffer_name
);
2875 tem
= Fmember (buffer_name
, Vspecial_display_buffer_names
);
2879 tem
= Fassoc (buffer_name
, Vspecial_display_buffer_names
);
2883 for (tem
= Vspecial_display_regexps
; CONSP (tem
); tem
= XCDR (tem
))
2885 Lisp_Object car
= XCAR (tem
);
2887 && fast_string_match (car
, buffer_name
) >= 0)
2889 else if (CONSP (car
)
2890 && STRINGP (XCAR (car
))
2891 && fast_string_match (XCAR (car
), buffer_name
) >= 0)
2897 DEFUN ("same-window-p", Fsame_window_p
, Ssame_window_p
, 1, 1, 0,
2898 doc
: /* Returns non-nil if a new buffer named BUFFER-NAME would use the same window.
2899 See `same-window-buffer-names' and `same-window-regexps'. */)
2901 Lisp_Object buffer_name
;
2905 CHECK_STRING (buffer_name
);
2907 tem
= Fmember (buffer_name
, Vsame_window_buffer_names
);
2911 tem
= Fassoc (buffer_name
, Vsame_window_buffer_names
);
2915 for (tem
= Vsame_window_regexps
; CONSP (tem
); tem
= XCDR (tem
))
2917 Lisp_Object car
= XCAR (tem
);
2919 && fast_string_match (car
, buffer_name
) >= 0)
2921 else if (CONSP (car
)
2922 && STRINGP (XCAR (car
))
2923 && fast_string_match (XCAR (car
), buffer_name
) >= 0)
2929 /* Use B so the default is (other-buffer). */
2930 DEFUN ("display-buffer", Fdisplay_buffer
, Sdisplay_buffer
, 1, 3,
2931 "BDisplay buffer: \nP",
2932 doc
: /* Make BUFFER appear in some window but don't select it.
2933 BUFFER can be a buffer or a buffer name.
2934 If BUFFER is shown already in some window, just use that one,
2935 unless the window is the selected window and the optional second
2936 argument NOT-THIS-WINDOW is non-nil (interactively, with prefix arg).
2937 If `pop-up-frames' is non-nil, make a new frame if no window shows BUFFER.
2938 Returns the window displaying BUFFER.
2939 If `display-buffer-reuse-frames' is non-nil, and another frame is currently
2940 displaying BUFFER, then simply raise that frame.
2942 The variables `special-display-buffer-names', `special-display-regexps',
2943 `same-window-buffer-names', and `same-window-regexps' customize how certain
2944 buffer names are handled.
2946 If optional argument FRAME is `visible', search all visible frames.
2947 If FRAME is 0, search all visible and iconified frames.
2948 If FRAME is t, search all frames.
2949 If FRAME is a frame, search only that frame.
2950 If FRAME is nil, search only the selected frame
2951 (actually the last nonminibuffer frame),
2952 unless `pop-up-frames' or `display-buffer-reuse-frames' is non-nil,
2953 which means search visible and iconified frames.
2955 If `even-window-heights' is non-nil, window heights will be evened out
2956 if displaying the buffer causes two vertically adjacent windows to be
2958 (buffer
, not_this_window
, frame
)
2959 register Lisp_Object buffer
, not_this_window
, frame
;
2961 register Lisp_Object window
, tem
, swp
;
2965 buffer
= Fget_buffer (buffer
);
2966 CHECK_BUFFER (buffer
);
2968 if (!NILP (Vdisplay_buffer_function
))
2969 return call2 (Vdisplay_buffer_function
, buffer
, not_this_window
);
2971 if (NILP (not_this_window
)
2972 && XBUFFER (XWINDOW (selected_window
)->buffer
) == XBUFFER (buffer
))
2973 return display_buffer_1 (selected_window
);
2975 /* See if the user has specified this buffer should appear
2976 in the selected window. */
2977 if (NILP (not_this_window
))
2979 swp
= Fsame_window_p (XBUFFER (buffer
)->name
);
2980 if (!NILP (swp
) && !no_switch_window (selected_window
))
2982 Fswitch_to_buffer (buffer
, Qnil
);
2983 return display_buffer_1 (selected_window
);
2987 /* If the user wants pop-up-frames or display-buffer-reuse-frames,
2988 look for a window showing BUFFER on any visible or iconified frame.
2989 Otherwise search only the current frame. */
2992 else if (pop_up_frames
2993 || display_buffer_reuse_frames
2994 || last_nonminibuf_frame
== 0)
2995 XSETFASTINT (tem
, 0);
2997 XSETFRAME (tem
, last_nonminibuf_frame
);
2999 window
= Fget_buffer_window (buffer
, tem
);
3001 && (NILP (not_this_window
) || !EQ (window
, selected_window
)))
3002 return display_buffer_1 (window
);
3004 /* Certain buffer names get special handling. */
3005 if (!NILP (Vspecial_display_function
) && NILP (swp
))
3007 tem
= Fspecial_display_p (XBUFFER (buffer
)->name
);
3009 return call1 (Vspecial_display_function
, buffer
);
3011 return call2 (Vspecial_display_function
, buffer
, tem
);
3014 /* If there are no frames open that have more than a minibuffer,
3015 we need to create a new frame. */
3016 if (pop_up_frames
|| last_nonminibuf_frame
== 0)
3018 window
= Fframe_selected_window (call0 (Vpop_up_frame_function
));
3019 Fset_window_buffer (window
, buffer
);
3020 return display_buffer_1 (window
);
3023 f
= SELECTED_FRAME ();
3025 || FRAME_MINIBUF_ONLY_P (f
)
3026 /* If the current frame is a special display frame,
3027 don't try to reuse its windows. */
3028 || !NILP (XWINDOW (FRAME_ROOT_WINDOW (f
))->dedicated
))
3033 if (FRAME_MINIBUF_ONLY_P (f
))
3034 XSETFRAME (frames
, last_nonminibuf_frame
);
3035 /* Don't try to create a window if would get an error */
3036 if (split_height_threshold
< window_min_height
<< 1)
3037 split_height_threshold
= window_min_height
<< 1;
3039 /* Note that both Fget_largest_window and Fget_lru_window
3040 ignore minibuffers and dedicated windows.
3041 This means they can return nil. */
3043 /* If the frame we would try to split cannot be split,
3044 try other frames. */
3045 if (FRAME_NO_SPLIT_P (NILP (frames
) ? f
: last_nonminibuf_frame
))
3047 /* Try visible frames first. */
3048 window
= Fget_largest_window (Qvisible
);
3049 /* If that didn't work, try iconified frames. */
3051 window
= Fget_largest_window (make_number (0));
3053 window
= Fget_largest_window (Qt
);
3056 window
= Fget_largest_window (frames
);
3058 /* If we got a tall enough full-width window that can be split,
3061 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window
)->frame
))
3062 && window_height (window
) >= split_height_threshold
3063 && WINDOW_FULL_WIDTH_P (XWINDOW (window
)))
3064 window
= Fsplit_window (window
, Qnil
, Qnil
);
3067 Lisp_Object upper
, lower
, other
;
3069 window
= Fget_lru_window (frames
);
3070 /* If the LRU window is selected, and big enough,
3071 and can be split, split it. */
3073 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window
)->frame
))
3074 && (EQ (window
, selected_window
)
3075 || EQ (XWINDOW (window
)->parent
, Qnil
))
3076 && window_height (window
) >= window_min_height
<< 1)
3077 window
= Fsplit_window (window
, Qnil
, Qnil
);
3078 /* If Fget_lru_window returned nil, try other approaches. */
3080 /* Try visible frames first. */
3082 window
= Fget_buffer_window (buffer
, Qvisible
);
3084 window
= Fget_largest_window (Qvisible
);
3085 /* If that didn't work, try iconified frames. */
3087 window
= Fget_buffer_window (buffer
, make_number (0));
3089 window
= Fget_largest_window (make_number (0));
3090 /* Try invisible frames. */
3092 window
= Fget_buffer_window (buffer
, Qt
);
3094 window
= Fget_largest_window (Qt
);
3095 /* As a last resort, make a new frame. */
3097 window
= Fframe_selected_window (call0 (Vpop_up_frame_function
));
3098 /* If window appears above or below another,
3099 even out their heights. */
3100 other
= upper
= lower
= Qnil
;
3101 if (!NILP (XWINDOW (window
)->prev
))
3102 other
= upper
= XWINDOW (window
)->prev
, lower
= window
;
3103 if (!NILP (XWINDOW (window
)->next
))
3104 other
= lower
= XWINDOW (window
)->next
, upper
= window
;
3106 && !NILP (Veven_window_heights
)
3107 /* Check that OTHER and WINDOW are vertically arrayed. */
3108 && !EQ (XWINDOW (other
)->top
, XWINDOW (window
)->top
)
3109 && (XFASTINT (XWINDOW (other
)->height
)
3110 > XFASTINT (XWINDOW (window
)->height
)))
3112 int total
= (XFASTINT (XWINDOW (other
)->height
)
3113 + XFASTINT (XWINDOW (window
)->height
));
3114 enlarge_window (upper
,
3115 total
/ 2 - XFASTINT (XWINDOW (upper
)->height
),
3121 window
= Fget_lru_window (Qnil
);
3123 Fset_window_buffer (window
, buffer
);
3124 return display_buffer_1 (window
);
3128 temp_output_buffer_show (buf
)
3129 register Lisp_Object buf
;
3131 register struct buffer
*old
= current_buffer
;
3132 register Lisp_Object window
;
3133 register struct window
*w
;
3135 XBUFFER (buf
)->directory
= current_buffer
->directory
;
3138 BUF_SAVE_MODIFF (XBUFFER (buf
)) = MODIFF
;
3142 #if 0 /* rms: there should be no reason for this. */
3143 XBUFFER (buf
)->prevent_redisplay_optimizations_p
= 1;
3145 set_buffer_internal (old
);
3147 if (!EQ (Vtemp_buffer_show_function
, Qnil
))
3148 call1 (Vtemp_buffer_show_function
, buf
);
3151 window
= Fdisplay_buffer (buf
, Qnil
, Qnil
);
3153 if (!EQ (XWINDOW (window
)->frame
, selected_frame
))
3154 Fmake_frame_visible (WINDOW_FRAME (XWINDOW (window
)));
3155 Vminibuf_scroll_window
= window
;
3156 w
= XWINDOW (window
);
3157 XSETFASTINT (w
->hscroll
, 0);
3158 XSETFASTINT (w
->min_hscroll
, 0);
3159 set_marker_restricted_both (w
->start
, buf
, 1, 1);
3160 set_marker_restricted_both (w
->pointm
, buf
, 1, 1);
3162 /* Run temp-buffer-show-hook, with the chosen window selected
3163 and it sbuffer current. */
3164 if (!NILP (Vrun_hooks
))
3167 tem
= Fboundp (Qtemp_buffer_show_hook
);
3170 tem
= Fsymbol_value (Qtemp_buffer_show_hook
);
3173 int count
= SPECPDL_INDEX ();
3174 Lisp_Object prev_window
;
3175 prev_window
= selected_window
;
3177 /* Select the window that was chosen, for running the hook. */
3178 record_unwind_protect (Fselect_window
, prev_window
);
3179 select_window_1 (window
, 0);
3180 Fset_buffer (w
->buffer
);
3181 call1 (Vrun_hooks
, Qtemp_buffer_show_hook
);
3182 select_window_1 (prev_window
, 0);
3183 unbind_to (count
, Qnil
);
3191 make_dummy_parent (window
)
3195 register struct window
*o
, *p
;
3198 o
= XWINDOW (window
);
3199 p
= allocate_window ();
3200 for (i
= 0; i
< VECSIZE (struct window
); ++i
)
3201 ((struct Lisp_Vector
*) p
)->contents
[i
]
3202 = ((struct Lisp_Vector
*)o
)->contents
[i
];
3203 XSETWINDOW (new, p
);
3205 XSETFASTINT (p
->sequence_number
, ++sequence_number
);
3207 /* Put new into window structure in place of window */
3208 replace_window (window
, new);
3221 DEFUN ("split-window", Fsplit_window
, Ssplit_window
, 0, 3, "",
3222 doc
: /* Split WINDOW, putting SIZE lines in the first of the pair.
3223 WINDOW defaults to selected one and SIZE to half its size.
3224 If optional third arg HORFLAG is non-nil, split side by side
3225 and put SIZE columns in the first of the pair. In that case,
3226 SIZE includes that window's scroll bar, or the divider column to its right. */)
3227 (window
, size
, horflag
)
3228 Lisp_Object window
, size
, horflag
;
3230 register Lisp_Object
new;
3231 register struct window
*o
, *p
;
3233 register int size_int
;
3236 window
= selected_window
;
3238 CHECK_LIVE_WINDOW (window
);
3240 o
= XWINDOW (window
);
3241 fo
= XFRAME (WINDOW_FRAME (o
));
3245 if (!NILP (horflag
))
3246 /* Calculate the size of the left-hand window, by dividing
3247 the usable space in columns by two.
3248 We round up, since the left-hand window may include
3249 a dividing line, while the right-hand may not. */
3250 size_int
= (XFASTINT (o
->width
) + 1) >> 1;
3252 size_int
= XFASTINT (o
->height
) >> 1;
3256 CHECK_NUMBER (size
);
3257 size_int
= XINT (size
);
3260 if (MINI_WINDOW_P (o
))
3261 error ("Attempt to split minibuffer window");
3262 else if (window_fixed_size_p (o
, !NILP (horflag
), 0))
3263 error ("Attempt to split fixed-size window");
3265 check_min_window_sizes ();
3269 if (size_int
< window_min_height
)
3270 error ("Window height %d too small (after splitting)", size_int
);
3271 if (size_int
+ window_min_height
> XFASTINT (o
->height
))
3272 error ("Window height %d too small (after splitting)",
3273 XFASTINT (o
->height
) - size_int
);
3274 if (NILP (o
->parent
)
3275 || NILP (XWINDOW (o
->parent
)->vchild
))
3277 make_dummy_parent (window
);
3279 XWINDOW (new)->vchild
= window
;
3284 if (size_int
< window_min_width
)
3285 error ("Window width %d too small (after splitting)", size_int
);
3287 if (size_int
+ window_min_width
> XFASTINT (o
->width
))
3288 error ("Window width %d too small (after splitting)",
3289 XFASTINT (o
->width
) - size_int
);
3290 if (NILP (o
->parent
)
3291 || NILP (XWINDOW (o
->parent
)->hchild
))
3293 make_dummy_parent (window
);
3295 XWINDOW (new)->hchild
= window
;
3299 /* Now we know that window's parent is a vertical combination
3300 if we are dividing vertically, or a horizontal combination
3301 if we are making side-by-side windows */
3303 windows_or_buffers_changed
++;
3304 FRAME_WINDOW_SIZES_CHANGED (fo
) = 1;
3305 new = make_window ();
3308 p
->frame
= o
->frame
;
3310 if (!NILP (p
->next
))
3311 XWINDOW (p
->next
)->prev
= new;
3314 p
->parent
= o
->parent
;
3316 p
->window_end_valid
= Qnil
;
3317 bzero (&p
->last_cursor
, sizeof p
->last_cursor
);
3319 /* Apportion the available frame space among the two new windows */
3321 if (!NILP (horflag
))
3323 p
->height
= o
->height
;
3325 XSETFASTINT (p
->width
, XFASTINT (o
->width
) - size_int
);
3326 XSETFASTINT (o
->width
, size_int
);
3327 XSETFASTINT (p
->left
, XFASTINT (o
->left
) + size_int
);
3332 p
->width
= o
->width
;
3333 XSETFASTINT (p
->height
, XFASTINT (o
->height
) - size_int
);
3334 XSETFASTINT (o
->height
, size_int
);
3335 XSETFASTINT (p
->top
, XFASTINT (o
->top
) + size_int
);
3338 /* Adjust glyph matrices. */
3340 Fset_window_buffer (new, o
->buffer
);
3344 DEFUN ("enlarge-window", Fenlarge_window
, Senlarge_window
, 1, 3, "p",
3345 doc
: /* Make current window ARG lines bigger.
3346 From program, optional second arg non-nil means grow sideways ARG columns.
3347 Interactively, if an argument is not given, make the window one line bigger.
3349 Optional third arg PRESERVE-BEFORE, if non-nil, means do not change the size
3350 of the siblings above or to the left of the selected window. Only
3351 siblings to the right or below are changed. */)
3352 (arg
, side
, preserve_before
)
3353 register Lisp_Object arg
, side
, preserve_before
;
3356 enlarge_window (selected_window
, XINT (arg
), !NILP (side
),
3357 !NILP (preserve_before
));
3359 if (! NILP (Vwindow_configuration_change_hook
))
3360 call1 (Vrun_hooks
, Qwindow_configuration_change_hook
);
3365 DEFUN ("shrink-window", Fshrink_window
, Sshrink_window
, 1, 2, "p",
3366 doc
: /* Make current window ARG lines smaller.
3367 From program, optional second arg non-nil means shrink sideways arg columns.
3368 Interactively, if an argument is not given, make the window one line smaller. */)
3370 register Lisp_Object arg
, side
;
3373 enlarge_window (selected_window
, -XINT (arg
), !NILP (side
), 0);
3375 if (! NILP (Vwindow_configuration_change_hook
))
3376 call1 (Vrun_hooks
, Qwindow_configuration_change_hook
);
3382 window_height (window
)
3385 register struct window
*p
= XWINDOW (window
);
3386 return XFASTINT (p
->height
);
3390 window_width (window
)
3393 register struct window
*p
= XWINDOW (window
);
3394 return XFASTINT (p
->width
);
3399 *(widthflag ? &(XWINDOW (w)->left) : &(XWINDOW (w)->top))
3401 #define CURSIZE(w) \
3402 *(widthflag ? &(XWINDOW (w)->width) : &(XWINDOW (w)->height))
3405 /* Enlarge WINDOW by DELTA. WIDTHFLAG non-zero means
3406 increase its width. Siblings of the selected window are resized to
3407 fulfill the size request. If they become too small in the process,
3408 they will be deleted.
3410 If PRESERVE_BEFORE is nonzero, that means don't alter
3411 the siblings to the left or above WINDOW. */
3414 enlarge_window (window
, delta
, widthflag
, preserve_before
)
3416 int delta
, widthflag
, preserve_before
;
3418 Lisp_Object parent
, next
, prev
;
3422 int (*sizefun
) P_ ((Lisp_Object
))
3423 = widthflag
? window_width
: window_height
;
3424 void (*setsizefun
) P_ ((Lisp_Object
, int, int))
3425 = (widthflag
? set_window_width
: set_window_height
);
3427 /* Check values of window_min_width and window_min_height for
3429 check_min_window_sizes ();
3431 /* Give up if this window cannot be resized. */
3432 if (window_fixed_size_p (XWINDOW (window
), widthflag
, 1))
3433 error ("Window is not resizable");
3435 /* Find the parent of the selected window. */
3438 p
= XWINDOW (window
);
3444 error ("No other window to side of this one");
3449 ? !NILP (XWINDOW (parent
)->hchild
)
3450 : !NILP (XWINDOW (parent
)->vchild
))
3456 sizep
= &CURSIZE (window
);
3459 register int maxdelta
;
3461 /* Compute the maximum size increment this window can have. */
3463 if (preserve_before
)
3467 maxdelta
= (*sizefun
) (parent
) - XINT (*sizep
);
3468 /* Subtract size of siblings before, since we can't take that. */
3469 maxdelta
-= XINT (CURBEG (window
)) - XINT (CURBEG (parent
));
3472 maxdelta
= (!NILP (p
->next
) ? ((*sizefun
) (p
->next
)
3473 - window_min_size (XWINDOW (p
->next
),
3478 maxdelta
= (!NILP (parent
) ? (*sizefun
) (parent
) - XINT (*sizep
)
3479 /* This is a main window followed by a minibuffer. */
3480 : !NILP (p
->next
) ? ((*sizefun
) (p
->next
)
3481 - window_min_size (XWINDOW (p
->next
),
3483 /* This is a minibuffer following a main window. */
3484 : !NILP (p
->prev
) ? ((*sizefun
) (p
->prev
)
3485 - window_min_size (XWINDOW (p
->prev
),
3487 /* This is a frame with only one window, a minibuffer-only
3488 or a minibufferless frame. */
3491 if (delta
> maxdelta
)
3492 /* This case traps trying to make the minibuffer
3493 the full frame, or make the only window aside from the
3494 minibuffer the full frame. */
3498 if (XINT (*sizep
) + delta
< window_min_size (XWINDOW (window
), widthflag
, 0, 0))
3500 delete_window (window
);
3507 /* Find the total we can get from other siblings without deleting them. */
3509 for (next
= p
->next
; ! NILP (next
); next
= XWINDOW (next
)->next
)
3510 maximum
+= (*sizefun
) (next
) - window_min_size (XWINDOW (next
),
3512 if (! preserve_before
)
3513 for (prev
= p
->prev
; ! NILP (prev
); prev
= XWINDOW (prev
)->prev
)
3514 maximum
+= (*sizefun
) (prev
) - window_min_size (XWINDOW (prev
),
3517 /* If we can get it all from them without deleting them, do so. */
3518 if (delta
<= maximum
)
3520 Lisp_Object first_unaffected
;
3521 Lisp_Object first_affected
;
3526 first_affected
= window
;
3527 /* Look at one sibling at a time,
3528 moving away from this window in both directions alternately,
3529 and take as much as we can get without deleting that sibling. */
3531 && (!NILP (next
) || (!preserve_before
&& !NILP (prev
))))
3535 int this_one
= ((*sizefun
) (next
)
3536 - window_min_size (XWINDOW (next
),
3537 widthflag
, 0, &fixed_p
));
3540 if (this_one
> delta
)
3543 (*setsizefun
) (next
, (*sizefun
) (next
) - this_one
, 0);
3544 (*setsizefun
) (window
, XINT (*sizep
) + this_one
, 0);
3549 next
= XWINDOW (next
)->next
;
3555 if (!preserve_before
&& ! NILP (prev
))
3557 int this_one
= ((*sizefun
) (prev
)
3558 - window_min_size (XWINDOW (prev
),
3559 widthflag
, 0, &fixed_p
));
3562 if (this_one
> delta
)
3565 first_affected
= prev
;
3567 (*setsizefun
) (prev
, (*sizefun
) (prev
) - this_one
, 0);
3568 (*setsizefun
) (window
, XINT (*sizep
) + this_one
, 0);
3573 prev
= XWINDOW (prev
)->prev
;
3577 xassert (delta
== 0);
3579 /* Now recalculate the edge positions of all the windows affected,
3580 based on the new sizes. */
3581 first_unaffected
= next
;
3582 prev
= first_affected
;
3583 for (next
= XWINDOW (prev
)->next
; ! EQ (next
, first_unaffected
);
3584 prev
= next
, next
= XWINDOW (next
)->next
)
3586 XSETINT (CURBEG (next
), XINT (CURBEG (prev
)) + (*sizefun
) (prev
));
3587 /* This does not change size of NEXT,
3588 but it propagates the new top edge to its children */
3589 (*setsizefun
) (next
, (*sizefun
) (next
), 0);
3594 register int delta1
;
3595 register int opht
= (*sizefun
) (parent
);
3597 if (opht
<= XINT (*sizep
) + delta
)
3599 /* If trying to grow this window to or beyond size of the parent,
3600 just delete all the sibling windows. */
3601 Lisp_Object start
, tem
, next
;
3603 start
= XWINDOW (parent
)->vchild
;
3605 start
= XWINDOW (parent
)->hchild
;
3607 /* Delete any siblings that come after WINDOW. */
3608 tem
= XWINDOW (window
)->next
;
3609 while (! NILP (tem
))
3611 next
= XWINDOW (tem
)->next
;
3612 delete_window (tem
);
3616 /* Delete any siblings that come after WINDOW.
3617 Note that if START is not WINDOW, then WINDOW still
3618 Fhas siblings, so WINDOW has not yet replaced its parent. */
3620 while (! EQ (tem
, window
))
3622 next
= XWINDOW (tem
)->next
;
3623 delete_window (tem
);
3629 /* Otherwise, make delta1 just right so that if we add
3630 delta1 lines to this window and to the parent, and then
3631 shrink the parent back to its original size, the new
3632 proportional size of this window will increase by delta.
3634 The function size_window will compute the new height h'
3635 of the window from delta1 as:
3638 x = delta1 - delta1/n * n for the 1st resizable child
3641 where n is the number of children that can be resized.
3642 We can ignore x by choosing a delta1 that is a multiple of
3643 n. We want the height of this window to come out as
3653 The number of children n rquals the number of resizable
3654 children of this window + 1 because we know window itself
3655 is resizable (otherwise we would have signalled an error. */
3657 struct window
*w
= XWINDOW (window
);
3661 for (s
= w
->next
; !NILP (s
); s
= XWINDOW (s
)->next
)
3662 if (!window_fixed_size_p (XWINDOW (s
), widthflag
, 0))
3664 for (s
= w
->prev
; !NILP (s
); s
= XWINDOW (s
)->prev
)
3665 if (!window_fixed_size_p (XWINDOW (s
), widthflag
, 0))
3670 /* Add delta1 lines or columns to this window, and to the parent,
3671 keeping things consistent while not affecting siblings. */
3672 XSETINT (CURSIZE (parent
), opht
+ delta1
);
3673 (*setsizefun
) (window
, XINT (*sizep
) + delta1
, 0);
3675 /* Squeeze out delta1 lines or columns from our parent,
3676 shriking this window and siblings proportionately.
3677 This brings parent back to correct size.
3678 Delta1 was calculated so this makes this window the desired size,
3679 taking it all out of the siblings. */
3680 (*setsizefun
) (parent
, opht
, 0);
3685 XSETFASTINT (p
->last_modified
, 0);
3686 XSETFASTINT (p
->last_overlay_modified
, 0);
3688 /* Adjust glyph matrices. */
3689 adjust_glyphs (XFRAME (WINDOW_FRAME (XWINDOW (window
))));
3697 /***********************************************************************
3698 Resizing Mini-Windows
3699 ***********************************************************************/
3701 static void shrink_window_lowest_first
P_ ((struct window
*, int));
3703 enum save_restore_action
3710 static int save_restore_orig_size
P_ ((struct window
*,
3711 enum save_restore_action
));
3713 /* Shrink windows rooted in window W to HEIGHT. Take the space needed
3714 from lowest windows first. */
3717 shrink_window_lowest_first (w
, height
)
3725 xassert (!MINI_WINDOW_P (w
));
3727 /* Set redisplay hints. */
3728 XSETFASTINT (w
->last_modified
, 0);
3729 XSETFASTINT (w
->last_overlay_modified
, 0);
3730 windows_or_buffers_changed
++;
3731 FRAME_WINDOW_SIZES_CHANGED (XFRAME (WINDOW_FRAME (w
))) = 1;
3733 old_height
= XFASTINT (w
->height
);
3734 XSETFASTINT (w
->height
, height
);
3736 if (!NILP (w
->hchild
))
3738 for (child
= w
->hchild
; !NILP (child
); child
= c
->next
)
3740 c
= XWINDOW (child
);
3742 shrink_window_lowest_first (c
, height
);
3745 else if (!NILP (w
->vchild
))
3747 Lisp_Object last_child
;
3748 int delta
= old_height
- height
;
3753 /* Find the last child. We are taking space from lowest windows
3754 first, so we iterate over children from the last child
3756 for (child
= w
->vchild
; !NILP (child
); child
= XWINDOW (child
)->next
)
3759 /* Assign new heights. We leave only MIN_SAFE_WINDOW_HEIGHT. */
3760 for (child
= last_child
; delta
&& !NILP (child
); child
= c
->prev
)
3764 c
= XWINDOW (child
);
3765 this_one
= XFASTINT (c
->height
) - MIN_SAFE_WINDOW_HEIGHT
;
3767 if (this_one
> delta
)
3770 shrink_window_lowest_first (c
, XFASTINT (c
->height
) - this_one
);
3774 /* Compute new positions. */
3775 last_top
= XINT (w
->top
);
3776 for (child
= w
->vchild
; !NILP (child
); child
= c
->next
)
3778 c
= XWINDOW (child
);
3779 c
->top
= make_number (last_top
);
3780 shrink_window_lowest_first (c
, XFASTINT (c
->height
));
3781 last_top
+= XFASTINT (c
->height
);
3787 /* Save, restore, or check positions and sizes in the window tree
3788 rooted at W. ACTION says what to do.
3790 If ACTION is CHECK_ORIG_SIZES, check if orig_top and orig_height
3791 members are valid for all windows in the window tree. Value is
3792 non-zero if they are valid.
3794 If ACTION is SAVE_ORIG_SIZES, save members top and height in
3795 orig_top and orig_height for all windows in the tree.
3797 If ACTION is RESTORE_ORIG_SIZES, restore top and height from
3798 values stored in orig_top and orig_height for all windows. */
3801 save_restore_orig_size (w
, action
)
3803 enum save_restore_action action
;
3809 if (!NILP (w
->hchild
))
3811 if (!save_restore_orig_size (XWINDOW (w
->hchild
), action
))
3814 else if (!NILP (w
->vchild
))
3816 if (!save_restore_orig_size (XWINDOW (w
->vchild
), action
))
3822 case CHECK_ORIG_SIZES
:
3823 if (!INTEGERP (w
->orig_top
) || !INTEGERP (w
->orig_height
))
3827 case SAVE_ORIG_SIZES
:
3828 w
->orig_top
= w
->top
;
3829 w
->orig_height
= w
->height
;
3830 XSETFASTINT (w
->last_modified
, 0);
3831 XSETFASTINT (w
->last_overlay_modified
, 0);
3834 case RESTORE_ORIG_SIZES
:
3835 xassert (INTEGERP (w
->orig_top
) && INTEGERP (w
->orig_height
));
3836 w
->top
= w
->orig_top
;
3837 w
->height
= w
->orig_height
;
3838 w
->orig_height
= w
->orig_top
= Qnil
;
3839 XSETFASTINT (w
->last_modified
, 0);
3840 XSETFASTINT (w
->last_overlay_modified
, 0);
3847 w
= NILP (w
->next
) ? NULL
: XWINDOW (w
->next
);
3854 /* Grow mini-window W by DELTA lines, DELTA >= 0, or as much as we can
3855 without deleting other windows. */
3858 grow_mini_window (w
, delta
)
3862 struct frame
*f
= XFRAME (w
->frame
);
3863 struct window
*root
;
3865 xassert (MINI_WINDOW_P (w
));
3866 xassert (delta
>= 0);
3868 /* Check values of window_min_width and window_min_height for
3870 check_min_window_sizes ();
3872 /* Compute how much we can enlarge the mini-window without deleting
3874 root
= XWINDOW (FRAME_ROOT_WINDOW (f
));
3877 int min_height
= window_min_size (root
, 0, 0, 0);
3878 if (XFASTINT (root
->height
) - delta
< min_height
)
3879 /* Note that the root window may already be smaller than
3881 delta
= max (0, XFASTINT (root
->height
) - min_height
);
3886 /* Save original window sizes and positions, if not already done. */
3887 if (!save_restore_orig_size (root
, CHECK_ORIG_SIZES
))
3888 save_restore_orig_size (root
, SAVE_ORIG_SIZES
);
3890 /* Shrink other windows. */
3891 shrink_window_lowest_first (root
, XFASTINT (root
->height
) - delta
);
3893 /* Grow the mini-window. */
3894 w
->top
= make_number (XFASTINT (root
->top
) + XFASTINT (root
->height
));
3895 w
->height
= make_number (XFASTINT (w
->height
) + delta
);
3896 XSETFASTINT (w
->last_modified
, 0);
3897 XSETFASTINT (w
->last_overlay_modified
, 0);
3904 /* Shrink mini-window W. If there is recorded info about window sizes
3905 before a call to grow_mini_window, restore recorded window sizes.
3906 Otherwise, if the mini-window is higher than 1 line, resize it to 1
3910 shrink_mini_window (w
)
3913 struct frame
*f
= XFRAME (w
->frame
);
3914 struct window
*root
= XWINDOW (FRAME_ROOT_WINDOW (f
));
3916 if (save_restore_orig_size (root
, CHECK_ORIG_SIZES
))
3918 save_restore_orig_size (root
, RESTORE_ORIG_SIZES
);
3920 FRAME_WINDOW_SIZES_CHANGED (f
) = 1;
3921 windows_or_buffers_changed
= 1;
3923 else if (XFASTINT (w
->height
) > 1)
3925 /* Distribute the additional lines of the mini-window
3926 among the other windows. */
3928 XSETWINDOW (window
, w
);
3929 enlarge_window (window
, 1 - XFASTINT (w
->height
), 0, 0);
3935 /* Mark window cursors off for all windows in the window tree rooted
3936 at W by setting their phys_cursor_on_p flag to zero. Called from
3937 xterm.c, e.g. when a frame is cleared and thereby all cursors on
3938 the frame are cleared. */
3941 mark_window_cursors_off (w
)
3946 if (!NILP (w
->hchild
))
3947 mark_window_cursors_off (XWINDOW (w
->hchild
));
3948 else if (!NILP (w
->vchild
))
3949 mark_window_cursors_off (XWINDOW (w
->vchild
));
3951 w
->phys_cursor_on_p
= 0;
3953 w
= NILP (w
->next
) ? 0 : XWINDOW (w
->next
);
3958 /* Return number of lines of text (not counting mode lines) in W. */
3961 window_internal_height (w
)
3964 int ht
= XFASTINT (w
->height
);
3966 if (!MINI_WINDOW_P (w
))
3968 if (!NILP (w
->parent
)
3969 || !NILP (w
->vchild
)
3970 || !NILP (w
->hchild
)
3973 || WINDOW_WANTS_MODELINE_P (w
))
3976 if (WINDOW_WANTS_HEADER_LINE_P (w
))
3984 /* Return the number of columns in W.
3985 Don't count columns occupied by scroll bars or the vertical bar
3986 separating W from the sibling to its right. */
3989 window_internal_width (w
)
3992 struct frame
*f
= XFRAME (WINDOW_FRAME (w
));
3993 int width
= XINT (w
->width
);
3995 if (FRAME_HAS_VERTICAL_SCROLL_BARS (f
))
3996 /* Scroll bars occupy a few columns. */
3997 width
-= FRAME_SCROLL_BAR_COLS (f
);
3998 else if (!WINDOW_RIGHTMOST_P (w
) && !WINDOW_FULL_WIDTH_P (w
))
3999 /* The column of `|' characters separating side-by-side windows
4000 occupies one column only. */
4003 /* On window-systems, areas to the left and right of the window
4004 are used as fringes. */
4005 if (FRAME_WINDOW_P (f
))
4006 width
-= FRAME_FRINGE_COLS (f
);
4012 /************************************************************************
4014 ***********************************************************************/
4016 /* Scroll contents of window WINDOW up. If WHOLE is non-zero, scroll
4017 N screen-fulls, which is defined as the height of the window minus
4018 next_screen_context_lines. If WHOLE is zero, scroll up N lines
4019 instead. Negative values of N mean scroll down. NOERROR non-zero
4020 means don't signal an error if we try to move over BEGV or ZV,
4024 window_scroll (window
, n
, whole
, noerror
)
4030 /* If we must, use the pixel-based version which is much slower than
4031 the line-based one but can handle varying line heights. */
4032 if (FRAME_WINDOW_P (XFRAME (XWINDOW (window
)->frame
)))
4033 window_scroll_pixel_based (window
, n
, whole
, noerror
);
4035 window_scroll_line_based (window
, n
, whole
, noerror
);
4039 /* Implementation of window_scroll that works based on pixel line
4040 heights. See the comment of window_scroll for parameter
4044 window_scroll_pixel_based (window
, n
, whole
, noerror
)
4051 struct window
*w
= XWINDOW (window
);
4052 struct text_pos start
;
4054 int this_scroll_margin
;
4056 /* True if we fiddled the window vscroll field without really scrolling. */
4059 SET_TEXT_POS_FROM_MARKER (start
, w
->start
);
4061 /* If PT is not visible in WINDOW, move back one half of
4062 the screen. Allow PT to be partially visible, otherwise
4063 something like (scroll-down 1) with PT in the line before
4064 the partially visible one would recenter. */
4065 tem
= Fpos_visible_in_window_p (make_number (PT
), window
, Qt
);
4068 /* Move backward half the height of the window. Performance note:
4069 vmotion used here is about 10% faster, but would give wrong
4070 results for variable height lines. */
4071 init_iterator (&it
, w
, PT
, PT_BYTE
, NULL
, DEFAULT_FACE_ID
);
4072 it
.current_y
= it
.last_visible_y
;
4073 move_it_vertically (&it
, - window_box_height (w
) / 2);
4075 /* The function move_iterator_vertically may move over more than
4076 the specified y-distance. If it->w is small, e.g. a
4077 mini-buffer window, we may end up in front of the window's
4078 display area. This is the case when Start displaying at the
4079 start of the line containing PT in this case. */
4080 if (it
.current_y
<= 0)
4082 init_iterator (&it
, w
, PT
, PT_BYTE
, NULL
, DEFAULT_FACE_ID
);
4083 move_it_vertically (&it
, 0);
4087 start
= it
.current
.pos
;
4090 /* If scroll_preserve_screen_position is non-zero, we try to set
4091 point in the same window line as it is now, so get that line. */
4092 if (!NILP (Vscroll_preserve_screen_position
))
4094 start_display (&it
, w
, start
);
4095 move_it_to (&it
, PT
, -1, -1, -1, MOVE_TO_POS
);
4096 preserve_y
= it
.current_y
;
4101 /* Move iterator it from start the specified distance forward or
4102 backward. The result is the new window start. */
4103 start_display (&it
, w
, start
);
4106 int screen_full
= (window_box_height (w
)
4107 - next_screen_context_lines
* CANON_Y_UNIT (it
.f
));
4108 int dy
= n
* screen_full
;
4110 /* Note that move_it_vertically always moves the iterator to the
4111 start of a line. So, if the last line doesn't have a newline,
4112 we would end up at the start of the line ending at ZV. */
4114 move_it_vertically_backward (&it
, -dy
);
4116 move_it_to (&it
, ZV
, -1, it
.current_y
+ dy
, -1,
4117 MOVE_TO_POS
| MOVE_TO_Y
);
4120 move_it_by_lines (&it
, n
, 1);
4122 /* End if we end up at ZV or BEGV. */
4123 if ((n
> 0 && IT_CHARPOS (it
) == ZV
)
4124 || (n
< 0 && IT_CHARPOS (it
) == CHARPOS (start
)))
4126 if (IT_CHARPOS (it
) == ZV
)
4128 if (it
.current_y
+ it
.max_ascent
+ it
.max_descent
4129 > it
.last_visible_y
)
4131 /* The last line was only partially visible, make it fully
4133 w
->vscroll
= (it
.last_visible_y
4134 - it
.current_y
+ it
.max_ascent
+ it
.max_descent
);
4135 adjust_glyphs (it
.f
);
4140 Fsignal (Qend_of_buffer
, Qnil
);
4144 if (w
->vscroll
!= 0)
4145 /* The first line was only partially visible, make it fully
4151 Fsignal (Qbeginning_of_buffer
, Qnil
);
4154 /* If control gets here, then we vscrolled. */
4156 XBUFFER (w
->buffer
)->prevent_redisplay_optimizations_p
= 1;
4158 /* Don't try to change the window start below. */
4164 int pos
= IT_CHARPOS (it
);
4166 /* Set the window start, and set up the window for redisplay. */
4167 set_marker_restricted (w
->start
, make_number (pos
),
4169 bytepos
= XMARKER (w
->start
)->bytepos
;
4170 w
->start_at_line_beg
= ((pos
== BEGV
|| FETCH_BYTE (bytepos
- 1) == '\n')
4172 w
->update_mode_line
= Qt
;
4173 XSETFASTINT (w
->last_modified
, 0);
4174 XSETFASTINT (w
->last_overlay_modified
, 0);
4175 /* Set force_start so that redisplay_window will run the
4176 window-scroll-functions. */
4177 w
->force_start
= Qt
;
4180 it
.current_y
= it
.vpos
= 0;
4182 /* Preserve the screen position if we must. */
4183 if (preserve_y
>= 0)
4185 move_it_to (&it
, -1, -1, preserve_y
, -1, MOVE_TO_Y
);
4186 SET_PT_BOTH (IT_CHARPOS (it
), IT_BYTEPOS (it
));
4190 /* Move PT out of scroll margins. */
4191 this_scroll_margin
= max (0, scroll_margin
);
4192 this_scroll_margin
= min (this_scroll_margin
, XFASTINT (w
->height
) / 4);
4193 this_scroll_margin
*= CANON_Y_UNIT (it
.f
);
4197 /* We moved the window start towards ZV, so PT may be now
4198 in the scroll margin at the top. */
4199 move_it_to (&it
, PT
, -1, -1, -1, MOVE_TO_POS
);
4200 while (it
.current_y
< this_scroll_margin
)
4202 int prev
= it
.current_y
;
4203 move_it_by_lines (&it
, 1, 1);
4204 if (prev
== it
.current_y
)
4207 SET_PT_BOTH (IT_CHARPOS (it
), IT_BYTEPOS (it
));
4211 int charpos
, bytepos
;
4213 /* We moved the window start towards BEGV, so PT may be now
4214 in the scroll margin at the bottom. */
4215 move_it_to (&it
, PT
, -1,
4216 it
.last_visible_y
- this_scroll_margin
- 1, -1,
4217 MOVE_TO_POS
| MOVE_TO_Y
);
4219 /* Save our position, in case it's correct. */
4220 charpos
= IT_CHARPOS (it
);
4221 bytepos
= IT_BYTEPOS (it
);
4223 /* See if point is on a partially visible line at the end. */
4224 move_it_by_lines (&it
, 1, 1);
4225 if (it
.current_y
> it
.last_visible_y
)
4226 /* The last line was only partially visible, so back up two
4227 lines to make sure we're on a fully visible line. */
4229 move_it_by_lines (&it
, -2, 0);
4230 SET_PT_BOTH (IT_CHARPOS (it
), IT_BYTEPOS (it
));
4233 /* No, the position we saved is OK, so use it. */
4234 SET_PT_BOTH (charpos
, bytepos
);
4240 /* Implementation of window_scroll that works based on screen lines.
4241 See the comment of window_scroll for parameter descriptions. */
4244 window_scroll_line_based (window
, n
, whole
, noerror
)
4250 register struct window
*w
= XWINDOW (window
);
4251 register int opoint
= PT
, opoint_byte
= PT_BYTE
;
4252 register int pos
, pos_byte
;
4253 register int ht
= window_internal_height (w
);
4254 register Lisp_Object tem
;
4258 struct position posit
;
4261 /* If scrolling screen-fulls, compute the number of lines to
4262 scroll from the window's height. */
4264 n
*= max (1, ht
- next_screen_context_lines
);
4266 startpos
= marker_position (w
->start
);
4268 posit
= *compute_motion (startpos
, 0, 0, 0,
4270 window_internal_width (w
), XINT (w
->hscroll
),
4272 original_vpos
= posit
.vpos
;
4274 XSETFASTINT (tem
, PT
);
4275 tem
= Fpos_visible_in_window_p (tem
, window
, Qnil
);
4279 Fvertical_motion (make_number (- (ht
/ 2)), window
);
4284 lose
= n
< 0 && PT
== BEGV
;
4285 Fvertical_motion (make_number (n
), window
);
4289 SET_PT_BOTH (opoint
, opoint_byte
);
4296 Fsignal (Qbeginning_of_buffer
, Qnil
);
4301 int this_scroll_margin
= scroll_margin
;
4303 /* Don't use a scroll margin that is negative or too large. */
4304 if (this_scroll_margin
< 0)
4305 this_scroll_margin
= 0;
4307 if (XINT (w
->height
) < 4 * scroll_margin
)
4308 this_scroll_margin
= XINT (w
->height
) / 4;
4310 set_marker_restricted_both (w
->start
, w
->buffer
, pos
, pos_byte
);
4311 w
->start_at_line_beg
= bolp
;
4312 w
->update_mode_line
= Qt
;
4313 XSETFASTINT (w
->last_modified
, 0);
4314 XSETFASTINT (w
->last_overlay_modified
, 0);
4315 /* Set force_start so that redisplay_window will run
4316 the window-scroll-functions. */
4317 w
->force_start
= Qt
;
4319 if (whole
&& !NILP (Vscroll_preserve_screen_position
))
4321 SET_PT_BOTH (pos
, pos_byte
);
4322 Fvertical_motion (make_number (original_vpos
), window
);
4324 /* If we scrolled forward, put point enough lines down
4325 that it is outside the scroll margin. */
4330 if (this_scroll_margin
> 0)
4332 SET_PT_BOTH (pos
, pos_byte
);
4333 Fvertical_motion (make_number (this_scroll_margin
), window
);
4339 if (top_margin
<= opoint
)
4340 SET_PT_BOTH (opoint
, opoint_byte
);
4341 else if (!NILP (Vscroll_preserve_screen_position
))
4343 SET_PT_BOTH (pos
, pos_byte
);
4344 Fvertical_motion (make_number (original_vpos
), window
);
4347 SET_PT (top_margin
);
4353 /* If we scrolled backward, put point near the end of the window
4354 but not within the scroll margin. */
4355 SET_PT_BOTH (pos
, pos_byte
);
4356 tem
= Fvertical_motion (make_number (ht
- this_scroll_margin
), window
);
4357 if (XFASTINT (tem
) == ht
- this_scroll_margin
)
4360 bottom_margin
= PT
+ 1;
4362 if (bottom_margin
> opoint
)
4363 SET_PT_BOTH (opoint
, opoint_byte
);
4366 if (!NILP (Vscroll_preserve_screen_position
))
4368 SET_PT_BOTH (pos
, pos_byte
);
4369 Fvertical_motion (make_number (original_vpos
), window
);
4372 Fvertical_motion (make_number (-1), window
);
4381 Fsignal (Qend_of_buffer
, Qnil
);
4386 /* Scroll selected_window up or down. If N is nil, scroll a
4387 screen-full which is defined as the height of the window minus
4388 next_screen_context_lines. If N is the symbol `-', scroll.
4389 DIRECTION may be 1 meaning to scroll down, or -1 meaning to scroll
4390 up. This is the guts of Fscroll_up and Fscroll_down. */
4393 scroll_command (n
, direction
)
4397 int count
= SPECPDL_INDEX ();
4399 xassert (abs (direction
) == 1);
4401 /* If selected window's buffer isn't current, make it current for
4402 the moment. But don't screw up if window_scroll gets an error. */
4403 if (XBUFFER (XWINDOW (selected_window
)->buffer
) != current_buffer
)
4405 record_unwind_protect (save_excursion_restore
, save_excursion_save ());
4406 Fset_buffer (XWINDOW (selected_window
)->buffer
);
4408 /* Make redisplay consider other windows than just selected_window. */
4409 ++windows_or_buffers_changed
;
4413 window_scroll (selected_window
, direction
, 1, 0);
4414 else if (EQ (n
, Qminus
))
4415 window_scroll (selected_window
, -direction
, 1, 0);
4418 n
= Fprefix_numeric_value (n
);
4419 window_scroll (selected_window
, XINT (n
) * direction
, 0, 0);
4422 unbind_to (count
, Qnil
);
4425 DEFUN ("scroll-up", Fscroll_up
, Sscroll_up
, 0, 1, "P",
4426 doc
: /* Scroll text of current window upward ARG lines; or near full screen if no ARG.
4427 A near full screen is `next-screen-context-lines' less than a full screen.
4428 Negative ARG means scroll downward.
4429 If ARG is the atom `-', scroll downward by nearly full screen.
4430 When calling from a program, supply as argument a number, nil, or `-'. */)
4434 scroll_command (arg
, 1);
4438 DEFUN ("scroll-down", Fscroll_down
, Sscroll_down
, 0, 1, "P",
4439 doc
: /* Scroll text of current window down ARG lines; or near full screen if no ARG.
4440 A near full screen is `next-screen-context-lines' less than a full screen.
4441 Negative ARG means scroll upward.
4442 If ARG is the atom `-', scroll upward by nearly full screen.
4443 When calling from a program, supply as argument a number, nil, or `-'. */)
4447 scroll_command (arg
, -1);
4451 DEFUN ("other-window-for-scrolling", Fother_window_for_scrolling
, Sother_window_for_scrolling
, 0, 0, 0,
4452 doc
: /* Return the other window for \"other window scroll\" commands.
4453 If in the minibuffer, `minibuffer-scroll-window' if non-nil
4454 specifies the window.
4455 If `other-window-scroll-buffer' is non-nil, a window
4456 showing that buffer is used. */)
4461 if (MINI_WINDOW_P (XWINDOW (selected_window
))
4462 && !NILP (Vminibuf_scroll_window
))
4463 window
= Vminibuf_scroll_window
;
4464 /* If buffer is specified, scroll that buffer. */
4465 else if (!NILP (Vother_window_scroll_buffer
))
4467 window
= Fget_buffer_window (Vother_window_scroll_buffer
, Qnil
);
4469 window
= Fdisplay_buffer (Vother_window_scroll_buffer
, Qt
, Qnil
);
4473 /* Nothing specified; look for a neighboring window on the same
4475 window
= Fnext_window (selected_window
, Qnil
, Qnil
);
4477 if (EQ (window
, selected_window
))
4478 /* That didn't get us anywhere; look for a window on another
4481 window
= Fnext_window (window
, Qnil
, Qt
);
4482 while (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (window
))))
4483 && ! EQ (window
, selected_window
));
4486 CHECK_LIVE_WINDOW (window
);
4488 if (EQ (window
, selected_window
))
4489 error ("There is no other window");
4494 DEFUN ("scroll-other-window", Fscroll_other_window
, Sscroll_other_window
, 0, 1, "P",
4495 doc
: /* Scroll next window upward ARG lines; or near full screen if no ARG.
4496 A near full screen is `next-screen-context-lines' less than a full screen.
4497 The next window is the one below the current one; or the one at the top
4498 if the current one is at the bottom. Negative ARG means scroll downward.
4499 If ARG is the atom `-', scroll downward by nearly full screen.
4500 When calling from a program, supply as argument a number, nil, or `-'.
4502 If in the minibuffer, `minibuffer-scroll-window' if non-nil
4503 specifies the window to scroll.
4504 If `other-window-scroll-buffer' is non-nil, scroll the window
4505 showing that buffer, popping the buffer up if necessary. */)
4511 int count
= SPECPDL_INDEX ();
4513 window
= Fother_window_for_scrolling ();
4514 w
= XWINDOW (window
);
4516 /* Don't screw up if window_scroll gets an error. */
4517 record_unwind_protect (save_excursion_restore
, save_excursion_save ());
4518 ++windows_or_buffers_changed
;
4520 Fset_buffer (w
->buffer
);
4521 SET_PT (marker_position (w
->pointm
));
4524 window_scroll (window
, 1, 1, 1);
4525 else if (EQ (arg
, Qminus
))
4526 window_scroll (window
, -1, 1, 1);
4532 window_scroll (window
, XINT (arg
), 0, 1);
4535 set_marker_both (w
->pointm
, Qnil
, PT
, PT_BYTE
);
4536 unbind_to (count
, Qnil
);
4541 DEFUN ("scroll-left", Fscroll_left
, Sscroll_left
, 0, 1, "P",
4542 doc
: /* Scroll selected window display ARG columns left.
4543 Default for ARG is window width minus 2.
4544 Value is the total amount of leftward horizontal scrolling in
4545 effect after the change.
4546 If `automatic-hscrolling' is non-nil, the argument ARG modifies
4547 a lower bound for automatic scrolling, i.e. automatic scrolling
4548 will not scroll a window to a column less than the value returned
4549 by this function. */)
4551 register Lisp_Object arg
;
4555 struct window
*w
= XWINDOW (selected_window
);
4558 XSETFASTINT (arg
, window_internal_width (w
) - 2);
4560 arg
= Fprefix_numeric_value (arg
);
4562 hscroll
= XINT (w
->hscroll
) + XINT (arg
);
4563 result
= Fset_window_hscroll (selected_window
, make_number (hscroll
));
4565 if (interactive_p (0))
4566 w
->min_hscroll
= w
->hscroll
;
4571 DEFUN ("scroll-right", Fscroll_right
, Sscroll_right
, 0, 1, "P",
4572 doc
: /* Scroll selected window display ARG columns right.
4573 Default for ARG is window width minus 2.
4574 Value is the total amount of leftward horizontal scrolling in
4575 effect after the change.
4576 If `automatic-hscrolling' is non-nil, the argument ARG modifies
4577 a lower bound for automatic scrolling, i.e. automatic scrolling
4578 will not scroll a window to a column less than the value returned
4579 by this function. */)
4581 register Lisp_Object arg
;
4585 struct window
*w
= XWINDOW (selected_window
);
4588 XSETFASTINT (arg
, window_internal_width (w
) - 2);
4590 arg
= Fprefix_numeric_value (arg
);
4592 hscroll
= XINT (w
->hscroll
) - XINT (arg
);
4593 result
= Fset_window_hscroll (selected_window
, make_number (hscroll
));
4595 if (interactive_p (0))
4596 w
->min_hscroll
= w
->hscroll
;
4601 DEFUN ("minibuffer-selected-window", Fminibuffer_selected_window
, Sminibuffer_selected_window
, 0, 0, 0,
4602 doc
: /* Return the window which was selected when entering the minibuffer.
4603 Returns nil, if current window is not a minibuffer window. */)
4606 if (minibuf_level
> 0
4607 && MINI_WINDOW_P (XWINDOW (selected_window
))
4608 && !NILP (minibuf_selected_window
)
4609 && WINDOW_LIVE_P (minibuf_selected_window
))
4610 return minibuf_selected_window
;
4615 /* Value is the number of lines actually displayed in window W,
4616 as opposed to its height. */
4619 displayed_window_lines (w
)
4623 struct text_pos start
;
4624 int height
= window_box_height (w
);
4625 struct buffer
*old_buffer
;
4628 if (XBUFFER (w
->buffer
) != current_buffer
)
4630 old_buffer
= current_buffer
;
4631 set_buffer_internal (XBUFFER (w
->buffer
));
4636 /* In case W->start is out of the accessible range, do something
4637 reasonable. This happens in Info mode when Info-scroll-down
4638 calls (recenter -1) while W->start is 1. */
4639 if (XMARKER (w
->start
)->charpos
< BEGV
)
4640 SET_TEXT_POS (start
, BEGV
, BEGV_BYTE
);
4641 else if (XMARKER (w
->start
)->charpos
> ZV
)
4642 SET_TEXT_POS (start
, ZV
, ZV_BYTE
);
4644 SET_TEXT_POS_FROM_MARKER (start
, w
->start
);
4646 start_display (&it
, w
, start
);
4647 move_it_vertically (&it
, height
);
4648 bottom_y
= line_bottom_y (&it
);
4650 /* rms: On a non-window display,
4651 the value of it.vpos at the bottom of the screen
4652 seems to be 1 larger than window_box_height (w).
4653 This kludge fixes a bug whereby (move-to-window-line -1)
4654 when ZV is on the last screen line
4655 moves to the previous screen line instead of the last one. */
4656 if (! FRAME_WINDOW_P (XFRAME (w
->frame
)))
4659 /* Add in empty lines at the bottom of the window. */
4660 if (bottom_y
< height
)
4662 int uy
= CANON_Y_UNIT (it
.f
);
4663 it
.vpos
+= (height
- bottom_y
+ uy
- 1) / uy
;
4667 set_buffer_internal (old_buffer
);
4673 DEFUN ("recenter", Frecenter
, Srecenter
, 0, 1, "P",
4674 doc
: /* Center point in window and redisplay frame.
4675 With prefix argument ARG, recenter putting point on screen line ARG
4676 relative to the current window. If ARG is negative, it counts up from the
4677 bottom of the window. (ARG should be less than the height of the window.)
4679 If ARG is omitted or nil, erase the entire frame and then
4680 redraw with point in the center of the current window.
4681 Just C-u as prefix means put point in the center of the window
4682 and redisplay normally--don't erase and redraw the frame. */)
4684 register Lisp_Object arg
;
4686 struct window
*w
= XWINDOW (selected_window
);
4687 struct buffer
*buf
= XBUFFER (w
->buffer
);
4688 struct buffer
*obuf
= current_buffer
;
4690 int charpos
, bytepos
;
4692 /* If redisplay is suppressed due to an error, try again. */
4693 obuf
->display_error_modiff
= 0;
4699 /* Invalidate pixel data calculated for all compositions. */
4700 for (i
= 0; i
< n_compositions
; i
++)
4701 composition_table
[i
]->font
= NULL
;
4703 Fredraw_frame (w
->frame
);
4704 SET_FRAME_GARBAGED (XFRAME (WINDOW_FRAME (w
)));
4707 else if (CONSP (arg
)) /* Just C-u. */
4711 arg
= Fprefix_numeric_value (arg
);
4715 set_buffer_internal (buf
);
4717 /* Handle centering on a graphical frame specially. Such frames can
4718 have variable-height lines and centering point on the basis of
4719 line counts would lead to strange effects. */
4720 if (FRAME_WINDOW_P (XFRAME (w
->frame
)))
4727 SET_TEXT_POS (pt
, PT
, PT_BYTE
);
4728 start_display (&it
, w
, pt
);
4729 move_it_vertically (&it
, - window_box_height (w
) / 2);
4730 charpos
= IT_CHARPOS (it
);
4731 bytepos
= IT_BYTEPOS (it
);
4733 else if (XINT (arg
) < 0)
4737 int y0
, y1
, h
, nlines
;
4739 SET_TEXT_POS (pt
, PT
, PT_BYTE
);
4740 start_display (&it
, w
, pt
);
4743 /* The amount of pixels we have to move back is the window
4744 height minus what's displayed in the line containing PT,
4745 and the lines below. */
4746 nlines
= - XINT (arg
) - 1;
4747 move_it_by_lines (&it
, nlines
, 1);
4749 y1
= line_bottom_y (&it
);
4751 /* If we can't move down NLINES lines because we hit
4752 the end of the buffer, count in some empty lines. */
4753 if (it
.vpos
< nlines
)
4754 y1
+= (nlines
- it
.vpos
) * CANON_Y_UNIT (it
.f
);
4756 h
= window_box_height (w
) - (y1
- y0
);
4758 start_display (&it
, w
, pt
);
4759 move_it_vertically (&it
, - h
);
4760 charpos
= IT_CHARPOS (it
);
4761 bytepos
= IT_BYTEPOS (it
);
4765 struct position pos
;
4766 pos
= *vmotion (PT
, - XINT (arg
), w
);
4767 charpos
= pos
.bufpos
;
4768 bytepos
= pos
.bytepos
;
4773 struct position pos
;
4774 int ht
= window_internal_height (w
);
4777 arg
= make_number (ht
/ 2);
4778 else if (XINT (arg
) < 0)
4779 arg
= make_number (XINT (arg
) + ht
);
4781 pos
= *vmotion (PT
, - XINT (arg
), w
);
4782 charpos
= pos
.bufpos
;
4783 bytepos
= pos
.bytepos
;
4786 /* Set the new window start. */
4787 set_marker_both (w
->start
, w
->buffer
, charpos
, bytepos
);
4788 w
->window_end_valid
= Qnil
;
4790 w
->optional_new_start
= Qt
;
4792 if (bytepos
== BEGV_BYTE
|| FETCH_BYTE (bytepos
- 1) == '\n')
4793 w
->start_at_line_beg
= Qt
;
4795 w
->start_at_line_beg
= Qnil
;
4797 set_buffer_internal (obuf
);
4802 DEFUN ("window-text-height", Fwindow_text_height
, Swindow_text_height
,
4804 doc
: /* Return the height in lines of the text display area of WINDOW.
4805 This doesn't include the mode-line (or header-line if any) or any
4806 partial-height lines in the text display area. */)
4810 struct window
*w
= decode_window (window
);
4811 int pixel_height
= window_box_height (w
);
4812 int line_height
= pixel_height
/ CANON_Y_UNIT (XFRAME (w
->frame
));
4813 return make_number (line_height
);
4818 DEFUN ("move-to-window-line", Fmove_to_window_line
, Smove_to_window_line
,
4820 doc
: /* Position point relative to window.
4821 With no argument, position point at center of window.
4822 An argument specifies vertical position within the window;
4823 zero means top of window, negative means relative to bottom of window. */)
4827 struct window
*w
= XWINDOW (selected_window
);
4831 window
= selected_window
;
4832 start
= marker_position (w
->start
);
4833 if (start
< BEGV
|| start
> ZV
)
4835 int height
= window_internal_height (w
);
4836 Fvertical_motion (make_number (- (height
/ 2)), window
);
4837 set_marker_both (w
->start
, w
->buffer
, PT
, PT_BYTE
);
4838 w
->start_at_line_beg
= Fbolp ();
4839 w
->force_start
= Qt
;
4842 Fgoto_char (w
->start
);
4844 lines
= displayed_window_lines (w
);
4846 XSETFASTINT (arg
, lines
/ 2);
4849 arg
= Fprefix_numeric_value (arg
);
4851 XSETINT (arg
, XINT (arg
) + lines
);
4854 /* Skip past a partially visible first line. */
4856 XSETINT (arg
, XINT (arg
) + 1);
4858 return Fvertical_motion (arg
, window
);
4863 /***********************************************************************
4864 Window Configuration
4865 ***********************************************************************/
4867 struct save_window_data
4869 EMACS_INT size_from_Lisp_Vector_struct
;
4870 struct Lisp_Vector
*next_from_Lisp_Vector_struct
;
4871 Lisp_Object frame_width
, frame_height
, frame_menu_bar_lines
;
4872 Lisp_Object frame_tool_bar_lines
;
4873 Lisp_Object selected_frame
;
4874 Lisp_Object current_window
;
4875 Lisp_Object current_buffer
;
4876 Lisp_Object minibuf_scroll_window
;
4877 Lisp_Object minibuf_selected_window
;
4878 Lisp_Object root_window
;
4879 Lisp_Object focus_frame
;
4880 /* Record the values of window-min-width and window-min-height
4881 so that window sizes remain consistent with them. */
4882 Lisp_Object min_width
, min_height
;
4883 /* A vector, each of whose elements is a struct saved_window
4885 Lisp_Object saved_windows
;
4888 /* This is saved as a Lisp_Vector */
4891 /* these first two must agree with struct Lisp_Vector in lisp.h */
4892 EMACS_INT size_from_Lisp_Vector_struct
;
4893 struct Lisp_Vector
*next_from_Lisp_Vector_struct
;
4896 Lisp_Object buffer
, start
, pointm
, mark
;
4897 Lisp_Object left
, top
, width
, height
, hscroll
, min_hscroll
;
4898 Lisp_Object parent
, prev
;
4899 Lisp_Object start_at_line_beg
;
4900 Lisp_Object display_table
;
4901 Lisp_Object orig_top
, orig_height
;
4904 #define SAVED_WINDOW_VECTOR_SIZE 17 /* Arg to Fmake_vector */
4906 #define SAVED_WINDOW_N(swv,n) \
4907 ((struct saved_window *) (XVECTOR ((swv)->contents[(n)])))
4909 DEFUN ("window-configuration-p", Fwindow_configuration_p
, Swindow_configuration_p
, 1, 1, 0,
4910 doc
: /* Return t if OBJECT is a window-configuration object. */)
4914 if (WINDOW_CONFIGURATIONP (object
))
4919 DEFUN ("window-configuration-frame", Fwindow_configuration_frame
, Swindow_configuration_frame
, 1, 1, 0,
4920 doc
: /* Return the frame that CONFIG, a window-configuration object, is about. */)
4924 register struct save_window_data
*data
;
4925 struct Lisp_Vector
*saved_windows
;
4927 if (! WINDOW_CONFIGURATIONP (config
))
4928 wrong_type_argument (Qwindow_configuration_p
, config
);
4930 data
= (struct save_window_data
*) XVECTOR (config
);
4931 saved_windows
= XVECTOR (data
->saved_windows
);
4932 return XWINDOW (SAVED_WINDOW_N (saved_windows
, 0)->window
)->frame
;
4935 DEFUN ("set-window-configuration", Fset_window_configuration
,
4936 Sset_window_configuration
, 1, 1, 0,
4937 doc
: /* Set the configuration of windows and buffers as specified by CONFIGURATION.
4938 CONFIGURATION must be a value previously returned
4939 by `current-window-configuration' (which see).
4940 If CONFIGURATION was made from a frame that is now deleted,
4941 only frame-independent values can be restored. In this case,
4942 the return value is nil. Otherwise the value is t. */)
4944 Lisp_Object configuration
;
4946 register struct save_window_data
*data
;
4947 struct Lisp_Vector
*saved_windows
;
4948 Lisp_Object new_current_buffer
;
4953 while (!WINDOW_CONFIGURATIONP (configuration
))
4954 wrong_type_argument (Qwindow_configuration_p
, configuration
);
4956 data
= (struct save_window_data
*) XVECTOR (configuration
);
4957 saved_windows
= XVECTOR (data
->saved_windows
);
4959 new_current_buffer
= data
->current_buffer
;
4960 if (NILP (XBUFFER (new_current_buffer
)->name
))
4961 new_current_buffer
= Qnil
;
4963 old_point
= BUF_PT (XBUFFER (new_current_buffer
));
4965 frame
= XWINDOW (SAVED_WINDOW_N (saved_windows
, 0)->window
)->frame
;
4968 /* If f is a dead frame, don't bother rebuilding its window tree.
4969 However, there is other stuff we should still try to do below. */
4970 if (FRAME_LIVE_P (f
))
4972 register struct window
*w
;
4973 register struct saved_window
*p
;
4974 struct window
*root_window
;
4975 struct window
**leaf_windows
;
4979 /* If the frame has been resized since this window configuration was
4980 made, we change the frame to the size specified in the
4981 configuration, restore the configuration, and then resize it
4982 back. We keep track of the prevailing height in these variables. */
4983 int previous_frame_height
= FRAME_HEIGHT (f
);
4984 int previous_frame_width
= FRAME_WIDTH (f
);
4985 int previous_frame_menu_bar_lines
= FRAME_MENU_BAR_LINES (f
);
4986 int previous_frame_tool_bar_lines
= FRAME_TOOL_BAR_LINES (f
);
4988 /* The mouse highlighting code could get screwed up
4989 if it runs during this. */
4992 if (XFASTINT (data
->frame_height
) != previous_frame_height
4993 || XFASTINT (data
->frame_width
) != previous_frame_width
)
4994 change_frame_size (f
, XFASTINT (data
->frame_height
),
4995 XFASTINT (data
->frame_width
), 0, 0, 0);
4996 #if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
4997 if (XFASTINT (data
->frame_menu_bar_lines
)
4998 != previous_frame_menu_bar_lines
)
4999 x_set_menu_bar_lines (f
, data
->frame_menu_bar_lines
, make_number (0));
5000 #ifdef HAVE_WINDOW_SYSTEM
5001 if (XFASTINT (data
->frame_tool_bar_lines
)
5002 != previous_frame_tool_bar_lines
)
5003 x_set_tool_bar_lines (f
, data
->frame_tool_bar_lines
, make_number (0));
5007 /* "Swap out" point from the selected window
5008 into its buffer. We do this now, before
5009 restoring the window contents, and prevent it from
5010 being done later on when we select a new window. */
5011 if (! NILP (XWINDOW (selected_window
)->buffer
))
5013 w
= XWINDOW (selected_window
);
5014 set_marker_both (w
->pointm
,
5016 BUF_PT (XBUFFER (w
->buffer
)),
5017 BUF_PT_BYTE (XBUFFER (w
->buffer
)));
5020 windows_or_buffers_changed
++;
5021 FRAME_WINDOW_SIZES_CHANGED (f
) = 1;
5023 /* Problem: Freeing all matrices and later allocating them again
5024 is a serious redisplay flickering problem. What we would
5025 really like to do is to free only those matrices not reused
5027 root_window
= XWINDOW (FRAME_ROOT_WINDOW (f
));
5029 = (struct window
**) alloca (count_windows (root_window
)
5030 * sizeof (struct window
*));
5031 n_leaf_windows
= get_leaf_windows (root_window
, leaf_windows
, 0);
5033 /* Temporarily avoid any problems with windows that are smaller
5034 than they are supposed to be. */
5035 window_min_height
= 1;
5036 window_min_width
= 1;
5039 Mark all windows now on frame as "deleted".
5040 Restoring the new configuration "undeletes" any that are in it.
5042 Save their current buffers in their height fields, since we may
5043 need it later, if a buffer saved in the configuration is now
5045 delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f
)));
5047 for (k
= 0; k
< saved_windows
->size
; k
++)
5049 p
= SAVED_WINDOW_N (saved_windows
, k
);
5050 w
= XWINDOW (p
->window
);
5053 if (!NILP (p
->parent
))
5054 w
->parent
= SAVED_WINDOW_N (saved_windows
,
5055 XFASTINT (p
->parent
))->window
;
5059 if (!NILP (p
->prev
))
5061 w
->prev
= SAVED_WINDOW_N (saved_windows
,
5062 XFASTINT (p
->prev
))->window
;
5063 XWINDOW (w
->prev
)->next
= p
->window
;
5068 if (!NILP (w
->parent
))
5070 if (EQ (p
->width
, XWINDOW (w
->parent
)->width
))
5072 XWINDOW (w
->parent
)->vchild
= p
->window
;
5073 XWINDOW (w
->parent
)->hchild
= Qnil
;
5077 XWINDOW (w
->parent
)->hchild
= p
->window
;
5078 XWINDOW (w
->parent
)->vchild
= Qnil
;
5083 /* If we squirreled away the buffer in the window's height,
5085 if (BUFFERP (w
->height
))
5086 w
->buffer
= w
->height
;
5089 w
->width
= p
->width
;
5090 w
->height
= p
->height
;
5091 w
->hscroll
= p
->hscroll
;
5092 w
->min_hscroll
= p
->min_hscroll
;
5093 w
->display_table
= p
->display_table
;
5094 w
->orig_top
= p
->orig_top
;
5095 w
->orig_height
= p
->orig_height
;
5096 XSETFASTINT (w
->last_modified
, 0);
5097 XSETFASTINT (w
->last_overlay_modified
, 0);
5099 /* Reinstall the saved buffer and pointers into it. */
5100 if (NILP (p
->buffer
))
5101 w
->buffer
= p
->buffer
;
5104 if (!NILP (XBUFFER (p
->buffer
)->name
))
5105 /* If saved buffer is alive, install it. */
5107 w
->buffer
= p
->buffer
;
5108 w
->start_at_line_beg
= p
->start_at_line_beg
;
5109 set_marker_restricted (w
->start
, p
->start
, w
->buffer
);
5110 set_marker_restricted (w
->pointm
, p
->pointm
, w
->buffer
);
5111 Fset_marker (XBUFFER (w
->buffer
)->mark
,
5112 p
->mark
, w
->buffer
);
5114 /* As documented in Fcurrent_window_configuration, don't
5115 restore the location of point in the buffer which was
5116 current when the window configuration was recorded. */
5117 if (!EQ (p
->buffer
, new_current_buffer
)
5118 && XBUFFER (p
->buffer
) == current_buffer
)
5119 Fgoto_char (w
->pointm
);
5121 else if (NILP (w
->buffer
) || NILP (XBUFFER (w
->buffer
)->name
))
5122 /* Else unless window has a live buffer, get one. */
5124 w
->buffer
= Fcdr (Fcar (Vbuffer_alist
));
5125 /* This will set the markers to beginning of visible
5127 set_marker_restricted (w
->start
, make_number (0), w
->buffer
);
5128 set_marker_restricted (w
->pointm
, make_number (0),w
->buffer
);
5129 w
->start_at_line_beg
= Qt
;
5132 /* Keeping window's old buffer; make sure the markers
5135 /* Set window markers at start of visible range. */
5136 if (XMARKER (w
->start
)->buffer
== 0)
5137 set_marker_restricted (w
->start
, make_number (0),
5139 if (XMARKER (w
->pointm
)->buffer
== 0)
5140 set_marker_restricted_both (w
->pointm
, w
->buffer
,
5141 BUF_PT (XBUFFER (w
->buffer
)),
5142 BUF_PT_BYTE (XBUFFER (w
->buffer
)));
5143 w
->start_at_line_beg
= Qt
;
5148 FRAME_ROOT_WINDOW (f
) = data
->root_window
;
5149 /* Prevent "swapping out point" in the old selected window
5150 using the buffer that has been restored into it.
5151 Use the point value from the beginning of this function
5152 since unshow_buffer (called from delete_all_subwindows)
5153 could have altered it. */
5154 selected_window
= Qnil
;
5155 if (EQ (XWINDOW (data
->current_window
)->buffer
, new_current_buffer
))
5156 set_marker_restricted (XWINDOW (data
->current_window
)->pointm
,
5157 make_number (old_point
),
5158 XWINDOW (data
->current_window
)->buffer
);
5160 Fselect_window (data
->current_window
);
5161 XBUFFER (XWINDOW (selected_window
)->buffer
)->last_selected_window
5164 if (NILP (data
->focus_frame
)
5165 || (FRAMEP (data
->focus_frame
)
5166 && FRAME_LIVE_P (XFRAME (data
->focus_frame
))))
5167 Fredirect_frame_focus (frame
, data
->focus_frame
);
5169 #if 0 /* I don't understand why this is needed, and it causes problems
5170 when the frame's old selected window has been deleted. */
5171 if (f
!= selected_frame
&& FRAME_WINDOW_P (f
))
5172 do_switch_frame (WINDOW_FRAME (XWINDOW (data
->root_window
)),
5176 /* Set the screen height to the value it had before this function. */
5177 if (previous_frame_height
!= FRAME_HEIGHT (f
)
5178 || previous_frame_width
!= FRAME_WIDTH (f
))
5179 change_frame_size (f
, previous_frame_height
, previous_frame_width
,
5181 #if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
5182 if (previous_frame_menu_bar_lines
!= FRAME_MENU_BAR_LINES (f
))
5183 x_set_menu_bar_lines (f
, make_number (previous_frame_menu_bar_lines
),
5185 #ifdef HAVE_WINDOW_SYSTEM
5186 if (previous_frame_tool_bar_lines
!= FRAME_TOOL_BAR_LINES (f
))
5187 x_set_tool_bar_lines (f
, make_number (previous_frame_tool_bar_lines
),
5192 /* Now, free glyph matrices in windows that were not reused. */
5193 for (i
= n
= 0; i
< n_leaf_windows
; ++i
)
5195 if (NILP (leaf_windows
[i
]->buffer
))
5197 /* Assert it's not reused as a combination. */
5198 xassert (NILP (leaf_windows
[i
]->hchild
)
5199 && NILP (leaf_windows
[i
]->vchild
));
5200 free_window_matrices (leaf_windows
[i
]);
5202 else if (EQ (leaf_windows
[i
]->buffer
, new_current_buffer
))
5210 /* Fselect_window will have made f the selected frame, so we
5211 reselect the proper frame here. Fhandle_switch_frame will change the
5212 selected window too, but that doesn't make the call to
5213 Fselect_window above totally superfluous; it still sets f's
5215 if (FRAME_LIVE_P (XFRAME (data
->selected_frame
)))
5216 do_switch_frame (data
->selected_frame
, 0, 0);
5218 if (! NILP (Vwindow_configuration_change_hook
)
5219 && ! NILP (Vrun_hooks
))
5220 call1 (Vrun_hooks
, Qwindow_configuration_change_hook
);
5223 if (!NILP (new_current_buffer
))
5224 Fset_buffer (new_current_buffer
);
5226 /* Restore the minimum heights recorded in the configuration. */
5227 window_min_height
= XINT (data
->min_height
);
5228 window_min_width
= XINT (data
->min_width
);
5230 Vminibuf_scroll_window
= data
->minibuf_scroll_window
;
5231 minibuf_selected_window
= data
->minibuf_selected_window
;
5233 return (FRAME_LIVE_P (f
) ? Qt
: Qnil
);
5236 /* Mark all windows now on frame as deleted
5237 by setting their buffers to nil. */
5240 delete_all_subwindows (w
)
5241 register struct window
*w
;
5243 if (!NILP (w
->next
))
5244 delete_all_subwindows (XWINDOW (w
->next
));
5245 if (!NILP (w
->vchild
))
5246 delete_all_subwindows (XWINDOW (w
->vchild
));
5247 if (!NILP (w
->hchild
))
5248 delete_all_subwindows (XWINDOW (w
->hchild
));
5250 w
->height
= w
->buffer
; /* See Fset_window_configuration for excuse. */
5252 if (!NILP (w
->buffer
))
5255 /* We set all three of these fields to nil, to make sure that we can
5256 distinguish this dead window from any live window. Live leaf
5257 windows will have buffer set, and combination windows will have
5258 vchild or hchild set. */
5263 Vwindow_list
= Qnil
;
5267 count_windows (window
)
5268 register struct window
*window
;
5270 register int count
= 1;
5271 if (!NILP (window
->next
))
5272 count
+= count_windows (XWINDOW (window
->next
));
5273 if (!NILP (window
->vchild
))
5274 count
+= count_windows (XWINDOW (window
->vchild
));
5275 if (!NILP (window
->hchild
))
5276 count
+= count_windows (XWINDOW (window
->hchild
));
5281 /* Fill vector FLAT with leaf windows under W, starting at index I.
5282 Value is last index + 1. */
5285 get_leaf_windows (w
, flat
, i
)
5287 struct window
**flat
;
5292 if (!NILP (w
->hchild
))
5293 i
= get_leaf_windows (XWINDOW (w
->hchild
), flat
, i
);
5294 else if (!NILP (w
->vchild
))
5295 i
= get_leaf_windows (XWINDOW (w
->vchild
), flat
, i
);
5299 w
= NILP (w
->next
) ? 0 : XWINDOW (w
->next
);
5306 /* Return a pointer to the glyph W's physical cursor is on. Value is
5307 null if W's current matrix is invalid, so that no meaningfull glyph
5311 get_phys_cursor_glyph (w
)
5314 struct glyph_row
*row
;
5315 struct glyph
*glyph
;
5317 if (w
->phys_cursor
.vpos
>= 0
5318 && w
->phys_cursor
.vpos
< w
->current_matrix
->nrows
5319 && (row
= MATRIX_ROW (w
->current_matrix
, w
->phys_cursor
.vpos
),
5321 && row
->used
[TEXT_AREA
] > w
->phys_cursor
.hpos
)
5322 glyph
= row
->glyphs
[TEXT_AREA
] + w
->phys_cursor
.hpos
;
5331 save_window_save (window
, vector
, i
)
5333 struct Lisp_Vector
*vector
;
5336 register struct saved_window
*p
;
5337 register struct window
*w
;
5338 register Lisp_Object tem
;
5340 for (;!NILP (window
); window
= w
->next
)
5342 p
= SAVED_WINDOW_N (vector
, i
);
5343 w
= XWINDOW (window
);
5345 XSETFASTINT (w
->temslot
, i
++);
5347 p
->buffer
= w
->buffer
;
5350 p
->width
= w
->width
;
5351 p
->height
= w
->height
;
5352 p
->hscroll
= w
->hscroll
;
5353 p
->min_hscroll
= w
->min_hscroll
;
5354 p
->display_table
= w
->display_table
;
5355 p
->orig_top
= w
->orig_top
;
5356 p
->orig_height
= w
->orig_height
;
5357 if (!NILP (w
->buffer
))
5359 /* Save w's value of point in the window configuration.
5360 If w is the selected window, then get the value of point
5361 from the buffer; pointm is garbage in the selected window. */
5362 if (EQ (window
, selected_window
))
5364 p
->pointm
= Fmake_marker ();
5365 set_marker_both (p
->pointm
, w
->buffer
,
5366 BUF_PT (XBUFFER (w
->buffer
)),
5367 BUF_PT_BYTE (XBUFFER (w
->buffer
)));
5370 p
->pointm
= Fcopy_marker (w
->pointm
, Qnil
);
5372 p
->start
= Fcopy_marker (w
->start
, Qnil
);
5373 p
->start_at_line_beg
= w
->start_at_line_beg
;
5375 tem
= XBUFFER (w
->buffer
)->mark
;
5376 p
->mark
= Fcopy_marker (tem
, Qnil
);
5383 p
->start_at_line_beg
= Qnil
;
5386 if (NILP (w
->parent
))
5389 p
->parent
= XWINDOW (w
->parent
)->temslot
;
5394 p
->prev
= XWINDOW (w
->prev
)->temslot
;
5396 if (!NILP (w
->vchild
))
5397 i
= save_window_save (w
->vchild
, vector
, i
);
5398 if (!NILP (w
->hchild
))
5399 i
= save_window_save (w
->hchild
, vector
, i
);
5405 DEFUN ("current-window-configuration", Fcurrent_window_configuration
,
5406 Scurrent_window_configuration
, 0, 1, 0,
5407 doc
: /* Return an object representing the current window configuration of FRAME.
5408 If FRAME is nil or omitted, use the selected frame.
5409 This describes the number of windows, their sizes and current buffers,
5410 and for each displayed buffer, where display starts, and the positions of
5411 point and mark. An exception is made for point in the current buffer:
5412 its value is -not- saved.
5413 This also records the currently selected frame, and FRAME's focus
5414 redirection (see `redirect-frame-focus'). */)
5418 register Lisp_Object tem
;
5419 register int n_windows
;
5420 register struct save_window_data
*data
;
5421 register struct Lisp_Vector
*vec
;
5426 frame
= selected_frame
;
5427 CHECK_LIVE_FRAME (frame
);
5430 n_windows
= count_windows (XWINDOW (FRAME_ROOT_WINDOW (f
)));
5431 vec
= allocate_other_vector (VECSIZE (struct save_window_data
));
5432 data
= (struct save_window_data
*)vec
;
5434 XSETFASTINT (data
->frame_width
, FRAME_WIDTH (f
));
5435 XSETFASTINT (data
->frame_height
, FRAME_HEIGHT (f
));
5436 XSETFASTINT (data
->frame_menu_bar_lines
, FRAME_MENU_BAR_LINES (f
));
5437 XSETFASTINT (data
->frame_tool_bar_lines
, FRAME_TOOL_BAR_LINES (f
));
5438 data
->selected_frame
= selected_frame
;
5439 data
->current_window
= FRAME_SELECTED_WINDOW (f
);
5440 XSETBUFFER (data
->current_buffer
, current_buffer
);
5441 data
->minibuf_scroll_window
= minibuf_level
> 0 ? Vminibuf_scroll_window
: Qnil
;
5442 data
->minibuf_selected_window
= minibuf_level
> 0 ? minibuf_selected_window
: Qnil
;
5443 data
->root_window
= FRAME_ROOT_WINDOW (f
);
5444 data
->focus_frame
= FRAME_FOCUS_FRAME (f
);
5445 XSETINT (data
->min_height
, window_min_height
);
5446 XSETINT (data
->min_width
, window_min_width
);
5447 tem
= Fmake_vector (make_number (n_windows
), Qnil
);
5448 data
->saved_windows
= tem
;
5449 for (i
= 0; i
< n_windows
; i
++)
5450 XVECTOR (tem
)->contents
[i
]
5451 = Fmake_vector (make_number (SAVED_WINDOW_VECTOR_SIZE
), Qnil
);
5452 save_window_save (FRAME_ROOT_WINDOW (f
), XVECTOR (tem
), 0);
5453 XSETWINDOW_CONFIGURATION (tem
, data
);
5457 DEFUN ("save-window-excursion", Fsave_window_excursion
, Ssave_window_excursion
,
5459 doc
: /* Execute body, preserving window sizes and contents.
5460 Restore which buffer appears in which window, where display starts,
5461 and the value of point and mark for each window.
5462 Also restore the choice of selected window.
5463 Also restore which buffer is current.
5464 Does not restore the value of point in current buffer.
5465 usage: (save-window-excursion BODY ...) */)
5469 register Lisp_Object val
;
5470 register int count
= SPECPDL_INDEX ();
5472 record_unwind_protect (Fset_window_configuration
,
5473 Fcurrent_window_configuration (Qnil
));
5474 val
= Fprogn (args
);
5475 return unbind_to (count
, val
);
5479 /***********************************************************************
5481 ***********************************************************************/
5483 DEFUN ("set-window-margins", Fset_window_margins
, Sset_window_margins
,
5485 doc
: /* Set width of marginal areas of window WINDOW.
5486 If window is nil, set margins of the currently selected window.
5487 First parameter LEFT-WIDTH specifies the number of character
5488 cells to reserve for the left marginal area. Second parameter
5489 RIGHT-WIDTH does the same for the right marginal area.
5490 A nil width parameter means no margin. */)
5491 (window
, left
, right
)
5492 Lisp_Object window
, left
, right
;
5494 struct window
*w
= decode_window (window
);
5497 CHECK_NUMBER_OR_FLOAT (left
);
5499 CHECK_NUMBER_OR_FLOAT (right
);
5501 /* Check widths < 0 and translate a zero width to nil.
5502 Margins that are too wide have to be checked elsewhere. */
5503 if ((INTEGERP (left
) && XINT (left
) < 0)
5504 || (FLOATP (left
) && XFLOAT_DATA (left
) <= 0))
5505 XSETFASTINT (left
, 0);
5506 if (INTEGERP (left
) && XFASTINT (left
) == 0)
5509 if ((INTEGERP (right
) && XINT (right
) < 0)
5510 || (FLOATP (right
) && XFLOAT_DATA (right
) <= 0))
5511 XSETFASTINT (right
, 0);
5512 if (INTEGERP (right
) && XFASTINT (right
) == 0)
5515 w
->left_margin_width
= left
;
5516 w
->right_margin_width
= right
;
5518 ++windows_or_buffers_changed
;
5519 adjust_glyphs (XFRAME (WINDOW_FRAME (w
)));
5524 DEFUN ("window-margins", Fwindow_margins
, Swindow_margins
,
5526 doc
: /* Get width of marginal areas of window WINDOW.
5527 If WINDOW is omitted or nil, use the currently selected window.
5528 Value is a cons of the form (LEFT-WIDTH . RIGHT-WIDTH).
5529 If a marginal area does not exist, its width will be returned
5534 struct window
*w
= decode_window (window
);
5535 return Fcons (w
->left_margin_width
, w
->right_margin_width
);
5540 /***********************************************************************
5542 ***********************************************************************/
5544 DEFUN ("window-vscroll", Fwindow_vscroll
, Swindow_vscroll
, 0, 1, 0,
5545 doc
: /* Return the amount by which WINDOW is scrolled vertically.
5546 Use the selected window if WINDOW is nil or omitted.
5547 Value is a multiple of the canonical character height of WINDOW. */)
5556 window
= selected_window
;
5558 CHECK_WINDOW (window
);
5559 w
= XWINDOW (window
);
5560 f
= XFRAME (w
->frame
);
5562 if (FRAME_WINDOW_P (f
))
5563 result
= CANON_Y_FROM_PIXEL_Y (f
, -w
->vscroll
);
5565 result
= make_number (0);
5570 DEFUN ("set-window-vscroll", Fset_window_vscroll
, Sset_window_vscroll
,
5572 doc
: /* Set amount by which WINDOW should be scrolled vertically to VSCROLL.
5573 WINDOW nil means use the selected window. VSCROLL is a non-negative
5574 multiple of the canonical character height of WINDOW. */)
5576 Lisp_Object window
, vscroll
;
5582 window
= selected_window
;
5584 CHECK_WINDOW (window
);
5585 CHECK_NUMBER_OR_FLOAT (vscroll
);
5587 w
= XWINDOW (window
);
5588 f
= XFRAME (w
->frame
);
5590 if (FRAME_WINDOW_P (f
))
5592 int old_dy
= w
->vscroll
;
5594 w
->vscroll
= - CANON_Y_UNIT (f
) * XFLOATINT (vscroll
);
5595 w
->vscroll
= min (w
->vscroll
, 0);
5597 /* Adjust glyph matrix of the frame if the virtual display
5598 area becomes larger than before. */
5599 if (w
->vscroll
< 0 && w
->vscroll
< old_dy
)
5602 /* Prevent redisplay shortcuts. */
5603 XBUFFER (w
->buffer
)->prevent_redisplay_optimizations_p
= 1;
5606 return Fwindow_vscroll (window
);
5610 /* Call FN for all leaf windows on frame F. FN is called with the
5611 first argument being a pointer to the leaf window, and with
5612 additional argument USER_DATA. Stops when FN returns 0. */
5615 foreach_window (f
, fn
, user_data
)
5617 int (* fn
) P_ ((struct window
*, void *));
5620 foreach_window_1 (XWINDOW (FRAME_ROOT_WINDOW (f
)), fn
, user_data
);
5624 /* Helper function for foreach_window. Call FN for all leaf windows
5625 reachable from W. FN is called with the first argument being a
5626 pointer to the leaf window, and with additional argument USER_DATA.
5627 Stop when FN returns 0. Value is 0 if stopped by FN. */
5630 foreach_window_1 (w
, fn
, user_data
)
5632 int (* fn
) P_ ((struct window
*, void *));
5637 for (cont
= 1; w
&& cont
;)
5639 if (!NILP (w
->hchild
))
5640 cont
= foreach_window_1 (XWINDOW (w
->hchild
), fn
, user_data
);
5641 else if (!NILP (w
->vchild
))
5642 cont
= foreach_window_1 (XWINDOW (w
->vchild
), fn
, user_data
);
5644 cont
= fn (w
, user_data
);
5646 w
= NILP (w
->next
) ? 0 : XWINDOW (w
->next
);
5653 /* Freeze or unfreeze the window start of W unless it is a
5654 mini-window or the selected window. FREEZE_P non-null means freeze
5655 the window start. */
5658 freeze_window_start (w
, freeze_p
)
5662 if (w
== XWINDOW (selected_window
)
5663 || MINI_WINDOW_P (w
)
5664 || (MINI_WINDOW_P (XWINDOW (selected_window
))
5665 && ! NILP (Vminibuf_scroll_window
)
5666 && w
== XWINDOW (Vminibuf_scroll_window
)))
5669 w
->frozen_window_start_p
= freeze_p
!= NULL
;
5674 /* Freeze or unfreeze the window starts of all leaf windows on frame
5675 F, except the selected window and a mini-window. FREEZE_P non-zero
5676 means freeze the window start. */
5679 freeze_window_starts (f
, freeze_p
)
5683 foreach_window (f
, freeze_window_start
, (void *) (freeze_p
? f
: 0));
5687 /***********************************************************************
5689 ***********************************************************************/
5691 /* Return 1 if window configurations C1 and C2
5692 describe the same state of affairs. This is used by Fequal. */
5695 compare_window_configurations (c1
, c2
, ignore_positions
)
5697 int ignore_positions
;
5699 register struct save_window_data
*d1
, *d2
;
5700 struct Lisp_Vector
*sw1
, *sw2
;
5703 if (!WINDOW_CONFIGURATIONP (c1
))
5704 wrong_type_argument (Qwindow_configuration_p
, c1
);
5705 if (!WINDOW_CONFIGURATIONP (c2
))
5706 wrong_type_argument (Qwindow_configuration_p
, c2
);
5708 d1
= (struct save_window_data
*) XVECTOR (c1
);
5709 d2
= (struct save_window_data
*) XVECTOR (c2
);
5710 sw1
= XVECTOR (d1
->saved_windows
);
5711 sw2
= XVECTOR (d2
->saved_windows
);
5713 if (! EQ (d1
->frame_width
, d2
->frame_width
))
5715 if (! EQ (d1
->frame_height
, d2
->frame_height
))
5717 if (! EQ (d1
->frame_menu_bar_lines
, d2
->frame_menu_bar_lines
))
5719 if (! EQ (d1
->selected_frame
, d2
->selected_frame
))
5721 /* Don't compare the current_window field directly.
5722 Instead see w1_is_current and w2_is_current, below. */
5723 if (! EQ (d1
->current_buffer
, d2
->current_buffer
))
5725 if (! ignore_positions
)
5727 if (! EQ (d1
->minibuf_scroll_window
, d2
->minibuf_scroll_window
))
5729 if (! EQ (d1
->minibuf_selected_window
, d2
->minibuf_selected_window
))
5732 /* Don't compare the root_window field.
5733 We don't require the two configurations
5734 to use the same window object,
5735 and the two root windows must be equivalent
5736 if everything else compares equal. */
5737 if (! EQ (d1
->focus_frame
, d2
->focus_frame
))
5739 if (! EQ (d1
->min_width
, d2
->min_width
))
5741 if (! EQ (d1
->min_height
, d2
->min_height
))
5744 /* Verify that the two confis have the same number of windows. */
5745 if (sw1
->size
!= sw2
->size
)
5748 for (i
= 0; i
< sw1
->size
; i
++)
5750 struct saved_window
*p1
, *p2
;
5751 int w1_is_current
, w2_is_current
;
5753 p1
= SAVED_WINDOW_N (sw1
, i
);
5754 p2
= SAVED_WINDOW_N (sw2
, i
);
5756 /* Verify that the current windows in the two
5757 configurations correspond to each other. */
5758 w1_is_current
= EQ (d1
->current_window
, p1
->window
);
5759 w2_is_current
= EQ (d2
->current_window
, p2
->window
);
5761 if (w1_is_current
!= w2_is_current
)
5764 /* Verify that the corresponding windows do match. */
5765 if (! EQ (p1
->buffer
, p2
->buffer
))
5767 if (! EQ (p1
->left
, p2
->left
))
5769 if (! EQ (p1
->top
, p2
->top
))
5771 if (! EQ (p1
->width
, p2
->width
))
5773 if (! EQ (p1
->height
, p2
->height
))
5775 if (! EQ (p1
->display_table
, p2
->display_table
))
5777 if (! EQ (p1
->parent
, p2
->parent
))
5779 if (! EQ (p1
->prev
, p2
->prev
))
5781 if (! ignore_positions
)
5783 if (! EQ (p1
->hscroll
, p2
->hscroll
))
5785 if (!EQ (p1
->min_hscroll
, p2
->min_hscroll
))
5787 if (! EQ (p1
->start_at_line_beg
, p2
->start_at_line_beg
))
5789 if (NILP (Fequal (p1
->start
, p2
->start
)))
5791 if (NILP (Fequal (p1
->pointm
, p2
->pointm
)))
5793 if (NILP (Fequal (p1
->mark
, p2
->mark
)))
5801 DEFUN ("compare-window-configurations", Fcompare_window_configurations
,
5802 Scompare_window_configurations
, 2, 2, 0,
5803 doc
: /* Compare two window configurations as regards the structure of windows.
5804 This function ignores details such as the values of point and mark
5805 and scrolling positions. */)
5809 if (compare_window_configurations (x
, y
, 1))
5817 struct frame
*f
= make_terminal_frame ();
5818 XSETFRAME (selected_frame
, f
);
5819 Vterminal_frame
= selected_frame
;
5820 minibuf_window
= f
->minibuffer_window
;
5821 selected_window
= f
->selected_window
;
5822 last_nonminibuf_frame
= f
;
5824 window_initialized
= 1;
5830 Vwindow_list
= Qnil
;
5836 Qwindow_size_fixed
= intern ("window-size-fixed");
5837 staticpro (&Qwindow_size_fixed
);
5839 staticpro (&Qwindow_configuration_change_hook
);
5840 Qwindow_configuration_change_hook
5841 = intern ("window-configuration-change-hook");
5843 Qwindowp
= intern ("windowp");
5844 staticpro (&Qwindowp
);
5846 Qwindow_configuration_p
= intern ("window-configuration-p");
5847 staticpro (&Qwindow_configuration_p
);
5849 Qwindow_live_p
= intern ("window-live-p");
5850 staticpro (&Qwindow_live_p
);
5852 Qtemp_buffer_show_hook
= intern ("temp-buffer-show-hook");
5853 staticpro (&Qtemp_buffer_show_hook
);
5855 staticpro (&Vwindow_list
);
5857 minibuf_selected_window
= Qnil
;
5858 staticpro (&minibuf_selected_window
);
5860 DEFVAR_LISP ("temp-buffer-show-function", &Vtemp_buffer_show_function
,
5861 doc
: /* Non-nil means call as function to display a help buffer.
5862 The function is called with one argument, the buffer to be displayed.
5863 Used by `with-output-to-temp-buffer'.
5864 If this function is used, then it must do the entire job of showing
5865 the buffer; `temp-buffer-show-hook' is not run unless this function runs it. */);
5866 Vtemp_buffer_show_function
= Qnil
;
5868 DEFVAR_LISP ("display-buffer-function", &Vdisplay_buffer_function
,
5869 doc
: /* If non-nil, function to call to handle `display-buffer'.
5870 It will receive two args, the buffer and a flag which if non-nil means
5871 that the currently selected window is not acceptable.
5872 Commands such as `switch-to-buffer-other-window' and `find-file-other-window'
5873 work using this function. */);
5874 Vdisplay_buffer_function
= Qnil
;
5876 DEFVAR_LISP ("even-window-heights", &Veven_window_heights
,
5877 doc
: /* *If non-nil, `display-buffer' should even the window heights.
5878 If nil, `display-buffer' will leave the window configuration alone. */);
5879 Veven_window_heights
= Qt
;
5881 DEFVAR_LISP ("minibuffer-scroll-window", &Vminibuf_scroll_window
,
5882 doc
: /* Non-nil means it is the window that C-M-v in minibuffer should scroll. */);
5883 Vminibuf_scroll_window
= Qnil
;
5885 DEFVAR_BOOL ("mode-line-in-non-selected-windows", &mode_line_in_non_selected_windows
,
5886 doc
: /* Non-nil means to use `mode-line-inactive' face in non-selected windows.
5887 If the minibuffer is active, the `minibuffer-scroll-window' mode line
5888 is displayed in the `mode-line' face. */);
5889 mode_line_in_non_selected_windows
= 1;
5891 DEFVAR_LISP ("other-window-scroll-buffer", &Vother_window_scroll_buffer
,
5892 doc
: /* If non-nil, this is a buffer and \\[scroll-other-window] should scroll its window. */);
5893 Vother_window_scroll_buffer
= Qnil
;
5895 DEFVAR_BOOL ("pop-up-frames", &pop_up_frames
,
5896 doc
: /* *Non-nil means `display-buffer' should make a separate frame. */);
5899 DEFVAR_BOOL ("display-buffer-reuse-frames", &display_buffer_reuse_frames
,
5900 doc
: /* *Non-nil means `display-buffer' should reuse frames.
5901 If the buffer in question is already displayed in a frame, raise that frame. */);
5902 display_buffer_reuse_frames
= 0;
5904 DEFVAR_LISP ("pop-up-frame-function", &Vpop_up_frame_function
,
5905 doc
: /* Function to call to handle automatic new frame creation.
5906 It is called with no arguments and should return a newly created frame.
5908 A typical value might be `(lambda () (new-frame pop-up-frame-alist))'
5909 where `pop-up-frame-alist' would hold the default frame parameters. */);
5910 Vpop_up_frame_function
= Qnil
;
5912 DEFVAR_LISP ("special-display-buffer-names", &Vspecial_display_buffer_names
,
5913 doc
: /* *List of buffer names that should have their own special frames.
5914 Displaying a buffer whose name is in this list makes a special frame for it
5915 using `special-display-function'. See also `special-display-regexps'.
5917 An element of the list can be a list instead of just a string.
5918 There are two ways to use a list as an element:
5919 (BUFFER FRAME-PARAMETERS...) (BUFFER FUNCTION OTHER-ARGS...)
5920 In the first case, FRAME-PARAMETERS are used to create the frame.
5921 In the latter case, FUNCTION is called with BUFFER as the first argument,
5922 followed by OTHER-ARGS--it can display BUFFER in any way it likes.
5923 All this is done by the function found in `special-display-function'.
5925 If this variable appears \"not to work\", because you add a name to it
5926 but that buffer still appears in the selected window, look at the
5927 values of `same-window-buffer-names' and `same-window-regexps'.
5928 Those variables take precedence over this one. */);
5929 Vspecial_display_buffer_names
= Qnil
;
5931 DEFVAR_LISP ("special-display-regexps", &Vspecial_display_regexps
,
5932 doc
: /* *List of regexps saying which buffers should have their own special frames.
5933 If a buffer name matches one of these regexps, it gets its own frame.
5934 Displaying a buffer whose name is in this list makes a special frame for it
5935 using `special-display-function'.
5937 An element of the list can be a list instead of just a string.
5938 There are two ways to use a list as an element:
5939 (REGEXP FRAME-PARAMETERS...) (REGEXP FUNCTION OTHER-ARGS...)
5940 In the first case, FRAME-PARAMETERS are used to create the frame.
5941 In the latter case, FUNCTION is called with the buffer as first argument,
5942 followed by OTHER-ARGS--it can display the buffer in any way it likes.
5943 All this is done by the function found in `special-display-function'.
5945 If this variable appears \"not to work\", because you add a regexp to it
5946 but the matching buffers still appear in the selected window, look at the
5947 values of `same-window-buffer-names' and `same-window-regexps'.
5948 Those variables take precedence over this one. */);
5949 Vspecial_display_regexps
= Qnil
;
5951 DEFVAR_LISP ("special-display-function", &Vspecial_display_function
,
5952 doc
: /* Function to call to make a new frame for a special buffer.
5953 It is called with two arguments, the buffer and optional buffer specific
5954 data, and should return a window displaying that buffer.
5955 The default value makes a separate frame for the buffer,
5956 using `special-display-frame-alist' to specify the frame parameters.
5958 A buffer is special if its is listed in `special-display-buffer-names'
5959 or matches a regexp in `special-display-regexps'. */);
5960 Vspecial_display_function
= Qnil
;
5962 DEFVAR_LISP ("same-window-buffer-names", &Vsame_window_buffer_names
,
5963 doc
: /* *List of buffer names that should appear in the selected window.
5964 Displaying one of these buffers using `display-buffer' or `pop-to-buffer'
5965 switches to it in the selected window, rather than making it appear
5966 in some other window.
5968 An element of the list can be a cons cell instead of just a string.
5969 Then the car must be a string, which specifies the buffer name.
5970 This is for compatibility with `special-display-buffer-names';
5971 the cdr of the cons cell is ignored.
5973 See also `same-window-regexps'. */);
5974 Vsame_window_buffer_names
= Qnil
;
5976 DEFVAR_LISP ("same-window-regexps", &Vsame_window_regexps
,
5977 doc
: /* *List of regexps saying which buffers should appear in the selected window.
5978 If a buffer name matches one of these regexps, then displaying it
5979 using `display-buffer' or `pop-to-buffer' switches to it
5980 in the selected window, rather than making it appear in some other window.
5982 An element of the list can be a cons cell instead of just a string.
5983 Then the car must be a string, which specifies the buffer name.
5984 This is for compatibility with `special-display-buffer-names';
5985 the cdr of the cons cell is ignored.
5987 See also `same-window-buffer-names'. */);
5988 Vsame_window_regexps
= Qnil
;
5990 DEFVAR_BOOL ("pop-up-windows", &pop_up_windows
,
5991 doc
: /* *Non-nil means display-buffer should make new windows. */);
5994 DEFVAR_INT ("next-screen-context-lines", &next_screen_context_lines
,
5995 doc
: /* *Number of lines of continuity when scrolling by screenfuls. */);
5996 next_screen_context_lines
= 2;
5998 DEFVAR_INT ("split-height-threshold", &split_height_threshold
,
5999 doc
: /* *display-buffer would prefer to split the largest window if this large.
6000 If there is only one window, it is split regardless of this value. */);
6001 split_height_threshold
= 500;
6003 DEFVAR_INT ("window-min-height", &window_min_height
,
6004 doc
: /* *Delete any window less than this tall (including its mode line). */);
6005 window_min_height
= 4;
6007 DEFVAR_INT ("window-min-width", &window_min_width
,
6008 doc
: /* *Delete any window less than this wide. */);
6009 window_min_width
= 10;
6011 DEFVAR_LISP ("scroll-preserve-screen-position",
6012 &Vscroll_preserve_screen_position
,
6013 doc
: /* *Non-nil means scroll commands move point to keep its screen line unchanged. */);
6014 Vscroll_preserve_screen_position
= Qnil
;
6016 DEFVAR_LISP ("window-configuration-change-hook",
6017 &Vwindow_configuration_change_hook
,
6018 doc
: /* Functions to call when window configuration changes.
6019 The selected frame is the one whose configuration has changed. */);
6020 Vwindow_configuration_change_hook
= Qnil
;
6022 DEFVAR_BOOL ("window-size-fixed", &window_size_fixed
,
6023 doc
: /* Non-nil in a buffer means windows displaying the buffer are fixed-size.
6024 Emacs won't change the size of any window displaying that buffer,
6025 unless you explicitly change the size, or Emacs has no other choice.
6026 This variable automatically becomes buffer-local when set. */);
6027 Fmake_variable_buffer_local (Qwindow_size_fixed
);
6028 window_size_fixed
= 0;
6030 defsubr (&Sselected_window
);
6031 defsubr (&Sminibuffer_window
);
6032 defsubr (&Swindow_minibuffer_p
);
6033 defsubr (&Swindowp
);
6034 defsubr (&Swindow_live_p
);
6035 defsubr (&Spos_visible_in_window_p
);
6036 defsubr (&Swindow_buffer
);
6037 defsubr (&Swindow_height
);
6038 defsubr (&Swindow_width
);
6039 defsubr (&Swindow_hscroll
);
6040 defsubr (&Sset_window_hscroll
);
6041 defsubr (&Swindow_redisplay_end_trigger
);
6042 defsubr (&Sset_window_redisplay_end_trigger
);
6043 defsubr (&Swindow_edges
);
6044 defsubr (&Scoordinates_in_window_p
);
6045 defsubr (&Swindow_at
);
6046 defsubr (&Swindow_point
);
6047 defsubr (&Swindow_start
);
6048 defsubr (&Swindow_end
);
6049 defsubr (&Sset_window_point
);
6050 defsubr (&Sset_window_start
);
6051 defsubr (&Swindow_dedicated_p
);
6052 defsubr (&Sset_window_dedicated_p
);
6053 defsubr (&Swindow_display_table
);
6054 defsubr (&Sset_window_display_table
);
6055 defsubr (&Snext_window
);
6056 defsubr (&Sprevious_window
);
6057 defsubr (&Sother_window
);
6058 defsubr (&Sget_lru_window
);
6059 defsubr (&Sget_largest_window
);
6060 defsubr (&Sget_buffer_window
);
6061 defsubr (&Sdelete_other_windows
);
6062 defsubr (&Sdelete_windows_on
);
6063 defsubr (&Sreplace_buffer_in_windows
);
6064 defsubr (&Sdelete_window
);
6065 defsubr (&Sset_window_buffer
);
6066 defsubr (&Sselect_window
);
6067 defsubr (&Sspecial_display_p
);
6068 defsubr (&Ssame_window_p
);
6069 defsubr (&Sdisplay_buffer
);
6070 defsubr (&Ssplit_window
);
6071 defsubr (&Senlarge_window
);
6072 defsubr (&Sshrink_window
);
6073 defsubr (&Sscroll_up
);
6074 defsubr (&Sscroll_down
);
6075 defsubr (&Sscroll_left
);
6076 defsubr (&Sscroll_right
);
6077 defsubr (&Sother_window_for_scrolling
);
6078 defsubr (&Sscroll_other_window
);
6079 defsubr (&Sminibuffer_selected_window
);
6080 defsubr (&Srecenter
);
6081 defsubr (&Swindow_text_height
);
6082 defsubr (&Smove_to_window_line
);
6083 defsubr (&Swindow_configuration_p
);
6084 defsubr (&Swindow_configuration_frame
);
6085 defsubr (&Sset_window_configuration
);
6086 defsubr (&Scurrent_window_configuration
);
6087 defsubr (&Ssave_window_excursion
);
6088 defsubr (&Sset_window_margins
);
6089 defsubr (&Swindow_margins
);
6090 defsubr (&Swindow_vscroll
);
6091 defsubr (&Sset_window_vscroll
);
6092 defsubr (&Scompare_window_configurations
);
6093 defsubr (&Swindow_list
);
6099 initial_define_key (control_x_map
, '1', "delete-other-windows");
6100 initial_define_key (control_x_map
, '2', "split-window");
6101 initial_define_key (control_x_map
, '0', "delete-window");
6102 initial_define_key (control_x_map
, 'o', "other-window");
6103 initial_define_key (control_x_map
, '^', "enlarge-window");
6104 initial_define_key (control_x_map
, '<', "scroll-left");
6105 initial_define_key (control_x_map
, '>', "scroll-right");
6107 initial_define_key (global_map
, Ctl ('V'), "scroll-up");
6108 initial_define_key (meta_map
, Ctl ('V'), "scroll-other-window");
6109 initial_define_key (meta_map
, 'v', "scroll-down");
6111 initial_define_key (global_map
, Ctl('L'), "recenter");
6112 initial_define_key (meta_map
, 'r', "move-to-window-line");