(eudc-install-menu): Use `define-key' 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. */
118ea242 1864 if (MINI_WINDOW_P (w) || !NILP (w->dedicated))
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. */
c930dfab 1917 if (MINI_WINDOW_P (w) || !NILP (w->dedicated))
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 }
38ab08d1 1957 else
38ab08d1
RS
1958 {
1959 /* Otherwise show a different buffer in the window. */
118ea242
GM
1960 w->dedicated = Qnil;
1961 Fset_window_buffer (window, buffer);
1962 if (EQ (window, selected_window))
1963 Fset_buffer (w->buffer);
38ab08d1 1964 }
7ab12479
JB
1965 }
1966 break;
3f8ab7bd
RS
1967
1968 /* Check for a window that has a killed buffer. */
1969 case CHECK_ALL_WINDOWS:
118ea242
GM
1970 if (! NILP (w->buffer)
1971 && NILP (XBUFFER (w->buffer)->name))
3f8ab7bd 1972 abort ();
118ea242 1973 break;
6bbd7a29
GM
1974
1975 case WINDOW_LOOP_UNUSED:
1976 break;
7ab12479 1977 }
7ab12479 1978 }
7ab12479 1979
118ea242 1980 UNGCPRO;
7ab12479 1981 return best_window;
37962e60 1982}
605be8af 1983
3f8ab7bd
RS
1984/* Used for debugging. Abort if any window has a dead buffer. */
1985
233a4a2c 1986void
3f8ab7bd
RS
1987check_all_windows ()
1988{
1989 window_loop (CHECK_ALL_WINDOWS, Qnil, 1, Qt);
1990}
1991
7ab12479 1992DEFUN ("get-lru-window", Fget_lru_window, Sget_lru_window, 0, 1, 0,
fdb82f93
PJ
1993 doc: /* Return the window least recently selected or used for display.
1994If optional argument FRAME is `visible', search all visible frames.
1995If FRAME is 0, search all visible and iconified frames.
1996If FRAME is t, search all frames.
1997If FRAME is nil, search only the selected frame.
1998If FRAME is a frame, search only that frame. */)
1999 (frame)
2000 Lisp_Object frame;
7ab12479
JB
2001{
2002 register Lisp_Object w;
2003 /* First try for a window that is full-width */
89bca612 2004 w = window_loop (GET_LRU_WINDOW, Qt, 0, frame);
265a9e55 2005 if (!NILP (w) && !EQ (w, selected_window))
7ab12479
JB
2006 return w;
2007 /* If none of them, try the rest */
89bca612 2008 return window_loop (GET_LRU_WINDOW, Qnil, 0, frame);
7ab12479
JB
2009}
2010
2011DEFUN ("get-largest-window", Fget_largest_window, Sget_largest_window, 0, 1, 0,
fdb82f93
PJ
2012 doc: /* Return the largest window in area.
2013If optional argument FRAME is `visible', search all visible frames.
2014If FRAME is 0, search all visible and iconified frames.
2015If FRAME is t, search all frames.
2016If FRAME is nil, search only the selected frame.
2017If FRAME is a frame, search only that frame. */)
2018 (frame)
2019 Lisp_Object frame;
7ab12479
JB
2020{
2021 return window_loop (GET_LARGEST_WINDOW, Qnil, 0,
44fa5b1e 2022 frame);
7ab12479
JB
2023}
2024
2025DEFUN ("get-buffer-window", Fget_buffer_window, Sget_buffer_window, 1, 2, 0,
fdb82f93
PJ
2026 doc: /* Return a window currently displaying BUFFER, or nil if none.
2027If optional argument FRAME is `visible', search all visible frames.
2028If optional argument FRAME is 0, search all visible and iconified frames.
2029If FRAME is t, search all frames.
2030If FRAME is nil, search only the selected frame.
2031If FRAME is a frame, search only that frame. */)
2032 (buffer, frame)
2033 Lisp_Object buffer, frame;
7ab12479
JB
2034{
2035 buffer = Fget_buffer (buffer);
017b2bad 2036 if (BUFFERP (buffer))
44fa5b1e 2037 return window_loop (GET_BUFFER_WINDOW, buffer, 1, frame);
7ab12479
JB
2038 else
2039 return Qnil;
2040}
2041
2042DEFUN ("delete-other-windows", Fdelete_other_windows, Sdelete_other_windows,
fdb82f93
PJ
2043 0, 1, "",
2044 doc: /* Make WINDOW (or the selected window) fill its frame.
2045Only the frame WINDOW is on is affected.
2046This function tries to reduce display jumps
2047by keeping the text previously visible in WINDOW
2048in the same place on the frame. Doing this depends on
2049the value of (window-start WINDOW), so if calling this function
2050in a program gives strange scrolling, make sure the window-start
2051value is reasonable when this function is called. */)
2052 (window)
7ab12479
JB
2053 Lisp_Object window;
2054{
2055 struct window *w;
00d3d838 2056 int startpos;
85fe3b5e 2057 int top, new_top;
7ab12479 2058
265a9e55 2059 if (NILP (window))
7ab12479
JB
2060 window = selected_window;
2061 else
b7826503 2062 CHECK_LIVE_WINDOW (window);
7ab12479 2063 w = XWINDOW (window);
a2b38b3c 2064
00d3d838 2065 startpos = marker_position (w->start);
5500c422 2066 top = XFASTINT (w->top) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w)));
7ab12479 2067
a2b38b3c
RS
2068 if (MINI_WINDOW_P (w) && top > 0)
2069 error ("Can't expand minibuffer to full frame");
2070
70728a80 2071 window_loop (DELETE_OTHER_WINDOWS, window, 0, WINDOW_FRAME (w));
7ab12479 2072
00d3d838
KH
2073 /* Try to minimize scrolling, by setting the window start to the point
2074 will cause the text at the old window start to be at the same place
2075 on the frame. But don't try to do this if the window start is
2076 outside the visible portion (as might happen when the display is
2077 not current, due to typeahead). */
85fe3b5e
GM
2078 new_top = XFASTINT (w->top) - FRAME_TOP_MARGIN (XFRAME (WINDOW_FRAME (w)));
2079 if (new_top != top
2080 && startpos >= BUF_BEGV (XBUFFER (w->buffer))
00d3d838
KH
2081 && startpos <= BUF_ZV (XBUFFER (w->buffer)))
2082 {
2083 struct position pos;
2084 struct buffer *obuf = current_buffer;
2085
2086 Fset_buffer (w->buffer);
2087 /* This computation used to temporarily move point, but that can
2088 have unwanted side effects due to text properties. */
0383eb57 2089 pos = *vmotion (startpos, -top, w);
4d047f50 2090
b73ea88e 2091 set_marker_both (w->start, w->buffer, pos.bufpos, pos.bytepos);
85fe3b5e 2092 w->window_end_valid = Qnil;
b73ea88e
RS
2093 w->start_at_line_beg = ((pos.bytepos == BEGV_BYTE
2094 || FETCH_BYTE (pos.bytepos - 1) == '\n') ? Qt
00d3d838 2095 : Qnil);
80622eec
RS
2096 /* We need to do this, so that the window-scroll-functions
2097 get called. */
4d047f50 2098 w->optional_new_start = Qt;
00d3d838
KH
2099
2100 set_buffer_internal (obuf);
2101 }
5500c422 2102
7ab12479
JB
2103 return Qnil;
2104}
2105
2106DEFUN ("delete-windows-on", Fdelete_windows_on, Sdelete_windows_on,
fdb82f93
PJ
2107 1, 2, "bDelete windows on (buffer): ",
2108 doc: /* Delete all windows showing BUFFER.
2109Optional second argument FRAME controls which frames are affected.
2110If optional argument FRAME is `visible', search all visible frames.
2111If FRAME is 0, search all visible and iconified frames.
2112If FRAME is nil, search all frames.
2113If FRAME is t, search only the selected frame.
2114If FRAME is a frame, search only that frame. */)
2115 (buffer, frame)
26f6279d 2116 Lisp_Object buffer, frame;
7ab12479 2117{
26f6279d
JB
2118 /* FRAME uses t and nil to mean the opposite of what window_loop
2119 expects. */
c520265e
RS
2120 if (NILP (frame))
2121 frame = Qt;
2122 else if (EQ (frame, Qt))
2123 frame = Qnil;
26f6279d 2124
265a9e55 2125 if (!NILP (buffer))
7ab12479
JB
2126 {
2127 buffer = Fget_buffer (buffer);
b7826503 2128 CHECK_BUFFER (buffer);
26f6279d 2129 window_loop (DELETE_BUFFER_WINDOWS, buffer, 0, frame);
7ab12479 2130 }
5500c422 2131
7ab12479
JB
2132 return Qnil;
2133}
2134
2135DEFUN ("replace-buffer-in-windows", Freplace_buffer_in_windows,
fdb82f93
PJ
2136 Sreplace_buffer_in_windows,
2137 1, 1, "bReplace buffer in windows: ",
2138 doc: /* Replace BUFFER with some other buffer in all windows showing it. */)
2139 (buffer)
7ab12479
JB
2140 Lisp_Object buffer;
2141{
265a9e55 2142 if (!NILP (buffer))
7ab12479
JB
2143 {
2144 buffer = Fget_buffer (buffer);
b7826503 2145 CHECK_BUFFER (buffer);
7ab12479
JB
2146 window_loop (UNSHOW_BUFFER, buffer, 0, Qt);
2147 }
2148 return Qnil;
2149}
ff58478b
RS
2150
2151/* Replace BUFFER with some other buffer in all windows
2152 of all frames, even those on other keyboards. */
2153
2154void
2155replace_buffer_in_all_windows (buffer)
2156 Lisp_Object buffer;
2157{
27abb84f 2158#ifdef MULTI_KBOARD
ff58478b
RS
2159 Lisp_Object tail, frame;
2160
ff58478b
RS
2161 /* A single call to window_loop won't do the job
2162 because it only considers frames on the current keyboard.
2163 So loop manually over frames, and handle each one. */
2164 FOR_EACH_FRAME (tail, frame)
db7f721d 2165 window_loop (UNSHOW_BUFFER, buffer, 1, frame);
ff58478b 2166#else
db7f721d 2167 window_loop (UNSHOW_BUFFER, buffer, 1, Qt);
ff58478b
RS
2168#endif
2169}
7ab12479
JB
2170\f
2171/* Set the height of WINDOW and all its inferiors. */
a481b3ea
JB
2172
2173/* The smallest acceptable dimensions for a window. Anything smaller
2174 might crash Emacs. */
5500c422 2175
a481b3ea 2176#define MIN_SAFE_WINDOW_WIDTH (2)
dc1ab1ee 2177#define MIN_SAFE_WINDOW_HEIGHT (1)
a481b3ea
JB
2178
2179/* Make sure that window_min_height and window_min_width are
2180 not too small; if they are, set them to safe minima. */
2181
2182static void
2183check_min_window_sizes ()
2184{
2185 /* Smaller values might permit a crash. */
2186 if (window_min_width < MIN_SAFE_WINDOW_WIDTH)
2187 window_min_width = MIN_SAFE_WINDOW_WIDTH;
2188 if (window_min_height < MIN_SAFE_WINDOW_HEIGHT)
2189 window_min_height = MIN_SAFE_WINDOW_HEIGHT;
2190}
2191
2192/* If *ROWS or *COLS are too small a size for FRAME, set them to the
2193 minimum allowable size. */
5500c422 2194
605be8af 2195void
a481b3ea 2196check_frame_size (frame, rows, cols)
605be8af
JB
2197 FRAME_PTR frame;
2198 int *rows, *cols;
a481b3ea 2199{
628df3bf 2200 /* For height, we have to see:
54b8bcb5
RS
2201 how many windows the frame has at minimum (one or two),
2202 and whether it has a menu bar or other special stuff at the top. */
2203 int min_height
2204 = ((FRAME_MINIBUF_ONLY_P (frame) || ! FRAME_HAS_MINIBUF_P (frame))
2205 ? MIN_SAFE_WINDOW_HEIGHT
2206 : 2 * MIN_SAFE_WINDOW_HEIGHT);
5500c422
GM
2207
2208 if (FRAME_TOP_MARGIN (frame) > 0)
2209 min_height += FRAME_TOP_MARGIN (frame);
a481b3ea
JB
2210
2211 if (*rows < min_height)
2212 *rows = min_height;
2213 if (*cols < MIN_SAFE_WINDOW_WIDTH)
2214 *cols = MIN_SAFE_WINDOW_WIDTH;
2215}
2216
c1636aa6 2217
233a4a2c
GM
2218/* Value is non-zero if window W is fixed-size. WIDTH_P non-zero means
2219 check if W's width can be changed, otherwise check W's height.
2220 CHECK_SIBLINGS_P non-zero means check resizablity of WINDOW's
2221 siblings, too. If none of the siblings is resizable, WINDOW isn't
2222 either. */
c1636aa6 2223
233a4a2c
GM
2224static int
2225window_fixed_size_p (w, width_p, check_siblings_p)
2226 struct window *w;
2227 int width_p, check_siblings_p;
2228{
2229 int fixed_p;
2230 struct window *c;
2231
2232 if (!NILP (w->hchild))
2233 {
2234 c = XWINDOW (w->hchild);
2235
2236 if (width_p)
2237 {
2238 /* A horiz. combination is fixed-width if all of if its
2239 children are. */
2240 while (c && window_fixed_size_p (c, width_p, 0))
2241 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2242 fixed_p = c == NULL;
2243 }
2244 else
2245 {
2246 /* A horiz. combination is fixed-height if one of if its
2247 children is. */
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 }
2253 else if (!NILP (w->vchild))
2254 {
2255 c = XWINDOW (w->vchild);
2256
2257 if (width_p)
2258 {
2259 /* A vert. combination is fixed-width if one of if its
2260 children is. */
2261 while (c && !window_fixed_size_p (c, width_p, 0))
2262 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2263 fixed_p = c != NULL;
2264 }
2265 else
2266 {
2267 /* A vert. combination is fixed-height if all of if its
2268 children are. */
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 }
2274 else if (BUFFERP (w->buffer))
2275 {
a34dfd12
GM
2276 if (w->height_fixed_p && !width_p)
2277 fixed_p = 1;
2278 else
233a4a2c 2279 {
a34dfd12
GM
2280 struct buffer *old = current_buffer;
2281 Lisp_Object val;
2282
2283 current_buffer = XBUFFER (w->buffer);
2284 val = find_symbol_value (Qwindow_size_fixed);
2285 current_buffer = old;
2286
2287 fixed_p = 0;
2288 if (!EQ (val, Qunbound))
2289 {
2290 fixed_p = !NILP (val);
2291
2292 if (fixed_p
2293 && ((EQ (val, Qheight) && width_p)
2294 || (EQ (val, Qwidth) && !width_p)))
2295 fixed_p = 0;
2296 }
233a4a2c
GM
2297 }
2298
2299 /* Can't tell if this one is resizable without looking at
2300 siblings. If all siblings are fixed-size this one is too. */
2301 if (!fixed_p && check_siblings_p && WINDOWP (w->parent))
2302 {
2303 Lisp_Object child;
2304
2305 for (child = w->prev; !NILP (child); child = XWINDOW (child)->prev)
2306 if (!window_fixed_size_p (XWINDOW (child), width_p, 0))
2307 break;
2308
2309 if (NILP (child))
2310 for (child = w->next; !NILP (child); child = XWINDOW (child)->next)
2311 if (!window_fixed_size_p (XWINDOW (child), width_p, 0))
2312 break;
2313
2314 if (NILP (child))
2315 fixed_p = 1;
2316 }
2317 }
2318 else
2319 fixed_p = 1;
2320
2321 return fixed_p;
2322}
2323
2324
2325/* Return the minimum size of window W, not taking fixed-width windows
2326 into account. WIDTH_P non-zero means return the minimum width,
2327 otherwise return the minimum height. If W is a combination window,
2328 compute the minimum size from the minimum sizes of W's children. */
2329
2330static int
2331window_min_size_1 (w, width_p)
c1636aa6
GM
2332 struct window *w;
2333 int width_p;
2334{
233a4a2c 2335 struct window *c;
c1636aa6
GM
2336 int size;
2337
233a4a2c
GM
2338 if (!NILP (w->hchild))
2339 {
2340 c = XWINDOW (w->hchild);
2341 size = 0;
2342
2343 if (width_p)
2344 {
2345 /* The min width of a horizontal combination is
2346 the sum of the min widths of its children. */
2347 while (c)
2348 {
2349 size += window_min_size_1 (c, width_p);
2350 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2351 }
2352 }
2353 else
2354 {
2355 /* The min height a horizontal combination equals
2356 the maximum of all min height of its children. */
2357 while (c)
2358 {
2359 int min_size = window_min_size_1 (c, width_p);
2360 size = max (min_size, size);
2361 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2362 }
2363 }
2364 }
2365 else if (!NILP (w->vchild))
2366 {
2367 c = XWINDOW (w->vchild);
2368 size = 0;
2369
2370 if (width_p)
2371 {
2372 /* The min width of a vertical combination is
2373 the maximum of the min widths of its children. */
2374 while (c)
2375 {
2376 int min_size = window_min_size_1 (c, width_p);
2377 size = max (min_size, size);
2378 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2379 }
2380 }
2381 else
2382 {
2383 /* The min height of a vertical combination equals
2384 the sum of the min height of its children. */
2385 while (c)
2386 {
2387 size += window_min_size_1 (c, width_p);
2388 c = WINDOWP (c->next) ? XWINDOW (c->next) : NULL;
2389 }
2390 }
2391 }
c1636aa6
GM
2392 else
2393 {
233a4a2c
GM
2394 if (width_p)
2395 size = window_min_width;
c1636aa6 2396 else
233a4a2c
GM
2397 {
2398 if (MINI_WINDOW_P (w)
2399 || (!WINDOW_WANTS_MODELINE_P (w)
045dee35 2400 && !WINDOW_WANTS_HEADER_LINE_P (w)))
233a4a2c
GM
2401 size = 1;
2402 else
2403 size = window_min_height;
2404 }
c1636aa6
GM
2405 }
2406
2407 return size;
2408}
2409
2410
233a4a2c
GM
2411/* Return the minimum size of window W, taking fixed-size windows into
2412 account. WIDTH_P non-zero means return the minimum width,
f984d4fc
GM
2413 otherwise return the minimum height. IGNORE_FIXED_P non-zero means
2414 ignore if W is fixed-size. Set *FIXED to 1 if W is fixed-size
2415 unless FIXED is null. */
7ab12479 2416
233a4a2c 2417static int
f984d4fc 2418window_min_size (w, width_p, ignore_fixed_p, fixed)
233a4a2c 2419 struct window *w;
f984d4fc 2420 int width_p, ignore_fixed_p, *fixed;
233a4a2c
GM
2421{
2422 int size, fixed_p;
2423
f984d4fc
GM
2424 if (ignore_fixed_p)
2425 fixed_p = 0;
2426 else
2427 fixed_p = window_fixed_size_p (w, width_p, 1);
2428
233a4a2c
GM
2429 if (fixed)
2430 *fixed = fixed_p;
2431
2432 if (fixed_p)
2433 size = width_p ? XFASTINT (w->width) : XFASTINT (w->height);
2434 else
2435 size = window_min_size_1 (w, width_p);
f984d4fc 2436
233a4a2c
GM
2437 return size;
2438}
2439
2440
2441/* Set WINDOW's height or width to SIZE. WIDTH_P non-zero means set
2442 WINDOW's width. Resize WINDOW's children, if any, so that they
2443 keep their proportionate size relative to WINDOW. Propagate
2444 WINDOW's top or left edge position to children. Delete windows
192c3131
RS
2445 that become too small unless NODELETE_P is non-zero.
2446
2447 If NODELETE_P is 2, that means we do delete windows that are
2448 too small, even if they were too small before! */
233a4a2c
GM
2449
2450static void
2451size_window (window, size, width_p, nodelete_p)
7ab12479 2452 Lisp_Object window;
233a4a2c 2453 int size, width_p, nodelete_p;
7ab12479 2454{
233a4a2c
GM
2455 struct window *w = XWINDOW (window);
2456 struct window *c;
2457 Lisp_Object child, *forward, *sideward;
2458 int old_size, min_size;
7ab12479 2459
192c3131
RS
2460 if (nodelete_p == 2)
2461 nodelete_p = 0;
2462
a481b3ea 2463 check_min_window_sizes ();
7ae2f10f 2464 size = max (0, size);
233a4a2c 2465
b5f05b50
GM
2466 /* If the window has been "too small" at one point,
2467 don't delete it for being "too small" in the future.
2468 Preserve it as long as that is at all possible. */
233a4a2c
GM
2469 if (width_p)
2470 {
8b6d9dc9 2471 old_size = XINT (w->width);
233a4a2c
GM
2472 min_size = window_min_width;
2473 }
2474 else
2475 {
8b6d9dc9 2476 old_size = XINT (w->height);
233a4a2c
GM
2477 min_size = window_min_height;
2478 }
192c3131
RS
2479
2480 if (old_size < min_size && nodelete_p != 2)
b5f05b50
GM
2481 w->too_small_ok = Qt;
2482
233a4a2c 2483 /* Maybe delete WINDOW if it's too small. */
192c3131 2484 if (nodelete_p != 1 && !NILP (w->parent))
7ab12479 2485 {
b5f05b50 2486 if (!MINI_WINDOW_P (w) && !NILP (w->too_small_ok))
233a4a2c 2487 min_size = width_p ? MIN_SAFE_WINDOW_WIDTH : MIN_SAFE_WINDOW_HEIGHT;
b5f05b50 2488 else
233a4a2c 2489 min_size = width_p ? window_min_width : window_min_height;
b5f05b50 2490
233a4a2c 2491 if (size < min_size)
c1636aa6
GM
2492 {
2493 delete_window (window);
2494 return;
2495 }
7ab12479
JB
2496 }
2497
233a4a2c 2498 /* Set redisplay hints. */
7ae2f10f
GM
2499 w->last_modified = make_number (0);
2500 w->last_overlay_modified = make_number (0);
7ab12479 2501 windows_or_buffers_changed++;
7ae2f10f 2502 FRAME_WINDOW_SIZES_CHANGED (XFRAME (w->frame)) = 1;
29aeee73 2503
233a4a2c
GM
2504 if (width_p)
2505 {
2506 sideward = &w->vchild;
2507 forward = &w->hchild;
7ae2f10f 2508 w->width = make_number (size);
233a4a2c
GM
2509 }
2510 else
2511 {
2512 sideward = &w->hchild;
2513 forward = &w->vchild;
7ae2f10f 2514 w->height = make_number (size);
0130fe1a 2515 w->orig_height = Qnil;
233a4a2c
GM
2516 }
2517
2518 if (!NILP (*sideward))
7ab12479 2519 {
233a4a2c 2520 for (child = *sideward; !NILP (child); child = c->next)
7ab12479 2521 {
233a4a2c
GM
2522 c = XWINDOW (child);
2523 if (width_p)
2524 c->left = w->left;
2525 else
2526 c->top = w->top;
2527 size_window (child, size, width_p, nodelete_p);
7ab12479
JB
2528 }
2529 }
233a4a2c 2530 else if (!NILP (*forward))
7ab12479 2531 {
233a4a2c
GM
2532 int fixed_size, each, extra, n;
2533 int resize_fixed_p, nfixed;
8b6d9dc9 2534 int last_pos, first_pos, nchildren, total;
233a4a2c
GM
2535
2536 /* Determine the fixed-size portion of the this window, and the
2537 number of child windows. */
8b6d9dc9 2538 fixed_size = nchildren = nfixed = total = 0;
233a4a2c 2539 for (child = *forward; !NILP (child); child = c->next, ++nchildren)
7ab12479 2540 {
8b6d9dc9
GM
2541 int child_size;
2542
7ab12479 2543 c = XWINDOW (child);
8b6d9dc9
GM
2544 child_size = width_p ? XINT (c->width) : XINT (c->height);
2545 total += child_size;
2546
233a4a2c
GM
2547 if (window_fixed_size_p (c, width_p, 0))
2548 {
8b6d9dc9 2549 fixed_size += child_size;
233a4a2c
GM
2550 ++nfixed;
2551 }
2552 }
7ab12479 2553
233a4a2c
GM
2554 /* If the new size is smaller than fixed_size, or if there
2555 aren't any resizable windows, allow resizing fixed-size
2556 windows. */
2557 resize_fixed_p = nfixed == nchildren || size < fixed_size;
2558
2559 /* Compute how many lines/columns to add to each child. The
2560 value of extra takes care of rounding errors. */
2561 n = resize_fixed_p ? nchildren : nchildren - nfixed;
8b6d9dc9
GM
2562 each = (size - total) / n;
2563 extra = (size - total) - n * each;
233a4a2c
GM
2564
2565 /* Compute new children heights and edge positions. */
8b6d9dc9 2566 first_pos = width_p ? XINT (w->left) : XINT (w->top);
233a4a2c
GM
2567 last_pos = first_pos;
2568 for (child = *forward; !NILP (child); child = c->next)
2569 {
2570 int new_size, old_size;
2571
2572 c = XWINDOW (child);
2573 old_size = width_p ? XFASTINT (c->width) : XFASTINT (c->height);
2574 new_size = old_size;
7ab12479 2575
233a4a2c
GM
2576 /* The top or left edge position of this child equals the
2577 bottom or right edge of its predecessor. */
2578 if (width_p)
2579 c->left = make_number (last_pos);
2580 else
2581 c->top = make_number (last_pos);
7ab12479 2582
233a4a2c
GM
2583 /* If this child can be resized, do it. */
2584 if (resize_fixed_p || !window_fixed_size_p (c, width_p, 0))
2585 {
2586 new_size = old_size + each + extra;
2587 extra = 0;
2588 }
2589
2590 /* Set new height. Note that size_window also propagates
2591 edge positions to children, so it's not a no-op if we
2592 didn't change the child's size. */
2593 size_window (child, new_size, width_p, 1);
2594
2595 /* Remember the bottom/right edge position of this child; it
2596 will be used to set the top/left edge of the next child. */
2597 last_pos += new_size;
7ab12479 2598 }
233a4a2c
GM
2599
2600 /* We should have covered the parent exactly with child windows. */
2601 xassert (size == last_pos - first_pos);
2602
7ab12479 2603 /* Now delete any children that became too small. */
233a4a2c
GM
2604 if (!nodelete_p)
2605 for (child = *forward; !NILP (child); child = c->next)
7ab12479 2606 {
233a4a2c
GM
2607 int child_size;
2608 c = XWINDOW (child);
8b6d9dc9 2609 child_size = width_p ? XINT (c->width) : XINT (c->height);
192c3131 2610 size_window (child, child_size, width_p, 2);
7ab12479
JB
2611 }
2612 }
2613}
2614
233a4a2c
GM
2615/* Set WINDOW's height to HEIGHT, and recursively change the height of
2616 WINDOW's children. NODELETE non-zero means don't delete windows
2617 that become too small in the process. (The caller should check
2618 later and do so if appropriate.) */
7ab12479 2619
5e14b1fc 2620void
233a4a2c 2621set_window_height (window, height, nodelete)
7ab12479 2622 Lisp_Object window;
233a4a2c 2623 int height;
7ab12479
JB
2624 int nodelete;
2625{
233a4a2c
GM
2626 size_window (window, height, 0, nodelete);
2627}
7ab12479 2628
7ab12479 2629
233a4a2c
GM
2630/* Set WINDOW's width to WIDTH, and recursively change the width of
2631 WINDOW's children. NODELETE non-zero means don't delete windows
2632 that become too small in the process. (The caller should check
2633 later and do so if appropriate.) */
7ab12479 2634
233a4a2c
GM
2635void
2636set_window_width (window, width, nodelete)
2637 Lisp_Object window;
2638 int width;
2639 int nodelete;
2640{
2641 size_window (window, width, 1, nodelete);
7ab12479 2642}
233a4a2c 2643
7ab12479 2644\f
1d8d96fa 2645int window_select_count;
7ab12479 2646
5b03d3c0
RS
2647Lisp_Object
2648Fset_window_buffer_unwind (obuf)
2649 Lisp_Object obuf;
2650{
2651 Fset_buffer (obuf);
2652 return Qnil;
2653}
2654
7ab12479 2655
5500c422
GM
2656/* Make WINDOW display BUFFER as its contents. RUN_HOOKS_P non-zero
2657 means it's allowed to run hooks. See make_frame for a case where
2658 it's not allowed. */
7ab12479 2659
5500c422
GM
2660void
2661set_window_buffer (window, buffer, run_hooks_p)
2662 Lisp_Object window, buffer;
2663 int run_hooks_p;
2664{
2665 struct window *w = XWINDOW (window);
2666 struct buffer *b = XBUFFER (buffer);
aed13378 2667 int count = SPECPDL_INDEX ();
7ab12479
JB
2668
2669 w->buffer = buffer;
86e48436
RS
2670
2671 if (EQ (window, selected_window))
5500c422 2672 b->last_selected_window = window;
beb4e312
RS
2673
2674 /* Update time stamps of buffer display. */
5500c422
GM
2675 if (INTEGERP (b->display_count))
2676 XSETINT (b->display_count, XINT (b->display_count) + 1);
2677 b->display_time = Fcurrent_time ();
86e48436 2678
d834a2e9 2679 XSETFASTINT (w->window_end_pos, 0);
5500c422
GM
2680 XSETFASTINT (w->window_end_vpos, 0);
2681 bzero (&w->last_cursor, sizeof w->last_cursor);
5a41ab94 2682 w->window_end_valid = Qnil;
1c686c99 2683 w->hscroll = w->min_hscroll = make_number (0);
224227d1 2684 w->vscroll = 0;
5500c422 2685 set_marker_both (w->pointm, buffer, BUF_PT (b), BUF_PT_BYTE (b));
7ab12479 2686 set_marker_restricted (w->start,
5500c422 2687 make_number (b->last_window_start),
7ab12479
JB
2688 buffer);
2689 w->start_at_line_beg = Qnil;
e36ab06b 2690 w->force_start = Qnil;
d834a2e9 2691 XSETFASTINT (w->last_modified, 0);
3cd21523 2692 XSETFASTINT (w->last_overlay_modified, 0);
7ab12479 2693 windows_or_buffers_changed++;
5b03d3c0
RS
2694
2695 /* We must select BUFFER for running the window-scroll-functions.
2696 If WINDOW is selected, switch permanently.
2697 Otherwise, switch but go back to the ambient buffer afterward. */
7ab12479
JB
2698 if (EQ (window, selected_window))
2699 Fset_buffer (buffer);
5b03d3c0
RS
2700 /* We can't check ! NILP (Vwindow_scroll_functions) here
2701 because that might itself be a local variable. */
2702 else if (window_initialized)
2703 {
2704 record_unwind_protect (Fset_window_buffer_unwind, Fcurrent_buffer ());
2705 Fset_buffer (buffer);
2706 }
2707
5500c422 2708 /* Set left and right marginal area width from buffer. */
cfa22082 2709 Fset_window_margins (window, b->left_margin_width, b->right_margin_width);
7ab12479 2710
5500c422
GM
2711 if (run_hooks_p)
2712 {
2713 if (! NILP (Vwindow_scroll_functions))
2714 run_hook_with_args_2 (Qwindow_scroll_functions, window,
2715 Fmarker_position (w->start));
2716
2717 if (! NILP (Vwindow_configuration_change_hook)
2718 && ! NILP (Vrun_hooks))
2719 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
2720 }
543f5fb1 2721
5b03d3c0 2722 unbind_to (count, Qnil);
5500c422 2723}
5b03d3c0 2724
5500c422
GM
2725
2726DEFUN ("set-window-buffer", Fset_window_buffer, Sset_window_buffer, 2, 2, 0,
fdb82f93
PJ
2727 doc: /* Make WINDOW display BUFFER as its contents.
2728BUFFER can be a buffer or buffer name. */)
2729 (window, buffer)
5500c422
GM
2730 register Lisp_Object window, buffer;
2731{
2732 register Lisp_Object tem;
2733 register struct window *w = decode_window (window);
5500c422 2734
bed0c171 2735 XSETWINDOW (window, w);
5500c422 2736 buffer = Fget_buffer (buffer);
b7826503 2737 CHECK_BUFFER (buffer);
5500c422
GM
2738
2739 if (NILP (XBUFFER (buffer)->name))
2740 error ("Attempt to display deleted buffer");
2741
2742 tem = w->buffer;
2743 if (NILP (tem))
2744 error ("Window is deleted");
2745 else if (! EQ (tem, Qt)) /* w->buffer is t when the window
2746 is first being set up. */
2747 {
2748 if (!NILP (w->dedicated) && !EQ (tem, buffer))
2749 error ("Window is dedicated to `%s'",
d5db4077 2750 SDATA (XBUFFER (tem)->name));
5500c422
GM
2751
2752 unshow_buffer (w);
2753 }
2754
2755 set_window_buffer (window, buffer, 1);
7ab12479
JB
2756 return Qnil;
2757}
2758
2759DEFUN ("select-window", Fselect_window, Sselect_window, 1, 1, 0,
fdb82f93
PJ
2760 doc: /* Select WINDOW. Most editing will apply to WINDOW's buffer.
2761If WINDOW is not already selected, also make WINDOW's buffer current.
2762Note that the main editor command loop
2763selects the buffer of the selected window before each command. */)
2764 (window)
7ab12479 2765 register Lisp_Object window;
b7354ddf
RS
2766{
2767 return select_window_1 (window, 1);
2768}
2769\f
719eaeb1
GM
2770/* Note that selected_window can be nil
2771 when this is called from Fset_window_configuration. */
2772
b7354ddf
RS
2773static Lisp_Object
2774select_window_1 (window, recordflag)
2775 register Lisp_Object window;
2776 int recordflag;
7ab12479
JB
2777{
2778 register struct window *w;
719eaeb1 2779 register struct window *ow;
1ae1a37d 2780 struct frame *sf;
7ab12479 2781
b7826503 2782 CHECK_LIVE_WINDOW (window);
7ab12479
JB
2783
2784 w = XWINDOW (window);
50e88778 2785 w->frozen_window_start_p = 0;
7ab12479 2786
d834a2e9 2787 XSETFASTINT (w->use_time, ++window_select_count);
7ab12479
JB
2788 if (EQ (window, selected_window))
2789 return window;
2790
719eaeb1
GM
2791 if (!NILP (selected_window))
2792 {
2793 ow = XWINDOW (selected_window);
2794 if (! NILP (ow->buffer))
2795 set_marker_both (ow->pointm, ow->buffer,
2796 BUF_PT (XBUFFER (ow->buffer)),
2797 BUF_PT_BYTE (XBUFFER (ow->buffer)));
2798 }
7ab12479
JB
2799
2800 selected_window = window;
1ae1a37d
GM
2801 sf = SELECTED_FRAME ();
2802 if (XFRAME (WINDOW_FRAME (w)) != sf)
7ab12479 2803 {
44fa5b1e 2804 XFRAME (WINDOW_FRAME (w))->selected_window = window;
147a6615
RS
2805 /* Use this rather than Fhandle_switch_frame
2806 so that FRAME_FOCUS_FRAME is moved appropriately as we
2807 move around in the state where a minibuffer in a separate
2808 frame is active. */
2809 Fselect_frame (WINDOW_FRAME (w), Qnil);
7ab12479
JB
2810 }
2811 else
1ae1a37d 2812 sf->selected_window = window;
7ab12479 2813
b7354ddf
RS
2814 if (recordflag)
2815 record_buffer (w->buffer);
7ab12479
JB
2816 Fset_buffer (w->buffer);
2817
86e48436
RS
2818 XBUFFER (w->buffer)->last_selected_window = window;
2819
7ab12479
JB
2820 /* Go to the point recorded in the window.
2821 This is important when the buffer is in more
2822 than one window. It also matters when
2823 redisplay_window has altered point after scrolling,
2824 because it makes the change only in the window. */
2825 {
2826 register int new_point = marker_position (w->pointm);
2827 if (new_point < BEGV)
2828 SET_PT (BEGV);
a9c95e08 2829 else if (new_point > ZV)
7ab12479
JB
2830 SET_PT (ZV);
2831 else
2832 SET_PT (new_point);
2833 }
2834
2835 windows_or_buffers_changed++;
2836 return window;
2837}
b7354ddf 2838\f
441a127e
RS
2839/* Deiconify the frame containing the window WINDOW,
2840 unless it is the selected frame;
2841 then return WINDOW.
2842
2843 The reason for the exception for the selected frame
2844 is that it seems better not to change the selected frames visibility
2845 merely because of displaying a different buffer in it.
2846 The deiconification is useful when a buffer gets shown in
2847 another frame that you were not using lately. */
d07f802a
RS
2848
2849static Lisp_Object
2850display_buffer_1 (window)
2851 Lisp_Object window;
2852{
1ae1a37d
GM
2853 Lisp_Object frame = XWINDOW (window)->frame;
2854 FRAME_PTR f = XFRAME (frame);
2855
d07f802a 2856 FRAME_SAMPLE_VISIBILITY (f);
1ae1a37d
GM
2857
2858 if (!EQ (frame, selected_frame))
60117126
RS
2859 {
2860 if (FRAME_ICONIFIED_P (f))
1ae1a37d 2861 Fmake_frame_visible (frame);
37962e60 2862 else if (FRAME_VISIBLE_P (f))
1ae1a37d 2863 Fraise_frame (frame);
60117126 2864 }
1ae1a37d 2865
d07f802a
RS
2866 return window;
2867}
2868
4628f7a4 2869DEFUN ("special-display-p", Fspecial_display_p, Sspecial_display_p, 1, 1, 0,
fdb82f93
PJ
2870 doc: /* Returns non-nil if a buffer named BUFFER-NAME would be created specially.
2871The value is actually t if the frame should be called with default frame
2872parameters, and a list of frame parameters if they were specified.
2873See `special-display-buffer-names', and `special-display-regexps'. */)
2874 (buffer_name)
4628f7a4
EN
2875 Lisp_Object buffer_name;
2876{
2877 Lisp_Object tem;
2878
b7826503 2879 CHECK_STRING (buffer_name);
4628f7a4
EN
2880
2881 tem = Fmember (buffer_name, Vspecial_display_buffer_names);
2882 if (!NILP (tem))
2883 return Qt;
2884
2885 tem = Fassoc (buffer_name, Vspecial_display_buffer_names);
2886 if (!NILP (tem))
2887 return XCDR (tem);
2888
2889 for (tem = Vspecial_display_regexps; CONSP (tem); tem = XCDR (tem))
2890 {
2891 Lisp_Object car = XCAR (tem);
2892 if (STRINGP (car)
2893 && fast_string_match (car, buffer_name) >= 0)
2894 return Qt;
2895 else if (CONSP (car)
2896 && STRINGP (XCAR (car))
2897 && fast_string_match (XCAR (car), buffer_name) >= 0)
0057b00a 2898 return XCDR (car);
4628f7a4
EN
2899 }
2900 return Qnil;
2901}
2902
2903DEFUN ("same-window-p", Fsame_window_p, Ssame_window_p, 1, 1, 0,
fdb82f93
PJ
2904 doc: /* Returns non-nil if a new buffer named BUFFER-NAME would use the same window.
2905See `same-window-buffer-names' and `same-window-regexps'. */)
2906 (buffer_name)
4628f7a4
EN
2907 Lisp_Object buffer_name;
2908{
2909 Lisp_Object tem;
2910
b7826503 2911 CHECK_STRING (buffer_name);
4628f7a4
EN
2912
2913 tem = Fmember (buffer_name, Vsame_window_buffer_names);
2914 if (!NILP (tem))
2915 return Qt;
2916
2917 tem = Fassoc (buffer_name, Vsame_window_buffer_names);
2918 if (!NILP (tem))
2919 return Qt;
2920
2921 for (tem = Vsame_window_regexps; CONSP (tem); tem = XCDR (tem))
2922 {
2923 Lisp_Object car = XCAR (tem);
2924 if (STRINGP (car)
2925 && fast_string_match (car, buffer_name) >= 0)
2926 return Qt;
2927 else if (CONSP (car)
2928 && STRINGP (XCAR (car))
2929 && fast_string_match (XCAR (car), buffer_name) >= 0)
2930 return Qt;
2931 }
2932 return Qnil;
2933}
2934
fdb82f93 2935/* Use B so the default is (other-buffer). */
53f76081 2936DEFUN ("display-buffer", Fdisplay_buffer, Sdisplay_buffer, 1, 3,
fdb82f93
PJ
2937 "BDisplay buffer: \nP",
2938 doc: /* Make BUFFER appear in some window but don't select it.
2939BUFFER can be a buffer or a buffer name.
2940If BUFFER is shown already in some window, just use that one,
2941unless the window is the selected window and the optional second
2942argument NOT-THIS-WINDOW is non-nil (interactively, with prefix arg).
2943If `pop-up-frames' is non-nil, make a new frame if no window shows BUFFER.
2944Returns the window displaying BUFFER.
079d288f 2945If `display-buffer-reuse-frames' is non-nil, and another frame is currently
fdb82f93
PJ
2946displaying BUFFER, then simply raise that frame.
2947
2948The variables `special-display-buffer-names', `special-display-regexps',
2949`same-window-buffer-names', and `same-window-regexps' customize how certain
2950buffer names are handled.
2951
2952If optional argument FRAME is `visible', search all visible frames.
2953If FRAME is 0, search all visible and iconified frames.
2954If FRAME is t, search all frames.
2955If FRAME is a frame, search only that frame.
2956If FRAME is nil, search only the selected frame
2957 (actually the last nonminibuffer frame),
2958 unless `pop-up-frames' or `display-buffer-reuse-frames' is non-nil,
2959 which means search visible and iconified frames.
2960
2961If `even-window-heights' is non-nil, window heights will be evened out
2962if displaying the buffer causes two vertically adjacent windows to be
2963displayed. */)
2964 (buffer, not_this_window, frame)
53f76081 2965 register Lisp_Object buffer, not_this_window, frame;
7ab12479 2966{
aee631c2 2967 register Lisp_Object window, tem, swp;
1ae1a37d 2968 struct frame *f;
7ab12479 2969
aee631c2 2970 swp = Qnil;
7ab12479 2971 buffer = Fget_buffer (buffer);
b7826503 2972 CHECK_BUFFER (buffer);
7ab12479 2973
265a9e55 2974 if (!NILP (Vdisplay_buffer_function))
7ab12479
JB
2975 return call2 (Vdisplay_buffer_function, buffer, not_this_window);
2976
265a9e55 2977 if (NILP (not_this_window)
7ab12479 2978 && XBUFFER (XWINDOW (selected_window)->buffer) == XBUFFER (buffer))
d07f802a 2979 return display_buffer_1 (selected_window);
7ab12479 2980
855d8627
RS
2981 /* See if the user has specified this buffer should appear
2982 in the selected window. */
2983 if (NILP (not_this_window))
2984 {
aee631c2
RS
2985 swp = Fsame_window_p (XBUFFER (buffer)->name);
2986 if (!NILP (swp) && !no_switch_window (selected_window))
c63dc4a2
RS
2987 {
2988 Fswitch_to_buffer (buffer, Qnil);
d07f802a 2989 return display_buffer_1 (selected_window);
c63dc4a2 2990 }
855d8627
RS
2991 }
2992
079d288f 2993 /* If the user wants pop-up-frames or display-buffer-reuse-frames,
73dc5198
KH
2994 look for a window showing BUFFER on any visible or iconified frame.
2995 Otherwise search only the current frame. */
53f76081
RS
2996 if (! NILP (frame))
2997 tem = frame;
9c3da604
GM
2998 else if (pop_up_frames
2999 || display_buffer_reuse_frames
3000 || last_nonminibuf_frame == 0)
73dc5198
KH
3001 XSETFASTINT (tem, 0);
3002 else
73dc5198 3003 XSETFRAME (tem, last_nonminibuf_frame);
9c3da604 3004
73dc5198 3005 window = Fget_buffer_window (buffer, tem);
265a9e55
JB
3006 if (!NILP (window)
3007 && (NILP (not_this_window) || !EQ (window, selected_window)))
9c3da604 3008 return display_buffer_1 (window);
7ab12479 3009
a90712c2 3010 /* Certain buffer names get special handling. */
aee631c2 3011 if (!NILP (Vspecial_display_function) && NILP (swp))
a90712c2 3012 {
4628f7a4
EN
3013 tem = Fspecial_display_p (XBUFFER (buffer)->name);
3014 if (EQ (tem, Qt))
a90712c2 3015 return call1 (Vspecial_display_function, buffer);
4628f7a4
EN
3016 if (CONSP (tem))
3017 return call2 (Vspecial_display_function, buffer, tem);
a90712c2
RS
3018 }
3019
44fa5b1e
JB
3020 /* If there are no frames open that have more than a minibuffer,
3021 we need to create a new frame. */
3022 if (pop_up_frames || last_nonminibuf_frame == 0)
7ab12479 3023 {
a90712c2 3024 window = Fframe_selected_window (call0 (Vpop_up_frame_function));
7ab12479 3025 Fset_window_buffer (window, buffer);
d07f802a 3026 return display_buffer_1 (window);
7ab12479 3027 }
7ab12479 3028
1ae1a37d 3029 f = SELECTED_FRAME ();
43bad991 3030 if (pop_up_windows
1ae1a37d 3031 || FRAME_MINIBUF_ONLY_P (f)
cee67da9
RS
3032 /* If the current frame is a special display frame,
3033 don't try to reuse its windows. */
1ae1a37d 3034 || !NILP (XWINDOW (FRAME_ROOT_WINDOW (f))->dedicated))
7ab12479 3035 {
12cae7c0
KH
3036 Lisp_Object frames;
3037
37962e60 3038 frames = Qnil;
1ae1a37d 3039 if (FRAME_MINIBUF_ONLY_P (f))
74112613 3040 XSETFRAME (frames, last_nonminibuf_frame);
7ab12479
JB
3041 /* Don't try to create a window if would get an error */
3042 if (split_height_threshold < window_min_height << 1)
3043 split_height_threshold = window_min_height << 1;
3044
cee67da9
RS
3045 /* Note that both Fget_largest_window and Fget_lru_window
3046 ignore minibuffers and dedicated windows.
3047 This means they can return nil. */
7ab12479 3048
cee67da9
RS
3049 /* If the frame we would try to split cannot be split,
3050 try other frames. */
1ae1a37d 3051 if (FRAME_NO_SPLIT_P (NILP (frames) ? f : last_nonminibuf_frame))
cee67da9
RS
3052 {
3053 /* Try visible frames first. */
3054 window = Fget_largest_window (Qvisible);
3055 /* If that didn't work, try iconified frames. */
3056 if (NILP (window))
3057 window = Fget_largest_window (make_number (0));
3058 if (NILP (window))
3059 window = Fget_largest_window (Qt);
3060 }
3061 else
3062 window = Fget_largest_window (frames);
3063
92cca945
RS
3064 /* If we got a tall enough full-width window that can be split,
3065 split it. */
265a9e55 3066 if (!NILP (window)
92cca945 3067 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window)->frame))
7ab12479 3068 && window_height (window) >= split_height_threshold
111e5992 3069 && WINDOW_FULL_WIDTH_P (XWINDOW (window)))
7ab12479
JB
3070 window = Fsplit_window (window, Qnil, Qnil);
3071 else
3072 {
1942f68f
RS
3073 Lisp_Object upper, lower, other;
3074
44fa5b1e 3075 window = Fget_lru_window (frames);
92cca945
RS
3076 /* If the LRU window is selected, and big enough,
3077 and can be split, split it. */
cee67da9 3078 if (!NILP (window)
92cca945 3079 && ! FRAME_NO_SPLIT_P (XFRAME (XWINDOW (window)->frame))
cee67da9
RS
3080 && (EQ (window, selected_window)
3081 || EQ (XWINDOW (window)->parent, Qnil))
7ab12479
JB
3082 && window_height (window) >= window_min_height << 1)
3083 window = Fsplit_window (window, Qnil, Qnil);
cee67da9 3084 /* If Fget_lru_window returned nil, try other approaches. */
48d9379d 3085
cee67da9 3086 /* Try visible frames first. */
48d9379d
RS
3087 if (NILP (window))
3088 window = Fget_buffer_window (buffer, Qvisible);
cee67da9
RS
3089 if (NILP (window))
3090 window = Fget_largest_window (Qvisible);
3091 /* If that didn't work, try iconified frames. */
48d9379d
RS
3092 if (NILP (window))
3093 window = Fget_buffer_window (buffer, make_number (0));
cee67da9
RS
3094 if (NILP (window))
3095 window = Fget_largest_window (make_number (0));
3096 /* Try invisible frames. */
48d9379d
RS
3097 if (NILP (window))
3098 window = Fget_buffer_window (buffer, Qt);
cee67da9
RS
3099 if (NILP (window))
3100 window = Fget_largest_window (Qt);
3101 /* As a last resort, make a new frame. */
3102 if (NILP (window))
3103 window = Fframe_selected_window (call0 (Vpop_up_frame_function));
1942f68f
RS
3104 /* If window appears above or below another,
3105 even out their heights. */
cac66e4f 3106 other = upper = lower = Qnil;
1942f68f
RS
3107 if (!NILP (XWINDOW (window)->prev))
3108 other = upper = XWINDOW (window)->prev, lower = window;
3109 if (!NILP (XWINDOW (window)->next))
3110 other = lower = XWINDOW (window)->next, upper = window;
3111 if (!NILP (other)
6529ed87 3112 && !NILP (Veven_window_heights)
1942f68f 3113 /* Check that OTHER and WINDOW are vertically arrayed. */
296b535c
KH
3114 && !EQ (XWINDOW (other)->top, XWINDOW (window)->top)
3115 && (XFASTINT (XWINDOW (other)->height)
3116 > XFASTINT (XWINDOW (window)->height)))
1942f68f 3117 {
296b535c
KH
3118 int total = (XFASTINT (XWINDOW (other)->height)
3119 + XFASTINT (XWINDOW (window)->height));
86c8e823
GM
3120 enlarge_window (upper,
3121 total / 2 - XFASTINT (XWINDOW (upper)->height),
f95284d2 3122 0, 0);
1942f68f 3123 }
7ab12479
JB
3124 }
3125 }
3126 else
3127 window = Fget_lru_window (Qnil);
3128
3129 Fset_window_buffer (window, buffer);
d07f802a 3130 return display_buffer_1 (window);
7ab12479
JB
3131}
3132
3133void
3134temp_output_buffer_show (buf)
3135 register Lisp_Object buf;
3136{
3137 register struct buffer *old = current_buffer;
3138 register Lisp_Object window;
3139 register struct window *w;
3140
bccd3dd1
RS
3141 XBUFFER (buf)->directory = current_buffer->directory;
3142
7ab12479 3143 Fset_buffer (buf);
c6367666 3144 BUF_SAVE_MODIFF (XBUFFER (buf)) = MODIFF;
7ab12479
JB
3145 BEGV = BEG;
3146 ZV = Z;
3147 SET_PT (BEG);
06be4f85 3148#if 0 /* rms: there should be no reason for this. */
b1599b4c 3149 XBUFFER (buf)->prevent_redisplay_optimizations_p = 1;
06be4f85 3150#endif
7ab12479
JB
3151 set_buffer_internal (old);
3152
3153 if (!EQ (Vtemp_buffer_show_function, Qnil))
3154 call1 (Vtemp_buffer_show_function, buf);
3155 else
3156 {
53f76081 3157 window = Fdisplay_buffer (buf, Qnil, Qnil);
7ab12479 3158
1ae1a37d 3159 if (!EQ (XWINDOW (window)->frame, selected_frame))
44fa5b1e 3160 Fmake_frame_visible (WINDOW_FRAME (XWINDOW (window)));
7ab12479
JB
3161 Vminibuf_scroll_window = window;
3162 w = XWINDOW (window);
d834a2e9 3163 XSETFASTINT (w->hscroll, 0);
ea68264b 3164 XSETFASTINT (w->min_hscroll, 0);
b73ea88e
RS
3165 set_marker_restricted_both (w->start, buf, 1, 1);
3166 set_marker_restricted_both (w->pointm, buf, 1, 1);
a58ec57d 3167
beb4e312
RS
3168 /* Run temp-buffer-show-hook, with the chosen window selected
3169 and it sbuffer current. */
f52cca03 3170 if (!NILP (Vrun_hooks))
2cccc823 3171 {
f52cca03
RS
3172 Lisp_Object tem;
3173 tem = Fboundp (Qtemp_buffer_show_hook);
2cccc823
RS
3174 if (!NILP (tem))
3175 {
f52cca03
RS
3176 tem = Fsymbol_value (Qtemp_buffer_show_hook);
3177 if (!NILP (tem))
3178 {
aed13378 3179 int count = SPECPDL_INDEX ();
b7354ddf
RS
3180 Lisp_Object prev_window;
3181 prev_window = selected_window;
2cccc823 3182
f52cca03 3183 /* Select the window that was chosen, for running the hook. */
65a04b96 3184 record_unwind_protect (Fselect_window, prev_window);
b7354ddf 3185 select_window_1 (window, 0);
beb4e312 3186 Fset_buffer (w->buffer);
f52cca03 3187 call1 (Vrun_hooks, Qtemp_buffer_show_hook);
b7354ddf 3188 select_window_1 (prev_window, 0);
f52cca03
RS
3189 unbind_to (count, Qnil);
3190 }
2cccc823
RS
3191 }
3192 }
3193 }
7ab12479
JB
3194}
3195\f
dfcf069d 3196static void
7ab12479
JB
3197make_dummy_parent (window)
3198 Lisp_Object window;
3199{
cffec418 3200 Lisp_Object new;
7ab12479 3201 register struct window *o, *p;
cffec418 3202 int i;
7ab12479 3203
cffec418 3204 o = XWINDOW (window);
26605be9 3205 p = allocate_window ();
cffec418 3206 for (i = 0; i < VECSIZE (struct window); ++i)
26605be9
GM
3207 ((struct Lisp_Vector *) p)->contents[i]
3208 = ((struct Lisp_Vector *)o)->contents[i];
cffec418 3209 XSETWINDOW (new, p);
7ab12479 3210
d834a2e9 3211 XSETFASTINT (p->sequence_number, ++sequence_number);
7ab12479
JB
3212
3213 /* Put new into window structure in place of window */
3214 replace_window (window, new);
3215
3216 o->next = Qnil;
3217 o->prev = Qnil;
3218 o->vchild = Qnil;
3219 o->hchild = Qnil;
3220 o->parent = new;
3221
3222 p->start = Qnil;
3223 p->pointm = Qnil;
3224 p->buffer = Qnil;
3225}
3226
3227DEFUN ("split-window", Fsplit_window, Ssplit_window, 0, 3, "",
fdb82f93
PJ
3228 doc: /* Split WINDOW, putting SIZE lines in the first of the pair.
3229WINDOW defaults to selected one and SIZE to half its size.
3230If optional third arg HORFLAG is non-nil, split side by side
3231and put SIZE columns in the first of the pair. In that case,
3232SIZE includes that window's scroll bar, or the divider column to its right. */)
3233 (window, size, horflag)
77ae0fe3 3234 Lisp_Object window, size, horflag;
7ab12479
JB
3235{
3236 register Lisp_Object new;
3237 register struct window *o, *p;
c0807608 3238 FRAME_PTR fo;
77ae0fe3 3239 register int size_int;
7ab12479 3240
265a9e55 3241 if (NILP (window))
7ab12479
JB
3242 window = selected_window;
3243 else
b7826503 3244 CHECK_LIVE_WINDOW (window);
7ab12479
JB
3245
3246 o = XWINDOW (window);
c0807608 3247 fo = XFRAME (WINDOW_FRAME (o));
7ab12479 3248
77ae0fe3 3249 if (NILP (size))
7ab12479 3250 {
265a9e55 3251 if (!NILP (horflag))
c0807608 3252 /* Calculate the size of the left-hand window, by dividing
25b33244
KH
3253 the usable space in columns by two.
3254 We round up, since the left-hand window may include
3255 a dividing line, while the right-hand may not. */
3256 size_int = (XFASTINT (o->width) + 1) >> 1;
7ab12479 3257 else
77ae0fe3 3258 size_int = XFASTINT (o->height) >> 1;
7ab12479
JB
3259 }
3260 else
3261 {
b7826503 3262 CHECK_NUMBER (size);
77ae0fe3 3263 size_int = XINT (size);
7ab12479
JB
3264 }
3265
3266 if (MINI_WINDOW_P (o))
3267 error ("Attempt to split minibuffer window");
233a4a2c
GM
3268 else if (window_fixed_size_p (o, !NILP (horflag), 0))
3269 error ("Attempt to split fixed-size window");
7ab12479 3270
a481b3ea 3271 check_min_window_sizes ();
7ab12479 3272
265a9e55 3273 if (NILP (horflag))
7ab12479 3274 {
77ae0fe3
KH
3275 if (size_int < window_min_height)
3276 error ("Window height %d too small (after splitting)", size_int);
3277 if (size_int + window_min_height > XFASTINT (o->height))
37962e60 3278 error ("Window height %d too small (after splitting)",
77ae0fe3 3279 XFASTINT (o->height) - size_int);
265a9e55
JB
3280 if (NILP (o->parent)
3281 || NILP (XWINDOW (o->parent)->vchild))
7ab12479
JB
3282 {
3283 make_dummy_parent (window);
3284 new = o->parent;
3285 XWINDOW (new)->vchild = window;
3286 }
3287 }
3288 else
3289 {
77ae0fe3
KH
3290 if (size_int < window_min_width)
3291 error ("Window width %d too small (after splitting)", size_int);
a59fed7e
RS
3292
3293 if (size_int + window_min_width > XFASTINT (o->width))
37962e60 3294 error ("Window width %d too small (after splitting)",
a59fed7e 3295 XFASTINT (o->width) - size_int);
265a9e55
JB
3296 if (NILP (o->parent)
3297 || NILP (XWINDOW (o->parent)->hchild))
7ab12479
JB
3298 {
3299 make_dummy_parent (window);
3300 new = o->parent;
3301 XWINDOW (new)->hchild = window;
3302 }
3303 }
3304
3305 /* Now we know that window's parent is a vertical combination
3306 if we are dividing vertically, or a horizontal combination
3307 if we are making side-by-side windows */
3308
3309 windows_or_buffers_changed++;
c0807608 3310 FRAME_WINDOW_SIZES_CHANGED (fo) = 1;
7ab12479
JB
3311 new = make_window ();
3312 p = XWINDOW (new);
3313
44fa5b1e 3314 p->frame = o->frame;
7ab12479 3315 p->next = o->next;
265a9e55 3316 if (!NILP (p->next))
7ab12479
JB
3317 XWINDOW (p->next)->prev = new;
3318 p->prev = window;
3319 o->next = new;
3320 p->parent = o->parent;
3321 p->buffer = Qt;
5500c422
GM
3322 p->window_end_valid = Qnil;
3323 bzero (&p->last_cursor, sizeof p->last_cursor);
7ab12479 3324
44fa5b1e 3325 /* Apportion the available frame space among the two new windows */
7ab12479 3326
265a9e55 3327 if (!NILP (horflag))
7ab12479
JB
3328 {
3329 p->height = o->height;
3330 p->top = o->top;
a59fed7e 3331 XSETFASTINT (p->width, XFASTINT (o->width) - size_int);
77ae0fe3
KH
3332 XSETFASTINT (o->width, size_int);
3333 XSETFASTINT (p->left, XFASTINT (o->left) + size_int);
7ab12479
JB
3334 }
3335 else
3336 {
3337 p->left = o->left;
3338 p->width = o->width;
77ae0fe3
KH
3339 XSETFASTINT (p->height, XFASTINT (o->height) - size_int);
3340 XSETFASTINT (o->height, size_int);
3341 XSETFASTINT (p->top, XFASTINT (o->top) + size_int);
7ab12479
JB
3342 }
3343
5500c422
GM
3344 /* Adjust glyph matrices. */
3345 adjust_glyphs (fo);
543f5fb1 3346 Fset_window_buffer (new, o->buffer);
7ab12479
JB
3347 return new;
3348}
3349\f
f95284d2 3350DEFUN ("enlarge-window", Fenlarge_window, Senlarge_window, 1, 3, "p",
fdb82f93
PJ
3351 doc: /* Make current window ARG lines bigger.
3352From program, optional second arg non-nil means grow sideways ARG columns.
f95284d2
RS
3353Interactively, if an argument is not given, make the window one line bigger.
3354
3355Optional third arg PRESERVE-BEFORE, if non-nil, means do not change the size
3356of the siblings above or to the left of the selected window. Only
3357siblings to the right or below are changed. */)
3358 (arg, side, preserve_before)
3359 register Lisp_Object arg, side, preserve_before;
7ab12479 3360{
b7826503 3361 CHECK_NUMBER (arg);
f95284d2
RS
3362 enlarge_window (selected_window, XINT (arg), !NILP (side),
3363 !NILP (preserve_before));
543f5fb1
RS
3364
3365 if (! NILP (Vwindow_configuration_change_hook))
3366 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
3367
7ab12479
JB
3368 return Qnil;
3369}
3370
3371DEFUN ("shrink-window", Fshrink_window, Sshrink_window, 1, 2, "p",
fdb82f93
PJ
3372 doc: /* Make current window ARG lines smaller.
3373From program, optional second arg non-nil means shrink sideways arg columns.
3374Interactively, if an argument is not given, make the window one line smaller. */)
3375 (arg, side)
413430c5 3376 register Lisp_Object arg, side;
7ab12479 3377{
b7826503 3378 CHECK_NUMBER (arg);
f95284d2 3379 enlarge_window (selected_window, -XINT (arg), !NILP (side), 0);
543f5fb1
RS
3380
3381 if (! NILP (Vwindow_configuration_change_hook))
3382 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
3383
7ab12479
JB
3384 return Qnil;
3385}
3386
3387int
3388window_height (window)
3389 Lisp_Object window;
3390{
3391 register struct window *p = XWINDOW (window);
3392 return XFASTINT (p->height);
3393}
3394
3395int
3396window_width (window)
3397 Lisp_Object window;
3398{
3399 register struct window *p = XWINDOW (window);
3400 return XFASTINT (p->width);
3401}
3402
c1636aa6 3403
7ab12479 3404#define CURBEG(w) \
3578db3c 3405 *(widthflag ? &(XWINDOW (w)->left) : &(XWINDOW (w)->top))
7ab12479
JB
3406
3407#define CURSIZE(w) \
3578db3c 3408 *(widthflag ? &(XWINDOW (w)->width) : &(XWINDOW (w)->height))
7ab12479 3409
233a4a2c 3410
f95284d2 3411/* Enlarge WINDOW by DELTA. WIDTHFLAG non-zero means
233a4a2c 3412 increase its width. Siblings of the selected window are resized to
f95284d2
RS
3413 fulfill the size request. If they become too small in the process,
3414 they will be deleted.
3415
3416 If PRESERVE_BEFORE is nonzero, that means don't alter
3417 the siblings to the left or above WINDOW. */
7ab12479 3418
f984d4fc 3419static void
f95284d2 3420enlarge_window (window, delta, widthflag, preserve_before)
86c8e823 3421 Lisp_Object window;
f95284d2 3422 int delta, widthflag, preserve_before;
7ab12479 3423{
86c8e823 3424 Lisp_Object parent, next, prev;
233a4a2c 3425 struct window *p;
3578db3c
KR
3426 Lisp_Object *sizep;
3427 int maximum;
5e14b1fc
AS
3428 int (*sizefun) P_ ((Lisp_Object))
3429 = widthflag ? window_width : window_height;
233a4a2c 3430 void (*setsizefun) P_ ((Lisp_Object, int, int))
5e14b1fc 3431 = (widthflag ? set_window_width : set_window_height);
7ab12479 3432
233a4a2c
GM
3433 /* Check values of window_min_width and window_min_height for
3434 validity. */
a481b3ea 3435 check_min_window_sizes ();
7ab12479 3436
233a4a2c 3437 /* Give up if this window cannot be resized. */
233a4a2c
GM
3438 if (window_fixed_size_p (XWINDOW (window), widthflag, 1))
3439 error ("Window is not resizable");
3440
3441 /* Find the parent of the selected window. */
7ab12479
JB
3442 while (1)
3443 {
3444 p = XWINDOW (window);
3445 parent = p->parent;
233a4a2c 3446
265a9e55 3447 if (NILP (parent))
7ab12479
JB
3448 {
3449 if (widthflag)
3450 error ("No other window to side of this one");
3451 break;
3452 }
233a4a2c
GM
3453
3454 if (widthflag
3455 ? !NILP (XWINDOW (parent)->hchild)
265a9e55 3456 : !NILP (XWINDOW (parent)->vchild))
7ab12479 3457 break;
233a4a2c 3458
7ab12479
JB
3459 window = parent;
3460 }
3461
05c2896a 3462 sizep = &CURSIZE (window);
7ab12479 3463
7ab12479
JB
3464 {
3465 register int maxdelta;
7ab12479 3466
f95284d2
RS
3467 /* Compute the maximum size increment this window can have. */
3468
3469 if (preserve_before)
3470 {
3471 if (!NILP (parent))
3472 {
3473 maxdelta = (*sizefun) (parent) - XINT (*sizep);
3474 /* Subtract size of siblings before, since we can't take that. */
1ab964d7 3475 maxdelta -= XINT (CURBEG (window)) - XINT (CURBEG (parent));
f95284d2
RS
3476 }
3477 else
3478 maxdelta = (!NILP (p->next) ? ((*sizefun) (p->next)
3479 - window_min_size (XWINDOW (p->next),
3480 widthflag, 0, 0))
3481 : (delta = 0));
3482 }
3483 else
3484 maxdelta = (!NILP (parent) ? (*sizefun) (parent) - XINT (*sizep)
3485 /* This is a main window followed by a minibuffer. */
3486 : !NILP (p->next) ? ((*sizefun) (p->next)
3487 - window_min_size (XWINDOW (p->next),
3488 widthflag, 0, 0))
3489 /* This is a minibuffer following a main window. */
3490 : !NILP (p->prev) ? ((*sizefun) (p->prev)
3491 - window_min_size (XWINDOW (p->prev),
3492 widthflag, 0, 0))
3493 /* This is a frame with only one window, a minibuffer-only
3494 or a minibufferless frame. */
3495 : (delta = 0));
7ab12479
JB
3496
3497 if (delta > maxdelta)
3498 /* This case traps trying to make the minibuffer
44fa5b1e
JB
3499 the full frame, or make the only window aside from the
3500 minibuffer the full frame. */
7ab12479 3501 delta = maxdelta;
6b54027b 3502 }
d5783c40 3503
3578db3c 3504 if (XINT (*sizep) + delta < window_min_size (XWINDOW (window), widthflag, 0, 0))
6b54027b 3505 {
543f5fb1 3506 delete_window (window);
d5783c40 3507 return;
6b54027b
RS
3508 }
3509
3510 if (delta == 0)
3511 return;
7ab12479 3512
f95284d2 3513 /* Find the total we can get from other siblings without deleting them. */
db98a733
RS
3514 maximum = 0;
3515 for (next = p->next; ! NILP (next); next = XWINDOW (next)->next)
c1636aa6 3516 maximum += (*sizefun) (next) - window_min_size (XWINDOW (next),
f984d4fc 3517 widthflag, 0, 0);
f95284d2
RS
3518 if (! preserve_before)
3519 for (prev = p->prev; ! NILP (prev); prev = XWINDOW (prev)->prev)
3520 maximum += (*sizefun) (prev) - window_min_size (XWINDOW (prev),
3521 widthflag, 0, 0);
db98a733 3522
f95284d2 3523 /* If we can get it all from them without deleting them, do so. */
c6b530ed 3524 if (delta <= maximum)
7ab12479 3525 {
db98a733
RS
3526 Lisp_Object first_unaffected;
3527 Lisp_Object first_affected;
233a4a2c 3528 int fixed_p;
db98a733
RS
3529
3530 next = p->next;
3531 prev = p->prev;
3532 first_affected = window;
3533 /* Look at one sibling at a time,
3534 moving away from this window in both directions alternately,
3535 and take as much as we can get without deleting that sibling. */
f95284d2
RS
3536 while (delta != 0
3537 && (!NILP (next) || (!preserve_before && !NILP (prev))))
db98a733 3538 {
db98a733
RS
3539 if (! NILP (next))
3540 {
c1636aa6 3541 int this_one = ((*sizefun) (next)
233a4a2c 3542 - window_min_size (XWINDOW (next),
f984d4fc 3543 widthflag, 0, &fixed_p));
233a4a2c
GM
3544 if (!fixed_p)
3545 {
3546 if (this_one > delta)
3547 this_one = delta;
3548
3549 (*setsizefun) (next, (*sizefun) (next) - this_one, 0);
3578db3c 3550 (*setsizefun) (window, XINT (*sizep) + this_one, 0);
db98a733 3551
233a4a2c
GM
3552 delta -= this_one;
3553 }
3554
db98a733
RS
3555 next = XWINDOW (next)->next;
3556 }
233a4a2c 3557
db98a733
RS
3558 if (delta == 0)
3559 break;
233a4a2c 3560
f95284d2 3561 if (!preserve_before && ! NILP (prev))
db98a733 3562 {
c1636aa6 3563 int this_one = ((*sizefun) (prev)
233a4a2c 3564 - window_min_size (XWINDOW (prev),
f984d4fc 3565 widthflag, 0, &fixed_p));
233a4a2c
GM
3566 if (!fixed_p)
3567 {
3568 if (this_one > delta)
3569 this_one = delta;
3570
3571 first_affected = prev;
3572
3573 (*setsizefun) (prev, (*sizefun) (prev) - this_one, 0);
3578db3c 3574 (*setsizefun) (window, XINT (*sizep) + this_one, 0);
233a4a2c
GM
3575
3576 delta -= this_one;
3577 }
3578
db98a733
RS
3579 prev = XWINDOW (prev)->prev;
3580 }
3581 }
3582
233a4a2c
GM
3583 xassert (delta == 0);
3584
db98a733
RS
3585 /* Now recalculate the edge positions of all the windows affected,
3586 based on the new sizes. */
3587 first_unaffected = next;
3588 prev = first_affected;
3589 for (next = XWINDOW (prev)->next; ! EQ (next, first_unaffected);
3590 prev = next, next = XWINDOW (next)->next)
3591 {
3578db3c 3592 XSETINT (CURBEG (next), XINT (CURBEG (prev)) + (*sizefun) (prev));
db98a733
RS
3593 /* This does not change size of NEXT,
3594 but it propagates the new top edge to its children */
3595 (*setsizefun) (next, (*sizefun) (next), 0);
3596 }
7ab12479
JB
3597 }
3598 else
3599 {
3600 register int delta1;
3601 register int opht = (*sizefun) (parent);
3602
3578db3c 3603 if (opht <= XINT (*sizep) + delta)
daf516d3
RS
3604 {
3605 /* If trying to grow this window to or beyond size of the parent,
3606 just delete all the sibling windows. */
dc1ab1ee 3607 Lisp_Object start, tem, next;
daf516d3 3608
dc1ab1ee
RS
3609 start = XWINDOW (parent)->vchild;
3610 if (NILP (start))
3611 start = XWINDOW (parent)->hchild;
daf516d3 3612
dc1ab1ee
RS
3613 /* Delete any siblings that come after WINDOW. */
3614 tem = XWINDOW (window)->next;
daf516d3
RS
3615 while (! NILP (tem))
3616 {
3617 next = XWINDOW (tem)->next;
dc1ab1ee
RS
3618 delete_window (tem);
3619 tem = next;
3620 }
3621
3622 /* Delete any siblings that come after WINDOW.
3623 Note that if START is not WINDOW, then WINDOW still
3624 Fhas siblings, so WINDOW has not yet replaced its parent. */
3625 tem = start;
3626 while (! EQ (tem, window))
3627 {
3628 next = XWINDOW (tem)->next;
3629 delete_window (tem);
daf516d3
RS
3630 tem = next;
3631 }
3632 }
7ab12479 3633 else
233a4a2c
GM
3634 {
3635 /* Otherwise, make delta1 just right so that if we add
3636 delta1 lines to this window and to the parent, and then
3637 shrink the parent back to its original size, the new
3638 proportional size of this window will increase by delta.
3639
3640 The function size_window will compute the new height h'
3641 of the window from delta1 as:
3642
3643 e = delta1/n
3644 x = delta1 - delta1/n * n for the 1st resizable child
3645 h' = h + e + x
3646
3647 where n is the number of children that can be resized.
3648 We can ignore x by choosing a delta1 that is a multiple of
3649 n. We want the height of this window to come out as
3650
3651 h' = h + delta
3652
3653 So, delta1 must be
3654
3655 h + e = h + delta
3656 delta1/n = delta
3657 delta1 = n * delta.
3658
3659 The number of children n rquals the number of resizable
3660 children of this window + 1 because we know window itself
3661 is resizable (otherwise we would have signalled an error. */
3662
3663 struct window *w = XWINDOW (window);
3664 Lisp_Object s;
3665 int n = 1;
3666
3667 for (s = w->next; !NILP (s); s = XWINDOW (s)->next)
3668 if (!window_fixed_size_p (XWINDOW (s), widthflag, 0))
3669 ++n;
3670 for (s = w->prev; !NILP (s); s = XWINDOW (s)->prev)
3671 if (!window_fixed_size_p (XWINDOW (s), widthflag, 0))
3672 ++n;
3673
3674 delta1 = n * delta;
7ab12479 3675
daf516d3
RS
3676 /* Add delta1 lines or columns to this window, and to the parent,
3677 keeping things consistent while not affecting siblings. */
3678 XSETINT (CURSIZE (parent), opht + delta1);
3679 (*setsizefun) (window, XINT (*sizep) + delta1, 0);
3680
3681 /* Squeeze out delta1 lines or columns from our parent,
3682 shriking this window and siblings proportionately.
3683 This brings parent back to correct size.
3684 Delta1 was calculated so this makes this window the desired size,
3685 taking it all out of the siblings. */
3686 (*setsizefun) (parent, opht, 0);
3687
3688 }
7ab12479
JB
3689 }
3690
d834a2e9 3691 XSETFASTINT (p->last_modified, 0);
3cd21523 3692 XSETFASTINT (p->last_overlay_modified, 0);
5500c422
GM
3693
3694 /* Adjust glyph matrices. */
3695 adjust_glyphs (XFRAME (WINDOW_FRAME (XWINDOW (window))));
7ab12479 3696}
c1636aa6 3697
7ab12479
JB
3698#undef CURBEG
3699#undef CURSIZE
3700
5500c422 3701
f984d4fc
GM
3702\f
3703/***********************************************************************
3704 Resizing Mini-Windows
3705 ***********************************************************************/
3706
3707static void shrink_window_lowest_first P_ ((struct window *, int));
f984d4fc 3708
43b4a21f
GM
3709enum save_restore_action
3710{
3711 CHECK_ORIG_SIZES,
3712 SAVE_ORIG_SIZES,
3713 RESTORE_ORIG_SIZES
3714};
3715
3716static int save_restore_orig_size P_ ((struct window *,
3717 enum save_restore_action));
f984d4fc
GM
3718
3719/* Shrink windows rooted in window W to HEIGHT. Take the space needed
3720 from lowest windows first. */
3721
3722static void
3723shrink_window_lowest_first (w, height)
3724 struct window *w;
3725 int height;
3726{
3727 struct window *c;
3728 Lisp_Object child;
3729 int old_height;
3730
3731 xassert (!MINI_WINDOW_P (w));
3732
3733 /* Set redisplay hints. */
3734 XSETFASTINT (w->last_modified, 0);
3735 XSETFASTINT (w->last_overlay_modified, 0);
3736 windows_or_buffers_changed++;
3737 FRAME_WINDOW_SIZES_CHANGED (XFRAME (WINDOW_FRAME (w))) = 1;
3738
3739 old_height = XFASTINT (w->height);
3740 XSETFASTINT (w->height, height);
3741
3742 if (!NILP (w->hchild))
3743 {
3744 for (child = w->hchild; !NILP (child); child = c->next)
3745 {
3746 c = XWINDOW (child);
3747 c->top = w->top;
3748 shrink_window_lowest_first (c, height);
3749 }
3750 }
3751 else if (!NILP (w->vchild))
3752 {
3753 Lisp_Object last_child;
3754 int delta = old_height - height;
3755 int last_top;
6bbd7a29
GM
3756
3757 last_child = Qnil;
f984d4fc
GM
3758
3759 /* Find the last child. We are taking space from lowest windows
3760 first, so we iterate over children from the last child
3761 backwards. */
3762 for (child = w->vchild; !NILP (child); child = XWINDOW (child)->next)
3763 last_child = child;
3764
3765 /* Assign new heights. We leave only MIN_SAFE_WINDOW_HEIGHT. */
3766 for (child = last_child; delta && !NILP (child); child = c->prev)
3767 {
3768 int this_one;
3769
3770 c = XWINDOW (child);
3771 this_one = XFASTINT (c->height) - MIN_SAFE_WINDOW_HEIGHT;
3772
3773 if (this_one > delta)
3774 this_one = delta;
3775
3776 shrink_window_lowest_first (c, XFASTINT (c->height) - this_one);
3777 delta -= this_one;
3778 }
3779
3780 /* Compute new positions. */
3578db3c 3781 last_top = XINT (w->top);
f984d4fc
GM
3782 for (child = w->vchild; !NILP (child); child = c->next)
3783 {
3784 c = XWINDOW (child);
3785 c->top = make_number (last_top);
3786 shrink_window_lowest_first (c, XFASTINT (c->height));
3787 last_top += XFASTINT (c->height);
3788 }
3789 }
3790}
3791
3792
43b4a21f
GM
3793/* Save, restore, or check positions and sizes in the window tree
3794 rooted at W. ACTION says what to do.
f984d4fc 3795
43b4a21f
GM
3796 If ACTION is CHECK_ORIG_SIZES, check if orig_top and orig_height
3797 members are valid for all windows in the window tree. Value is
3798 non-zero if they are valid.
3799
3800 If ACTION is SAVE_ORIG_SIZES, save members top and height in
3801 orig_top and orig_height for all windows in the tree.
3802
3803 If ACTION is RESTORE_ORIG_SIZES, restore top and height from
3804 values stored in orig_top and orig_height for all windows. */
3805
3806static int
3807save_restore_orig_size (w, action)
f984d4fc 3808 struct window *w;
43b4a21f 3809 enum save_restore_action action;
f984d4fc 3810{
43b4a21f
GM
3811 int success_p = 1;
3812
f984d4fc
GM
3813 while (w)
3814 {
3815 if (!NILP (w->hchild))
43b4a21f
GM
3816 {
3817 if (!save_restore_orig_size (XWINDOW (w->hchild), action))
3818 success_p = 0;
3819 }
f984d4fc 3820 else if (!NILP (w->vchild))
43b4a21f
GM
3821 {
3822 if (!save_restore_orig_size (XWINDOW (w->vchild), action))
3823 success_p = 0;
3824 }
f984d4fc 3825
43b4a21f 3826 switch (action)
f984d4fc 3827 {
43b4a21f
GM
3828 case CHECK_ORIG_SIZES:
3829 if (!INTEGERP (w->orig_top) || !INTEGERP (w->orig_height))
3830 return 0;
3831 break;
3832
3833 case SAVE_ORIG_SIZES:
f984d4fc
GM
3834 w->orig_top = w->top;
3835 w->orig_height = w->height;
43b4a21f
GM
3836 XSETFASTINT (w->last_modified, 0);
3837 XSETFASTINT (w->last_overlay_modified, 0);
3838 break;
3839
3840 case RESTORE_ORIG_SIZES:
f984d4fc
GM
3841 xassert (INTEGERP (w->orig_top) && INTEGERP (w->orig_height));
3842 w->top = w->orig_top;
3843 w->height = w->orig_height;
3844 w->orig_height = w->orig_top = Qnil;
43b4a21f
GM
3845 XSETFASTINT (w->last_modified, 0);
3846 XSETFASTINT (w->last_overlay_modified, 0);
3847 break;
3848
3849 default:
3850 abort ();
f984d4fc 3851 }
43b4a21f 3852
f984d4fc
GM
3853 w = NILP (w->next) ? NULL : XWINDOW (w->next);
3854 }
43b4a21f
GM
3855
3856 return success_p;
f984d4fc
GM
3857}
3858
3859
3860/* Grow mini-window W by DELTA lines, DELTA >= 0, or as much as we can
3861 without deleting other windows. */
3862
3863void
3864grow_mini_window (w, delta)
3865 struct window *w;
3866 int delta;
3867{
3868 struct frame *f = XFRAME (w->frame);
3869 struct window *root;
3870
3871 xassert (MINI_WINDOW_P (w));
3872 xassert (delta >= 0);
3873
3874 /* Check values of window_min_width and window_min_height for
3875 validity. */
3876 check_min_window_sizes ();
3877
3878 /* Compute how much we can enlarge the mini-window without deleting
3879 other windows. */
3880 root = XWINDOW (FRAME_ROOT_WINDOW (f));
3881 if (delta)
3882 {
3883 int min_height = window_min_size (root, 0, 0, 0);
3884 if (XFASTINT (root->height) - delta < min_height)
8b8bd9c6 3885 /* Note that the root window may already be smaller than
eafa3196
GM
3886 min_height. */
3887 delta = max (0, XFASTINT (root->height) - min_height);
f984d4fc
GM
3888 }
3889
3890 if (delta)
3891 {
3892 /* Save original window sizes and positions, if not already done. */
43b4a21f
GM
3893 if (!save_restore_orig_size (root, CHECK_ORIG_SIZES))
3894 save_restore_orig_size (root, SAVE_ORIG_SIZES);
f984d4fc
GM
3895
3896 /* Shrink other windows. */
3897 shrink_window_lowest_first (root, XFASTINT (root->height) - delta);
3898
3899 /* Grow the mini-window. */
3578db3c 3900 w->top = make_number (XFASTINT (root->top) + XFASTINT (root->height));
f984d4fc
GM
3901 w->height = make_number (XFASTINT (w->height) + delta);
3902 XSETFASTINT (w->last_modified, 0);
3903 XSETFASTINT (w->last_overlay_modified, 0);
3904
3905 adjust_glyphs (f);
3906 }
3907}
3908
3909
86c8e823
GM
3910/* Shrink mini-window W. If there is recorded info about window sizes
3911 before a call to grow_mini_window, restore recorded window sizes.
3912 Otherwise, if the mini-window is higher than 1 line, resize it to 1
3913 line. */
f984d4fc
GM
3914
3915void
3916shrink_mini_window (w)
3917 struct window *w;
3918{
3919 struct frame *f = XFRAME (w->frame);
3920 struct window *root = XWINDOW (FRAME_ROOT_WINDOW (f));
3921
43b4a21f 3922 if (save_restore_orig_size (root, CHECK_ORIG_SIZES))
f984d4fc 3923 {
43b4a21f 3924 save_restore_orig_size (root, RESTORE_ORIG_SIZES);
f984d4fc
GM
3925 adjust_glyphs (f);
3926 FRAME_WINDOW_SIZES_CHANGED (f) = 1;
3927 windows_or_buffers_changed = 1;
3928 }
86c8e823
GM
3929 else if (XFASTINT (w->height) > 1)
3930 {
0130fe1a
GM
3931 /* Distribute the additional lines of the mini-window
3932 among the other windows. */
86c8e823
GM
3933 Lisp_Object window;
3934 XSETWINDOW (window, w);
f95284d2 3935 enlarge_window (window, 1 - XFASTINT (w->height), 0, 0);
86c8e823 3936 }
f984d4fc
GM
3937}
3938
3939
3940\f
5500c422
GM
3941/* Mark window cursors off for all windows in the window tree rooted
3942 at W by setting their phys_cursor_on_p flag to zero. Called from
3943 xterm.c, e.g. when a frame is cleared and thereby all cursors on
3944 the frame are cleared. */
3945
3946void
3947mark_window_cursors_off (w)
3948 struct window *w;
3949{
3950 while (w)
3951 {
3952 if (!NILP (w->hchild))
3953 mark_window_cursors_off (XWINDOW (w->hchild));
3954 else if (!NILP (w->vchild))
3955 mark_window_cursors_off (XWINDOW (w->vchild));
3956 else
3957 w->phys_cursor_on_p = 0;
3958
3959 w = NILP (w->next) ? 0 : XWINDOW (w->next);
3960 }
3961}
3962
3963
e9c195b1 3964/* Return number of lines of text (not counting mode lines) in W. */
7ab12479
JB
3965
3966int
3967window_internal_height (w)
3968 struct window *w;
3969{
3970 int ht = XFASTINT (w->height);
3971
e9c195b1
GM
3972 if (!MINI_WINDOW_P (w))
3973 {
3974 if (!NILP (w->parent)
3975 || !NILP (w->vchild)
3976 || !NILP (w->hchild)
3977 || !NILP (w->next)
3978 || !NILP (w->prev)
3979 || WINDOW_WANTS_MODELINE_P (w))
3980 --ht;
7ab12479 3981
e9c195b1
GM
3982 if (WINDOW_WANTS_HEADER_LINE_P (w))
3983 --ht;
3984 }
7ab12479
JB
3985
3986 return ht;
3987}
3988
535e0b8e
JB
3989
3990/* Return the number of columns in W.
a3c87d4e 3991 Don't count columns occupied by scroll bars or the vertical bar
535e0b8e 3992 separating W from the sibling to its right. */
5500c422 3993
535e0b8e
JB
3994int
3995window_internal_width (w)
3996 struct window *w;
3997{
5500c422 3998 struct frame *f = XFRAME (WINDOW_FRAME (w));
535e0b8e
JB
3999 int width = XINT (w->width);
4000
a3c87d4e 4001 if (FRAME_HAS_VERTICAL_SCROLL_BARS (f))
5500c422
GM
4002 /* Scroll bars occupy a few columns. */
4003 width -= FRAME_SCROLL_BAR_COLS (f);
4004 else if (!WINDOW_RIGHTMOST_P (w) && !WINDOW_FULL_WIDTH_P (w))
4005 /* The column of `|' characters separating side-by-side windows
4006 occupies one column only. */
4007 width -= 1;
4008
4009 /* On window-systems, areas to the left and right of the window
81d189fd 4010 are used as fringes. */
5500c422 4011 if (FRAME_WINDOW_P (f))
81d189fd 4012 width -= FRAME_FRINGE_COLS (f);
111e5992
RS
4013
4014 return width;
535e0b8e
JB
4015}
4016
5500c422
GM
4017\f
4018/************************************************************************
4019 Window Scrolling
4020 ***********************************************************************/
535e0b8e 4021
5500c422 4022/* Scroll contents of window WINDOW up. If WHOLE is non-zero, scroll
d4e7cf01 4023 N screen-fulls, which is defined as the height of the window minus
5500c422
GM
4024 next_screen_context_lines. If WHOLE is zero, scroll up N lines
4025 instead. Negative values of N mean scroll down. NOERROR non-zero
4026 means don't signal an error if we try to move over BEGV or ZV,
4027 respectively. */
7ab12479 4028
101d1605
RS
4029static void
4030window_scroll (window, n, whole, noerror)
7ab12479
JB
4031 Lisp_Object window;
4032 int n;
101d1605 4033 int whole;
f8026fd8 4034 int noerror;
5500c422
GM
4035{
4036 /* If we must, use the pixel-based version which is much slower than
4037 the line-based one but can handle varying line heights. */
4038 if (FRAME_WINDOW_P (XFRAME (XWINDOW (window)->frame)))
4039 window_scroll_pixel_based (window, n, whole, noerror);
4040 else
4041 window_scroll_line_based (window, n, whole, noerror);
4042}
4043
4044
4045/* Implementation of window_scroll that works based on pixel line
4046 heights. See the comment of window_scroll for parameter
4047 descriptions. */
4048
4049static void
4050window_scroll_pixel_based (window, n, whole, noerror)
4051 Lisp_Object window;
4052 int n;
4053 int whole;
4054 int noerror;
4055{
4056 struct it it;
4057 struct window *w = XWINDOW (window);
4058 struct text_pos start;
4059 Lisp_Object tem;
4060 int this_scroll_margin;
4061 int preserve_y;
5cdb3cf3
MB
4062 /* True if we fiddled the window vscroll field without really scrolling. */
4063 int vscrolled = 0;
5500c422
GM
4064
4065 SET_TEXT_POS_FROM_MARKER (start, w->start);
4066
4067 /* If PT is not visible in WINDOW, move back one half of
288d4e06
GM
4068 the screen. Allow PT to be partially visible, otherwise
4069 something like (scroll-down 1) with PT in the line before
4070 the partially visible one would recenter. */
4071 tem = Fpos_visible_in_window_p (make_number (PT), window, Qt);
5500c422
GM
4072 if (NILP (tem))
4073 {
4074 /* Move backward half the height of the window. Performance note:
4075 vmotion used here is about 10% faster, but would give wrong
4076 results for variable height lines. */
4077 init_iterator (&it, w, PT, PT_BYTE, NULL, DEFAULT_FACE_ID);
4078 it.current_y = it.last_visible_y;
202379cf 4079 move_it_vertically (&it, - window_box_height (w) / 2);
5500c422
GM
4080
4081 /* The function move_iterator_vertically may move over more than
4082 the specified y-distance. If it->w is small, e.g. a
4083 mini-buffer window, we may end up in front of the window's
4084 display area. This is the case when Start displaying at the
4085 start of the line containing PT in this case. */
4086 if (it.current_y <= 0)
4087 {
4088 init_iterator (&it, w, PT, PT_BYTE, NULL, DEFAULT_FACE_ID);
4089 move_it_vertically (&it, 0);
4090 it.current_y = 0;
4091 }
4092
4093 start = it.current.pos;
4094 }
4095
4096 /* If scroll_preserve_screen_position is non-zero, we try to set
4097 point in the same window line as it is now, so get that line. */
4098 if (!NILP (Vscroll_preserve_screen_position))
4099 {
4100 start_display (&it, w, start);
4101 move_it_to (&it, PT, -1, -1, -1, MOVE_TO_POS);
4102 preserve_y = it.current_y;
4103 }
4104 else
4105 preserve_y = -1;
4106
4107 /* Move iterator it from start the specified distance forward or
4108 backward. The result is the new window start. */
4109 start_display (&it, w, start);
4110 if (whole)
4111 {
202379cf 4112 int screen_full = (window_box_height (w)
5500c422 4113 - next_screen_context_lines * CANON_Y_UNIT (it.f));
d4e7cf01 4114 int dy = n * screen_full;
d72340d4
GM
4115
4116 /* Note that move_it_vertically always moves the iterator to the
4117 start of a line. So, if the last line doesn't have a newline,
4118 we would end up at the start of the line ending at ZV. */
4119 if (dy <= 0)
4120 move_it_vertically_backward (&it, -dy);
4121 else if (dy > 0)
4122 move_it_to (&it, ZV, -1, it.current_y + dy, -1,
4123 MOVE_TO_POS | MOVE_TO_Y);
5500c422
GM
4124 }
4125 else
4126 move_it_by_lines (&it, n, 1);
4127
4128 /* End if we end up at ZV or BEGV. */
4129 if ((n > 0 && IT_CHARPOS (it) == ZV)
4130 || (n < 0 && IT_CHARPOS (it) == CHARPOS (start)))
4131 {
5cdb3cf3
MB
4132 if (IT_CHARPOS (it) == ZV)
4133 {
4134 if (it.current_y + it.max_ascent + it.max_descent
4135 > it.last_visible_y)
a74eca50
GM
4136 {
4137 /* The last line was only partially visible, make it fully
4138 visible. */
4139 w->vscroll = (it.last_visible_y
4140 - it.current_y + it.max_ascent + it.max_descent);
4141 adjust_glyphs (it.f);
4142 }
5cdb3cf3
MB
4143 else if (noerror)
4144 return;
4145 else
4146 Fsignal (Qend_of_buffer, Qnil);
4147 }
5500c422 4148 else
5cdb3cf3
MB
4149 {
4150 if (w->vscroll != 0)
4151 /* The first line was only partially visible, make it fully
4152 visible. */
4153 w->vscroll = 0;
4154 else if (noerror)
4155 return;
4156 else
4157 Fsignal (Qbeginning_of_buffer, Qnil);
4158 }
4159
4160 /* If control gets here, then we vscrolled. */
4161
4162 XBUFFER (w->buffer)->prevent_redisplay_optimizations_p = 1;
4163
4164 /* Don't try to change the window start below. */
4165 vscrolled = 1;
5500c422
GM
4166 }
4167
5cdb3cf3
MB
4168 if (! vscrolled)
4169 {
dad67609
RS
4170 int pos = IT_CHARPOS (it);
4171 int bytepos;
5cdb3cf3 4172 /* Set the window start, and set up the window for redisplay. */
dad67609 4173 set_marker_restricted (w->start, make_number (pos),
5cdb3cf3 4174 w->buffer);
dad67609
RS
4175 bytepos = XMARKER (w->start)->bytepos;
4176 w->start_at_line_beg = ((pos == BEGV || FETCH_BYTE (bytepos - 1) == '\n')
4177 ? Qt : Qnil);
5cdb3cf3
MB
4178 w->update_mode_line = Qt;
4179 XSETFASTINT (w->last_modified, 0);
4180 XSETFASTINT (w->last_overlay_modified, 0);
4181 /* Set force_start so that redisplay_window will run the
4182 window-scroll-functions. */
4183 w->force_start = Qt;
4184 }
5500c422
GM
4185
4186 it.current_y = it.vpos = 0;
4187
4188 /* Preserve the screen position if we must. */
4189 if (preserve_y >= 0)
4190 {
4191 move_it_to (&it, -1, -1, preserve_y, -1, MOVE_TO_Y);
4192 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4193 }
4194 else
4195 {
4196 /* Move PT out of scroll margins. */
4197 this_scroll_margin = max (0, scroll_margin);
4198 this_scroll_margin = min (this_scroll_margin, XFASTINT (w->height) / 4);
4199 this_scroll_margin *= CANON_Y_UNIT (it.f);
4200
4201 if (n > 0)
4202 {
4203 /* We moved the window start towards ZV, so PT may be now
4204 in the scroll margin at the top. */
4205 move_it_to (&it, PT, -1, -1, -1, MOVE_TO_POS);
4206 while (it.current_y < this_scroll_margin)
e9b2c961
RS
4207 {
4208 int prev = it.current_y;
4209 move_it_by_lines (&it, 1, 1);
4210 if (prev == it.current_y)
4211 break;
4212 }
5500c422
GM
4213 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4214 }
4215 else if (n < 0)
4216 {
5cdb3cf3
MB
4217 int charpos, bytepos;
4218
5500c422
GM
4219 /* We moved the window start towards BEGV, so PT may be now
4220 in the scroll margin at the bottom. */
4221 move_it_to (&it, PT, -1,
4222 it.last_visible_y - this_scroll_margin - 1, -1,
4223 MOVE_TO_POS | MOVE_TO_Y);
5cdb3cf3
MB
4224
4225 /* Save our position, in case it's correct. */
4226 charpos = IT_CHARPOS (it);
4227 bytepos = IT_BYTEPOS (it);
5500c422 4228
5cdb3cf3
MB
4229 /* See if point is on a partially visible line at the end. */
4230 move_it_by_lines (&it, 1, 1);
4231 if (it.current_y > it.last_visible_y)
4232 /* The last line was only partially visible, so back up two
4233 lines to make sure we're on a fully visible line. */
4234 {
4235 move_it_by_lines (&it, -2, 0);
4236 SET_PT_BOTH (IT_CHARPOS (it), IT_BYTEPOS (it));
4237 }
4238 else
4239 /* No, the position we saved is OK, so use it. */
4240 SET_PT_BOTH (charpos, bytepos);
5500c422
GM
4241 }
4242 }
4243}
4244
4245
4246/* Implementation of window_scroll that works based on screen lines.
4247 See the comment of window_scroll for parameter descriptions. */
4248
4249static void
4250window_scroll_line_based (window, n, whole, noerror)
4251 Lisp_Object window;
4252 int n;
4253 int whole;
4254 int noerror;
7ab12479
JB
4255{
4256 register struct window *w = XWINDOW (window);
5500c422 4257 register int opoint = PT, opoint_byte = PT_BYTE;
b73ea88e 4258 register int pos, pos_byte;
7ab12479
JB
4259 register int ht = window_internal_height (w);
4260 register Lisp_Object tem;
4261 int lose;
5500c422 4262 Lisp_Object bolp;
345d45b2 4263 int startpos;
101d1605
RS
4264 struct position posit;
4265 int original_vpos;
4266
d4e7cf01
GM
4267 /* If scrolling screen-fulls, compute the number of lines to
4268 scroll from the window's height. */
4269 if (whole)
4270 n *= max (1, ht - next_screen_context_lines);
4271
101d1605
RS
4272 startpos = marker_position (w->start);
4273
4274 posit = *compute_motion (startpos, 0, 0, 0,
4275 PT, ht, 0,
4276 window_internal_width (w), XINT (w->hscroll),
4277 0, w);
4278 original_vpos = posit.vpos;
0a1f771a 4279
d834a2e9 4280 XSETFASTINT (tem, PT);
6ffdb539 4281 tem = Fpos_visible_in_window_p (tem, window, Qnil);
7ab12479 4282
265a9e55 4283 if (NILP (tem))
7ab12479 4284 {
cd2be1dd 4285 Fvertical_motion (make_number (- (ht / 2)), window);
345d45b2 4286 startpos = PT;
7ab12479
JB
4287 }
4288
345d45b2 4289 SET_PT (startpos);
5ce7b543 4290 lose = n < 0 && PT == BEGV;
540b6aa0 4291 Fvertical_motion (make_number (n), window);
5ce7b543 4292 pos = PT;
b73ea88e 4293 pos_byte = PT_BYTE;
7ab12479 4294 bolp = Fbolp ();
b73ea88e 4295 SET_PT_BOTH (opoint, opoint_byte);
7ab12479
JB
4296
4297 if (lose)
f8026fd8
JB
4298 {
4299 if (noerror)
4300 return;
4301 else
4302 Fsignal (Qbeginning_of_buffer, Qnil);
4303 }
7ab12479
JB
4304
4305 if (pos < ZV)
7ab12479 4306 {
0c7da84e
RS
4307 int this_scroll_margin = scroll_margin;
4308
4309 /* Don't use a scroll margin that is negative or too large. */
4310 if (this_scroll_margin < 0)
4311 this_scroll_margin = 0;
4312
4313 if (XINT (w->height) < 4 * scroll_margin)
4314 this_scroll_margin = XINT (w->height) / 4;
4315
b73ea88e 4316 set_marker_restricted_both (w->start, w->buffer, pos, pos_byte);
7ab12479
JB
4317 w->start_at_line_beg = bolp;
4318 w->update_mode_line = Qt;
d834a2e9 4319 XSETFASTINT (w->last_modified, 0);
3cd21523 4320 XSETFASTINT (w->last_overlay_modified, 0);
345d45b2
RS
4321 /* Set force_start so that redisplay_window will run
4322 the window-scroll-functions. */
4323 w->force_start = Qt;
0c7da84e 4324
5500c422 4325 if (whole && !NILP (Vscroll_preserve_screen_position))
0c7da84e 4326 {
b73ea88e 4327 SET_PT_BOTH (pos, pos_byte);
101d1605 4328 Fvertical_motion (make_number (original_vpos), window);
0c7da84e 4329 }
101d1605
RS
4330 /* If we scrolled forward, put point enough lines down
4331 that it is outside the scroll margin. */
4332 else if (n > 0)
0c7da84e 4333 {
101d1605
RS
4334 int top_margin;
4335
4336 if (this_scroll_margin > 0)
4337 {
b73ea88e 4338 SET_PT_BOTH (pos, pos_byte);
101d1605
RS
4339 Fvertical_motion (make_number (this_scroll_margin), window);
4340 top_margin = PT;
4341 }
4342 else
4343 top_margin = pos;
4344
4345 if (top_margin <= opoint)
b73ea88e 4346 SET_PT_BOTH (opoint, opoint_byte);
5500c422 4347 else if (!NILP (Vscroll_preserve_screen_position))
101d1605 4348 {
b73ea88e 4349 SET_PT_BOTH (pos, pos_byte);
101d1605
RS
4350 Fvertical_motion (make_number (original_vpos), window);
4351 }
9317a85d 4352 else
335406fc 4353 SET_PT (top_margin);
0c7da84e 4354 }
101d1605 4355 else if (n < 0)
7ab12479 4356 {
101d1605
RS
4357 int bottom_margin;
4358
0c7da84e
RS
4359 /* If we scrolled backward, put point near the end of the window
4360 but not within the scroll margin. */
b73ea88e 4361 SET_PT_BOTH (pos, pos_byte);
0c7da84e 4362 tem = Fvertical_motion (make_number (ht - this_scroll_margin), window);
101d1605
RS
4363 if (XFASTINT (tem) == ht - this_scroll_margin)
4364 bottom_margin = PT;
4365 else
4366 bottom_margin = PT + 1;
4367
4368 if (bottom_margin > opoint)
b73ea88e 4369 SET_PT_BOTH (opoint, opoint_byte);
7ab12479 4370 else
101d1605 4371 {
5500c422 4372 if (!NILP (Vscroll_preserve_screen_position))
9317a85d 4373 {
b73ea88e 4374 SET_PT_BOTH (pos, pos_byte);
9317a85d
RS
4375 Fvertical_motion (make_number (original_vpos), window);
4376 }
4377 else
4378 Fvertical_motion (make_number (-1), window);
101d1605 4379 }
7ab12479
JB
4380 }
4381 }
4382 else
f8026fd8
JB
4383 {
4384 if (noerror)
4385 return;
4386 else
4387 Fsignal (Qend_of_buffer, Qnil);
4388 }
7ab12479 4389}
5500c422
GM
4390
4391
4392/* Scroll selected_window up or down. If N is nil, scroll a
4393 screen-full which is defined as the height of the window minus
4394 next_screen_context_lines. If N is the symbol `-', scroll.
4395 DIRECTION may be 1 meaning to scroll down, or -1 meaning to scroll
4396 up. This is the guts of Fscroll_up and Fscroll_down. */
7ab12479
JB
4397
4398static void
4399scroll_command (n, direction)
5500c422 4400 Lisp_Object n;
7ab12479
JB
4401 int direction;
4402{
331379bf 4403 int count = SPECPDL_INDEX ();
7ab12479 4404
5500c422
GM
4405 xassert (abs (direction) == 1);
4406
4407 /* If selected window's buffer isn't current, make it current for
4408 the moment. But don't screw up if window_scroll gets an error. */
7ab12479 4409 if (XBUFFER (XWINDOW (selected_window)->buffer) != current_buffer)
95605e15
JB
4410 {
4411 record_unwind_protect (save_excursion_restore, save_excursion_save ());
4412 Fset_buffer (XWINDOW (selected_window)->buffer);
5500c422
GM
4413
4414 /* Make redisplay consider other windows than just selected_window. */
4415 ++windows_or_buffers_changed;
95605e15 4416 }
7ab12479 4417
265a9e55 4418 if (NILP (n))
d4e7cf01 4419 window_scroll (selected_window, direction, 1, 0);
7ab12479 4420 else if (EQ (n, Qminus))
d4e7cf01 4421 window_scroll (selected_window, -direction, 1, 0);
7ab12479
JB
4422 else
4423 {
4424 n = Fprefix_numeric_value (n);
101d1605 4425 window_scroll (selected_window, XINT (n) * direction, 0, 0);
7ab12479 4426 }
95605e15
JB
4427
4428 unbind_to (count, Qnil);
7ab12479
JB
4429}
4430
4431DEFUN ("scroll-up", Fscroll_up, Sscroll_up, 0, 1, "P",
fdb82f93
PJ
4432 doc: /* Scroll text of current window upward ARG lines; or near full screen if no ARG.
4433A near full screen is `next-screen-context-lines' less than a full screen.
4434Negative ARG means scroll downward.
4435If ARG is the atom `-', scroll downward by nearly full screen.
4436When calling from a program, supply as argument a number, nil, or `-'. */)
4437 (arg)
413430c5 4438 Lisp_Object arg;
7ab12479 4439{
413430c5 4440 scroll_command (arg, 1);
7ab12479
JB
4441 return Qnil;
4442}
4443
4444DEFUN ("scroll-down", Fscroll_down, Sscroll_down, 0, 1, "P",
fdb82f93
PJ
4445 doc: /* Scroll text of current window down ARG lines; or near full screen if no ARG.
4446A near full screen is `next-screen-context-lines' less than a full screen.
4447Negative ARG means scroll upward.
4448If ARG is the atom `-', scroll upward by nearly full screen.
4449When calling from a program, supply as argument a number, nil, or `-'. */)
4450 (arg)
413430c5 4451 Lisp_Object arg;
7ab12479 4452{
413430c5 4453 scroll_command (arg, -1);
7ab12479
JB
4454 return Qnil;
4455}
ccd0664b
RS
4456\f
4457DEFUN ("other-window-for-scrolling", Fother_window_for_scrolling, Sother_window_for_scrolling, 0, 0, 0,
fdb82f93
PJ
4458 doc: /* Return the other window for \"other window scroll\" commands.
4459If in the minibuffer, `minibuffer-scroll-window' if non-nil
4460specifies the window.
4461If `other-window-scroll-buffer' is non-nil, a window
4462showing that buffer is used. */)
4463 ()
7ab12479 4464{
ccd0664b 4465 Lisp_Object window;
7ab12479
JB
4466
4467 if (MINI_WINDOW_P (XWINDOW (selected_window))
265a9e55 4468 && !NILP (Vminibuf_scroll_window))
7ab12479
JB
4469 window = Vminibuf_scroll_window;
4470 /* If buffer is specified, scroll that buffer. */
265a9e55 4471 else if (!NILP (Vother_window_scroll_buffer))
7ab12479
JB
4472 {
4473 window = Fget_buffer_window (Vother_window_scroll_buffer, Qnil);
265a9e55 4474 if (NILP (window))
53f76081 4475 window = Fdisplay_buffer (Vother_window_scroll_buffer, Qt, Qnil);
7ab12479
JB
4476 }
4477 else
dbc4e1c1
JB
4478 {
4479 /* Nothing specified; look for a neighboring window on the same
4480 frame. */
4481 window = Fnext_window (selected_window, Qnil, Qnil);
4482
4483 if (EQ (window, selected_window))
4484 /* That didn't get us anywhere; look for a window on another
4485 visible frame. */
4486 do
4487 window = Fnext_window (window, Qnil, Qt);
4488 while (! FRAME_VISIBLE_P (XFRAME (WINDOW_FRAME (XWINDOW (window))))
4489 && ! EQ (window, selected_window));
4490 }
4491
b7826503 4492 CHECK_LIVE_WINDOW (window);
7ab12479
JB
4493
4494 if (EQ (window, selected_window))
4495 error ("There is no other window");
4496
ccd0664b
RS
4497 return window;
4498}
4499
4500DEFUN ("scroll-other-window", Fscroll_other_window, Sscroll_other_window, 0, 1, "P",
fdb82f93
PJ
4501 doc: /* Scroll next window upward ARG lines; or near full screen if no ARG.
4502A near full screen is `next-screen-context-lines' less than a full screen.
4503The next window is the one below the current one; or the one at the top
4504if the current one is at the bottom. Negative ARG means scroll downward.
4505If ARG is the atom `-', scroll downward by nearly full screen.
4506When calling from a program, supply as argument a number, nil, or `-'.
4507
4508If in the minibuffer, `minibuffer-scroll-window' if non-nil
4509specifies the window to scroll.
4510If `other-window-scroll-buffer' is non-nil, scroll the window
4511showing that buffer, popping the buffer up if necessary. */)
4512 (arg)
d4e7cf01 4513 Lisp_Object arg;
ccd0664b 4514{
d4e7cf01
GM
4515 Lisp_Object window;
4516 struct window *w;
331379bf 4517 int count = SPECPDL_INDEX ();
ccd0664b
RS
4518
4519 window = Fother_window_for_scrolling ();
7ab12479 4520 w = XWINDOW (window);
7ab12479
JB
4521
4522 /* Don't screw up if window_scroll gets an error. */
4523 record_unwind_protect (save_excursion_restore, save_excursion_save ());
5500c422 4524 ++windows_or_buffers_changed;
7ab12479
JB
4525
4526 Fset_buffer (w->buffer);
4527 SET_PT (marker_position (w->pointm));
4528
413430c5 4529 if (NILP (arg))
d4e7cf01 4530 window_scroll (window, 1, 1, 1);
413430c5 4531 else if (EQ (arg, Qminus))
d4e7cf01 4532 window_scroll (window, -1, 1, 1);
7ab12479
JB
4533 else
4534 {
413430c5
EN
4535 if (CONSP (arg))
4536 arg = Fcar (arg);
b7826503 4537 CHECK_NUMBER (arg);
101d1605 4538 window_scroll (window, XINT (arg), 0, 1);
7ab12479
JB
4539 }
4540
b73ea88e 4541 set_marker_both (w->pointm, Qnil, PT, PT_BYTE);
f4e7b2c2 4542 unbind_to (count, Qnil);
7ab12479
JB
4543
4544 return Qnil;
4545}
4546\f
644b477c 4547DEFUN ("scroll-left", Fscroll_left, Sscroll_left, 0, 1, "P",
fdb82f93 4548 doc: /* Scroll selected window display ARG columns left.
2fe41216
EZ
4549Default for ARG is window width minus 2.
4550Value is the total amount of leftward horizontal scrolling in
4551effect after the change.
4552If `automatic-hscrolling' is non-nil, the argument ARG modifies
4553a lower bound for automatic scrolling, i.e. automatic scrolling
4554will not scroll a window to a column less than the value returned
4555by this function. */)
fdb82f93 4556 (arg)
7ab12479
JB
4557 register Lisp_Object arg;
4558{
c67fa410
GM
4559 Lisp_Object result;
4560 int hscroll;
4561 struct window *w = XWINDOW (selected_window);
4562
265a9e55 4563 if (NILP (arg))
c67fa410 4564 XSETFASTINT (arg, window_internal_width (w) - 2);
7ab12479
JB
4565 else
4566 arg = Fprefix_numeric_value (arg);
4567
c67fa410
GM
4568 hscroll = XINT (w->hscroll) + XINT (arg);
4569 result = Fset_window_hscroll (selected_window, make_number (hscroll));
4570
f5686fbd 4571 if (interactive_p (0))
c67fa410
GM
4572 w->min_hscroll = w->hscroll;
4573
4574 return result;
7ab12479
JB
4575}
4576
644b477c 4577DEFUN ("scroll-right", Fscroll_right, Sscroll_right, 0, 1, "P",
fdb82f93 4578 doc: /* Scroll selected window display ARG columns right.
2fe41216
EZ
4579Default for ARG is window width minus 2.
4580Value is the total amount of leftward horizontal scrolling in
4581effect after the change.
4582If `automatic-hscrolling' is non-nil, the argument ARG modifies
4583a lower bound for automatic scrolling, i.e. automatic scrolling
4584will not scroll a window to a column less than the value returned
4585by this function. */)
fdb82f93 4586 (arg)
7ab12479
JB
4587 register Lisp_Object arg;
4588{
c67fa410
GM
4589 Lisp_Object result;
4590 int hscroll;
4591 struct window *w = XWINDOW (selected_window);
4592
265a9e55 4593 if (NILP (arg))
c67fa410 4594 XSETFASTINT (arg, window_internal_width (w) - 2);
7ab12479
JB
4595 else
4596 arg = Fprefix_numeric_value (arg);
4597
c67fa410
GM
4598 hscroll = XINT (w->hscroll) - XINT (arg);
4599 result = Fset_window_hscroll (selected_window, make_number (hscroll));
4600
f5686fbd 4601 if (interactive_p (0))
c67fa410
GM
4602 w->min_hscroll = w->hscroll;
4603
4604 return result;
7ab12479
JB
4605}
4606
fa832261
KS
4607DEFUN ("minibuffer-selected-window", Fminibuffer_selected_window, Sminibuffer_selected_window, 0, 0, 0,
4608 doc: /* Return the window which was selected when entering the minibuffer.
4609Returns nil, if current window is not a minibuffer window. */)
4610 ()
4611{
4612 if (minibuf_level > 0
4613 && MINI_WINDOW_P (XWINDOW (selected_window))
4614 && !NILP (minibuf_selected_window)
4615 && WINDOW_LIVE_P (minibuf_selected_window))
4616 return minibuf_selected_window;
4617
4618 return Qnil;
4619}
4620
12c8b416
GM
4621/* Value is the number of lines actually displayed in window W,
4622 as opposed to its height. */
4623
4624static int
4625displayed_window_lines (w)
4626 struct window *w;
4627{
4628 struct it it;
4629 struct text_pos start;
4630 int height = window_box_height (w);
4631 struct buffer *old_buffer;
4632 int bottom_y;
4633
4634 if (XBUFFER (w->buffer) != current_buffer)
4635 {
4636 old_buffer = current_buffer;
4637 set_buffer_internal (XBUFFER (w->buffer));
4638 }
4639 else
4640 old_buffer = NULL;
4641
521b203e
GM
4642 /* In case W->start is out of the accessible range, do something
4643 reasonable. This happens in Info mode when Info-scroll-down
4644 calls (recenter -1) while W->start is 1. */
4645 if (XMARKER (w->start)->charpos < BEGV)
4646 SET_TEXT_POS (start, BEGV, BEGV_BYTE);
4647 else if (XMARKER (w->start)->charpos > ZV)
4648 SET_TEXT_POS (start, ZV, ZV_BYTE);
4649 else
4650 SET_TEXT_POS_FROM_MARKER (start, w->start);
4651
12c8b416
GM
4652 start_display (&it, w, start);
4653 move_it_vertically (&it, height);
c8bc6f65 4654 bottom_y = line_bottom_y (&it);
12c8b416 4655
1de65f51
RS
4656 /* rms: On a non-window display,
4657 the value of it.vpos at the bottom of the screen
4658 seems to be 1 larger than window_box_height (w).
4659 This kludge fixes a bug whereby (move-to-window-line -1)
4660 when ZV is on the last screen line
4661 moves to the previous screen line instead of the last one. */
4662 if (! FRAME_WINDOW_P (XFRAME (w->frame)))
4663 height++;
4664
12c8b416
GM
4665 /* Add in empty lines at the bottom of the window. */
4666 if (bottom_y < height)
4667 {
c8bc6f65
GM
4668 int uy = CANON_Y_UNIT (it.f);
4669 it.vpos += (height - bottom_y + uy - 1) / uy;
12c8b416
GM
4670 }
4671
c8bc6f65
GM
4672 if (old_buffer)
4673 set_buffer_internal (old_buffer);
4674
12c8b416
GM
4675 return it.vpos;
4676}
4677
4678
7ab12479 4679DEFUN ("recenter", Frecenter, Srecenter, 0, 1, "P",
fdb82f93
PJ
4680 doc: /* Center point in window and redisplay frame.
4681With prefix argument ARG, recenter putting point on screen line ARG
4682relative to the current window. If ARG is negative, it counts up from the
4683bottom of the window. (ARG should be less than the height of the window.)
4684
4685If ARG is omitted or nil, erase the entire frame and then
4686redraw with point in the center of the current window.
4687Just C-u as prefix means put point in the center of the window
4688and redisplay normally--don't erase and redraw the frame. */)
4689 (arg)
413430c5 4690 register Lisp_Object arg;
7ab12479 4691{
6df47b59 4692 struct window *w = XWINDOW (selected_window);
478292ed
RS
4693 struct buffer *buf = XBUFFER (w->buffer);
4694 struct buffer *obuf = current_buffer;
6df47b59
GM
4695 int center_p = 0;
4696 int charpos, bytepos;
7ab12479 4697
0fa5d25b
RS
4698 /* If redisplay is suppressed due to an error, try again. */
4699 obuf->display_error_modiff = 0;
4700
413430c5 4701 if (NILP (arg))
7ab12479 4702 {
f02d6d5c
KH
4703 int i;
4704
4705 /* Invalidate pixel data calculated for all compositions. */
4706 for (i = 0; i < n_compositions; i++)
4707 composition_table[i]->font = NULL;
7ab12479 4708
527b6458 4709 Fredraw_frame (w->frame);
44fa5b1e 4710 SET_FRAME_GARBAGED (XFRAME (WINDOW_FRAME (w)));
6df47b59 4711 center_p = 1;
7ab12479 4712 }
413430c5 4713 else if (CONSP (arg)) /* Just C-u. */
6df47b59 4714 center_p = 1;
7ab12479
JB
4715 else
4716 {
413430c5 4717 arg = Fprefix_numeric_value (arg);
b7826503 4718 CHECK_NUMBER (arg);
7ab12479
JB
4719 }
4720
478292ed 4721 set_buffer_internal (buf);
7ab12479 4722
521b203e 4723 /* Handle centering on a graphical frame specially. Such frames can
6df47b59
GM
4724 have variable-height lines and centering point on the basis of
4725 line counts would lead to strange effects. */
521b203e 4726 if (FRAME_WINDOW_P (XFRAME (w->frame)))
6df47b59 4727 {
6df47b59
GM
4728 if (center_p)
4729 {
521b203e
GM
4730 struct it it;
4731 struct text_pos pt;
4732
4733 SET_TEXT_POS (pt, PT, PT_BYTE);
4734 start_display (&it, w, pt);
202379cf 4735 move_it_vertically (&it, - window_box_height (w) / 2);
521b203e
GM
4736 charpos = IT_CHARPOS (it);
4737 bytepos = IT_BYTEPOS (it);
6df47b59
GM
4738 }
4739 else if (XINT (arg) < 0)
4740 {
521b203e
GM
4741 struct it it;
4742 struct text_pos pt;
d466fa4d 4743 int y0, y1, h, nlines;
521b203e
GM
4744
4745 SET_TEXT_POS (pt, PT, PT_BYTE);
4746 start_display (&it, w, pt);
4747 y0 = it.current_y;
4748
d466fa4d 4749 /* The amount of pixels we have to move back is the window
521b203e
GM
4750 height minus what's displayed in the line containing PT,
4751 and the lines below. */
d466fa4d
GM
4752 nlines = - XINT (arg) - 1;
4753 move_it_by_lines (&it, nlines, 1);
4754
201c831a 4755 y1 = line_bottom_y (&it);
d466fa4d
GM
4756
4757 /* If we can't move down NLINES lines because we hit
4758 the end of the buffer, count in some empty lines. */
4759 if (it.vpos < nlines)
4760 y1 += (nlines - it.vpos) * CANON_Y_UNIT (it.f);
4761
201c831a
GM
4762 h = window_box_height (w) - (y1 - y0);
4763
521b203e 4764 start_display (&it, w, pt);
201c831a 4765 move_it_vertically (&it, - h);
521b203e
GM
4766 charpos = IT_CHARPOS (it);
4767 bytepos = IT_BYTEPOS (it);
6df47b59 4768 }
521b203e
GM
4769 else
4770 {
4771 struct position pos;
4772 pos = *vmotion (PT, - XINT (arg), w);
4773 charpos = pos.bufpos;
4774 bytepos = pos.bytepos;
4775 }
4776 }
4777 else
4778 {
4779 struct position pos;
4780 int ht = window_internal_height (w);
4781
4782 if (center_p)
4783 arg = make_number (ht / 2);
4784 else if (XINT (arg) < 0)
4785 arg = make_number (XINT (arg) + ht);
6df47b59
GM
4786
4787 pos = *vmotion (PT, - XINT (arg), w);
4788 charpos = pos.bufpos;
4789 bytepos = pos.bytepos;
4790 }
4791
4792 /* Set the new window start. */
4793 set_marker_both (w->start, w->buffer, charpos, bytepos);
2f3cad6c 4794 w->window_end_valid = Qnil;
95605b1b
RS
4795
4796 w->optional_new_start = Qt;
4797
6df47b59
GM
4798 if (bytepos == BEGV_BYTE || FETCH_BYTE (bytepos - 1) == '\n')
4799 w->start_at_line_beg = Qt;
4800 else
4801 w->start_at_line_beg = Qnil;
4802
478292ed 4803 set_buffer_internal (obuf);
7ab12479
JB
4804 return Qnil;
4805}
b7617575
GM
4806
4807
81fe0836 4808DEFUN ("window-text-height", Fwindow_text_height, Swindow_text_height,
fdb82f93
PJ
4809 0, 1, 0,
4810 doc: /* Return the height in lines of the text display area of WINDOW.
4811This doesn't include the mode-line (or header-line if any) or any
4812partial-height lines in the text display area. */)
4813 (window)
81fe0836
MB
4814 Lisp_Object window;
4815{
4816 struct window *w = decode_window (window);
4817 int pixel_height = window_box_height (w);
4818 int line_height = pixel_height / CANON_Y_UNIT (XFRAME (w->frame));
4819 return make_number (line_height);
4820}
4821
4822
7ab12479
JB
4823\f
4824DEFUN ("move-to-window-line", Fmove_to_window_line, Smove_to_window_line,
fdb82f93
PJ
4825 1, 1, "P",
4826 doc: /* Position point relative to window.
4827With no argument, position point at center of window.
4828An argument specifies vertical position within the window;
4829zero means top of window, negative means relative to bottom of window. */)
4830 (arg)
b7617575 4831 Lisp_Object arg;
7ab12479 4832{
b7617575
GM
4833 struct window *w = XWINDOW (selected_window);
4834 int lines, start;
540b6aa0 4835 Lisp_Object window;
7ab12479 4836
b7617575 4837 window = selected_window;
7ab12479
JB
4838 start = marker_position (w->start);
4839 if (start < BEGV || start > ZV)
4840 {
b7617575 4841 int height = window_internal_height (w);
cd2be1dd 4842 Fvertical_motion (make_number (- (height / 2)), window);
b73ea88e 4843 set_marker_both (w->start, w->buffer, PT, PT_BYTE);
7ab12479
JB
4844 w->start_at_line_beg = Fbolp ();
4845 w->force_start = Qt;
4846 }
4847 else
b73ea88e 4848 Fgoto_char (w->start);
7ab12479 4849
b7617575
GM
4850 lines = displayed_window_lines (w);
4851 if (NILP (arg))
4852 XSETFASTINT (arg, lines / 2);
4853 else
4854 {
4855 arg = Fprefix_numeric_value (arg);
4856 if (XINT (arg) < 0)
4857 XSETINT (arg, XINT (arg) + lines);
4858 }
4859
c8bc6f65 4860 /* Skip past a partially visible first line. */
163784df 4861 if (w->vscroll)
163784df
MB
4862 XSETINT (arg, XINT (arg) + 1);
4863
540b6aa0 4864 return Fvertical_motion (arg, window);
7ab12479 4865}
5500c422
GM
4866
4867
7ab12479 4868\f
5500c422
GM
4869/***********************************************************************
4870 Window Configuration
4871 ***********************************************************************/
4872
7ab12479
JB
4873struct save_window_data
4874 {
f5ccc0cc 4875 EMACS_INT size_from_Lisp_Vector_struct;
7ab12479 4876 struct Lisp_Vector *next_from_Lisp_Vector_struct;
8f6ea2e9 4877 Lisp_Object frame_width, frame_height, frame_menu_bar_lines;
9ea173e8 4878 Lisp_Object frame_tool_bar_lines;
bdc727bf 4879 Lisp_Object selected_frame;
7ab12479
JB
4880 Lisp_Object current_window;
4881 Lisp_Object current_buffer;
4882 Lisp_Object minibuf_scroll_window;
3f49fddc 4883 Lisp_Object minibuf_selected_window;
7ab12479 4884 Lisp_Object root_window;
bdc727bf 4885 Lisp_Object focus_frame;
756b6edc
RS
4886 /* Record the values of window-min-width and window-min-height
4887 so that window sizes remain consistent with them. */
4888 Lisp_Object min_width, min_height;
cbff28e8
RS
4889 /* A vector, each of whose elements is a struct saved_window
4890 for one window. */
7ab12479
JB
4891 Lisp_Object saved_windows;
4892 };
ff06df24 4893
cbff28e8 4894/* This is saved as a Lisp_Vector */
7ab12479 4895struct saved_window
ea68264b
GM
4896{
4897 /* these first two must agree with struct Lisp_Vector in lisp.h */
4898 EMACS_INT size_from_Lisp_Vector_struct;
4899 struct Lisp_Vector *next_from_Lisp_Vector_struct;
7ab12479 4900
ea68264b
GM
4901 Lisp_Object window;
4902 Lisp_Object buffer, start, pointm, mark;
4903 Lisp_Object left, top, width, height, hscroll, min_hscroll;
4904 Lisp_Object parent, prev;
4905 Lisp_Object start_at_line_beg;
4906 Lisp_Object display_table;
4907 Lisp_Object orig_top, orig_height;
4908};
4909
4910#define SAVED_WINDOW_VECTOR_SIZE 17 /* Arg to Fmake_vector */
7ab12479
JB
4911
4912#define SAVED_WINDOW_N(swv,n) \
4913 ((struct saved_window *) (XVECTOR ((swv)->contents[(n)])))
4914
4915DEFUN ("window-configuration-p", Fwindow_configuration_p, Swindow_configuration_p, 1, 1, 0,
fdb82f93
PJ
4916 doc: /* Return t if OBJECT is a window-configuration object. */)
4917 (object)
413430c5 4918 Lisp_Object object;
7ab12479 4919{
413430c5 4920 if (WINDOW_CONFIGURATIONP (object))
7ab12479
JB
4921 return Qt;
4922 return Qnil;
4923}
4924
3f8ab7bd 4925DEFUN ("window-configuration-frame", Fwindow_configuration_frame, Swindow_configuration_frame, 1, 1, 0,
fdb82f93
PJ
4926 doc: /* Return the frame that CONFIG, a window-configuration object, is about. */)
4927 (config)
3f8ab7bd
RS
4928 Lisp_Object config;
4929{
4930 register struct save_window_data *data;
4931 struct Lisp_Vector *saved_windows;
4932
4933 if (! WINDOW_CONFIGURATIONP (config))
4934 wrong_type_argument (Qwindow_configuration_p, config);
4935
4936 data = (struct save_window_data *) XVECTOR (config);
4937 saved_windows = XVECTOR (data->saved_windows);
4938 return XWINDOW (SAVED_WINDOW_N (saved_windows, 0)->window)->frame;
4939}
4940
d5b2799e 4941DEFUN ("set-window-configuration", Fset_window_configuration,
fdb82f93
PJ
4942 Sset_window_configuration, 1, 1, 0,
4943 doc: /* Set the configuration of windows and buffers as specified by CONFIGURATION.
4944CONFIGURATION must be a value previously returned
4945by `current-window-configuration' (which see).
4946If CONFIGURATION was made from a frame that is now deleted,
4947only frame-independent values can be restored. In this case,
4948the return value is nil. Otherwise the value is t. */)
4949 (configuration)
2f83aebe 4950 Lisp_Object configuration;
7ab12479 4951{
7ab12479
JB
4952 register struct save_window_data *data;
4953 struct Lisp_Vector *saved_windows;
7ab12479 4954 Lisp_Object new_current_buffer;
fd482be5 4955 Lisp_Object frame;
44fa5b1e 4956 FRAME_PTR f;
72695e47 4957 int old_point = -1;
7ab12479 4958
017b2bad 4959 while (!WINDOW_CONFIGURATIONP (configuration))
3f8ab7bd 4960 wrong_type_argument (Qwindow_configuration_p, configuration);
7ab12479 4961
2f83aebe 4962 data = (struct save_window_data *) XVECTOR (configuration);
7ab12479
JB
4963 saved_windows = XVECTOR (data->saved_windows);
4964
7ab12479 4965 new_current_buffer = data->current_buffer;
265a9e55 4966 if (NILP (XBUFFER (new_current_buffer)->name))
7ab12479 4967 new_current_buffer = Qnil;
72695e47
RS
4968 else
4969 old_point = BUF_PT (XBUFFER (new_current_buffer));
7ab12479 4970
fd482be5
JB
4971 frame = XWINDOW (SAVED_WINDOW_N (saved_windows, 0)->window)->frame;
4972 f = XFRAME (frame);
aa35ba9e 4973
fd482be5
JB
4974 /* If f is a dead frame, don't bother rebuilding its window tree.
4975 However, there is other stuff we should still try to do below. */
4976 if (FRAME_LIVE_P (f))
7ab12479 4977 {
fd482be5
JB
4978 register struct window *w;
4979 register struct saved_window *p;
5500c422
GM
4980 struct window *root_window;
4981 struct window **leaf_windows;
4982 int n_leaf_windows;
c4280705 4983 int k, i, n;
fd482be5
JB
4984
4985 /* If the frame has been resized since this window configuration was
4986 made, we change the frame to the size specified in the
4987 configuration, restore the configuration, and then resize it
4988 back. We keep track of the prevailing height in these variables. */
4989 int previous_frame_height = FRAME_HEIGHT (f);
4990 int previous_frame_width = FRAME_WIDTH (f);
8f6ea2e9 4991 int previous_frame_menu_bar_lines = FRAME_MENU_BAR_LINES (f);
9ea173e8 4992 int previous_frame_tool_bar_lines = FRAME_TOOL_BAR_LINES (f);
fd482be5 4993
d2b35234
RS
4994 /* The mouse highlighting code could get screwed up
4995 if it runs during this. */
4996 BLOCK_INPUT;
4997
fd482be5
JB
4998 if (XFASTINT (data->frame_height) != previous_frame_height
4999 || XFASTINT (data->frame_width) != previous_frame_width)
f8ad443a 5000 change_frame_size (f, XFASTINT (data->frame_height),
2b653806 5001 XFASTINT (data->frame_width), 0, 0, 0);
e3678b64 5002#if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
8f6ea2e9
KH
5003 if (XFASTINT (data->frame_menu_bar_lines)
5004 != previous_frame_menu_bar_lines)
f8ad443a 5005 x_set_menu_bar_lines (f, data->frame_menu_bar_lines, make_number (0));
4314246f 5006#ifdef HAVE_WINDOW_SYSTEM
9ea173e8
GM
5007 if (XFASTINT (data->frame_tool_bar_lines)
5008 != previous_frame_tool_bar_lines)
5009 x_set_tool_bar_lines (f, data->frame_tool_bar_lines, make_number (0));
4314246f 5010#endif
217f2871 5011#endif
fd482be5 5012
719eaeb1
GM
5013 /* "Swap out" point from the selected window
5014 into its buffer. We do this now, before
5015 restoring the window contents, and prevent it from
5016 being done later on when we select a new window. */
596ae0cf
RS
5017 if (! NILP (XWINDOW (selected_window)->buffer))
5018 {
5019 w = XWINDOW (selected_window);
5020 set_marker_both (w->pointm,
5021 w->buffer,
5022 BUF_PT (XBUFFER (w->buffer)),
5023 BUF_PT_BYTE (XBUFFER (w->buffer)));
5024 }
5025
fd482be5 5026 windows_or_buffers_changed++;
29aeee73 5027 FRAME_WINDOW_SIZES_CHANGED (f) = 1;
fd482be5 5028
5500c422
GM
5029 /* Problem: Freeing all matrices and later allocating them again
5030 is a serious redisplay flickering problem. What we would
5031 really like to do is to free only those matrices not reused
5032 below. */
5033 root_window = XWINDOW (FRAME_ROOT_WINDOW (f));
5034 leaf_windows
5035 = (struct window **) alloca (count_windows (root_window)
5036 * sizeof (struct window *));
5037 n_leaf_windows = get_leaf_windows (root_window, leaf_windows, 0);
5038
756b6edc
RS
5039 /* Temporarily avoid any problems with windows that are smaller
5040 than they are supposed to be. */
5041 window_min_height = 1;
5042 window_min_width = 1;
5043
fd482be5
JB
5044 /* Kludge Alert!
5045 Mark all windows now on frame as "deleted".
5046 Restoring the new configuration "undeletes" any that are in it.
37962e60 5047
fd482be5
JB
5048 Save their current buffers in their height fields, since we may
5049 need it later, if a buffer saved in the configuration is now
5050 dead. */
5051 delete_all_subwindows (XWINDOW (FRAME_ROOT_WINDOW (f)));
5052
5053 for (k = 0; k < saved_windows->size; k++)
5054 {
5055 p = SAVED_WINDOW_N (saved_windows, k);
5056 w = XWINDOW (p->window);
5057 w->next = Qnil;
7ab12479 5058
fd482be5
JB
5059 if (!NILP (p->parent))
5060 w->parent = SAVED_WINDOW_N (saved_windows,
5061 XFASTINT (p->parent))->window;
5062 else
5063 w->parent = Qnil;
7ab12479 5064
fd482be5 5065 if (!NILP (p->prev))
7ab12479 5066 {
fd482be5
JB
5067 w->prev = SAVED_WINDOW_N (saved_windows,
5068 XFASTINT (p->prev))->window;
5069 XWINDOW (w->prev)->next = p->window;
5070 }
5071 else
5072 {
5073 w->prev = Qnil;
5074 if (!NILP (w->parent))
5075 {
5076 if (EQ (p->width, XWINDOW (w->parent)->width))
5077 {
5078 XWINDOW (w->parent)->vchild = p->window;
5079 XWINDOW (w->parent)->hchild = Qnil;
5080 }
5081 else
5082 {
5083 XWINDOW (w->parent)->hchild = p->window;
5084 XWINDOW (w->parent)->vchild = Qnil;
5085 }
5086 }
5087 }
5088
5089 /* If we squirreled away the buffer in the window's height,
5090 restore it now. */
017b2bad 5091 if (BUFFERP (w->height))
fd482be5
JB
5092 w->buffer = w->height;
5093 w->left = p->left;
5094 w->top = p->top;
5095 w->width = p->width;
5096 w->height = p->height;
5097 w->hscroll = p->hscroll;
ea68264b 5098 w->min_hscroll = p->min_hscroll;
fd482be5 5099 w->display_table = p->display_table;
a1d58e5b
GM
5100 w->orig_top = p->orig_top;
5101 w->orig_height = p->orig_height;
d834a2e9 5102 XSETFASTINT (w->last_modified, 0);
3cd21523 5103 XSETFASTINT (w->last_overlay_modified, 0);
fd482be5
JB
5104
5105 /* Reinstall the saved buffer and pointers into it. */
5106 if (NILP (p->buffer))
5107 w->buffer = p->buffer;
5108 else
5109 {
5110 if (!NILP (XBUFFER (p->buffer)->name))
5111 /* If saved buffer is alive, install it. */
5112 {
5113 w->buffer = p->buffer;
5114 w->start_at_line_beg = p->start_at_line_beg;
b73ea88e
RS
5115 set_marker_restricted (w->start, p->start, w->buffer);
5116 set_marker_restricted (w->pointm, p->pointm, w->buffer);
fd482be5 5117 Fset_marker (XBUFFER (w->buffer)->mark,
b73ea88e 5118 p->mark, w->buffer);
fd482be5
JB
5119
5120 /* As documented in Fcurrent_window_configuration, don't
243a5ce6
RS
5121 restore the location of point in the buffer which was
5122 current when the window configuration was recorded. */
6b54027b
RS
5123 if (!EQ (p->buffer, new_current_buffer)
5124 && XBUFFER (p->buffer) == current_buffer)
fd482be5
JB
5125 Fgoto_char (w->pointm);
5126 }
52a68e98
RS
5127 else if (NILP (w->buffer) || NILP (XBUFFER (w->buffer)->name))
5128 /* Else unless window has a live buffer, get one. */
7ab12479 5129 {
fd482be5
JB
5130 w->buffer = Fcdr (Fcar (Vbuffer_alist));
5131 /* This will set the markers to beginning of visible
5132 range. */
5133 set_marker_restricted (w->start, make_number (0), w->buffer);
5134 set_marker_restricted (w->pointm, make_number (0),w->buffer);
5135 w->start_at_line_beg = Qt;
7ab12479
JB
5136 }
5137 else
fd482be5 5138 /* Keeping window's old buffer; make sure the markers
52a68e98 5139 are real. */
7ab12479 5140 {
fd482be5
JB
5141 /* Set window markers at start of visible range. */
5142 if (XMARKER (w->start)->buffer == 0)
5143 set_marker_restricted (w->start, make_number (0),
5144 w->buffer);
5145 if (XMARKER (w->pointm)->buffer == 0)
b73ea88e
RS
5146 set_marker_restricted_both (w->pointm, w->buffer,
5147 BUF_PT (XBUFFER (w->buffer)),
5148 BUF_PT_BYTE (XBUFFER (w->buffer)));
fd482be5 5149 w->start_at_line_beg = Qt;
7ab12479
JB
5150 }
5151 }
5152 }
9ace597f 5153
fd482be5 5154 FRAME_ROOT_WINDOW (f) = data->root_window;
719eaeb1
GM
5155 /* Prevent "swapping out point" in the old selected window
5156 using the buffer that has been restored into it.
72695e47
RS
5157 Use the point value from the beginning of this function
5158 since unshow_buffer (called from delete_all_subwindows)
5159 could have altered it. */
719eaeb1 5160 selected_window = Qnil;
243a5ce6
RS
5161 if (EQ (XWINDOW (data->current_window)->buffer, new_current_buffer))
5162 set_marker_restricted (XWINDOW (data->current_window)->pointm,
18787f5e 5163 make_number (old_point),
243a5ce6
RS
5164 XWINDOW (data->current_window)->buffer);
5165
fd482be5 5166 Fselect_window (data->current_window);
396a830c
RS
5167 XBUFFER (XWINDOW (selected_window)->buffer)->last_selected_window
5168 = selected_window;
7ab12479 5169
db269683 5170 if (NILP (data->focus_frame)
017b2bad 5171 || (FRAMEP (data->focus_frame)
db269683
JB
5172 && FRAME_LIVE_P (XFRAME (data->focus_frame))))
5173 Fredirect_frame_focus (frame, data->focus_frame);
7ab12479 5174
fd482be5
JB
5175#if 0 /* I don't understand why this is needed, and it causes problems
5176 when the frame's old selected window has been deleted. */
e4e59717 5177 if (f != selected_frame && FRAME_WINDOW_P (f))
9a7c6fc3 5178 do_switch_frame (WINDOW_FRAME (XWINDOW (data->root_window)),
d12f6f83 5179 0, 0);
fd482be5
JB
5180#endif
5181
5182 /* Set the screen height to the value it had before this function. */
5183 if (previous_frame_height != FRAME_HEIGHT (f)
5184 || previous_frame_width != FRAME_WIDTH (f))
5185 change_frame_size (f, previous_frame_height, previous_frame_width,
2b653806 5186 0, 0, 0);
e3678b64 5187#if defined (HAVE_WINDOW_SYSTEM) || defined (MSDOS)
8f6ea2e9 5188 if (previous_frame_menu_bar_lines != FRAME_MENU_BAR_LINES (f))
f8ad443a
AS
5189 x_set_menu_bar_lines (f, make_number (previous_frame_menu_bar_lines),
5190 make_number (0));
4314246f 5191#ifdef HAVE_WINDOW_SYSTEM
9ea173e8
GM
5192 if (previous_frame_tool_bar_lines != FRAME_TOOL_BAR_LINES (f))
5193 x_set_tool_bar_lines (f, make_number (previous_frame_tool_bar_lines),
5194 make_number (0));
4314246f 5195#endif
217f2871 5196#endif
d2b35234 5197
5500c422 5198 /* Now, free glyph matrices in windows that were not reused. */
c4280705
GM
5199 for (i = n = 0; i < n_leaf_windows; ++i)
5200 {
5201 if (NILP (leaf_windows[i]->buffer))
5202 {
5203 /* Assert it's not reused as a combination. */
5204 xassert (NILP (leaf_windows[i]->hchild)
5205 && NILP (leaf_windows[i]->vchild));
5206 free_window_matrices (leaf_windows[i]);
c4280705
GM
5207 }
5208 else if (EQ (leaf_windows[i]->buffer, new_current_buffer))
5209 ++n;
5210 }
5500c422
GM
5211
5212 adjust_glyphs (f);
5213
d2b35234 5214 UNBLOCK_INPUT;
756b6edc 5215
478292ed
RS
5216 /* Fselect_window will have made f the selected frame, so we
5217 reselect the proper frame here. Fhandle_switch_frame will change the
5218 selected window too, but that doesn't make the call to
5219 Fselect_window above totally superfluous; it still sets f's
5220 selected window. */
5221 if (FRAME_LIVE_P (XFRAME (data->selected_frame)))
d12f6f83 5222 do_switch_frame (data->selected_frame, 0, 0);
478292ed
RS
5223
5224 if (! NILP (Vwindow_configuration_change_hook)
5225 && ! NILP (Vrun_hooks))
5226 call1 (Vrun_hooks, Qwindow_configuration_change_hook);
5227 }
bdc727bf
JB
5228
5229 if (!NILP (new_current_buffer))
243a5ce6 5230 Fset_buffer (new_current_buffer);
bdc727bf 5231
478292ed
RS
5232 /* Restore the minimum heights recorded in the configuration. */
5233 window_min_height = XINT (data->min_height);
5234 window_min_width = XINT (data->min_width);
543f5fb1 5235
478292ed 5236 Vminibuf_scroll_window = data->minibuf_scroll_window;
3dbab091 5237 minibuf_selected_window = data->minibuf_selected_window;
543f5fb1 5238
3f8ab7bd 5239 return (FRAME_LIVE_P (f) ? Qt : Qnil);
7ab12479
JB
5240}
5241
44fa5b1e 5242/* Mark all windows now on frame as deleted
7ab12479
JB
5243 by setting their buffers to nil. */
5244
fd482be5 5245void
7ab12479
JB
5246delete_all_subwindows (w)
5247 register struct window *w;
5248{
265a9e55 5249 if (!NILP (w->next))
7ab12479 5250 delete_all_subwindows (XWINDOW (w->next));
265a9e55 5251 if (!NILP (w->vchild))
7ab12479 5252 delete_all_subwindows (XWINDOW (w->vchild));
265a9e55 5253 if (!NILP (w->hchild))
7ab12479 5254 delete_all_subwindows (XWINDOW (w->hchild));
605be8af
JB
5255
5256 w->height = w->buffer; /* See Fset_window_configuration for excuse. */
5257
86e48436
RS
5258 if (!NILP (w->buffer))
5259 unshow_buffer (w);
5260
605be8af
JB
5261 /* We set all three of these fields to nil, to make sure that we can
5262 distinguish this dead window from any live window. Live leaf
5263 windows will have buffer set, and combination windows will have
5264 vchild or hchild set. */
5265 w->buffer = Qnil;
5266 w->vchild = Qnil;
5267 w->hchild = Qnil;
acf70840
GM
5268
5269 Vwindow_list = Qnil;
7ab12479
JB
5270}
5271\f
5272static int
5273count_windows (window)
5274 register struct window *window;
5275{
5276 register int count = 1;
265a9e55 5277 if (!NILP (window->next))
7ab12479 5278 count += count_windows (XWINDOW (window->next));
265a9e55 5279 if (!NILP (window->vchild))
7ab12479 5280 count += count_windows (XWINDOW (window->vchild));
265a9e55 5281 if (!NILP (window->hchild))
7ab12479
JB
5282 count += count_windows (XWINDOW (window->hchild));
5283 return count;
5284}
5285
5500c422
GM
5286
5287/* Fill vector FLAT with leaf windows under W, starting at index I.
5288 Value is last index + 1. */
5289
5290static int
5291get_leaf_windows (w, flat, i)
5292 struct window *w;
5293 struct window **flat;
5294 int i;
5295{
5296 while (w)
5297 {
5298 if (!NILP (w->hchild))
5299 i = get_leaf_windows (XWINDOW (w->hchild), flat, i);
5300 else if (!NILP (w->vchild))
5301 i = get_leaf_windows (XWINDOW (w->vchild), flat, i);
5302 else
5303 flat[i++] = w;
5304
5305 w = NILP (w->next) ? 0 : XWINDOW (w->next);
5306 }
5307
5308 return i;
5309}
5310
5311
5312/* Return a pointer to the glyph W's physical cursor is on. Value is
5313 null if W's current matrix is invalid, so that no meaningfull glyph
5314 can be returned. */
5315
5316struct glyph *
5317get_phys_cursor_glyph (w)
5318 struct window *w;
5319{
5320 struct glyph_row *row;
5321 struct glyph *glyph;
5322
5323 if (w->phys_cursor.vpos >= 0
5324 && w->phys_cursor.vpos < w->current_matrix->nrows
5325 && (row = MATRIX_ROW (w->current_matrix, w->phys_cursor.vpos),
5326 row->enabled_p)
5327 && row->used[TEXT_AREA] > w->phys_cursor.hpos)
5328 glyph = row->glyphs[TEXT_AREA] + w->phys_cursor.hpos;
5329 else
5330 glyph = NULL;
5331
5332 return glyph;
5333}
5334
5335
7ab12479
JB
5336static int
5337save_window_save (window, vector, i)
5338 Lisp_Object window;
5339 struct Lisp_Vector *vector;
5340 int i;
5341{
5342 register struct saved_window *p;
5343 register struct window *w;
5344 register Lisp_Object tem;
5345
265a9e55 5346 for (;!NILP (window); window = w->next)
7ab12479
JB
5347 {
5348 p = SAVED_WINDOW_N (vector, i);
5349 w = XWINDOW (window);
5350
d834a2e9 5351 XSETFASTINT (w->temslot, i++);
7ab12479
JB
5352 p->window = window;
5353 p->buffer = w->buffer;
5354 p->left = w->left;
5355 p->top = w->top;
5356 p->width = w->width;
5357 p->height = w->height;
5358 p->hscroll = w->hscroll;
ea68264b 5359 p->min_hscroll = w->min_hscroll;
7ab12479 5360 p->display_table = w->display_table;
a1d58e5b
GM
5361 p->orig_top = w->orig_top;
5362 p->orig_height = w->orig_height;
265a9e55 5363 if (!NILP (w->buffer))
7ab12479
JB
5364 {
5365 /* Save w's value of point in the window configuration.
5366 If w is the selected window, then get the value of point
5367 from the buffer; pointm is garbage in the selected window. */
5368 if (EQ (window, selected_window))
5369 {
5370 p->pointm = Fmake_marker ();
b73ea88e
RS
5371 set_marker_both (p->pointm, w->buffer,
5372 BUF_PT (XBUFFER (w->buffer)),
5373 BUF_PT_BYTE (XBUFFER (w->buffer)));
7ab12479
JB
5374 }
5375 else
eeb82665 5376 p->pointm = Fcopy_marker (w->pointm, Qnil);
7ab12479 5377
eeb82665 5378 p->start = Fcopy_marker (w->start, Qnil);
7ab12479
JB
5379 p->start_at_line_beg = w->start_at_line_beg;
5380
5381 tem = XBUFFER (w->buffer)->mark;
eeb82665 5382 p->mark = Fcopy_marker (tem, Qnil);
7ab12479
JB
5383 }
5384 else
5385 {
5386 p->pointm = Qnil;
5387 p->start = Qnil;
5388 p->mark = Qnil;
5389 p->start_at_line_beg = Qnil;
5390 }
5391
265a9e55 5392 if (NILP (w->parent))
7ab12479
JB
5393 p->parent = Qnil;
5394 else
5395 p->parent = XWINDOW (w->parent)->temslot;
5396
265a9e55 5397 if (NILP (w->prev))
7ab12479
JB
5398 p->prev = Qnil;
5399 else
5400 p->prev = XWINDOW (w->prev)->temslot;
5401
265a9e55 5402 if (!NILP (w->vchild))
7ab12479 5403 i = save_window_save (w->vchild, vector, i);
265a9e55 5404 if (!NILP (w->hchild))
7ab12479
JB
5405 i = save_window_save (w->hchild, vector, i);
5406 }
5407
5408 return i;
5409}
5410
a0d76c27 5411DEFUN ("current-window-configuration", Fcurrent_window_configuration,
fdb82f93
PJ
5412 Scurrent_window_configuration, 0, 1, 0,
5413 doc: /* Return an object representing the current window configuration of FRAME.
5414If FRAME is nil or omitted, use the selected frame.
5415This describes the number of windows, their sizes and current buffers,
5416and for each displayed buffer, where display starts, and the positions of
5417point and mark. An exception is made for point in the current buffer:
5418its value is -not- saved.
5419This also records the currently selected frame, and FRAME's focus
5420redirection (see `redirect-frame-focus'). */)
5421 (frame)
44fa5b1e 5422 Lisp_Object frame;
7ab12479
JB
5423{
5424 register Lisp_Object tem;
5425 register int n_windows;
5426 register struct save_window_data *data;
da2792e0 5427 register struct Lisp_Vector *vec;
7ab12479 5428 register int i;
44fa5b1e 5429 FRAME_PTR f;
43bad991 5430
44fa5b1e 5431 if (NILP (frame))
1ae1a37d 5432 frame = selected_frame;
b7826503 5433 CHECK_LIVE_FRAME (frame);
1ae1a37d 5434 f = XFRAME (frame);
7ab12479 5435
44fa5b1e 5436 n_windows = count_windows (XWINDOW (FRAME_ROOT_WINDOW (f)));
26605be9 5437 vec = allocate_other_vector (VECSIZE (struct save_window_data));
da2792e0
KH
5438 data = (struct save_window_data *)vec;
5439
d834a2e9
KH
5440 XSETFASTINT (data->frame_width, FRAME_WIDTH (f));
5441 XSETFASTINT (data->frame_height, FRAME_HEIGHT (f));
5442 XSETFASTINT (data->frame_menu_bar_lines, FRAME_MENU_BAR_LINES (f));
9ea173e8 5443 XSETFASTINT (data->frame_tool_bar_lines, FRAME_TOOL_BAR_LINES (f));
1ae1a37d 5444 data->selected_frame = selected_frame;
44fa5b1e 5445 data->current_window = FRAME_SELECTED_WINDOW (f);
74112613 5446 XSETBUFFER (data->current_buffer, current_buffer);
3f49fddc 5447 data->minibuf_scroll_window = minibuf_level > 0 ? Vminibuf_scroll_window : Qnil;
3dbab091 5448 data->minibuf_selected_window = minibuf_level > 0 ? minibuf_selected_window : Qnil;
44fa5b1e 5449 data->root_window = FRAME_ROOT_WINDOW (f);
bdc727bf 5450 data->focus_frame = FRAME_FOCUS_FRAME (f);
74112613
KH
5451 XSETINT (data->min_height, window_min_height);
5452 XSETINT (data->min_width, window_min_width);
7ab12479
JB
5453 tem = Fmake_vector (make_number (n_windows), Qnil);
5454 data->saved_windows = tem;
5455 for (i = 0; i < n_windows; i++)
5456 XVECTOR (tem)->contents[i]
5457 = Fmake_vector (make_number (SAVED_WINDOW_VECTOR_SIZE), Qnil);
a1d58e5b 5458 save_window_save (FRAME_ROOT_WINDOW (f), XVECTOR (tem), 0);
74112613 5459 XSETWINDOW_CONFIGURATION (tem, data);
7ab12479
JB
5460 return (tem);
5461}
5462
5463DEFUN ("save-window-excursion", Fsave_window_excursion, Ssave_window_excursion,
fdb82f93
PJ
5464 0, UNEVALLED, 0,
5465 doc: /* Execute body, preserving window sizes and contents.
5466Restore which buffer appears in which window, where display starts,
5467and the value of point and mark for each window.
5468Also restore the choice of selected window.
5469Also restore which buffer is current.
6924dda6
MB
5470Does not restore the value of point in current buffer.
5471usage: (save-window-excursion BODY ...) */)
fdb82f93 5472 (args)
7ab12479
JB
5473 Lisp_Object args;
5474{
5475 register Lisp_Object val;
aed13378 5476 register int count = SPECPDL_INDEX ();
7ab12479
JB
5477
5478 record_unwind_protect (Fset_window_configuration,
43bad991 5479 Fcurrent_window_configuration (Qnil));
7ab12479
JB
5480 val = Fprogn (args);
5481 return unbind_to (count, val);
5482}
5500c422
GM
5483
5484\f
5485/***********************************************************************
5486 Marginal Areas
5487 ***********************************************************************/
5488
5489DEFUN ("set-window-margins", Fset_window_margins, Sset_window_margins,
a7bdfc08 5490 2, 3, 0,
fdb82f93
PJ
5491 doc: /* Set width of marginal areas of window WINDOW.
5492If window is nil, set margins of the currently selected window.
5493First parameter LEFT-WIDTH specifies the number of character
5494cells to reserve for the left marginal area. Second parameter
5495RIGHT-WIDTH does the same for the right marginal area.
5496A nil width parameter means no margin. */)
5497 (window, left, right)
5500c422
GM
5498 Lisp_Object window, left, right;
5499{
5500 struct window *w = decode_window (window);
5500c422
GM
5501
5502 if (!NILP (left))
b7826503 5503 CHECK_NUMBER_OR_FLOAT (left);
5500c422 5504 if (!NILP (right))
b7826503 5505 CHECK_NUMBER_OR_FLOAT (right);
5500c422
GM
5506
5507 /* Check widths < 0 and translate a zero width to nil.
5508 Margins that are too wide have to be checked elsewhere. */
5509 if ((INTEGERP (left) && XINT (left) < 0)
7539e11f 5510 || (FLOATP (left) && XFLOAT_DATA (left) <= 0))
5500c422
GM
5511 XSETFASTINT (left, 0);
5512 if (INTEGERP (left) && XFASTINT (left) == 0)
5513 left = Qnil;
5514
5515 if ((INTEGERP (right) && XINT (right) < 0)
7539e11f 5516 || (FLOATP (right) && XFLOAT_DATA (right) <= 0))
5500c422
GM
5517 XSETFASTINT (right, 0);
5518 if (INTEGERP (right) && XFASTINT (right) == 0)
5519 right = Qnil;
5520
5521 w->left_margin_width = left;
5522 w->right_margin_width = right;
5523
5524 ++windows_or_buffers_changed;
5525 adjust_glyphs (XFRAME (WINDOW_FRAME (w)));
5526 return Qnil;
5527}
5528
5529
5530DEFUN ("window-margins", Fwindow_margins, Swindow_margins,
5531 0, 1, 0,
fdb82f93
PJ
5532 doc: /* Get width of marginal areas of window WINDOW.
5533If WINDOW is omitted or nil, use the currently selected window.
5534Value is a cons of the form (LEFT-WIDTH . RIGHT-WIDTH).
5535If a marginal area does not exist, its width will be returned
5536as nil. */)
5537 (window)
5500c422
GM
5538 Lisp_Object window;
5539{
5540 struct window *w = decode_window (window);
5541 return Fcons (w->left_margin_width, w->right_margin_width);
5542}
5543
5544
7ab12479 5545\f
5500c422
GM
5546/***********************************************************************
5547 Smooth scrolling
5548 ***********************************************************************/
5549
5550DEFUN ("window-vscroll", Fwindow_vscroll, Swindow_vscroll, 0, 1, 0,
fdb82f93
PJ
5551 doc: /* Return the amount by which WINDOW is scrolled vertically.
5552Use the selected window if WINDOW is nil or omitted.
5553Value is a multiple of the canonical character height of WINDOW. */)
5554 (window)
5500c422
GM
5555 Lisp_Object window;
5556{
47004952 5557 Lisp_Object result;
5500c422
GM
5558 struct frame *f;
5559 struct window *w;
5560
5561 if (NILP (window))
5562 window = selected_window;
47004952 5563 else
b7826503 5564 CHECK_WINDOW (window);
5500c422
GM
5565 w = XWINDOW (window);
5566 f = XFRAME (w->frame);
5567
5568 if (FRAME_WINDOW_P (f))
47004952 5569 result = CANON_Y_FROM_PIXEL_Y (f, -w->vscroll);
5500c422 5570 else
47004952
GM
5571 result = make_number (0);
5572 return result;
5500c422
GM
5573}
5574
5575
5576DEFUN ("set-window-vscroll", Fset_window_vscroll, Sset_window_vscroll,
47004952 5577 2, 2, 0,
fdb82f93 5578 doc: /* Set amount by which WINDOW should be scrolled vertically to VSCROLL.
b0f906c8
EZ
5579WINDOW nil means use the selected window. VSCROLL is a non-negative
5580multiple of the canonical character height of WINDOW. */)
fdb82f93 5581 (window, vscroll)
47004952 5582 Lisp_Object window, vscroll;
5500c422
GM
5583{
5584 struct window *w;
5585 struct frame *f;
5586
5500c422
GM
5587 if (NILP (window))
5588 window = selected_window;
47004952 5589 else
b7826503
PJ
5590 CHECK_WINDOW (window);
5591 CHECK_NUMBER_OR_FLOAT (vscroll);
47004952 5592
5500c422
GM
5593 w = XWINDOW (window);
5594 f = XFRAME (w->frame);
5595
5596 if (FRAME_WINDOW_P (f))
5597 {
5598 int old_dy = w->vscroll;
47004952
GM
5599
5600 w->vscroll = - CANON_Y_UNIT (f) * XFLOATINT (vscroll);
5601 w->vscroll = min (w->vscroll, 0);
5500c422
GM
5602
5603 /* Adjust glyph matrix of the frame if the virtual display
5604 area becomes larger than before. */
5605 if (w->vscroll < 0 && w->vscroll < old_dy)
5606 adjust_glyphs (f);
5607
5608 /* Prevent redisplay shortcuts. */
b1599b4c 5609 XBUFFER (w->buffer)->prevent_redisplay_optimizations_p = 1;
5500c422
GM
5610 }
5611
47004952 5612 return Fwindow_vscroll (window);
5500c422
GM
5613}
5614
7bbb5782
GM
5615\f
5616/* Call FN for all leaf windows on frame F. FN is called with the
5617 first argument being a pointer to the leaf window, and with
f95464e4 5618 additional argument USER_DATA. Stops when FN returns 0. */
7bbb5782
GM
5619
5620void
f95464e4 5621foreach_window (f, fn, user_data)
7bbb5782 5622 struct frame *f;
f95464e4
GM
5623 int (* fn) P_ ((struct window *, void *));
5624 void *user_data;
7bbb5782 5625{
f95464e4 5626 foreach_window_1 (XWINDOW (FRAME_ROOT_WINDOW (f)), fn, user_data);
7bbb5782
GM
5627}
5628
5629
5630/* Helper function for foreach_window. Call FN for all leaf windows
5631 reachable from W. FN is called with the first argument being a
f95464e4 5632 pointer to the leaf window, and with additional argument USER_DATA.
67492200 5633 Stop when FN returns 0. Value is 0 if stopped by FN. */
7bbb5782 5634
67492200 5635static int
f95464e4 5636foreach_window_1 (w, fn, user_data)
7bbb5782 5637 struct window *w;
f95464e4
GM
5638 int (* fn) P_ ((struct window *, void *));
5639 void *user_data;
7bbb5782 5640{
67492200
GM
5641 int cont;
5642
5643 for (cont = 1; w && cont;)
7bbb5782
GM
5644 {
5645 if (!NILP (w->hchild))
f95464e4 5646 cont = foreach_window_1 (XWINDOW (w->hchild), fn, user_data);
7bbb5782 5647 else if (!NILP (w->vchild))
f95464e4 5648 cont = foreach_window_1 (XWINDOW (w->vchild), fn, user_data);
0f532a9a
GM
5649 else
5650 cont = fn (w, user_data);
7bbb5782
GM
5651
5652 w = NILP (w->next) ? 0 : XWINDOW (w->next);
5653 }
67492200
GM
5654
5655 return cont;
7bbb5782
GM
5656}
5657
5658
6d194a45 5659/* Freeze or unfreeze the window start of W unless it is a
f95464e4 5660 mini-window or the selected window. FREEZE_P non-null means freeze
7bbb5782
GM
5661 the window start. */
5662
67492200 5663static int
7bbb5782
GM
5664freeze_window_start (w, freeze_p)
5665 struct window *w;
f95464e4 5666 void *freeze_p;
7bbb5782
GM
5667{
5668 if (w == XWINDOW (selected_window)
5669 || MINI_WINDOW_P (w)
5670 || (MINI_WINDOW_P (XWINDOW (selected_window))
24d744ac 5671 && ! NILP (Vminibuf_scroll_window)
7bbb5782 5672 && w == XWINDOW (Vminibuf_scroll_window)))
f95464e4 5673 freeze_p = NULL;
7bbb5782 5674
f95464e4 5675 w->frozen_window_start_p = freeze_p != NULL;
67492200 5676 return 1;
7bbb5782
GM
5677}
5678
5679
5680/* Freeze or unfreeze the window starts of all leaf windows on frame
5681 F, except the selected window and a mini-window. FREEZE_P non-zero
5682 means freeze the window start. */
5683
5684void
5685freeze_window_starts (f, freeze_p)
5686 struct frame *f;
5687 int freeze_p;
5688{
cbccabec 5689 foreach_window (f, freeze_window_start, (void *) (freeze_p ? f : 0));
7bbb5782 5690}
5500c422
GM
5691
5692\f
5693/***********************************************************************
5694 Initialization
5695 ***********************************************************************/
5696
cbff28e8
RS
5697/* Return 1 if window configurations C1 and C2
5698 describe the same state of affairs. This is used by Fequal. */
5699
5700int
2f8274be 5701compare_window_configurations (c1, c2, ignore_positions)
cbff28e8 5702 Lisp_Object c1, c2;
2f8274be 5703 int ignore_positions;
cbff28e8
RS
5704{
5705 register struct save_window_data *d1, *d2;
5706 struct Lisp_Vector *sw1, *sw2;
5707 int i;
5708
4d3edcb4
GM
5709 if (!WINDOW_CONFIGURATIONP (c1))
5710 wrong_type_argument (Qwindow_configuration_p, c1);
5711 if (!WINDOW_CONFIGURATIONP (c2))
5712 wrong_type_argument (Qwindow_configuration_p, c2);
5713
cbff28e8
RS
5714 d1 = (struct save_window_data *) XVECTOR (c1);
5715 d2 = (struct save_window_data *) XVECTOR (c2);
5716 sw1 = XVECTOR (d1->saved_windows);
5717 sw2 = XVECTOR (d2->saved_windows);
5718
5719 if (! EQ (d1->frame_width, d2->frame_width))
5720 return 0;
5721 if (! EQ (d1->frame_height, d2->frame_height))
5722 return 0;
5723 if (! EQ (d1->frame_menu_bar_lines, d2->frame_menu_bar_lines))
5724 return 0;
5725 if (! EQ (d1->selected_frame, d2->selected_frame))
5726 return 0;
5727 /* Don't compare the current_window field directly.
5728 Instead see w1_is_current and w2_is_current, below. */
5729 if (! EQ (d1->current_buffer, d2->current_buffer))
5730 return 0;
2f8274be 5731 if (! ignore_positions)
3f49fddc
KS
5732 {
5733 if (! EQ (d1->minibuf_scroll_window, d2->minibuf_scroll_window))
5734 return 0;
5735 if (! EQ (d1->minibuf_selected_window, d2->minibuf_selected_window))
5736 return 0;
5737 }
cbff28e8
RS
5738 /* Don't compare the root_window field.
5739 We don't require the two configurations
5740 to use the same window object,
5741 and the two root windows must be equivalent
5742 if everything else compares equal. */
5743 if (! EQ (d1->focus_frame, d2->focus_frame))
5744 return 0;
5745 if (! EQ (d1->min_width, d2->min_width))
5746 return 0;
5747 if (! EQ (d1->min_height, d2->min_height))
5748 return 0;
5749
5750 /* Verify that the two confis have the same number of windows. */
5751 if (sw1->size != sw2->size)
5752 return 0;
5753
5754 for (i = 0; i < sw1->size; i++)
5755 {
5756 struct saved_window *p1, *p2;
5757 int w1_is_current, w2_is_current;
5758
5759 p1 = SAVED_WINDOW_N (sw1, i);
5760 p2 = SAVED_WINDOW_N (sw2, i);
5761
5762 /* Verify that the current windows in the two
5763 configurations correspond to each other. */
5764 w1_is_current = EQ (d1->current_window, p1->window);
5765 w2_is_current = EQ (d2->current_window, p2->window);
5766
5767 if (w1_is_current != w2_is_current)
5768 return 0;
5769
5770 /* Verify that the corresponding windows do match. */
5771 if (! EQ (p1->buffer, p2->buffer))
5772 return 0;
5773 if (! EQ (p1->left, p2->left))
5774 return 0;
5775 if (! EQ (p1->top, p2->top))
5776 return 0;
5777 if (! EQ (p1->width, p2->width))
5778 return 0;
5779 if (! EQ (p1->height, p2->height))
5780 return 0;
cbff28e8
RS
5781 if (! EQ (p1->display_table, p2->display_table))
5782 return 0;
5783 if (! EQ (p1->parent, p2->parent))
5784 return 0;
5785 if (! EQ (p1->prev, p2->prev))
5786 return 0;
2f8274be
RS
5787 if (! ignore_positions)
5788 {
5789 if (! EQ (p1->hscroll, p2->hscroll))
5790 return 0;
ea68264b
GM
5791 if (!EQ (p1->min_hscroll, p2->min_hscroll))
5792 return 0;
2f8274be
RS
5793 if (! EQ (p1->start_at_line_beg, p2->start_at_line_beg))
5794 return 0;
5795 if (NILP (Fequal (p1->start, p2->start)))
5796 return 0;
5797 if (NILP (Fequal (p1->pointm, p2->pointm)))
5798 return 0;
5799 if (NILP (Fequal (p1->mark, p2->mark)))
5800 return 0;
5801 }
cbff28e8
RS
5802 }
5803
5804 return 1;
5805}
2f8274be
RS
5806
5807DEFUN ("compare-window-configurations", Fcompare_window_configurations,
5808 Scompare_window_configurations, 2, 2, 0,
fdb82f93
PJ
5809 doc: /* Compare two window configurations as regards the structure of windows.
5810This function ignores details such as the values of point and mark
5811and scrolling positions. */)
5812 (x, y)
2f8274be
RS
5813 Lisp_Object x, y;
5814{
5815 if (compare_window_configurations (x, y, 1))
5816 return Qt;
5817 return Qnil;
5818}
cbff28e8 5819\f
dfcf069d 5820void
7ab12479
JB
5821init_window_once ()
5822{
1ae1a37d
GM
5823 struct frame *f = make_terminal_frame ();
5824 XSETFRAME (selected_frame, f);
5825 Vterminal_frame = selected_frame;
5826 minibuf_window = f->minibuffer_window;
5827 selected_window = f->selected_window;
5828 last_nonminibuf_frame = f;
5b03d3c0
RS
5829
5830 window_initialized = 1;
7ab12479
JB
5831}
5832
67492200
GM
5833void
5834init_window ()
5835{
5836 Vwindow_list = Qnil;
5837}
5838
dfcf069d 5839void
7ab12479
JB
5840syms_of_window ()
5841{
8a37516b
GM
5842 Qwindow_size_fixed = intern ("window-size-fixed");
5843 staticpro (&Qwindow_size_fixed);
233a4a2c 5844
543f5fb1
RS
5845 staticpro (&Qwindow_configuration_change_hook);
5846 Qwindow_configuration_change_hook
5847 = intern ("window-configuration-change-hook");
5848
7ab12479
JB
5849 Qwindowp = intern ("windowp");
5850 staticpro (&Qwindowp);
5851
3f8ab7bd
RS
5852 Qwindow_configuration_p = intern ("window-configuration-p");
5853 staticpro (&Qwindow_configuration_p);
5854
806b4d9b
JB
5855 Qwindow_live_p = intern ("window-live-p");
5856 staticpro (&Qwindow_live_p);
605be8af 5857
2cccc823 5858 Qtemp_buffer_show_hook = intern ("temp-buffer-show-hook");
a58ec57d
RS
5859 staticpro (&Qtemp_buffer_show_hook);
5860
67492200 5861 staticpro (&Vwindow_list);
8bfb170b
KS
5862
5863 minibuf_selected_window = Qnil;
3dbab091 5864 staticpro (&minibuf_selected_window);
67492200 5865
7ab12479 5866 DEFVAR_LISP ("temp-buffer-show-function", &Vtemp_buffer_show_function,
fdb82f93
PJ
5867 doc: /* Non-nil means call as function to display a help buffer.
5868The function is called with one argument, the buffer to be displayed.
5869Used by `with-output-to-temp-buffer'.
5870If this function is used, then it must do the entire job of showing
5871the buffer; `temp-buffer-show-hook' is not run unless this function runs it. */);
7ab12479
JB
5872 Vtemp_buffer_show_function = Qnil;
5873
5874 DEFVAR_LISP ("display-buffer-function", &Vdisplay_buffer_function,
fdb82f93
PJ
5875 doc: /* If non-nil, function to call to handle `display-buffer'.
5876It will receive two args, the buffer and a flag which if non-nil means
5877 that the currently selected window is not acceptable.
5878Commands such as `switch-to-buffer-other-window' and `find-file-other-window'
5879work using this function. */);
7ab12479
JB
5880 Vdisplay_buffer_function = Qnil;
5881
6529ed87 5882 DEFVAR_LISP ("even-window-heights", &Veven_window_heights,
fdb82f93
PJ
5883 doc: /* *If non-nil, `display-buffer' should even the window heights.
5884If nil, `display-buffer' will leave the window configuration alone. */);
6529ed87
GM
5885 Veven_window_heights = Qt;
5886
7ab12479 5887 DEFVAR_LISP ("minibuffer-scroll-window", &Vminibuf_scroll_window,
fdb82f93 5888 doc: /* Non-nil means it is the window that C-M-v in minibuffer should scroll. */);
7ab12479
JB
5889 Vminibuf_scroll_window = Qnil;
5890
cc91894c 5891 DEFVAR_BOOL ("mode-line-in-non-selected-windows", &mode_line_in_non_selected_windows,
26124d5e 5892 doc: /* Non-nil means to use `mode-line-inactive' face in non-selected windows.
cc91894c
KS
5893If the minibuffer is active, the `minibuffer-scroll-window' mode line
5894is displayed in the `mode-line' face. */);
5895 mode_line_in_non_selected_windows = 1;
26124d5e 5896
7ab12479 5897 DEFVAR_LISP ("other-window-scroll-buffer", &Vother_window_scroll_buffer,
fdb82f93 5898 doc: /* If non-nil, this is a buffer and \\[scroll-other-window] should scroll its window. */);
7ab12479
JB
5899 Vother_window_scroll_buffer = Qnil;
5900
44fa5b1e 5901 DEFVAR_BOOL ("pop-up-frames", &pop_up_frames,
fdb82f93 5902 doc: /* *Non-nil means `display-buffer' should make a separate frame. */);
44fa5b1e 5903 pop_up_frames = 0;
7ab12479 5904
9c3da604 5905 DEFVAR_BOOL ("display-buffer-reuse-frames", &display_buffer_reuse_frames,
fdb82f93
PJ
5906 doc: /* *Non-nil means `display-buffer' should reuse frames.
5907If the buffer in question is already displayed in a frame, raise that frame. */);
9c3da604
GM
5908 display_buffer_reuse_frames = 0;
5909
44fa5b1e 5910 DEFVAR_LISP ("pop-up-frame-function", &Vpop_up_frame_function,
fdb82f93
PJ
5911 doc: /* Function to call to handle automatic new frame creation.
5912It is called with no arguments and should return a newly created frame.
5913
5914A typical value might be `(lambda () (new-frame pop-up-frame-alist))'
5915where `pop-up-frame-alist' would hold the default frame parameters. */);
44fa5b1e 5916 Vpop_up_frame_function = Qnil;
7ab12479 5917
a90712c2 5918 DEFVAR_LISP ("special-display-buffer-names", &Vspecial_display_buffer_names,
fdb82f93
PJ
5919 doc: /* *List of buffer names that should have their own special frames.
5920Displaying a buffer whose name is in this list makes a special frame for it
5921using `special-display-function'. See also `special-display-regexps'.
5922
5923An element of the list can be a list instead of just a string.
5924There are two ways to use a list as an element:
5925 (BUFFER FRAME-PARAMETERS...) (BUFFER FUNCTION OTHER-ARGS...)
5926In the first case, FRAME-PARAMETERS are used to create the frame.
5927In the latter case, FUNCTION is called with BUFFER as the first argument,
5928followed by OTHER-ARGS--it can display BUFFER in any way it likes.
5929All this is done by the function found in `special-display-function'.
5930
5931If this variable appears \"not to work\", because you add a name to it
5932but that buffer still appears in the selected window, look at the
5933values of `same-window-buffer-names' and `same-window-regexps'.
5934Those variables take precedence over this one. */);
a90712c2
RS
5935 Vspecial_display_buffer_names = Qnil;
5936
5937 DEFVAR_LISP ("special-display-regexps", &Vspecial_display_regexps,
fdb82f93
PJ
5938 doc: /* *List of regexps saying which buffers should have their own special frames.
5939If a buffer name matches one of these regexps, it gets its own frame.
5940Displaying a buffer whose name is in this list makes a special frame for it
5941using `special-display-function'.
5942
5943An element of the list can be a list instead of just a string.
5944There are two ways to use a list as an element:
5945 (REGEXP FRAME-PARAMETERS...) (REGEXP FUNCTION OTHER-ARGS...)
5946In the first case, FRAME-PARAMETERS are used to create the frame.
5947In the latter case, FUNCTION is called with the buffer as first argument,
5948followed by OTHER-ARGS--it can display the buffer in any way it likes.
5949All this is done by the function found in `special-display-function'.
5950
5951If this variable appears \"not to work\", because you add a regexp to it
5952but the matching buffers still appear in the selected window, look at the
5953values of `same-window-buffer-names' and `same-window-regexps'.
5954Those variables take precedence over this one. */);
a90712c2
RS
5955 Vspecial_display_regexps = Qnil;
5956
5957 DEFVAR_LISP ("special-display-function", &Vspecial_display_function,
fdb82f93
PJ
5958 doc: /* Function to call to make a new frame for a special buffer.
5959It is called with two arguments, the buffer and optional buffer specific
5960data, and should return a window displaying that buffer.
5961The default value makes a separate frame for the buffer,
5962using `special-display-frame-alist' to specify the frame parameters.
5963
5964A buffer is special if its is listed in `special-display-buffer-names'
5965or matches a regexp in `special-display-regexps'. */);
a90712c2
RS
5966 Vspecial_display_function = Qnil;
5967
855d8627 5968 DEFVAR_LISP ("same-window-buffer-names", &Vsame_window_buffer_names,
fdb82f93
PJ
5969 doc: /* *List of buffer names that should appear in the selected window.
5970Displaying one of these buffers using `display-buffer' or `pop-to-buffer'
5971switches to it in the selected window, rather than making it appear
5972in some other window.
5973
5974An element of the list can be a cons cell instead of just a string.
5975Then the car must be a string, which specifies the buffer name.
5976This is for compatibility with `special-display-buffer-names';
5977the cdr of the cons cell is ignored.
5978
5979See also `same-window-regexps'. */);
855d8627
RS
5980 Vsame_window_buffer_names = Qnil;
5981
5982 DEFVAR_LISP ("same-window-regexps", &Vsame_window_regexps,
fdb82f93
PJ
5983 doc: /* *List of regexps saying which buffers should appear in the selected window.
5984If a buffer name matches one of these regexps, then displaying it
5985using `display-buffer' or `pop-to-buffer' switches to it
5986in the selected window, rather than making it appear in some other window.
5987
5988An element of the list can be a cons cell instead of just a string.
5989Then the car must be a string, which specifies the buffer name.
5990This is for compatibility with `special-display-buffer-names';
5991the cdr of the cons cell is ignored.
5992
5993See also `same-window-buffer-names'. */);
855d8627
RS
5994 Vsame_window_regexps = Qnil;
5995
7ab12479 5996 DEFVAR_BOOL ("pop-up-windows", &pop_up_windows,
fdb82f93 5997 doc: /* *Non-nil means display-buffer should make new windows. */);
7ab12479
JB
5998 pop_up_windows = 1;
5999
6000 DEFVAR_INT ("next-screen-context-lines", &next_screen_context_lines,
fdb82f93 6001 doc: /* *Number of lines of continuity when scrolling by screenfuls. */);
7ab12479
JB
6002 next_screen_context_lines = 2;
6003
6004 DEFVAR_INT ("split-height-threshold", &split_height_threshold,
fdb82f93
PJ
6005 doc: /* *display-buffer would prefer to split the largest window if this large.
6006If there is only one window, it is split regardless of this value. */);
7ab12479
JB
6007 split_height_threshold = 500;
6008
6009 DEFVAR_INT ("window-min-height", &window_min_height,
fdb82f93 6010 doc: /* *Delete any window less than this tall (including its mode line). */);
7ab12479
JB
6011 window_min_height = 4;
6012
6013 DEFVAR_INT ("window-min-width", &window_min_width,
fdb82f93 6014 doc: /* *Delete any window less than this wide. */);
7ab12479
JB
6015 window_min_width = 10;
6016
5500c422
GM
6017 DEFVAR_LISP ("scroll-preserve-screen-position",
6018 &Vscroll_preserve_screen_position,
fdb82f93 6019 doc: /* *Non-nil means scroll commands move point to keep its screen line unchanged. */);
5500c422 6020 Vscroll_preserve_screen_position = Qnil;
9317a85d 6021
543f5fb1
RS
6022 DEFVAR_LISP ("window-configuration-change-hook",
6023 &Vwindow_configuration_change_hook,
fdb82f93
PJ
6024 doc: /* Functions to call when window configuration changes.
6025The selected frame is the one whose configuration has changed. */);
543f5fb1
RS
6026 Vwindow_configuration_change_hook = Qnil;
6027
b0b7ed0e 6028 DEFVAR_BOOL ("window-size-fixed", &window_size_fixed,
fdb82f93
PJ
6029 doc: /* Non-nil in a buffer means windows displaying the buffer are fixed-size.
6030Emacs won't change the size of any window displaying that buffer,
6031unless you explicitly change the size, or Emacs has no other choice.
6032This variable automatically becomes buffer-local when set. */);
b0b7ed0e
GM
6033 Fmake_variable_buffer_local (Qwindow_size_fixed);
6034 window_size_fixed = 0;
6035
7ab12479
JB
6036 defsubr (&Sselected_window);
6037 defsubr (&Sminibuffer_window);
6038 defsubr (&Swindow_minibuffer_p);
6039 defsubr (&Swindowp);
806b4d9b 6040 defsubr (&Swindow_live_p);
7ab12479
JB
6041 defsubr (&Spos_visible_in_window_p);
6042 defsubr (&Swindow_buffer);
6043 defsubr (&Swindow_height);
6044 defsubr (&Swindow_width);
6045 defsubr (&Swindow_hscroll);
6046 defsubr (&Sset_window_hscroll);
190eb263
RS
6047 defsubr (&Swindow_redisplay_end_trigger);
6048 defsubr (&Sset_window_redisplay_end_trigger);
7ab12479 6049 defsubr (&Swindow_edges);
d5783c40
JB
6050 defsubr (&Scoordinates_in_window_p);
6051 defsubr (&Swindow_at);
7ab12479
JB
6052 defsubr (&Swindow_point);
6053 defsubr (&Swindow_start);
6054 defsubr (&Swindow_end);
6055 defsubr (&Sset_window_point);
6056 defsubr (&Sset_window_start);
6057 defsubr (&Swindow_dedicated_p);
d207b766 6058 defsubr (&Sset_window_dedicated_p);
7ab12479
JB
6059 defsubr (&Swindow_display_table);
6060 defsubr (&Sset_window_display_table);
6061 defsubr (&Snext_window);
6062 defsubr (&Sprevious_window);
6063 defsubr (&Sother_window);
6064 defsubr (&Sget_lru_window);
6065 defsubr (&Sget_largest_window);
6066 defsubr (&Sget_buffer_window);
6067 defsubr (&Sdelete_other_windows);
6068 defsubr (&Sdelete_windows_on);
6069 defsubr (&Sreplace_buffer_in_windows);
6070 defsubr (&Sdelete_window);
6071 defsubr (&Sset_window_buffer);
6072 defsubr (&Sselect_window);
4628f7a4
EN
6073 defsubr (&Sspecial_display_p);
6074 defsubr (&Ssame_window_p);
7ab12479
JB
6075 defsubr (&Sdisplay_buffer);
6076 defsubr (&Ssplit_window);
6077 defsubr (&Senlarge_window);
6078 defsubr (&Sshrink_window);
6079 defsubr (&Sscroll_up);
6080 defsubr (&Sscroll_down);
6081 defsubr (&Sscroll_left);
6082 defsubr (&Sscroll_right);
ccd0664b 6083 defsubr (&Sother_window_for_scrolling);
7ab12479 6084 defsubr (&Sscroll_other_window);
fa832261 6085 defsubr (&Sminibuffer_selected_window);
7ab12479 6086 defsubr (&Srecenter);
81fe0836 6087 defsubr (&Swindow_text_height);
7ab12479
JB
6088 defsubr (&Smove_to_window_line);
6089 defsubr (&Swindow_configuration_p);
3f8ab7bd 6090 defsubr (&Swindow_configuration_frame);
7ab12479
JB
6091 defsubr (&Sset_window_configuration);
6092 defsubr (&Scurrent_window_configuration);
6093 defsubr (&Ssave_window_excursion);
5500c422
GM
6094 defsubr (&Sset_window_margins);
6095 defsubr (&Swindow_margins);
6096 defsubr (&Swindow_vscroll);
6097 defsubr (&Sset_window_vscroll);
2f8274be 6098 defsubr (&Scompare_window_configurations);
67492200 6099 defsubr (&Swindow_list);
7ab12479
JB
6100}
6101
dfcf069d 6102void
7ab12479
JB
6103keys_of_window ()
6104{
6105 initial_define_key (control_x_map, '1', "delete-other-windows");
6106 initial_define_key (control_x_map, '2', "split-window");
6107 initial_define_key (control_x_map, '0', "delete-window");
6108 initial_define_key (control_x_map, 'o', "other-window");
6109 initial_define_key (control_x_map, '^', "enlarge-window");
6110 initial_define_key (control_x_map, '<', "scroll-left");
6111 initial_define_key (control_x_map, '>', "scroll-right");
6112
6113 initial_define_key (global_map, Ctl ('V'), "scroll-up");
6114 initial_define_key (meta_map, Ctl ('V'), "scroll-other-window");
6115 initial_define_key (meta_map, 'v', "scroll-down");
6116
6117 initial_define_key (global_map, Ctl('L'), "recenter");
6118 initial_define_key (meta_map, 'r', "move-to-window-line");
6119}