Mercurial > emacs
view src/frame.c @ 406:bd0533ed9b5a
Initial revision
author | Jim Blandy <jimb@redhat.com> |
---|---|
date | Fri, 23 Aug 1991 03:01:50 +0000 |
parents | 6acd2a203aa8 |
children | ca410e0b4797 |
line wrap: on
line source
/* Generic screen functions. Copyright (C) 1989 Free Software Foundation. This file is part of GNU Emacs. GNU Emacs is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 1, or (at your option) any later version. GNU Emacs is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with GNU Emacs; see the file COPYING. If not, write to the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA. */ #include <stdio.h> #include "config.h" #include "lisp.h" #include "screen.h" #include "window.h" #include "termhooks.h" Lisp_Object Vemacs_iconified; Lisp_Object Qscreenp; Lisp_Object Vscreen_list; Lisp_Object Vterminal_screen; Lisp_Object Vglobal_minibuffer_screen; Lisp_Object Vdefault_screen_alist; /* A screen which is not just a minibuffer, or 0 if there are no such screens. This is usually the most recent such screen that was selected. */ struct screen *last_nonminibuf_screen; extern Lisp_Object Vminibuffer_list; extern Lisp_Object get_minibuffer (); DEFUN ("screenp", Fscreenp, Sscreenp, 1, 1, 0, "Return non-nil if OBJECT is a screen.\n\ Value is t for a termcap screen (a character-only terminal),\n\ `x' for an Emacs screen that is really an X window.") (screen) Lisp_Object screen; { if (XTYPE (screen) != Lisp_Screen) return Qnil; switch (XSCREEN (screen)->output_method) { case output_termcap: return Qt; case output_x_window: return intern ("x"); default: abort (); } } struct screen * make_screen (mini_p) int mini_p; { Lisp_Object screen; register struct screen *s; register Lisp_Object root_window; register Lisp_Object mini_window; screen = Fmake_vector (((sizeof (struct screen) - (sizeof (Lisp_Vector) - sizeof (Lisp_Object))) / sizeof (Lisp_Object)), make_number (0)); XSETTYPE (screen, Lisp_Screen); s = XSCREEN (screen); s->cursor_x = 0; s->cursor_y = 0; s->current_glyphs = 0; s->desired_glyphs = 0; s->visible = 0; s->display.nothing = 0; s->iconified = 0; s->wants_modeline = 1; s->auto_raise = 0; s->auto_lower = 0; s->no_split = 0; s->garbaged = 0; s->has_minibuffer = mini_p; s->focus_screen = screen; s->param_alist = Qnil; root_window = make_window (0); if (mini_p) { mini_window = make_window (0); XWINDOW (root_window)->next = mini_window; XWINDOW (mini_window)->prev = root_window; XWINDOW (mini_window)->mini_p = Qt; XWINDOW (mini_window)->screen = screen; s->minibuffer_window = mini_window; } else { mini_window = Qnil; XWINDOW (root_window)->next = Qnil; s->minibuffer_window = Qnil; } XWINDOW (root_window)->screen = screen; /* 10 is arbitrary, just so that there is "something there." Correct size will be set up later with change_screen_size. */ s->width = 10; s->height = 10; XFASTINT (XWINDOW (root_window)->width) = 10; XFASTINT (XWINDOW (root_window)->height) = (mini_p ? 9 : 10); if (mini_p) { XFASTINT (XWINDOW (mini_window)->width) = 10; XFASTINT (XWINDOW (mini_window)->top) = 9; XFASTINT (XWINDOW (mini_window)->height) = 1; } /* Choose a buffer for the screen's root window. */ { Lisp_Object buf; XWINDOW (root_window)->buffer = Qt; buf = Fcurrent_buffer (); /* If buf is a 'hidden' buffer (i.e. one whose name starts with a space), try to find another one. */ if (XSTRING (Fbuffer_name (buf))->data[0] == ' ') buf = Fother_buffer (buf); Fset_window_buffer (root_window, buf); } if (mini_p) { XWINDOW (mini_window)->buffer = Qt; Fset_window_buffer (mini_window, (NULL (Vminibuffer_list) ? get_minibuffer (0) : Fcar (Vminibuffer_list))); } s->root_window = root_window; s->selected_window = root_window; /* Make sure this window seems more recently used than a newly-created, never-selected window. */ XFASTINT (XWINDOW (s->selected_window)->use_time) = ++window_select_count; Vscreen_list = Fcons (screen, Vscreen_list); return s; } /* Make a screen using a separate minibuffer window on another screen. MINI_WINDOW is the minibuffer window to use. nil means use the default (the global minibuffer). */ struct screen * make_screen_without_minibuffer (mini_window) register Lisp_Object mini_window; { register struct screen *s; /* Choose the minibuffer window to use. */ if (NULL (mini_window)) { if (XTYPE (Vglobal_minibuffer_screen) != Lisp_Screen) error ("global-minibuffer-screen must be set to create minibufferless screens."); mini_window = XSCREEN (Vglobal_minibuffer_screen)->minibuffer_window; } else { CHECK_WINDOW (mini_window, 0); } /* Make a screen containing just a root window. */ s = make_screen (0); /* Install the chosen minibuffer window, with proper buffer. */ s->minibuffer_window = mini_window; Fset_window_buffer (mini_window, (NULL (Vminibuffer_list) ? get_minibuffer (0) : Fcar (Vminibuffer_list))); return s; } /* Make a screen containing only a minibuffer window. */ struct screen * make_minibuffer_screen () { /* First make a screen containing just a root window, no minibuffer. */ register struct screen *s = make_screen (0); register Lisp_Object mini_window; register Lisp_Object screen; XSET (screen, Lisp_Screen, s); /* ??? Perhaps leave it to the user program to set auto_raise. */ s->auto_raise = 1; s->auto_lower = 0; s->no_split = 1; s->wants_modeline = 0; /* Note we leave has_minibuffer as 0. This is a little strange. */ /* Now label the root window as also being the minibuffer. Avoid infinite looping on the window chain by marking next pointer as nil. */ mini_window = s->minibuffer_window = s->root_window; XWINDOW (mini_window)->mini_p = Qt; XWINDOW (mini_window)->next = Qnil; XWINDOW (mini_window)->prev = mini_window; XWINDOW (mini_window)->screen = screen; /* Put the proper buffer in that window. */ Fset_window_buffer (mini_window, (NULL (Vminibuffer_list) ? get_minibuffer (0) : Fcar (Vminibuffer_list))); return s; } /* Construct a screen that refers to the terminal (stdin and stdout). */ struct screen * make_terminal_screen () { register struct screen *s; Vscreen_list = Qnil; s = make_screen (1); s->name = build_string ("terminal"); s->visible = 1; s->display.nothing = 1; /* Nonzero means screen isn't deleted. */ XSET (Vterminal_screen, Lisp_Screen, s); return s; } DEFUN ("select-screen", Fselect_screen, Sselect_screen, 1, 2, 0, "Select the screen S. S's selected window becomes \"the\"\n\ selected window. If the optional parameter NO-ENTER is non-nil, don't\n\ focus on that screen.") (screen, no_enter) Lisp_Object screen, no_enter; { CHECK_SCREEN (screen, 0); if (selected_screen == XSCREEN (screen)) return screen; selected_screen = XSCREEN (screen); if (!EQ (SCREEN_ROOT_WINDOW (selected_screen), SCREEN_MINIBUF_WINDOW (selected_screen))) last_nonminibuf_screen = selected_screen; Fselect_window (XSCREEN (screen)->selected_window); #ifdef HAVE_X_WINDOWS #ifdef MULTI_SCREEN if (XSCREEN (screen)->output_method == output_x_window && NULL (no_enter)) { Ffocus_screen (screen); } #endif #endif choose_minibuf_screen (); return screen; } DEFUN ("selected-screen", Fselected_screen, Sselected_screen, 0, 0, 0, "Return the screen that is now selected.") () { Lisp_Object tem; XSET (tem, Lisp_Screen, selected_screen); return tem; } DEFUN ("window-screen", Fwindow_screen, Swindow_screen, 1, 1, 0, "Return the screen object that window WINDOW is on.") (window) Lisp_Object window; { CHECK_WINDOW (window, 0); return XWINDOW (window)->screen; } DEFUN ("screen-root-window", Fscreen_root_window, Sscreen_root_window, 0, 1, 0, "Returns the root-window of SCREEN.") (screen) Lisp_Object screen; { if (NULL (screen)) XSET (screen, Lisp_Screen, selected_screen); CHECK_SCREEN (screen, 0); return XSCREEN (screen)->root_window; } DEFUN ("screen-selected-window", Fscreen_selected_window, Sscreen_selected_window, 0, 1, 0, "Return the selected window of screen object SCREEN.") (screen) Lisp_Object screen; { if (NULL (screen)) XSET (screen, Lisp_Screen, selected_screen); CHECK_SCREEN (screen, 0); return XSCREEN (screen)->selected_window; } DEFUN ("screen-list", Fscreen_list, Sscreen_list, 0, 0, 0, "Return a list of all screens.") () { return Fcopy_sequence (Vscreen_list); } #ifdef MULTI_SCREEN Lisp_Object next_screen (screen, mini_screen) Lisp_Object screen; int mini_screen; { Lisp_Object tail; int passed = 0; while (1) for (tail = Vscreen_list; CONSP (tail); tail = XCONS (tail)->cdr) { if (passed) { SCREEN_PTR s = XSCREEN (XCONS (tail)->car); if (!mini_screen /* Is this screen only a minibuffer? */ && EQ (SCREEN_ROOT_WINDOW (s), SCREEN_MINIBUF_WINDOW (s)) /* If we have wrapped all the way around the list (i.e. the only screen is an all-minibuffer screen), return it anyway. */ && s != XSCREEN (screen)) continue; else return XCONS (tail)->car; } if (EQ (screen, XCONS (tail)->car)) passed++; } } Lisp_Object prev_screen (screen, mini_screen) Lisp_Object screen; int mini_screen; { Lisp_Object tail; Lisp_Object prev; prev = Qnil; while (1) for (tail = Vscreen_list; CONSP (tail); tail = XCONS (tail)->cdr) { if (EQ (screen, XCONS (tail)->car)) { if (!NULL (prev) && (mini_screen || !EQ (XCONS (tail)->car, Vglobal_minibuffer_screen))) return prev; } prev = XCONS (tail)->car; } } DEFUN ("next-screen", Fnext_screen, Snext_screen, 0, 2, 0, "Return the next screen in the screen list after SCREEN.\n\ If MINISCREEN is non-nil, include screens whose only window is a minibuffer.\n\ If MINISCREEN is nil or omitted, these screens are skipped.") (screen, miniscreen) Lisp_Object screen, miniscreen; { Lisp_Object tail; if (NULL (screen)) XSET (screen, Lisp_Screen, selected_screen); CHECK_SCREEN (screen, 0); return next_screen (screen, (NULL (miniscreen) ? 0 : 1)); } #endif /* MULTI_SCREEN */ DEFUN ("delete-screen", Fdelete_screen, Sdelete_screen, 0, 1, "", "Delete SCREEN, permanently eliminating it from use.\n\ Default is current screen.") (screen) Lisp_Object screen; { struct screen *s; union display displ; if (EQ (screen, Qnil)) { s = selected_screen; XSET (screen, Lisp_Screen, s); } else { CHECK_SCREEN (screen, 0); s = XSCREEN (screen); } /* Are there any other screens besides this one? */ if (s == selected_screen && EQ (next_screen (screen, 1), screen)) error ("Attempt to delete the only screen"); /* Does this screen have a minibuffer, and is it the surrogate minibuffer for any other screen? */ if (EQ (screen, WINDOW_SCREEN (XWINDOW (SCREEN_MINIBUF_WINDOW (XSCREEN (screen)))))) { Lisp_Object screen2; for (screen2 = Vscreen_list; CONSP (2); screen2 = XCONS (screen2)->cdr) if (! EQ (screen2, screen) && EQ (screen, WINDOW_SCREEN (XWINDOW (SCREEN_MINIBUF_WINDOW (XSCREEN (screen2)))))) error ("Attempt to delete a surrogate minibuffer screen"); } /* Don't let the screen remain selected. */ if (s == selected_screen) Fselect_screen (next_screen (screen, 1)); /* Don't allow minibuf_window to remain on a deleted screen. */ if (EQ (s->minibuffer_window, minibuf_window)) { Fset_window_buffer (selected_screen->minibuffer_window, XWINDOW (minibuf_window)->buffer); minibuf_window = selected_screen->minibuffer_window; } Vscreen_list = Fdelq (screen, Vscreen_list); s->visible = 0; displ = s->display; s->display.nothing = 0; #ifdef HAVE_X_WINDOWS if (s->output_method == output_x_window) x_destroy_window (s, displ); #endif /* If we've deleted the last_nonminibuf_screen, then try to find another one. */ if (s == last_nonminibuf_screen) { last_nonminibuf_screen = 0; for (screen = Vscreen_list; CONSP (screen); screen = XCONS (screen)->cdr) { s = XSCREEN (XCONS (screen)->cdr); if (!EQ (SCREEN_ROOT_WINDOW (s), SCREEN_MINIBUF_WINDOW (s))) { last_nonminibuf_screen = s; break; } } } return Qnil; } /* Return mouse position in character cell units. */ static read_mouse_position (screen, x, y) Lisp_Object screen; int *x, *y; { CHECK_SCREEN (screen, 0); *x = 1; *y = 1; #ifdef HAVE_X_WINDOWS if (XSCREEN (screen)->output_method == output_x_window) x_read_mouse_position (XSCREEN (screen), x, y); #endif } DEFUN ("read-mouse-position", Fread_mouse_position, Sread_mouse_position, 1, 1, 0, "Return a cons (x . y) which represents the position of the mouse.") (screen) Lisp_Object screen; { int x, y; struct screen *s; CHECK_SCREEN (screen, 0); read_mouse_position (screen, &x, &y); return Fcons (make_number (x), make_number (y)); } DEFUN ("set-mouse-position", Fset_mouse_position, Sset_mouse_position, 3, 3, 0, "Move the mouse pointer to the center of cell (X,Y) in SCREEN.\n\ WARNING: If you use this under X, you should do unfocus-screen afterwards.") (screen, x, y) Lisp_Object screen, x, y; { CHECK_SCREEN (screen, 0); CHECK_NUMBER (x, 2); CHECK_NUMBER (y, 1); #ifdef HAVE_X_WINDOWS if (XSCREEN (screen)->output_method == output_x_window) /* Warping the mouse will cause enternotify and focus events. */ x_set_mouse_position (XSCREEN (screen), x, y); #endif return Qnil; } #if 0 /* ??? Can this be replaced with a Lisp function? It is used in minibuf.c. Can we get rid of that? */ DEFUN ("screen-configuration", Fscreen_configuration, Sscreen_configuration, 0, 0, 0, "Return object describing current screen configuration.\n\ The screen configuration is the current mouse position and selected screen.\n\ This object can be given to `restore-screen-configuration'\n\ to restore this screen configuration.") () { int x, y; Lisp_Object c, screen; struct screen *s; c = Fmake_vector (make_number(3), Qnil); XVECTOR (c)->contents[0] = screen = Fselected_screen(); read_mouse_position (screen, &x, &y); XVECTOR (c)->contents[1] = make_number (x); XVECTOR (c)->contents[2] = make_number (y); return c; } DEFUN ("restore-screen-configuration", Frestore_screen_configuration, Srestore_screen_configuration, 1, 1, 0, "Restores screen configuration CONFIGURATION.") (config) Lisp_Object config; { Lisp_Object x_pos, y_pos, screen; CHECK_VECTOR (config, 0); if (XVECTOR (config)->size != 3) { error ("Wrong size vector passed to restore-screen-configuration"); } screen = XVECTOR (config)->contents[0]; CHECK_SCREEN (screen, 0); Fselect_screen (screen, Qnil); #if 0 /* This seems to interfere with the screen selection mechanism. jla */ x_pos = XVECTOR (config)->contents[1]; y_pos = XVECTOR (config)->contents[2]; set_mouse_position (screen, XINT (x_pos), XINT (y_pos)); #endif return screen; } #endif DEFUN ("make-screen-visible", Fmake_screen_visible, Smake_screen_visible, 1, 1, 0, "Make the screen SCREEN visible (assuming it is an X-window).\n\ Also raises the screen so that nothing obscures it.") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); if (XSCREEN (screen)->display.nothing == 0) error ("Cannot make a dead screen object visible"); if (XSCREEN (screen)->output_method == output_x_window) x_make_screen_visible (XSCREEN (screen)); return screen; } DEFUN ("make-screen-invisible", Fmake_screen_invisible, Smake_screen_invisible, 1, 1, 0, "Make the screen SCREEN invisible (assuming it is an X-window).") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); if (XSCREEN (screen)->output_method == output_x_window) x_make_screen_invisible (XSCREEN (screen)); return Qnil; } DEFUN ("iconify-screen", Ficonify_screen, Siconify_screen, 1, 1, 0, "Make the screen SCREEN into an icon.") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); if (XSCREEN (screen)->display.nothing == 0) error ("Cannot make a dead screen object iconified."); if (XSCREEN (screen)->output_method == output_x_window) x_iconify_screen (XSCREEN (screen)); return Qnil; } DEFUN ("deiconify-screen", Fdeiconify_screen, Sdeiconify_screen, 1, 1, 0, "Open (de-iconify) the iconified screen SCREEN.") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); if (XSCREEN (screen)->display.nothing == 0) error ("Cannot deiconify a dead screen object."); if (XSCREEN (screen)->output_method == output_x_window) x_make_screen_visible (XSCREEN (screen)); return screen; } DEFUN ("screen-visible-p", Fscreen_visible_p, Sscreen_visible_p, 1, 1, 0, "Return t if SCREEN is now \"visible\" (actually in use for display).\n\ A screen that is not \"visible\" is not updated and, if it works through\n\ a window system, it may not show at all.\n\ Return the symbol `icon' if window is visible only as an icon.") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); if (XSCREEN (screen)->visible) return Qt; if (XSCREEN (screen)->iconified) return intern ("icon"); return Qnil; } DEFUN ("visible-screen-list", Fvisible_screen_list, Svisible_screen_list, 0, 0, 0, "Return a list of all screens now \"visible\" (being updated).") () { Lisp_Object tail, screen; struct screen *s; Lisp_Object value; value = Qnil; for (tail = Vscreen_list; CONSP (tail); tail = XCONS (tail)->cdr) { screen = XCONS (tail)->car; if (XTYPE (screen) != Lisp_Screen) continue; s = XSCREEN (screen); if (s->visible) value = Fcons (screen, value); } return value; } DEFUN ("redirect-screen-focus", Fredirect_screen_focus, Sredirect_screen_focus, 1, 2, 0, "Arrange for keystrokes typed at SCREEN to be sent to FOCUS-SCREEN.\n\ This means that, after reading a keystroke typed at SCREEN,\n\ last-event-screen will be FOCUS-SCREEN.\n\ \n\ If FOCUS-SCREEN is omitted or eq to SCREEN, any existing redirection is\n\ cancelled, and the screen again receives its own keystrokes.\n\ \n\ The redirection lasts until the next call to redirect-screen-focus\n\ or select-screen.\n\ \n\ This is useful for temporarily redirecting keystrokes to the minibuffer\n\ window when a screen doesn't have its own minibuffer.") (screen, focus_screen) Lisp_Object screen, focus_screen; { CHECK_SCREEN (screen, 0); if (NULL (focus_screen)) focus_screen = screen; else CHECK_SCREEN (focus_screen, 1); XSCREEN (screen)->focus_screen = focus_screen; if (screen_rehighlight_hook) (*screen_rehighlight_hook) (); return Qnil; } DEFUN ("screen-focus", Fscreen_focus, Sscreen_focus, 1, 1, 0, "Return the screen to which SCREEN's keystrokes are currently being sent.\n\ See redirect-screen-focus.") (screen) Lisp_Object screen; { CHECK_SCREEN (screen, 0); return SCREEN_FOCUS_SCREEN (XSCREEN (screen)); } Lisp_Object get_screen_param (screen, prop) register struct screen *screen; Lisp_Object prop; { register Lisp_Object tem; tem = Fassq (prop, screen->param_alist); if (EQ (tem, Qnil)) return tem; return Fcdr (tem); } void store_in_alist (alistptr, propname, val) Lisp_Object *alistptr, val; char *propname; { register Lisp_Object tem; register Lisp_Object prop; prop = intern (propname); tem = Fassq (prop, *alistptr); if (EQ (tem, Qnil)) *alistptr = Fcons (Fcons (prop, val), *alistptr); else Fsetcdr (tem, val); } void store_screen_param (s, prop, val) struct screen *s; Lisp_Object prop, val; { register Lisp_Object tem; tem = Fassq (prop, s->param_alist); if (EQ (tem, Qnil)) s->param_alist = Fcons (Fcons (prop, val), s->param_alist); else Fsetcdr (tem, val); } DEFUN ("screen-parameters", Fscreen_parameters, Sscreen_parameters, 0, 1, 0, "Return the parameters-alist of screen SCREEN.\n\ It is a list of elements of the form (PARM . VALUE), where PARM is a symbol.\n\ The meaningful PARMs depend on the kind of screen.") (screen) Lisp_Object screen; { Lisp_Object alist; struct screen *s; if (EQ (screen, Qnil)) s = selected_screen; else { CHECK_SCREEN (screen, 0); s = XSCREEN (screen); } if (s->display.nothing == 0) return Qnil; alist = Fcopy_alist (s->param_alist); store_in_alist (&alist, "name", s->name); store_in_alist (&alist, "height", make_number (s->height)); store_in_alist (&alist, "width", make_number (s->width)); store_in_alist (&alist, "modeline", (s->wants_modeline ? Qt : Qnil)); store_in_alist (&alist, "minibuffer", (s->has_minibuffer ? Qt : Qnil)); store_in_alist (&alist, "unsplittable", (s->no_split ? Qt : Qnil)); if (s->output_method == output_x_window) x_report_screen_params (s, &alist); return alist; } DEFUN ("modify-screen-parameters", Fmodify_screen_parameters, Smodify_screen_parameters, 2, 2, 0, "Modify the parameters of screen SCREEN according to ALIST.\n\ ALIST is an alist of parameters to change and their new values.\n\ Each element of ALIST has the form (PARM . VALUE), where PARM is a symbol.\n\ The meaningful PARMs depend on the kind of screen; undefined PARMs are ignored.") (screen, alist) Lisp_Object screen, alist; { register struct screen *s; register Lisp_Object tail, elt, prop, val; if (EQ (screen, Qnil)) s = selected_screen; else { CHECK_SCREEN (screen, 0); s = XSCREEN (screen); } if (s->display.nothing == 0) error ("Cannot modify parameters of a deleted screen"); if (s->output_method == output_x_window) for (tail = alist; !EQ (tail, Qnil); tail = Fcdr (tail)) { elt = Fcar (tail); prop = Fcar (elt); val = Fcdr (elt); x_set_screen_param (s, prop, val, get_screen_param (s, prop)); store_screen_param (s, prop, val); } return Qnil; } DEFUN ("screen-pixel-size", Fscreen_pixel_size, Sscreen_pixel_size, 1, 1, 0, "Return a cons (width . height) of screen SCREEN's dimensions.") (screen) Lisp_Object screen; { register struct screen *s; int width, height; CHECK_SCREEN (screen, 0); s = XSCREEN (screen); return Fcons (make_number (x_pixel_width (s)), make_number (x_pixel_height (s))); } DEFUN ("screen-height", Fscreen_height, Sscreen_height, 0, 0, 0, "Return number of lines available for display on selected screen.") () { return make_number (SCREEN_HEIGHT (selected_screen)); } DEFUN ("screen-width", Fscreen_width, Sscreen_width, 0, 0, 0, "Return number of columns available for display on selected screen.") () { return make_number (SCREEN_WIDTH (selected_screen)); } DEFUN ("set-screen-height", Fset_screen_height, Sset_screen_height, 2, 3, 0, "Specify that the screen SCREEN has LINES lines.\n\ Optional third arg non-nil means that redisplay should use LINES lines\n\ but that the idea of the actual height of the screen should not be changed.") (screen, rows, pretend) Lisp_Object rows, pretend; { register struct screen *s; CHECK_NUMBER (rows, 0); if (NULL (screen)) s = selected_screen; else { CHECK_SCREEN (screen, 0); s = XSCREEN (screen); } if (s->output_method == output_x_window) { if (XINT (rows) != s->width) x_set_window_size (s, s->width, XINT (rows)); } else change_screen_size (s, XINT (rows), 0, !NULL (pretend)); return Qnil; } DEFUN ("set-screen-width", Fset_screen_width, Sset_screen_width, 2, 3, 0, "Specify that the screen SCREEN has COLS columns.\n\ Optional third arg non-nil means that redisplay should use COLS columns\n\ but that the idea of the actual width of the screen should not be changed.") (screen, cols, pretend) Lisp_Object cols, pretend; { register struct screen *s; CHECK_NUMBER (cols, 0); if (NULL (screen)) s = selected_screen; else { CHECK_SCREEN (screen, 0); s = XSCREEN (screen); } if (s->output_method == output_x_window) { if (XINT (cols) != s->width) x_set_window_size (s, XINT (cols), s->height); } else change_screen_size (selected_screen, 0, XINT (cols), !NULL (pretend)); return Qnil; } DEFUN ("set-screen-size", Fset_screen_size, Sset_screen_size, 3, 3, 0, "Sets size of SCREEN to COLS by ROWS, measured in characters.") (screen, cols, rows) Lisp_Object screen, cols, rows; { register struct screen *s; int mask; CHECK_SCREEN (screen, 0); CHECK_NUMBER (cols, 2); CHECK_NUMBER (rows, 1); s = XSCREEN (screen); if (s->output_method == output_x_window) { if (XINT (rows) != s->height || XINT (cols) != s->width) x_set_window_size (s, XINT (cols), XINT (rows)); } else change_screen_size (s, XINT (rows), XINT (cols), 0); return Qnil; } DEFUN ("set-screen-position", Fset_screen_position, Sset_screen_position, 3, 3, 0, "Sets size of SCREEN in pixels to XOFFSET by YOFFSET.") (screen, xoffset, yoffset) Lisp_Object screen, xoffset, yoffset; { register struct screen *s; int mask; CHECK_SCREEN (screen, 0); CHECK_NUMBER (xoffset, 1); CHECK_NUMBER (yoffset, 2); s = XSCREEN (screen); if (s->output_method == output_x_window) x_set_offset (s, XINT (xoffset), XINT (yoffset)); return Qt; } /* Test if column *x, row *y is within window *w. If they are not, return 0; if they are on the window's modeline, return -1; if they are in the window's text area (the only other alternative) set *x and *y to their locations relative to the upper left corner of the window, and return 1. */ int coordinates_in_window (w, x, y) register struct window *w; register int *x, *y; { register int left = XINT (w->left); register int width = XINT (w->width); register int window_height = XINT (w->height); register int top = XFASTINT (w->top); if (*x < left || *x >= left + width || *y < top || *y > top + window_height - 1) return 0; if (*y == top + window_height - 1 && window_height > 1) /* 1 line => minibuffer */ /* in modeline */ return -1; *x -= left; *y -= top; return 1; } DEFUN ("coordinates-in-window-p", Fcoordinates_in_window_p, Scoordinates_in_window_p, 2, 2, 0, "Return non-nil if COORDINATES are in WINDOW.\n\ COORDINATES is a cons of the form (X Y), X and Y being screen-relative.\n\ If COORDINATES are in the text portion of WINDOW, the coordinates relative\n\ to the window are returned. If they are in the modeline of WINDOW, t is\n\ returned.") (coordinates, window) register Lisp_Object coordinates, window; { int x, y; CHECK_WINDOW (window, 0); CHECK_CONS (coordinates, 1); x = XINT (Fcar (coordinates)); y = XINT (Fcar (Fcdr (coordinates))); switch (coordinates_in_window (XWINDOW (window), &x, &y)) { case -1: /* In modeline of window. */ return Qt; case 0: /* NOT in window at all. */ return Qnil; case 1: /* In text part of window. */ return Fcons (x, Fcons (y, Qnil)); default: abort (); } } #ifndef HAVE_X11 DEFUN ("rubber-band-rectangle", Frubber_band_rectangle, Srubber_band_rectangle, 3, 3, "", "Ask user to specify a window position and size on SCREEN with the mouse.\n\ Arguments are SCREEN, NAME and GEO. NAME is a name to be displayed as\n\ the purpose of this rectangle. GEO is an X-windows size spec that can\n\ specify defaults for some sizes/positions. If GEO specifies everything,\n\ the mouse is not used.\n\ Returns a list of five values: (SCREEN LEFT TOP WIDTH HEIGHT).") (screen, name, geo) Lisp_Object screen; Lisp_Object name; Lisp_Object geo; { int vals[4]; Lisp_Object nums[4]; int i; CHECK_SCREEN (screen, 0); CHECK_STRING (name, 1); CHECK_STRING (geo, 2); switch (XSCREEN (screen)->output_method) { case output_x_window: x_rubber_band (XSCREEN (screen), &vals[0], &vals[1], &vals[2], &vals[3], XSTRING (geo)->data, XSTRING (name)->data); break; default: return Qnil; } for (i = 0; i < 4; i++) XFASTINT (nums[i]) = vals[i]; return Fcons (screen, Flist (4, nums)); return Qnil; } #endif /* not HAVE_X11 */ choose_minibuf_screen () { /* For lowest-level minibuf, put it on currently selected screen if screen has a minibuffer. */ if (minibuf_level == 0 && selected_screen != 0 && !EQ (minibuf_window, selected_screen->minibuffer_window) && !EQ (Qnil, selected_screen->minibuffer_window)) { Fset_window_buffer (selected_screen->minibuffer_window, XWINDOW (minibuf_window)->buffer); minibuf_window = selected_screen->minibuffer_window; } } syms_of_screen () { Qscreenp = intern ("screenp"); staticpro (&Vscreen_list); DEFVAR_LISP ("terminal-screen", &Vterminal_screen, "The initial screen-object, which represents Emacs's stdout."); DEFVAR_LISP ("emacs-iconified", &Vemacs_iconified, "Non-nil if all of emacs is iconified and not screen updates are needed."); Vemacs_iconified = Qnil; DEFVAR_LISP ("global-minibuffer-screen", &Vglobal_minibuffer_screen, "A screen whose minibuffer is used by minibufferless screens.\n\ When you create a minibufferless screen, by default it will use the\n\ minibuffer of this screen. It is up to you to create a suitable screen\n\ and store it in this variable."); Vglobal_minibuffer_screen = Qnil; DEFVAR_LISP ("default-screen-alist", &Vdefault_screen_alist, "Alist of default values for screen creation.\n\ These may be set in your init file, like this:\n\ (setq default-screen-alist '((width . 80) (height . 55)))\n\ These override values given in window system configuration data, like\n\ X Windows' defaults database.\n\ For values specific to the first emacs screen, see initial-screen-alist.\n\ For values specific to the separate minibuffer screen, see\n\ minibuffer-screen-alist."); Vdefault_screen_alist = Qnil; defsubr (&Sscreenp); defsubr (&Sselect_screen); defsubr (&Sselected_screen); defsubr (&Swindow_screen); defsubr (&Sscreen_root_window); defsubr (&Sscreen_selected_window); defsubr (&Sscreen_list); defsubr (&Snext_screen); defsubr (&Sdelete_screen); defsubr (&Sread_mouse_position); defsubr (&Sset_mouse_position); #if 0 defsubr (&Sscreen_configuration); defsubr (&Srestore_screen_configuration); #endif defsubr (&Smake_screen_visible); defsubr (&Smake_screen_invisible); defsubr (&Siconify_screen); defsubr (&Sdeiconify_screen); defsubr (&Sscreen_visible_p); defsubr (&Svisible_screen_list); defsubr (&Sredirect_screen_focus); defsubr (&Sscreen_focus); defsubr (&Sscreen_parameters); defsubr (&Smodify_screen_parameters); defsubr (&Sscreen_pixel_size); defsubr (&Sscreen_height); defsubr (&Sscreen_width); defsubr (&Sset_screen_height); defsubr (&Sset_screen_width); defsubr (&Sset_screen_size); defsubr (&Sset_screen_position); defsubr (&Scoordinates_in_window_p); #ifndef HAVE_X11 defsubr (&Srubber_band_rectangle); #endif /* HAVE_X11 */ }