* macterm.c (XTread_socket): Call KeyTranslate for control and
[bpt/emacs.git] / src / window.c
CommitLineData
7ab12479
JB
1/* Window creation, deletion and examination for GNU Emacs.
2 Does not include redisplay.
49b996e7 3 Copyright (C) 1985,86,87,93,94,95,96,97,1998,2000, 2001, 2002
6529ed87 4 Free Software Foundation, Inc.
7ab12479
JB
5
6This file is part of GNU Emacs.
7
8GNU Emacs is free software; you can redistribute it and/or modify
9it under the terms of the GNU General Public License as published by
1113d9db 10the Free Software Foundation; either version 2, or (at your option)
7ab12479
JB
11any later version.
12
13GNU Emacs is distributed in the hope that it will be useful,
14but WITHOUT ANY WARRANTY; without even the implied warranty of
15MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
16GNU General Public License for more details.
17
18You should have received a copy of the GNU General Public License
19along with GNU Emacs; see the file COPYING. If not, write to
3b7ad313
EN
20the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
21Boston, MA 02111-1307, USA. */
7ab12479 22
18160b98 23#include <config.h>
7ab12479
JB
24#include "lisp.h"
25#include "buffer.h"
3e4731a3 26#include "keyboard.h"
e35f6ff7 27#include "keymap.h"
44fa5b1e 28#include "frame.h"
7ab12479
JB
29#include "window.h"
30#include "commands.h"
31#include "indent.h"
32#include "termchar.h"
33#include "disptab.h"
dfcf069d 34#include "dispextern.h"
5500c422
GM
35#include "blockinput.h"
36#include "intervals.h"
37
6d55d620 38#ifdef HAVE_X_WINDOWS
dfcf069d 39#include "xterm.h"
5500c422 40#endif /* HAVE_X_WINDOWS */
8f23f280
AI
41#ifdef WINDOWSNT
42#include "w32term.h"
43#endif
1e3c8885
EZ
44#ifdef MSDOS
45#include "msdos.h"
46#endif
e0f712ba 47#ifdef MAC_OS
1a578e9b
AC
48#include "macterm.h"
49#endif
5500c422 50
7442878f
GM
51/* Values returned from coordinates_in_window. */
52
53enum window_part
54{
55 ON_NOTHING,
56 ON_TEXT,
57 ON_MODE_LINE,
58 ON_VERTICAL_BORDER,
59 ON_HEADER_LINE,
60 ON_LEFT_FRINGE,
49b996e7
GM
61 ON_RIGHT_FRINGE,
62 ON_LEFT_MARGIN,
63 ON_RIGHT_MARGIN
7442878f
GM
64};
65
5500c422 66
3f8ab7bd 67Lisp_Object Qwindowp, Qwindow_live_p, Qwindow_configuration_p;
90647b07 68Lisp_Object Qwindow_size_fixed;
49b996e7 69extern Lisp_Object Qleft_margin, Qright_margin;
233a4a2c 70extern Lisp_Object Qheight, Qwidth;
7ab12479 71
521b203e 72static int displayed_window_lines P_ ((struct window *));
5e14b1fc 73static struct window *decode_window P_ ((Lisp_Object));
b7354ddf 74static Lisp_Object select_window_1 P_ ((Lisp_Object, int));
5500c422
GM
75static int count_windows P_ ((struct window *));
76static int get_leaf_windows P_ ((struct window *, struct window **, int));
77static void window_scroll P_ ((Lisp_Object, int, int, int));
78static void window_scroll_pixel_based P_ ((Lisp_Object, int, int, int));
79static void window_scroll_line_based P_ ((Lisp_Object, int, int, int));
233a4a2c 80static int window_min_size_1 P_ ((struct window *, int));
f984d4fc 81static int window_min_size P_ ((struct window *, int, int, int *));
233a4a2c 82static void size_window P_ ((Lisp_Object, int, int, int));
f95464e4 83static int freeze_window_start P_ ((struct window *, void *));
f984d4fc 84static int window_fixed_size_p P_ ((struct window *, int, int));
f95284d2 85static void enlarge_window P_ ((Lisp_Object, int, int, int));
67492200 86static Lisp_Object window_list P_ ((void));
f95464e4 87static int add_window_to_list P_ ((struct window *, void *));
118ea242
GM
88static int candidate_window_p P_ ((Lisp_Object, Lisp_Object, Lisp_Object,
89 Lisp_Object));
67492200
GM
90static Lisp_Object next_window P_ ((Lisp_Object, Lisp_Object,
91 Lisp_Object, int));
92static void decode_next_window_args P_ ((Lisp_Object *, Lisp_Object *,
93 Lisp_Object *));
f95464e4
GM
94static int foreach_window_1 P_ ((struct window *,
95 int (* fn) (struct window *, void *),
96 void *));
cd2904bd 97static Lisp_Object window_list_1 P_ ((Lisp_Object, Lisp_Object, Lisp_Object));
b7354ddf 98
b0b7ed0e
GM
99/* The value of `window-size-fixed'. */
100
101int window_size_fixed;
102
7ab12479
JB
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.
107
108 This value is always the same as
44fa5b1e 109 FRAME_SELECTED_WINDOW (selected_frame). */
7ab12479
JB
110
111Lisp_Object selected_window;
112
67492200
GM
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. */
116
117Lisp_Object Vwindow_list;
118
5500c422
GM
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
7ab12479 122 the selected window. */
5500c422 123
7ab12479
JB
124Lisp_Object minibuf_window;
125
3f49fddc
KS
126/* Non-nil means it is the window whose mode line should be
127 shown as the selected window when the minibuffer is selected. */
128
3dbab091 129Lisp_Object minibuf_selected_window;
3f49fddc 130
7ab12479 131/* Non-nil means it is the window for C-M-v to scroll
5500c422
GM
132 when the mini-buffer is selected. */
133
7ab12479
JB
134Lisp_Object Vminibuf_scroll_window;
135
136/* Non-nil means this is the buffer whose window C-M-v should scroll. */
5500c422 137
7ab12479
JB
138Lisp_Object Vother_window_scroll_buffer;
139
7ab12479 140/* Non-nil means it's function to call to display temp buffers. */
5500c422 141
7ab12479
JB
142Lisp_Object Vtemp_buffer_show_function;
143
cc91894c
KS
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. */
147int mode_line_in_non_selected_windows;
26124d5e 148
7ab12479 149/* If a window gets smaller than either of these, it is removed. */
5500c422 150
31ade731
SM
151EMACS_INT window_min_height;
152EMACS_INT window_min_width;
7ab12479
JB
153
154/* Nonzero implies Fdisplay_buffer should create windows. */
5500c422 155
7ab12479
JB
156int pop_up_windows;
157
44fa5b1e 158/* Nonzero implies make new frames for Fdisplay_buffer. */
5500c422 159
44fa5b1e 160int pop_up_frames;
7ab12479 161
9c3da604
GM
162/* Nonzero means reuse existing frames for displaying buffers. */
163
164int display_buffer_reuse_frames;
165
7ab12479 166/* Non-nil means use this function instead of default */
5500c422 167
44fa5b1e 168Lisp_Object Vpop_up_frame_function;
7ab12479
JB
169
170/* Function to call to handle Fdisplay_buffer. */
5500c422 171
7ab12479
JB
172Lisp_Object Vdisplay_buffer_function;
173
6529ed87
GM
174/* Non-nil means that Fdisplay_buffer should even the heights of windows. */
175
176Lisp_Object Veven_window_heights;
177
a90712c2 178/* List of buffer *names* for buffers that should have their own frames. */
5500c422 179
a90712c2
RS
180Lisp_Object Vspecial_display_buffer_names;
181
182/* List of regexps for buffer names that should have their own frames. */
5500c422 183
a90712c2
RS
184Lisp_Object Vspecial_display_regexps;
185
186/* Function to pop up a special frame. */
5500c422 187
a90712c2
RS
188Lisp_Object Vspecial_display_function;
189
855d8627 190/* List of buffer *names* for buffers to appear in selected window. */
5500c422 191
855d8627
RS
192Lisp_Object Vsame_window_buffer_names;
193
194/* List of regexps for buffer names to appear in selected window. */
5500c422 195
855d8627
RS
196Lisp_Object Vsame_window_regexps;
197
a58ec57d 198/* Hook run at end of temp_output_buffer_show. */
5500c422 199
a58ec57d
RS
200Lisp_Object Qtemp_buffer_show_hook;
201
37962e60 202/* Fdisplay_buffer always splits the largest window
7ab12479 203 if that window is more than this high. */
5500c422 204
31ade731 205EMACS_INT split_height_threshold;
7ab12479
JB
206
207/* Number of lines of continuity in scrolling by screenfuls. */
5500c422 208
31ade731 209EMACS_INT next_screen_context_lines;
7ab12479
JB
210
211/* Incremented for each window created. */
5500c422 212
7ab12479
JB
213static int sequence_number;
214
5b03d3c0 215/* Nonzero after init_window_once has finished. */
5500c422 216
5b03d3c0
RS
217static int window_initialized;
218
543f5fb1 219/* Hook to run when window config changes. */
5500c422 220
543f5fb1
RS
221Lisp_Object Qwindow_configuration_change_hook;
222Lisp_Object Vwindow_configuration_change_hook;
223
9317a85d
RS
224/* Nonzero means scroll commands try to put point
225 at the same screen height as previously. */
9317a85d 226
5500c422
GM
227Lisp_Object Vscroll_preserve_screen_position;
228
229#if 0 /* This isn't used anywhere. */
91a78190 230/* Nonzero means we can split a frame even if it is "unsplittable". */
eddd51c2 231static int inhibit_frame_unsplittable;
5500c422 232#endif /* 0 */
7d601aaa 233
31ade731 234extern EMACS_INT scroll_margin;
9317a85d 235
dba06815 236extern Lisp_Object Qwindow_scroll_functions, Vwindow_scroll_functions;
7ab12479
JB
237\f
238DEFUN ("windowp", Fwindowp, Swindowp, 1, 1, 0,
fdb82f93
PJ
239 doc: /* Returns t if OBJECT is a window. */)
240 (object)
413430c5 241 Lisp_Object object;
7ab12479 242{
413430c5 243 return WINDOWP (object) ? Qt : Qnil;
7ab12479
JB
244}
245
806b4d9b 246DEFUN ("window-live-p", Fwindow_live_p, Swindow_live_p, 1, 1, 0,
fdb82f93 247 doc: /* Returns t if OBJECT is a window which is currently visible. */)
413430c5
EN
248 (object)
249 Lisp_Object object;
605be8af 250{
9e571f49 251 return WINDOW_LIVE_P (object) ? Qt : Qnil;
605be8af
JB
252}
253
7ab12479
JB
254Lisp_Object
255make_window ()
256{
cffec418 257 Lisp_Object val;
7ab12479 258 register struct window *p;
cffec418 259
26605be9 260 p = allocate_window ();
d834a2e9
KH
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);
ea68264b 267 XSETFASTINT (p->min_hscroll, 0);
f984d4fc 268 p->orig_top = p->orig_height = Qnil;
7ab12479
JB
269 p->start = Fmake_marker ();
270 p->pointm = Fmake_marker ();
d834a2e9 271 XSETFASTINT (p->use_time, 0);
44fa5b1e 272 p->frame = Qnil;
7ab12479
JB
273 p->display_table = Qnil;
274 p->dedicated = Qnil;
5500c422
GM
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;
4b26579e 281 p->phys_cursor_width = -1;
5500c422
GM
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;
286 p->vscroll = 0;
cffec418 287 XSETWINDOW (val, p);
5500c422 288 XSETFASTINT (p->last_point, 0);
7bbb5782 289 p->frozen_window_start_p = 0;
6b448755
RS
290 p->height_fixed_p = 0;
291 p->last_cursor_off_p = p->cursor_off_p = 0;
67492200
GM
292
293 Vwindow_list = Qnil;
7ab12479
JB
294 return val;
295}
296
297DEFUN ("selected-window", Fselected_window, Sselected_window, 0, 0, 0,
fdb82f93
PJ
298 doc: /* Return the window that the cursor now appears in and commands apply to. */)
299 ()
7ab12479
JB
300{
301 return selected_window;
302}
303
83762ba4 304DEFUN ("minibuffer-window", Fminibuffer_window, Sminibuffer_window, 0, 1, 0,
fdb82f93
PJ
305 doc: /* Return the window used now for minibuffers.
306If the optional argument FRAME is specified, return the minibuffer window
307used by that frame. */)
308 (frame)
83762ba4 309 Lisp_Object frame;
7ab12479 310{
83762ba4 311 if (NILP (frame))
1ae1a37d 312 frame = selected_frame;
b7826503 313 CHECK_LIVE_FRAME (frame);
83762ba4 314 return FRAME_MINIBUF_WINDOW (XFRAME (frame));
7ab12479
JB
315}
316
605be8af 317DEFUN ("window-minibuffer-p", Fwindow_minibuffer_p, Swindow_minibuffer_p, 0, 1, 0,
fdb82f93
PJ
318 doc: /* Returns non-nil if WINDOW is a minibuffer window. */)
319 (window)
7ab12479
JB
320 Lisp_Object window;
321{
322 struct window *w = decode_window (window);
53bb6b99 323 return MINI_WINDOW_P (w) ? Qt : Qnil;
81e4d465
MB
324}
325
326
7ab12479 327DEFUN ("pos-visible-in-window-p", Fpos_visible_in_window_p,
fdb82f93
PJ
328 Spos_visible_in_window_p, 0, 3, 0,
329 doc: /* Return t if position POS is currently on the frame in WINDOW.
330Return nil if that position is scrolled vertically out of view.
331If a character is only partially visible, nil is returned, unless the
332optional argument PARTIALLY is non-nil.
333POS defaults to point in WINDOW; WINDOW defaults to the selected window. */)
334 (pos, window, partially)
5cdb3cf3 335 Lisp_Object pos, window, partially;
7ab12479
JB
336{
337 register struct window *w;
7ab12479
JB
338 register int posint;
339 register struct buffer *buf;
9a132b1f
GM
340 struct text_pos top;
341 Lisp_Object in_window;
53bb6b99 342 int fully_p;
7ab12479 343
f28b75a9
MB
344 w = decode_window (window);
345 buf = XBUFFER (w->buffer);
346 SET_TEXT_POS_FROM_MARKER (top, w->start);
347
348 if (!NILP (pos))
7ab12479 349 {
b7826503 350 CHECK_NUMBER_COERCE_MARKER (pos);
7ab12479
JB
351 posint = XINT (pos);
352 }
f28b75a9
MB
353 else if (w == XWINDOW (selected_window))
354 posint = PT;
355 else
356 posint = XMARKER (w->pointm)->charpos;
7ab12479 357
53bb6b99 358 /* If position is above window start, it's not visible. */
5500c422 359 if (posint < CHARPOS (top))
9a132b1f
GM
360 in_window = Qnil;
361 else if (XFASTINT (w->last_modified) >= BUF_MODIFF (buf)
81e4d465
MB
362 && XFASTINT (w->last_overlay_modified) >= BUF_OVERLAY_MODIFF (buf)
363 && posint < BUF_Z (buf) - XFASTINT (w->window_end_pos))
81e4d465 364 {
53bb6b99
GM
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. */
5cdb3cf3 369 if (NILP (partially))
53bb6b99 370 {
5cdb3cf3 371 pos_visible_p (w, posint, &fully_p, NILP (partially));
53bb6b99
GM
372 in_window = fully_p ? Qt : Qnil;
373 }
81e4d465 374 else
53bb6b99 375 in_window = Qt;
81e4d465 376 }
9a132b1f
GM
377 else if (posint > BUF_ZV (buf))
378 in_window = Qnil;
379 else if (CHARPOS (top) < BUF_BEGV (buf) || CHARPOS (top) > BUF_ZV (buf))
380 /* If window start is out of range, do something reasonable. */
381 in_window = Qnil;
7ab12479
JB
382 else
383 {
5cdb3cf3
MB
384 if (pos_visible_p (w, posint, &fully_p, NILP (partially)))
385 in_window = !NILP (partially) || fully_p ? Qt : Qnil;
81e4d465
MB
386 else
387 in_window = Qnil;
7ab12479 388 }
9a132b1f
GM
389
390 return in_window;
7ab12479 391}
53bb6b99 392
7ab12479
JB
393\f
394static struct window *
395decode_window (window)
396 register Lisp_Object window;
397{
265a9e55 398 if (NILP (window))
7ab12479
JB
399 return XWINDOW (selected_window);
400
b7826503 401 CHECK_LIVE_WINDOW (window);
7ab12479
JB
402 return XWINDOW (window);
403}
404
405DEFUN ("window-buffer", Fwindow_buffer, Swindow_buffer, 0, 1, 0,
fdb82f93
PJ
406 doc: /* Return the buffer that WINDOW is displaying. */)
407 (window)
7ab12479
JB
408 Lisp_Object window;
409{
410 return decode_window (window)->buffer;
411}
412
413DEFUN ("window-height", Fwindow_height, Swindow_height, 0, 1, 0,
fdb82f93
PJ
414 doc: /* Return the number of lines in WINDOW (including its mode line). */)
415 (window)
7ab12479
JB
416 Lisp_Object window;
417{
418 return decode_window (window)->height;
419}
420
421DEFUN ("window-width", Fwindow_width, Swindow_width, 0, 1, 0,
fdb82f93
PJ
422 doc: /* Return the number of display columns in WINDOW.
423This is the width that is usable columns available for text in WINDOW.
424If you want to find out how many columns WINDOW takes up,
425use (let ((edges (window-edges))) (- (nth 2 edges) (nth 0 edges))). */)
426 (window)
7ab12479
JB
427 Lisp_Object window;
428{
ee61d94e 429 return make_number (window_internal_width (decode_window (window)));
7ab12479
JB
430}
431
432DEFUN ("window-hscroll", Fwindow_hscroll, Swindow_hscroll, 0, 1, 0,
fdb82f93
PJ
433 doc: /* Return the number of columns by which WINDOW is scrolled from left margin. */)
434 (window)
7ab12479
JB
435 Lisp_Object window;
436{
437 return decode_window (window)->hscroll;
438}
439
440DEFUN ("set-window-hscroll", Fset_window_hscroll, Sset_window_hscroll, 2, 2, 0,
fdb82f93 441 doc: /* Set number of columns WINDOW is scrolled from left margin to NCOL.
ebadb1e4
EZ
442NCOL should be zero or positive.
443
444Note that if `automatic-hscrolling' is non-nil, you cannot scroll the
445window so that the location of point becomes invisible. */)
fdb82f93 446 (window, ncol)
ea68264b 447 Lisp_Object window, ncol;
7ab12479 448{
ea68264b
GM
449 struct window *w = decode_window (window);
450 int hscroll;
7ab12479 451
b7826503 452 CHECK_NUMBER (ncol);
ea68264b
GM
453 hscroll = max (0, XINT (ncol));
454
455 /* Prevent redisplay shortcuts when changing the hscroll. */
456 if (XINT (w->hscroll) != hscroll)
b1599b4c 457 XBUFFER (w->buffer)->prevent_redisplay_optimizations_p = 1;
ea68264b 458
c67fa410 459 w->hscroll = make_number (hscroll);
7ab12479
JB
460 return ncol;
461}
462
190eb263
RS
463DEFUN ("window-redisplay-end-trigger", Fwindow_redisplay_end_trigger,
464 Swindow_redisplay_end_trigger, 0, 1, 0,
fdb82f93
PJ
465 doc: /* Return WINDOW's redisplay end trigger value.
466See `set-window-redisplay-end-trigger' for more information. */)
467 (window)
190eb263
RS
468 Lisp_Object window;
469{
470 return decode_window (window)->redisplay_end_trigger;
471}
472
473DEFUN ("set-window-redisplay-end-trigger", Fset_window_redisplay_end_trigger,
474 Sset_window_redisplay_end_trigger, 2, 2, 0,
fdb82f93
PJ
475 doc: /* Set WINDOW's redisplay end trigger value to VALUE.
476VALUE should be a buffer position (typically a marker) or nil.
477If it is a buffer position, then if redisplay in WINDOW reaches a position
478beyond VALUE, the functions in `redisplay-end-trigger-functions' are called
479with two arguments: WINDOW, and the end trigger value.
480Afterwards the end-trigger value is reset to nil. */)
481 (window, value)
190eb263
RS
482 register Lisp_Object window, value;
483{
484 register struct window *w;
485
486 w = decode_window (window);
487 w->redisplay_end_trigger = value;
488 return value;
489}
490
7ab12479 491DEFUN ("window-edges", Fwindow_edges, Swindow_edges, 0, 1, 0,
fdb82f93
PJ
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.
494RIGHT is one more than the rightmost column used by WINDOW,
495and BOTTOM is one more than the bottommost row used by WINDOW
496 and its mode-line. */)
497 (window)
7ab12479
JB
498 Lisp_Object window;
499{
500 register struct window *w = decode_window (window);
501
502 return Fcons (w->left, Fcons (w->top,
111e5992 503 Fcons (make_number (WINDOW_RIGHT_EDGE (w)),
7ab12479
JB
504 Fcons (make_number (XFASTINT (w->top)
505 + XFASTINT (w->height)),
506 Qnil))));
507}
508
fbad6f9a 509/* Test if the character at column *X, row *Y is within window W.
d5783c40
JB
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
513 of the window, and
514 return 1;
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,
5500c422
GM
517 return 3.
518 if it is on the window's top line, return 4;
81d189fd 519 if it is in left or right fringe of the window,
49b996e7
GM
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.
5500c422
GM
523
524 X and Y are frame relative pixel coordinates. */
525
7442878f 526static enum window_part
d5783c40
JB
527coordinates_in_window (w, x, y)
528 register struct window *w;
529 register int *x, *y;
530{
1cb794ba
GM
531 /* Let's make this a global enum later, instead of using numbers
532 everywhere. */
b0228ace
GM
533 struct frame *f = XFRAME (WINDOW_FRAME (w));
534 int left_x, right_x, top_y, bottom_y;
7442878f 535 enum window_part part;
15c565ac 536 int ux = CANON_X_UNIT (f);
b0228ace
GM
537 int x0 = XFASTINT (w->left) * ux;
538 int x1 = x0 + XFASTINT (w->width) * ux;
dad75588
GM
539 /* The width of the area where the vertical line can be dragged.
540 (Between mode lines for instance. */
541 int grabbable_width = ux;
ac431108 542 int lmargin_width = 0, rmargin_width = 0;
b0228ace
GM
543
544 if (*x < x0 || *x >= x1)
545 return ON_NOTHING;
546
466539bc
EZ
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. */
5500c422
GM
549 if (w->pseudo_window_p)
550 {
551 left_x = 0;
b0228ace 552 right_x = XFASTINT (w->width) * CANON_X_UNIT (f) - 1;
5500c422
GM
553 top_y = WINDOW_DISPLAY_TOP_EDGE_PIXEL_Y (w);
554 bottom_y = WINDOW_DISPLAY_BOTTOM_EDGE_PIXEL_Y (w);
555 }
556 else
557 {
2add4349 558 left_x = (WINDOW_DISPLAY_LEFT_EDGE_PIXEL_X (w)
50df5e5a 559 - FRAME_INTERNAL_BORDER_WIDTH_SAFE (f));
466539bc 560 right_x = WINDOW_DISPLAY_RIGHT_EDGE_PIXEL_X (w) - 1;
2add4349 561 top_y = (WINDOW_DISPLAY_TOP_EDGE_PIXEL_Y (w)
50df5e5a 562 - FRAME_INTERNAL_BORDER_WIDTH_SAFE (f));
5500c422
GM
563 bottom_y = WINDOW_DISPLAY_BOTTOM_EDGE_PIXEL_Y (w);
564 }
37962e60 565
1cb794ba
GM
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
570 scroll bars. */
b0228ace 571
1cb794ba 572 if (WINDOW_WANTS_MODELINE_P (w)
b0228ace
GM
573 && *y >= bottom_y - CURRENT_MODE_LINE_HEIGHT (w)
574 && *y < bottom_y)
1cb794ba 575 {
b0228ace
GM
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. */
580 part = ON_MODE_LINE;
581
582 if (FRAME_HAS_VERTICAL_SCROLL_BARS_ON_LEFT (f))
583 {
dad75588 584 if (abs (*x - x0) < grabbable_width)
b0228ace
GM
585 part = ON_VERTICAL_BORDER;
586 }
dad75588 587 else if (!WINDOW_RIGHTMOST_P (w) && abs (*x - x1) < grabbable_width)
b0228ace 588 part = ON_VERTICAL_BORDER;
1cb794ba 589 }
b0228ace
GM
590 else if (WINDOW_WANTS_HEADER_LINE_P (w)
591 && *y < top_y + CURRENT_HEADER_LINE_HEIGHT (w)
592 && *y >= top_y)
593 {
594 part = ON_HEADER_LINE;
595
596 if (FRAME_HAS_VERTICAL_SCROLL_BARS_ON_LEFT (f))
597 {
dad75588 598 if (abs (*x - x0) < grabbable_width)
b0228ace
GM
599 part = ON_VERTICAL_BORDER;
600 }
dad75588 601 else if (!WINDOW_RIGHTMOST_P (w) && abs (*x - x1) < grabbable_width)
b0228ace
GM
602 part = ON_VERTICAL_BORDER;
603 }
604 /* Outside anything interesting? */
605 else if (*y < top_y
606 || *y >= bottom_y
607 || *x < (left_x
c13418ac 608 - FRAME_LEFT_FRINGE_WIDTH (f)
b0228ace 609 - FRAME_LEFT_SCROLL_BAR_WIDTH (f) * ux)
7442878f 610 || *x > (right_x
c13418ac 611 + FRAME_RIGHT_FRINGE_WIDTH (f)
7442878f 612 + FRAME_RIGHT_SCROLL_BAR_WIDTH (f) * ux))
1cb794ba 613 {
b0228ace 614 part = ON_NOTHING;
1cb794ba 615 }
b0228ace 616 else if (FRAME_WINDOW_P (f))
fbad6f9a 617 {
447e9da0
GM
618 if (!w->pseudo_window_p
619 && !FRAME_HAS_VERTICAL_SCROLL_BARS (f)
620 && !WINDOW_RIGHTMOST_P (w)
c13418ac 621 && (abs (*x - right_x - FRAME_RIGHT_FRINGE_WIDTH (f)) < grabbable_width))
b0228ace
GM
622 {
623 part = ON_VERTICAL_BORDER;
624 }
625 else if (*x < left_x || *x > right_x)
447e9da0 626 {
81d189fd 627 /* Other lines than the mode line don't include fringes and
447e9da0 628 scroll bars on the left. */
fbad6f9a 629
447e9da0
GM
630 /* Convert X and Y to window-relative pixel coordinates. */
631 *x -= left_x;
632 *y -= top_y;
b0228ace
GM
633 part = *x < left_x ? ON_LEFT_FRINGE : ON_RIGHT_FRINGE;
634 }
635 else
636 {
ac431108
EZ
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. */
640 if (lmargin_width
641 && *x <= window_box_right (w, LEFT_MARGIN_AREA))
49b996e7 642 part = ON_LEFT_MARGIN;
ac431108
EZ
643 else if (rmargin_width
644 && *x >= window_box_left (w, RIGHT_MARGIN_AREA))
49b996e7
GM
645 part = ON_RIGHT_MARGIN;
646 else
647 {
648 part = ON_TEXT;
649 *x -= left_x;
650 *y -= top_y;
651 }
447e9da0 652 }
fbad6f9a 653 }
447e9da0
GM
654 else
655 {
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)
659 {
81d189fd 660 /* Other lines than the mode line don't include fringes and
447e9da0
GM
661 scroll bars on the left. */
662
663 /* Convert X and Y to window-relative pixel coordinates. */
664 *x -= left_x;
665 *y -= top_y;
b0228ace 666 part = *x < left_x ? ON_LEFT_FRINGE : ON_RIGHT_FRINGE;
447e9da0 667 }
447e9da0 668 /* Here, too, "*x > right_x" is because of character terminals. */
b0228ace
GM
669 else if (!w->pseudo_window_p
670 && !WINDOW_RIGHTMOST_P (w)
671 && *x > right_x - ux)
672 {
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;
676 }
677 else
678 {
ac431108
EZ
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. */
683 if (lmargin_width
684 && *x <= window_box_right (w, LEFT_MARGIN_AREA))
49b996e7 685 part = ON_LEFT_MARGIN;
ac431108
EZ
686 else if (rmargin_width
687 && *x >= window_box_left (w, RIGHT_MARGIN_AREA))
49b996e7
GM
688 part = ON_RIGHT_MARGIN;
689 else
690 {
691 part = ON_TEXT;
692 /* Convert X and Y to window-relative pixel coordinates. */
693 *x -= left_x;
694 *y -= top_y;
695 }
b0228ace 696 }
447e9da0 697 }
b0228ace
GM
698
699 return part;
d5783c40
JB
700}
701
b0228ace 702
d5783c40 703DEFUN ("coordinates-in-window-p", Fcoordinates_in_window_p,
fdb82f93
PJ
704 Scoordinates_in_window_p, 2, 2, 0,
705 doc: /* Return non-nil if COORDINATES are in WINDOW.
706COORDINATES is a cons of the form (X . Y), X and Y being distances
707measured in characters from the upper-left corner of the frame.
708\(0 . 0) denotes the character in the upper left corner of the
709frame.
710If COORDINATES are in the text portion of WINDOW,
711 the coordinates relative to the window are returned.
712If they are in the mode line of WINDOW, `mode-line' is returned.
713If they are in the top mode line of WINDOW, `header-line' is returned.
81d189fd
KS
714If they are in the left fringe of WINDOW, `left-fringe' is returned.
715If they are in the right fringe of WINDOW, `right-fringe' is returned.
fdb82f93 716If they are on the border between WINDOW and its right sibling,
49b996e7
GM
717 `vertical-line' is returned.
718If they are in the windows's left or right marginal areas, `left-margin'\n\
719 or `right-margin' is returned. */)
fdb82f93 720 (coordinates, window)
d5783c40
JB
721 register Lisp_Object coordinates, window;
722{
5500c422
GM
723 struct window *w;
724 struct frame *f;
d5783c40 725 int x, y;
5500c422 726 Lisp_Object lx, ly;
d5783c40 727
b7826503 728 CHECK_LIVE_WINDOW (window);
5500c422
GM
729 w = XWINDOW (window);
730 f = XFRAME (w->frame);
b7826503 731 CHECK_CONS (coordinates);
5500c422
GM
732 lx = Fcar (coordinates);
733 ly = Fcdr (coordinates);
b7826503
PJ
734 CHECK_NUMBER_OR_FLOAT (lx);
735 CHECK_NUMBER_OR_FLOAT (ly);
5500c422
GM
736 x = PIXEL_X_FROM_CANON_X (f, lx);
737 y = PIXEL_Y_FROM_CANON_Y (f, ly);
738
739 switch (coordinates_in_window (w, &x, &y))
d5783c40 740 {
7442878f 741 case ON_NOTHING:
d5783c40
JB
742 return Qnil;
743
7442878f
GM
744 case ON_TEXT:
745 /* X and Y are now window relative pixel coordinates. Convert
746 them to canonical char units before returning them. */
5500c422
GM
747 return Fcons (CANON_X_FROM_PIXEL_X (f, x),
748 CANON_Y_FROM_PIXEL_Y (f, y));
d5783c40 749
7442878f 750 case ON_MODE_LINE:
d5783c40 751 return Qmode_line;
37962e60 752
7442878f 753 case ON_VERTICAL_BORDER:
e5d77022 754 return Qvertical_line;
d5783c40 755
7442878f 756 case ON_HEADER_LINE:
045dee35 757 return Qheader_line;
5500c422 758
7442878f
GM
759 case ON_LEFT_FRINGE:
760 return Qleft_fringe;
fbad6f9a 761
7442878f
GM
762 case ON_RIGHT_FRINGE:
763 return Qright_fringe;
fbad6f9a 764
49b996e7
GM
765 case ON_LEFT_MARGIN:
766 return Qleft_margin;
767
768 case ON_RIGHT_MARGIN:
769 return Qright_margin;
770
d5783c40
JB
771 default:
772 abort ();
773 }
774}
775
67492200
GM
776
777/* Callback for foreach_window, used in window_from_coordinates.
f95464e4
GM
778 Check if window W contains coordinates specified by USER_DATA which
779 is actually a pointer to a struct check_window_data CW.
780
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
5372262f 783 *CW->part the part of the window under coordinates *X,*Y. Return
f95464e4
GM
784 zero from this function to stop iterating over windows. */
785
786struct check_window_data
787{
788 Lisp_Object *window;
789 int *x, *y, *part;
790};
67492200
GM
791
792static int
f95464e4 793check_window_containing (w, user_data)
67492200 794 struct window *w;
f95464e4 795 void *user_data;
67492200 796{
f95464e4 797 struct check_window_data *cw = (struct check_window_data *) user_data;
7442878f
GM
798 enum window_part found;
799 int continue_p = 1;
67492200 800
f95464e4 801 found = coordinates_in_window (w, cw->x, cw->y);
7442878f 802 if (found != ON_NOTHING)
67492200 803 {
f95464e4
GM
804 *cw->part = found - 1;
805 XSETWINDOW (*cw->window, w);
7442878f 806 continue_p = 0;
67492200
GM
807 }
808
7442878f 809 return continue_p;
67492200
GM
810}
811
812
5500c422
GM
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
67492200
GM
818 unmodified. TOOL_BAR_P non-zero means detect tool-bar windows.
819
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
827 case. */
5500c422 828
7ab12479 829Lisp_Object
67492200
GM
830window_from_coordinates (f, x, y, part, tool_bar_p)
831 struct frame *f;
7ab12479 832 int x, y;
d5783c40 833 int *part;
9ea173e8 834 int tool_bar_p;
7ab12479 835{
67492200 836 Lisp_Object window;
f95464e4 837 struct check_window_data cw;
5500c422 838
67492200 839 window = Qnil;
f95464e4
GM
840 cw.window = &window, cw.x = &x, cw.y = &y; cw.part = part;
841 foreach_window (f, check_window_containing, &cw);
67492200
GM
842
843 /* If not found above, see if it's in the tool bar window, if a tool
844 bar exists. */
845 if (NILP (window)
846 && tool_bar_p
847 && WINDOWP (f->tool_bar_window)
848 && XINT (XWINDOW (f->tool_bar_window)->height) > 0
7442878f
GM
849 && (coordinates_in_window (XWINDOW (f->tool_bar_window), &x, &y)
850 != ON_NOTHING))
5500c422
GM
851 {
852 *part = 0;
67492200 853 window = f->tool_bar_window;
5500c422 854 }
37962e60 855
67492200 856 return window;
7ab12479
JB
857}
858
ab17c3f2 859DEFUN ("window-at", Fwindow_at, Swindow_at, 2, 3, 0,
fdb82f93
PJ
860 doc: /* Return window containing coordinates X and Y on FRAME.
861If omitted, FRAME defaults to the currently selected frame.
862The top left corner of the frame is considered to be row 0,
863column 0. */)
864 (x, y, frame)
865 Lisp_Object x, y, frame;
7ab12479
JB
866{
867 int part;
5500c422 868 struct frame *f;
7ab12479 869
44fa5b1e 870 if (NILP (frame))
1ae1a37d 871 frame = selected_frame;
b7826503 872 CHECK_LIVE_FRAME (frame);
5500c422 873 f = XFRAME (frame);
7ab12479 874
5500c422 875 /* Check that arguments are integers or floats. */
b7826503
PJ
876 CHECK_NUMBER_OR_FLOAT (x);
877 CHECK_NUMBER_OR_FLOAT (y);
5500c422
GM
878
879 return window_from_coordinates (f,
880 PIXEL_X_FROM_CANON_X (f, x),
881 PIXEL_Y_FROM_CANON_Y (f, y),
882 &part, 0);
7ab12479
JB
883}
884
885DEFUN ("window-point", Fwindow_point, Swindow_point, 0, 1, 0,
fdb82f93
PJ
886 doc: /* Return current value of point in WINDOW.
887For a nonselected window, this is the value point would have
888if that window were selected.
889
890Note that, when WINDOW is the selected window and its buffer
891is also currently selected, the value returned is the same as (point).
892It would be more strictly correct to return the `top-level' value
893of point, outside of any save-excursion forms.
894But that is hard to define. */)
895 (window)
7ab12479
JB
896 Lisp_Object window;
897{
898 register struct window *w = decode_window (window);
899
900 if (w == XWINDOW (selected_window)
901 && current_buffer == XBUFFER (w->buffer))
902 return Fpoint ();
903 return Fmarker_position (w->pointm);
904}
905
906DEFUN ("window-start", Fwindow_start, Swindow_start, 0, 1, 0,
fdb82f93
PJ
907 doc: /* Return position at which display currently starts in WINDOW.
908This is updated by redisplay or by calling `set-window-start'. */)
909 (window)
7ab12479
JB
910 Lisp_Object window;
911{
912 return Fmarker_position (decode_window (window)->start);
913}
914
8646118f
RS
915/* This is text temporarily removed from the doc string below.
916
fdb82f93
PJ
917This function returns nil if the position is not currently known.
918That happens when redisplay is preempted and doesn't finish.
919If in that case you want to compute where the end of the window would
920have been if redisplay had finished, do this:
921 (save-excursion
922 (goto-char (window-start window))
923 (vertical-motion (1- (window-height window)) window)
8646118f
RS
924 (point))") */
925
478292ed 926DEFUN ("window-end", Fwindow_end, Swindow_end, 0, 2, 0,
fdb82f93
PJ
927 doc: /* Return position at which display currently ends in WINDOW.
928This is updated by redisplay, when it runs to completion.
929Simply changing the buffer text or setting `window-start'
930does not update this value.
931If UPDATE is non-nil, compute the up-to-date position
932if it isn't already recorded. */)
933 (window, update)
478292ed 934 Lisp_Object window, update;
7ab12479
JB
935{
936 Lisp_Object value;
937 struct window *w = decode_window (window);
5a41ab94
RS
938 Lisp_Object buf;
939
940 buf = w->buffer;
b7826503 941 CHECK_BUFFER (buf);
5a41ab94 942
8646118f 943#if 0 /* This change broke some things. We should make it later. */
7250968e
RS
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))
949 return Qnil;
8646118f 950#endif
7250968e 951
478292ed
RS
952 if (! NILP (update)
953 && ! (! NILP (w->window_end_valid)
954 && XFASTINT (w->last_modified) >= MODIFF))
955 {
cbc099e5
GM
956 struct text_pos startp;
957 struct it it;
99593a9d 958 struct buffer *old_buffer = NULL, *b = XBUFFER (buf);
2d6d9df0
GM
959
960 /* In case W->start is out of the range, use something
cb795ec4 961 reasonable. This situation occurred when loading a file with
2d6d9df0
GM
962 `-l' containing a call to `rmail' with subsequent other
963 commands. At the end, W->start happened to be BEG, while
cbc099e5 964 rmail had already narrowed the buffer. */
2d6d9df0 965 if (XMARKER (w->start)->charpos < BEGV)
cbc099e5 966 SET_TEXT_POS (startp, BEGV, BEGV_BYTE);
2d6d9df0 967 else if (XMARKER (w->start)->charpos > ZV)
cbc099e5 968 SET_TEXT_POS (startp, ZV, ZV_BYTE);
2d6d9df0 969 else
cbc099e5
GM
970 SET_TEXT_POS_FROM_MARKER (startp, w->start);
971
972 /* Cannot use Fvertical_motion because that function doesn't
973 cope with variable-height lines. */
99593a9d
GM
974 if (b != current_buffer)
975 {
976 old_buffer = current_buffer;
977 set_buffer_internal (b);
978 }
979
cbc099e5
GM
980 start_display (&it, w, startp);
981 move_it_vertically (&it, window_box_height (w));
c3c45f65
GM
982 if (it.current_y < it.last_visible_y)
983 move_it_past_eol (&it);
cbc099e5 984 value = make_number (IT_CHARPOS (it));
99593a9d
GM
985
986 if (old_buffer)
987 set_buffer_internal (old_buffer);
478292ed
RS
988 }
989 else
cbc099e5 990 XSETINT (value, BUF_Z (XBUFFER (buf)) - XFASTINT (w->window_end_pos));
7ab12479
JB
991
992 return value;
993}
994
995DEFUN ("set-window-point", Fset_window_point, Sset_window_point, 2, 2, 0,
fdb82f93
PJ
996 doc: /* Make point value in WINDOW be at position POS in WINDOW's buffer. */)
997 (window, pos)
7ab12479
JB
998 Lisp_Object window, pos;
999{
1000 register struct window *w = decode_window (window);
1001
b7826503 1002 CHECK_NUMBER_COERCE_MARKER (pos);
e90c4fe6
RS
1003 if (w == XWINDOW (selected_window)
1004 && XBUFFER (w->buffer) == current_buffer)
7ab12479
JB
1005 Fgoto_char (pos);
1006 else
1007 set_marker_restricted (w->pointm, pos, w->buffer);
799417bd 1008
0685cb3c
GM
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))
799417bd 1012 ++windows_or_buffers_changed;
ce7fae7d 1013
7ab12479
JB
1014 return pos;
1015}
1016
1017DEFUN ("set-window-start", Fset_window_start, Sset_window_start, 2, 3, 0,
fdb82f93
PJ
1018 doc: /* Make display in WINDOW start at position POS in WINDOW's buffer.
1019Optional third arg NOFORCE non-nil inhibits next redisplay
1020from overriding motion of point in order to display at this exact start. */)
1021 (window, pos, noforce)
7ab12479
JB
1022 Lisp_Object window, pos, noforce;
1023{
1024 register struct window *w = decode_window (window);
1025
b7826503 1026 CHECK_NUMBER_COERCE_MARKER (pos);
7ab12479
JB
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;
265a9e55 1030 if (NILP (noforce))
7ab12479
JB
1031 w->force_start = Qt;
1032 w->update_mode_line = Qt;
d834a2e9 1033 XSETFASTINT (w->last_modified, 0);
3cd21523 1034 XSETFASTINT (w->last_overlay_modified, 0);
62c07cc7
JB
1035 if (!EQ (window, selected_window))
1036 windows_or_buffers_changed++;
ce7fae7d 1037
7ab12479
JB
1038 return pos;
1039}
1040
1041DEFUN ("window-dedicated-p", Fwindow_dedicated_p, Swindow_dedicated_p,
1042 1, 1, 0,
fdb82f93
PJ
1043 doc: /* Return WINDOW's dedicated object, usually t or nil.
1044See also `set-window-dedicated-p'. */)
1045 (window)
7ab12479
JB
1046 Lisp_Object window;
1047{
1048 return decode_window (window)->dedicated;
1049}
1050
d207b766
RS
1051DEFUN ("set-window-dedicated-p", Fset_window_dedicated_p,
1052 Sset_window_dedicated_p, 2, 2, 0,
fdb82f93
PJ
1053 doc: /* Control whether WINDOW is dedicated to the buffer it displays.
1054If it is dedicated, Emacs will not automatically change
1055which buffer appears in it.
1056The second argument is the new value for the dedication flag;
1057non-nil means yes. */)
1058 (window, arg)
1059 Lisp_Object window, arg;
7ab12479
JB
1060{
1061 register struct window *w = decode_window (window);
1062
265a9e55 1063 if (NILP (arg))
7ab12479
JB
1064 w->dedicated = Qnil;
1065 else
d207b766 1066 w->dedicated = Qt;
7ab12479
JB
1067
1068 return w->dedicated;
1069}
1070
1071DEFUN ("window-display-table", Fwindow_display_table, Swindow_display_table,
1072 0, 1, 0,
fdb82f93
PJ
1073 doc: /* Return the display-table that WINDOW is using. */)
1074 (window)
7ab12479
JB
1075 Lisp_Object window;
1076{
1077 return decode_window (window)->display_table;
1078}
1079
5500c422
GM
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,
1083 return 0. */
7ab12479 1084
319315f1 1085struct Lisp_Char_Table *
7ab12479
JB
1086window_display_table (w)
1087 struct window *w;
1088{
c756cdbe
GM
1089 struct Lisp_Char_Table *dp = NULL;
1090
1091 if (DISP_TABLE_P (w->display_table))
1092 dp = XCHAR_TABLE (w->display_table);
1093 else if (BUFFERP (w->buffer))
1094 {
1095 struct buffer *b = XBUFFER (w->buffer);
1096
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);
1101 }
171d003c 1102
c756cdbe 1103 return dp;
7ab12479
JB
1104}
1105
3a2712f9 1106DEFUN ("set-window-display-table", Fset_window_display_table, Sset_window_display_table, 2, 2, 0,
fdb82f93
PJ
1107 doc: /* Set WINDOW's display-table to TABLE. */)
1108 (window, table)
7ab12479
JB
1109 register Lisp_Object window, table;
1110{
1111 register struct window *w;
7ab12479
JB
1112
1113 w = decode_window (window);
1114 w->display_table = table;
1115 return table;
1116}
1117\f
1118/* Record info on buffer window w is displaying
1119 when it is about to cease to display that buffer. */
dfcf069d 1120static void
7ab12479
JB
1121unshow_buffer (w)
1122 register struct window *w;
1123{
12cae7c0 1124 Lisp_Object buf;
b73ea88e 1125 struct buffer *b;
7ab12479 1126
12cae7c0 1127 buf = w->buffer;
b73ea88e
RS
1128 b = XBUFFER (buf);
1129 if (b != XMARKER (w->pointm)->buffer)
7ab12479
JB
1130 abort ();
1131
573f41ab 1132#if 0
7ab12479
JB
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. */
573f41ab
RS
1137#endif
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. */
b73ea88e 1145 b->last_window_start = marker_position (w->start);
7ab12479
JB
1146
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. */
719eaeb1
GM
1150 if (! EQ (buf, XWINDOW (selected_window)->buffer)
1151 /* This line helps to fix Horsley's testbug.el bug. */
8801a864
KR
1152 && !(WINDOWP (b->last_selected_window)
1153 && w != XWINDOW (b->last_selected_window)
719eaeb1 1154 && EQ (buf, XWINDOW (b->last_selected_window)->buffer)))
b73ea88e
RS
1155 temp_set_point_both (b,
1156 clip_to_bounds (BUF_BEGV (b),
1157 XMARKER (w->pointm)->charpos,
1158 BUF_ZV (b)),
1159 clip_to_bounds (BUF_BEGV_BYTE (b),
1160 marker_byte_position (w->pointm),
1161 BUF_ZV_BYTE (b)));
719eaeb1 1162
8801a864
KR
1163 if (WINDOWP (b->last_selected_window)
1164 && w == XWINDOW (b->last_selected_window))
719eaeb1 1165 b->last_selected_window = Qnil;
7ab12479
JB
1166}
1167
1168/* Put replacement into the window structure in place of old. */
dfcf069d 1169static void
7ab12479
JB
1170replace_window (old, replacement)
1171 Lisp_Object old, replacement;
1172{
1173 register Lisp_Object tem;
1174 register struct window *o = XWINDOW (old), *p = XWINDOW (replacement);
1175
44fa5b1e
JB
1176 /* If OLD is its frame's root_window, then replacement is the new
1177 root_window for that frame. */
7ab12479 1178
7f4161e0 1179 if (EQ (old, FRAME_ROOT_WINDOW (XFRAME (o->frame))))
44fa5b1e 1180 FRAME_ROOT_WINDOW (XFRAME (o->frame)) = replacement;
7ab12479
JB
1181
1182 p->left = o->left;
1183 p->top = o->top;
1184 p->width = o->width;
1185 p->height = o->height;
5500c422
GM
1186 p->desired_matrix = p->current_matrix = 0;
1187 p->vscroll = 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;
4b26579e 1192 p->phys_cursor_width = -1;
5500c422
GM
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;
7bbb5782 1198 p->frozen_window_start_p = 0;
f984d4fc 1199 p->orig_top = p->orig_height = Qnil;
7ab12479
JB
1200
1201 p->next = tem = o->next;
265a9e55 1202 if (!NILP (tem))
7ab12479
JB
1203 XWINDOW (tem)->prev = replacement;
1204
1205 p->prev = tem = o->prev;
265a9e55 1206 if (!NILP (tem))
7ab12479
JB
1207 XWINDOW (tem)->next = replacement;
1208
1209 p->parent = tem = o->parent;
265a9e55 1210 if (!NILP (tem))
7ab12479
JB
1211 {
1212 if (EQ (XWINDOW (tem)->vchild, old))
1213 XWINDOW (tem)->vchild = replacement;
1214 if (EQ (XWINDOW (tem)->hchild, old))
1215 XWINDOW (tem)->hchild = replacement;
1216 }
1217
1218/*** Here, if replacement is a vertical combination
1219and so is its new parent, we should make replacement's
1220children be children of that parent instead. ***/
1221}
1222
1223DEFUN ("delete-window", Fdelete_window, Sdelete_window, 0, 1, "",
fdb82f93
PJ
1224 doc: /* Remove WINDOW from the display. Default is selected window. */)
1225 (window)
7ab12479 1226 register Lisp_Object window;
543f5fb1
RS
1227{
1228 delete_window (window);
1229
1230 if (! NILP (Vwindow_configuration_change_hook)
1231 && ! NILP (Vrun_hooks))
1232 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
1233
1234 return Qnil;
1235}
1236
5e14b1fc 1237void
543f5fb1
RS
1238delete_window (window)
1239 register Lisp_Object window;
7ab12479
JB
1240{
1241 register Lisp_Object tem, parent, sib;
1242 register struct window *p;
1243 register struct window *par;
951f9df5 1244 struct frame *f;
7ab12479 1245
605be8af
JB
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. */
265a9e55 1249 if (NILP (window))
7ab12479
JB
1250 window = selected_window;
1251 else
b7826503 1252 CHECK_WINDOW (window);
7ab12479 1253 p = XWINDOW (window);
605be8af
JB
1254
1255 /* It's okay to delete an already-deleted window. */
1256 if (NILP (p->buffer)
1257 && NILP (p->hchild)
1258 && NILP (p->vchild))
296b535c 1259 return;
605be8af 1260
7ab12479 1261 parent = p->parent;
265a9e55 1262 if (NILP (parent))
7ab12479
JB
1263 error ("Attempt to delete minibuffer or sole ordinary window");
1264 par = XWINDOW (parent);
1265
1266 windows_or_buffers_changed++;
67492200 1267 Vwindow_list = Qnil;
951f9df5
GM
1268 f = XFRAME (WINDOW_FRAME (p));
1269 FRAME_WINDOW_SIZES_CHANGED (f) = 1;
7ab12479 1270
605be8af
JB
1271 /* Are we trying to delete any frame's selected window? */
1272 {
192c3131 1273 Lisp_Object swindow, pwindow;
605be8af 1274
0def0403
RS
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. */
192c3131 1278 swindow = FRAME_SELECTED_WINDOW (f);
0def0403 1279
192c3131 1280 while (1)
0def0403 1281 {
192c3131
RS
1282 pwindow = swindow;
1283 while (!NILP (pwindow))
1284 {
1285 if (EQ (window, pwindow))
1286 break;
1287 pwindow = XWINDOW (pwindow)->parent;
1288 }
1289
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))
0def0403 1293 break;
192c3131
RS
1294 /* Otherwise, try another window for SWINDOW. */
1295 swindow = Fnext_window (swindow, Qlambda, Qnil);;
1296
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");
0def0403
RS
1302 }
1303
192c3131
RS
1304 /* If we need to change SWINDOW, do it. */
1305 if (! EQ (swindow, FRAME_SELECTED_WINDOW (f)))
605be8af 1306 {
605be8af
JB
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. */
192c3131
RS
1312 if (EQ (FRAME_SELECTED_WINDOW (f), selected_window))
1313 Fselect_window (swindow);
605be8af 1314 else
192c3131 1315 FRAME_SELECTED_WINDOW (f) = swindow;
605be8af
JB
1316 }
1317 }
7ab12479
JB
1318
1319 tem = p->buffer;
1320 /* tem is null for dummy parent windows
1321 (which have inferiors but not any contents themselves) */
265a9e55 1322 if (!NILP (tem))
7ab12479
JB
1323 {
1324 unshow_buffer (p);
1325 unchain_marker (p->pointm);
1326 unchain_marker (p->start);
7ab12479
JB
1327 }
1328
05e71564
GM
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. */
1333 BLOCK_INPUT;
951f9df5 1334 free_window_matrices (XWINDOW (FRAME_ROOT_WINDOW (f)));
5500c422 1335
7ab12479 1336 tem = p->next;
265a9e55 1337 if (!NILP (tem))
7ab12479
JB
1338 XWINDOW (tem)->prev = p->prev;
1339
1340 tem = p->prev;
265a9e55 1341 if (!NILP (tem))
7ab12479
JB
1342 XWINDOW (tem)->next = p->next;
1343
1344 if (EQ (window, par->hchild))
1345 par->hchild = p->next;
1346 if (EQ (window, par->vchild))
1347 par->vchild = p->next;
1348
1349 /* Find one of our siblings to give our space to. */
1350 sib = p->prev;
265a9e55 1351 if (NILP (sib))
7ab12479
JB
1352 {
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
1356 children. */
1357 sib = p->next;
7f4161e0
JB
1358 XWINDOW (sib)->top = p->top;
1359 XWINDOW (sib)->left = p->left;
7ab12479
JB
1360 }
1361
1362 /* Stretch that sibling. */
265a9e55 1363 if (!NILP (par->vchild))
7ab12479
JB
1364 set_window_height (sib,
1365 XFASTINT (XWINDOW (sib)->height) + XFASTINT (p->height),
1366 1);
265a9e55 1367 if (!NILP (par->hchild))
7ab12479
JB
1368 set_window_width (sib,
1369 XFASTINT (XWINDOW (sib)->width) + XFASTINT (p->width),
1370 1);
1371
1372 /* If parent now has only one child,
1373 put the child into the parent's place. */
7ab12479 1374 tem = par->hchild;
265a9e55 1375 if (NILP (tem))
7ab12479 1376 tem = par->vchild;
265a9e55 1377 if (NILP (XWINDOW (tem)->next))
7ab12479 1378 replace_window (parent, tem);
605be8af
JB
1379
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));
1386
1387 /* Mark this window as deleted. */
1388 p->buffer = p->hchild = p->vchild = Qnil;
5500c422
GM
1389
1390 /* Adjust glyph matrices. */
951f9df5 1391 adjust_glyphs (f);
05e71564 1392 UNBLOCK_INPUT;
7ab12479 1393}
67492200
GM
1394
1395
7ab12479 1396\f
67492200
GM
1397/***********************************************************************
1398 Window List
1399 ***********************************************************************/
1400
f95464e4
GM
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. */
67492200
GM
1404
1405static int
f95464e4 1406add_window_to_list (w, user_data)
67492200 1407 struct window *w;
f95464e4 1408 void *user_data;
67492200 1409{
f95464e4 1410 Lisp_Object *list = (Lisp_Object *) user_data;
67492200
GM
1411 Lisp_Object window;
1412 XSETWINDOW (window, w);
212116d6 1413 *list = Fcons (window, *list);
67492200
GM
1414 return 1;
1415}
1416
1417
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. */
1421
1422static Lisp_Object
1423window_list ()
1424{
1425 if (!CONSP (Vwindow_list))
1426 {
1427 Lisp_Object tail;
212116d6 1428
67492200
GM
1429 Vwindow_list = Qnil;
1430 for (tail = Vframe_list; CONSP (tail); tail = XCDR (tail))
212116d6
GM
1431 {
1432 Lisp_Object args[2];
1433
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. */
1437 args[1] = Qnil;
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);
1442 }
67492200
GM
1443 }
1444
1445 return Vwindow_list;
1446}
1447
1448
118ea242
GM
1449/* Value is non-zero if WINDOW satisfies the constraints given by
1450 OWINDOW, MINIBUF and ALL_FRAMES.
67492200 1451
118ea242
GM
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
67492200 1455
118ea242
GM
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. */
67492200
GM
1462
1463static int
118ea242
GM
1464candidate_window_p (window, owindow, minibuf, all_frames)
1465 Lisp_Object window, owindow, minibuf, all_frames;
67492200
GM
1466{
1467 struct window *w = XWINDOW (window);
1468 struct frame *f = XFRAME (w->frame);
1469 int candidate_p = 1;
1470
1471 if (!BUFFERP (w->buffer))
1472 candidate_p = 0;
1473 else if (MINI_WINDOW_P (w)
1474 && (EQ (minibuf, Qlambda)
1475 || (WINDOWP (minibuf) && !EQ (minibuf, window))))
1476 {
1477 /* If MINIBUF is `lambda' don't consider any mini-windows.
1478 If it is a window, consider only that one. */
1479 candidate_p = 0;
1480 }
118ea242
GM
1481 else if (EQ (all_frames, Qt))
1482 candidate_p = 1;
67492200 1483 else if (NILP (all_frames))
118ea242
GM
1484 {
1485 xassert (WINDOWP (owindow));
1486 candidate_p = EQ (w->frame, XWINDOW (owindow)->frame);
1487 }
67492200
GM
1488 else if (EQ (all_frames, Qvisible))
1489 {
1490 FRAME_SAMPLE_VISIBILITY (f);
1491 candidate_p = FRAME_VISIBLE_P (f);
1492 }
1493 else if (INTEGERP (all_frames) && XINT (all_frames) == 0)
1494 {
1495 FRAME_SAMPLE_VISIBILITY (f);
1496 candidate_p = FRAME_VISIBLE_P (f) || FRAME_ICONIFIED_P (f);
1497 }
67492200
GM
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)));
118ea242
GM
1502 else if (FRAMEP (all_frames))
1503 candidate_p = EQ (all_frames, w->frame);
67492200
GM
1504
1505 return candidate_p;
1506}
1507
1508
1509/* Decode arguments as allowed by Fnext_window, Fprevious_window, and
1510 Fwindow_list. See there for the meaning of WINDOW, MINIBUF, and
1511 ALL_FRAMES. */
1512
1513static void
1514decode_next_window_args (window, minibuf, all_frames)
1515 Lisp_Object *window, *minibuf, *all_frames;
1516{
1517 if (NILP (*window))
1518 *window = selected_window;
1519 else
b7826503 1520 CHECK_LIVE_WINDOW (*window);
67492200
GM
1521
1522 /* MINIBUF nil may or may not include minibuffers. Decide if it
1523 does. */
1524 if (NILP (*minibuf))
1525 *minibuf = minibuf_level ? minibuf_window : Qlambda;
1526 else if (!EQ (*minibuf, Qt))
1527 *minibuf = Qlambda;
1528
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. */
1532
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))
1537 : Qnil);
1538 else if (EQ (*all_frames, Qvisible))
1539 ;
1540 else if (XFASTINT (*all_frames) == 0)
1541 ;
1542 else if (FRAMEP (*all_frames))
1543 ;
1544 else if (!EQ (*all_frames, Qt))
1545 *all_frames = Qnil;
1546
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. */
1552}
1553
1554
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
1558 ALL_FRAMES. */
1559
1560static Lisp_Object
1561next_window (window, minibuf, all_frames, next_p)
1562 Lisp_Object window, minibuf, all_frames;
1563 int next_p;
1564{
1565 decode_next_window_args (&window, &minibuf, &all_frames);
1566
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);
1572
212116d6 1573 if (next_p)
67492200
GM
1574 {
1575 Lisp_Object list;
1576
1577 /* Find WINDOW in the list of all windows. */
1578 list = Fmemq (window, window_list ());
1579
1580 /* Scan forward from WINDOW to the end of the window list. */
1581 if (CONSP (list))
1582 for (list = XCDR (list); CONSP (list); list = XCDR (list))
118ea242 1583 if (candidate_window_p (XCAR (list), window, minibuf, all_frames))
67492200
GM
1584 break;
1585
1586 /* Scan from the start of the window list up to WINDOW. */
1587 if (!CONSP (list))
1588 for (list = Vwindow_list;
1589 CONSP (list) && !EQ (XCAR (list), window);
1590 list = XCDR (list))
118ea242 1591 if (candidate_window_p (XCAR (list), window, minibuf, all_frames))
67492200
GM
1592 break;
1593
1594 if (CONSP (list))
1595 window = XCAR (list);
1596 }
1597 else
1598 {
1599 Lisp_Object candidate, list;
1600
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. */
1605 candidate = Qnil;
1606 for (list = window_list (); CONSP (list); list = XCDR (list))
1607 {
1608 if (EQ (XCAR (list), window))
1609 {
1610 if (WINDOWP (candidate))
1611 break;
1612 }
118ea242
GM
1613 else if (candidate_window_p (XCAR (list), window, minibuf,
1614 all_frames))
67492200
GM
1615 candidate = XCAR (list);
1616 }
1617
1618 if (WINDOWP (candidate))
1619 window = candidate;
1620 }
1621
1622 return window;
1623}
7ab12479 1624
7ab12479 1625
26f6279d 1626DEFUN ("next-window", Fnext_window, Snext_window, 0, 3, 0,
fdb82f93
PJ
1627 doc: /* Return next window after WINDOW in canonical ordering of windows.
1628If omitted, WINDOW defaults to the selected window.
1629
1630Optional second arg MINIBUF t means count the minibuffer window even
1631if not active. MINIBUF nil or omitted means count the minibuffer iff
1632it is active. MINIBUF neither t nor nil means not to count the
1633minibuffer even if it is active.
1634
1635Several frames may share a single minibuffer; if the minibuffer
1636counts, all windows on all frames that share that minibuffer count
1637too. Therefore, `next-window' can be used to iterate through the
1638set of windows even when the minibuffer is on another frame. If the
1639minibuffer does not count, only windows from WINDOW's frame count.
1640
1641Optional third arg ALL-FRAMES t means include windows on all frames.
1642ALL-FRAMES nil or omitted means cycle within the frames as specified
1643above. ALL-FRAMES = `visible' means include windows on all visible frames.
1644ALL-FRAMES = 0 means include windows on all visible and iconified frames.
1645If ALL-FRAMES is a frame, restrict search to windows on that frame.
1646Anything else means restrict to WINDOW's frame.
1647
1648If you use consistent values for MINIBUF and ALL-FRAMES, you can use
1649`next-window' to iterate through the entire cycle of acceptable
1650windows, 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)
67492200 1653 Lisp_Object window, minibuf, all_frames;
7ab12479 1654{
67492200 1655 return next_window (window, minibuf, all_frames, 1);
7ab12479
JB
1656}
1657
67492200 1658
26f6279d 1659DEFUN ("previous-window", Fprevious_window, Sprevious_window, 0, 3, 0,
fdb82f93
PJ
1660 doc: /* Return the window preceding WINDOW in canonical ordering of windows.
1661If omitted, WINDOW defaults to the selected window.
1662
1663Optional second arg MINIBUF t means count the minibuffer window even
1664if not active. MINIBUF nil or omitted means count the minibuffer iff
1665it is active. MINIBUF neither t nor nil means not to count the
1666minibuffer even if it is active.
1667
1668Several frames may share a single minibuffer; if the minibuffer
1669counts, all windows on all frames that share that minibuffer count
1670too. Therefore, `previous-window' can be used to iterate through
1671the set of windows even when the minibuffer is on another frame. If
1672the minibuffer does not count, only windows from WINDOW's frame count
1673
1674Optional third arg ALL-FRAMES t means include windows on all frames.
1675ALL-FRAMES nil or omitted means cycle within the frames as specified
1676above. ALL-FRAMES = `visible' means include windows on all visible frames.
1677ALL-FRAMES = 0 means include windows on all visible and iconified frames.
1678If ALL-FRAMES is a frame, restrict search to windows on that frame.
1679Anything else means restrict to WINDOW's frame.
1680
1681If you use consistent values for MINIBUF and ALL-FRAMES, you can use
1682`previous-window' to iterate through the entire cycle of acceptable
1683windows, 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)
67492200 1686 Lisp_Object window, minibuf, all_frames;
7ab12479 1687{
67492200 1688 return next_window (window, minibuf, all_frames, 0);
7ab12479
JB
1689}
1690
67492200 1691
62c07cc7 1692DEFUN ("other-window", Fother_window, Sother_window, 1, 2, "p",
fdb82f93
PJ
1693 doc: /* Select the ARG'th different window on this frame.
1694All windows on current frame are arranged in a cyclic order.
1695This command selects the window ARG steps away in that order.
1696A negative ARG moves in the opposite order. If the optional second
1697argument ALL_FRAMES is non-nil, cycle through all frames. */)
1698 (arg, all_frames)
67492200 1699 Lisp_Object arg, all_frames;
7ab12479 1700{
67492200
GM
1701 Lisp_Object window;
1702 int i;
7ab12479 1703
b7826503 1704 CHECK_NUMBER (arg);
67492200
GM
1705 window = selected_window;
1706
1707 for (i = XINT (arg); i > 0; --i)
1708 window = Fnext_window (window, Qnil, all_frames);
1709 for (; i < 0; ++i)
1710 window = Fprevious_window (window, Qnil, all_frames);
50e88778 1711
67492200 1712 Fselect_window (window);
7ab12479
JB
1713 return Qnil;
1714}
67492200
GM
1715
1716
1717DEFUN ("window-list", Fwindow_list, Swindow_list, 0, 3, 0,
fdb82f93
PJ
1718 doc: /* Return a list of windows on FRAME, starting with WINDOW.
1719FRAME nil or omitted means use the selected frame.
1720WINDOW nil or omitted means use the selected window.
1721MINIBUF t means include the minibuffer window, even if it isn't active.
1722MINIBUF nil or omitted means include the minibuffer window only
1723if it's active.
1724MINIBUF neither nil nor t means never include the minibuffer window. */)
1725 (frame, minibuf, window)
cd2904bd
GM
1726 Lisp_Object frame, minibuf, window;
1727{
cd2904bd
GM
1728 if (NILP (window))
1729 window = selected_window;
1730 if (NILP (frame))
1731 frame = selected_frame;
1732
1733 if (!EQ (frame, XWINDOW (window)->frame))
1734 error ("Window is on a different frame");
1735
1736 return window_list_1 (window, minibuf, frame);
1737}
1738
1739
1740/* Return a list of windows in canonical ordering. Arguments are like
1741 for `next-window'. */
1742
1743static Lisp_Object
1744window_list_1 (window, minibuf, all_frames)
069f5950 1745 Lisp_Object window, minibuf, all_frames;
67492200
GM
1746{
1747 Lisp_Object tail, list;
1748
1749 decode_next_window_args (&window, &minibuf, &all_frames);
1750 list = Qnil;
1751
1752 for (tail = window_list (); CONSP (tail); tail = XCDR (tail))
118ea242 1753 if (candidate_window_p (XCAR (tail), window, minibuf, all_frames))
67492200
GM
1754 list = Fcons (XCAR (tail), list);
1755
118ea242 1756 return Fnreverse (list);
67492200
GM
1757}
1758
1759
7ab12479
JB
1760\f
1761/* Look at all windows, performing an operation specified by TYPE
1762 with argument OBJ.
75d8f668 1763 If FRAMES is Qt, look at all frames;
75d8f668 1764 Qnil, look at just the selected frame;
89bca612 1765 Qvisible, look at visible frames;
75d8f668 1766 a frame, just look at windows on that frame.
85fe3b5e 1767 If MINI is non-zero, perform the operation on minibuffer windows too. */
7ab12479
JB
1768
1769enum window_loop
1770{
1771 WINDOW_LOOP_UNUSED,
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 */
1776 GET_LARGEST_WINDOW,
3f8ab7bd
RS
1777 UNSHOW_BUFFER, /* Arg is buffer */
1778 CHECK_ALL_WINDOWS
7ab12479
JB
1779};
1780
1781static Lisp_Object
44fa5b1e 1782window_loop (type, obj, mini, frames)
7ab12479 1783 enum window_loop type;
118ea242 1784 Lisp_Object obj, frames;
7ab12479
JB
1785 int mini;
1786{
118ea242
GM
1787 Lisp_Object window, windows, best_window, frame_arg;
1788 struct frame *f;
ffdc852d 1789 struct gcpro gcpro1;
118ea242 1790
44fa5b1e
JB
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))
118ea242 1795 f = XFRAME (frames);
44fa5b1e 1796 else if (NILP (frames))
118ea242 1797 f = SELECTED_FRAME ();
7ab12479 1798 else
118ea242
GM
1799 f = NULL;
1800
1801 if (f)
89bca612 1802 frame_arg = Qlambda;
f812f9c6
RS
1803 else if (XFASTINT (frames) == 0)
1804 frame_arg = frames;
89bca612
RS
1805 else if (EQ (frames, Qvisible))
1806 frame_arg = frames;
118ea242
GM
1807 else
1808 frame_arg = Qt;
7ab12479 1809
89bca612
RS
1810 /* frame_arg is Qlambda to stick to one frame,
1811 Qvisible to consider all visible frames,
1812 or Qt otherwise. */
1813
7ab12479 1814 /* Pick a window to start with. */
017b2bad 1815 if (WINDOWP (obj))
118ea242
GM
1816 window = obj;
1817 else if (f)
1818 window = FRAME_SELECTED_WINDOW (f);
7ab12479 1819 else
118ea242 1820 window = FRAME_SELECTED_WINDOW (SELECTED_FRAME ());
4b206065 1821
cd2904bd 1822 windows = window_list_1 (window, mini ? Qt : Qnil, frame_arg);
118ea242 1823 GCPRO1 (windows);
7ab12479 1824 best_window = Qnil;
118ea242
GM
1825
1826 for (; CONSP (windows); windows = CDR (windows))
7ab12479 1827 {
118ea242
GM
1828 struct window *w;
1829
1830 window = XCAR (windows);
1831 w = XWINDOW (window);
1832
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)
65a04b96
RS
1837 /* For UNSHOW_BUFFER, we must always consider all windows. */
1838 || type == UNSHOW_BUFFER
7ab12479
JB
1839 || (mini && minibuf_level > 0))
1840 switch (type)
1841 {
1842 case GET_BUFFER_WINDOW:
118ea242 1843 if (EQ (w->buffer, obj)
5c204627
RS
1844 /* Don't find any minibuffer window
1845 except the one that is currently in use. */
118ea242
GM
1846 && (MINI_WINDOW_P (w)
1847 ? EQ (window, minibuf_window)
1848 : 1))
1849 {
5ddc1b75
GM
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
1854 selected-window. */
1855 best_window = window;
118ea242 1856 }
7ab12479
JB
1857 break;
1858
1859 case GET_LRU_WINDOW:
1860 /* t as arg means consider only full-width windows */
118ea242 1861 if (!NILP (obj) && !WINDOW_FULL_WIDTH_P (w))
7ab12479 1862 break;
7ab12479 1863 /* Ignore dedicated windows and minibuffers. */
a5731348 1864 if (MINI_WINDOW_P (w) || EQ (w->dedicated, Qt))
7ab12479 1865 break;
265a9e55 1866 if (NILP (best_window)
7ab12479 1867 || (XFASTINT (XWINDOW (best_window)->use_time)
118ea242
GM
1868 > XFASTINT (w->use_time)))
1869 best_window = window;
7ab12479
JB
1870 break;
1871
1872 case DELETE_OTHER_WINDOWS:
118ea242
GM
1873 if (!EQ (window, obj))
1874 Fdelete_window (window);
7ab12479
JB
1875 break;
1876
1877 case DELETE_BUFFER_WINDOWS:
118ea242 1878 if (EQ (w->buffer, obj))
7ab12479 1879 {
118ea242 1880 struct frame *f = XFRAME (WINDOW_FRAME (w));
3548e138
RS
1881
1882 /* If this window is dedicated, and in a frame of its own,
1883 kill the frame. */
118ea242
GM
1884 if (EQ (window, FRAME_ROOT_WINDOW (f))
1885 && !NILP (w->dedicated)
3548e138 1886 && other_visible_frames (f))
7ab12479 1887 {
3548e138
RS
1888 /* Skip the other windows on this frame.
1889 There might be one, the minibuffer! */
118ea242
GM
1890 while (CONSP (XCDR (windows))
1891 && EQ (XWINDOW (XCAR (windows))->frame,
1892 XWINDOW (XCAR (XCDR (windows)))->frame))
1893 windows = XCDR (windows);
1894
3548e138 1895 /* Now we can safely delete the frame. */
118ea242
GM
1896 Fdelete_frame (w->frame, Qnil);
1897 }
1898 else if (NILP (w->parent))
1899 {
1900 /* If we're deleting the buffer displayed in the
1901 only window on the frame, find a new buffer to
1902 display there. */
1903 Lisp_Object buffer;
1904 buffer = Fother_buffer (obj, Qnil, w->frame);
118ea242
GM
1905 Fset_window_buffer (window, buffer);
1906 if (EQ (window, selected_window))
1907 Fset_buffer (w->buffer);
7ab12479
JB
1908 }
1909 else
118ea242 1910 Fdelete_window (window);
7ab12479
JB
1911 }
1912 break;
1913
1914 case GET_LARGEST_WINDOW:
7ab12479 1915 {
118ea242 1916 /* Ignore dedicated windows and minibuffers. */
a5731348 1917 if (MINI_WINDOW_P (w) || EQ (w->dedicated, Qt))
118ea242
GM
1918 break;
1919
c930dfab 1920 if (NILP (best_window))
118ea242 1921 best_window = window;
c930dfab
GM
1922 else
1923 {
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;
1928 }
7ab12479
JB
1929 }
1930 break;
1931
1932 case UNSHOW_BUFFER:
118ea242 1933 if (EQ (w->buffer, obj))
7ab12479 1934 {
118ea242
GM
1935 Lisp_Object buffer;
1936 struct frame *f = XFRAME (w->frame);
1937
7ab12479 1938 /* Find another buffer to show in this window. */
118ea242 1939 buffer = Fother_buffer (obj, Qnil, w->frame);
118ea242 1940
38ab08d1
RS
1941 /* If this window is dedicated, and in a frame of its own,
1942 kill the frame. */
118ea242
GM
1943 if (EQ (window, FRAME_ROOT_WINDOW (f))
1944 && !NILP (w->dedicated)
38ab08d1 1945 && other_visible_frames (f))
45945a7b
RS
1946 {
1947 /* Skip the other windows on this frame.
1948 There might be one, the minibuffer! */
118ea242
GM
1949 while (CONSP (XCDR (windows))
1950 && EQ (XWINDOW (XCAR (windows))->frame,
1951 XWINDOW (XCAR (XCDR (windows)))->frame))
1952 windows = XCDR (windows);
1953
45945a7b 1954 /* Now we can safely delete the frame. */
118ea242 1955 Fdelete_frame (w->frame, Qnil);
45945a7b 1956 }
a5731348
SM
1957 else if (!NILP (w->dedicated) && !NILP (w->parent))
1958 {
1959 Lisp_Object window;
1960 XSETWINDOW (window, w);
1961 /* If this window is dedicated and not the only window
1962 in its frame, then kill it. */
1963 Fdelete_window (window);
1964 }
38ab08d1 1965 else
38ab08d1
RS
1966 {
1967 /* Otherwise show a different buffer in the window. */
118ea242
GM
1968 w->dedicated = Qnil;
1969 Fset_window_buffer (window, buffer);
1970 if (EQ (window, selected_window))
1971 Fset_buffer (w->buffer);
38ab08d1 1972 }
7ab12479
JB
1973 }
1974 break;
3f8ab7bd
RS
1975
1976 /* Check for a window that has a killed buffer. */
1977 case CHECK_ALL_WINDOWS:
118ea242
GM
1978 if (! NILP (w->buffer)
1979 && NILP (XBUFFER (w->buffer)->name))
3f8ab7bd 1980 abort ();
118ea242 1981 break;
6bbd7a29
GM
1982
1983 case WINDOW_LOOP_UNUSED:
1984 break;
7ab12479 1985 }
7ab12479 1986 }
7ab12479 1987
118ea242 1988 UNGCPRO;
7ab12479 1989 return best_window;
37962e60 1990}
605be8af 1991
3f8ab7bd
RS
1992/* Used for debugging. Abort if any window has a dead buffer. */
1993
233a4a2c 1994void
3f8ab7bd
RS
1995check_all_windows ()
1996{
1997 window_loop (CHECK_ALL_WINDOWS, Qnil, 1, Qt);
1998}
1999
7ab12479 2000DEFUN ("get-lru-window", Fget_lru_window, Sget_lru_window, 0, 1, 0,
fdb82f93
PJ
2001 doc: /* Return the window least recently selected or used for display.
2002If optional argument FRAME is `visible', search all visible frames.
2003If FRAME is 0, search all visible and iconified frames.
2004If FRAME is t, search all frames.
2005If FRAME is nil, search only the selected frame.
2006If FRAME is a frame, search only that frame. */)
2007 (frame)
2008 Lisp_Object frame;
7ab12479
JB
2009{
2010 register Lisp_Object w;
2011 /* First try for a window that is full-width */
89bca612 2012 w = window_loop (GET_LRU_WINDOW, Qt, 0, frame);
265a9e55 2013 if (!NILP (w) && !EQ (w, selected_window))
7ab12479
JB
2014 return w;
2015 /* If none of them, try the rest */
89bca612 2016 return window_loop (GET_LRU_WINDOW, Qnil, 0, frame);
7ab12479
JB
2017}
2018
2019DEFUN ("get-largest-window", Fget_largest_window, Sget_largest_window, 0, 1, 0,
fdb82f93
PJ
2020 doc: /* Return the largest window in area.
2021If optional argument FRAME is `visible', search all visible frames.
2022If FRAME is 0, search all visible and iconified frames.
2023If FRAME is t, search all frames.
2024If FRAME is nil, search only the selected frame.
2025If FRAME is a frame, search only that frame. */)
2026 (frame)
2027 Lisp_Object frame;
7ab12479
JB
2028{
2029 return window_loop (GET_LARGEST_WINDOW, Qnil, 0,
44fa5b1e 2030 frame);
7ab12479
JB
2031}
2032
2033DEFUN ("get-buffer-window", Fget_buffer_window, Sget_buffer_window, 1, 2, 0,
fdb82f93
PJ
2034 doc: /* Return a window currently displaying BUFFER, or nil if none.
2035If optional argument FRAME is `visible', search all visible frames.
2036If optional argument FRAME is 0, search all visible and iconified frames.
2037If FRAME is t, search all frames.
2038If FRAME is nil, search only the selected frame.
2039If FRAME is a frame, search only that frame. */)
2040 (buffer, frame)
2041 Lisp_Object buffer, frame;
7ab12479
JB
2042{
2043 buffer = Fget_buffer (buffer);
017b2bad 2044 if (BUFFERP (buffer))
44fa5b1e 2045 return window_loop (GET_BUFFER_WINDOW, buffer, 1, frame);
7ab12479
JB
2046 else
2047 return Qnil;
2048}
2049
2050DEFUN ("delete-other-windows", Fdelete_other_windows, Sdelete_other_windows,
fdb82f93
PJ
2051 0, 1, "",
2052 doc: /* Make WINDOW (or the selected window) fill its frame.
2053Only the frame WINDOW is on is affected.
2054This function tries to reduce display jumps
2055by keeping the text previously visible in WINDOW
2056in the same place on the frame. Doing this depends on
2057the value of (window-start WINDOW), so if calling this function
2058in a program gives strange scrolling, make sure the window-start
2059value is reasonable when this function is called. */)
2060 (window)
7ab12479
JB
2061 Lisp_Object window;
2062{
2063 struct window *w;
00d3d838 2064 int startpos;
85fe3b5e 2065 int top, new_top;
7ab12479 2066
265a9e55 2067 if (NILP (window))
7ab12479
JB
2068 window = selected_window;
2069 else
b7826503 2070 CHECK_LIVE_WINDOW (window);
7ab12479 2071 w = XWINDOW (window);
a2b38b3c 2072
00d3d838 2073 startpos = marker_position (w->start);
5500c422 2074 top = XFASTINT (w->top) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w)));
7ab12479 2075
a2b38b3c
RS
2076 if (MINI_WINDOW_P (w) && top > 0)
2077 error ("Can't expand minibuffer to full frame");
2078
70728a80 2079 window_loop (DELETE_OTHER_WINDOWS, window, 0, WINDOW_FRAME (w));
7ab12479 2080
00d3d838
KH
2081 /* Try to minimize scrolling, by setting the window start to the point
2082 will cause the text at the old window start to be at the same place
2083 on the frame. But don't try to do this if the window start is
2084 outside the visible portion (as might happen when the display is
2085 not current, due to typeahead). */
85fe3b5e
GM
2086 new_top = XFASTINT (w->top) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w)));
2087 if (new_top != top
2088 && startpos >= BUF_BEGV (XBUFFER (w->buffer))
00d3d838
KH
2089 && startpos <= BUF_ZV (XBUFFER (w->buffer)))
2090 {
2091 struct position pos;
2092 struct buffer *obuf = current_buffer;
2093
2094 Fset_buffer (w->buffer);
2095 /* This computation used to temporarily move point, but that can
2096 have unwanted side effects due to text properties. */
0383eb57 2097 pos = *vmotion (startpos, -top, w);
4d047f50 2098
b73ea88e 2099 set_marker_both (w->start, w->buffer, pos.bufpos, pos.bytepos);
85fe3b5e 2100 w->window_end_valid = Qnil;
b73ea88e
RS
2101 w->start_at_line_beg = ((pos.bytepos == BEGV_BYTE
2102 || FETCH_BYTE (pos.bytepos - 1) == '\n') ? Qt
00d3d838 2103 : Qnil);
80622eec
RS
2104 /* We need to do this, so that the window-scroll-functions
2105 get called. */
4d047f50 2106 w->optional_new_start = Qt;
00d3d838
KH
2107
2108 set_buffer_internal (obuf);
2109 }
5500c422 2110
7ab12479
JB
2111 return Qnil;
2112}
2113
2114DEFUN ("delete-windows-on", Fdelete_windows_on, Sdelete_windows_on,
fdb82f93
PJ
2115 1, 2, "bDelete windows on (buffer): ",
2116 doc: /* Delete all windows showing BUFFER.
2117Optional second argument FRAME controls which frames are affected.
2118If optional argument FRAME is `visible', search all visible frames.
2119If FRAME is 0, search all visible and iconified frames.
2120If FRAME is nil, search all frames.
2121If FRAME is t, search only the selected frame.
2122If FRAME is a frame, search only that frame. */)
2123 (buffer, frame)
26f6279d 2124 Lisp_Object buffer, frame;
7ab12479 2125{
26f6279d
JB
2126 /* FRAME uses t and nil to mean the opposite of what window_loop
2127 expects. */
c520265e
RS
2128 if (NILP (frame))
2129 frame = Qt;
2130 else if (EQ (frame, Qt))
2131 frame = Qnil;
26f6279d 2132
265a9e55 2133 if (!NILP (buffer))
7ab12479
JB
2134 {
2135 buffer = Fget_buffer (buffer);
b7826503 2136 CHECK_BUFFER (buffer);
26f6279d 2137 window_loop (DELETE_BUFFER_WINDOWS, buffer, 0, frame);
7ab12479 2138 }
5500c422 2139
7ab12479
JB
2140 return Qnil;
2141}
2142
2143DEFUN ("replace-buffer-in-windows", Freplace_buffer_in_windows,
fdb82f93
PJ
2144 Sreplace_buffer_in_windows,
2145 1, 1, "bReplace buffer in windows: ",
2146 doc: /* Replace BUFFER with some other buffer in all windows showing it. */)
2147 (buffer)
7ab12479
JB
2148 Lisp_Object buffer;
2149{
265a9e55 2150 if (!NILP (buffer))
7ab12479
JB
2151 {
2152 buffer = Fget_buffer (buffer);
b7826503 2153 CHECK_BUFFER (buffer);
7ab12479
JB
2154 window_loop (UNSHOW_BUFFER, buffer, 0, Qt);
2155 }
2156 return Qnil;
2157}
ff58478b
RS
2158
2159/* Replace BUFFER with some other buffer in all windows
2160 of all frames, even those on other keyboards. */
2161
2162void
2163replace_buffer_in_all_windows (buffer)
2164 Lisp_Object buffer;
2165{
27abb84f 2166#ifdef MULTI_KBOARD
ff58478b
RS
2167 Lisp_Object tail, frame;
2168
ff58478b
RS
2169 /* A single call to window_loop won't do the job
2170 because it only considers frames on the current keyboard.
2171 So loop manually over frames, and handle each one. */
2172 FOR_EACH_FRAME (tail, frame)
db7f721d 2173 window_loop (UNSHOW_BUFFER, buffer, 1, frame);
ff58478b 2174#else
db7f721d 2175 window_loop (UNSHOW_BUFFER, buffer, 1, Qt);
ff58478b
RS
2176#endif
2177}
7ab12479
JB
2178\f
2179/* Set the height of WINDOW and all its inferiors. */
a481b3ea
JB
2180
2181/* The smallest acceptable dimensions for a window. Anything smaller
2182 might crash Emacs. */
5500c422 2183
a481b3ea 2184#define MIN_SAFE_WINDOW_WIDTH (2)
dc1ab1ee 2185#define MIN_SAFE_WINDOW_HEIGHT (1)
a481b3ea
JB
2186
2187/* Make sure that window_min_height and window_min_width are
2188 not too small; if they are, set them to safe minima. */
2189
2190static void
2191check_min_window_sizes ()
2192{
2193 /* Smaller values might permit a crash. */
2194 if (window_min_width < MIN_SAFE_WINDOW_WIDTH)
2195 window_min_width = MIN_SAFE_WINDOW_WIDTH;
2196 if (window_min_height < MIN_SAFE_WINDOW_HEIGHT)
2197 window_min_height = MIN_SAFE_WINDOW_HEIGHT;
2198}
2199
2200/* If *ROWS or *COLS are too small a size for FRAME, set them to the
2201 minimum allowable size. */
5500c422 2202
605be8af 2203void
a481b3ea 2204check_frame_size (frame, rows, cols)
605be8af
JB
2205 FRAME_PTR frame;
2206 int *rows, *cols;
a481b3ea 2207{
628df3bf 2208 /* For height, we have to see:
54b8bcb5
RS
2209 how many windows the frame has at minimum (one or two),
2210 and whether it has a menu bar or other special stuff at the top. */
2211 int min_height
2212 = ((FRAME_MINIBUF_ONLY_P (frame) || ! FRAME_HAS_MINIBUF_P (frame))
2213 ? MIN_SAFE_WINDOW_HEIGHT
2214 : 2 * MIN_SAFE_WINDOW_HEIGHT);
5500c422
GM
2215
2216 if (FRAME_TOP_MARGIN (frame) > 0)
2217 min_height += FRAME_TOP_MARGIN (frame);
a481b3ea
JB
2218
2219 if (*rows < min_height)
2220 *rows = min_height;
2221 if (*cols < MIN_SAFE_WINDOW_WIDTH)
2222 *cols = MIN_SAFE_WINDOW_WIDTH;
2223}
2224
c1636aa6 2225
233a4a2c
GM
2226/* Value is non-zero if window W is fixed-size. WIDTH_P non-zero means
2227 check if W's width can be changed, otherwise check W's height.
2228 CHECK_SIBLINGS_P non-zero means check resizablity of WINDOW's
2229 siblings, too. If none of the siblings is resizable, WINDOW isn't
2230 either. */
c1636aa6 2231
233a4a2c
GM
2232static int
2233window_fixed_size_p (w, width_p, check_siblings_p)
2234 struct window *w;
2235 int width_p, check_siblings_p;
2236{
2237 int fixed_p;
2238 struct window *c;
2239
2240 if (!NILP (w->hchild))
2241 {
2242 c = XWINDOW (w->hchild);
2243
2244 if (width_p)
2245 {
2246 /* A horiz. combination is fixed-width if all of if its
2247 children are. */
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;
2251 }
2252 else
2253 {
2254 /* A horiz. combination is fixed-height if one of if its
2255 children is. */
2256 while (c && !window_fixed_size_p (c, width_p, 0))
2257 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2258 fixed_p = c != NULL;
2259 }
2260 }
2261 else if (!NILP (w->vchild))
2262 {
2263 c = XWINDOW (w->vchild);
2264
2265 if (width_p)
2266 {
2267 /* A vert. combination is fixed-width if one of if its
2268 children is. */
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;
2272 }
2273 else
2274 {
2275 /* A vert. combination is fixed-height if all of if its
2276 children are. */
2277 while (c && window_fixed_size_p (c, width_p, 0))
2278 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2279 fixed_p = c == NULL;
2280 }
2281 }
2282 else if (BUFFERP (w->buffer))
2283 {
a34dfd12
GM
2284 if (w->height_fixed_p && !width_p)
2285 fixed_p = 1;
2286 else
233a4a2c 2287 {
a34dfd12
GM
2288 struct buffer *old = current_buffer;
2289 Lisp_Object val;
2290
2291 current_buffer = XBUFFER (w->buffer);
2292 val = find_symbol_value (Qwindow_size_fixed);
2293 current_buffer = old;
2294
2295 fixed_p = 0;
2296 if (!EQ (val, Qunbound))
2297 {
2298 fixed_p = !NILP (val);
2299
2300 if (fixed_p
2301 && ((EQ (val, Qheight) && width_p)
2302 || (EQ (val, Qwidth) && !width_p)))
2303 fixed_p = 0;
2304 }
233a4a2c
GM
2305 }
2306
2307 /* Can't tell if this one is resizable without looking at
2308 siblings. If all siblings are fixed-size this one is too. */
2309 if (!fixed_p && check_siblings_p && WINDOWP (w->parent))
2310 {
2311 Lisp_Object child;
2312
2313 for (child = w->prev; !NILP (child); child = XWINDOW (child)->prev)
2314 if (!window_fixed_size_p (XWINDOW (child), width_p, 0))
2315 break;
2316
2317 if (NILP (child))
2318 for (child = w->next; !NILP (child); child = XWINDOW (child)->next)
2319 if (!window_fixed_size_p (XWINDOW (child), width_p, 0))
2320 break;
2321
2322 if (NILP (child))
2323 fixed_p = 1;
2324 }
2325 }
2326 else
2327 fixed_p = 1;
2328
2329 return fixed_p;
2330}
2331
2332
2333/* Return the minimum size of window W, not taking fixed-width windows
2334 into account. WIDTH_P non-zero means return the minimum width,
2335 otherwise return the minimum height. If W is a combination window,
2336 compute the minimum size from the minimum sizes of W's children. */
2337
2338static int
2339window_min_size_1 (w, width_p)
c1636aa6
GM
2340 struct window *w;
2341 int width_p;
2342{
233a4a2c 2343 struct window *c;
c1636aa6
GM
2344 int size;
2345
233a4a2c
GM
2346 if (!NILP (w->hchild))
2347 {
2348 c = XWINDOW (w->hchild);
2349 size = 0;
2350
2351 if (width_p)
2352 {
2353 /* The min width of a horizontal combination is
2354 the sum of the min widths of its children. */
2355 while (c)
2356 {
2357 size += window_min_size_1 (c, width_p);
2358 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2359 }
2360 }
2361 else
2362 {
2363 /* The min height a horizontal combination equals
2364 the maximum of all min height of its children. */
2365 while (c)
2366 {
2367 int min_size = window_min_size_1 (c, width_p);
2368 size = max (min_size, size);
2369 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2370 }
2371 }
2372 }
2373 else if (!NILP (w->vchild))
2374 {
2375 c = XWINDOW (w->vchild);
2376 size = 0;
2377
2378 if (width_p)
2379 {
2380 /* The min width of a vertical combination is
2381 the maximum of the min widths of its children. */
2382 while (c)
2383 {
2384 int min_size = window_min_size_1 (c, width_p);
2385 size = max (min_size, size);
2386 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2387 }
2388 }
2389 else
2390 {
2391 /* The min height of a vertical combination equals
2392 the sum of the min height of its children. */
2393 while (c)
2394 {
2395 size += window_min_size_1 (c, width_p);
2396 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2397 }
2398 }
2399 }
c1636aa6
GM
2400 else
2401 {
233a4a2c
GM
2402 if (width_p)
2403 size = window_min_width;
c1636aa6 2404 else
233a4a2c
GM
2405 {
2406 if (MINI_WINDOW_P (w)
2407 || (!WINDOW_WANTS_MODELINE_P (w)
045dee35 2408 && !WINDOW_WANTS_HEADER_LINE_P (w)))
233a4a2c
GM
2409 size = 1;
2410 else
2411 size = window_min_height;
2412 }
c1636aa6
GM
2413 }
2414
2415 return size;
2416}
2417
2418
233a4a2c
GM
2419/* Return the minimum size of window W, taking fixed-size windows into
2420 account. WIDTH_P non-zero means return the minimum width,
f984d4fc
GM
2421 otherwise return the minimum height. IGNORE_FIXED_P non-zero means
2422 ignore if W is fixed-size. Set *FIXED to 1 if W is fixed-size
2423 unless FIXED is null. */
7ab12479 2424
233a4a2c 2425static int
f984d4fc 2426window_min_size (w, width_p, ignore_fixed_p, fixed)
233a4a2c 2427 struct window *w;
f984d4fc 2428 int width_p, ignore_fixed_p, *fixed;
233a4a2c
GM
2429{
2430 int size, fixed_p;
2431
f984d4fc
GM
2432 if (ignore_fixed_p)
2433 fixed_p = 0;
2434 else
2435 fixed_p = window_fixed_size_p (w, width_p, 1);
2436
233a4a2c
GM
2437 if (fixed)
2438 *fixed = fixed_p;
2439
2440 if (fixed_p)
2441 size = width_p ? XFASTINT (w->width) : XFASTINT (w->height);
2442 else
2443 size = window_min_size_1 (w, width_p);
f984d4fc 2444
233a4a2c
GM
2445 return size;
2446}
2447
2448
2449/* Set WINDOW's height or width to SIZE. WIDTH_P non-zero means set
2450 WINDOW's width. Resize WINDOW's children, if any, so that they
2451 keep their proportionate size relative to WINDOW. Propagate
2452 WINDOW's top or left edge position to children. Delete windows
192c3131
RS
2453 that become too small unless NODELETE_P is non-zero.
2454
2455 If NODELETE_P is 2, that means we do delete windows that are
2456 too small, even if they were too small before! */
233a4a2c
GM
2457
2458static void
2459size_window (window, size, width_p, nodelete_p)
7ab12479 2460 Lisp_Object window;
233a4a2c 2461 int size, width_p, nodelete_p;
7ab12479 2462{
233a4a2c
GM
2463 struct window *w = XWINDOW (window);
2464 struct window *c;
2465 Lisp_Object child, *forward, *sideward;
2466 int old_size, min_size;
7ab12479 2467
192c3131
RS
2468 if (nodelete_p == 2)
2469 nodelete_p = 0;
2470
a481b3ea 2471 check_min_window_sizes ();
7ae2f10f 2472 size = max (0, size);
233a4a2c 2473
b5f05b50
GM
2474 /* If the window has been "too small" at one point,
2475 don't delete it for being "too small" in the future.
2476 Preserve it as long as that is at all possible. */
233a4a2c
GM
2477 if (width_p)
2478 {
8b6d9dc9 2479 old_size = XINT (w->width);
233a4a2c
GM
2480 min_size = window_min_width;
2481 }
2482 else
2483 {
8b6d9dc9 2484 old_size = XINT (w->height);
233a4a2c
GM
2485 min_size = window_min_height;
2486 }
192c3131
RS
2487
2488 if (old_size < min_size && nodelete_p != 2)
b5f05b50
GM
2489 w->too_small_ok = Qt;
2490
233a4a2c 2491 /* Maybe delete WINDOW if it's too small. */
192c3131 2492 if (nodelete_p != 1 && !NILP (w->parent))
7ab12479 2493 {
b5f05b50 2494 if (!MINI_WINDOW_P (w) && !NILP (w->too_small_ok))
233a4a2c 2495 min_size = width_p ? MIN_SAFE_WINDOW_WIDTH : MIN_SAFE_WINDOW_HEIGHT;
b5f05b50 2496 else
233a4a2c 2497 min_size = width_p ? window_min_width : window_min_height;
b5f05b50 2498
233a4a2c 2499 if (size < min_size)
c1636aa6
GM
2500 {
2501 delete_window (window);
2502 return;
2503 }
7ab12479
JB
2504 }
2505
233a4a2c 2506 /* Set redisplay hints. */
7ae2f10f
GM
2507 w->last_modified = make_number (0);
2508 w->last_overlay_modified = make_number (0);
7ab12479 2509 windows_or_buffers_changed++;
7ae2f10f 2510 FRAME_WINDOW_SIZES_CHANGED (XFRAME (w->frame)) = 1;
29aeee73 2511
233a4a2c
GM
2512 if (width_p)
2513 {
2514 sideward = &w->vchild;
2515 forward = &w->hchild;
7ae2f10f 2516 w->width = make_number (size);
233a4a2c
GM
2517 }
2518 else
2519 {
2520 sideward = &w->hchild;
2521 forward = &w->vchild;
7ae2f10f 2522 w->height = make_number (size);
0130fe1a 2523 w->orig_height = Qnil;
233a4a2c
GM
2524 }
2525
2526 if (!NILP (*sideward))
7ab12479 2527 {
233a4a2c 2528 for (child = *sideward; !NILP (child); child = c->next)
7ab12479 2529 {
233a4a2c
GM
2530 c = XWINDOW (child);
2531 if (width_p)
2532 c->left = w->left;
2533 else
2534 c->top = w->top;
2535 size_window (child, size, width_p, nodelete_p);
7ab12479
JB
2536 }
2537 }
233a4a2c 2538 else if (!NILP (*forward))
7ab12479 2539 {
233a4a2c
GM
2540 int fixed_size, each, extra, n;
2541 int resize_fixed_p, nfixed;
8b6d9dc9 2542 int last_pos, first_pos, nchildren, total;
233a4a2c
GM
2543
2544 /* Determine the fixed-size portion of the this window, and the
2545 number of child windows. */
8b6d9dc9 2546 fixed_size = nchildren = nfixed = total = 0;
233a4a2c 2547 for (child = *forward; !NILP (child); child = c->next, ++nchildren)
7ab12479 2548 {
8b6d9dc9
GM
2549 int child_size;
2550
7ab12479 2551 c = XWINDOW (child);
8b6d9dc9
GM
2552 child_size = width_p ? XINT (c->width) : XINT (c->height);
2553 total += child_size;
2554
233a4a2c
GM
2555 if (window_fixed_size_p (c, width_p, 0))
2556 {
8b6d9dc9 2557 fixed_size += child_size;
233a4a2c
GM
2558 ++nfixed;
2559 }
2560 }
7ab12479 2561
233a4a2c
GM
2562 /* If the new size is smaller than fixed_size, or if there
2563 aren't any resizable windows, allow resizing fixed-size
2564 windows. */
2565 resize_fixed_p = nfixed == nchildren || size < fixed_size;
2566
2567 /* Compute how many lines/columns to add to each child. The
2568 value of extra takes care of rounding errors. */
2569 n = resize_fixed_p ? nchildren : nchildren - nfixed;
8b6d9dc9
GM
2570 each = (size - total) / n;
2571 extra = (size - total) - n * each;
233a4a2c
GM
2572
2573 /* Compute new children heights and edge positions. */
8b6d9dc9 2574 first_pos = width_p ? XINT (w->left) : XINT (w->top);
233a4a2c
GM
2575 last_pos = first_pos;
2576 for (child = *forward; !NILP (child); child = c->next)
2577 {
2578 int new_size, old_size;
2579
2580 c = XWINDOW (child);
2581 old_size = width_p ? XFASTINT (c->width) : XFASTINT (c->height);
2582 new_size = old_size;
7ab12479 2583
233a4a2c
GM
2584 /* The top or left edge position of this child equals the
2585 bottom or right edge of its predecessor. */
2586 if (width_p)
2587 c->left = make_number (last_pos);
2588 else
2589 c->top = make_number (last_pos);
7ab12479 2590
233a4a2c
GM
2591 /* If this child can be resized, do it. */
2592 if (resize_fixed_p || !window_fixed_size_p (c, width_p, 0))
2593 {
2594 new_size = old_size + each + extra;
2595 extra = 0;
2596 }
2597
2598 /* Set new height. Note that size_window also propagates
2599 edge positions to children, so it's not a no-op if we
2600 didn't change the child's size. */
2601 size_window (child, new_size, width_p, 1);
2602
2603 /* Remember the bottom/right edge position of this child; it
2604 will be used to set the top/left edge of the next child. */
2605 last_pos += new_size;
7ab12479 2606 }
233a4a2c
GM
2607
2608 /* We should have covered the parent exactly with child windows. */
2609 xassert (size == last_pos - first_pos);
2610
7ab12479 2611 /* Now delete any children that became too small. */
233a4a2c
GM
2612 if (!nodelete_p)
2613 for (child = *forward; !NILP (child); child = c->next)
7ab12479 2614 {
233a4a2c
GM
2615 int child_size;
2616 c = XWINDOW (child);
8b6d9dc9 2617 child_size = width_p ? XINT (c->width) : XINT (c->height);
192c3131 2618 size_window (child, child_size, width_p, 2);
7ab12479
JB
2619 }
2620 }
2621}
2622
233a4a2c
GM
2623/* Set WINDOW's height to HEIGHT, and recursively change the height of
2624 WINDOW's children. NODELETE non-zero means don't delete windows
2625 that become too small in the process. (The caller should check
2626 later and do so if appropriate.) */
7ab12479 2627
5e14b1fc 2628void
233a4a2c 2629set_window_height (window, height, nodelete)
7ab12479 2630 Lisp_Object window;
233a4a2c 2631 int height;
7ab12479
JB
2632 int nodelete;
2633{
233a4a2c
GM
2634 size_window (window, height, 0, nodelete);
2635}
7ab12479 2636
7ab12479 2637
233a4a2c
GM
2638/* Set WINDOW's width to WIDTH, and recursively change the width of
2639 WINDOW's children. NODELETE non-zero means don't delete windows
2640 that become too small in the process. (The caller should check
2641 later and do so if appropriate.) */
7ab12479 2642
233a4a2c
GM
2643void
2644set_window_width (window, width, nodelete)
2645 Lisp_Object window;
2646 int width;
2647 int nodelete;
2648{
2649 size_window (window, width, 1, nodelete);
7ab12479 2650}
233a4a2c 2651
7ab12479 2652\f
1d8d96fa 2653int window_select_count;
7ab12479 2654
5b03d3c0
RS
2655Lisp_Object
2656Fset_window_buffer_unwind (obuf)
2657 Lisp_Object obuf;
2658{
2659 Fset_buffer (obuf);
2660 return Qnil;
2661}
2662
7ab12479 2663
5500c422
GM
2664/* Make WINDOW display BUFFER as its contents. RUN_HOOKS_P non-zero
2665 means it's allowed to run hooks. See make_frame for a case where
2666 it's not allowed. */
7ab12479 2667
5500c422
GM
2668void
2669set_window_buffer (window, buffer, run_hooks_p)
2670 Lisp_Object window, buffer;
2671 int run_hooks_p;
2672{
2673 struct window *w = XWINDOW (window);
2674 struct buffer *b = XBUFFER (buffer);
aed13378 2675 int count = SPECPDL_INDEX ();
7ab12479
JB
2676
2677 w->buffer = buffer;
86e48436
RS
2678
2679 if (EQ (window, selected_window))
5500c422 2680 b->last_selected_window = window;
beb4e312
RS
2681
2682 /* Update time stamps of buffer display. */
5500c422
GM
2683 if (INTEGERP (b->display_count))
2684 XSETINT (b->display_count, XINT (b->display_count) + 1);
2685 b->display_time = Fcurrent_time ();
86e48436 2686
d834a2e9 2687 XSETFASTINT (w->window_end_pos, 0);
5500c422
GM
2688 XSETFASTINT (w->window_end_vpos, 0);
2689 bzero (&w->last_cursor, sizeof w->last_cursor);
5a41ab94 2690 w->window_end_valid = Qnil;
1c686c99 2691 w->hscroll = w->min_hscroll = make_number (0);
224227d1 2692 w->vscroll = 0;
5500c422 2693 set_marker_both (w->pointm, buffer, BUF_PT (b), BUF_PT_BYTE (b));
7ab12479 2694 set_marker_restricted (w->start,
5500c422 2695 make_number (b->last_window_start),
7ab12479
JB
2696 buffer);
2697 w->start_at_line_beg = Qnil;
e36ab06b 2698 w->force_start = Qnil;
d834a2e9 2699 XSETFASTINT (w->last_modified, 0);
3cd21523 2700 XSETFASTINT (w->last_overlay_modified, 0);
7ab12479 2701 windows_or_buffers_changed++;
5b03d3c0
RS
2702
2703 /* We must select BUFFER for running the window-scroll-functions.
2704 If WINDOW is selected, switch permanently.
2705 Otherwise, switch but go back to the ambient buffer afterward. */
7ab12479
JB
2706 if (EQ (window, selected_window))
2707 Fset_buffer (buffer);
5b03d3c0
RS
2708 /* We can't check ! NILP (Vwindow_scroll_functions) here
2709 because that might itself be a local variable. */
2710 else if (window_initialized)
2711 {
2712 record_unwind_protect (Fset_window_buffer_unwind, Fcurrent_buffer ());
2713 Fset_buffer (buffer);
2714 }
2715
5500c422 2716 /* Set left and right marginal area width from buffer. */
cfa22082 2717 Fset_window_margins (window, b->left_margin_width, b->right_margin_width);
7ab12479 2718
5500c422
GM
2719 if (run_hooks_p)
2720 {
2721 if (! NILP (Vwindow_scroll_functions))
2722 run_hook_with_args_2 (Qwindow_scroll_functions, window,
2723 Fmarker_position (w->start));
2724
2725 if (! NILP (Vwindow_configuration_change_hook)
2726 && ! NILP (Vrun_hooks))
2727 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
2728 }
543f5fb1 2729
5b03d3c0 2730 unbind_to (count, Qnil);
5500c422 2731}
5b03d3c0 2732
5500c422
GM
2733
2734DEFUN ("set-window-buffer", Fset_window_buffer, Sset_window_buffer, 2, 2, 0,
fdb82f93
PJ
2735 doc: /* Make WINDOW display BUFFER as its contents.
2736BUFFER can be a buffer or buffer name. */)
2737 (window, buffer)
5500c422
GM
2738 register Lisp_Object window, buffer;
2739{
2740 register Lisp_Object tem;
2741 register struct window *w = decode_window (window);
5500c422 2742
bed0c171 2743 XSETWINDOW (window, w);
5500c422 2744 buffer = Fget_buffer (buffer);
b7826503 2745 CHECK_BUFFER (buffer);
5500c422
GM
2746
2747 if (NILP (XBUFFER (buffer)->name))
2748 error ("Attempt to display deleted buffer");
2749
2750 tem = w->buffer;
2751 if (NILP (tem))
2752 error ("Window is deleted");
2753 else if (! EQ (tem, Qt)) /* w->buffer is t when the window
2754 is first being set up. */
2755 {
2756 if (!NILP (w->dedicated) && !EQ (tem, buffer))
2757 error ("Window is dedicated to `%s'",
d5db4077 2758 SDATA (XBUFFER (tem)->name));
5500c422
GM
2759
2760 unshow_buffer (w);
2761 }
2762
2763 set_window_buffer (window, buffer, 1);
7ab12479
JB
2764 return Qnil;
2765}
2766
2767DEFUN ("select-window", Fselect_window, Sselect_window, 1, 1, 0,
fdb82f93
PJ
2768 doc: /* Select WINDOW. Most editing will apply to WINDOW's buffer.
2769If WINDOW is not already selected, also make WINDOW's buffer current.
282f7831
RS
2770If WINDOW's frame is the selected frame, also make WINDOW the frame's
2771selected window.
2772
fdb82f93
PJ
2773Note that the main editor command loop
2774selects the buffer of the selected window before each command. */)
2775 (window)
7ab12479 2776 register Lisp_Object window;
b7354ddf
RS
2777{
2778 return select_window_1 (window, 1);
2779}
2780\f
719eaeb1
GM
2781/* Note that selected_window can be nil
2782 when this is called from Fset_window_configuration. */
2783
b7354ddf
RS
2784static Lisp_Object
2785select_window_1 (window, recordflag)
2786 register Lisp_Object window;
2787 int recordflag;
7ab12479
JB
2788{
2789 register struct window *w;
719eaeb1 2790 register struct window *ow;
1ae1a37d 2791 struct frame *sf;
7ab12479 2792
b7826503 2793 CHECK_LIVE_WINDOW (window);
7ab12479
JB
2794
2795 w = XWINDOW (window);
50e88778 2796 w->frozen_window_start_p = 0;
7ab12479 2797
d834a2e9 2798 XSETFASTINT (w->use_time, ++window_select_count);
7ab12479
JB
2799 if (EQ (window, selected_window))
2800 return window;
2801
719eaeb1
GM
2802 if (!NILP (selected_window))
2803 {
2804 ow = XWINDOW (selected_window);
2805 if (! NILP (ow->buffer))
2806 set_marker_both (ow->pointm, ow->buffer,
2807 BUF_PT (XBUFFER (ow->buffer)),
2808 BUF_PT_BYTE (XBUFFER (ow->buffer)));
2809 }
7ab12479
JB
2810
2811 selected_window = window;
1ae1a37d 2812 sf = SELECTED_FRAME ();
282f7831 2813 if (XFRAME (WINDOW_FRAME (w)) == sf)
1ae1a37d 2814 sf->selected_window = window;
7ab12479 2815
b7354ddf
RS
2816 if (recordflag)
2817 record_buffer (w->buffer);
7ab12479
JB
2818 Fset_buffer (w->buffer);
2819
86e48436
RS
2820 XBUFFER (w->buffer)->last_selected_window = window;
2821
7ab12479
JB
2822 /* Go to the point recorded in the window.
2823 This is important when the buffer is in more
2824 than one window. It also matters when
2825 redisplay_window has altered point after scrolling,
2826 because it makes the change only in the window. */
2827 {
2828 register int new_point = marker_position (w->pointm);
2829 if (new_point < BEGV)
2830 SET_PT (BEGV);
a9c95e08 2831 else if (new_point > ZV)
7ab12479
JB
2832 SET_PT (ZV);
2833 else
2834 SET_PT (new_point);
2835 }
2836
2837 windows_or_buffers_changed++;
2838 return window;
2839}
b7354ddf 2840\f
441a127e
RS
2841/* Deiconify the frame containing the window WINDOW,
2842 unless it is the selected frame;
2843 then return WINDOW.
2844
2845 The reason for the exception for the selected frame
2846 is that it seems better not to change the selected frames visibility
2847 merely because of displaying a different buffer in it.
2848 The deiconification is useful when a buffer gets shown in
2849 another frame that you were not using lately. */
d07f802a
RS
2850
2851static Lisp_Object
2852display_buffer_1 (window)
2853 Lisp_Object window;
2854{
1ae1a37d
GM
2855 Lisp_Object frame = XWINDOW (window)->frame;
2856 FRAME_PTR f = XFRAME (frame);
2857
d07f802a 2858 FRAME_SAMPLE_VISIBILITY (f);
1ae1a37d
GM
2859
2860 if (!EQ (frame, selected_frame))
60117126
RS
2861 {
2862 if (FRAME_ICONIFIED_P (f))
1ae1a37d 2863 Fmake_frame_visible (frame);
37962e60 2864 else if (FRAME_VISIBLE_P (f))
1ae1a37d 2865 Fraise_frame (frame);
60117126 2866 }
1ae1a37d 2867
d07f802a
RS
2868 return window;
2869}
2870
4628f7a4 2871DEFUN ("special-display-p", Fspecial_display_p, Sspecial_display_p, 1, 1, 0,
fdb82f93
PJ
2872 doc: /* Returns non-nil if a buffer named BUFFER-NAME would be created specially.
2873The value is actually t if the frame should be called with default frame
2874parameters, and a list of frame parameters if they were specified.
2875See `special-display-buffer-names', and `special-display-regexps'. */)
2876 (buffer_name)
4628f7a4
EN
2877 Lisp_Object buffer_name;
2878{
2879 Lisp_Object tem;
2880
b7826503 2881 CHECK_STRING (buffer_name);
4628f7a4
EN
2882
2883 tem = Fmember (buffer_name, Vspecial_display_buffer_names);
2884 if (!NILP (tem))
2885 return Qt;
2886
2887 tem = Fassoc (buffer_name, Vspecial_display_buffer_names);
2888 if (!NILP (tem))
2889 return XCDR (tem);
2890
2891 for (tem = Vspecial_display_regexps; CONSP (tem); tem = XCDR (tem))
2892 {
2893 Lisp_Object car = XCAR (tem);
2894 if (STRINGP (car)
2895 && fast_string_match (car, buffer_name) >= 0)
2896 return Qt;
2897 else if (CONSP (car)
2898 && STRINGP (XCAR (car))
2899 && fast_string_match (XCAR (car), buffer_name) >= 0)
0057b00a 2900 return XCDR (car);
4628f7a4
EN
2901 }
2902 return Qnil;
2903}
2904
2905DEFUN ("same-window-p", Fsame_window_p, Ssame_window_p, 1, 1, 0,
fdb82f93
PJ
2906 doc: /* Returns non-nil if a new buffer named BUFFER-NAME would use the same window.
2907See `same-window-buffer-names' and `same-window-regexps'. */)
2908 (buffer_name)
4628f7a4
EN
2909 Lisp_Object buffer_name;
2910{
2911 Lisp_Object tem;
2912
b7826503 2913 CHECK_STRING (buffer_name);
4628f7a4
EN
2914
2915 tem = Fmember (buffer_name, Vsame_window_buffer_names);
2916 if (!NILP (tem))
2917 return Qt;
2918
2919 tem = Fassoc (buffer_name, Vsame_window_buffer_names);
2920 if (!NILP (tem))
2921 return Qt;
2922
2923 for (tem = Vsame_window_regexps; CONSP (tem); tem = XCDR (tem))
2924 {
2925 Lisp_Object car = XCAR (tem);
2926 if (STRINGP (car)
2927 && fast_string_match (car, buffer_name) >= 0)
2928 return Qt;
2929 else if (CONSP (car)
2930 && STRINGP (XCAR (car))
2931 && fast_string_match (XCAR (car), buffer_name) >= 0)
2932 return Qt;
2933 }
2934 return Qnil;
2935}
2936
fdb82f93 2937/* Use B so the default is (other-buffer). */
53f76081 2938DEFUN ("display-buffer", Fdisplay_buffer, Sdisplay_buffer, 1, 3,
fdb82f93
PJ
2939 "BDisplay buffer: \nP",
2940 doc: /* Make BUFFER appear in some window but don't select it.
2941BUFFER can be a buffer or a buffer name.
2942If BUFFER is shown already in some window, just use that one,
2943unless the window is the selected window and the optional second
2944argument NOT-THIS-WINDOW is non-nil (interactively, with prefix arg).
2945If `pop-up-frames' is non-nil, make a new frame if no window shows BUFFER.
2946Returns the window displaying BUFFER.
079d288f 2947If `display-buffer-reuse-frames' is non-nil, and another frame is currently
fdb82f93
PJ
2948displaying BUFFER, then simply raise that frame.
2949
2950The variables `special-display-buffer-names', `special-display-regexps',
2951`same-window-buffer-names', and `same-window-regexps' customize how certain
2952buffer names are handled.
2953
2954If optional argument FRAME is `visible', search all visible frames.
2955If FRAME is 0, search all visible and iconified frames.
2956If FRAME is t, search all frames.
2957If FRAME is a frame, search only that frame.
2958If FRAME is nil, search only the selected frame
2959 (actually the last nonminibuffer frame),
2960 unless `pop-up-frames' or `display-buffer-reuse-frames' is non-nil,
2961 which means search visible and iconified frames.
2962
2963If `even-window-heights' is non-nil, window heights will be evened out
2964if displaying the buffer causes two vertically adjacent windows to be
2965displayed. */)
2966 (buffer, not_this_window, frame)
53f76081 2967 register Lisp_Object buffer, not_this_window, frame;
7ab12479 2968{
aee631c2 2969 register Lisp_Object window, tem, swp;
1ae1a37d 2970 struct frame *f;
7ab12479 2971
aee631c2 2972 swp = Qnil;
7ab12479 2973 buffer = Fget_buffer (buffer);
b7826503 2974 CHECK_BUFFER (buffer);
7ab12479 2975
265a9e55 2976 if (!NILP (Vdisplay_buffer_function))
7ab12479
JB
2977 return call2 (Vdisplay_buffer_function, buffer, not_this_window);
2978
265a9e55 2979 if (NILP (not_this_window)
7ab12479 2980 && XBUFFER (XWINDOW (selected_window)->buffer) == XBUFFER (buffer))
d07f802a 2981 return display_buffer_1 (selected_window);
7ab12479 2982
855d8627
RS
2983 /* See if the user has specified this buffer should appear
2984 in the selected window. */
2985 if (NILP (not_this_window))
2986 {
aee631c2
RS
2987 swp = Fsame_window_p (XBUFFER (buffer)->name);
2988 if (!NILP (swp) && !no_switch_window (selected_window))
c63dc4a2
RS
2989 {
2990 Fswitch_to_buffer (buffer, Qnil);
d07f802a 2991 return display_buffer_1 (selected_window);
c63dc4a2 2992 }
855d8627
RS
2993 }
2994
079d288f 2995 /* If the user wants pop-up-frames or display-buffer-reuse-frames,
73dc5198
KH
2996 look for a window showing BUFFER on any visible or iconified frame.
2997 Otherwise search only the current frame. */
53f76081
RS
2998 if (! NILP (frame))
2999 tem = frame;
9c3da604
GM
3000 else if (pop_up_frames
3001 || display_buffer_reuse_frames
3002 || last_nonminibuf_frame == 0)
73dc5198
KH
3003 XSETFASTINT (tem, 0);
3004 else
73dc5198 3005 XSETFRAME (tem, last_nonminibuf_frame);
9c3da604 3006
73dc5198 3007 window = Fget_buffer_window (buffer, tem);
265a9e55
JB
3008 if (!NILP (window)
3009 && (NILP (not_this_window) || !EQ (window, selected_window)))
9c3da604 3010 return display_buffer_1 (window);
7ab12479 3011
a90712c2 3012 /* Certain buffer names get special handling. */
aee631c2 3013 if (!NILP (Vspecial_display_function) && NILP (swp))
a90712c2 3014 {
4628f7a4
EN
3015 tem = Fspecial_display_p (XBUFFER (buffer)->name);
3016 if (EQ (tem, Qt))
a90712c2 3017 return call1 (Vspecial_display_function, buffer);
4628f7a4
EN
3018 if (CONSP (tem))
3019 return call2 (Vspecial_display_function, buffer, tem);
a90712c2
RS
3020 }
3021
44fa5b1e
JB
3022 /* If there are no frames open that have more than a minibuffer,
3023 we need to create a new frame. */
3024 if (pop_up_frames || last_nonminibuf_frame == 0)
7ab12479 3025 {
a90712c2 3026 window = Fframe_selected_window (call0 (Vpop_up_frame_function));
7ab12479 3027 Fset_window_buffer (window, buffer);
d07f802a 3028 return display_buffer_1 (window);
7ab12479 3029 }
7ab12479 3030
1ae1a37d 3031 f = SELECTED_FRAME ();
43bad991 3032 if (pop_up_windows
1ae1a37d 3033 || FRAME_MINIBUF_ONLY_P (f)
cee67da9
RS
3034 /* If the current frame is a special display frame,
3035 don't try to reuse its windows. */
1ae1a37d 3036 || !NILP (XWINDOW (FRAME_ROOT_WINDOW (f))->dedicated))
7ab12479 3037 {
12cae7c0
KH
3038 Lisp_Object frames;
3039
37962e60 3040 frames = Qnil;
1ae1a37d 3041 if (FRAME_MINIBUF_ONLY_P (f))
74112613 3042 XSETFRAME (frames, last_nonminibuf_frame);
a5731348 3043 /* Don't try to create a window if we would get an error. */
7ab12479
JB
3044 if (split_height_threshold < window_min_height << 1)
3045 split_height_threshold = window_min_height << 1;
3046
cee67da9
RS
3047 /* Note that both Fget_largest_window and Fget_lru_window
3048 ignore minibuffers and dedicated windows.
3049 This means they can return nil. */
7ab12479 3050
cee67da9
RS
3051 /* If the frame we would try to split cannot be split,
3052 try other frames. */
1ae1a37d 3053 if (FRAME_NO_SPLIT_P (NILP (frames) ? f : last_nonminibuf_frame))
cee67da9
RS
3054 {
3055 /* Try visible frames first. */
3056 window = Fget_largest_window (Qvisible);
3057 /* If that didn't work, try iconified frames. */
3058 if (NILP (window))
3059 window = Fget_largest_window (make_number (0));
3060 if (NILP (window))
3061 window = Fget_largest_window (Qt);
3062 }
3063 else
3064 window = Fget_largest_window (frames);
3065
92cca945
RS
3066 /* If we got a tall enough full-width window that can be split,
3067 split it. */
265a9e55 3068 if (!NILP (window)
92cca945 3069 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window)->frame))
7ab12479 3070 && window_height (window) >= split_height_threshold
111e5992 3071 && WINDOW_FULL_WIDTH_P (XWINDOW (window)))
7ab12479
JB
3072 window = Fsplit_window (window, Qnil, Qnil);
3073 else
3074 {
1942f68f
RS
3075 Lisp_Object upper, lower, other;
3076
44fa5b1e 3077 window = Fget_lru_window (frames);
92cca945
RS
3078 /* If the LRU window is selected, and big enough,
3079 and can be split, split it. */
cee67da9 3080 if (!NILP (window)
92cca945 3081 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window)->frame))
cee67da9
RS
3082 && (EQ (window, selected_window)
3083 || EQ (XWINDOW (window)->parent, Qnil))
7ab12479
JB
3084 && window_height (window) >= window_min_height << 1)
3085 window = Fsplit_window (window, Qnil, Qnil);
cee67da9 3086 /* If Fget_lru_window returned nil, try other approaches. */
48d9379d 3087
cee67da9 3088 /* Try visible frames first. */
48d9379d
RS
3089 if (NILP (window))
3090 window = Fget_buffer_window (buffer, Qvisible);
cee67da9
RS
3091 if (NILP (window))
3092 window = Fget_largest_window (Qvisible);
3093 /* If that didn't work, try iconified frames. */
48d9379d
RS
3094 if (NILP (window))
3095 window = Fget_buffer_window (buffer, make_number (0));
cee67da9
RS
3096 if (NILP (window))
3097 window = Fget_largest_window (make_number (0));
3098 /* Try invisible frames. */
48d9379d
RS
3099 if (NILP (window))
3100 window = Fget_buffer_window (buffer, Qt);
cee67da9
RS
3101 if (NILP (window))
3102 window = Fget_largest_window (Qt);
3103 /* As a last resort, make a new frame. */
3104 if (NILP (window))
3105 window = Fframe_selected_window (call0 (Vpop_up_frame_function));
1942f68f
RS
3106 /* If window appears above or below another,
3107 even out their heights. */
cac66e4f 3108 other = upper = lower = Qnil;
1942f68f
RS
3109 if (!NILP (XWINDOW (window)->prev))
3110 other = upper = XWINDOW (window)->prev, lower = window;
3111 if (!NILP (XWINDOW (window)->next))
3112 other = lower = XWINDOW (window)->next, upper = window;
3113 if (!NILP (other)
6529ed87 3114 && !NILP (Veven_window_heights)
1942f68f 3115 /* Check that OTHER and WINDOW are vertically arrayed. */
296b535c
KH
3116 && !EQ (XWINDOW (other)->top, XWINDOW (window)->top)
3117 && (XFASTINT (XWINDOW (other)->height)
3118 > XFASTINT (XWINDOW (window)->height)))
1942f68f 3119 {
296b535c
KH
3120 int total = (XFASTINT (XWINDOW (other)->height)
3121 + XFASTINT (XWINDOW (window)->height));
86c8e823
GM
3122 enlarge_window (upper,
3123 total / 2 - XFASTINT (XWINDOW (upper)->height),
f95284d2 3124 0, 0);
1942f68f 3125 }
7ab12479
JB
3126 }
3127 }
3128 else
3129 window = Fget_lru_window (Qnil);
3130
3131 Fset_window_buffer (window, buffer);
d07f802a 3132 return display_buffer_1 (window);
7ab12479
JB
3133}
3134
3135void
3136temp_output_buffer_show (buf)
3137 register Lisp_Object buf;
3138{
3139 register struct buffer *old = current_buffer;
3140 register Lisp_Object window;
3141 register struct window *w;
3142
bccd3dd1
RS
3143 XBUFFER (buf)->directory = current_buffer->directory;
3144
7ab12479 3145 Fset_buffer (buf);
c6367666 3146 BUF_SAVE_MODIFF (XBUFFER (buf)) = MODIFF;
7ab12479
JB
3147 BEGV = BEG;
3148 ZV = Z;
3149 SET_PT (BEG);
06be4f85 3150#if 0 /* rms: there should be no reason for this. */
b1599b4c 3151 XBUFFER (buf)->prevent_redisplay_optimizations_p = 1;
06be4f85 3152#endif
7ab12479
JB
3153 set_buffer_internal (old);
3154
3155 if (!EQ (Vtemp_buffer_show_function, Qnil))
3156 call1 (Vtemp_buffer_show_function, buf);
3157 else
3158 {
53f76081 3159 window = Fdisplay_buffer (buf, Qnil, Qnil);
7ab12479 3160
1ae1a37d 3161 if (!EQ (XWINDOW (window)->frame, selected_frame))
44fa5b1e 3162 Fmake_frame_visible (WINDOW_FRAME (XWINDOW (window)));
7ab12479
JB
3163 Vminibuf_scroll_window = window;
3164 w = XWINDOW (window);
d834a2e9 3165 XSETFASTINT (w->hscroll, 0);
ea68264b 3166 XSETFASTINT (w->min_hscroll, 0);
b73ea88e
RS
3167 set_marker_restricted_both (w->start, buf, 1, 1);
3168 set_marker_restricted_both (w->pointm, buf, 1, 1);
a58ec57d 3169
beb4e312 3170 /* Run temp-buffer-show-hook, with the chosen window selected
a5731348 3171 and its buffer current. */
f52cca03 3172 if (!NILP (Vrun_hooks))
2cccc823 3173 {
f52cca03
RS
3174 Lisp_Object tem;
3175 tem = Fboundp (Qtemp_buffer_show_hook);
2cccc823
RS
3176 if (!NILP (tem))
3177 {
f52cca03
RS
3178 tem = Fsymbol_value (Qtemp_buffer_show_hook);
3179 if (!NILP (tem))
3180 {
aed13378 3181 int count = SPECPDL_INDEX ();
b7354ddf
RS
3182 Lisp_Object prev_window;
3183 prev_window = selected_window;
2cccc823 3184
f52cca03 3185 /* Select the window that was chosen, for running the hook. */
a5731348
SM
3186 /* Both this Fselect_window and the select_window_1
3187 below will (may) incorrectly set-buffer to the buffer
3188 displayed in the window. --stef */
65a04b96 3189 record_unwind_protect (Fselect_window, prev_window);
b7354ddf 3190 select_window_1 (window, 0);
beb4e312 3191 Fset_buffer (w->buffer);
f52cca03 3192 call1 (Vrun_hooks, Qtemp_buffer_show_hook);
b7354ddf 3193 select_window_1 (prev_window, 0);
f52cca03
RS
3194 unbind_to (count, Qnil);
3195 }
2cccc823
RS
3196 }
3197 }
3198 }
7ab12479
JB
3199}
3200\f
dfcf069d 3201static void
7ab12479
JB
3202make_dummy_parent (window)
3203 Lisp_Object window;
3204{
cffec418 3205 Lisp_Object new;
7ab12479 3206 register struct window *o, *p;
cffec418 3207 int i;
7ab12479 3208
cffec418 3209 o = XWINDOW (window);
26605be9 3210 p = allocate_window ();
cffec418 3211 for (i = 0; i < VECSIZE (struct window); ++i)
26605be9
GM
3212 ((struct Lisp_Vector *) p)->contents[i]
3213 = ((struct Lisp_Vector *)o)->contents[i];
cffec418 3214 XSETWINDOW (new, p);
7ab12479 3215
d834a2e9 3216 XSETFASTINT (p->sequence_number, ++sequence_number);
7ab12479
JB
3217
3218 /* Put new into window structure in place of window */
3219 replace_window (window, new);
3220
3221 o->next = Qnil;
3222 o->prev = Qnil;
3223 o->vchild = Qnil;
3224 o->hchild = Qnil;
3225 o->parent = new;
3226
3227 p->start = Qnil;
3228 p->pointm = Qnil;
3229 p->buffer = Qnil;
3230}
3231
3232DEFUN ("split-window", Fsplit_window, Ssplit_window, 0, 3, "",
fdb82f93
PJ
3233 doc: /* Split WINDOW, putting SIZE lines in the first of the pair.
3234WINDOW defaults to selected one and SIZE to half its size.
3235If optional third arg HORFLAG is non-nil, split side by side
3236and put SIZE columns in the first of the pair. In that case,
3237SIZE includes that window's scroll bar, or the divider column to its right. */)
3238 (window, size, horflag)
77ae0fe3 3239 Lisp_Object window, size, horflag;
7ab12479
JB
3240{
3241 register Lisp_Object new;
3242 register struct window *o, *p;
c0807608 3243 FRAME_PTR fo;
77ae0fe3 3244 register int size_int;
7ab12479 3245
265a9e55 3246 if (NILP (window))
7ab12479
JB
3247 window = selected_window;
3248 else
b7826503 3249 CHECK_LIVE_WINDOW (window);
7ab12479
JB
3250
3251 o = XWINDOW (window);
c0807608 3252 fo = XFRAME (WINDOW_FRAME (o));
7ab12479 3253
77ae0fe3 3254 if (NILP (size))
7ab12479 3255 {
265a9e55 3256 if (!NILP (horflag))
c0807608 3257 /* Calculate the size of the left-hand window, by dividing
25b33244
KH
3258 the usable space in columns by two.
3259 We round up, since the left-hand window may include
3260 a dividing line, while the right-hand may not. */
3261 size_int = (XFASTINT (o->width) + 1) >> 1;
7ab12479 3262 else
77ae0fe3 3263 size_int = XFASTINT (o->height) >> 1;
7ab12479
JB
3264 }
3265 else
3266 {
b7826503 3267 CHECK_NUMBER (size);
77ae0fe3 3268 size_int = XINT (size);
7ab12479
JB
3269 }
3270
3271 if (MINI_WINDOW_P (o))
3272 error ("Attempt to split minibuffer window");
233a4a2c
GM
3273 else if (window_fixed_size_p (o, !NILP (horflag), 0))
3274 error ("Attempt to split fixed-size window");
7ab12479 3275
a481b3ea 3276 check_min_window_sizes ();
7ab12479 3277
265a9e55 3278 if (NILP (horflag))
7ab12479 3279 {
77ae0fe3
KH
3280 if (size_int < window_min_height)
3281 error ("Window height %d too small (after splitting)", size_int);
3282 if (size_int + window_min_height > XFASTINT (o->height))
37962e60 3283 error ("Window height %d too small (after splitting)",
77ae0fe3 3284 XFASTINT (o->height) - size_int);
265a9e55
JB
3285 if (NILP (o->parent)
3286 || NILP (XWINDOW (o->parent)->vchild))
7ab12479
JB
3287 {
3288 make_dummy_parent (window);
3289 new = o->parent;
3290 XWINDOW (new)->vchild = window;
3291 }
3292 }
3293 else
3294 {
77ae0fe3
KH
3295 if (size_int < window_min_width)
3296 error ("Window width %d too small (after splitting)", size_int);
a59fed7e
RS
3297
3298 if (size_int + window_min_width > XFASTINT (o->width))
37962e60 3299 error ("Window width %d too small (after splitting)",
a59fed7e 3300 XFASTINT (o->width) - size_int);
265a9e55
JB
3301 if (NILP (o->parent)
3302 || NILP (XWINDOW (o->parent)->hchild))
7ab12479
JB
3303 {
3304 make_dummy_parent (window);
3305 new = o->parent;
3306 XWINDOW (new)->hchild = window;
3307 }
3308 }
3309
3310 /* Now we know that window's parent is a vertical combination
3311 if we are dividing vertically, or a horizontal combination
3312 if we are making side-by-side windows */
3313
3314 windows_or_buffers_changed++;
c0807608 3315 FRAME_WINDOW_SIZES_CHANGED (fo) = 1;
7ab12479
JB
3316 new = make_window ();
3317 p = XWINDOW (new);
3318
44fa5b1e 3319 p->frame = o->frame;
7ab12479 3320 p->next = o->next;
265a9e55 3321 if (!NILP (p->next))
7ab12479
JB
3322 XWINDOW (p->next)->prev = new;
3323 p->prev = window;
3324 o->next = new;
3325 p->parent = o->parent;
3326 p->buffer = Qt;
5500c422
GM
3327 p->window_end_valid = Qnil;
3328 bzero (&p->last_cursor, sizeof p->last_cursor);
7ab12479 3329
44fa5b1e 3330 /* Apportion the available frame space among the two new windows */
7ab12479 3331
265a9e55 3332 if (!NILP (horflag))
7ab12479
JB
3333 {
3334 p->height = o->height;
3335 p->top = o->top;
a59fed7e 3336 XSETFASTINT (p->width, XFASTINT (o->width) - size_int);
77ae0fe3
KH
3337 XSETFASTINT (o->width, size_int);
3338 XSETFASTINT (p->left, XFASTINT (o->left) + size_int);
7ab12479
JB
3339 }
3340 else
3341 {
3342 p->left = o->left;
3343 p->width = o->width;
77ae0fe3
KH
3344 XSETFASTINT (p->height, XFASTINT (o->height) - size_int);
3345 XSETFASTINT (o->height, size_int);
3346 XSETFASTINT (p->top, XFASTINT (o->top) + size_int);
7ab12479
JB
3347 }
3348
5500c422
GM
3349 /* Adjust glyph matrices. */
3350 adjust_glyphs (fo);
543f5fb1 3351 Fset_window_buffer (new, o->buffer);
7ab12479
JB
3352 return new;
3353}
3354\f
f95284d2 3355DEFUN ("enlarge-window", Fenlarge_window, Senlarge_window, 1, 3, "p",
fdb82f93
PJ
3356 doc: /* Make current window ARG lines bigger.
3357From program, optional second arg non-nil means grow sideways ARG columns.
f95284d2
RS
3358Interactively, if an argument is not given, make the window one line bigger.
3359
3360Optional third arg PRESERVE-BEFORE, if non-nil, means do not change the size
3361of the siblings above or to the left of the selected window. Only
3362siblings to the right or below are changed. */)
3363 (arg, side, preserve_before)
3364 register Lisp_Object arg, side, preserve_before;
7ab12479 3365{
b7826503 3366 CHECK_NUMBER (arg);
f95284d2
RS
3367 enlarge_window (selected_window, XINT (arg), !NILP (side),
3368 !NILP (preserve_before));
543f5fb1
RS
3369
3370 if (! NILP (Vwindow_configuration_change_hook))
3371 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
3372
7ab12479
JB
3373 return Qnil;
3374}
3375
a5731348 3376DEFUN ("shrink-window", Fshrink_window, Sshrink_window, 1, 3, "p",
fdb82f93
PJ
3377 doc: /* Make current window ARG lines smaller.
3378From program, optional second arg non-nil means shrink sideways arg columns.
a5731348
SM
3379Interactively, if an argument is not given, make the window one line smaller.
3380
3381Optional third arg PRESERVE-BEFORE, if non-nil, means do not change the size
3382of the siblings above or to the left of the selected window. Only
3383siblings to the right or below are changed. */)
3384 (arg, side, preserve_before)
3385 register Lisp_Object arg, side, preserve_before;
7ab12479 3386{
b7826503 3387 CHECK_NUMBER (arg);
a5731348
SM
3388 enlarge_window (selected_window, -XINT (arg), !NILP (side),
3389 !NILP (preserve_before));
543f5fb1
RS
3390
3391 if (! NILP (Vwindow_configuration_change_hook))
3392 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
3393
7ab12479
JB
3394 return Qnil;
3395}
3396
3397int
3398window_height (window)
3399 Lisp_Object window;
3400{
3401 register struct window *p = XWINDOW (window);
3402 return XFASTINT (p->height);
3403}
3404
3405int
3406window_width (window)
3407 Lisp_Object window;
3408{
3409 register struct window *p = XWINDOW (window);
3410 return XFASTINT (p->width);
3411}
3412
c1636aa6 3413
7ab12479 3414#define CURBEG(w) \
3578db3c 3415 *(widthflag ? &(XWINDOW (w)->left) : &(XWINDOW (w)->top))
7ab12479
JB
3416
3417#define CURSIZE(w) \
3578db3c 3418 *(widthflag ? &(XWINDOW (w)->width) : &(XWINDOW (w)->height))
7ab12479 3419
233a4a2c 3420
f95284d2 3421/* Enlarge WINDOW by DELTA. WIDTHFLAG non-zero means
233a4a2c 3422 increase its width. Siblings of the selected window are resized to
f95284d2
RS
3423 fulfill the size request. If they become too small in the process,
3424 they will be deleted.
3425
3426 If PRESERVE_BEFORE is nonzero, that means don't alter
3427 the siblings to the left or above WINDOW. */
7ab12479 3428
f984d4fc 3429static void
f95284d2 3430enlarge_window (window, delta, widthflag, preserve_before)
86c8e823 3431 Lisp_Object window;
f95284d2 3432 int delta, widthflag, preserve_before;
7ab12479 3433{
86c8e823 3434 Lisp_Object parent, next, prev;
233a4a2c 3435 struct window *p;
3578db3c
KR
3436 Lisp_Object *sizep;
3437 int maximum;
5e14b1fc
AS
3438 int (*sizefun) P_ ((Lisp_Object))
3439 = widthflag ? window_width : window_height;
233a4a2c 3440 void (*setsizefun) P_ ((Lisp_Object, int, int))
5e14b1fc 3441 = (widthflag ? set_window_width : set_window_height);
7ab12479 3442
233a4a2c
GM
3443 /* Check values of window_min_width and window_min_height for
3444 validity. */
a481b3ea 3445 check_min_window_sizes ();
7ab12479 3446
233a4a2c 3447 /* Give up if this window cannot be resized. */
233a4a2c
GM
3448 if (window_fixed_size_p (XWINDOW (window), widthflag, 1))
3449 error ("Window is not resizable");
3450
3451 /* Find the parent of the selected window. */
7ab12479
JB
3452 while (1)
3453 {
3454 p = XWINDOW (window);
3455 parent = p->parent;
233a4a2c 3456
265a9e55 3457 if (NILP (parent))
7ab12479
JB
3458 {
3459 if (widthflag)
3460 error ("No other window to side of this one");
3461 break;
3462 }
233a4a2c
GM
3463
3464 if (widthflag
3465 ? !NILP (XWINDOW (parent)->hchild)
265a9e55 3466 : !NILP (XWINDOW (parent)->vchild))
7ab12479 3467 break;
233a4a2c 3468
7ab12479
JB
3469 window = parent;
3470 }
3471
05c2896a 3472 sizep = &CURSIZE (window);
7ab12479 3473
7ab12479
JB
3474 {
3475 register int maxdelta;
7ab12479 3476
f95284d2
RS
3477 /* Compute the maximum size increment this window can have. */
3478
3479 if (preserve_before)
3480 {
3481 if (!NILP (parent))
3482 {
3483 maxdelta = (*sizefun) (parent) - XINT (*sizep);
3484 /* Subtract size of siblings before, since we can't take that. */
1ab964d7 3485 maxdelta -= XINT (CURBEG (window)) - XINT (CURBEG (parent));
f95284d2
RS
3486 }
3487 else
3488 maxdelta = (!NILP (p->next) ? ((*sizefun) (p->next)
3489 - window_min_size (XWINDOW (p->next),
3490 widthflag, 0, 0))
3491 : (delta = 0));
3492 }
3493 else
3494 maxdelta = (!NILP (parent) ? (*sizefun) (parent) - XINT (*sizep)
3495 /* This is a main window followed by a minibuffer. */
3496 : !NILP (p->next) ? ((*sizefun) (p->next)
3497 - window_min_size (XWINDOW (p->next),
3498 widthflag, 0, 0))
3499 /* This is a minibuffer following a main window. */
3500 : !NILP (p->prev) ? ((*sizefun) (p->prev)
3501 - window_min_size (XWINDOW (p->prev),
3502 widthflag, 0, 0))
3503 /* This is a frame with only one window, a minibuffer-only
3504 or a minibufferless frame. */
3505 : (delta = 0));
7ab12479
JB
3506
3507 if (delta > maxdelta)
3508 /* This case traps trying to make the minibuffer
44fa5b1e
JB
3509 the full frame, or make the only window aside from the
3510 minibuffer the full frame. */
7ab12479 3511 delta = maxdelta;
6b54027b 3512 }
d5783c40 3513
3578db3c 3514 if (XINT (*sizep) + delta < window_min_size (XWINDOW (window), widthflag, 0, 0))
6b54027b 3515 {
543f5fb1 3516 delete_window (window);
d5783c40 3517 return;
6b54027b
RS
3518 }
3519
3520 if (delta == 0)
3521 return;
7ab12479 3522
f95284d2 3523 /* Find the total we can get from other siblings without deleting them. */
db98a733
RS
3524 maximum = 0;
3525 for (next = p->next; ! NILP (next); next = XWINDOW (next)->next)
c1636aa6 3526 maximum += (*sizefun) (next) - window_min_size (XWINDOW (next),
f984d4fc 3527 widthflag, 0, 0);
f95284d2
RS
3528 if (! preserve_before)
3529 for (prev = p->prev; ! NILP (prev); prev = XWINDOW (prev)->prev)
3530 maximum += (*sizefun) (prev) - window_min_size (XWINDOW (prev),
3531 widthflag, 0, 0);
db98a733 3532
f95284d2 3533 /* If we can get it all from them without deleting them, do so. */
c6b530ed 3534 if (delta <= maximum)
7ab12479 3535 {
db98a733
RS
3536 Lisp_Object first_unaffected;
3537 Lisp_Object first_affected;
233a4a2c 3538 int fixed_p;
db98a733
RS
3539
3540 next = p->next;
3541 prev = p->prev;
3542 first_affected = window;
3543 /* Look at one sibling at a time,
3544 moving away from this window in both directions alternately,
3545 and take as much as we can get without deleting that sibling. */
f95284d2
RS
3546 while (delta != 0
3547 && (!NILP (next) || (!preserve_before && !NILP (prev))))
db98a733 3548 {
db98a733
RS
3549 if (! NILP (next))
3550 {
c1636aa6 3551 int this_one = ((*sizefun) (next)
233a4a2c 3552 - window_min_size (XWINDOW (next),
f984d4fc 3553 widthflag, 0, &fixed_p));
233a4a2c
GM
3554 if (!fixed_p)
3555 {
3556 if (this_one > delta)
3557 this_one = delta;
3558
3559 (*setsizefun) (next, (*sizefun) (next) - this_one, 0);
3578db3c 3560 (*setsizefun) (window, XINT (*sizep) + this_one, 0);
db98a733 3561
233a4a2c
GM
3562 delta -= this_one;
3563 }
3564
db98a733
RS
3565 next = XWINDOW (next)->next;
3566 }
233a4a2c 3567
db98a733
RS
3568 if (delta == 0)
3569 break;
233a4a2c 3570
f95284d2 3571 if (!preserve_before && ! NILP (prev))
db98a733 3572 {
c1636aa6 3573 int this_one = ((*sizefun) (prev)
233a4a2c 3574 - window_min_size (XWINDOW (prev),
f984d4fc 3575 widthflag, 0, &fixed_p));
233a4a2c
GM
3576 if (!fixed_p)
3577 {
3578 if (this_one > delta)
3579 this_one = delta;
3580
3581 first_affected = prev;
3582
3583 (*setsizefun) (prev, (*sizefun) (prev) - this_one, 0);
3578db3c 3584 (*setsizefun) (window, XINT (*sizep) + this_one, 0);
233a4a2c
GM
3585
3586 delta -= this_one;
3587 }
3588
db98a733
RS
3589 prev = XWINDOW (prev)->prev;
3590 }
3591 }
3592
233a4a2c
GM
3593 xassert (delta == 0);
3594
db98a733
RS
3595 /* Now recalculate the edge positions of all the windows affected,
3596 based on the new sizes. */
3597 first_unaffected = next;
3598 prev = first_affected;
3599 for (next = XWINDOW (prev)->next; ! EQ (next, first_unaffected);
3600 prev = next, next = XWINDOW (next)->next)
3601 {
3578db3c 3602 XSETINT (CURBEG (next), XINT (CURBEG (prev)) + (*sizefun) (prev));
db98a733
RS
3603 /* This does not change size of NEXT,
3604 but it propagates the new top edge to its children */
3605 (*setsizefun) (next, (*sizefun) (next), 0);
3606 }
7ab12479
JB
3607 }
3608 else
3609 {
3610 register int delta1;
3611 register int opht = (*sizefun) (parent);
3612
3578db3c 3613 if (opht <= XINT (*sizep) + delta)
daf516d3
RS
3614 {
3615 /* If trying to grow this window to or beyond size of the parent,
3616 just delete all the sibling windows. */
dc1ab1ee 3617 Lisp_Object start, tem, next;
daf516d3 3618
dc1ab1ee
RS
3619 start = XWINDOW (parent)->vchild;
3620 if (NILP (start))
3621 start = XWINDOW (parent)->hchild;
daf516d3 3622
dc1ab1ee
RS
3623 /* Delete any siblings that come after WINDOW. */
3624 tem = XWINDOW (window)->next;
daf516d3
RS
3625 while (! NILP (tem))
3626 {
3627 next = XWINDOW (tem)->next;
dc1ab1ee
RS
3628 delete_window (tem);
3629 tem = next;
3630 }
3631
3632 /* Delete any siblings that come after WINDOW.
3633 Note that if START is not WINDOW, then WINDOW still
3634 Fhas siblings, so WINDOW has not yet replaced its parent. */
3635 tem = start;
3636 while (! EQ (tem, window))
3637 {
3638 next = XWINDOW (tem)->next;
3639 delete_window (tem);
daf516d3
RS
3640 tem = next;
3641 }
3642 }
7ab12479 3643 else
233a4a2c
GM
3644 {
3645 /* Otherwise, make delta1 just right so that if we add
3646 delta1 lines to this window and to the parent, and then
3647 shrink the parent back to its original size, the new
3648 proportional size of this window will increase by delta.
3649
3650 The function size_window will compute the new height h'
3651 of the window from delta1 as:
3652
3653 e = delta1/n
3654 x = delta1 - delta1/n * n for the 1st resizable child
3655 h' = h + e + x
3656
3657 where n is the number of children that can be resized.
3658 We can ignore x by choosing a delta1 that is a multiple of
3659 n. We want the height of this window to come out as
3660
3661 h' = h + delta
3662
3663 So, delta1 must be
3664
3665 h + e = h + delta
3666 delta1/n = delta
3667 delta1 = n * delta.
3668
a5731348 3669 The number of children n equals the number of resizable
233a4a2c
GM
3670 children of this window + 1 because we know window itself
3671 is resizable (otherwise we would have signalled an error. */
3672
3673 struct window *w = XWINDOW (window);
3674 Lisp_Object s;
3675 int n = 1;
3676
3677 for (s = w->next; !NILP (s); s = XWINDOW (s)->next)
3678 if (!window_fixed_size_p (XWINDOW (s), widthflag, 0))
3679 ++n;
3680 for (s = w->prev; !NILP (s); s = XWINDOW (s)->prev)
3681 if (!window_fixed_size_p (XWINDOW (s), widthflag, 0))
3682 ++n;
3683
3684 delta1 = n * delta;
7ab12479 3685
daf516d3
RS
3686 /* Add delta1 lines or columns to this window, and to the parent,
3687 keeping things consistent while not affecting siblings. */
3688 XSETINT (CURSIZE (parent), opht + delta1);
3689 (*setsizefun) (window, XINT (*sizep) + delta1, 0);
3690
3691 /* Squeeze out delta1 lines or columns from our parent,
3692 shriking this window and siblings proportionately.
3693 This brings parent back to correct size.
3694 Delta1 was calculated so this makes this window the desired size,
3695 taking it all out of the siblings. */
3696 (*setsizefun) (parent, opht, 0);
3697
3698 }
7ab12479
JB
3699 }
3700
d834a2e9 3701 XSETFASTINT (p->last_modified, 0);
3cd21523 3702 XSETFASTINT (p->last_overlay_modified, 0);
5500c422
GM
3703
3704 /* Adjust glyph matrices. */
3705 adjust_glyphs (XFRAME (WINDOW_FRAME (XWINDOW (window))));
7ab12479 3706}
c1636aa6 3707
7ab12479
JB
3708#undef CURBEG
3709#undef CURSIZE
3710
5500c422 3711
f984d4fc
GM
3712\f
3713/***********************************************************************
3714 Resizing Mini-Windows
3715 ***********************************************************************/
3716
3717static void shrink_window_lowest_first P_ ((struct window *, int));
f984d4fc 3718
43b4a21f
GM
3719enum save_restore_action
3720{
3721 CHECK_ORIG_SIZES,
3722 SAVE_ORIG_SIZES,
3723 RESTORE_ORIG_SIZES
3724};
3725
3726static int save_restore_orig_size P_ ((struct window *,
3727 enum save_restore_action));
f984d4fc
GM
3728
3729/* Shrink windows rooted in window W to HEIGHT. Take the space needed
3730 from lowest windows first. */
3731
3732static void
3733shrink_window_lowest_first (w, height)
3734 struct window *w;
3735 int height;
3736{
3737 struct window *c;
3738 Lisp_Object child;
3739 int old_height;
3740
3741 xassert (!MINI_WINDOW_P (w));
3742
3743 /* Set redisplay hints. */
3744 XSETFASTINT (w->last_modified, 0);
3745 XSETFASTINT (w->last_overlay_modified, 0);
3746 windows_or_buffers_changed++;
3747 FRAME_WINDOW_SIZES_CHANGED (XFRAME (WINDOW_FRAME (w))) = 1;
3748
3749 old_height = XFASTINT (w->height);
3750 XSETFASTINT (w->height, height);
3751
3752 if (!NILP (w->hchild))
3753 {
3754 for (child = w->hchild; !NILP (child); child = c->next)
3755 {
3756 c = XWINDOW (child);
3757 c->top = w->top;
3758 shrink_window_lowest_first (c, height);
3759 }
3760 }
3761 else if (!NILP (w->vchild))
3762 {
3763 Lisp_Object last_child;
3764 int delta = old_height - height;
3765 int last_top;
6bbd7a29
GM
3766
3767 last_child = Qnil;
f984d4fc
GM
3768
3769 /* Find the last child. We are taking space from lowest windows
3770 first, so we iterate over children from the last child
3771 backwards. */
3772 for (child = w->vchild; !NILP (child); child = XWINDOW (child)->next)
3773 last_child = child;
3774
3775 /* Assign new heights. We leave only MIN_SAFE_WINDOW_HEIGHT. */
3776 for (child = last_child; delta && !NILP (child); child = c->prev)
3777 {
3778 int this_one;
3779
3780 c = XWINDOW (child);
3781 this_one = XFASTINT (c->height) - MIN_SAFE_WINDOW_HEIGHT;
3782
3783 if (this_one > delta)
3784 this_one = delta;
3785
3786 shrink_window_lowest_first (c, XFASTINT (c->height) - this_one);
3787 delta -= this_one;
3788 }
3789
3790 /* Compute new positions. */
3578db3c 3791 last_top = XINT (w->top);
f984d4fc
GM
3792 for (child = w->vchild; !NILP (child); child = c->next)
3793 {
3794 c = XWINDOW (child);
3795 c->top = make_number (last_top);
3796 shrink_window_lowest_first (c, XFASTINT (c->height));
3797 last_top += XFASTINT (c->height);
3798 }
3799 }
3800}
3801
3802
43b4a21f
GM
3803/* Save, restore, or check positions and sizes in the window tree
3804 rooted at W. ACTION says what to do.
f984d4fc 3805
43b4a21f
GM
3806 If ACTION is CHECK_ORIG_SIZES, check if orig_top and orig_height
3807 members are valid for all windows in the window tree. Value is
3808 non-zero if they are valid.
3809
3810 If ACTION is SAVE_ORIG_SIZES, save members top and height in
3811 orig_top and orig_height for all windows in the tree.
3812
3813 If ACTION is RESTORE_ORIG_SIZES, restore top and height from
3814 values stored in orig_top and orig_height for all windows. */
3815
3816static int
3817save_restore_orig_size (w, action)
f984d4fc 3818 struct window *w;
43b4a21f 3819 enum save_restore_action action;
f984d4fc 3820{
43b4a21f
GM
3821 int success_p = 1;
3822
f984d4fc
GM
3823 while (w)
3824 {
3825 if (!NILP (w->hchild))
43b4a21f
GM
3826 {
3827 if (!save_restore_orig_size (XWINDOW (w->hchild), action))
3828 success_p = 0;
3829 }
f984d4fc 3830 else if (!NILP (w->vchild))
43b4a21f
GM
3831 {
3832 if (!save_restore_orig_size (XWINDOW (w->vchild), action))
3833 success_p = 0;
3834 }
f984d4fc 3835
43b4a21f 3836 switch (action)
f984d4fc 3837 {
43b4a21f
GM
3838 case CHECK_ORIG_SIZES:
3839 if (!INTEGERP (w->orig_top) || !INTEGERP (w->orig_height))
3840 return 0;
3841 break;
3842
3843 case SAVE_ORIG_SIZES:
f984d4fc
GM
3844 w->orig_top = w->top;
3845 w->orig_height = w->height;
43b4a21f
GM
3846 XSETFASTINT (w->last_modified, 0);
3847 XSETFASTINT (w->last_overlay_modified, 0);
3848 break;
3849
3850 case RESTORE_ORIG_SIZES:
f984d4fc
GM
3851 xassert (INTEGERP (w->orig_top) && INTEGERP (w->orig_height));
3852 w->top = w->orig_top;
3853 w->height = w->orig_height;
3854 w->orig_height = w->orig_top = Qnil;
43b4a21f
GM
3855 XSETFASTINT (w->last_modified, 0);
3856 XSETFASTINT (w->last_overlay_modified, 0);
3857 break;
3858
3859 default:
3860 abort ();
f984d4fc 3861 }
43b4a21f 3862
f984d4fc
GM
3863 w = NILP (w->next) ? NULL : XWINDOW (w->next);
3864 }
43b4a21f
GM
3865
3866 return success_p;
f984d4fc
GM
3867}
3868
3869
3870/* Grow mini-window W by DELTA lines, DELTA >= 0, or as much as we can
3871 without deleting other windows. */
3872
3873void
3874grow_mini_window (w, delta)
3875 struct window *w;
3876 int delta;
3877{
3878 struct frame *f = XFRAME (w->frame);
3879 struct window *root;
3880
3881 xassert (MINI_WINDOW_P (w));
3882 xassert (delta >= 0);
3883
3884 /* Check values of window_min_width and window_min_height for
3885 validity. */
3886 check_min_window_sizes ();
3887
3888 /* Compute how much we can enlarge the mini-window without deleting
3889 other windows. */
3890 root = XWINDOW (FRAME_ROOT_WINDOW (f));
3891 if (delta)
3892 {
3893 int min_height = window_min_size (root, 0, 0, 0);
3894 if (XFASTINT (root->height) - delta < min_height)
8b8bd9c6 3895 /* Note that the root window may already be smaller than
eafa3196
GM
3896 min_height. */
3897 delta = max (0, XFASTINT (root->height) - min_height);
f984d4fc
GM
3898 }
3899
3900 if (delta)
3901 {
3902 /* Save original window sizes and positions, if not already done. */
43b4a21f
GM
3903 if (!save_restore_orig_size (root, CHECK_ORIG_SIZES))
3904 save_restore_orig_size (root, SAVE_ORIG_SIZES);
f984d4fc
GM
3905
3906 /* Shrink other windows. */
3907 shrink_window_lowest_first (root, XFASTINT (root->height) - delta);
3908
3909 /* Grow the mini-window. */
3578db3c 3910 w->top = make_number (XFASTINT (root->top) + XFASTINT (root->height));
f984d4fc
GM
3911 w->height = make_number (XFASTINT (w->height) + delta);
3912 XSETFASTINT (w->last_modified, 0);
3913 XSETFASTINT (w->last_overlay_modified, 0);
3914
3915 adjust_glyphs (f);
3916 }
3917}
3918
3919
86c8e823
GM
3920/* Shrink mini-window W. If there is recorded info about window sizes
3921 before a call to grow_mini_window, restore recorded window sizes.
3922 Otherwise, if the mini-window is higher than 1 line, resize it to 1
3923 line. */
f984d4fc
GM
3924
3925void
3926shrink_mini_window (w)
3927 struct window *w;
3928{
3929 struct frame *f = XFRAME (w->frame);
3930 struct window *root = XWINDOW (FRAME_ROOT_WINDOW (f));
3931
43b4a21f 3932 if (save_restore_orig_size (root, CHECK_ORIG_SIZES))
f984d4fc 3933 {
43b4a21f 3934 save_restore_orig_size (root, RESTORE_ORIG_SIZES);
f984d4fc
GM
3935 adjust_glyphs (f);
3936 FRAME_WINDOW_SIZES_CHANGED (f) = 1;
3937 windows_or_buffers_changed = 1;
3938 }
86c8e823
GM
3939 else if (XFASTINT (w->height) > 1)
3940 {
0130fe1a
GM
3941 /* Distribute the additional lines of the mini-window
3942 among the other windows. */
86c8e823
GM
3943 Lisp_Object window;
3944 XSETWINDOW (window, w);
f95284d2 3945 enlarge_window (window, 1 - XFASTINT (w->height), 0, 0);
86c8e823 3946 }
f984d4fc
GM
3947}
3948
3949
3950\f
5500c422
GM
3951/* Mark window cursors off for all windows in the window tree rooted
3952 at W by setting their phys_cursor_on_p flag to zero. Called from
3953 xterm.c, e.g. when a frame is cleared and thereby all cursors on
3954 the frame are cleared. */
3955
3956void
3957mark_window_cursors_off (w)
3958 struct window *w;
3959{
3960 while (w)
3961 {
3962 if (!NILP (w->hchild))
3963 mark_window_cursors_off (XWINDOW (w->hchild));
3964 else if (!NILP (w->vchild))
3965 mark_window_cursors_off (XWINDOW (w->vchild));
3966 else
3967 w->phys_cursor_on_p = 0;
3968
3969 w = NILP (w->next) ? 0 : XWINDOW (w->next);
3970 }
3971}
3972
3973
e9c195b1 3974/* Return number of lines of text (not counting mode lines) in W. */
7ab12479
JB
3975
3976int
3977window_internal_height (w)
3978 struct window *w;
3979{
3980 int ht = XFASTINT (w->height);
3981
e9c195b1
GM
3982 if (!MINI_WINDOW_P (w))
3983 {
3984 if (!NILP (w->parent)
3985 || !NILP (w->vchild)
3986 || !NILP (w->hchild)
3987 || !NILP (w->next)
3988 || !NILP (w->prev)
3989 || WINDOW_WANTS_MODELINE_P (w))
3990 --ht;
7ab12479 3991
e9c195b1
GM
3992 if (WINDOW_WANTS_HEADER_LINE_P (w))
3993 --ht;
3994 }
7ab12479
JB
3995
3996 return ht;
3997}
3998
535e0b8e
JB
3999
4000/* Return the number of columns in W.
a3c87d4e 4001 Don't count columns occupied by scroll bars or the vertical bar
535e0b8e 4002 separating W from the sibling to its right. */
5500c422 4003
535e0b8e
JB
4004int
4005window_internal_width (w)
4006 struct window *w;
4007{
5500c422 4008 struct frame *f = XFRAME (WINDOW_FRAME (w));
535e0b8e
JB
4009 int width = XINT (w->width);
4010
a3c87d4e 4011 if (FRAME_HAS_VERTICAL_SCROLL_BARS (f))
5500c422
GM
4012 /* Scroll bars occupy a few columns. */
4013 width -= FRAME_SCROLL_BAR_COLS (f);
4014 else if (!WINDOW_RIGHTMOST_P (w) && !WINDOW_FULL_WIDTH_P (w))
4015 /* The column of `|' characters separating side-by-side windows
4016 occupies one column only. */
4017 width -= 1;
4018
4019 /* On window-systems, areas to the left and right of the window
81d189fd 4020 are used as fringes. */
5500c422 4021 if (FRAME_WINDOW_P (f))
81d189fd 4022 width -= FRAME_FRINGE_COLS (f);
111e5992
RS
4023
4024 return width;
535e0b8e
JB
4025}
4026
5500c422
GM
4027\f
4028/************************************************************************
4029 Window Scrolling
4030 ***********************************************************************/
535e0b8e 4031
5500c422 4032/* Scroll contents of window WINDOW up. If WHOLE is non-zero, scroll
d4e7cf01 4033 N screen-fulls, which is defined as the height of the window minus
5500c422
GM
4034 next_screen_context_lines. If WHOLE is zero, scroll up N lines
4035 instead. Negative values of N mean scroll down. NOERROR non-zero
4036 means don't signal an error if we try to move over BEGV or ZV,
4037 respectively. */
7ab12479 4038
101d1605
RS
4039static void
4040window_scroll (window, n, whole, noerror)
7ab12479
JB
4041 Lisp_Object window;
4042 int n;
101d1605 4043 int whole;
f8026fd8 4044 int noerror;
5500c422 4045{
cba59f77
RS
4046 immediate_quit = 1;
4047
5500c422
GM
4048 /* If we must, use the pixel-based version which is much slower than
4049 the line-based one but can handle varying line heights. */
4050 if (FRAME_WINDOW_P (XFRAME (XWINDOW (window)->frame)))
4051 window_scroll_pixel_based (window, n, whole, noerror);
4052 else
4053 window_scroll_line_based (window, n, whole, noerror);
cba59f77
RS
4054
4055 immediate_quit = 0;
5500c422
GM
4056}
4057
4058
4059/* Implementation of window_scroll that works based on pixel line
4060 heights. See the comment of window_scroll for parameter
4061 descriptions. */
4062
4063static void
4064window_scroll_pixel_based (window, n, whole, noerror)
4065 Lisp_Object window;
4066 int n;
4067 int whole;
4068 int noerror;
4069{
4070 struct it it;
4071 struct window *w = XWINDOW (window);
4072 struct text_pos start;
4073 Lisp_Object tem;
4074 int this_scroll_margin;
4075 int preserve_y;
5cdb3cf3
MB
4076 /* True if we fiddled the window vscroll field without really scrolling. */
4077 int vscrolled = 0;
5500c422
GM
4078
4079 SET_TEXT_POS_FROM_MARKER (start, w->start);
4080
4081 /* If PT is not visible in WINDOW, move back one half of
288d4e06
GM
4082 the screen. Allow PT to be partially visible, otherwise
4083 something like (scroll-down 1) with PT in the line before
4084 the partially visible one would recenter. */
4085 tem = Fpos_visible_in_window_p (make_number (PT), window, Qt);
5500c422
GM
4086 if (NILP (tem))
4087 {
4088 /* Move backward half the height of the window. Performance note:
4089 vmotion used here is about 10% faster, but would give wrong
4090 results for variable height lines. */
4091 init_iterator (&it, w, PT, PT_BYTE, NULL, DEFAULT_FACE_ID);
4092 it.current_y = it.last_visible_y;
202379cf 4093 move_it_vertically (&it, - window_box_height (w) / 2);
5500c422
GM
4094
4095 /* The function move_iterator_vertically may move over more than
4096 the specified y-distance. If it->w is small, e.g. a
4097 mini-buffer window, we may end up in front of the window's
4098 display area. This is the case when Start displaying at the
4099 start of the line containing PT in this case. */
4100 if (it.current_y <= 0)
4101 {
4102 init_iterator (&it, w, PT, PT_BYTE, NULL, DEFAULT_FACE_ID);
4103 move_it_vertically (&it, 0);
4104 it.current_y = 0;
4105 }
4106
4107 start = it.current.pos;
4108 }
4109
4110 /* If scroll_preserve_screen_position is non-zero, we try to set
4111 point in the same window line as it is now, so get that line. */
4112 if (!NILP (Vscroll_preserve_screen_position))
4113 {
4114 start_display (&it, w, start);
4115 move_it_to (&it, PT, -1, -1, -1, MOVE_TO_POS);
4116 preserve_y = it.current_y;
4117 }
4118 else
4119 preserve_y = -1;
4120
4121 /* Move iterator it from start the specified distance forward or
4122 backward. The result is the new window start. */
4123 start_display (&it, w, start);
4124 if (whole)
4125 {
202379cf 4126 int screen_full = (window_box_height (w)
5500c422 4127 - next_screen_context_lines * CANON_Y_UNIT (it.f));
d4e7cf01 4128 int dy = n * screen_full;
d72340d4
GM
4129
4130 /* Note that move_it_vertically always moves the iterator to the
4131 start of a line. So, if the last line doesn't have a newline,
4132 we would end up at the start of the line ending at ZV. */
4133 if (dy <= 0)
4134 move_it_vertically_backward (&it, -dy);
4135 else if (dy > 0)
4136 move_it_to (&it, ZV, -1, it.current_y + dy, -1,
4137 MOVE_TO_POS | MOVE_TO_Y);
5500c422
GM
4138 }
4139 else
4140 move_it_by_lines (&it, n, 1);
4141
4142 /* End if we end up at ZV or BEGV. */
4143 if ((n > 0 && IT_CHARPOS (it) == ZV)
4144 || (n < 0 && IT_CHARPOS (it) == CHARPOS (start)))
4145 {
5cdb3cf3
MB
4146 if (IT_CHARPOS (it) == ZV)
4147 {
4148 if (it.current_y + it.max_ascent + it.max_descent
4149 > it.last_visible_y)
a74eca50
GM
4150 {
4151 /* The last line was only partially visible, make it fully
4152 visible. */
4153 w->vscroll = (it.last_visible_y
4154 - it.current_y + it.max_ascent + it.max_descent);
4155 adjust_glyphs (it.f);
4156 }
5cdb3cf3
MB
4157 else if (noerror)
4158 return;
4159 else
4160 Fsignal (Qend_of_buffer, Qnil);
4161 }
5500c422 4162 else
5cdb3cf3
MB
4163 {
4164 if (w->vscroll != 0)
4165 /* The first line was only partially visible, make it fully
4166 visible. */
4167 w->vscroll = 0;
4168 else if (noerror)
4169 return;
4170 else
4171 Fsignal (Qbeginning_of_buffer, Qnil);
4172 }
4173
4174 /* If control gets here, then we vscrolled. */
4175
4176 XBUFFER (w->buffer)->prevent_redisplay_optimizations_p = 1;
4177
4178 /* Don't try to change the window start below. */
4179 vscrolled = 1;
5500c422
GM
4180 }
4181
5cdb3cf3
MB
4182 if (! vscrolled)
4183 {
dad67609
RS
4184 int pos = IT_CHARPOS (it);
4185 int bytepos;
5cdb3cf3 4186 /* Set the window start, and set up the window for redisplay. */
dad67609 4187 set_marker_restricted (w->start, make_number (pos),
5cdb3cf3 4188 w->buffer);
dad67609
RS
4189 bytepos = XMARKER (w->start)->bytepos;
4190 w->start_at_line_beg = ((pos == BEGV || FETCH_BYTE (bytepos - 1) == '\n')
4191 ? Qt : Qnil);
5cdb3cf3
MB
4192 w->update_mode_line = Qt;
4193 XSETFASTINT (w->last_modified, 0);
4194 XSETFASTINT (w->last_overlay_modified, 0);
4195 /* Set force_start so that redisplay_window will run the
4196 window-scroll-functions. */
4197 w->force_start = Qt;
4198 }
5500c422
GM
4199
4200 it.current_y = it.vpos = 0;
4201
4202 /* Preserve the screen position if we must. */
4203 if (preserve_y >= 0)
4204 {
4205 move_it_to (&it, -1, -1, preserve_y, -1, MOVE_TO_Y);
4206 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4207 }
4208 else
4209 {
4210 /* Move PT out of scroll margins. */
4211 this_scroll_margin = max (0, scroll_margin);
4212 this_scroll_margin = min (this_scroll_margin, XFASTINT (w->height) / 4);
4213 this_scroll_margin *= CANON_Y_UNIT (it.f);
4214
4215 if (n > 0)
4216 {
4217 /* We moved the window start towards ZV, so PT may be now
4218 in the scroll margin at the top. */
4219 move_it_to (&it, PT, -1, -1, -1, MOVE_TO_POS);
4220 while (it.current_y < this_scroll_margin)
e9b2c961
RS
4221 {
4222 int prev = it.current_y;
4223 move_it_by_lines (&it, 1, 1);
4224 if (prev == it.current_y)
4225 break;
4226 }
5500c422
GM
4227 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4228 }
4229 else if (n < 0)
4230 {
5cdb3cf3
MB
4231 int charpos, bytepos;
4232
5500c422
GM
4233 /* We moved the window start towards BEGV, so PT may be now
4234 in the scroll margin at the bottom. */
4235 move_it_to (&it, PT, -1,
4236 it.last_visible_y - this_scroll_margin - 1, -1,
4237 MOVE_TO_POS | MOVE_TO_Y);
5cdb3cf3
MB
4238
4239 /* Save our position, in case it's correct. */
4240 charpos = IT_CHARPOS (it);
4241 bytepos = IT_BYTEPOS (it);
5500c422 4242
5cdb3cf3
MB
4243 /* See if point is on a partially visible line at the end. */
4244 move_it_by_lines (&it, 1, 1);
4245 if (it.current_y > it.last_visible_y)
4246 /* The last line was only partially visible, so back up two
4247 lines to make sure we're on a fully visible line. */
4248 {
4249 move_it_by_lines (&it, -2, 0);
4250 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4251 }
4252 else
4253 /* No, the position we saved is OK, so use it. */
4254 SET_PT_BOTH (charpos, bytepos);
5500c422
GM
4255 }
4256 }
4257}
4258
4259
4260/* Implementation of window_scroll that works based on screen lines.
4261 See the comment of window_scroll for parameter descriptions. */
4262
4263static void
4264window_scroll_line_based (window, n, whole, noerror)
4265 Lisp_Object window;
4266 int n;
4267 int whole;
4268 int noerror;
7ab12479
JB
4269{
4270 register struct window *w = XWINDOW (window);
5500c422 4271 register int opoint = PT, opoint_byte = PT_BYTE;
b73ea88e 4272 register int pos, pos_byte;
7ab12479
JB
4273 register int ht = window_internal_height (w);
4274 register Lisp_Object tem;
4275 int lose;
5500c422 4276 Lisp_Object bolp;
345d45b2 4277 int startpos;
101d1605
RS
4278 struct position posit;
4279 int original_vpos;
4280
d4e7cf01
GM
4281 /* If scrolling screen-fulls, compute the number of lines to
4282 scroll from the window's height. */
4283 if (whole)
4284 n *= max (1, ht - next_screen_context_lines);
4285
101d1605
RS
4286 startpos = marker_position (w->start);
4287
4288 posit = *compute_motion (startpos, 0, 0, 0,
4289 PT, ht, 0,
4290 window_internal_width (w), XINT (w->hscroll),
4291 0, w);
4292 original_vpos = posit.vpos;
0a1f771a 4293
d834a2e9 4294 XSETFASTINT (tem, PT);
6ffdb539 4295 tem = Fpos_visible_in_window_p (tem, window, Qnil);
7ab12479 4296
265a9e55 4297 if (NILP (tem))
7ab12479 4298 {
cd2be1dd 4299 Fvertical_motion (make_number (- (ht / 2)), window);
345d45b2 4300 startpos = PT;
7ab12479
JB
4301 }
4302
345d45b2 4303 SET_PT (startpos);
5ce7b543 4304 lose = n < 0 && PT == BEGV;
540b6aa0 4305 Fvertical_motion (make_number (n), window);
5ce7b543 4306 pos = PT;
b73ea88e 4307 pos_byte = PT_BYTE;
7ab12479 4308 bolp = Fbolp ();
b73ea88e 4309 SET_PT_BOTH (opoint, opoint_byte);
7ab12479
JB
4310
4311 if (lose)
f8026fd8
JB
4312 {
4313 if (noerror)
4314 return;
4315 else
4316 Fsignal (Qbeginning_of_buffer, Qnil);
4317 }
7ab12479
JB
4318
4319 if (pos < ZV)
7ab12479 4320 {
0c7da84e
RS
4321 int this_scroll_margin = scroll_margin;
4322
4323 /* Don't use a scroll margin that is negative or too large. */
4324 if (this_scroll_margin < 0)
4325 this_scroll_margin = 0;
4326
4327 if (XINT (w->height) < 4 * scroll_margin)
4328 this_scroll_margin = XINT (w->height) / 4;
4329
b73ea88e 4330 set_marker_restricted_both (w->start, w->buffer, pos, pos_byte);
7ab12479
JB
4331 w->start_at_line_beg = bolp;
4332 w->update_mode_line = Qt;
d834a2e9 4333 XSETFASTINT (w->last_modified, 0);
3cd21523 4334 XSETFASTINT (w->last_overlay_modified, 0);
345d45b2
RS
4335 /* Set force_start so that redisplay_window will run
4336 the window-scroll-functions. */
4337 w->force_start = Qt;
0c7da84e 4338
5500c422 4339 if (whole && !NILP (Vscroll_preserve_screen_position))
0c7da84e 4340 {
b73ea88e 4341 SET_PT_BOTH (pos, pos_byte);
101d1605 4342 Fvertical_motion (make_number (original_vpos), window);
0c7da84e 4343 }
101d1605
RS
4344 /* If we scrolled forward, put point enough lines down
4345 that it is outside the scroll margin. */
4346 else if (n > 0)
0c7da84e 4347 {
101d1605
RS
4348 int top_margin;
4349
4350 if (this_scroll_margin > 0)
4351 {
b73ea88e 4352 SET_PT_BOTH (pos, pos_byte);
101d1605
RS
4353 Fvertical_motion (make_number (this_scroll_margin), window);
4354 top_margin = PT;
4355 }
4356 else
4357 top_margin = pos;
4358
4359 if (top_margin <= opoint)
b73ea88e 4360 SET_PT_BOTH (opoint, opoint_byte);
5500c422 4361 else if (!NILP (Vscroll_preserve_screen_position))
101d1605 4362 {
b73ea88e 4363 SET_PT_BOTH (pos, pos_byte);
101d1605
RS
4364 Fvertical_motion (make_number (original_vpos), window);
4365 }
9317a85d 4366 else
335406fc 4367 SET_PT (top_margin);
0c7da84e 4368 }
101d1605 4369 else if (n < 0)
7ab12479 4370 {
101d1605
RS
4371 int bottom_margin;
4372
0c7da84e
RS
4373 /* If we scrolled backward, put point near the end of the window
4374 but not within the scroll margin. */
b73ea88e 4375 SET_PT_BOTH (pos, pos_byte);
0c7da84e 4376 tem = Fvertical_motion (make_number (ht - this_scroll_margin), window);
101d1605
RS
4377 if (XFASTINT (tem) == ht - this_scroll_margin)
4378 bottom_margin = PT;
4379 else
4380 bottom_margin = PT + 1;
4381
4382 if (bottom_margin > opoint)
b73ea88e 4383 SET_PT_BOTH (opoint, opoint_byte);
7ab12479 4384 else
101d1605 4385 {
5500c422 4386 if (!NILP (Vscroll_preserve_screen_position))
9317a85d 4387 {
b73ea88e 4388 SET_PT_BOTH (pos, pos_byte);
9317a85d
RS
4389 Fvertical_motion (make_number (original_vpos), window);
4390 }
4391 else
4392 Fvertical_motion (make_number (-1), window);
101d1605 4393 }
7ab12479
JB
4394 }
4395 }
4396 else
f8026fd8
JB
4397 {
4398 if (noerror)
4399 return;
4400 else
4401 Fsignal (Qend_of_buffer, Qnil);
4402 }
7ab12479 4403}
5500c422
GM
4404
4405
4406/* Scroll selected_window up or down. If N is nil, scroll a
4407 screen-full which is defined as the height of the window minus
4408 next_screen_context_lines. If N is the symbol `-', scroll.
4409 DIRECTION may be 1 meaning to scroll down, or -1 meaning to scroll
4410 up. This is the guts of Fscroll_up and Fscroll_down. */
7ab12479
JB
4411
4412static void
4413scroll_command (n, direction)
5500c422 4414 Lisp_Object n;
7ab12479
JB
4415 int direction;
4416{
331379bf 4417 int count = SPECPDL_INDEX ();
7ab12479 4418
5500c422
GM
4419 xassert (abs (direction) == 1);
4420
4421 /* If selected window's buffer isn't current, make it current for
4422 the moment. But don't screw up if window_scroll gets an error. */
7ab12479 4423 if (XBUFFER (XWINDOW (selected_window)->buffer) != current_buffer)
95605e15
JB
4424 {
4425 record_unwind_protect (save_excursion_restore, save_excursion_save ());
4426 Fset_buffer (XWINDOW (selected_window)->buffer);
5500c422
GM
4427
4428 /* Make redisplay consider other windows than just selected_window. */
4429 ++windows_or_buffers_changed;
95605e15 4430 }
7ab12479 4431
265a9e55 4432 if (NILP (n))
d4e7cf01 4433 window_scroll (selected_window, direction, 1, 0);
7ab12479 4434 else if (EQ (n, Qminus))
d4e7cf01 4435 window_scroll (selected_window, -direction, 1, 0);
7ab12479
JB
4436 else
4437 {
4438 n = Fprefix_numeric_value (n);
101d1605 4439 window_scroll (selected_window, XINT (n) * direction, 0, 0);
7ab12479 4440 }
95605e15
JB
4441
4442 unbind_to (count, Qnil);
7ab12479
JB
4443}
4444
4445DEFUN ("scroll-up", Fscroll_up, Sscroll_up, 0, 1, "P",
fdb82f93
PJ
4446 doc: /* Scroll text of current window upward ARG lines; or near full screen if no ARG.
4447A near full screen is `next-screen-context-lines' less than a full screen.
4448Negative ARG means scroll downward.
4449If ARG is the atom `-', scroll downward by nearly full screen.
4450When calling from a program, supply as argument a number, nil, or `-'. */)
4451 (arg)
413430c5 4452 Lisp_Object arg;
7ab12479 4453{
413430c5 4454 scroll_command (arg, 1);
7ab12479
JB
4455 return Qnil;
4456}
4457
4458DEFUN ("scroll-down", Fscroll_down, Sscroll_down, 0, 1, "P",
fdb82f93
PJ
4459 doc: /* Scroll text of current window down ARG lines; or near full screen if no ARG.
4460A near full screen is `next-screen-context-lines' less than a full screen.
4461Negative ARG means scroll upward.
4462If ARG is the atom `-', scroll upward by nearly full screen.
4463When calling from a program, supply as argument a number, nil, or `-'. */)
4464 (arg)
413430c5 4465 Lisp_Object arg;
7ab12479 4466{
413430c5 4467 scroll_command (arg, -1);
7ab12479
JB
4468 return Qnil;
4469}
ccd0664b
RS
4470\f
4471DEFUN ("other-window-for-scrolling", Fother_window_for_scrolling, Sother_window_for_scrolling, 0, 0, 0,
fdb82f93
PJ
4472 doc: /* Return the other window for \"other window scroll\" commands.
4473If in the minibuffer, `minibuffer-scroll-window' if non-nil
4474specifies the window.
4475If `other-window-scroll-buffer' is non-nil, a window
4476showing that buffer is used. */)
4477 ()
7ab12479 4478{
ccd0664b 4479 Lisp_Object window;
7ab12479
JB
4480
4481 if (MINI_WINDOW_P (XWINDOW (selected_window))
265a9e55 4482 && !NILP (Vminibuf_scroll_window))
7ab12479
JB
4483 window = Vminibuf_scroll_window;
4484 /* If buffer is specified, scroll that buffer. */
265a9e55 4485 else if (!NILP (Vother_window_scroll_buffer))
7ab12479
JB
4486 {
4487 window = Fget_buffer_window (Vother_window_scroll_buffer, Qnil);
265a9e55 4488 if (NILP (window))
53f76081 4489 window = Fdisplay_buffer (Vother_window_scroll_buffer, Qt, Qnil);
7ab12479
JB
4490 }
4491 else
dbc4e1c1
JB
4492 {
4493 /* Nothing specified; look for a neighboring window on the same
4494 frame. */
4495 window = Fnext_window (selected_window, Qnil, Qnil);
4496
4497 if (EQ (window, selected_window))
4498 /* That didn't get us anywhere; look for a window on another
4499 visible frame. */
4500 do
4501 window = Fnext_window (window, Qnil, Qt);
4502 while (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (window))))
4503 && ! EQ (window, selected_window));
4504 }
4505
b7826503 4506 CHECK_LIVE_WINDOW (window);
7ab12479
JB
4507
4508 if (EQ (window, selected_window))
4509 error ("There is no other window");
4510
ccd0664b
RS
4511 return window;
4512}
4513
4514DEFUN ("scroll-other-window", Fscroll_other_window, Sscroll_other_window, 0, 1, "P",
fdb82f93
PJ
4515 doc: /* Scroll next window upward ARG lines; or near full screen if no ARG.
4516A near full screen is `next-screen-context-lines' less than a full screen.
4517The next window is the one below the current one; or the one at the top
4518if the current one is at the bottom. Negative ARG means scroll downward.
4519If ARG is the atom `-', scroll downward by nearly full screen.
4520When calling from a program, supply as argument a number, nil, or `-'.
4521
4522If in the minibuffer, `minibuffer-scroll-window' if non-nil
4523specifies the window to scroll.
4524If `other-window-scroll-buffer' is non-nil, scroll the window
4525showing that buffer, popping the buffer up if necessary. */)
4526 (arg)
d4e7cf01 4527 Lisp_Object arg;
ccd0664b 4528{
d4e7cf01
GM
4529 Lisp_Object window;
4530 struct window *w;
331379bf 4531 int count = SPECPDL_INDEX ();
ccd0664b
RS
4532
4533 window = Fother_window_for_scrolling ();
7ab12479 4534 w = XWINDOW (window);
7ab12479
JB
4535
4536 /* Don't screw up if window_scroll gets an error. */
4537 record_unwind_protect (save_excursion_restore, save_excursion_save ());
5500c422 4538 ++windows_or_buffers_changed;
7ab12479
JB
4539
4540 Fset_buffer (w->buffer);
4541 SET_PT (marker_position (w->pointm));
4542
413430c5 4543 if (NILP (arg))
d4e7cf01 4544 window_scroll (window, 1, 1, 1);
413430c5 4545 else if (EQ (arg, Qminus))
d4e7cf01 4546 window_scroll (window, -1, 1, 1);
7ab12479
JB
4547 else
4548 {
413430c5
EN
4549 if (CONSP (arg))
4550 arg = Fcar (arg);
b7826503 4551 CHECK_NUMBER (arg);
101d1605 4552 window_scroll (window, XINT (arg), 0, 1);
7ab12479
JB
4553 }
4554
b73ea88e 4555 set_marker_both (w->pointm, Qnil, PT, PT_BYTE);
f4e7b2c2 4556 unbind_to (count, Qnil);
7ab12479
JB
4557
4558 return Qnil;
4559}
4560\f
644b477c 4561DEFUN ("scroll-left", Fscroll_left, Sscroll_left, 0, 1, "P",
fdb82f93 4562 doc: /* Scroll selected window display ARG columns left.
2fe41216
EZ
4563Default for ARG is window width minus 2.
4564Value is the total amount of leftward horizontal scrolling in
4565effect after the change.
4566If `automatic-hscrolling' is non-nil, the argument ARG modifies
4567a lower bound for automatic scrolling, i.e. automatic scrolling
4568will not scroll a window to a column less than the value returned
4569by this function. */)
fdb82f93 4570 (arg)
7ab12479
JB
4571 register Lisp_Object arg;
4572{
c67fa410
GM
4573 Lisp_Object result;
4574 int hscroll;
4575 struct window *w = XWINDOW (selected_window);
4576
265a9e55 4577 if (NILP (arg))
c67fa410 4578 XSETFASTINT (arg, window_internal_width (w) - 2);
7ab12479
JB
4579 else
4580 arg = Fprefix_numeric_value (arg);
4581
c67fa410
GM
4582 hscroll = XINT (w->hscroll) + XINT (arg);
4583 result = Fset_window_hscroll (selected_window, make_number (hscroll));
4584
f5686fbd 4585 if (interactive_p (0))
c67fa410
GM
4586 w->min_hscroll = w->hscroll;
4587
4588 return result;
7ab12479
JB
4589}
4590
644b477c 4591DEFUN ("scroll-right", Fscroll_right, Sscroll_right, 0, 1, "P",
fdb82f93 4592 doc: /* Scroll selected window display ARG columns right.
2fe41216
EZ
4593Default for ARG is window width minus 2.
4594Value is the total amount of leftward horizontal scrolling in
4595effect after the change.
4596If `automatic-hscrolling' is non-nil, the argument ARG modifies
4597a lower bound for automatic scrolling, i.e. automatic scrolling
4598will not scroll a window to a column less than the value returned
4599by this function. */)
fdb82f93 4600 (arg)
7ab12479
JB
4601 register Lisp_Object arg;
4602{
c67fa410
GM
4603 Lisp_Object result;
4604 int hscroll;
4605 struct window *w = XWINDOW (selected_window);
4606
265a9e55 4607 if (NILP (arg))
c67fa410 4608 XSETFASTINT (arg, window_internal_width (w) - 2);
7ab12479
JB
4609 else
4610 arg = Fprefix_numeric_value (arg);
4611
c67fa410
GM
4612 hscroll = XINT (w->hscroll) - XINT (arg);
4613 result = Fset_window_hscroll (selected_window, make_number (hscroll));
4614
f5686fbd 4615 if (interactive_p (0))
c67fa410
GM
4616 w->min_hscroll = w->hscroll;
4617
4618 return result;
7ab12479
JB
4619}
4620
fa832261
KS
4621DEFUN ("minibuffer-selected-window", Fminibuffer_selected_window, Sminibuffer_selected_window, 0, 0, 0,
4622 doc: /* Return the window which was selected when entering the minibuffer.
4623Returns nil, if current window is not a minibuffer window. */)
4624 ()
4625{
4626 if (minibuf_level > 0
4627 && MINI_WINDOW_P (XWINDOW (selected_window))
4628 && !NILP (minibuf_selected_window)
4629 && WINDOW_LIVE_P (minibuf_selected_window))
4630 return minibuf_selected_window;
4631
4632 return Qnil;
4633}
4634
12c8b416
GM
4635/* Value is the number of lines actually displayed in window W,
4636 as opposed to its height. */
4637
4638static int
4639displayed_window_lines (w)
4640 struct window *w;
4641{
4642 struct it it;
4643 struct text_pos start;
4644 int height = window_box_height (w);
4645 struct buffer *old_buffer;
4646 int bottom_y;
4647
4648 if (XBUFFER (w->buffer) != current_buffer)
4649 {
4650 old_buffer = current_buffer;
4651 set_buffer_internal (XBUFFER (w->buffer));
4652 }
4653 else
4654 old_buffer = NULL;
4655
521b203e
GM
4656 /* In case W->start is out of the accessible range, do something
4657 reasonable. This happens in Info mode when Info-scroll-down
4658 calls (recenter -1) while W->start is 1. */
4659 if (XMARKER (w->start)->charpos < BEGV)
4660 SET_TEXT_POS (start, BEGV, BEGV_BYTE);
4661 else if (XMARKER (w->start)->charpos > ZV)
4662 SET_TEXT_POS (start, ZV, ZV_BYTE);
4663 else
4664 SET_TEXT_POS_FROM_MARKER (start, w->start);
4665
12c8b416
GM
4666 start_display (&it, w, start);
4667 move_it_vertically (&it, height);
c8bc6f65 4668 bottom_y = line_bottom_y (&it);
12c8b416 4669
1de65f51
RS
4670 /* rms: On a non-window display,
4671 the value of it.vpos at the bottom of the screen
4672 seems to be 1 larger than window_box_height (w).
4673 This kludge fixes a bug whereby (move-to-window-line -1)
4674 when ZV is on the last screen line
4675 moves to the previous screen line instead of the last one. */
4676 if (! FRAME_WINDOW_P (XFRAME (w->frame)))
4677 height++;
4678
12c8b416
GM
4679 /* Add in empty lines at the bottom of the window. */
4680 if (bottom_y < height)
4681 {
c8bc6f65
GM
4682 int uy = CANON_Y_UNIT (it.f);
4683 it.vpos += (height - bottom_y + uy - 1) / uy;
12c8b416
GM
4684 }
4685
c8bc6f65
GM
4686 if (old_buffer)
4687 set_buffer_internal (old_buffer);
4688
12c8b416
GM
4689 return it.vpos;
4690}
4691
4692
7ab12479 4693DEFUN ("recenter", Frecenter, Srecenter, 0, 1, "P",
fdb82f93
PJ
4694 doc: /* Center point in window and redisplay frame.
4695With prefix argument ARG, recenter putting point on screen line ARG
4696relative to the current window. If ARG is negative, it counts up from the
4697bottom of the window. (ARG should be less than the height of the window.)
4698
4699If ARG is omitted or nil, erase the entire frame and then
4700redraw with point in the center of the current window.
4701Just C-u as prefix means put point in the center of the window
4702and redisplay normally--don't erase and redraw the frame. */)
4703 (arg)
413430c5 4704 register Lisp_Object arg;
7ab12479 4705{
6df47b59 4706 struct window *w = XWINDOW (selected_window);
478292ed
RS
4707 struct buffer *buf = XBUFFER (w->buffer);
4708 struct buffer *obuf = current_buffer;
6df47b59
GM
4709 int center_p = 0;
4710 int charpos, bytepos;
7ab12479 4711
0fa5d25b
RS
4712 /* If redisplay is suppressed due to an error, try again. */
4713 obuf->display_error_modiff = 0;
4714
413430c5 4715 if (NILP (arg))
7ab12479 4716 {
f02d6d5c
KH
4717 int i;
4718
4719 /* Invalidate pixel data calculated for all compositions. */
4720 for (i = 0; i < n_compositions; i++)
4721 composition_table[i]->font = NULL;
7ab12479 4722
527b6458 4723 Fredraw_frame (w->frame);
44fa5b1e 4724 SET_FRAME_GARBAGED (XFRAME (WINDOW_FRAME (w)));
6df47b59 4725 center_p = 1;
7ab12479 4726 }
413430c5 4727 else if (CONSP (arg)) /* Just C-u. */
6df47b59 4728 center_p = 1;
7ab12479
JB
4729 else
4730 {
413430c5 4731 arg = Fprefix_numeric_value (arg);
b7826503 4732 CHECK_NUMBER (arg);
7ab12479
JB
4733 }
4734
478292ed 4735 set_buffer_internal (buf);
7ab12479 4736
521b203e 4737 /* Handle centering on a graphical frame specially. Such frames can
6df47b59
GM
4738 have variable-height lines and centering point on the basis of
4739 line counts would lead to strange effects. */
521b203e 4740 if (FRAME_WINDOW_P (XFRAME (w->frame)))
6df47b59 4741 {
6df47b59
GM
4742 if (center_p)
4743 {
521b203e
GM
4744 struct it it;
4745 struct text_pos pt;
4746
4747 SET_TEXT_POS (pt, PT, PT_BYTE);
4748 start_display (&it, w, pt);
202379cf 4749 move_it_vertically (&it, - window_box_height (w) / 2);
521b203e
GM
4750 charpos = IT_CHARPOS (it);
4751 bytepos = IT_BYTEPOS (it);
6df47b59
GM
4752 }
4753 else if (XINT (arg) < 0)
4754 {
521b203e
GM
4755 struct it it;
4756 struct text_pos pt;
d466fa4d 4757 int y0, y1, h, nlines;
521b203e
GM
4758
4759 SET_TEXT_POS (pt, PT, PT_BYTE);
4760 start_display (&it, w, pt);
4761 y0 = it.current_y;
4762
d466fa4d 4763 /* The amount of pixels we have to move back is the window
521b203e
GM
4764 height minus what's displayed in the line containing PT,
4765 and the lines below. */
d466fa4d
GM
4766 nlines = - XINT (arg) - 1;
4767 move_it_by_lines (&it, nlines, 1);
4768
201c831a 4769 y1 = line_bottom_y (&it);
d466fa4d
GM
4770
4771 /* If we can't move down NLINES lines because we hit
4772 the end of the buffer, count in some empty lines. */
4773 if (it.vpos < nlines)
4774 y1 += (nlines - it.vpos) * CANON_Y_UNIT (it.f);
4775
201c831a
GM
4776 h = window_box_height (w) - (y1 - y0);
4777
521b203e 4778 start_display (&it, w, pt);
201c831a 4779 move_it_vertically (&it, - h);
521b203e
GM
4780 charpos = IT_CHARPOS (it);
4781 bytepos = IT_BYTEPOS (it);
6df47b59 4782 }
521b203e
GM
4783 else
4784 {
4785 struct position pos;
4786 pos = *vmotion (PT, - XINT (arg), w);
4787 charpos = pos.bufpos;
4788 bytepos = pos.bytepos;
4789 }
4790 }
4791 else
4792 {
4793 struct position pos;
4794 int ht = window_internal_height (w);
4795
4796 if (center_p)
4797 arg = make_number (ht / 2);
4798 else if (XINT (arg) < 0)
4799 arg = make_number (XINT (arg) + ht);
6df47b59
GM
4800
4801 pos = *vmotion (PT, - XINT (arg), w);
4802 charpos = pos.bufpos;
4803 bytepos = pos.bytepos;
4804 }
4805
4806 /* Set the new window start. */
4807 set_marker_both (w->start, w->buffer, charpos, bytepos);
2f3cad6c 4808 w->window_end_valid = Qnil;
95605b1b
RS
4809
4810 w->optional_new_start = Qt;
4811
6df47b59
GM
4812 if (bytepos == BEGV_BYTE || FETCH_BYTE (bytepos - 1) == '\n')
4813 w->start_at_line_beg = Qt;
4814 else
4815 w->start_at_line_beg = Qnil;
4816
478292ed 4817 set_buffer_internal (obuf);
7ab12479
JB
4818 return Qnil;
4819}
b7617575
GM
4820
4821
81fe0836 4822DEFUN ("window-text-height", Fwindow_text_height, Swindow_text_height,
fdb82f93
PJ
4823 0, 1, 0,
4824 doc: /* Return the height in lines of the text display area of WINDOW.
4825This doesn't include the mode-line (or header-line if any) or any
4826partial-height lines in the text display area. */)
4827 (window)
81fe0836
MB
4828 Lisp_Object window;
4829{
4830 struct window *w = decode_window (window);
4831 int pixel_height = window_box_height (w);
4832 int line_height = pixel_height / CANON_Y_UNIT (XFRAME (w->frame));
4833 return make_number (line_height);
4834}
4835
4836
7ab12479
JB
4837\f
4838DEFUN ("move-to-window-line", Fmove_to_window_line, Smove_to_window_line,
fdb82f93
PJ
4839 1, 1, "P",
4840 doc: /* Position point relative to window.
4841With no argument, position point at center of window.
4842An argument specifies vertical position within the window;
4843zero means top of window, negative means relative to bottom of window. */)
4844 (arg)
b7617575 4845 Lisp_Object arg;
7ab12479 4846{
b7617575
GM
4847 struct window *w = XWINDOW (selected_window);
4848 int lines, start;
540b6aa0 4849 Lisp_Object window;
7ab12479 4850
b7617575 4851 window = selected_window;
7ab12479
JB
4852 start = marker_position (w->start);
4853 if (start < BEGV || start > ZV)
4854 {
b7617575 4855 int height = window_internal_height (w);
cd2be1dd 4856 Fvertical_motion (make_number (- (height / 2)), window);
b73ea88e 4857 set_marker_both (w->start, w->buffer, PT, PT_BYTE);
7ab12479
JB
4858 w->start_at_line_beg = Fbolp ();
4859 w->force_start = Qt;
4860 }
4861 else
b73ea88e 4862 Fgoto_char (w->start);
7ab12479 4863
b7617575
GM
4864 lines = displayed_window_lines (w);
4865 if (NILP (arg))
4866 XSETFASTINT (arg, lines / 2);
4867 else
4868 {
4869 arg = Fprefix_numeric_value (arg);
4870 if (XINT (arg) < 0)
4871 XSETINT (arg, XINT (arg) + lines);
4872 }
4873
c8bc6f65 4874 /* Skip past a partially visible first line. */
163784df 4875 if (w->vscroll)
163784df
MB
4876 XSETINT (arg, XINT (arg) + 1);
4877
540b6aa0 4878 return Fvertical_motion (arg, window);
7ab12479 4879}
5500c422
GM
4880
4881
7ab12479 4882\f
5500c422
GM
4883/***********************************************************************
4884 Window Configuration
4885 ***********************************************************************/
4886
7ab12479
JB
4887struct save_window_data
4888 {
f5ccc0cc 4889 EMACS_INT size_from_Lisp_Vector_struct;
7ab12479 4890 struct Lisp_Vector *next_from_Lisp_Vector_struct;
8f6ea2e9 4891 Lisp_Object frame_width, frame_height, frame_menu_bar_lines;
9ea173e8 4892 Lisp_Object frame_tool_bar_lines;
bdc727bf 4893 Lisp_Object selected_frame;
7ab12479
JB
4894 Lisp_Object current_window;
4895 Lisp_Object current_buffer;
4896 Lisp_Object minibuf_scroll_window;
3f49fddc 4897 Lisp_Object minibuf_selected_window;
7ab12479 4898 Lisp_Object root_window;
bdc727bf 4899 Lisp_Object focus_frame;
756b6edc
RS
4900 /* Record the values of window-min-width and window-min-height
4901 so that window sizes remain consistent with them. */
4902 Lisp_Object min_width, min_height;
cbff28e8
RS
4903 /* A vector, each of whose elements is a struct saved_window
4904 for one window. */
7ab12479
JB
4905 Lisp_Object saved_windows;
4906 };
ff06df24 4907
cbff28e8 4908/* This is saved as a Lisp_Vector */
7ab12479 4909struct saved_window
ea68264b
GM
4910{
4911 /* these first two must agree with struct Lisp_Vector in lisp.h */
4912 EMACS_INT size_from_Lisp_Vector_struct;
4913 struct Lisp_Vector *next_from_Lisp_Vector_struct;
7ab12479 4914
ea68264b
GM
4915 Lisp_Object window;
4916 Lisp_Object buffer, start, pointm, mark;
4917 Lisp_Object left, top, width, height, hscroll, min_hscroll;
4918 Lisp_Object parent, prev;
4919 Lisp_Object start_at_line_beg;
4920 Lisp_Object display_table;
4921 Lisp_Object orig_top, orig_height;
4922};
4923
4924#define SAVED_WINDOW_VECTOR_SIZE 17 /* Arg to Fmake_vector */
7ab12479
JB
4925
4926#define SAVED_WINDOW_N(swv,n) \
4927 ((struct saved_window *) (XVECTOR ((swv)->contents[(n)])))
4928
4929DEFUN ("window-configuration-p", Fwindow_configuration_p, Swindow_configuration_p, 1, 1, 0,
fdb82f93
PJ
4930 doc: /* Return t if OBJECT is a window-configuration object. */)
4931 (object)
413430c5 4932 Lisp_Object object;
7ab12479 4933{
413430c5 4934 if (WINDOW_CONFIGURATIONP (object))
7ab12479
JB
4935 return Qt;
4936 return Qnil;
4937}
4938
3f8ab7bd 4939DEFUN ("window-configuration-frame", Fwindow_configuration_frame, Swindow_configuration_frame, 1, 1, 0,
fdb82f93
PJ
4940 doc: /* Return the frame that CONFIG, a window-configuration object, is about. */)
4941 (config)
3f8ab7bd
RS
4942 Lisp_Object config;
4943{
4944 register struct save_window_data *data;
4945 struct Lisp_Vector *saved_windows;
4946
4947 if (! WINDOW_CONFIGURATIONP (config))
4948 wrong_type_argument (Qwindow_configuration_p, config);
4949
4950 data = (struct save_window_data *) XVECTOR (config);
4951 saved_windows = XVECTOR (data->saved_windows);
4952 return XWINDOW (SAVED_WINDOW_N (saved_windows, 0)->window)->frame;
4953}
4954
d5b2799e 4955DEFUN ("set-window-configuration", Fset_window_configuration,
fdb82f93
PJ
4956 Sset_window_configuration, 1, 1, 0,
4957 doc: /* Set the configuration of windows and buffers as specified by CONFIGURATION.
4958CONFIGURATION must be a value previously returned
4959by `current-window-configuration' (which see).
4960If CONFIGURATION was made from a frame that is now deleted,
4961only frame-independent values can be restored. In this case,
4962the return value is nil. Otherwise the value is t. */)
4963 (configuration)
2f83aebe 4964 Lisp_Object configuration;
7ab12479 4965{
7ab12479
JB
4966 register struct save_window_data *data;
4967 struct Lisp_Vector *saved_windows;
7ab12479 4968 Lisp_Object new_current_buffer;
fd482be5 4969 Lisp_Object frame;
44fa5b1e 4970 FRAME_PTR f;
72695e47 4971 int old_point = -1;
7ab12479 4972
017b2bad 4973 while (!WINDOW_CONFIGURATIONP (configuration))
3f8ab7bd 4974 wrong_type_argument (Qwindow_configuration_p, configuration);
7ab12479 4975
2f83aebe 4976 data = (struct save_window_data *) XVECTOR (configuration);
7ab12479
JB
4977 saved_windows = XVECTOR (data->saved_windows);
4978
7ab12479 4979 new_current_buffer = data->current_buffer;
265a9e55 4980 if (NILP (XBUFFER (new_current_buffer)->name))
7ab12479 4981 new_current_buffer = Qnil;
72695e47 4982 else
73cadfc1
DK
4983 {
4984 if (XBUFFER (new_current_buffer) == current_buffer)
4985 old_point = PT;
4986 else
4987 old_point = BUF_PT (XBUFFER (new_current_buffer));
4988 }
7ab12479 4989
fd482be5
JB
4990 frame = XWINDOW (SAVED_WINDOW_N (saved_windows, 0)->window)->frame;
4991 f = XFRAME (frame);
aa35ba9e 4992
fd482be5
JB
4993 /* If f is a dead frame, don't bother rebuilding its window tree.
4994 However, there is other stuff we should still try to do below. */
4995 if (FRAME_LIVE_P (f))
7ab12479 4996 {
fd482be5
JB
4997 register struct window *w;
4998 register struct saved_window *p;
5500c422
GM
4999 struct window *root_window;
5000 struct window **leaf_windows;
5001 int n_leaf_windows;
c4280705 5002 int k, i, n;
fd482be5
JB
5003
5004 /* If the frame has been resized since this window configuration was
5005 made, we change the frame to the size specified in the
5006 configuration, restore the configuration, and then resize it
5007 back. We keep track of the prevailing height in these variables. */
5008 int previous_frame_height = FRAME_HEIGHT (f);
5009 int previous_frame_width = FRAME_WIDTH (f);
8f6ea2e9 5010 int previous_frame_menu_bar_lines = FRAME_MENU_BAR_LINES (f);
9ea173e8 5011 int previous_frame_tool_bar_lines = FRAME_TOOL_BAR_LINES (f);
fd482be5 5012
d2b35234
RS
5013 /* The mouse highlighting code could get screwed up
5014 if it runs during this. */
5015 BLOCK_INPUT;
5016
fd482be5
JB
5017 if (XFASTINT (data->frame_height) != previous_frame_height
5018 || XFASTINT (data->frame_width) != previous_frame_width)
f8ad443a 5019 change_frame_size (f, XFASTINT (data->frame_height),
2b653806 5020 XFASTINT (data->frame_width), 0, 0, 0);
e3678b64 5021#if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
8f6ea2e9
KH
5022 if (XFASTINT (data->frame_menu_bar_lines)
5023 != previous_frame_menu_bar_lines)
f8ad443a 5024 x_set_menu_bar_lines (f, data->frame_menu_bar_lines, make_number (0));
4314246f 5025#ifdef HAVE_WINDOW_SYSTEM
9ea173e8
GM
5026 if (XFASTINT (data->frame_tool_bar_lines)
5027 != previous_frame_tool_bar_lines)
5028 x_set_tool_bar_lines (f, data->frame_tool_bar_lines, make_number (0));
4314246f 5029#endif
217f2871 5030#endif
fd482be5 5031
719eaeb1
GM
5032 /* "Swap out" point from the selected window
5033 into its buffer. We do this now, before
5034 restoring the window contents, and prevent it from
5035 being done later on when we select a new window. */
596ae0cf
RS
5036 if (! NILP (XWINDOW (selected_window)->buffer))
5037 {
5038 w = XWINDOW (selected_window);
5039 set_marker_both (w->pointm,
5040 w->buffer,
5041 BUF_PT (XBUFFER (w->buffer)),
5042 BUF_PT_BYTE (XBUFFER (w->buffer)));
5043 }
5044
fd482be5 5045 windows_or_buffers_changed++;
29aeee73 5046 FRAME_WINDOW_SIZES_CHANGED (f) = 1;
fd482be5 5047
5500c422
GM
5048 /* Problem: Freeing all matrices and later allocating them again
5049 is a serious redisplay flickering problem. What we would
5050 really like to do is to free only those matrices not reused
5051 below. */
5052 root_window = XWINDOW (FRAME_ROOT_WINDOW (f));
5053 leaf_windows
5054 = (struct window **) alloca (count_windows (root_window)
5055 * sizeof (struct window *));
5056 n_leaf_windows = get_leaf_windows (root_window, leaf_windows, 0);
5057
756b6edc
RS
5058 /* Temporarily avoid any problems with windows that are smaller
5059 than they are supposed to be. */
5060 window_min_height = 1;
5061 window_min_width = 1;
5062
fd482be5
JB
5063 /* Kludge Alert!
5064 Mark all windows now on frame as "deleted".
5065 Restoring the new configuration "undeletes" any that are in it.
37962e60 5066
fd482be5
JB
5067 Save their current buffers in their height fields, since we may
5068 need it later, if a buffer saved in the configuration is now
5069 dead. */
5070 delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f)));
5071
5072 for (k = 0; k < saved_windows->size; k++)
5073 {
5074 p = SAVED_WINDOW_N (saved_windows, k);
5075 w = XWINDOW (p->window);
5076 w->next = Qnil;
7ab12479 5077
fd482be5
JB
5078 if (!NILP (p->parent))
5079 w->parent = SAVED_WINDOW_N (saved_windows,
5080 XFASTINT (p->parent))->window;
5081 else
5082 w->parent = Qnil;
7ab12479 5083
fd482be5 5084 if (!NILP (p->prev))
7ab12479 5085 {
fd482be5
JB
5086 w->prev = SAVED_WINDOW_N (saved_windows,
5087 XFASTINT (p->prev))->window;
5088 XWINDOW (w->prev)->next = p->window;
5089 }
5090 else
5091 {
5092 w->prev = Qnil;
5093 if (!NILP (w->parent))
5094 {
5095 if (EQ (p->width, XWINDOW (w->parent)->width))
5096 {
5097 XWINDOW (w->parent)->vchild = p->window;
5098 XWINDOW (w->parent)->hchild = Qnil;
5099 }
5100 else
5101 {
5102 XWINDOW (w->parent)->hchild = p->window;
5103 XWINDOW (w->parent)->vchild = Qnil;
5104 }
5105 }
5106 }
5107
5108 /* If we squirreled away the buffer in the window's height,
5109 restore it now. */
017b2bad 5110 if (BUFFERP (w->height))
fd482be5
JB
5111 w->buffer = w->height;
5112 w->left = p->left;
5113 w->top = p->top;
5114 w->width = p->width;
5115 w->height = p->height;
5116 w->hscroll = p->hscroll;
ea68264b 5117 w->min_hscroll = p->min_hscroll;
fd482be5 5118 w->display_table = p->display_table;
a1d58e5b
GM
5119 w->orig_top = p->orig_top;
5120 w->orig_height = p->orig_height;
d834a2e9 5121 XSETFASTINT (w->last_modified, 0);
3cd21523 5122 XSETFASTINT (w->last_overlay_modified, 0);
fd482be5
JB
5123
5124 /* Reinstall the saved buffer and pointers into it. */
5125 if (NILP (p->buffer))
5126 w->buffer = p->buffer;
5127 else
5128 {
5129 if (!NILP (XBUFFER (p->buffer)->name))
5130 /* If saved buffer is alive, install it. */
5131 {
5132 w->buffer = p->buffer;
5133 w->start_at_line_beg = p->start_at_line_beg;
b73ea88e
RS
5134 set_marker_restricted (w->start, p->start, w->buffer);
5135 set_marker_restricted (w->pointm, p->pointm, w->buffer);
fd482be5 5136 Fset_marker (XBUFFER (w->buffer)->mark,
b73ea88e 5137 p->mark, w->buffer);
fd482be5
JB
5138
5139 /* As documented in Fcurrent_window_configuration, don't
243a5ce6
RS
5140 restore the location of point in the buffer which was
5141 current when the window configuration was recorded. */
6b54027b
RS
5142 if (!EQ (p->buffer, new_current_buffer)
5143 && XBUFFER (p->buffer) == current_buffer)
fd482be5
JB
5144 Fgoto_char (w->pointm);
5145 }
52a68e98
RS
5146 else if (NILP (w->buffer) || NILP (XBUFFER (w->buffer)->name))
5147 /* Else unless window has a live buffer, get one. */
7ab12479 5148 {
fd482be5
JB
5149 w->buffer = Fcdr (Fcar (Vbuffer_alist));
5150 /* This will set the markers to beginning of visible
5151 range. */
5152 set_marker_restricted (w->start, make_number (0), w->buffer);
5153 set_marker_restricted (w->pointm, make_number (0),w->buffer);
5154 w->start_at_line_beg = Qt;
7ab12479
JB
5155 }
5156 else
fd482be5 5157 /* Keeping window's old buffer; make sure the markers
52a68e98 5158 are real. */
7ab12479 5159 {
fd482be5
JB
5160 /* Set window markers at start of visible range. */
5161 if (XMARKER (w->start)->buffer == 0)
5162 set_marker_restricted (w->start, make_number (0),
5163 w->buffer);
5164 if (XMARKER (w->pointm)->buffer == 0)
b73ea88e
RS
5165 set_marker_restricted_both (w->pointm, w->buffer,
5166 BUF_PT (XBUFFER (w->buffer)),
5167 BUF_PT_BYTE (XBUFFER (w->buffer)));
fd482be5 5168 w->start_at_line_beg = Qt;
7ab12479
JB
5169 }
5170 }
5171 }
9ace597f 5172
fd482be5 5173 FRAME_ROOT_WINDOW (f) = data->root_window;
719eaeb1
GM
5174 /* Prevent "swapping out point" in the old selected window
5175 using the buffer that has been restored into it.
72695e47
RS
5176 Use the point value from the beginning of this function
5177 since unshow_buffer (called from delete_all_subwindows)
5178 could have altered it. */
719eaeb1 5179 selected_window = Qnil;
243a5ce6
RS
5180 if (EQ (XWINDOW (data->current_window)->buffer, new_current_buffer))
5181 set_marker_restricted (XWINDOW (data->current_window)->pointm,
18787f5e 5182 make_number (old_point),
243a5ce6
RS
5183 XWINDOW (data->current_window)->buffer);
5184
fd482be5 5185 Fselect_window (data->current_window);
396a830c
RS
5186 XBUFFER (XWINDOW (selected_window)->buffer)->last_selected_window
5187 = selected_window;
7ab12479 5188
db269683 5189 if (NILP (data->focus_frame)
017b2bad 5190 || (FRAMEP (data->focus_frame)
db269683
JB
5191 && FRAME_LIVE_P (XFRAME (data->focus_frame))))
5192 Fredirect_frame_focus (frame, data->focus_frame);
7ab12479 5193
fd482be5
JB
5194#if 0 /* I don't understand why this is needed, and it causes problems
5195 when the frame's old selected window has been deleted. */
e4e59717 5196 if (f != selected_frame && FRAME_WINDOW_P (f))
9a7c6fc3 5197 do_switch_frame (WINDOW_FRAME (XWINDOW (data->root_window)),
d12f6f83 5198 0, 0);
fd482be5
JB
5199#endif
5200
5201 /* Set the screen height to the value it had before this function. */
5202 if (previous_frame_height != FRAME_HEIGHT (f)
5203 || previous_frame_width != FRAME_WIDTH (f))
5204 change_frame_size (f, previous_frame_height, previous_frame_width,
2b653806 5205 0, 0, 0);
e3678b64 5206#if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
8f6ea2e9 5207 if (previous_frame_menu_bar_lines != FRAME_MENU_BAR_LINES (f))
f8ad443a
AS
5208 x_set_menu_bar_lines (f, make_number (previous_frame_menu_bar_lines),
5209 make_number (0));
4314246f 5210#ifdef HAVE_WINDOW_SYSTEM
9ea173e8
GM
5211 if (previous_frame_tool_bar_lines != FRAME_TOOL_BAR_LINES (f))
5212 x_set_tool_bar_lines (f, make_number (previous_frame_tool_bar_lines),
5213 make_number (0));
4314246f 5214#endif
217f2871 5215#endif
d2b35234 5216
5500c422 5217 /* Now, free glyph matrices in windows that were not reused. */
c4280705
GM
5218 for (i = n = 0; i < n_leaf_windows; ++i)
5219 {
5220 if (NILP (leaf_windows[i]->buffer))
5221 {
5222 /* Assert it's not reused as a combination. */
5223 xassert (NILP (leaf_windows[i]->hchild)
5224 && NILP (leaf_windows[i]->vchild));
5225 free_window_matrices (leaf_windows[i]);
c4280705
GM
5226 }
5227 else if (EQ (leaf_windows[i]->buffer, new_current_buffer))
5228 ++n;
5229 }
5500c422
GM
5230
5231 adjust_glyphs (f);
5232
d2b35234 5233 UNBLOCK_INPUT;
756b6edc 5234
478292ed
RS
5235 /* Fselect_window will have made f the selected frame, so we
5236 reselect the proper frame here. Fhandle_switch_frame will change the
5237 selected window too, but that doesn't make the call to
5238 Fselect_window above totally superfluous; it still sets f's
5239 selected window. */
5240 if (FRAME_LIVE_P (XFRAME (data->selected_frame)))
d12f6f83 5241 do_switch_frame (data->selected_frame, 0, 0);
478292ed
RS
5242
5243 if (! NILP (Vwindow_configuration_change_hook)
5244 && ! NILP (Vrun_hooks))
5245 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
5246 }
bdc727bf
JB
5247
5248 if (!NILP (new_current_buffer))
243a5ce6 5249 Fset_buffer (new_current_buffer);
bdc727bf 5250
478292ed
RS
5251 /* Restore the minimum heights recorded in the configuration. */
5252 window_min_height = XINT (data->min_height);
5253 window_min_width = XINT (data->min_width);
543f5fb1 5254
478292ed 5255 Vminibuf_scroll_window = data->minibuf_scroll_window;
3dbab091 5256 minibuf_selected_window = data->minibuf_selected_window;
543f5fb1 5257
3f8ab7bd 5258 return (FRAME_LIVE_P (f) ? Qt : Qnil);
7ab12479
JB
5259}
5260
44fa5b1e 5261/* Mark all windows now on frame as deleted
7ab12479
JB
5262 by setting their buffers to nil. */
5263
fd482be5 5264void
7ab12479
JB
5265delete_all_subwindows (w)
5266 register struct window *w;
5267{
265a9e55 5268 if (!NILP (w->next))
7ab12479 5269 delete_all_subwindows (XWINDOW (w->next));
265a9e55 5270 if (!NILP (w->vchild))
7ab12479 5271 delete_all_subwindows (XWINDOW (w->vchild));
265a9e55 5272 if (!NILP (w->hchild))
7ab12479 5273 delete_all_subwindows (XWINDOW (w->hchild));
605be8af
JB
5274
5275 w->height = w->buffer; /* See Fset_window_configuration for excuse. */
5276
86e48436
RS
5277 if (!NILP (w->buffer))
5278 unshow_buffer (w);
5279
605be8af
JB
5280 /* We set all three of these fields to nil, to make sure that we can
5281 distinguish this dead window from any live window. Live leaf
5282 windows will have buffer set, and combination windows will have
5283 vchild or hchild set. */
5284 w->buffer = Qnil;
5285 w->vchild = Qnil;
5286 w->hchild = Qnil;
acf70840
GM
5287
5288 Vwindow_list = Qnil;
7ab12479
JB
5289}
5290\f
5291static int
5292count_windows (window)
5293 register struct window *window;
5294{
5295 register int count = 1;
265a9e55 5296 if (!NILP (window->next))
7ab12479 5297 count += count_windows (XWINDOW (window->next));
265a9e55 5298 if (!NILP (window->vchild))
7ab12479 5299 count += count_windows (XWINDOW (window->vchild));
265a9e55 5300 if (!NILP (window->hchild))
7ab12479
JB
5301 count += count_windows (XWINDOW (window->hchild));
5302 return count;
5303}
5304
5500c422
GM
5305
5306/* Fill vector FLAT with leaf windows under W, starting at index I.
5307 Value is last index + 1. */
5308
5309static int
5310get_leaf_windows (w, flat, i)
5311 struct window *w;
5312 struct window **flat;
5313 int i;
5314{
5315 while (w)
5316 {
5317 if (!NILP (w->hchild))
5318 i = get_leaf_windows (XWINDOW (w->hchild), flat, i);
5319 else if (!NILP (w->vchild))
5320 i = get_leaf_windows (XWINDOW (w->vchild), flat, i);
5321 else
5322 flat[i++] = w;
5323
5324 w = NILP (w->next) ? 0 : XWINDOW (w->next);
5325 }
5326
5327 return i;
5328}
5329
5330
5331/* Return a pointer to the glyph W's physical cursor is on. Value is
5332 null if W's current matrix is invalid, so that no meaningfull glyph
5333 can be returned. */
5334
5335struct glyph *
5336get_phys_cursor_glyph (w)
5337 struct window *w;
5338{
5339 struct glyph_row *row;
5340 struct glyph *glyph;
5341
5342 if (w->phys_cursor.vpos >= 0
5343 && w->phys_cursor.vpos < w->current_matrix->nrows
5344 && (row = MATRIX_ROW (w->current_matrix, w->phys_cursor.vpos),
5345 row->enabled_p)
5346 && row->used[TEXT_AREA] > w->phys_cursor.hpos)
5347 glyph = row->glyphs[TEXT_AREA] + w->phys_cursor.hpos;
5348 else
5349 glyph = NULL;
5350
5351 return glyph;
5352}
5353
5354
7ab12479
JB
5355static int
5356save_window_save (window, vector, i)
5357 Lisp_Object window;
5358 struct Lisp_Vector *vector;
5359 int i;
5360{
5361 register struct saved_window *p;
5362 register struct window *w;
5363 register Lisp_Object tem;
5364
265a9e55 5365 for (;!NILP (window); window = w->next)
7ab12479
JB
5366 {
5367 p = SAVED_WINDOW_N (vector, i);
5368 w = XWINDOW (window);
5369
d834a2e9 5370 XSETFASTINT (w->temslot, i++);
7ab12479
JB
5371 p->window = window;
5372 p->buffer = w->buffer;
5373 p->left = w->left;
5374 p->top = w->top;
5375 p->width = w->width;
5376 p->height = w->height;
5377 p->hscroll = w->hscroll;
ea68264b 5378 p->min_hscroll = w->min_hscroll;
7ab12479 5379 p->display_table = w->display_table;
a1d58e5b
GM
5380 p->orig_top = w->orig_top;
5381 p->orig_height = w->orig_height;
265a9e55 5382 if (!NILP (w->buffer))
7ab12479
JB
5383 {
5384 /* Save w's value of point in the window configuration.
5385 If w is the selected window, then get the value of point
5386 from the buffer; pointm is garbage in the selected window. */
5387 if (EQ (window, selected_window))
5388 {
5389 p->pointm = Fmake_marker ();
b73ea88e
RS
5390 set_marker_both (p->pointm, w->buffer,
5391 BUF_PT (XBUFFER (w->buffer)),
5392 BUF_PT_BYTE (XBUFFER (w->buffer)));
7ab12479
JB
5393 }
5394 else
eeb82665 5395 p->pointm = Fcopy_marker (w->pointm, Qnil);
7ab12479 5396
eeb82665 5397 p->start = Fcopy_marker (w->start, Qnil);
7ab12479
JB
5398 p->start_at_line_beg = w->start_at_line_beg;
5399
5400 tem = XBUFFER (w->buffer)->mark;
eeb82665 5401 p->mark = Fcopy_marker (tem, Qnil);
7ab12479
JB
5402 }
5403 else
5404 {
5405 p->pointm = Qnil;
5406 p->start = Qnil;
5407 p->mark = Qnil;
5408 p->start_at_line_beg = Qnil;
5409 }
5410
265a9e55 5411 if (NILP (w->parent))
7ab12479
JB
5412 p->parent = Qnil;
5413 else
5414 p->parent = XWINDOW (w->parent)->temslot;
5415
265a9e55 5416 if (NILP (w->prev))
7ab12479
JB
5417 p->prev = Qnil;
5418 else
5419 p->prev = XWINDOW (w->prev)->temslot;
5420
265a9e55 5421 if (!NILP (w->vchild))
7ab12479 5422 i = save_window_save (w->vchild, vector, i);
265a9e55 5423 if (!NILP (w->hchild))
7ab12479
JB
5424 i = save_window_save (w->hchild, vector, i);
5425 }
5426
5427 return i;
5428}
5429
a0d76c27 5430DEFUN ("current-window-configuration", Fcurrent_window_configuration,
fdb82f93
PJ
5431 Scurrent_window_configuration, 0, 1, 0,
5432 doc: /* Return an object representing the current window configuration of FRAME.
5433If FRAME is nil or omitted, use the selected frame.
5434This describes the number of windows, their sizes and current buffers,
5435and for each displayed buffer, where display starts, and the positions of
5436point and mark. An exception is made for point in the current buffer:
5437its value is -not- saved.
5438This also records the currently selected frame, and FRAME's focus
5439redirection (see `redirect-frame-focus'). */)
5440 (frame)
44fa5b1e 5441 Lisp_Object frame;
7ab12479
JB
5442{
5443 register Lisp_Object tem;
5444 register int n_windows;
5445 register struct save_window_data *data;
da2792e0 5446 register struct Lisp_Vector *vec;
7ab12479 5447 register int i;
44fa5b1e 5448 FRAME_PTR f;
43bad991 5449
44fa5b1e 5450 if (NILP (frame))
1ae1a37d 5451 frame = selected_frame;
b7826503 5452 CHECK_LIVE_FRAME (frame);
1ae1a37d 5453 f = XFRAME (frame);
7ab12479 5454
44fa5b1e 5455 n_windows = count_windows (XWINDOW (FRAME_ROOT_WINDOW (f)));
26605be9 5456 vec = allocate_other_vector (VECSIZE (struct save_window_data));
da2792e0
KH
5457 data = (struct save_window_data *)vec;
5458
d834a2e9
KH
5459 XSETFASTINT (data->frame_width, FRAME_WIDTH (f));
5460 XSETFASTINT (data->frame_height, FRAME_HEIGHT (f));
5461 XSETFASTINT (data->frame_menu_bar_lines, FRAME_MENU_BAR_LINES (f));
9ea173e8 5462 XSETFASTINT (data->frame_tool_bar_lines, FRAME_TOOL_BAR_LINES (f));
1ae1a37d 5463 data->selected_frame = selected_frame;
44fa5b1e 5464 data->current_window = FRAME_SELECTED_WINDOW (f);
74112613 5465 XSETBUFFER (data->current_buffer, current_buffer);
3f49fddc 5466 data->minibuf_scroll_window = minibuf_level > 0 ? Vminibuf_scroll_window : Qnil;
3dbab091 5467 data->minibuf_selected_window = minibuf_level > 0 ? minibuf_selected_window : Qnil;
44fa5b1e 5468 data->root_window = FRAME_ROOT_WINDOW (f);
bdc727bf 5469 data->focus_frame = FRAME_FOCUS_FRAME (f);
74112613
KH
5470 XSETINT (data->min_height, window_min_height);
5471 XSETINT (data->min_width, window_min_width);
7ab12479
JB
5472 tem = Fmake_vector (make_number (n_windows), Qnil);
5473 data->saved_windows = tem;
5474 for (i = 0; i < n_windows; i++)
5475 XVECTOR (tem)->contents[i]
5476 = Fmake_vector (make_number (SAVED_WINDOW_VECTOR_SIZE), Qnil);
a1d58e5b 5477 save_window_save (FRAME_ROOT_WINDOW (f), XVECTOR (tem), 0);
74112613 5478 XSETWINDOW_CONFIGURATION (tem, data);
7ab12479
JB
5479 return (tem);
5480}
5481
5482DEFUN ("save-window-excursion", Fsave_window_excursion, Ssave_window_excursion,
fdb82f93
PJ
5483 0, UNEVALLED, 0,
5484 doc: /* Execute body, preserving window sizes and contents.
5485Restore which buffer appears in which window, where display starts,
5486and the value of point and mark for each window.
5487Also restore the choice of selected window.
5488Also restore which buffer is current.
6924dda6
MB
5489Does not restore the value of point in current buffer.
5490usage: (save-window-excursion BODY ...) */)
fdb82f93 5491 (args)
7ab12479
JB
5492 Lisp_Object args;
5493{
5494 register Lisp_Object val;
aed13378 5495 register int count = SPECPDL_INDEX ();
7ab12479
JB
5496
5497 record_unwind_protect (Fset_window_configuration,
43bad991 5498 Fcurrent_window_configuration (Qnil));
7ab12479
JB
5499 val = Fprogn (args);
5500 return unbind_to (count, val);
5501}
5500c422
GM
5502
5503\f
5504/***********************************************************************
5505 Marginal Areas
5506 ***********************************************************************/
5507
5508DEFUN ("set-window-margins", Fset_window_margins, Sset_window_margins,
a7bdfc08 5509 2, 3, 0,
fdb82f93
PJ
5510 doc: /* Set width of marginal areas of window WINDOW.
5511If window is nil, set margins of the currently selected window.
5512First parameter LEFT-WIDTH specifies the number of character
5513cells to reserve for the left marginal area. Second parameter
5514RIGHT-WIDTH does the same for the right marginal area.
5515A nil width parameter means no margin. */)
5516 (window, left, right)
5500c422
GM
5517 Lisp_Object window, left, right;
5518{
5519 struct window *w = decode_window (window);
5500c422
GM
5520
5521 if (!NILP (left))
b7826503 5522 CHECK_NUMBER_OR_FLOAT (left);
5500c422 5523 if (!NILP (right))
b7826503 5524 CHECK_NUMBER_OR_FLOAT (right);
5500c422
GM
5525
5526 /* Check widths < 0 and translate a zero width to nil.
5527 Margins that are too wide have to be checked elsewhere. */
5528 if ((INTEGERP (left) && XINT (left) < 0)
7539e11f 5529 || (FLOATP (left) && XFLOAT_DATA (left) <= 0))
5500c422
GM
5530 XSETFASTINT (left, 0);
5531 if (INTEGERP (left) && XFASTINT (left) == 0)
5532 left = Qnil;
5533
5534 if ((INTEGERP (right) && XINT (right) < 0)
7539e11f 5535 || (FLOATP (right) && XFLOAT_DATA (right) <= 0))
5500c422
GM
5536 XSETFASTINT (right, 0);
5537 if (INTEGERP (right) && XFASTINT (right) == 0)
5538 right = Qnil;
5539
5540 w->left_margin_width = left;
5541 w->right_margin_width = right;
5542
5543 ++windows_or_buffers_changed;
5544 adjust_glyphs (XFRAME (WINDOW_FRAME (w)));
5545 return Qnil;
5546}
5547
5548
5549DEFUN ("window-margins", Fwindow_margins, Swindow_margins,
5550 0, 1, 0,
fdb82f93
PJ
5551 doc: /* Get width of marginal areas of window WINDOW.
5552If WINDOW is omitted or nil, use the currently selected window.
5553Value is a cons of the form (LEFT-WIDTH . RIGHT-WIDTH).
5554If a marginal area does not exist, its width will be returned
5555as nil. */)
5556 (window)
5500c422
GM
5557 Lisp_Object window;
5558{
5559 struct window *w = decode_window (window);
5560 return Fcons (w->left_margin_width, w->right_margin_width);
5561}
5562
5563
7ab12479 5564\f
5500c422
GM
5565/***********************************************************************
5566 Smooth scrolling
5567 ***********************************************************************/
5568
5569DEFUN ("window-vscroll", Fwindow_vscroll, Swindow_vscroll, 0, 1, 0,
fdb82f93
PJ
5570 doc: /* Return the amount by which WINDOW is scrolled vertically.
5571Use the selected window if WINDOW is nil or omitted.
5572Value is a multiple of the canonical character height of WINDOW. */)
5573 (window)
5500c422
GM
5574 Lisp_Object window;
5575{
47004952 5576 Lisp_Object result;
5500c422
GM
5577 struct frame *f;
5578 struct window *w;
5579
5580 if (NILP (window))
5581 window = selected_window;
47004952 5582 else
b7826503 5583 CHECK_WINDOW (window);
5500c422
GM
5584 w = XWINDOW (window);
5585 f = XFRAME (w->frame);
5586
5587 if (FRAME_WINDOW_P (f))
47004952 5588 result = CANON_Y_FROM_PIXEL_Y (f, -w->vscroll);
5500c422 5589 else
47004952
GM
5590 result = make_number (0);
5591 return result;
5500c422
GM
5592}
5593
5594
5595DEFUN ("set-window-vscroll", Fset_window_vscroll, Sset_window_vscroll,
47004952 5596 2, 2, 0,
fdb82f93 5597 doc: /* Set amount by which WINDOW should be scrolled vertically to VSCROLL.
b0f906c8
EZ
5598WINDOW nil means use the selected window. VSCROLL is a non-negative
5599multiple of the canonical character height of WINDOW. */)
fdb82f93 5600 (window, vscroll)
47004952 5601 Lisp_Object window, vscroll;
5500c422
GM
5602{
5603 struct window *w;
5604 struct frame *f;
5605
5500c422
GM
5606 if (NILP (window))
5607 window = selected_window;
47004952 5608 else
b7826503
PJ
5609 CHECK_WINDOW (window);
5610 CHECK_NUMBER_OR_FLOAT (vscroll);
47004952 5611
5500c422
GM
5612 w = XWINDOW (window);
5613 f = XFRAME (w->frame);
5614
5615 if (FRAME_WINDOW_P (f))
5616 {
5617 int old_dy = w->vscroll;
47004952
GM
5618
5619 w->vscroll = - CANON_Y_UNIT (f) * XFLOATINT (vscroll);
5620 w->vscroll = min (w->vscroll, 0);
5500c422
GM
5621
5622 /* Adjust glyph matrix of the frame if the virtual display
5623 area becomes larger than before. */
5624 if (w->vscroll < 0 && w->vscroll < old_dy)
5625 adjust_glyphs (f);
5626
5627 /* Prevent redisplay shortcuts. */
b1599b4c 5628 XBUFFER (w->buffer)->prevent_redisplay_optimizations_p = 1;
5500c422
GM
5629 }
5630
47004952 5631 return Fwindow_vscroll (window);
5500c422
GM
5632}
5633
7bbb5782
GM
5634\f
5635/* Call FN for all leaf windows on frame F. FN is called with the
5636 first argument being a pointer to the leaf window, and with
f95464e4 5637 additional argument USER_DATA. Stops when FN returns 0. */
7bbb5782
GM
5638
5639void
f95464e4 5640foreach_window (f, fn, user_data)
7bbb5782 5641 struct frame *f;
f95464e4
GM
5642 int (* fn) P_ ((struct window *, void *));
5643 void *user_data;
7bbb5782 5644{
f95464e4 5645 foreach_window_1 (XWINDOW (FRAME_ROOT_WINDOW (f)), fn, user_data);
7bbb5782
GM
5646}
5647
5648
5649/* Helper function for foreach_window. Call FN for all leaf windows
5650 reachable from W. FN is called with the first argument being a
f95464e4 5651 pointer to the leaf window, and with additional argument USER_DATA.
67492200 5652 Stop when FN returns 0. Value is 0 if stopped by FN. */
7bbb5782 5653
67492200 5654static int
f95464e4 5655foreach_window_1 (w, fn, user_data)
7bbb5782 5656 struct window *w;
f95464e4
GM
5657 int (* fn) P_ ((struct window *, void *));
5658 void *user_data;
7bbb5782 5659{
67492200
GM
5660 int cont;
5661
5662 for (cont = 1; w && cont;)
7bbb5782
GM
5663 {
5664 if (!NILP (w->hchild))
f95464e4 5665 cont = foreach_window_1 (XWINDOW (w->hchild), fn, user_data);
7bbb5782 5666 else if (!NILP (w->vchild))
f95464e4 5667 cont = foreach_window_1 (XWINDOW (w->vchild), fn, user_data);
0f532a9a
GM
5668 else
5669 cont = fn (w, user_data);
7bbb5782
GM
5670
5671 w = NILP (w->next) ? 0 : XWINDOW (w->next);
5672 }
67492200
GM
5673
5674 return cont;
7bbb5782
GM
5675}
5676
5677
6d194a45 5678/* Freeze or unfreeze the window start of W unless it is a
f95464e4 5679 mini-window or the selected window. FREEZE_P non-null means freeze
7bbb5782
GM
5680 the window start. */
5681
67492200 5682static int
7bbb5782
GM
5683freeze_window_start (w, freeze_p)
5684 struct window *w;
f95464e4 5685 void *freeze_p;
7bbb5782
GM
5686{
5687 if (w == XWINDOW (selected_window)
5688 || MINI_WINDOW_P (w)
5689 || (MINI_WINDOW_P (XWINDOW (selected_window))
24d744ac 5690 && ! NILP (Vminibuf_scroll_window)
7bbb5782 5691 && w == XWINDOW (Vminibuf_scroll_window)))
f95464e4 5692 freeze_p = NULL;
7bbb5782 5693
f95464e4 5694 w->frozen_window_start_p = freeze_p != NULL;
67492200 5695 return 1;
7bbb5782
GM
5696}
5697
5698
5699/* Freeze or unfreeze the window starts of all leaf windows on frame
5700 F, except the selected window and a mini-window. FREEZE_P non-zero
5701 means freeze the window start. */
5702
5703void
5704freeze_window_starts (f, freeze_p)
5705 struct frame *f;
5706 int freeze_p;
5707{
cbccabec 5708 foreach_window (f, freeze_window_start, (void *) (freeze_p ? f : 0));
7bbb5782 5709}
5500c422
GM
5710
5711\f
5712/***********************************************************************
5713 Initialization
5714 ***********************************************************************/
5715
cbff28e8
RS
5716/* Return 1 if window configurations C1 and C2
5717 describe the same state of affairs. This is used by Fequal. */
5718
5719int
2f8274be 5720compare_window_configurations (c1, c2, ignore_positions)
cbff28e8 5721 Lisp_Object c1, c2;
2f8274be 5722 int ignore_positions;
cbff28e8
RS
5723{
5724 register struct save_window_data *d1, *d2;
5725 struct Lisp_Vector *sw1, *sw2;
5726 int i;
5727
4d3edcb4
GM
5728 if (!WINDOW_CONFIGURATIONP (c1))
5729 wrong_type_argument (Qwindow_configuration_p, c1);
5730 if (!WINDOW_CONFIGURATIONP (c2))
5731 wrong_type_argument (Qwindow_configuration_p, c2);
5732
cbff28e8
RS
5733 d1 = (struct save_window_data *) XVECTOR (c1);
5734 d2 = (struct save_window_data *) XVECTOR (c2);
5735 sw1 = XVECTOR (d1->saved_windows);
5736 sw2 = XVECTOR (d2->saved_windows);
5737
5738 if (! EQ (d1->frame_width, d2->frame_width))
5739 return 0;
5740 if (! EQ (d1->frame_height, d2->frame_height))
5741 return 0;
5742 if (! EQ (d1->frame_menu_bar_lines, d2->frame_menu_bar_lines))
5743 return 0;
5744 if (! EQ (d1->selected_frame, d2->selected_frame))
5745 return 0;
5746 /* Don't compare the current_window field directly.
5747 Instead see w1_is_current and w2_is_current, below. */
5748 if (! EQ (d1->current_buffer, d2->current_buffer))
5749 return 0;
2f8274be 5750 if (! ignore_positions)
3f49fddc
KS
5751 {
5752 if (! EQ (d1->minibuf_scroll_window, d2->minibuf_scroll_window))
5753 return 0;
5754 if (! EQ (d1->minibuf_selected_window, d2->minibuf_selected_window))
5755 return 0;
5756 }
cbff28e8
RS
5757 /* Don't compare the root_window field.
5758 We don't require the two configurations
5759 to use the same window object,
5760 and the two root windows must be equivalent
5761 if everything else compares equal. */
5762 if (! EQ (d1->focus_frame, d2->focus_frame))
5763 return 0;
5764 if (! EQ (d1->min_width, d2->min_width))
5765 return 0;
5766 if (! EQ (d1->min_height, d2->min_height))
5767 return 0;
5768
5769 /* Verify that the two confis have the same number of windows. */
5770 if (sw1->size != sw2->size)
5771 return 0;
5772
5773 for (i = 0; i < sw1->size; i++)
5774 {
5775 struct saved_window *p1, *p2;
5776 int w1_is_current, w2_is_current;
5777
5778 p1 = SAVED_WINDOW_N (sw1, i);
5779 p2 = SAVED_WINDOW_N (sw2, i);
5780
5781 /* Verify that the current windows in the two
5782 configurations correspond to each other. */
5783 w1_is_current = EQ (d1->current_window, p1->window);
5784 w2_is_current = EQ (d2->current_window, p2->window);
5785
5786 if (w1_is_current != w2_is_current)
5787 return 0;
5788
5789 /* Verify that the corresponding windows do match. */
5790 if (! EQ (p1->buffer, p2->buffer))
5791 return 0;
5792 if (! EQ (p1->left, p2->left))
5793 return 0;
5794 if (! EQ (p1->top, p2->top))
5795 return 0;
5796 if (! EQ (p1->width, p2->width))
5797 return 0;
5798 if (! EQ (p1->height, p2->height))
5799 return 0;
cbff28e8
RS
5800 if (! EQ (p1->display_table, p2->display_table))
5801 return 0;
5802 if (! EQ (p1->parent, p2->parent))
5803 return 0;
5804 if (! EQ (p1->prev, p2->prev))
5805 return 0;
2f8274be
RS
5806 if (! ignore_positions)
5807 {
5808 if (! EQ (p1->hscroll, p2->hscroll))
5809 return 0;
ea68264b
GM
5810 if (!EQ (p1->min_hscroll, p2->min_hscroll))
5811 return 0;
2f8274be
RS
5812 if (! EQ (p1->start_at_line_beg, p2->start_at_line_beg))
5813 return 0;
5814 if (NILP (Fequal (p1->start, p2->start)))
5815 return 0;
5816 if (NILP (Fequal (p1->pointm, p2->pointm)))
5817 return 0;
5818 if (NILP (Fequal (p1->mark, p2->mark)))
5819 return 0;
5820 }
cbff28e8
RS
5821 }
5822
5823 return 1;
5824}
2f8274be
RS
5825
5826DEFUN ("compare-window-configurations", Fcompare_window_configurations,
5827 Scompare_window_configurations, 2, 2, 0,
fdb82f93
PJ
5828 doc: /* Compare two window configurations as regards the structure of windows.
5829This function ignores details such as the values of point and mark
5830and scrolling positions. */)
5831 (x, y)
2f8274be
RS
5832 Lisp_Object x, y;
5833{
5834 if (compare_window_configurations (x, y, 1))
5835 return Qt;
5836 return Qnil;
5837}
cbff28e8 5838\f
dfcf069d 5839void
7ab12479
JB
5840init_window_once ()
5841{
1ae1a37d
GM
5842 struct frame *f = make_terminal_frame ();
5843 XSETFRAME (selected_frame, f);
5844 Vterminal_frame = selected_frame;
5845 minibuf_window = f->minibuffer_window;
5846 selected_window = f->selected_window;
5847 last_nonminibuf_frame = f;
5b03d3c0
RS
5848
5849 window_initialized = 1;
7ab12479
JB
5850}
5851
67492200
GM
5852void
5853init_window ()
5854{
5855 Vwindow_list = Qnil;
5856}
5857
dfcf069d 5858void
7ab12479
JB
5859syms_of_window ()
5860{
8a37516b
GM
5861 Qwindow_size_fixed = intern ("window-size-fixed");
5862 staticpro (&Qwindow_size_fixed);
233a4a2c 5863
543f5fb1
RS
5864 staticpro (&Qwindow_configuration_change_hook);
5865 Qwindow_configuration_change_hook
5866 = intern ("window-configuration-change-hook");
5867
7ab12479
JB
5868 Qwindowp = intern ("windowp");
5869 staticpro (&Qwindowp);
5870
3f8ab7bd
RS
5871 Qwindow_configuration_p = intern ("window-configuration-p");
5872 staticpro (&Qwindow_configuration_p);
5873
806b4d9b
JB
5874 Qwindow_live_p = intern ("window-live-p");
5875 staticpro (&Qwindow_live_p);
605be8af 5876
2cccc823 5877 Qtemp_buffer_show_hook = intern ("temp-buffer-show-hook");
a58ec57d
RS
5878 staticpro (&Qtemp_buffer_show_hook);
5879
67492200 5880 staticpro (&Vwindow_list);
8bfb170b
KS
5881
5882 minibuf_selected_window = Qnil;
3dbab091 5883 staticpro (&minibuf_selected_window);
67492200 5884
7ab12479 5885 DEFVAR_LISP ("temp-buffer-show-function", &Vtemp_buffer_show_function,
fdb82f93
PJ
5886 doc: /* Non-nil means call as function to display a help buffer.
5887The function is called with one argument, the buffer to be displayed.
5888Used by `with-output-to-temp-buffer'.
5889If this function is used, then it must do the entire job of showing
5890the buffer; `temp-buffer-show-hook' is not run unless this function runs it. */);
7ab12479
JB
5891 Vtemp_buffer_show_function = Qnil;
5892
5893 DEFVAR_LISP ("display-buffer-function", &Vdisplay_buffer_function,
fdb82f93
PJ
5894 doc: /* If non-nil, function to call to handle `display-buffer'.
5895It will receive two args, the buffer and a flag which if non-nil means
5896 that the currently selected window is not acceptable.
5897Commands such as `switch-to-buffer-other-window' and `find-file-other-window'
5898work using this function. */);
7ab12479
JB
5899 Vdisplay_buffer_function = Qnil;
5900
6529ed87 5901 DEFVAR_LISP ("even-window-heights", &Veven_window_heights,
fdb82f93
PJ
5902 doc: /* *If non-nil, `display-buffer' should even the window heights.
5903If nil, `display-buffer' will leave the window configuration alone. */);
6529ed87
GM
5904 Veven_window_heights = Qt;
5905
7ab12479 5906 DEFVAR_LISP ("minibuffer-scroll-window", &Vminibuf_scroll_window,
fdb82f93 5907 doc: /* Non-nil means it is the window that C-M-v in minibuffer should scroll. */);
7ab12479
JB
5908 Vminibuf_scroll_window = Qnil;
5909
cc91894c 5910 DEFVAR_BOOL ("mode-line-in-non-selected-windows", &mode_line_in_non_selected_windows,
26124d5e 5911 doc: /* Non-nil means to use `mode-line-inactive' face in non-selected windows.
cc91894c
KS
5912If the minibuffer is active, the `minibuffer-scroll-window' mode line
5913is displayed in the `mode-line' face. */);
5914 mode_line_in_non_selected_windows = 1;
26124d5e 5915
7ab12479 5916 DEFVAR_LISP ("other-window-scroll-buffer", &Vother_window_scroll_buffer,
fdb82f93 5917 doc: /* If non-nil, this is a buffer and \\[scroll-other-window] should scroll its window. */);
7ab12479
JB
5918 Vother_window_scroll_buffer = Qnil;
5919
44fa5b1e 5920 DEFVAR_BOOL ("pop-up-frames", &pop_up_frames,
fdb82f93 5921 doc: /* *Non-nil means `display-buffer' should make a separate frame. */);
44fa5b1e 5922 pop_up_frames = 0;
7ab12479 5923
9c3da604 5924 DEFVAR_BOOL ("display-buffer-reuse-frames", &display_buffer_reuse_frames,
fdb82f93
PJ
5925 doc: /* *Non-nil means `display-buffer' should reuse frames.
5926If the buffer in question is already displayed in a frame, raise that frame. */);
9c3da604
GM
5927 display_buffer_reuse_frames = 0;
5928
44fa5b1e 5929 DEFVAR_LISP ("pop-up-frame-function", &Vpop_up_frame_function,
fdb82f93
PJ
5930 doc: /* Function to call to handle automatic new frame creation.
5931It is called with no arguments and should return a newly created frame.
5932
5933A typical value might be `(lambda () (new-frame pop-up-frame-alist))'
5934where `pop-up-frame-alist' would hold the default frame parameters. */);
44fa5b1e 5935 Vpop_up_frame_function = Qnil;
7ab12479 5936
a90712c2 5937 DEFVAR_LISP ("special-display-buffer-names", &Vspecial_display_buffer_names,
fdb82f93
PJ
5938 doc: /* *List of buffer names that should have their own special frames.
5939Displaying a buffer whose name is in this list makes a special frame for it
5940using `special-display-function'. See also `special-display-regexps'.
5941
5942An element of the list can be a list instead of just a string.
5943There are two ways to use a list as an element:
5944 (BUFFER FRAME-PARAMETERS...) (BUFFER FUNCTION OTHER-ARGS...)
5945In the first case, FRAME-PARAMETERS are used to create the frame.
5946In the latter case, FUNCTION is called with BUFFER as the first argument,
5947followed by OTHER-ARGS--it can display BUFFER in any way it likes.
5948All this is done by the function found in `special-display-function'.
5949
5950If this variable appears \"not to work\", because you add a name to it
5951but that buffer still appears in the selected window, look at the
5952values of `same-window-buffer-names' and `same-window-regexps'.
5953Those variables take precedence over this one. */);
a90712c2
RS
5954 Vspecial_display_buffer_names = Qnil;
5955
5956 DEFVAR_LISP ("special-display-regexps", &Vspecial_display_regexps,
fdb82f93
PJ
5957 doc: /* *List of regexps saying which buffers should have their own special frames.
5958If a buffer name matches one of these regexps, it gets its own frame.
5959Displaying a buffer whose name is in this list makes a special frame for it
5960using `special-display-function'.
5961
5962An element of the list can be a list instead of just a string.
5963There are two ways to use a list as an element:
5964 (REGEXP FRAME-PARAMETERS...) (REGEXP FUNCTION OTHER-ARGS...)
5965In the first case, FRAME-PARAMETERS are used to create the frame.
5966In the latter case, FUNCTION is called with the buffer as first argument,
5967followed by OTHER-ARGS--it can display the buffer in any way it likes.
5968All this is done by the function found in `special-display-function'.
5969
5970If this variable appears \"not to work\", because you add a regexp to it
5971but the matching buffers still appear in the selected window, look at the
5972values of `same-window-buffer-names' and `same-window-regexps'.
5973Those variables take precedence over this one. */);
a90712c2
RS
5974 Vspecial_display_regexps = Qnil;
5975
5976 DEFVAR_LISP ("special-display-function", &Vspecial_display_function,
fdb82f93
PJ
5977 doc: /* Function to call to make a new frame for a special buffer.
5978It is called with two arguments, the buffer and optional buffer specific
5979data, and should return a window displaying that buffer.
a5731348
SM
5980The default value normally makes a separate frame for the buffer,
5981 using `special-display-frame-alist' to specify the frame parameters.
5982But if the buffer specific data includes (same-buffer . t) then the
5983 buffer is displayed in the current selected window.
5984Otherwise if it includes (same-frame . t) then the buffer is displayed in
5985 a new window in the currently selected frame.
5986
5987A buffer is special if it is listed in `special-display-buffer-names'
fdb82f93 5988or matches a regexp in `special-display-regexps'. */);
a90712c2
RS
5989 Vspecial_display_function = Qnil;
5990
855d8627 5991 DEFVAR_LISP ("same-window-buffer-names", &Vsame_window_buffer_names,
fdb82f93
PJ
5992 doc: /* *List of buffer names that should appear in the selected window.
5993Displaying one of these buffers using `display-buffer' or `pop-to-buffer'
5994switches to it in the selected window, rather than making it appear
5995in some other window.
5996
5997An element of the list can be a cons cell instead of just a string.
5998Then the car must be a string, which specifies the buffer name.
5999This is for compatibility with `special-display-buffer-names';
6000the cdr of the cons cell is ignored.
6001
6002See also `same-window-regexps'. */);
855d8627
RS
6003 Vsame_window_buffer_names = Qnil;
6004
6005 DEFVAR_LISP ("same-window-regexps", &Vsame_window_regexps,
fdb82f93
PJ
6006 doc: /* *List of regexps saying which buffers should appear in the selected window.
6007If a buffer name matches one of these regexps, then displaying it
6008using `display-buffer' or `pop-to-buffer' switches to it
6009in the selected window, rather than making it appear in some other window.
6010
6011An element of the list can be a cons cell instead of just a string.
6012Then the car must be a string, which specifies the buffer name.
6013This is for compatibility with `special-display-buffer-names';
6014the cdr of the cons cell is ignored.
6015
6016See also `same-window-buffer-names'. */);
855d8627
RS
6017 Vsame_window_regexps = Qnil;
6018
7ab12479 6019 DEFVAR_BOOL ("pop-up-windows", &pop_up_windows,
fdb82f93 6020 doc: /* *Non-nil means display-buffer should make new windows. */);
7ab12479
JB
6021 pop_up_windows = 1;
6022
6023 DEFVAR_INT ("next-screen-context-lines", &next_screen_context_lines,
fdb82f93 6024 doc: /* *Number of lines of continuity when scrolling by screenfuls. */);
7ab12479
JB
6025 next_screen_context_lines = 2;
6026
6027 DEFVAR_INT ("split-height-threshold", &split_height_threshold,
fdb82f93
PJ
6028 doc: /* *display-buffer would prefer to split the largest window if this large.
6029If there is only one window, it is split regardless of this value. */);
7ab12479
JB
6030 split_height_threshold = 500;
6031
6032 DEFVAR_INT ("window-min-height", &window_min_height,
fdb82f93 6033 doc: /* *Delete any window less than this tall (including its mode line). */);
7ab12479
JB
6034 window_min_height = 4;
6035
6036 DEFVAR_INT ("window-min-width", &window_min_width,
fdb82f93 6037 doc: /* *Delete any window less than this wide. */);
7ab12479
JB
6038 window_min_width = 10;
6039
5500c422
GM
6040 DEFVAR_LISP ("scroll-preserve-screen-position",
6041 &Vscroll_preserve_screen_position,
fdb82f93 6042 doc: /* *Non-nil means scroll commands move point to keep its screen line unchanged. */);
5500c422 6043 Vscroll_preserve_screen_position = Qnil;
9317a85d 6044
543f5fb1
RS
6045 DEFVAR_LISP ("window-configuration-change-hook",
6046 &Vwindow_configuration_change_hook,
fdb82f93
PJ
6047 doc: /* Functions to call when window configuration changes.
6048The selected frame is the one whose configuration has changed. */);
543f5fb1
RS
6049 Vwindow_configuration_change_hook = Qnil;
6050
b0b7ed0e 6051 DEFVAR_BOOL ("window-size-fixed", &window_size_fixed,
fdb82f93
PJ
6052 doc: /* Non-nil in a buffer means windows displaying the buffer are fixed-size.
6053Emacs won't change the size of any window displaying that buffer,
6054unless you explicitly change the size, or Emacs has no other choice.
6055This variable automatically becomes buffer-local when set. */);
b0b7ed0e
GM
6056 Fmake_variable_buffer_local (Qwindow_size_fixed);
6057 window_size_fixed = 0;
6058
7ab12479
JB
6059 defsubr (&Sselected_window);
6060 defsubr (&Sminibuffer_window);
6061 defsubr (&Swindow_minibuffer_p);
6062 defsubr (&Swindowp);
806b4d9b 6063 defsubr (&Swindow_live_p);
7ab12479
JB
6064 defsubr (&Spos_visible_in_window_p);
6065 defsubr (&Swindow_buffer);
6066 defsubr (&Swindow_height);
6067 defsubr (&Swindow_width);
6068 defsubr (&Swindow_hscroll);
6069 defsubr (&Sset_window_hscroll);
190eb263
RS
6070 defsubr (&Swindow_redisplay_end_trigger);
6071 defsubr (&Sset_window_redisplay_end_trigger);
7ab12479 6072 defsubr (&Swindow_edges);
d5783c40
JB
6073 defsubr (&Scoordinates_in_window_p);
6074 defsubr (&Swindow_at);
7ab12479
JB
6075 defsubr (&Swindow_point);
6076 defsubr (&Swindow_start);
6077 defsubr (&Swindow_end);
6078 defsubr (&Sset_window_point);
6079 defsubr (&Sset_window_start);
6080 defsubr (&Swindow_dedicated_p);
d207b766 6081 defsubr (&Sset_window_dedicated_p);
7ab12479
JB
6082 defsubr (&Swindow_display_table);
6083 defsubr (&Sset_window_display_table);
6084 defsubr (&Snext_window);
6085 defsubr (&Sprevious_window);
6086 defsubr (&Sother_window);
6087 defsubr (&Sget_lru_window);
6088 defsubr (&Sget_largest_window);
6089 defsubr (&Sget_buffer_window);
6090 defsubr (&Sdelete_other_windows);
6091 defsubr (&Sdelete_windows_on);
6092 defsubr (&Sreplace_buffer_in_windows);
6093 defsubr (&Sdelete_window);
6094 defsubr (&Sset_window_buffer);
6095 defsubr (&Sselect_window);
4628f7a4
EN
6096 defsubr (&Sspecial_display_p);
6097 defsubr (&Ssame_window_p);
7ab12479
JB
6098 defsubr (&Sdisplay_buffer);
6099 defsubr (&Ssplit_window);
6100 defsubr (&Senlarge_window);
6101 defsubr (&Sshrink_window);
6102 defsubr (&Sscroll_up);
6103 defsubr (&Sscroll_down);
6104 defsubr (&Sscroll_left);
6105 defsubr (&Sscroll_right);
ccd0664b 6106 defsubr (&Sother_window_for_scrolling);
7ab12479 6107 defsubr (&Sscroll_other_window);
fa832261 6108 defsubr (&Sminibuffer_selected_window);
7ab12479 6109 defsubr (&Srecenter);
81fe0836 6110 defsubr (&Swindow_text_height);
7ab12479
JB
6111 defsubr (&Smove_to_window_line);
6112 defsubr (&Swindow_configuration_p);
3f8ab7bd 6113 defsubr (&Swindow_configuration_frame);
7ab12479
JB
6114 defsubr (&Sset_window_configuration);
6115 defsubr (&Scurrent_window_configuration);
6116 defsubr (&Ssave_window_excursion);
5500c422
GM
6117 defsubr (&Sset_window_margins);
6118 defsubr (&Swindow_margins);
6119 defsubr (&Swindow_vscroll);
6120 defsubr (&Sset_window_vscroll);
2f8274be 6121 defsubr (&Scompare_window_configurations);
67492200 6122 defsubr (&Swindow_list);
7ab12479
JB
6123}
6124
dfcf069d 6125void
7ab12479
JB
6126keys_of_window ()
6127{
6128 initial_define_key (control_x_map, '1', "delete-other-windows");
6129 initial_define_key (control_x_map, '2', "split-window");
6130 initial_define_key (control_x_map, '0', "delete-window");
6131 initial_define_key (control_x_map, 'o', "other-window");
6132 initial_define_key (control_x_map, '^', "enlarge-window");
6133 initial_define_key (control_x_map, '<', "scroll-left");
6134 initial_define_key (control_x_map, '>', "scroll-right");
6135
6136 initial_define_key (global_map, Ctl ('V'), "scroll-up");
6137 initial_define_key (meta_map, Ctl ('V'), "scroll-other-window");
6138 initial_define_key (meta_map, 'v', "scroll-down");
6139
6140 initial_define_key (global_map, Ctl('L'), "recenter");
6141 initial_define_key (meta_map, 'r', "move-to-window-line");
6142}