1991-07-03 12:10:07 +00:00
|
|
|
|
/* Updating of data structures for redisplay.
|
Add support for large files, 64-bit Solaris, system locale codings.
* Makefile.in (emacs): Set the LC_ALL environment variable to "C"
when dumping, so that the dumped Emacs doesn't have stray locale info.
(dired.o): Depend on systime.h.
(editfns.o): Depend on coding.h.
* alloc.c, buffer.c, callproc.c, ccl.c, charset.c, coding.c, data.c,
dispnew.c, editfns.c, emacs.c, filelock.c, floatfns.c, hftctl.c,
keyboard.c, process.c, sysdep.c, unexelf.c, unexhp9k800.c,
unexsunos4.c, vmsfns.c, vmsgmalloc.c, w32faces.c, w32menu.c, w32term.c,
w32xfns.c, xfaces.c, xfns.c, xmenu.c, xterm.c:
Include <config.h> before any system include files.
* alloc.c, buffer.c, ccl.c, data.c, editfns.c, emacs.c, eval.c,
fileio.c, filelock.c, frame.c, insdel.c, keymap.c, lread.c,
m/alpha.h, print.c, search.c, sysdep.c, xdisp.c, xfaces.c, xfns.c,
xmenu.c, xterm.c:
Do not include <stdlib.h>, as <config.h> does this now.
* callproc.c (Fcall_process):
Synchronize messages locale before invoking strerror.
Decode resulting string with locale-coding-system.
* coding.c (Vlocale_coding_system): New var.
(syms_of_coding): Adjust to above change.
(emacs_strerror): New function.
* coding.h (emacs_strerror, Vlocale_coding_system): New decls.
* config.in (HAVE_STDIO_EXT_H, HAVE_TM_GMTOFF, HAVE___FPENDING,
HAVE_FTELLO, HAVE_GETLOADAVG, HAVE_MBLEN, HAVE_MBRLEN,
HAVE_STRSIGNAL): New macros.
(BITS_PER_LONG): Default to 64 if _LP64 is defined.
<stdlib.h>: Include if HAVE_STDLIB_H is defined and NOT_C_CODE isn't.
* dired.c: Include "systime.h".
(Ffile_attributes): Do not cast s.st_size to int; this loses
information if int is 32 bits but st_size and EMACS_INT are larger.
Treat large device numbers like large inode numbers.
* dispnew.c (PENDING_OUTPUT_COUNT): Use __fpending if available.
* editfns.c: Include coding.h.
(emacs_strftime): Remove decl.
(emacs_strftimeu): New decl.
(emacs_memftimeu): Renamed from emacs_memftime; new arg UT.
Use emacs_strftimeu instead of emacs_strftime.
(Fformat_time_string): Convert format string using
Vlocale_coding_system, and convert result back. Synchronize time
locale before invoking lower level function. Invoke
emacs_memftimeu, passing ut, instead of emacs_memftime.
* emacs.c: Include <locale.h> if HAVE_SETLOCALE is defined.
(Vmessages_locale, Vprevious_messages_locale, Vtime_locale,
Vprevious_time_locale): New variables.
(main): Invoke setlocale early, so that initial error messages are
localized properly. But skip locale-setting if LC_ALL is "C".
Fix up locale when it's safe to do so.
(fixup_locale): Moved here from xterm.c.
(synchronize_locale, synchronize_time_locale,
synchronize_messages_locale): New functions.
(syms_of_emacs): Accommodate above changes.
* fileio.c (report_file_error): Convert strerror output according
to Vlocale_coding_system.
(Finsert_file_contents): Check for arithmetic overflow in
computations that depend on file size. Report IO errors
with emacs_strerror, not strerror.
* fns.c (Fgethash): Declare dflt parameter.
* gmalloc.c: Do not define const to nothing if HAVE_CONFIG_H
is defined; that's config.h's job.
* lisp.h (EMACS_INT, BITS_PER_EMACS_INT, EMACS_UINT): If _LP64,
default these values to long, BITS_PER_LONG, and unsigned long.
(VALBITS, MARKBIT, XINT): Do not assume 32-bit EMACS_INT.
(PNTR_COMPARISON_TYPE): Default to EMACS_UINT, not to unsigned int.
(code_convert_string_norecord, fixup_locale,
synchronize_messages_locale, synchronize_time_locale,
emacs_open, emacs_close, emacs_read, emacs_write): New decls.
All Emacs callers of open, close, read, write changed to use
emacs_open, emacs_close, emacs_read, emacs_write.
* lread.c (file_offset, file_tell): New macros. All uses of ftell
changed to file_tell.
(saved_doc_string_position, prev_saved_doc_string_position): Now
of type file_offset.
(init_lread): Do not fix locale here; fixup_locale now does this.
* m/amdahl.h, s/usg5-4.h:
(NSIG): Remove.
(NSIG_MINIMUM): New macro.
* m/cydra5.h, m/dpx2.h, m/mips.h, m/pfa50.h, m/sps7.h, m/stride.h,
m/ustation.h, s/gnu-linux.h, s/hpux.h, s/iris3-5.h, s/iris3-6.h,
s/umips.h, s/usg5-4.h:
(SIGIO): Do not undef.
(BROKEN_SIGIO): New macro.
* m/ustation.h:
(SIGTSTP): Do not undef.
(BROKEN_SIGTSTP): New macro.
* s/gnu-linux.h:
(SIGPOLL, SIGURG): Do not undef.
(BROKEN_SIGPOLL, BROKEN_SIGURG): New macros.
* s/ptx4.h:
(SIGINFO): Do not undef.
(BROKEN_SIGINFO): New macros.
* m/delta.h, s/ptx.h, s/template.h: Doc fix.
* mktime.c, strftime.c: Update to glibc 2.1.2 version, with
some Emacs-related changes merged.
* print.c (float_to_string): Prepend "-" to representation of a
NaN if the NaN is negative.
* process.c (sys_siglist): Omit if HAVE_STRSIGNAL.
(wait_reading_process_input): Use emacs_strerror, not strerror.
* process.c (status_message, sigchld_handler): Synchronize locale,
then use strsignal istead of sys_siglist.
* w32proc.c (sys_wait): Likewise.
* s/aix3-1.h, s/bsd4-1.h, s/dgux.h, s/gnu-linux.h, s/hiuxmpp.h,
s/hpux.h, s/iris3-5.h, s/iris3-6.h, s/irix3-3.h, s/osf1.h, s/rtu.h,
s/sunos4-1.h, s/unipl5-0.h, s/unipl5-2.h, s/usg5-0.h, s/usg5-2-2.h,
s/usg5-2.h, s/usg5-3.h, s/xenix.h:
(open, close, read, write, INTERRUPTIBLE_OPEN,
INTERRUPTIBLE_CLOSE, INTERRUPTIBLE_IO): Remove.
* s/sol2-5.h (_LARGEFILE_SOURCE, _FILE_OFFSET_BITS): New macros.
* sysdep.c (sys_read, sys_write, read, write, sys_close, close,
sys_open, open): Remove.
(emacs_open, emacs_close, emacs_read, emacs_write): Always define;
the old INTERRUPTIBLE_OPEN, INTERRUPTIBLE_CLOSE, and INTERRUPTIBLE_IO
macros are no longer used.
(emacs_open): Renamed from sys_open. Merge BSD4_1 version.
(emacs_close): Renamed from sys_close.
(emacs_read): Renamed from sys_read.
(emacs_write): Renamed from sys_write.
(sys_siglist): Do not declare if HAVE_STRSIGNAL.
(dup2): Do not print error on failure; the real dup2 doesn't.
(strsignal): New function, defined if !HAVE_STRSIGNAL.
* syssignal.h (SIGINFO): Undef if defined and if BROKEN_SIGINFO
is defined.
(SIGIO, SIGPOLL, SIGTSTP, SIGURG): Likewise.
(NSIG): If less than NSIG_MINIMUM, define to NSIG_MINIMUM.
(strsignal): Declare if !HAVE_STRSIGNAL.
* unexelf.c (ElfBitsW, ELFSIZE, ElfExpandBitsW): New macros.
(ElfW): Define in terms of ElfExpandBitsW.
* w32proc.c (sys_siglist): Remove decl.
* xdisp.c (decode_mode_spec): 3rd arg is int, not char, to comply
with ANSI C.
(display_string): Declare face_string_pos arg.
* xfns.c (Fx_show_tip): Declare timeout param.
* xterm.c: No need to include locale.h.
(x_alloc_lighter_color, x_setup_relief_color):
Pass arg as double, not float, for compatibility with ANSI C.
(fixup_locale): Move to emacs.c.
(x_term_init): Do not setlocale or fixup locale; the main program
does this now.
1999-10-19 07:25:11 +00:00
|
|
|
|
Copyright (C) 1985, 86, 87, 88, 93, 94, 95, 97, 98, 1999
|
1997-07-18 17:50:12 +00:00
|
|
|
|
Free Software Foundation, Inc.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
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
|
* window.c (window_internal_width): New function.
* lisp.h (window_internal_height, window_internal_width): Add
extern declarations for these.
* dispnew.c (direct_output_for_insert, direct_output_forward_char,
buffer_posn_from_coords): Use window_internal_width instead of
writing out its definition.
* indent.c (compute_motion): Doc fix; mention scrollbars and
window_internal_width.
(pos_tab_offset, Fvertical_motion): Use window_internal_width
instead of writing it out.
* window.c (Fpos_visible_in_window_p, Fwindow_width, Fscroll_left,
Fscroll_right): Same.
* xdisp.c (redisplay, try_window, try_window_id,
display_text_line): Same.
1993-01-14 15:10:14 +00:00
|
|
|
|
the Free Software Foundation; either version 2, or (at your option)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
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
|
1996-01-15 09:18:04 +00:00
|
|
|
|
the Free Software Foundation, Inc., 59 Temple Place - Suite 330,
|
|
|
|
|
Boston, MA 02111-1307, USA. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1993-09-10 06:15:46 +00:00
|
|
|
|
#include <config.h>
|
Add support for large files, 64-bit Solaris, system locale codings.
* Makefile.in (emacs): Set the LC_ALL environment variable to "C"
when dumping, so that the dumped Emacs doesn't have stray locale info.
(dired.o): Depend on systime.h.
(editfns.o): Depend on coding.h.
* alloc.c, buffer.c, callproc.c, ccl.c, charset.c, coding.c, data.c,
dispnew.c, editfns.c, emacs.c, filelock.c, floatfns.c, hftctl.c,
keyboard.c, process.c, sysdep.c, unexelf.c, unexhp9k800.c,
unexsunos4.c, vmsfns.c, vmsgmalloc.c, w32faces.c, w32menu.c, w32term.c,
w32xfns.c, xfaces.c, xfns.c, xmenu.c, xterm.c:
Include <config.h> before any system include files.
* alloc.c, buffer.c, ccl.c, data.c, editfns.c, emacs.c, eval.c,
fileio.c, filelock.c, frame.c, insdel.c, keymap.c, lread.c,
m/alpha.h, print.c, search.c, sysdep.c, xdisp.c, xfaces.c, xfns.c,
xmenu.c, xterm.c:
Do not include <stdlib.h>, as <config.h> does this now.
* callproc.c (Fcall_process):
Synchronize messages locale before invoking strerror.
Decode resulting string with locale-coding-system.
* coding.c (Vlocale_coding_system): New var.
(syms_of_coding): Adjust to above change.
(emacs_strerror): New function.
* coding.h (emacs_strerror, Vlocale_coding_system): New decls.
* config.in (HAVE_STDIO_EXT_H, HAVE_TM_GMTOFF, HAVE___FPENDING,
HAVE_FTELLO, HAVE_GETLOADAVG, HAVE_MBLEN, HAVE_MBRLEN,
HAVE_STRSIGNAL): New macros.
(BITS_PER_LONG): Default to 64 if _LP64 is defined.
<stdlib.h>: Include if HAVE_STDLIB_H is defined and NOT_C_CODE isn't.
* dired.c: Include "systime.h".
(Ffile_attributes): Do not cast s.st_size to int; this loses
information if int is 32 bits but st_size and EMACS_INT are larger.
Treat large device numbers like large inode numbers.
* dispnew.c (PENDING_OUTPUT_COUNT): Use __fpending if available.
* editfns.c: Include coding.h.
(emacs_strftime): Remove decl.
(emacs_strftimeu): New decl.
(emacs_memftimeu): Renamed from emacs_memftime; new arg UT.
Use emacs_strftimeu instead of emacs_strftime.
(Fformat_time_string): Convert format string using
Vlocale_coding_system, and convert result back. Synchronize time
locale before invoking lower level function. Invoke
emacs_memftimeu, passing ut, instead of emacs_memftime.
* emacs.c: Include <locale.h> if HAVE_SETLOCALE is defined.
(Vmessages_locale, Vprevious_messages_locale, Vtime_locale,
Vprevious_time_locale): New variables.
(main): Invoke setlocale early, so that initial error messages are
localized properly. But skip locale-setting if LC_ALL is "C".
Fix up locale when it's safe to do so.
(fixup_locale): Moved here from xterm.c.
(synchronize_locale, synchronize_time_locale,
synchronize_messages_locale): New functions.
(syms_of_emacs): Accommodate above changes.
* fileio.c (report_file_error): Convert strerror output according
to Vlocale_coding_system.
(Finsert_file_contents): Check for arithmetic overflow in
computations that depend on file size. Report IO errors
with emacs_strerror, not strerror.
* fns.c (Fgethash): Declare dflt parameter.
* gmalloc.c: Do not define const to nothing if HAVE_CONFIG_H
is defined; that's config.h's job.
* lisp.h (EMACS_INT, BITS_PER_EMACS_INT, EMACS_UINT): If _LP64,
default these values to long, BITS_PER_LONG, and unsigned long.
(VALBITS, MARKBIT, XINT): Do not assume 32-bit EMACS_INT.
(PNTR_COMPARISON_TYPE): Default to EMACS_UINT, not to unsigned int.
(code_convert_string_norecord, fixup_locale,
synchronize_messages_locale, synchronize_time_locale,
emacs_open, emacs_close, emacs_read, emacs_write): New decls.
All Emacs callers of open, close, read, write changed to use
emacs_open, emacs_close, emacs_read, emacs_write.
* lread.c (file_offset, file_tell): New macros. All uses of ftell
changed to file_tell.
(saved_doc_string_position, prev_saved_doc_string_position): Now
of type file_offset.
(init_lread): Do not fix locale here; fixup_locale now does this.
* m/amdahl.h, s/usg5-4.h:
(NSIG): Remove.
(NSIG_MINIMUM): New macro.
* m/cydra5.h, m/dpx2.h, m/mips.h, m/pfa50.h, m/sps7.h, m/stride.h,
m/ustation.h, s/gnu-linux.h, s/hpux.h, s/iris3-5.h, s/iris3-6.h,
s/umips.h, s/usg5-4.h:
(SIGIO): Do not undef.
(BROKEN_SIGIO): New macro.
* m/ustation.h:
(SIGTSTP): Do not undef.
(BROKEN_SIGTSTP): New macro.
* s/gnu-linux.h:
(SIGPOLL, SIGURG): Do not undef.
(BROKEN_SIGPOLL, BROKEN_SIGURG): New macros.
* s/ptx4.h:
(SIGINFO): Do not undef.
(BROKEN_SIGINFO): New macros.
* m/delta.h, s/ptx.h, s/template.h: Doc fix.
* mktime.c, strftime.c: Update to glibc 2.1.2 version, with
some Emacs-related changes merged.
* print.c (float_to_string): Prepend "-" to representation of a
NaN if the NaN is negative.
* process.c (sys_siglist): Omit if HAVE_STRSIGNAL.
(wait_reading_process_input): Use emacs_strerror, not strerror.
* process.c (status_message, sigchld_handler): Synchronize locale,
then use strsignal istead of sys_siglist.
* w32proc.c (sys_wait): Likewise.
* s/aix3-1.h, s/bsd4-1.h, s/dgux.h, s/gnu-linux.h, s/hiuxmpp.h,
s/hpux.h, s/iris3-5.h, s/iris3-6.h, s/irix3-3.h, s/osf1.h, s/rtu.h,
s/sunos4-1.h, s/unipl5-0.h, s/unipl5-2.h, s/usg5-0.h, s/usg5-2-2.h,
s/usg5-2.h, s/usg5-3.h, s/xenix.h:
(open, close, read, write, INTERRUPTIBLE_OPEN,
INTERRUPTIBLE_CLOSE, INTERRUPTIBLE_IO): Remove.
* s/sol2-5.h (_LARGEFILE_SOURCE, _FILE_OFFSET_BITS): New macros.
* sysdep.c (sys_read, sys_write, read, write, sys_close, close,
sys_open, open): Remove.
(emacs_open, emacs_close, emacs_read, emacs_write): Always define;
the old INTERRUPTIBLE_OPEN, INTERRUPTIBLE_CLOSE, and INTERRUPTIBLE_IO
macros are no longer used.
(emacs_open): Renamed from sys_open. Merge BSD4_1 version.
(emacs_close): Renamed from sys_close.
(emacs_read): Renamed from sys_read.
(emacs_write): Renamed from sys_write.
(sys_siglist): Do not declare if HAVE_STRSIGNAL.
(dup2): Do not print error on failure; the real dup2 doesn't.
(strsignal): New function, defined if !HAVE_STRSIGNAL.
* syssignal.h (SIGINFO): Undef if defined and if BROKEN_SIGINFO
is defined.
(SIGIO, SIGPOLL, SIGTSTP, SIGURG): Likewise.
(NSIG): If less than NSIG_MINIMUM, define to NSIG_MINIMUM.
(strsignal): Declare if !HAVE_STRSIGNAL.
* unexelf.c (ElfBitsW, ELFSIZE, ElfExpandBitsW): New macros.
(ElfW): Define in terms of ElfExpandBitsW.
* w32proc.c (sys_siglist): Remove decl.
* xdisp.c (decode_mode_spec): 3rd arg is int, not char, to comply
with ANSI C.
(display_string): Declare face_string_pos arg.
* xfns.c (Fx_show_tip): Declare timeout param.
* xterm.c: No need to include locale.h.
(x_alloc_lighter_color, x_setup_relief_color):
Pass arg as double, not float, for compatibility with ANSI C.
(fixup_locale): Move to emacs.c.
(x_term_init): Do not setlocale or fixup locale; the main program
does this now.
1999-10-19 07:25:11 +00:00
|
|
|
|
#include <signal.h>
|
1994-06-15 03:54:06 +00:00
|
|
|
|
#include <stdio.h>
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#include <ctype.h>
|
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
1993-06-06 21:16:51 +00:00
|
|
|
|
#include "lisp.h"
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#include "termchar.h"
|
|
|
|
|
#include "termopts.h"
|
1993-03-15 06:06:56 +00:00
|
|
|
|
#include "termhooks.h"
|
1995-11-12 05:20:12 +00:00
|
|
|
|
/* cm.h must come after dispextern.h on Windows. */
|
1995-11-07 07:14:59 +00:00
|
|
|
|
#include "dispextern.h"
|
|
|
|
|
#include "cm.h"
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#include "buffer.h"
|
1997-02-20 06:35:00 +00:00
|
|
|
|
#include "charset.h"
|
2000-08-22 22:14:00 +00:00
|
|
|
|
#include "keyboard.h"
|
1992-07-13 19:54:34 +00:00
|
|
|
|
#include "frame.h"
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#include "window.h"
|
|
|
|
|
#include "commands.h"
|
|
|
|
|
#include "disptab.h"
|
|
|
|
|
#include "indent.h"
|
1993-07-31 21:58:45 +00:00
|
|
|
|
#include "intervals.h"
|
1996-04-23 20:32:58 +00:00
|
|
|
|
#include "blockinput.h"
|
1998-04-14 12:25:56 +00:00
|
|
|
|
#include "process.h"
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1997-02-20 06:35:00 +00:00
|
|
|
|
/* I don't know why DEC Alpha OSF1 fail to compile this file if we
|
|
|
|
|
include the following file. */
|
|
|
|
|
/* #include "systty.h" */
|
1995-08-21 21:56:40 +00:00
|
|
|
|
#include "syssignal.h"
|
1992-02-20 04:15:20 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#ifdef HAVE_X_WINDOWS
|
|
|
|
|
#include "xterm.h"
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#endif /* HAVE_X_WINDOWS */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1995-11-07 07:14:59 +00:00
|
|
|
|
#ifdef HAVE_NTGUI
|
|
|
|
|
#include "w32term.h"
|
|
|
|
|
#endif /* HAVE_NTGUI */
|
|
|
|
|
|
Initial check-in: changes for building Emacs under Mac OS.
2000-10-23 Andrew Choi <akochoi@i-cable.com>
* dispextern.h [macintosh]: Include macgui.h instead of macterm.h.
* dispnew.c [macintosh]: Include macterm.h.
(init_display) [macintosh]: initialization for window system.
* emacs.c (main) [macintosh]: Call syms_of_textprop,
syms_of_macfns, syms_of_ccl, syms_of_fontset, syms_of_xterm,
syms_of_search, x_term_init, and init_keyboard before calling
init_window_once. Also, call syms_of_xmenu.
* fontset.c (syms_of_fontset) [macintosh]: Set ASCII font of
default fontset to Monaco.
* frame.c [macintosh]: Include macterm.h. Remove declarations of
NewMacWindow and DisposeMacWindow.
(make_terminal_frame) [macintosh]: Call make_mac_terminal_frame
instead of calling NewMacWindow and setting fields of
f->output_data.mac directly. Call init_frame_faces.
(Fdelete_frame) [macintosh]: Remove unused code.
(Fmodify_frame_parameters) [macintosh]: Call
x_set_frame_parameters instead of mac_set_frame_parameters.
* frame.h [macintosh]: Define menu_bar_lines field in struct
frame. Define FRAME_EXTERNAL_MENU_BAR macro.
* keyboard.c [macintosh]: Include macterm.h.
(kbd_buffer_get_event) [macintosh]: Generate delete_window_event
and menu_bar_activate_event type events as for X and NT.
(make_lispy_event) [macintosh]: Construct lisp events of type
MENU_BAR_EVENT as for X and NT.
* sysdep.c [macintosh]: Remove declaration for sys_signal.
Include stdlib.h. Remove definition of Vx_bitmap_file_path.
(sys_subshell) [macintosh]: Remove definition entirely.
(init_sys_modes) [macintosh]: Do not initialize Vwindow_system and
Vwindow_system_version here. Remove initialization of
Vx_bitmap_file_path.
(read_input_waiting): Correct the number of parameters passed to
read_socket_hook.
Move all Macintosh functions to mac/mac.c.
* term.c [macintosh]: Include macterm.h.
* window.c [macintosh]: Include macterm.h.
* xdisp.c [macintosh]: Include macterm.h. Declare
set_frame_menubar and pending_menu_activation.
(echo_area_display) [macintosh]: Do not return if terminal frame
is the selected frame.
(update_menu_bar) [macintosh]: Check FRAME_EXTERNAL_MENU_BAR (f).
Allow only the selected frame to set menu bar.
(redisplay_window) [macintosh]: Obtain menu bar to redisplay by
calling FRAME_EXTERNAL_MENU_BAR (f).
(display_menu_bar) [macintosh]: Check FRAME_MAC_P (f).
* xfaces.c [macintosh]: Include macterm.h. Define x_display_info
and check_x. Declare XCreateGC. Define x_create_gc and
x_free_gc. Initialize font_sort_order.
(x_face_list_fonts) [macintosh]: Use the same code as WINDOWSNT,
but call x_list_fonts instead of w32_list_fonts.
(Finternal_face_x_get_resource) [macintosh]: Do not call
display_x_get_resource.
(prepare_face_for_display) [macintosh]: Set xgcv.font.
(realize_x_face) [macintosh]: Load the font if it is specified in
ATTRS.
(syms_of_xfaces) [macintosh]: Initialize Vscalable_fonts_allowed
to Qt.
* cus-edit.el (custom-button-face): Use 3D look for mac.
(custom-button-pressed-face): Likewise.
* faces.el (set-face-attributes-from-resources): Handle mac frames
in the same way as x and w32 frames.
(face-valid-attribute-values): Likewise.
(read-face-attribute): Likewise.
(defined-colors): Likewise.
(color-defined-p): Likewise.
(color-values): Likewise.
(display-grayscale-p): Likewise.
(face-set-after-frame-default): Likewise.
(mode-line): Same default face as for x and w32.
(tool-bar): Likewise.
* frame.el: Remove call to frame-notice-user-settings at end of
the file.
* info.el (Info-fontify-node): make underlines invisible for mac
as for x, pc, and w32 frame types.
* term/mac-win.el: New file.
2000-10-22 16:50:16 +00:00
|
|
|
|
#ifdef macintosh
|
|
|
|
|
#include "macterm.h"
|
|
|
|
|
#endif /* macintosh */
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Include systime.h after xterm.h to avoid double inclusion of time.h. */
|
1994-06-05 18:29:46 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#include "systime.h"
|
1994-05-19 18:42:37 +00:00
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* To get the prototype for `sleep'. */
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#define max(a, b) ((a) > (b) ? (a) : (b))
|
|
|
|
|
#define min(a, b) ((a) < (b) ? (a) : (b))
|
|
|
|
|
|
|
|
|
|
/* Get number of chars of output now in the buffer of a stdio stream.
|
1999-07-21 21:43:52 +00:00
|
|
|
|
This ought to be built in in stdio, but it isn't. Some s- files
|
|
|
|
|
override this because their stdio internals differ. */
|
|
|
|
|
|
1993-12-08 23:49:24 +00:00
|
|
|
|
#ifdef __GNU_LIBRARY__
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* The s- file might have overridden the definition with one that
|
|
|
|
|
works for the system's C library. But we are using the GNU C
|
|
|
|
|
library, so this is the right definition for every system. */
|
|
|
|
|
|
1994-05-19 18:42:37 +00:00
|
|
|
|
#ifdef GNU_LIBRARY_PENDING_OUTPUT_COUNT
|
|
|
|
|
#define PENDING_OUTPUT_COUNT GNU_LIBRARY_PENDING_OUTPUT_COUNT
|
|
|
|
|
#else
|
1994-05-10 22:39:29 +00:00
|
|
|
|
#undef PENDING_OUTPUT_COUNT
|
1993-12-08 23:49:24 +00:00
|
|
|
|
#define PENDING_OUTPUT_COUNT(FILE) ((FILE)->__bufp - (FILE)->__buffer)
|
1994-05-19 18:42:37 +00:00
|
|
|
|
#endif
|
|
|
|
|
#else /* not __GNU_LIBRARY__ */
|
Add support for large files, 64-bit Solaris, system locale codings.
* Makefile.in (emacs): Set the LC_ALL environment variable to "C"
when dumping, so that the dumped Emacs doesn't have stray locale info.
(dired.o): Depend on systime.h.
(editfns.o): Depend on coding.h.
* alloc.c, buffer.c, callproc.c, ccl.c, charset.c, coding.c, data.c,
dispnew.c, editfns.c, emacs.c, filelock.c, floatfns.c, hftctl.c,
keyboard.c, process.c, sysdep.c, unexelf.c, unexhp9k800.c,
unexsunos4.c, vmsfns.c, vmsgmalloc.c, w32faces.c, w32menu.c, w32term.c,
w32xfns.c, xfaces.c, xfns.c, xmenu.c, xterm.c:
Include <config.h> before any system include files.
* alloc.c, buffer.c, ccl.c, data.c, editfns.c, emacs.c, eval.c,
fileio.c, filelock.c, frame.c, insdel.c, keymap.c, lread.c,
m/alpha.h, print.c, search.c, sysdep.c, xdisp.c, xfaces.c, xfns.c,
xmenu.c, xterm.c:
Do not include <stdlib.h>, as <config.h> does this now.
* callproc.c (Fcall_process):
Synchronize messages locale before invoking strerror.
Decode resulting string with locale-coding-system.
* coding.c (Vlocale_coding_system): New var.
(syms_of_coding): Adjust to above change.
(emacs_strerror): New function.
* coding.h (emacs_strerror, Vlocale_coding_system): New decls.
* config.in (HAVE_STDIO_EXT_H, HAVE_TM_GMTOFF, HAVE___FPENDING,
HAVE_FTELLO, HAVE_GETLOADAVG, HAVE_MBLEN, HAVE_MBRLEN,
HAVE_STRSIGNAL): New macros.
(BITS_PER_LONG): Default to 64 if _LP64 is defined.
<stdlib.h>: Include if HAVE_STDLIB_H is defined and NOT_C_CODE isn't.
* dired.c: Include "systime.h".
(Ffile_attributes): Do not cast s.st_size to int; this loses
information if int is 32 bits but st_size and EMACS_INT are larger.
Treat large device numbers like large inode numbers.
* dispnew.c (PENDING_OUTPUT_COUNT): Use __fpending if available.
* editfns.c: Include coding.h.
(emacs_strftime): Remove decl.
(emacs_strftimeu): New decl.
(emacs_memftimeu): Renamed from emacs_memftime; new arg UT.
Use emacs_strftimeu instead of emacs_strftime.
(Fformat_time_string): Convert format string using
Vlocale_coding_system, and convert result back. Synchronize time
locale before invoking lower level function. Invoke
emacs_memftimeu, passing ut, instead of emacs_memftime.
* emacs.c: Include <locale.h> if HAVE_SETLOCALE is defined.
(Vmessages_locale, Vprevious_messages_locale, Vtime_locale,
Vprevious_time_locale): New variables.
(main): Invoke setlocale early, so that initial error messages are
localized properly. But skip locale-setting if LC_ALL is "C".
Fix up locale when it's safe to do so.
(fixup_locale): Moved here from xterm.c.
(synchronize_locale, synchronize_time_locale,
synchronize_messages_locale): New functions.
(syms_of_emacs): Accommodate above changes.
* fileio.c (report_file_error): Convert strerror output according
to Vlocale_coding_system.
(Finsert_file_contents): Check for arithmetic overflow in
computations that depend on file size. Report IO errors
with emacs_strerror, not strerror.
* fns.c (Fgethash): Declare dflt parameter.
* gmalloc.c: Do not define const to nothing if HAVE_CONFIG_H
is defined; that's config.h's job.
* lisp.h (EMACS_INT, BITS_PER_EMACS_INT, EMACS_UINT): If _LP64,
default these values to long, BITS_PER_LONG, and unsigned long.
(VALBITS, MARKBIT, XINT): Do not assume 32-bit EMACS_INT.
(PNTR_COMPARISON_TYPE): Default to EMACS_UINT, not to unsigned int.
(code_convert_string_norecord, fixup_locale,
synchronize_messages_locale, synchronize_time_locale,
emacs_open, emacs_close, emacs_read, emacs_write): New decls.
All Emacs callers of open, close, read, write changed to use
emacs_open, emacs_close, emacs_read, emacs_write.
* lread.c (file_offset, file_tell): New macros. All uses of ftell
changed to file_tell.
(saved_doc_string_position, prev_saved_doc_string_position): Now
of type file_offset.
(init_lread): Do not fix locale here; fixup_locale now does this.
* m/amdahl.h, s/usg5-4.h:
(NSIG): Remove.
(NSIG_MINIMUM): New macro.
* m/cydra5.h, m/dpx2.h, m/mips.h, m/pfa50.h, m/sps7.h, m/stride.h,
m/ustation.h, s/gnu-linux.h, s/hpux.h, s/iris3-5.h, s/iris3-6.h,
s/umips.h, s/usg5-4.h:
(SIGIO): Do not undef.
(BROKEN_SIGIO): New macro.
* m/ustation.h:
(SIGTSTP): Do not undef.
(BROKEN_SIGTSTP): New macro.
* s/gnu-linux.h:
(SIGPOLL, SIGURG): Do not undef.
(BROKEN_SIGPOLL, BROKEN_SIGURG): New macros.
* s/ptx4.h:
(SIGINFO): Do not undef.
(BROKEN_SIGINFO): New macros.
* m/delta.h, s/ptx.h, s/template.h: Doc fix.
* mktime.c, strftime.c: Update to glibc 2.1.2 version, with
some Emacs-related changes merged.
* print.c (float_to_string): Prepend "-" to representation of a
NaN if the NaN is negative.
* process.c (sys_siglist): Omit if HAVE_STRSIGNAL.
(wait_reading_process_input): Use emacs_strerror, not strerror.
* process.c (status_message, sigchld_handler): Synchronize locale,
then use strsignal istead of sys_siglist.
* w32proc.c (sys_wait): Likewise.
* s/aix3-1.h, s/bsd4-1.h, s/dgux.h, s/gnu-linux.h, s/hiuxmpp.h,
s/hpux.h, s/iris3-5.h, s/iris3-6.h, s/irix3-3.h, s/osf1.h, s/rtu.h,
s/sunos4-1.h, s/unipl5-0.h, s/unipl5-2.h, s/usg5-0.h, s/usg5-2-2.h,
s/usg5-2.h, s/usg5-3.h, s/xenix.h:
(open, close, read, write, INTERRUPTIBLE_OPEN,
INTERRUPTIBLE_CLOSE, INTERRUPTIBLE_IO): Remove.
* s/sol2-5.h (_LARGEFILE_SOURCE, _FILE_OFFSET_BITS): New macros.
* sysdep.c (sys_read, sys_write, read, write, sys_close, close,
sys_open, open): Remove.
(emacs_open, emacs_close, emacs_read, emacs_write): Always define;
the old INTERRUPTIBLE_OPEN, INTERRUPTIBLE_CLOSE, and INTERRUPTIBLE_IO
macros are no longer used.
(emacs_open): Renamed from sys_open. Merge BSD4_1 version.
(emacs_close): Renamed from sys_close.
(emacs_read): Renamed from sys_read.
(emacs_write): Renamed from sys_write.
(sys_siglist): Do not declare if HAVE_STRSIGNAL.
(dup2): Do not print error on failure; the real dup2 doesn't.
(strsignal): New function, defined if !HAVE_STRSIGNAL.
* syssignal.h (SIGINFO): Undef if defined and if BROKEN_SIGINFO
is defined.
(SIGIO, SIGPOLL, SIGTSTP, SIGURG): Likewise.
(NSIG): If less than NSIG_MINIMUM, define to NSIG_MINIMUM.
(strsignal): Declare if !HAVE_STRSIGNAL.
* unexelf.c (ElfBitsW, ELFSIZE, ElfExpandBitsW): New macros.
(ElfW): Define in terms of ElfExpandBitsW.
* w32proc.c (sys_siglist): Remove decl.
* xdisp.c (decode_mode_spec): 3rd arg is int, not char, to comply
with ANSI C.
(display_string): Declare face_string_pos arg.
* xfns.c (Fx_show_tip): Declare timeout param.
* xterm.c: No need to include locale.h.
(x_alloc_lighter_color, x_setup_relief_color):
Pass arg as double, not float, for compatibility with ANSI C.
(fixup_locale): Move to emacs.c.
(x_term_init): Do not setlocale or fixup locale; the main program
does this now.
1999-10-19 07:25:11 +00:00
|
|
|
|
#if !defined (PENDING_OUTPUT_COUNT) && HAVE_STDIO_EXT_H && HAVE___FPENDING
|
|
|
|
|
#include <stdio_ext.h>
|
|
|
|
|
#define PENDING_OUTPUT_COUNT(FILE) __fpending (FILE)
|
|
|
|
|
#endif
|
1994-05-10 22:39:29 +00:00
|
|
|
|
#ifndef PENDING_OUTPUT_COUNT
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#define PENDING_OUTPUT_COUNT(FILE) ((FILE)->_ptr - (FILE)->_base)
|
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#endif /* not __GNU_LIBRARY__ */
|
|
|
|
|
|
2000-06-05 14:22:26 +00:00
|
|
|
|
#if defined (LINUX) && defined (HAVE_LIBNCURSES)
|
|
|
|
|
#include <term.h> /* for tgetent */
|
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Structure to pass dimensions around. Used for character bounding
|
|
|
|
|
boxes, glyph matrix dimensions and alike. */
|
|
|
|
|
|
|
|
|
|
struct dim
|
|
|
|
|
{
|
|
|
|
|
int width;
|
|
|
|
|
int height;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Function prototypes. */
|
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
static void redraw_overlapping_rows P_ ((struct window *, int));
|
|
|
|
|
static void redraw_overlapped_rows P_ ((struct window *, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static int count_blanks P_ ((struct glyph *, int));
|
|
|
|
|
static int count_match P_ ((struct glyph *, struct glyph *,
|
|
|
|
|
struct glyph *, struct glyph *));
|
|
|
|
|
static unsigned line_draw_cost P_ ((struct glyph_matrix *, int));
|
|
|
|
|
static void update_frame_line P_ ((struct frame *, int));
|
|
|
|
|
static struct dim allocate_matrices_for_frame_redisplay
|
|
|
|
|
P_ ((Lisp_Object, int, int, struct dim, int, int *));
|
|
|
|
|
static void allocate_matrices_for_window_redisplay P_ ((struct window *,
|
|
|
|
|
struct dim));
|
|
|
|
|
static int realloc_glyph_pool P_ ((struct glyph_pool *, struct dim));
|
|
|
|
|
static void adjust_frame_glyphs P_ ((struct frame *));
|
|
|
|
|
struct glyph_matrix *new_glyph_matrix P_ ((struct glyph_pool *));
|
|
|
|
|
static void free_glyph_matrix P_ ((struct glyph_matrix *));
|
|
|
|
|
static void adjust_glyph_matrix P_ ((struct window *, struct glyph_matrix *,
|
|
|
|
|
int, int, struct dim));
|
1999-08-21 19:29:02 +00:00
|
|
|
|
static void change_frame_size_1 P_ ((struct frame *, int, int, int, int, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void swap_glyph_pointers P_ ((struct glyph_row *, struct glyph_row *));
|
2000-07-19 15:47:52 +00:00
|
|
|
|
#ifdef GLYPH_DEBUG
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static int glyph_row_slice_p P_ ((struct glyph_row *, struct glyph_row *));
|
2000-07-19 15:47:52 +00:00
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void fill_up_frame_row_with_spaces P_ ((struct glyph_row *, int));
|
|
|
|
|
static void build_frame_matrix_from_window_tree P_ ((struct glyph_matrix *,
|
|
|
|
|
struct window *));
|
|
|
|
|
static void build_frame_matrix_from_leaf_window P_ ((struct glyph_matrix *,
|
|
|
|
|
struct window *));
|
|
|
|
|
static struct glyph_pool *new_glyph_pool P_ ((void));
|
|
|
|
|
static void free_glyph_pool P_ ((struct glyph_pool *));
|
|
|
|
|
static void adjust_frame_glyphs_initially P_ ((void));
|
|
|
|
|
static void adjust_frame_message_buffer P_ ((struct frame *));
|
|
|
|
|
static void adjust_decode_mode_spec_buffer P_ ((struct frame *));
|
|
|
|
|
static void fill_up_glyph_row_with_spaces P_ ((struct glyph_row *));
|
|
|
|
|
static void build_frame_matrix P_ ((struct frame *));
|
|
|
|
|
void clear_current_matrices P_ ((struct frame *));
|
|
|
|
|
void scroll_glyph_matrix_range P_ ((struct glyph_matrix *, int, int,
|
|
|
|
|
int, int));
|
|
|
|
|
static void clear_window_matrices P_ ((struct window *, int));
|
|
|
|
|
static void fill_up_glyph_row_area_with_spaces P_ ((struct glyph_row *, int));
|
|
|
|
|
static int scrolling_window P_ ((struct window *, int));
|
2000-07-12 15:10:30 +00:00
|
|
|
|
static int update_window_line P_ ((struct window *, int, int *));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void update_marginal_area P_ ((struct window *, int, int));
|
1999-08-06 13:59:32 +00:00
|
|
|
|
static int update_text_area P_ ((struct window *, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void make_current P_ ((struct glyph_matrix *, struct glyph_matrix *,
|
|
|
|
|
int));
|
|
|
|
|
static void mirror_make_current P_ ((struct window *, int));
|
|
|
|
|
void check_window_matrix_pointers P_ ((struct window *));
|
1999-08-21 19:29:02 +00:00
|
|
|
|
#if GLYPH_DEBUG
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void check_matrix_pointers P_ ((struct glyph_matrix *,
|
|
|
|
|
struct glyph_matrix *));
|
1999-08-21 19:29:02 +00:00
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void mirror_line_dance P_ ((struct window *, int, int, int *, char *));
|
|
|
|
|
static int update_window_tree P_ ((struct window *, int));
|
|
|
|
|
static int update_window P_ ((struct window *, int));
|
|
|
|
|
static int update_frame_1 P_ ((struct frame *, int, int));
|
|
|
|
|
static void set_window_cursor_after_update P_ ((struct window *));
|
|
|
|
|
static int row_equal_p P_ ((struct window *, struct glyph_row *,
|
2000-07-11 19:57:54 +00:00
|
|
|
|
struct glyph_row *, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void adjust_frame_glyphs_for_window_redisplay P_ ((struct frame *));
|
|
|
|
|
static void adjust_frame_glyphs_for_frame_redisplay P_ ((struct frame *));
|
|
|
|
|
static void reverse_rows P_ ((struct glyph_matrix *, int, int));
|
|
|
|
|
static int margin_glyphs_to_reserve P_ ((struct window *, int, Lisp_Object));
|
2000-02-25 13:20:36 +00:00
|
|
|
|
static void sync_window_with_frame_matrix_rows P_ ((struct window *));
|
|
|
|
|
struct window *frame_row_to_window P_ ((struct window *, int));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Non-zero means don't pause redisplay for pending input. (This is
|
|
|
|
|
for debugging and for a future implementation of EDT-like
|
|
|
|
|
scrolling. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int redisplay_dont_pause;
|
1995-02-17 08:40:18 +00:00
|
|
|
|
|
1992-02-20 04:15:20 +00:00
|
|
|
|
/* Nonzero upon entry to redisplay means do not assume anything about
|
1992-07-13 19:54:34 +00:00
|
|
|
|
current contents of actual terminal frame; clear and redraw it. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
int frame_garbaged;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Nonzero means last display completed. Zero means it was preempted. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
int display_completed;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Lisp variable visible-bell; enables use of screen-flash instead of
|
|
|
|
|
audible bell. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
int visible_bell;
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* Invert the color of the whole frame, at a low level. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
int inverse_video;
|
|
|
|
|
|
|
|
|
|
/* Line speed of the terminal. */
|
|
|
|
|
|
|
|
|
|
int baud_rate;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Either nil or a symbol naming the window system under which Emacs
|
|
|
|
|
is running. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
Lisp_Object Vwindow_system;
|
|
|
|
|
|
|
|
|
|
/* Version number of X windows: 10, 11 or nil. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
Lisp_Object Vwindow_system_version;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Vector of glyph definitions. Indexed by glyph number, the contents
|
|
|
|
|
are a string which is how to output the glyph.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
If Vglyph_table is nil, a glyph is output by using its low 8 bits
|
1999-07-21 21:43:52 +00:00
|
|
|
|
as a character code.
|
|
|
|
|
|
|
|
|
|
This is an obsolete feature that is no longer used. The variable
|
|
|
|
|
is retained for compatibility. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
Lisp_Object Vglyph_table;
|
|
|
|
|
|
|
|
|
|
/* Display table to use for vectors that don't specify their own. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object Vstandard_display_table;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Nonzero means reading single-character input with prompt so put
|
|
|
|
|
cursor on mini-buffer after the prompt. positive means at end of
|
|
|
|
|
text in echo area; negative means at beginning of line. */
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
int cursor_in_echo_area;
|
1995-10-14 05:54:16 +00:00
|
|
|
|
|
2000-08-09 18:39:58 +00:00
|
|
|
|
Lisp_Object Qdisplay_table, Qredisplay_dont_pause;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* The currently selected frame. In a single-frame version, this
|
1999-09-13 11:13:24 +00:00
|
|
|
|
variable always equals the_only_frame. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-09-13 11:13:24 +00:00
|
|
|
|
Lisp_Object selected_frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* A frame which is not just a mini-buffer, or 0 if there are no such
|
1992-07-13 19:54:34 +00:00
|
|
|
|
frames. This is usually the most recent such frame that was
|
1994-10-17 08:42:36 +00:00
|
|
|
|
selected. In a single-frame version, this variable always holds
|
|
|
|
|
the address of the_only_frame. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct frame *last_nonminibuf_frame;
|
1994-04-01 10:13:32 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Stdio stream being used for copy of all output. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
FILE *termscript;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Structure for info on cursor positioning. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct cm Wcm;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* 1 means SIGWINCH happened when not safe. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int delayed_size_change;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* 1 means glyph initialization has been completed at startup. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static int glyphs_initialized_initially_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Updated window if != 0. Set by update_window. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *updated_window;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Glyph row updated in update_window_line, and area that is updated. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *updated_row;
|
|
|
|
|
int updated_area;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* A glyph for a space. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph space_glyph;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Non-zero means update has been performed directly, so that there's
|
|
|
|
|
no need for redisplay_internal to do much work. Set by
|
|
|
|
|
direct_output_for_insert. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int redisplay_performed_directly_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Counts of allocated structures. These counts serve to diagnose
|
|
|
|
|
memory leaks and double frees. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int glyph_matrix_count;
|
|
|
|
|
int glyph_pool_count;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If non-null, the frame whose frame matrices are manipulated. If
|
|
|
|
|
null, window matrices are worked on. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static struct frame *frame_matrix_frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Current interface for window-based redisplay. Set from init_xterm.
|
|
|
|
|
A null value means we are not using window-based redisplay. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct redisplay_interface *rif;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Non-zero means that fonts have been loaded since the last glyph
|
|
|
|
|
matrix adjustments. Redisplay must stop, and glyph matrices must
|
|
|
|
|
be adjusted when this flag becomes non-zero during display. The
|
|
|
|
|
reason fonts can be loaded so late is that fonts of fontsets are
|
|
|
|
|
loaded on demand. */
|
1995-04-13 07:47:27 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int fonts_changed_p;
|
1995-04-13 07:47:27 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Convert vpos and hpos from frame to window and vice versa.
|
|
|
|
|
This may only be used for terminal frames. */
|
1995-04-13 07:47:27 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#if GLYPH_DEBUG
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static int window_to_frame_vpos P_ ((struct window *, int));
|
|
|
|
|
static int window_to_frame_hpos P_ ((struct window *, int));
|
|
|
|
|
#define WINDOW_TO_FRAME_VPOS(W, VPOS) window_to_frame_vpos ((W), (VPOS))
|
|
|
|
|
#define WINDOW_TO_FRAME_HPOS(W, HPOS) window_to_frame_hpos ((W), (HPOS))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#else /* GLYPH_DEBUG == 0 */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#define WINDOW_TO_FRAME_VPOS(W, VPOS) ((VPOS) + XFASTINT ((W)->top))
|
|
|
|
|
#define WINDOW_TO_FRAME_HPOS(W, HPOS) ((HPOS) + XFASTINT ((W)->left))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#endif /* GLYPH_DEBUG == 0 */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Like bcopy except never gets confused by overlap. Let this be the
|
|
|
|
|
first function defined in this file, or change emacs.c where the
|
|
|
|
|
address of this function is used. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
safe_bcopy (from, to, size)
|
|
|
|
|
char *from, *to;
|
|
|
|
|
int size;
|
|
|
|
|
{
|
1992-11-16 00:42:24 +00:00
|
|
|
|
if (size <= 0 || from == to)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return;
|
|
|
|
|
|
1992-11-16 00:42:24 +00:00
|
|
|
|
/* If the source and destination don't overlap, then bcopy can
|
|
|
|
|
handle it. If they do overlap, but the destination is lower in
|
|
|
|
|
memory than the source, we'll assume bcopy can handle that. */
|
|
|
|
|
if (to < from || from + size <= to)
|
|
|
|
|
bcopy (from, to, size);
|
|
|
|
|
|
|
|
|
|
/* Otherwise, we'll copy from the end. */
|
|
|
|
|
else
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1992-11-16 00:42:24 +00:00
|
|
|
|
register char *endf = from + size;
|
|
|
|
|
register char *endt = to + size;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* If TO - FROM is large, then we should break the copy into
|
|
|
|
|
nonoverlapping chunks of TO - FROM bytes each. However, if
|
|
|
|
|
TO - FROM is small, then the bcopy function call overhead
|
|
|
|
|
makes this not worth it. The crossover point could be about
|
1992-11-16 00:42:24 +00:00
|
|
|
|
anywhere. Since I don't think the obvious copy loop is too
|
|
|
|
|
bad, I'm trying to err in its favor. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
if (to - from < 64)
|
|
|
|
|
{
|
|
|
|
|
do
|
|
|
|
|
*--endt = *--endf;
|
|
|
|
|
while (endf != from);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1992-11-16 00:42:24 +00:00
|
|
|
|
for (;;)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
endt -= (to - from);
|
|
|
|
|
endf -= (to - from);
|
|
|
|
|
|
1992-11-16 00:42:24 +00:00
|
|
|
|
if (endt < to)
|
|
|
|
|
break;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
bcopy (endf, endt, to - from);
|
|
|
|
|
}
|
1992-11-16 00:42:24 +00:00
|
|
|
|
|
|
|
|
|
/* If SIZE wasn't a multiple of TO - FROM, there will be a
|
1999-07-21 21:43:52 +00:00
|
|
|
|
little left over. The amount left over is (endt + (to -
|
|
|
|
|
from)) - to, which is endt - from. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
bcopy (from, to, endt - from);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Glyph Matrices
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Allocate and return a glyph_matrix structure. POOL is the glyph
|
|
|
|
|
pool from which memory for the matrix should be allocated, or null
|
|
|
|
|
for window-based redisplay where no glyph pools are used. The
|
|
|
|
|
member `pool' of the glyph matrix structure returned is set to
|
|
|
|
|
POOL, the structure is otherwise zeroed. */
|
|
|
|
|
|
|
|
|
|
struct glyph_matrix *
|
|
|
|
|
new_glyph_matrix (pool)
|
|
|
|
|
struct glyph_pool *pool;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *result;
|
|
|
|
|
|
|
|
|
|
/* Allocate and clear. */
|
|
|
|
|
result = (struct glyph_matrix *) xmalloc (sizeof *result);
|
|
|
|
|
bzero (result, sizeof *result);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Increment number of allocated matrices. This count is used
|
|
|
|
|
to detect memory leaks. */
|
|
|
|
|
++glyph_matrix_count;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Set pool and return. */
|
|
|
|
|
result->pool = pool;
|
|
|
|
|
return result;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Free glyph matrix MATRIX. Passing in a null MATRIX is allowed.
|
|
|
|
|
|
|
|
|
|
The global counter glyph_matrix_count is decremented when a matrix
|
|
|
|
|
is freed. If the count gets negative, more structures were freed
|
|
|
|
|
than allocated, i.e. one matrix was freed more than once or a bogus
|
|
|
|
|
pointer was passed to this function.
|
|
|
|
|
|
|
|
|
|
If MATRIX->pool is null, this means that the matrix manages its own
|
|
|
|
|
glyph memory---this is done for matrices on X frames. Freeing the
|
|
|
|
|
matrix also frees the glyph memory in this case. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
free_glyph_matrix (matrix)
|
|
|
|
|
struct glyph_matrix *matrix;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (matrix)
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* Detect the case that more matrices are freed than were
|
|
|
|
|
allocated. */
|
|
|
|
|
if (--glyph_matrix_count < 0)
|
|
|
|
|
abort ();
|
|
|
|
|
|
|
|
|
|
/* Free glyph memory if MATRIX owns it. */
|
|
|
|
|
if (matrix->pool == NULL)
|
|
|
|
|
for (i = 0; i < matrix->rows_allocated; ++i)
|
|
|
|
|
xfree (matrix->rows[i].glyphs[LEFT_MARGIN_AREA]);
|
|
|
|
|
|
|
|
|
|
/* Free row structures and the matrix itself. */
|
|
|
|
|
xfree (matrix->rows);
|
|
|
|
|
xfree (matrix);
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Return the number of glyphs to reserve for a marginal area of
|
|
|
|
|
window W. TOTAL_GLYPHS is the number of glyphs in a complete
|
|
|
|
|
display line of window W. MARGIN gives the width of the marginal
|
|
|
|
|
area in canonical character units. MARGIN should be an integer
|
|
|
|
|
or a float. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
margin_glyphs_to_reserve (w, total_glyphs, margin)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int total_glyphs;
|
|
|
|
|
Lisp_Object margin;
|
|
|
|
|
{
|
|
|
|
|
int n;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (NUMBERP (margin))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int width = XFASTINT (w->width);
|
|
|
|
|
double d = max (0, XFLOATINT (margin));
|
|
|
|
|
d = min (width / 2 - 1, d);
|
|
|
|
|
n = (int) ((double) total_glyphs / width * d);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
n = 0;
|
|
|
|
|
|
|
|
|
|
return n;
|
|
|
|
|
}
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Adjust glyph matrix MATRIX on window W or on a frame to changed
|
|
|
|
|
window sizes.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
W is null if the function is called for a frame glyph matrix.
|
|
|
|
|
Otherwise it is the window MATRIX is a member of. X and Y are the
|
|
|
|
|
indices of the first column and row of MATRIX within the frame
|
|
|
|
|
matrix, if such a matrix exists. They are zero for purely
|
|
|
|
|
window-based redisplay. DIM is the needed size of the matrix.
|
1994-04-01 10:13:32 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
In window-based redisplay, where no frame matrices exist, glyph
|
|
|
|
|
matrices manage their own glyph storage. Otherwise, they allocate
|
|
|
|
|
storage from a common frame glyph pool which can be found in
|
|
|
|
|
MATRIX->pool.
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
The reason for this memory management strategy is to avoid complete
|
|
|
|
|
frame redraws if possible. When we allocate from a common pool, a
|
|
|
|
|
change of the location or size of a sub-matrix within the pool
|
|
|
|
|
requires a complete redisplay of the frame because we cannot easily
|
|
|
|
|
make sure that the current matrices of all windows still agree with
|
|
|
|
|
what is displayed on the screen. While this is usually fast, it
|
|
|
|
|
leads to screen flickering. */
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void
|
|
|
|
|
adjust_glyph_matrix (w, matrix, x, y, dim)
|
|
|
|
|
struct window *w;
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int x, y;
|
|
|
|
|
struct dim dim;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
int new_rows;
|
|
|
|
|
int marginal_areas_changed_p = 0;
|
1999-09-05 16:39:19 +00:00
|
|
|
|
int header_line_changed_p = 0;
|
|
|
|
|
int header_line_p = 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int left = -1, right = -1;
|
|
|
|
|
int window_x, window_y, window_width, window_height;
|
|
|
|
|
|
|
|
|
|
/* See if W had a top line that has disappeared now, or vice versa. */
|
|
|
|
|
if (w)
|
|
|
|
|
{
|
1999-09-05 16:39:19 +00:00
|
|
|
|
header_line_p = WINDOW_WANTS_HEADER_LINE_P (w);
|
|
|
|
|
header_line_changed_p = header_line_p != matrix->header_line_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
1999-09-05 16:39:19 +00:00
|
|
|
|
matrix->header_line_p = header_line_p;
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Do nothing if MATRIX' size, position, vscroll, and marginal areas
|
|
|
|
|
haven't changed. This optimization is important because preserving
|
|
|
|
|
the matrix means preventing redisplay. */
|
|
|
|
|
if (matrix->pool == NULL)
|
|
|
|
|
{
|
|
|
|
|
window_box (w, -1, &window_x, &window_y, &window_width, &window_height);
|
|
|
|
|
left = margin_glyphs_to_reserve (w, dim.width, w->left_margin_width);
|
|
|
|
|
right = margin_glyphs_to_reserve (w, dim.width, w->right_margin_width);
|
|
|
|
|
xassert (left >= 0 && right >= 0);
|
|
|
|
|
marginal_areas_changed_p = (left != matrix->left_margin_glyphs
|
|
|
|
|
|| right != matrix->right_margin_glyphs);
|
|
|
|
|
|
|
|
|
|
if (!marginal_areas_changed_p
|
|
|
|
|
&& !fonts_changed_p
|
1999-09-05 16:39:19 +00:00
|
|
|
|
&& !header_line_changed_p
|
1999-07-21 21:43:52 +00:00
|
|
|
|
&& matrix->window_top_y == XFASTINT (w->top)
|
|
|
|
|
&& matrix->window_height == window_height
|
|
|
|
|
&& matrix->window_vscroll == w->vscroll
|
|
|
|
|
&& matrix->window_width == window_width)
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Enlarge MATRIX->rows if necessary. New rows are cleared. */
|
|
|
|
|
if (matrix->rows_allocated < dim.height)
|
|
|
|
|
{
|
|
|
|
|
int size = dim.height * sizeof (struct glyph_row);
|
|
|
|
|
new_rows = dim.height - matrix->rows_allocated;
|
|
|
|
|
matrix->rows = (struct glyph_row *) xrealloc (matrix->rows, size);
|
|
|
|
|
bzero (matrix->rows + matrix->rows_allocated,
|
|
|
|
|
new_rows * sizeof *matrix->rows);
|
|
|
|
|
matrix->rows_allocated = dim.height;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
new_rows = 0;
|
1994-03-31 23:22:23 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If POOL is not null, MATRIX is a frame matrix or a window matrix
|
|
|
|
|
on a frame not using window-based redisplay. Set up pointers for
|
|
|
|
|
each row into the glyph pool. */
|
|
|
|
|
if (matrix->pool)
|
|
|
|
|
{
|
|
|
|
|
xassert (matrix->pool->glyphs);
|
|
|
|
|
|
|
|
|
|
if (w)
|
1994-03-31 23:22:23 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
left = margin_glyphs_to_reserve (w, dim.width,
|
|
|
|
|
w->left_margin_width);
|
|
|
|
|
right = margin_glyphs_to_reserve (w, dim.width,
|
|
|
|
|
w->right_margin_width);
|
1994-03-31 23:22:23 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
else
|
|
|
|
|
left = right = 0;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < dim.height; ++i)
|
1994-03-31 23:22:23 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *row = &matrix->rows[i];
|
|
|
|
|
|
|
|
|
|
row->glyphs[LEFT_MARGIN_AREA]
|
|
|
|
|
= (matrix->pool->glyphs
|
|
|
|
|
+ (y + i) * matrix->pool->ncolumns
|
|
|
|
|
+ x);
|
|
|
|
|
|
|
|
|
|
if (w == NULL
|
|
|
|
|
|| row == matrix->rows + dim.height - 1
|
1999-09-05 16:39:19 +00:00
|
|
|
|
|| (row == matrix->rows && matrix->header_line_p))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
row->glyphs[TEXT_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA];
|
|
|
|
|
row->glyphs[RIGHT_MARGIN_AREA]
|
|
|
|
|
= row->glyphs[TEXT_AREA] + dim.width;
|
|
|
|
|
row->glyphs[LAST_AREA]
|
|
|
|
|
= row->glyphs[RIGHT_MARGIN_AREA];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
row->glyphs[TEXT_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA] + left;
|
|
|
|
|
row->glyphs[RIGHT_MARGIN_AREA]
|
|
|
|
|
= row->glyphs[TEXT_AREA] + dim.width - left - right;
|
|
|
|
|
row->glyphs[LAST_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA] + dim.width;
|
|
|
|
|
}
|
1994-03-31 23:22:23 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
matrix->left_margin_glyphs = left;
|
|
|
|
|
matrix->right_margin_glyphs = right;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* If MATRIX->pool is null, MATRIX is responsible for managing
|
|
|
|
|
its own memory. Allocate glyph memory from the heap. */
|
|
|
|
|
if (dim.width > matrix->matrix_w
|
|
|
|
|
|| new_rows
|
1999-09-05 16:39:19 +00:00
|
|
|
|
|| header_line_changed_p
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| marginal_areas_changed_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *row = matrix->rows;
|
|
|
|
|
struct glyph_row *end = row + matrix->rows_allocated;
|
|
|
|
|
|
|
|
|
|
while (row < end)
|
|
|
|
|
{
|
|
|
|
|
row->glyphs[LEFT_MARGIN_AREA]
|
|
|
|
|
= (struct glyph *) xrealloc (row->glyphs[LEFT_MARGIN_AREA],
|
|
|
|
|
(dim.width
|
|
|
|
|
* sizeof (struct glyph)));
|
|
|
|
|
|
|
|
|
|
/* The mode line never has marginal areas. */
|
|
|
|
|
if (row == matrix->rows + dim.height - 1
|
1999-09-05 16:39:19 +00:00
|
|
|
|
|| (row == matrix->rows && matrix->header_line_p))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
row->glyphs[TEXT_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA];
|
|
|
|
|
row->glyphs[RIGHT_MARGIN_AREA]
|
|
|
|
|
= row->glyphs[TEXT_AREA] + dim.width;
|
|
|
|
|
row->glyphs[LAST_AREA]
|
|
|
|
|
= row->glyphs[RIGHT_MARGIN_AREA];
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
row->glyphs[TEXT_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA] + left;
|
|
|
|
|
row->glyphs[RIGHT_MARGIN_AREA]
|
|
|
|
|
= row->glyphs[TEXT_AREA] + dim.width - left - right;
|
|
|
|
|
row->glyphs[LAST_AREA]
|
|
|
|
|
= row->glyphs[LEFT_MARGIN_AREA] + dim.width;
|
|
|
|
|
}
|
|
|
|
|
++row;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
xassert (left >= 0 && right >= 0);
|
|
|
|
|
matrix->left_margin_glyphs = left;
|
|
|
|
|
matrix->right_margin_glyphs = right;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Number of rows to be used by MATRIX. */
|
|
|
|
|
matrix->nrows = dim.height;
|
2000-09-27 11:46:33 +00:00
|
|
|
|
xassert (matrix->nrows >= 0);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Mark rows in a current matrix of a window as not having valid
|
|
|
|
|
contents. It's important to not do this for desired matrices.
|
|
|
|
|
When Emacs starts, it may already be building desired matrices
|
|
|
|
|
when this function runs. */
|
|
|
|
|
if (w && matrix == w->current_matrix)
|
|
|
|
|
{
|
|
|
|
|
/* Optimize the case that only the height has changed (C-x 2,
|
|
|
|
|
upper window). Invalidate all rows that are no longer part
|
|
|
|
|
of the window. */
|
|
|
|
|
if (!marginal_areas_changed_p
|
|
|
|
|
&& matrix->window_top_y == XFASTINT (w->top)
|
|
|
|
|
&& matrix->window_width == window_width)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
i = 0;
|
|
|
|
|
while (matrix->rows[i].enabled_p
|
|
|
|
|
&& (MATRIX_ROW_BOTTOM_Y (matrix->rows + i)
|
|
|
|
|
< matrix->window_height))
|
|
|
|
|
++i;
|
|
|
|
|
|
|
|
|
|
/* Window end is invalid, if inside of the rows that
|
|
|
|
|
are invalidated. */
|
|
|
|
|
if (INTEGERP (w->window_end_vpos)
|
|
|
|
|
&& XFASTINT (w->window_end_vpos) >= i)
|
|
|
|
|
w->window_end_valid = Qnil;
|
|
|
|
|
|
|
|
|
|
while (i < matrix->nrows)
|
|
|
|
|
matrix->rows[i++].enabled_p = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
for (i = 0; i < matrix->nrows; ++i)
|
|
|
|
|
matrix->rows[i].enabled_p = 0;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Remember last values to be able to optimize frame redraws. */
|
|
|
|
|
matrix->matrix_x = x;
|
|
|
|
|
matrix->matrix_y = y;
|
|
|
|
|
matrix->matrix_w = dim.width;
|
|
|
|
|
matrix->matrix_h = dim.height;
|
|
|
|
|
|
|
|
|
|
/* Record the top y location and height of W at the time the matrix
|
|
|
|
|
was last adjusted. This is used to optimize redisplay above. */
|
|
|
|
|
if (w)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
matrix->window_top_y = XFASTINT (w->top);
|
|
|
|
|
matrix->window_height = window_height;
|
|
|
|
|
matrix->window_width = window_width;
|
|
|
|
|
matrix->window_vscroll = w->vscroll;
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Reverse the contents of rows in MATRIX between START and END. The
|
|
|
|
|
contents of the row at END - 1 end up at START, END - 2 at START +
|
|
|
|
|
1 etc. This is part of the implementation of rotate_matrix (see
|
|
|
|
|
below). */
|
1994-04-01 10:13:32 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static void
|
|
|
|
|
reverse_rows (matrix, start, end)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int start, end;
|
|
|
|
|
{
|
|
|
|
|
int i, j;
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (i = start, j = end - 1; i < j; ++i, --j)
|
|
|
|
|
{
|
|
|
|
|
/* Non-ISO HP/UX compiler doesn't like auto struct
|
|
|
|
|
initialization. */
|
|
|
|
|
struct glyph_row temp;
|
|
|
|
|
temp = matrix->rows[i];
|
|
|
|
|
matrix->rows[i] = matrix->rows[j];
|
|
|
|
|
matrix->rows[j] = temp;
|
|
|
|
|
}
|
|
|
|
|
}
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Rotate the contents of rows in MATRIX in the range FIRST .. LAST -
|
|
|
|
|
1 by BY positions. BY < 0 means rotate left, i.e. towards lower
|
|
|
|
|
indices. (Note: this does not copy glyphs, only glyph pointers in
|
|
|
|
|
row structures are moved around).
|
1994-03-31 23:22:23 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
The algorithm used for rotating the vector was, I believe, first
|
|
|
|
|
described by Kernighan. See the vector R as consisting of two
|
|
|
|
|
sub-vectors AB, where A has length BY for BY >= 0. The result
|
|
|
|
|
after rotating is then BA. Reverse both sub-vectors to get ArBr
|
|
|
|
|
and reverse the result to get (ArBr)r which is BA. Similar for
|
|
|
|
|
rotating right. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
rotate_matrix (matrix, first, last, by)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int first, last, by;
|
|
|
|
|
{
|
|
|
|
|
if (by < 0)
|
|
|
|
|
{
|
|
|
|
|
/* Up (rotate left, i.e. towards lower indices). */
|
|
|
|
|
by = -by;
|
|
|
|
|
reverse_rows (matrix, first, first + by);
|
|
|
|
|
reverse_rows (matrix, first + by, last);
|
|
|
|
|
reverse_rows (matrix, first, last);
|
|
|
|
|
}
|
|
|
|
|
else if (by > 0)
|
|
|
|
|
{
|
|
|
|
|
/* Down (rotate right, i.e. towards higher indices). */
|
|
|
|
|
reverse_rows (matrix, last - by, last);
|
|
|
|
|
reverse_rows (matrix, first, last - by);
|
|
|
|
|
reverse_rows (matrix, first, last);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Increment buffer positions in glyph rows of MATRIX. Do it for rows
|
|
|
|
|
with indices START <= index < END. Increment positions by DELTA/
|
|
|
|
|
DELTA_BYTES. */
|
|
|
|
|
|
|
|
|
|
void
|
2000-04-25 19:39:59 +00:00
|
|
|
|
increment_matrix_positions (matrix, start, end, delta, delta_bytes)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int start, end, delta, delta_bytes;
|
|
|
|
|
{
|
|
|
|
|
/* Check that START and END are reasonable values. */
|
|
|
|
|
xassert (start >= 0 && start <= matrix->nrows);
|
|
|
|
|
xassert (end >= 0 && end <= matrix->nrows);
|
|
|
|
|
xassert (start <= end);
|
|
|
|
|
|
|
|
|
|
for (; start < end; ++start)
|
2000-04-25 19:39:59 +00:00
|
|
|
|
increment_row_positions (matrix->rows + start, delta, delta_bytes);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Enable a range of rows in glyph matrix MATRIX. START and END are
|
|
|
|
|
the row indices of the first and last + 1 row to enable. If
|
|
|
|
|
ENABLED_P is non-zero, enabled_p flags in rows will be set to 1. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
enable_glyph_matrix_rows (matrix, start, end, enabled_p)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int start, end;
|
|
|
|
|
int enabled_p;
|
|
|
|
|
{
|
2000-09-27 11:46:33 +00:00
|
|
|
|
xassert (start <= end);
|
|
|
|
|
xassert (start >= 0 && start < matrix->nrows);
|
|
|
|
|
xassert (end >= 0 && end <= matrix->nrows);
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (; start < end; ++start)
|
|
|
|
|
matrix->rows[start].enabled_p = enabled_p != 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Clear MATRIX.
|
|
|
|
|
|
|
|
|
|
This empties all rows in MATRIX by setting the enabled_p flag for
|
|
|
|
|
all rows of the matrix to zero. The function prepare_desired_row
|
|
|
|
|
will eventually really clear a row when it sees one with a zero
|
|
|
|
|
enabled_p flag.
|
|
|
|
|
|
|
|
|
|
Resets update hints to defaults value. The only update hint
|
|
|
|
|
currently present is the flag MATRIX->no_scrolling_p. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
clear_glyph_matrix (matrix)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
{
|
|
|
|
|
if (matrix)
|
|
|
|
|
{
|
|
|
|
|
enable_glyph_matrix_rows (matrix, 0, matrix->nrows, 0);
|
|
|
|
|
matrix->no_scrolling_p = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Shift part of the glyph matrix MATRIX of window W up or down.
|
|
|
|
|
Increment y-positions in glyph rows between START and END by DY,
|
|
|
|
|
and recompute their visible height. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
shift_glyph_matrix (w, matrix, start, end, dy)
|
|
|
|
|
struct window *w;
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int start, end, dy;
|
|
|
|
|
{
|
|
|
|
|
int min_y, max_y;
|
|
|
|
|
|
|
|
|
|
xassert (start <= end);
|
|
|
|
|
xassert (start >= 0 && start < matrix->nrows);
|
|
|
|
|
xassert (end >= 0 && end <= matrix->nrows);
|
|
|
|
|
|
1999-09-05 16:39:19 +00:00
|
|
|
|
min_y = WINDOW_DISPLAY_HEADER_LINE_HEIGHT (w);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
max_y = WINDOW_DISPLAY_HEIGHT_NO_MODE_LINE (w);
|
|
|
|
|
|
|
|
|
|
for (; start < end; ++start)
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *row = &matrix->rows[start];
|
|
|
|
|
|
|
|
|
|
row->y += dy;
|
|
|
|
|
|
|
|
|
|
if (row->y < min_y)
|
|
|
|
|
row->visible_height = row->height - (min_y - row->y);
|
|
|
|
|
else if (row->y + row->height > max_y)
|
|
|
|
|
row->visible_height = row->height - (row->y + row->height - max_y);
|
|
|
|
|
else
|
|
|
|
|
row->visible_height = row->height;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Mark all rows in current matrices of frame F as invalid. Marking
|
|
|
|
|
invalid is done by setting enabled_p to zero for all rows in a
|
|
|
|
|
current matrix. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
clear_current_matrices (f)
|
|
|
|
|
register struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
/* Clear frame current matrix, if we have one. */
|
|
|
|
|
if (f->current_matrix)
|
|
|
|
|
clear_glyph_matrix (f->current_matrix);
|
|
|
|
|
|
|
|
|
|
/* Clear the matrix of the menu bar window, if such a window exists.
|
|
|
|
|
The menu bar window is currently used to display menus on X when
|
|
|
|
|
no toolkit support is compiled in. */
|
|
|
|
|
if (WINDOWP (f->menu_bar_window))
|
|
|
|
|
clear_glyph_matrix (XWINDOW (f->menu_bar_window)->current_matrix);
|
|
|
|
|
|
1999-09-05 15:49:07 +00:00
|
|
|
|
/* Clear the matrix of the tool-bar window, if any. */
|
|
|
|
|
if (WINDOWP (f->tool_bar_window))
|
|
|
|
|
clear_glyph_matrix (XWINDOW (f->tool_bar_window)->current_matrix);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Clear current window matrices. */
|
|
|
|
|
xassert (WINDOWP (FRAME_ROOT_WINDOW (f)));
|
|
|
|
|
clear_window_matrices (XWINDOW (FRAME_ROOT_WINDOW (f)), 0);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Clear out all display lines of F for a coming redisplay. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
clear_desired_matrices (f)
|
|
|
|
|
register struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
if (f->desired_matrix)
|
|
|
|
|
clear_glyph_matrix (f->desired_matrix);
|
|
|
|
|
|
|
|
|
|
if (WINDOWP (f->menu_bar_window))
|
|
|
|
|
clear_glyph_matrix (XWINDOW (f->menu_bar_window)->desired_matrix);
|
|
|
|
|
|
1999-09-05 15:49:07 +00:00
|
|
|
|
if (WINDOWP (f->tool_bar_window))
|
|
|
|
|
clear_glyph_matrix (XWINDOW (f->tool_bar_window)->desired_matrix);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Do it for window matrices. */
|
|
|
|
|
xassert (WINDOWP (FRAME_ROOT_WINDOW (f)));
|
|
|
|
|
clear_window_matrices (XWINDOW (FRAME_ROOT_WINDOW (f)), 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Clear matrices in window tree rooted in W. If DESIRED_P is
|
|
|
|
|
non-zero clear desired matrices, otherwise clear current matrices. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
clear_window_matrices (w, desired_p)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int desired_p;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
{
|
|
|
|
|
xassert (WINDOWP (w->hchild));
|
|
|
|
|
clear_window_matrices (XWINDOW (w->hchild), desired_p);
|
|
|
|
|
}
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
{
|
|
|
|
|
xassert (WINDOWP (w->vchild));
|
|
|
|
|
clear_window_matrices (XWINDOW (w->vchild), desired_p);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (desired_p)
|
|
|
|
|
clear_glyph_matrix (w->desired_matrix);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
clear_glyph_matrix (w->current_matrix);
|
|
|
|
|
w->window_end_valid = Qnil;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Glyph Rows
|
|
|
|
|
|
|
|
|
|
See dispextern.h for an overall explanation of glyph rows.
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Clear glyph row ROW. Do it in a way that makes it robust against
|
|
|
|
|
changes in the glyph_row structure, i.e. addition or removal of
|
|
|
|
|
structure members. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
clear_glyph_row (row)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
{
|
|
|
|
|
struct glyph *p[1 + LAST_AREA];
|
|
|
|
|
static struct glyph_row null_row;
|
|
|
|
|
|
|
|
|
|
/* Save pointers. */
|
|
|
|
|
p[LEFT_MARGIN_AREA] = row->glyphs[LEFT_MARGIN_AREA];
|
|
|
|
|
p[TEXT_AREA] = row->glyphs[TEXT_AREA];
|
|
|
|
|
p[RIGHT_MARGIN_AREA] = row->glyphs[RIGHT_MARGIN_AREA];
|
|
|
|
|
p[LAST_AREA] = row->glyphs[LAST_AREA];
|
|
|
|
|
|
|
|
|
|
/* Clear. */
|
|
|
|
|
*row = null_row;
|
|
|
|
|
|
|
|
|
|
/* Restore pointers. */
|
|
|
|
|
row->glyphs[LEFT_MARGIN_AREA] = p[LEFT_MARGIN_AREA];
|
|
|
|
|
row->glyphs[TEXT_AREA] = p[TEXT_AREA];
|
|
|
|
|
row->glyphs[RIGHT_MARGIN_AREA] = p[RIGHT_MARGIN_AREA];
|
|
|
|
|
row->glyphs[LAST_AREA] = p[LAST_AREA];
|
2000-07-05 11:39:35 +00:00
|
|
|
|
|
|
|
|
|
#if 0 /* At some point, some bit-fields of struct glyph were not set,
|
|
|
|
|
which made glyphs unequal when compared with GLYPH_EQUAL_P.
|
|
|
|
|
Redisplay outputs such glyphs, and flickering effects were
|
|
|
|
|
the result. This also depended on the contents of memory
|
|
|
|
|
returned by xmalloc. If flickering happens again, activate
|
|
|
|
|
the code below If the flickering is gone with that, chances
|
|
|
|
|
are that the flickering has the same reason as here. */
|
|
|
|
|
bzero (p[0], (char *) p[LAST_AREA] - (char *) p[0]);
|
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Make ROW an empty, enabled row of canonical character height,
|
|
|
|
|
in window W starting at y-position Y. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
blank_row (w, row, y)
|
|
|
|
|
struct window *w;
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
int y;
|
|
|
|
|
{
|
|
|
|
|
int min_y, max_y;
|
|
|
|
|
|
1999-09-05 16:39:19 +00:00
|
|
|
|
min_y = WINDOW_DISPLAY_HEADER_LINE_HEIGHT (w);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
max_y = WINDOW_DISPLAY_HEIGHT_NO_MODE_LINE (w);
|
|
|
|
|
|
|
|
|
|
clear_glyph_row (row);
|
|
|
|
|
row->y = y;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
row->ascent = row->phys_ascent = 0;
|
|
|
|
|
row->height = row->phys_height = CANON_Y_UNIT (XFRAME (w->frame));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
if (row->y < min_y)
|
|
|
|
|
row->visible_height = row->height - (min_y - row->y);
|
|
|
|
|
else if (row->y + row->height > max_y)
|
|
|
|
|
row->visible_height = row->height - (row->y + row->height - max_y);
|
|
|
|
|
else
|
|
|
|
|
row->visible_height = row->height;
|
|
|
|
|
|
|
|
|
|
row->enabled_p = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Increment buffer positions in glyph row ROW. DELTA and DELTA_BYTES
|
|
|
|
|
are the amounts by which to change positions. Note that the first
|
|
|
|
|
glyph of the text area of a row can have a buffer position even if
|
|
|
|
|
the used count of the text area is zero. Such rows display line
|
|
|
|
|
ends. */
|
|
|
|
|
|
|
|
|
|
void
|
2000-04-25 19:39:59 +00:00
|
|
|
|
increment_row_positions (row, delta, delta_bytes)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
int delta, delta_bytes;
|
|
|
|
|
{
|
|
|
|
|
int area, i;
|
|
|
|
|
|
|
|
|
|
/* Increment start and end positions. */
|
|
|
|
|
MATRIX_ROW_START_CHARPOS (row) += delta;
|
|
|
|
|
MATRIX_ROW_START_BYTEPOS (row) += delta_bytes;
|
|
|
|
|
MATRIX_ROW_END_CHARPOS (row) += delta;
|
|
|
|
|
MATRIX_ROW_END_BYTEPOS (row) += delta_bytes;
|
|
|
|
|
|
|
|
|
|
/* Increment positions in glyphs. */
|
|
|
|
|
for (area = 0; area < LAST_AREA; ++area)
|
|
|
|
|
for (i = 0; i < row->used[area]; ++i)
|
|
|
|
|
if (BUFFERP (row->glyphs[area][i].object)
|
|
|
|
|
&& row->glyphs[area][i].charpos > 0)
|
|
|
|
|
row->glyphs[area][i].charpos += delta;
|
|
|
|
|
|
|
|
|
|
/* Capture the case of rows displaying a line end. */
|
|
|
|
|
if (row->used[TEXT_AREA] == 0
|
|
|
|
|
&& MATRIX_ROW_DISPLAYS_TEXT_P (row))
|
|
|
|
|
row->glyphs[TEXT_AREA]->charpos += delta;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2000-05-31 19:16:25 +00:00
|
|
|
|
#if 0
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Swap glyphs between two glyph rows A and B. This exchanges glyph
|
|
|
|
|
contents, i.e. glyph structure contents are exchanged between A and
|
|
|
|
|
B without changing glyph pointers in A and B. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
swap_glyphs_in_rows (a, b)
|
|
|
|
|
struct glyph_row *a, *b;
|
|
|
|
|
{
|
|
|
|
|
int area;
|
|
|
|
|
|
|
|
|
|
for (area = 0; area < LAST_AREA; ++area)
|
|
|
|
|
{
|
|
|
|
|
/* Number of glyphs to swap. */
|
|
|
|
|
int max_used = max (a->used[area], b->used[area]);
|
|
|
|
|
|
|
|
|
|
/* Start of glyphs in area of row A. */
|
|
|
|
|
struct glyph *glyph_a = a->glyphs[area];
|
|
|
|
|
|
|
|
|
|
/* End + 1 of glyphs in area of row A. */
|
|
|
|
|
struct glyph *glyph_a_end = a->glyphs[max_used];
|
|
|
|
|
|
|
|
|
|
/* Start of glyphs in area of row B. */
|
|
|
|
|
struct glyph *glyph_b = b->glyphs[area];
|
|
|
|
|
|
|
|
|
|
while (glyph_a < glyph_a_end)
|
|
|
|
|
{
|
|
|
|
|
/* Non-ISO HP/UX compiler doesn't like auto struct
|
|
|
|
|
initialization. */
|
|
|
|
|
struct glyph temp;
|
|
|
|
|
temp = *glyph_a;
|
|
|
|
|
*glyph_a = *glyph_b;
|
|
|
|
|
*glyph_b = temp;
|
|
|
|
|
++glyph_a;
|
|
|
|
|
++glyph_b;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2000-05-31 19:16:25 +00:00
|
|
|
|
#endif /* 0 */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Exchange pointers to glyph memory between glyph rows A and B. */
|
|
|
|
|
|
|
|
|
|
static INLINE void
|
|
|
|
|
swap_glyph_pointers (a, b)
|
|
|
|
|
struct glyph_row *a, *b;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
for (i = 0; i < LAST_AREA + 1; ++i)
|
|
|
|
|
{
|
|
|
|
|
struct glyph *temp = a->glyphs[i];
|
|
|
|
|
a->glyphs[i] = b->glyphs[i];
|
|
|
|
|
b->glyphs[i] = temp;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Copy glyph row structure FROM to glyph row structure TO, except
|
|
|
|
|
that glyph pointers in the structures are left unchanged. */
|
|
|
|
|
|
|
|
|
|
INLINE void
|
|
|
|
|
copy_row_except_pointers (to, from)
|
|
|
|
|
struct glyph_row *to, *from;
|
|
|
|
|
{
|
|
|
|
|
struct glyph *pointers[1 + LAST_AREA];
|
|
|
|
|
|
|
|
|
|
/* Save glyph pointers of TO. */
|
|
|
|
|
bcopy (to->glyphs, pointers, sizeof to->glyphs);
|
|
|
|
|
|
|
|
|
|
/* Do a structure assignment. */
|
|
|
|
|
*to = *from;
|
|
|
|
|
|
|
|
|
|
/* Restore original pointers of TO. */
|
|
|
|
|
bcopy (pointers, to->glyphs, sizeof to->glyphs);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Copy contents of glyph row FROM to glyph row TO. Glyph pointers in
|
|
|
|
|
TO and FROM are left unchanged. Glyph contents are copied from the
|
|
|
|
|
glyph memory of FROM to the glyph memory of TO. Increment buffer
|
|
|
|
|
positions in row TO by DELTA/ DELTA_BYTES. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
copy_glyph_row_contents (to, from, delta, delta_bytes)
|
|
|
|
|
struct glyph_row *to, *from;
|
|
|
|
|
int delta, delta_bytes;
|
|
|
|
|
{
|
|
|
|
|
int area;
|
|
|
|
|
|
|
|
|
|
/* This is like a structure assignment TO = FROM, except that
|
|
|
|
|
glyph pointers in the rows are left unchanged. */
|
|
|
|
|
copy_row_except_pointers (to, from);
|
|
|
|
|
|
|
|
|
|
/* Copy glyphs from FROM to TO. */
|
|
|
|
|
for (area = 0; area < LAST_AREA; ++area)
|
|
|
|
|
if (from->used[area])
|
|
|
|
|
bcopy (from->glyphs[area], to->glyphs[area],
|
|
|
|
|
from->used[area] * sizeof (struct glyph));
|
|
|
|
|
|
|
|
|
|
/* Increment buffer positions in TO by DELTA. */
|
2000-04-25 19:39:59 +00:00
|
|
|
|
increment_row_positions (to, delta, delta_bytes);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Assign glyph row FROM to glyph row TO. This works like a structure
|
|
|
|
|
assignment TO = FROM, except that glyph pointers are not copied but
|
|
|
|
|
exchanged between TO and FROM. Pointers must be exchanged to avoid
|
|
|
|
|
a memory leak. */
|
|
|
|
|
|
|
|
|
|
static INLINE void
|
|
|
|
|
assign_row (to, from)
|
|
|
|
|
struct glyph_row *to, *from;
|
|
|
|
|
{
|
|
|
|
|
swap_glyph_pointers (to, from);
|
|
|
|
|
copy_row_except_pointers (to, from);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Test whether the glyph memory of the glyph row WINDOW_ROW, which is
|
|
|
|
|
a row in a window matrix, is a slice of the glyph memory of the
|
|
|
|
|
glyph row FRAME_ROW which is a row in a frame glyph matrix. Value
|
|
|
|
|
is non-zero if the glyph memory of WINDOW_ROW is part of the glyph
|
|
|
|
|
memory of FRAME_ROW. */
|
|
|
|
|
|
2000-07-19 15:47:52 +00:00
|
|
|
|
#ifdef GLYPH_DEBUG
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
static int
|
|
|
|
|
glyph_row_slice_p (window_row, frame_row)
|
|
|
|
|
struct glyph_row *window_row, *frame_row;
|
|
|
|
|
{
|
|
|
|
|
struct glyph *window_glyph_start = window_row->glyphs[0];
|
|
|
|
|
struct glyph *frame_glyph_start = frame_row->glyphs[0];
|
|
|
|
|
struct glyph *frame_glyph_end = frame_row->glyphs[LAST_AREA];
|
|
|
|
|
|
|
|
|
|
return (frame_glyph_start <= window_glyph_start
|
|
|
|
|
&& window_glyph_start < frame_glyph_end);
|
|
|
|
|
}
|
|
|
|
|
|
2000-07-19 15:47:52 +00:00
|
|
|
|
#endif /* GLYPH_DEBUG */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
2000-05-31 19:16:25 +00:00
|
|
|
|
#if 0
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Find the row in the window glyph matrix WINDOW_MATRIX being a slice
|
|
|
|
|
of ROW in the frame matrix FRAME_MATRIX. Value is null if no row
|
|
|
|
|
in WINDOW_MATRIX is found satisfying the condition. */
|
|
|
|
|
|
|
|
|
|
static struct glyph_row *
|
|
|
|
|
find_glyph_row_slice (window_matrix, frame_matrix, row)
|
|
|
|
|
struct glyph_matrix *window_matrix, *frame_matrix;
|
|
|
|
|
int row;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
xassert (row >= 0 && row < frame_matrix->nrows);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < window_matrix->nrows; ++i)
|
|
|
|
|
if (glyph_row_slice_p (window_matrix->rows + i,
|
|
|
|
|
frame_matrix->rows + row))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
return i < window_matrix->nrows ? window_matrix->rows + i : 0;
|
|
|
|
|
}
|
|
|
|
|
|
2000-05-31 19:16:25 +00:00
|
|
|
|
#endif /* 0 */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Prepare ROW for display. Desired rows are cleared lazily,
|
|
|
|
|
i.e. they are only marked as to be cleared by setting their
|
|
|
|
|
enabled_p flag to zero. When a row is to be displayed, a prior
|
|
|
|
|
call to this function really clears it. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
prepare_desired_row (row)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
{
|
|
|
|
|
if (!row->enabled_p)
|
|
|
|
|
{
|
|
|
|
|
clear_glyph_row (row);
|
|
|
|
|
row->enabled_p = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return a hash code for glyph row ROW. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
line_hash_code (row)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
{
|
|
|
|
|
int hash = 0;
|
|
|
|
|
|
|
|
|
|
if (row->enabled_p)
|
|
|
|
|
{
|
|
|
|
|
if (row->inverse_p)
|
|
|
|
|
{
|
|
|
|
|
/* Give all highlighted lines the same hash code
|
|
|
|
|
so as to encourage scrolling to leave them in place. */
|
|
|
|
|
hash = -1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
struct glyph *glyph = row->glyphs[TEXT_AREA];
|
|
|
|
|
struct glyph *end = glyph + row->used[TEXT_AREA];
|
|
|
|
|
|
|
|
|
|
while (glyph < end)
|
|
|
|
|
{
|
1999-12-27 05:06:03 +00:00
|
|
|
|
int c = glyph->u.ch;
|
|
|
|
|
int face_id = glyph->face_id;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (must_write_spaces)
|
1999-12-27 05:06:03 +00:00
|
|
|
|
c -= SPACEGLYPH;
|
|
|
|
|
hash = (((hash << 4) + (hash >> 24)) & 0x0fffffff) + c;
|
|
|
|
|
hash = (((hash << 4) + (hash >> 24)) & 0x0fffffff) + face_id;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
++glyph;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (hash == 0)
|
|
|
|
|
hash = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return hash;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the cost of drawing line VPOS In MATRIX. The cost equals
|
|
|
|
|
the number of characters in the line. If must_write_spaces is
|
|
|
|
|
zero, leading and trailing spaces are ignored. */
|
|
|
|
|
|
|
|
|
|
static unsigned int
|
|
|
|
|
line_draw_cost (matrix, vpos)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int vpos;
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *row = matrix->rows + vpos;
|
|
|
|
|
struct glyph *beg = row->glyphs[TEXT_AREA];
|
|
|
|
|
struct glyph *end = beg + row->used[TEXT_AREA];
|
|
|
|
|
int len;
|
|
|
|
|
Lisp_Object *glyph_table_base = GLYPH_TABLE_BASE;
|
|
|
|
|
int glyph_table_len = GLYPH_TABLE_LENGTH;
|
|
|
|
|
|
|
|
|
|
/* Ignore trailing and leading spaces if we can. */
|
|
|
|
|
if (!must_write_spaces)
|
|
|
|
|
{
|
|
|
|
|
/* Skip from the end over trailing spaces. */
|
|
|
|
|
while (end != beg && CHAR_GLYPH_SPACE_P (*end))
|
|
|
|
|
--end;
|
|
|
|
|
|
|
|
|
|
/* All blank line. */
|
|
|
|
|
if (end == beg)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Skip over leading spaces. */
|
|
|
|
|
while (CHAR_GLYPH_SPACE_P (*beg))
|
|
|
|
|
++beg;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If we don't have a glyph-table, each glyph is one character,
|
|
|
|
|
so return the number of glyphs. */
|
|
|
|
|
if (glyph_table_base == 0)
|
|
|
|
|
len = end - beg;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Otherwise, scan the glyphs and accumulate their total length
|
|
|
|
|
in LEN. */
|
|
|
|
|
len = 0;
|
|
|
|
|
while (beg < end)
|
|
|
|
|
{
|
|
|
|
|
GLYPH g = GLYPH_FROM_CHAR_GLYPH (*beg);
|
|
|
|
|
|
1999-12-27 05:06:03 +00:00
|
|
|
|
if (g < 0
|
|
|
|
|
|| GLYPH_SIMPLE_P (glyph_table_base, glyph_table_len, g))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
len += 1;
|
|
|
|
|
else
|
|
|
|
|
len += GLYPH_LENGTH (glyph_table_base, g);
|
|
|
|
|
|
|
|
|
|
++beg;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return len;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Test two glyph rows A and B for equality. Value is non-zero if A
|
|
|
|
|
and B have equal contents. W is the window to which the glyphs
|
|
|
|
|
rows A and B belong. It is needed here to test for partial row
|
2000-07-11 19:57:54 +00:00
|
|
|
|
visibility. MOUSE_FACE_P non-zero means compare the mouse_face_p
|
|
|
|
|
flags of A and B, too. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
static INLINE int
|
2000-07-11 19:57:54 +00:00
|
|
|
|
row_equal_p (w, a, b, mouse_face_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *w;
|
|
|
|
|
struct glyph_row *a, *b;
|
2000-07-11 19:57:54 +00:00
|
|
|
|
int mouse_face_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
if (a == b)
|
|
|
|
|
return 1;
|
|
|
|
|
else if (a->hash != b->hash)
|
|
|
|
|
return 0;
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
struct glyph *a_glyph, *b_glyph, *a_end;
|
|
|
|
|
int area;
|
|
|
|
|
|
2000-07-11 19:57:54 +00:00
|
|
|
|
if (mouse_face_p && a->mouse_face_p != b->mouse_face_p)
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Compare glyphs. */
|
|
|
|
|
for (area = LEFT_MARGIN_AREA; area < LAST_AREA; ++area)
|
|
|
|
|
{
|
|
|
|
|
if (a->used[area] != b->used[area])
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
a_glyph = a->glyphs[area];
|
|
|
|
|
a_end = a_glyph + a->used[area];
|
|
|
|
|
b_glyph = b->glyphs[area];
|
|
|
|
|
|
|
|
|
|
while (a_glyph < a_end
|
|
|
|
|
&& GLYPH_EQUAL_P (a_glyph, b_glyph))
|
|
|
|
|
++a_glyph, ++b_glyph;
|
|
|
|
|
|
|
|
|
|
if (a_glyph != a_end)
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (a->truncated_on_left_p != b->truncated_on_left_p
|
|
|
|
|
|| a->inverse_p != b->inverse_p
|
|
|
|
|
|| a->fill_line_p != b->fill_line_p
|
|
|
|
|
|| a->truncated_on_right_p != b->truncated_on_right_p
|
|
|
|
|
|| a->overlay_arrow_p != b->overlay_arrow_p
|
|
|
|
|
|| a->continued_p != b->continued_p
|
|
|
|
|
|| a->indicate_empty_line_p != b->indicate_empty_line_p
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| a->overlapped_p != b->overlapped_p
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| (MATRIX_ROW_CONTINUATION_LINE_P (a)
|
|
|
|
|
!= MATRIX_ROW_CONTINUATION_LINE_P (b))
|
|
|
|
|
/* Different partially visible characters on left margin. */
|
|
|
|
|
|| a->x != b->x
|
|
|
|
|
/* Different height. */
|
|
|
|
|
|| a->ascent != b->ascent
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| a->phys_ascent != b->phys_ascent
|
|
|
|
|
|| a->phys_height != b->phys_height
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| a->visible_height != b->visible_height)
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Glyph Pool
|
|
|
|
|
|
|
|
|
|
See dispextern.h for an overall explanation of glyph pools.
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Allocate a glyph_pool structure. The structure returned is
|
|
|
|
|
initialized with zeros. The global variable glyph_pool_count is
|
|
|
|
|
incremented for each pool allocated. */
|
|
|
|
|
|
|
|
|
|
static struct glyph_pool *
|
|
|
|
|
new_glyph_pool ()
|
|
|
|
|
{
|
|
|
|
|
struct glyph_pool *result;
|
|
|
|
|
|
|
|
|
|
/* Allocate a new glyph_pool and clear it. */
|
|
|
|
|
result = (struct glyph_pool *) xmalloc (sizeof *result);
|
|
|
|
|
bzero (result, sizeof *result);
|
|
|
|
|
|
|
|
|
|
/* For memory leak and double deletion checking. */
|
|
|
|
|
++glyph_pool_count;
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Free a glyph_pool structure POOL. The function may be called with
|
|
|
|
|
a null POOL pointer. The global variable glyph_pool_count is
|
|
|
|
|
decremented with every pool structure freed. If this count gets
|
|
|
|
|
negative, more structures were freed than allocated, i.e. one
|
|
|
|
|
structure must have been freed more than once or a bogus pointer
|
|
|
|
|
was passed to free_glyph_pool. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
free_glyph_pool (pool)
|
|
|
|
|
struct glyph_pool *pool;
|
|
|
|
|
{
|
|
|
|
|
if (pool)
|
|
|
|
|
{
|
|
|
|
|
/* More freed than allocated? */
|
|
|
|
|
--glyph_pool_count;
|
|
|
|
|
xassert (glyph_pool_count >= 0);
|
|
|
|
|
|
|
|
|
|
xfree (pool->glyphs);
|
|
|
|
|
xfree (pool);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Enlarge a glyph pool POOL. MATRIX_DIM gives the number of rows and
|
|
|
|
|
columns we need. This function never shrinks a pool. The only
|
|
|
|
|
case in which this would make sense, would be when a frame's size
|
|
|
|
|
is changed from a large value to a smaller one. But, if someone
|
|
|
|
|
does it once, we can expect that he will do it again.
|
|
|
|
|
|
|
|
|
|
Value is non-zero if the pool changed in a way which makes
|
|
|
|
|
re-adjusting window glyph matrices necessary. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
realloc_glyph_pool (pool, matrix_dim)
|
|
|
|
|
struct glyph_pool *pool;
|
|
|
|
|
struct dim matrix_dim;
|
|
|
|
|
{
|
|
|
|
|
int needed;
|
|
|
|
|
int changed_p;
|
|
|
|
|
|
|
|
|
|
changed_p = (pool->glyphs == 0
|
|
|
|
|
|| matrix_dim.height != pool->nrows
|
|
|
|
|
|| matrix_dim.width != pool->ncolumns);
|
|
|
|
|
|
|
|
|
|
/* Enlarge the glyph pool. */
|
|
|
|
|
needed = matrix_dim.width * matrix_dim.height;
|
|
|
|
|
if (needed > pool->nglyphs)
|
|
|
|
|
{
|
|
|
|
|
int size = needed * sizeof (struct glyph);
|
|
|
|
|
|
|
|
|
|
if (pool->glyphs)
|
|
|
|
|
pool->glyphs = (struct glyph *) xrealloc (pool->glyphs, size);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
pool->glyphs = (struct glyph *) xmalloc (size);
|
|
|
|
|
bzero (pool->glyphs, size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pool->nglyphs = needed;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Remember the number of rows and columns because (a) we use then
|
|
|
|
|
to do sanity checks, and (b) the number of columns determines
|
|
|
|
|
where rows in the frame matrix start---this must be available to
|
|
|
|
|
determine pointers to rows of window sub-matrices. */
|
|
|
|
|
pool->nrows = matrix_dim.height;
|
|
|
|
|
pool->ncolumns = matrix_dim.width;
|
|
|
|
|
|
|
|
|
|
return changed_p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Debug Code
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
|
2000-02-25 13:20:36 +00:00
|
|
|
|
|
|
|
|
|
/* Flush standard output. This is sometimes useful to call from
|
|
|
|
|
the debugger. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
flush_stdout ()
|
|
|
|
|
{
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Check that no glyph pointers have been lost in MATRIX. If a
|
|
|
|
|
pointer has been lost, e.g. by using a structure assignment between
|
|
|
|
|
rows, at least one pointer must occur more than once in the rows of
|
|
|
|
|
MATRIX. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
check_matrix_pointer_lossage (matrix)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
{
|
|
|
|
|
int i, j;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < matrix->nrows; ++i)
|
|
|
|
|
for (j = 0; j < matrix->nrows; ++j)
|
|
|
|
|
xassert (i == j
|
|
|
|
|
|| (matrix->rows[i].glyphs[TEXT_AREA]
|
|
|
|
|
!= matrix->rows[j].glyphs[TEXT_AREA]));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Get a pointer to glyph row ROW in MATRIX, with bounds checks. */
|
|
|
|
|
|
|
|
|
|
struct glyph_row *
|
|
|
|
|
matrix_row (matrix, row)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int row;
|
|
|
|
|
{
|
|
|
|
|
xassert (matrix && matrix->rows);
|
|
|
|
|
xassert (row >= 0 && row < matrix->nrows);
|
|
|
|
|
|
|
|
|
|
/* That's really too slow for normal testing because this function
|
|
|
|
|
is called almost everywhere. Although---it's still astonishingly
|
|
|
|
|
fast, so it is valuable to have for debugging purposes. */
|
|
|
|
|
#if 0
|
|
|
|
|
check_matrix_pointer_lossage (matrix);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return matrix->rows + row;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#if 0 /* This function makes invalid assumptions when text is
|
|
|
|
|
partially invisible. But it might come handy for debugging
|
|
|
|
|
nevertheless. */
|
|
|
|
|
|
|
|
|
|
/* Check invariants that must hold for an up to date current matrix of
|
|
|
|
|
window W. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
check_matrix_invariants (w)
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
struct glyph_matrix *matrix = w->current_matrix;
|
|
|
|
|
int yb = window_text_bottom_y (w);
|
|
|
|
|
struct glyph_row *row = matrix->rows;
|
|
|
|
|
struct glyph_row *last_text_row = NULL;
|
|
|
|
|
struct buffer *saved = current_buffer;
|
|
|
|
|
struct buffer *buffer = XBUFFER (w->buffer);
|
|
|
|
|
int c;
|
|
|
|
|
|
|
|
|
|
/* This can sometimes happen for a fresh window. */
|
|
|
|
|
if (matrix->nrows < 2)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
set_buffer_temp (buffer);
|
|
|
|
|
|
|
|
|
|
/* Note: last row is always reserved for the mode line. */
|
|
|
|
|
while (MATRIX_ROW_DISPLAYS_TEXT_P (row)
|
|
|
|
|
&& MATRIX_ROW_BOTTOM_Y (row) < yb)
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *next = row + 1;
|
|
|
|
|
|
|
|
|
|
if (MATRIX_ROW_DISPLAYS_TEXT_P (row))
|
|
|
|
|
last_text_row = row;
|
|
|
|
|
|
|
|
|
|
/* Check that character and byte positions are in sync. */
|
|
|
|
|
xassert (MATRIX_ROW_START_BYTEPOS (row)
|
|
|
|
|
== CHAR_TO_BYTE (MATRIX_ROW_START_CHARPOS (row)));
|
|
|
|
|
|
|
|
|
|
/* CHAR_TO_BYTE aborts when invoked for a position > Z. We can
|
|
|
|
|
have such a position temporarily in case of a minibuffer
|
|
|
|
|
displaying something like `[Sole completion]' at its end. */
|
|
|
|
|
if (MATRIX_ROW_END_CHARPOS (row) < BUF_ZV (current_buffer))
|
|
|
|
|
xassert (MATRIX_ROW_END_BYTEPOS (row)
|
|
|
|
|
== CHAR_TO_BYTE (MATRIX_ROW_END_CHARPOS (row)));
|
|
|
|
|
|
|
|
|
|
/* Check that end position of `row' is equal to start position
|
|
|
|
|
of next row. */
|
|
|
|
|
if (next->enabled_p && MATRIX_ROW_DISPLAYS_TEXT_P (next))
|
|
|
|
|
{
|
|
|
|
|
xassert (MATRIX_ROW_END_CHARPOS (row)
|
|
|
|
|
== MATRIX_ROW_START_CHARPOS (next));
|
|
|
|
|
xassert (MATRIX_ROW_END_BYTEPOS (row)
|
|
|
|
|
== MATRIX_ROW_START_BYTEPOS (next));
|
|
|
|
|
}
|
|
|
|
|
row = next;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
xassert (w->current_matrix->nrows == w->desired_matrix->nrows);
|
|
|
|
|
xassert (w->desired_matrix->rows != NULL);
|
|
|
|
|
set_buffer_temp (saved);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* 0 */
|
|
|
|
|
|
|
|
|
|
#endif /* GLYPH_DEBUG != 0 */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
Allocating/ Adjusting Glyph Matrices
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Allocate glyph matrices over a window tree for a frame-based
|
|
|
|
|
redisplay
|
|
|
|
|
|
|
|
|
|
X and Y are column/row within the frame glyph matrix where
|
|
|
|
|
sub-matrices for the window tree rooted at WINDOW must be
|
|
|
|
|
allocated. CH_DIM contains the dimensions of the smallest
|
|
|
|
|
character that could be used during display. DIM_ONLY_P non-zero
|
|
|
|
|
means that the caller of this function is only interested in the
|
|
|
|
|
result matrix dimension, and matrix adjustments should not be
|
|
|
|
|
performed.
|
|
|
|
|
|
|
|
|
|
The function returns the total width/height of the sub-matrices of
|
|
|
|
|
the window tree. If called on a frame root window, the computation
|
|
|
|
|
will take the mini-buffer window into account.
|
|
|
|
|
|
|
|
|
|
*WINDOW_CHANGE_FLAGS is set to a bit mask with bits
|
|
|
|
|
|
|
|
|
|
NEW_LEAF_MATRIX set if any window in the tree did not have a
|
|
|
|
|
glyph matrices yet, and
|
|
|
|
|
|
|
|
|
|
CHANGED_LEAF_MATRIX set if the dimension or location of a matrix of
|
|
|
|
|
any window in the tree will be changed or have been changed (see
|
|
|
|
|
DIM_ONLY_P).
|
|
|
|
|
|
|
|
|
|
*WINDOW_CHANGE_FLAGS must be initialized by the caller of this
|
|
|
|
|
function.
|
|
|
|
|
|
|
|
|
|
Windows are arranged into chains of windows on the same level
|
|
|
|
|
through the next fields of window structures. Such a level can be
|
|
|
|
|
either a sequence of horizontally adjacent windows from left to
|
|
|
|
|
right, or a sequence of vertically adjacent windows from top to
|
|
|
|
|
bottom. Each window in a horizontal sequence can be either a leaf
|
|
|
|
|
window or a vertical sequence; a window in a vertical sequence can
|
|
|
|
|
be either a leaf or a horizontal sequence. All windows in a
|
|
|
|
|
horizontal sequence have the same height, and all windows in a
|
|
|
|
|
vertical sequence have the same width.
|
|
|
|
|
|
|
|
|
|
This function uses, for historical reasons, a more general
|
|
|
|
|
algorithm to determine glyph matrix dimensions that would be
|
|
|
|
|
necessary.
|
|
|
|
|
|
|
|
|
|
The matrix height of a horizontal sequence is determined by the
|
|
|
|
|
maximum height of any matrix in the sequence. The matrix width of
|
|
|
|
|
a horizontal sequence is computed by adding up matrix widths of
|
|
|
|
|
windows in the sequence.
|
|
|
|
|
|
|
|
|
|
|<------- result width ------->|
|
|
|
|
|
+---------+----------+---------+ ---
|
|
|
|
|
| | | | |
|
|
|
|
|
| | | |
|
|
|
|
|
+---------+ | | result height
|
|
|
|
|
| +---------+
|
|
|
|
|
| | |
|
|
|
|
|
+----------+ ---
|
|
|
|
|
|
|
|
|
|
The matrix width of a vertical sequence is the maximum matrix width
|
|
|
|
|
of any window in the sequence. Its height is computed by adding up
|
|
|
|
|
matrix heights of windows in the sequence.
|
|
|
|
|
|
|
|
|
|
|<---- result width -->|
|
|
|
|
|
+---------+ ---
|
|
|
|
|
| | |
|
|
|
|
|
| | |
|
|
|
|
|
+---------+--+ |
|
|
|
|
|
| | |
|
|
|
|
|
| | result height
|
|
|
|
|
| |
|
|
|
|
|
+------------+---------+ |
|
|
|
|
|
| | |
|
|
|
|
|
| | |
|
|
|
|
|
+------------+---------+ --- */
|
|
|
|
|
|
|
|
|
|
/* Bit indicating that a new matrix will be allocated or has been
|
|
|
|
|
allocated. */
|
|
|
|
|
|
|
|
|
|
#define NEW_LEAF_MATRIX (1 << 0)
|
|
|
|
|
|
|
|
|
|
/* Bit indicating that a matrix will or has changed its location or
|
|
|
|
|
size. */
|
|
|
|
|
|
|
|
|
|
#define CHANGED_LEAF_MATRIX (1 << 1)
|
|
|
|
|
|
|
|
|
|
static struct dim
|
|
|
|
|
allocate_matrices_for_frame_redisplay (window, x, y, ch_dim,
|
|
|
|
|
dim_only_p, window_change_flags)
|
|
|
|
|
Lisp_Object window;
|
|
|
|
|
int x, y;
|
|
|
|
|
struct dim ch_dim;
|
|
|
|
|
int dim_only_p;
|
|
|
|
|
int *window_change_flags;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (WINDOW_FRAME (XWINDOW (window)));
|
|
|
|
|
int x0 = x, y0 = y;
|
|
|
|
|
int wmax = 0, hmax = 0;
|
|
|
|
|
struct dim total;
|
|
|
|
|
struct dim dim;
|
|
|
|
|
struct window *w;
|
|
|
|
|
int in_horz_combination_p;
|
|
|
|
|
|
|
|
|
|
/* What combination is WINDOW part of? Compute this once since the
|
|
|
|
|
result is the same for all windows in the `next' chain. The
|
|
|
|
|
special case of a root window (parent equal to nil) is treated
|
|
|
|
|
like a vertical combination because a root window's `next'
|
|
|
|
|
points to the mini-buffer window, if any, which is arranged
|
|
|
|
|
vertically below other windows. */
|
|
|
|
|
in_horz_combination_p
|
|
|
|
|
= (!NILP (XWINDOW (window)->parent)
|
|
|
|
|
&& !NILP (XWINDOW (XWINDOW (window)->parent)->hchild));
|
|
|
|
|
|
|
|
|
|
/* For WINDOW and all windows on the same level. */
|
|
|
|
|
do
|
|
|
|
|
{
|
|
|
|
|
w = XWINDOW (window);
|
|
|
|
|
|
|
|
|
|
/* Get the dimension of the window sub-matrix for W, depending
|
|
|
|
|
on whether this a combination or a leaf window. */
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
dim = allocate_matrices_for_frame_redisplay (w->hchild, x, y, ch_dim,
|
|
|
|
|
dim_only_p,
|
|
|
|
|
window_change_flags);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
dim = allocate_matrices_for_frame_redisplay (w->vchild, x, y, ch_dim,
|
|
|
|
|
dim_only_p,
|
|
|
|
|
window_change_flags);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* If not already done, allocate sub-matrix structures. */
|
|
|
|
|
if (w->desired_matrix == NULL)
|
|
|
|
|
{
|
|
|
|
|
w->desired_matrix = new_glyph_matrix (f->desired_pool);
|
|
|
|
|
w->current_matrix = new_glyph_matrix (f->current_pool);
|
|
|
|
|
*window_change_flags |= NEW_LEAF_MATRIX;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Width and height MUST be chosen so that there are no
|
|
|
|
|
holes in the frame matrix. */
|
2000-04-05 17:29:31 +00:00
|
|
|
|
dim.width = XINT (w->width);
|
|
|
|
|
dim.height = XINT (w->height);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Will matrix be re-allocated? */
|
|
|
|
|
if (x != w->desired_matrix->matrix_x
|
|
|
|
|
|| y != w->desired_matrix->matrix_y
|
|
|
|
|
|| dim.width != w->desired_matrix->matrix_w
|
|
|
|
|
|| dim.height != w->desired_matrix->matrix_h
|
|
|
|
|
|| (margin_glyphs_to_reserve (w, dim.width,
|
|
|
|
|
w->right_margin_width)
|
|
|
|
|
!= w->desired_matrix->left_margin_glyphs)
|
|
|
|
|
|| (margin_glyphs_to_reserve (w, dim.width,
|
|
|
|
|
w->left_margin_width)
|
|
|
|
|
!= w->desired_matrix->right_margin_glyphs))
|
|
|
|
|
*window_change_flags |= CHANGED_LEAF_MATRIX;
|
|
|
|
|
|
|
|
|
|
/* Actually change matrices, if allowed. Do not consider
|
|
|
|
|
CHANGED_LEAF_MATRIX computed above here because the pool
|
|
|
|
|
may have been changed which we don't now here. We trust
|
|
|
|
|
that we only will be called with DIM_ONLY_P != 0 when
|
|
|
|
|
necessary. */
|
|
|
|
|
if (!dim_only_p)
|
|
|
|
|
{
|
|
|
|
|
adjust_glyph_matrix (w, w->desired_matrix, x, y, dim);
|
|
|
|
|
adjust_glyph_matrix (w, w->current_matrix, x, y, dim);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If we are part of a horizontal combination, advance x for
|
|
|
|
|
windows to the right of W; otherwise advance y for windows
|
|
|
|
|
below W. */
|
|
|
|
|
if (in_horz_combination_p)
|
|
|
|
|
x += dim.width;
|
|
|
|
|
else
|
|
|
|
|
y += dim.height;
|
|
|
|
|
|
|
|
|
|
/* Remember maximum glyph matrix dimensions. */
|
|
|
|
|
wmax = max (wmax, dim.width);
|
|
|
|
|
hmax = max (hmax, dim.height);
|
|
|
|
|
|
|
|
|
|
/* Next window on same level. */
|
|
|
|
|
window = w->next;
|
|
|
|
|
}
|
|
|
|
|
while (!NILP (window));
|
|
|
|
|
|
|
|
|
|
/* Set `total' to the total glyph matrix dimension of this window
|
|
|
|
|
level. In a vertical combination, the width is the width of the
|
|
|
|
|
widest window; the height is the y we finally reached, corrected
|
|
|
|
|
by the y we started with. In a horizontal combination, the total
|
|
|
|
|
height is the height of the tallest window, and the width is the
|
|
|
|
|
x we finally reached, corrected by the x we started with. */
|
|
|
|
|
if (in_horz_combination_p)
|
|
|
|
|
{
|
|
|
|
|
total.width = x - x0;
|
|
|
|
|
total.height = hmax;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
total.width = wmax;
|
|
|
|
|
total.height = y - y0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return total;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Allocate window matrices for window-based redisplay. W is the
|
|
|
|
|
window whose matrices must be allocated/reallocated. CH_DIM is the
|
|
|
|
|
size of the smallest character that could potentially be used on W. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
allocate_matrices_for_window_redisplay (w, ch_dim)
|
|
|
|
|
struct window *w;
|
|
|
|
|
struct dim ch_dim;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->vchild))
|
|
|
|
|
allocate_matrices_for_window_redisplay (XWINDOW (w->vchild), ch_dim);
|
|
|
|
|
else if (!NILP (w->hchild))
|
|
|
|
|
allocate_matrices_for_window_redisplay (XWINDOW (w->hchild), ch_dim);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* W is a leaf window. */
|
|
|
|
|
int window_pixel_width = XFLOATINT (w->width) * CANON_X_UNIT (f);
|
|
|
|
|
int window_pixel_height = window_box_height (w) + abs (w->vscroll);
|
|
|
|
|
struct dim dim;
|
|
|
|
|
|
|
|
|
|
/* If matrices are not yet allocated, allocate them now. */
|
|
|
|
|
if (w->desired_matrix == NULL)
|
|
|
|
|
{
|
|
|
|
|
w->desired_matrix = new_glyph_matrix (NULL);
|
|
|
|
|
w->current_matrix = new_glyph_matrix (NULL);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Compute number of glyphs needed in a glyph row. */
|
|
|
|
|
dim.width = (((window_pixel_width + ch_dim.width - 1)
|
|
|
|
|
/ ch_dim.width)
|
|
|
|
|
/* 2 partially visible columns in the text area. */
|
|
|
|
|
+ 2
|
|
|
|
|
/* One partially visible column at the right
|
|
|
|
|
edge of each marginal area. */
|
|
|
|
|
+ 1 + 1);
|
|
|
|
|
|
|
|
|
|
/* Compute number of glyph rows needed. */
|
|
|
|
|
dim.height = (((window_pixel_height + ch_dim.height - 1)
|
|
|
|
|
/ ch_dim.height)
|
|
|
|
|
/* One partially visible line at the top and
|
|
|
|
|
bottom of the window. */
|
|
|
|
|
+ 2
|
|
|
|
|
/* 2 for top and mode line. */
|
|
|
|
|
+ 2);
|
|
|
|
|
|
|
|
|
|
/* Change matrices. */
|
|
|
|
|
adjust_glyph_matrix (w, w->desired_matrix, 0, 0, dim);
|
|
|
|
|
adjust_glyph_matrix (w, w->current_matrix, 0, 0, dim);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? NULL : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Re-allocate/ re-compute glyph matrices on frame F. If F is null,
|
|
|
|
|
do it for all frames; otherwise do it just for the given frame.
|
|
|
|
|
This function must be called when a new frame is created, its size
|
|
|
|
|
changes, or its window configuration changes. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
adjust_glyphs (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* Block input so that expose events and other events that access
|
|
|
|
|
glyph matrices are not processed while we are changing them. */
|
|
|
|
|
BLOCK_INPUT;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (f)
|
|
|
|
|
adjust_frame_glyphs (f);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, lisp_frame;
|
|
|
|
|
|
|
|
|
|
FOR_EACH_FRAME (tail, lisp_frame)
|
|
|
|
|
adjust_frame_glyphs (XFRAME (lisp_frame));
|
|
|
|
|
}
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|
|
|
|
|
UNBLOCK_INPUT;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Adjust frame glyphs when Emacs is initialized.
|
|
|
|
|
|
|
|
|
|
To be called from init_display.
|
|
|
|
|
|
|
|
|
|
We need a glyph matrix because redraw will happen soon.
|
|
|
|
|
Unfortunately, window sizes on selected_frame are not yet set to
|
|
|
|
|
meaningful values. I believe we can assume that there are only two
|
|
|
|
|
windows on the frame---the mini-buffer and the root window. Frame
|
|
|
|
|
height and width seem to be correct so far. So, set the sizes of
|
|
|
|
|
windows to estimated values. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_frame_glyphs_initially ()
|
|
|
|
|
{
|
1999-09-13 11:13:24 +00:00
|
|
|
|
struct frame *sf = SELECTED_FRAME ();
|
|
|
|
|
struct window *root = XWINDOW (sf->root_window);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *mini = XWINDOW (root->next);
|
1999-09-13 11:13:24 +00:00
|
|
|
|
int frame_height = FRAME_HEIGHT (sf);
|
|
|
|
|
int frame_width = FRAME_WIDTH (sf);
|
|
|
|
|
int top_margin = FRAME_TOP_MARGIN (sf);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Do it for the root window. */
|
|
|
|
|
XSETFASTINT (root->top, top_margin);
|
|
|
|
|
XSETFASTINT (root->width, frame_width);
|
1999-09-13 11:13:24 +00:00
|
|
|
|
set_window_height (sf->root_window, frame_height - 1 - top_margin, 0);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Do it for the mini-buffer window. */
|
|
|
|
|
XSETFASTINT (mini->top, frame_height - 1);
|
|
|
|
|
XSETFASTINT (mini->width, frame_width);
|
|
|
|
|
set_window_height (root->next, 1, 0);
|
|
|
|
|
|
1999-09-13 11:13:24 +00:00
|
|
|
|
adjust_frame_glyphs (sf);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
glyphs_initialized_initially_p = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Allocate/reallocate glyph matrices of a single frame F. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_frame_glyphs (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
if (FRAME_WINDOW_P (f))
|
|
|
|
|
adjust_frame_glyphs_for_window_redisplay (f);
|
|
|
|
|
else
|
|
|
|
|
adjust_frame_glyphs_for_frame_redisplay (f);
|
|
|
|
|
|
|
|
|
|
/* Don't forget the message buffer and the buffer for
|
|
|
|
|
decode_mode_spec. */
|
|
|
|
|
adjust_frame_message_buffer (f);
|
|
|
|
|
adjust_decode_mode_spec_buffer (f);
|
|
|
|
|
|
|
|
|
|
f->glyphs_initialized_p = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Allocate/reallocate glyph matrices of a single frame F for
|
|
|
|
|
frame-based redisplay. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_frame_glyphs_for_frame_redisplay (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
struct dim ch_dim;
|
|
|
|
|
struct dim matrix_dim;
|
|
|
|
|
int pool_changed_p;
|
|
|
|
|
int window_change_flags;
|
|
|
|
|
int top_window_y;
|
|
|
|
|
|
|
|
|
|
if (!FRAME_LIVE_P (f))
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
/* Determine the smallest character in any font for F. On
|
|
|
|
|
console windows, all characters have dimension (1, 1). */
|
|
|
|
|
ch_dim.width = ch_dim.height = 1;
|
|
|
|
|
|
|
|
|
|
top_window_y = FRAME_TOP_MARGIN (f);
|
|
|
|
|
|
|
|
|
|
/* Allocate glyph pool structures if not already done. */
|
|
|
|
|
if (f->desired_pool == NULL)
|
|
|
|
|
{
|
|
|
|
|
f->desired_pool = new_glyph_pool ();
|
|
|
|
|
f->current_pool = new_glyph_pool ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Allocate frames matrix structures if needed. */
|
|
|
|
|
if (f->desired_matrix == NULL)
|
|
|
|
|
{
|
|
|
|
|
f->desired_matrix = new_glyph_matrix (f->desired_pool);
|
|
|
|
|
f->current_matrix = new_glyph_matrix (f->current_pool);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Compute window glyph matrices. (This takes the mini-buffer
|
|
|
|
|
window into account). The result is the size of the frame glyph
|
|
|
|
|
matrix needed. The variable window_change_flags is set to a bit
|
|
|
|
|
mask indicating whether new matrices will be allocated or
|
|
|
|
|
existing matrices change their size or location within the frame
|
|
|
|
|
matrix. */
|
|
|
|
|
window_change_flags = 0;
|
|
|
|
|
matrix_dim
|
|
|
|
|
= allocate_matrices_for_frame_redisplay (FRAME_ROOT_WINDOW (f),
|
|
|
|
|
0, top_window_y,
|
|
|
|
|
ch_dim, 1,
|
|
|
|
|
&window_change_flags);
|
|
|
|
|
|
|
|
|
|
/* Add in menu bar lines, if any. */
|
|
|
|
|
matrix_dim.height += top_window_y;
|
|
|
|
|
|
|
|
|
|
/* Enlarge pools as necessary. */
|
|
|
|
|
pool_changed_p = realloc_glyph_pool (f->desired_pool, matrix_dim);
|
|
|
|
|
realloc_glyph_pool (f->current_pool, matrix_dim);
|
|
|
|
|
|
|
|
|
|
/* Set up glyph pointers within window matrices. Do this only if
|
|
|
|
|
absolutely necessary since it requires a frame redraw. */
|
|
|
|
|
if (pool_changed_p || window_change_flags)
|
|
|
|
|
{
|
|
|
|
|
/* Do it for window matrices. */
|
|
|
|
|
allocate_matrices_for_frame_redisplay (FRAME_ROOT_WINDOW (f),
|
|
|
|
|
0, top_window_y, ch_dim, 0,
|
|
|
|
|
&window_change_flags);
|
|
|
|
|
|
|
|
|
|
/* Size of frame matrices must equal size of frame. Note
|
|
|
|
|
that we are called for X frames with window widths NOT equal
|
|
|
|
|
to the frame width (from CHANGE_FRAME_SIZE_1). */
|
|
|
|
|
xassert (matrix_dim.width == FRAME_WIDTH (f)
|
|
|
|
|
&& matrix_dim.height == FRAME_HEIGHT (f));
|
|
|
|
|
|
|
|
|
|
/* Resize frame matrices. */
|
|
|
|
|
adjust_glyph_matrix (NULL, f->desired_matrix, 0, 0, matrix_dim);
|
|
|
|
|
adjust_glyph_matrix (NULL, f->current_matrix, 0, 0, matrix_dim);
|
|
|
|
|
|
|
|
|
|
/* Since location and size of sub-matrices within the pool may
|
|
|
|
|
have changed, and current matrices don't have meaningful
|
|
|
|
|
contents anymore, mark the frame garbaged. */
|
|
|
|
|
SET_FRAME_GARBAGED (f);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Allocate/reallocate glyph matrices of a single frame F for
|
|
|
|
|
window-based redisplay. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_frame_glyphs_for_window_redisplay (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
struct dim ch_dim;
|
|
|
|
|
struct window *w;
|
|
|
|
|
|
|
|
|
|
xassert (FRAME_WINDOW_P (f) && FRAME_LIVE_P (f));
|
|
|
|
|
|
|
|
|
|
/* Get minimum sizes. */
|
|
|
|
|
#ifdef HAVE_WINDOW_SYSTEM
|
|
|
|
|
ch_dim.width = FRAME_SMALLEST_CHAR_WIDTH (f);
|
|
|
|
|
ch_dim.height = FRAME_SMALLEST_FONT_HEIGHT (f);
|
|
|
|
|
#else
|
|
|
|
|
ch_dim.width = ch_dim.height = 1;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
/* Allocate/reallocate window matrices. */
|
|
|
|
|
allocate_matrices_for_window_redisplay (XWINDOW (FRAME_ROOT_WINDOW (f)),
|
|
|
|
|
ch_dim);
|
|
|
|
|
|
|
|
|
|
/* Allocate/ reallocate matrices of the dummy window used to display
|
|
|
|
|
the menu bar under X when no X toolkit support is available. */
|
|
|
|
|
#ifndef USE_X_TOOLKIT
|
|
|
|
|
{
|
|
|
|
|
/* Allocate a dummy window if not already done. */
|
|
|
|
|
if (NILP (f->menu_bar_window))
|
|
|
|
|
{
|
|
|
|
|
f->menu_bar_window = make_window ();
|
|
|
|
|
w = XWINDOW (f->menu_bar_window);
|
|
|
|
|
XSETFRAME (w->frame, f);
|
|
|
|
|
w->pseudo_window_p = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
w = XWINDOW (f->menu_bar_window);
|
|
|
|
|
|
|
|
|
|
/* Set window dimensions to frame dimensions and allocate or
|
|
|
|
|
adjust glyph matrices of W. */
|
|
|
|
|
XSETFASTINT (w->top, 0);
|
|
|
|
|
XSETFASTINT (w->left, 0);
|
|
|
|
|
XSETFASTINT (w->height, FRAME_MENU_BAR_LINES (f));
|
|
|
|
|
XSETFASTINT (w->width, FRAME_WINDOW_WIDTH (f));
|
|
|
|
|
allocate_matrices_for_window_redisplay (w, ch_dim);
|
|
|
|
|
}
|
|
|
|
|
#endif /* not USE_X_TOOLKIT */
|
|
|
|
|
|
1999-09-05 15:49:07 +00:00
|
|
|
|
/* Allocate/ reallocate matrices of the tool bar window. If we
|
|
|
|
|
don't have a tool bar window yet, make one. */
|
|
|
|
|
if (NILP (f->tool_bar_window))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-09-05 15:49:07 +00:00
|
|
|
|
f->tool_bar_window = make_window ();
|
|
|
|
|
w = XWINDOW (f->tool_bar_window);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
XSETFRAME (w->frame, f);
|
|
|
|
|
w->pseudo_window_p = 1;
|
|
|
|
|
}
|
|
|
|
|
else
|
1999-09-05 15:49:07 +00:00
|
|
|
|
w = XWINDOW (f->tool_bar_window);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
XSETFASTINT (w->top, FRAME_MENU_BAR_LINES (f));
|
|
|
|
|
XSETFASTINT (w->left, 0);
|
1999-09-05 15:49:07 +00:00
|
|
|
|
XSETFASTINT (w->height, FRAME_TOOL_BAR_LINES (f));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
XSETFASTINT (w->width, FRAME_WINDOW_WIDTH (f));
|
|
|
|
|
allocate_matrices_for_window_redisplay (w, ch_dim);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Adjust/ allocate message buffer of frame F.
|
|
|
|
|
|
|
|
|
|
Note that the message buffer is never freed. Since I could not
|
|
|
|
|
find a free in 19.34, I assume that freeing it would be
|
|
|
|
|
problematic in some way and don't do it either.
|
|
|
|
|
|
|
|
|
|
(Implementation note: It should be checked if we can free it
|
|
|
|
|
eventually without causing trouble). */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_frame_message_buffer (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
int size = FRAME_MESSAGE_BUF_SIZE (f) + 1;
|
|
|
|
|
|
|
|
|
|
if (FRAME_MESSAGE_BUF (f))
|
|
|
|
|
{
|
|
|
|
|
char *buffer = FRAME_MESSAGE_BUF (f);
|
|
|
|
|
char *new_buffer = (char *) xrealloc (buffer, size);
|
|
|
|
|
FRAME_MESSAGE_BUF (f) = new_buffer;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
FRAME_MESSAGE_BUF (f) = (char *) xmalloc (size);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Re-allocate buffer for decode_mode_spec on frame F. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
adjust_decode_mode_spec_buffer (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
f->decode_mode_spec_buffer
|
|
|
|
|
= (char *) xrealloc (f->decode_mode_spec_buffer,
|
|
|
|
|
FRAME_MESSAGE_BUF_SIZE (f) + 1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
Freeing Glyph Matrices
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Free glyph memory for a frame F. F may be null. This function can
|
|
|
|
|
be called for the same frame more than once. The root window of
|
|
|
|
|
F may be nil when this function is called. This is the case when
|
|
|
|
|
the function is called when F is destroyed. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
free_glyphs (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
if (f && f->glyphs_initialized_p)
|
|
|
|
|
{
|
2000-03-02 20:11:02 +00:00
|
|
|
|
/* Block interrupt input so that we don't get surprised by an X
|
|
|
|
|
event while we're in an inconsistent state. */
|
|
|
|
|
BLOCK_INPUT;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
f->glyphs_initialized_p = 0;
|
|
|
|
|
|
|
|
|
|
/* Release window sub-matrices. */
|
|
|
|
|
if (!NILP (f->root_window))
|
|
|
|
|
free_window_matrices (XWINDOW (f->root_window));
|
|
|
|
|
|
|
|
|
|
/* Free the dummy window for menu bars without X toolkit and its
|
|
|
|
|
glyph matrices. */
|
|
|
|
|
if (!NILP (f->menu_bar_window))
|
|
|
|
|
{
|
|
|
|
|
struct window *w = XWINDOW (f->menu_bar_window);
|
|
|
|
|
free_glyph_matrix (w->desired_matrix);
|
|
|
|
|
free_glyph_matrix (w->current_matrix);
|
|
|
|
|
w->desired_matrix = w->current_matrix = NULL;
|
|
|
|
|
f->menu_bar_window = Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1999-09-05 15:49:07 +00:00
|
|
|
|
/* Free the tool bar window and its glyph matrices. */
|
|
|
|
|
if (!NILP (f->tool_bar_window))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-09-05 15:49:07 +00:00
|
|
|
|
struct window *w = XWINDOW (f->tool_bar_window);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
free_glyph_matrix (w->desired_matrix);
|
|
|
|
|
free_glyph_matrix (w->current_matrix);
|
|
|
|
|
w->desired_matrix = w->current_matrix = NULL;
|
1999-09-05 15:49:07 +00:00
|
|
|
|
f->tool_bar_window = Qnil;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Release frame glyph matrices. Reset fields to zero in
|
|
|
|
|
case we are called a second time. */
|
|
|
|
|
if (f->desired_matrix)
|
|
|
|
|
{
|
|
|
|
|
free_glyph_matrix (f->desired_matrix);
|
|
|
|
|
free_glyph_matrix (f->current_matrix);
|
|
|
|
|
f->desired_matrix = f->current_matrix = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Release glyph pools. */
|
|
|
|
|
if (f->desired_pool)
|
|
|
|
|
{
|
|
|
|
|
free_glyph_pool (f->desired_pool);
|
|
|
|
|
free_glyph_pool (f->current_pool);
|
|
|
|
|
f->desired_pool = f->current_pool = NULL;
|
|
|
|
|
}
|
2000-03-02 20:11:02 +00:00
|
|
|
|
|
|
|
|
|
UNBLOCK_INPUT;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Free glyph sub-matrices in the window tree rooted at W. This
|
|
|
|
|
function may be called with a null pointer, and it may be called on
|
|
|
|
|
the same tree more than once. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
free_window_matrices (w)
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
free_window_matrices (XWINDOW (w->hchild));
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
free_window_matrices (XWINDOW (w->vchild));
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* This is a leaf window. Free its memory and reset fields
|
|
|
|
|
to zero in case this function is called a second time for
|
|
|
|
|
W. */
|
|
|
|
|
free_glyph_matrix (w->current_matrix);
|
|
|
|
|
free_glyph_matrix (w->desired_matrix);
|
|
|
|
|
w->current_matrix = w->desired_matrix = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Next window on same level. */
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Check glyph memory leaks. This function is called from
|
|
|
|
|
shut_down_emacs. Note that frames are not destroyed when Emacs
|
|
|
|
|
exits. We therefore free all glyph memory for all active frames
|
|
|
|
|
explicitly and check that nothing is left allocated. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
check_glyph_memory ()
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame;
|
|
|
|
|
|
|
|
|
|
/* Free glyph memory for all frames. */
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
free_glyphs (XFRAME (frame));
|
|
|
|
|
|
|
|
|
|
/* Check that nothing is left allocated. */
|
|
|
|
|
if (glyph_matrix_count)
|
|
|
|
|
abort ();
|
|
|
|
|
if (glyph_pool_count)
|
|
|
|
|
abort ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
Building a Frame Matrix
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Most of the redisplay code works on glyph matrices attached to
|
|
|
|
|
windows. This is a good solution most of the time, but it is not
|
|
|
|
|
suitable for terminal code. Terminal output functions cannot rely
|
|
|
|
|
on being able to set an arbitrary terminal window. Instead they
|
|
|
|
|
must be provided with a view of the whole frame, i.e. the whole
|
|
|
|
|
screen. We build such a view by constructing a frame matrix from
|
|
|
|
|
window matrices in this section.
|
|
|
|
|
|
|
|
|
|
Windows that must be updated have their must_be_update_p flag set.
|
|
|
|
|
For all such windows, their desired matrix is made part of the
|
|
|
|
|
desired frame matrix. For other windows, their current matrix is
|
|
|
|
|
made part of the desired frame matrix.
|
|
|
|
|
|
|
|
|
|
+-----------------+----------------+
|
|
|
|
|
| desired | desired |
|
|
|
|
|
| | |
|
|
|
|
|
+-----------------+----------------+
|
|
|
|
|
| current |
|
|
|
|
|
| |
|
|
|
|
|
+----------------------------------+
|
|
|
|
|
|
|
|
|
|
Desired window matrices can be made part of the frame matrix in a
|
|
|
|
|
cheap way: We exploit the fact that the desired frame matrix and
|
|
|
|
|
desired window matrices share their glyph memory. This is not
|
|
|
|
|
possible for current window matrices. Their glyphs are copied to
|
|
|
|
|
the desired frame matrix. The latter is equivalent to
|
|
|
|
|
preserve_other_columns in the old redisplay.
|
|
|
|
|
|
|
|
|
|
Used glyphs counters for frame matrix rows are the result of adding
|
|
|
|
|
up glyph lengths of the window matrices. A line in the frame
|
|
|
|
|
matrix is enabled, if a corresponding line in a window matrix is
|
|
|
|
|
enabled.
|
|
|
|
|
|
|
|
|
|
After building the desired frame matrix, it will be passed to
|
|
|
|
|
terminal code, which will manipulate both the desired and current
|
|
|
|
|
frame matrix. Changes applied to the frame's current matrix have
|
|
|
|
|
to be visible in current window matrices afterwards, of course.
|
|
|
|
|
|
|
|
|
|
This problem is solved like this:
|
|
|
|
|
|
|
|
|
|
1. Window and frame matrices share glyphs. Window matrices are
|
|
|
|
|
constructed in a way that their glyph contents ARE the glyph
|
|
|
|
|
contents needed in a frame matrix. Thus, any modification of
|
|
|
|
|
glyphs done in terminal code will be reflected in window matrices
|
|
|
|
|
automatically.
|
|
|
|
|
|
|
|
|
|
2. Exchanges of rows in a frame matrix done by terminal code are
|
|
|
|
|
intercepted by hook functions so that corresponding row operations
|
|
|
|
|
on window matrices can be performed. This is necessary because we
|
|
|
|
|
use pointers to glyphs in glyph row structures. To satisfy the
|
|
|
|
|
assumption of point 1 above that glyphs are updated implicitly in
|
|
|
|
|
window matrices when they are manipulated via the frame matrix,
|
|
|
|
|
window and frame matrix must of course agree where to find the
|
|
|
|
|
glyphs for their rows. Possible manipulations that must be
|
|
|
|
|
mirrored are assignments of rows of the desired frame matrix to the
|
|
|
|
|
current frame matrix and scrolling the current frame matrix. */
|
|
|
|
|
|
|
|
|
|
/* Build frame F's desired matrix from window matrices. Only windows
|
|
|
|
|
which have the flag must_be_updated_p set have to be updated. Menu
|
|
|
|
|
bar lines of a frame are not covered by window matrices, so make
|
|
|
|
|
sure not to touch them in this function. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
build_frame_matrix (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* F must have a frame matrix when this function is called. */
|
|
|
|
|
xassert (!FRAME_WINDOW_P (f));
|
|
|
|
|
|
|
|
|
|
/* Clear all rows in the frame matrix covered by window matrices.
|
|
|
|
|
Menu bar lines are not covered by windows. */
|
|
|
|
|
for (i = FRAME_TOP_MARGIN (f); i < f->desired_matrix->nrows; ++i)
|
|
|
|
|
clear_glyph_row (MATRIX_ROW (f->desired_matrix, i));
|
|
|
|
|
|
|
|
|
|
/* Build the matrix by walking the window tree. */
|
|
|
|
|
build_frame_matrix_from_window_tree (f->desired_matrix,
|
|
|
|
|
XWINDOW (FRAME_ROOT_WINDOW (f)));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Walk a window tree, building a frame matrix MATRIX from window
|
|
|
|
|
matrices. W is the root of a window tree. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
build_frame_matrix_from_window_tree (matrix, w)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
build_frame_matrix_from_window_tree (matrix, XWINDOW (w->hchild));
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
build_frame_matrix_from_window_tree (matrix, XWINDOW (w->vchild));
|
|
|
|
|
else
|
|
|
|
|
build_frame_matrix_from_leaf_window (matrix, w);
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Add a window's matrix to a frame matrix. FRAME_MATRIX is the
|
|
|
|
|
desired frame matrix built. W is a leaf window whose desired or
|
|
|
|
|
current matrix is to be added to FRAME_MATRIX. W's flag
|
|
|
|
|
must_be_updated_p determines which matrix it contributes to
|
|
|
|
|
FRAME_MATRIX. If must_be_updated_p is non-zero, W's desired matrix
|
|
|
|
|
is added to FRAME_MATRIX, otherwise W's current matrix is added.
|
|
|
|
|
Adding a desired matrix means setting up used counters and such in
|
|
|
|
|
frame rows, while adding a current window matrix to FRAME_MATRIX
|
|
|
|
|
means copying glyphs. The latter case corresponds to
|
|
|
|
|
preserve_other_columns in the old redisplay. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
build_frame_matrix_from_leaf_window (frame_matrix, w)
|
|
|
|
|
struct glyph_matrix *frame_matrix;
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
struct glyph_matrix *window_matrix;
|
|
|
|
|
int window_y, frame_y;
|
|
|
|
|
/* If non-zero, a glyph to insert at the right border of W. */
|
|
|
|
|
GLYPH right_border_glyph = 0;
|
|
|
|
|
|
|
|
|
|
/* Set window_matrix to the matrix we have to add to FRAME_MATRIX. */
|
|
|
|
|
if (w->must_be_updated_p)
|
|
|
|
|
{
|
|
|
|
|
window_matrix = w->desired_matrix;
|
|
|
|
|
|
|
|
|
|
/* Decide whether we want to add a vertical border glyph. */
|
|
|
|
|
if (!WINDOW_RIGHTMOST_P (w))
|
|
|
|
|
{
|
|
|
|
|
struct Lisp_Char_Table *dp = window_display_table (w);
|
|
|
|
|
right_border_glyph = (dp && INTEGERP (DISP_BORDER_GLYPH (dp))
|
|
|
|
|
? XINT (DISP_BORDER_GLYPH (dp))
|
|
|
|
|
: '|');
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
window_matrix = w->current_matrix;
|
|
|
|
|
|
|
|
|
|
/* For all rows in the window matrix and corresponding rows in the
|
|
|
|
|
frame matrix. */
|
|
|
|
|
window_y = 0;
|
|
|
|
|
frame_y = window_matrix->matrix_y;
|
|
|
|
|
while (window_y < window_matrix->nrows)
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *frame_row = frame_matrix->rows + frame_y;
|
|
|
|
|
struct glyph_row *window_row = window_matrix->rows + window_y;
|
2000-08-29 15:04:48 +00:00
|
|
|
|
int current_row_p = window_matrix == w->current_matrix;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Fill up the frame row with spaces up to the left margin of the
|
|
|
|
|
window row. */
|
|
|
|
|
fill_up_frame_row_with_spaces (frame_row, window_matrix->matrix_x);
|
|
|
|
|
|
|
|
|
|
/* Fill up areas in the window matrix row with spaces. */
|
|
|
|
|
fill_up_glyph_row_with_spaces (window_row);
|
2000-08-29 15:04:48 +00:00
|
|
|
|
|
|
|
|
|
/* If only part of W's desired matrix has been built, and
|
|
|
|
|
window_row wasn't displayed, use the corresponding current
|
|
|
|
|
row instead. */
|
|
|
|
|
if (window_matrix == w->desired_matrix
|
|
|
|
|
&& !window_row->enabled_p)
|
|
|
|
|
{
|
|
|
|
|
window_row = w->current_matrix->rows + window_y;
|
|
|
|
|
current_row_p = 1;
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
2000-08-29 15:04:48 +00:00
|
|
|
|
if (current_row_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
2000-08-29 15:04:48 +00:00
|
|
|
|
/* Copy window row to frame row. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
bcopy (window_row->glyphs[0],
|
|
|
|
|
frame_row->glyphs[TEXT_AREA] + window_matrix->matrix_x,
|
|
|
|
|
window_matrix->matrix_w * sizeof (struct glyph));
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
2000-08-29 15:04:48 +00:00
|
|
|
|
xassert (window_row->enabled_p);
|
|
|
|
|
|
|
|
|
|
/* Only when a desired row has been displayed, we want
|
|
|
|
|
the corresponding frame row to be updated. */
|
|
|
|
|
frame_row->enabled_p = 1;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Maybe insert a vertical border between horizontally adjacent
|
|
|
|
|
windows. */
|
|
|
|
|
if (right_border_glyph)
|
|
|
|
|
{
|
|
|
|
|
struct glyph *border = window_row->glyphs[LAST_AREA] - 1;
|
|
|
|
|
SET_CHAR_GLYPH_FROM_GLYPH (*border, right_border_glyph);
|
|
|
|
|
}
|
|
|
|
|
|
2000-02-25 13:20:36 +00:00
|
|
|
|
#if 0 /* This shouldn't be necessary. Let's check it. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Due to hooks installed, it normally doesn't happen that
|
|
|
|
|
window rows and frame rows of the same matrix are out of
|
|
|
|
|
sync, i.e. have a different understanding of where to
|
|
|
|
|
find glyphs for the row. The following is a safety-belt
|
|
|
|
|
that doesn't cost much and makes absolutely sure that
|
|
|
|
|
window and frame matrices are in sync. */
|
|
|
|
|
if (!glyph_row_slice_p (window_row, frame_row))
|
|
|
|
|
{
|
|
|
|
|
/* Find the row in the window being a slice. There
|
|
|
|
|
should exist one from program logic. */
|
|
|
|
|
struct glyph_row *slice_row
|
|
|
|
|
= find_glyph_row_slice (window_matrix, frame_matrix, frame_y);
|
|
|
|
|
xassert (slice_row != 0);
|
|
|
|
|
|
|
|
|
|
/* Exchange glyphs between both window rows. */
|
|
|
|
|
swap_glyphs_in_rows (window_row, slice_row);
|
|
|
|
|
|
|
|
|
|
/* Exchange pointers between both rows. */
|
|
|
|
|
swap_glyph_pointers (window_row, slice_row);
|
|
|
|
|
}
|
2000-02-25 13:20:36 +00:00
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
2000-02-25 13:20:36 +00:00
|
|
|
|
/* Window row window_y must be a slice of frame row
|
|
|
|
|
frame_y. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
xassert (glyph_row_slice_p (window_row, frame_row));
|
2000-02-25 13:20:36 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If rows are in sync, we don't have to copy glyphs because
|
|
|
|
|
frame and window share glyphs. */
|
1999-08-11 10:47:38 +00:00
|
|
|
|
|
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
strcpy (w->current_matrix->method, w->desired_matrix->method);
|
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Set number of used glyphs in the frame matrix. Since we fill
|
|
|
|
|
up with spaces, and visit leaf windows from left to right it
|
|
|
|
|
can be done simply. */
|
|
|
|
|
frame_row->used[TEXT_AREA]
|
|
|
|
|
= window_matrix->matrix_x + window_matrix->matrix_w;
|
|
|
|
|
|
2000-08-29 15:04:48 +00:00
|
|
|
|
/* Or in other flags. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
frame_row->inverse_p |= window_row->inverse_p;
|
|
|
|
|
|
|
|
|
|
/* Next row. */
|
|
|
|
|
++window_y;
|
|
|
|
|
++frame_y;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Add spaces to a glyph row ROW in a window matrix.
|
|
|
|
|
|
|
|
|
|
Each row has the form:
|
|
|
|
|
|
|
|
|
|
+---------+-----------------------------+------------+
|
|
|
|
|
| left | text | right |
|
|
|
|
|
+---------+-----------------------------+------------+
|
|
|
|
|
|
|
|
|
|
Left and right marginal areas are optional. This function adds
|
|
|
|
|
spaces to areas so that there are no empty holes between areas.
|
|
|
|
|
In other words: If the right area is not empty, the text area
|
|
|
|
|
is filled up with spaces up to the right area. If the text area
|
|
|
|
|
is not empty, the left area is filled up.
|
|
|
|
|
|
|
|
|
|
To be called for frame-based redisplay, only. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fill_up_glyph_row_with_spaces (row)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
{
|
|
|
|
|
fill_up_glyph_row_area_with_spaces (row, LEFT_MARGIN_AREA);
|
|
|
|
|
fill_up_glyph_row_area_with_spaces (row, TEXT_AREA);
|
|
|
|
|
fill_up_glyph_row_area_with_spaces (row, RIGHT_MARGIN_AREA);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Fill area AREA of glyph row ROW with spaces. To be called for
|
|
|
|
|
frame-based redisplay only. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fill_up_glyph_row_area_with_spaces (row, area)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
int area;
|
|
|
|
|
{
|
|
|
|
|
if (row->glyphs[area] < row->glyphs[area + 1])
|
|
|
|
|
{
|
|
|
|
|
struct glyph *end = row->glyphs[area + 1];
|
|
|
|
|
struct glyph *text = row->glyphs[area] + row->used[area];
|
|
|
|
|
|
|
|
|
|
while (text < end)
|
|
|
|
|
*text++ = space_glyph;
|
|
|
|
|
row->used[area] = text - row->glyphs[area];
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Add spaces to the end of ROW in a frame matrix until index UPTO is
|
|
|
|
|
reached. In frame matrices only one area, TEXT_AREA, is used. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
fill_up_frame_row_with_spaces (row, upto)
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
int upto;
|
|
|
|
|
{
|
|
|
|
|
int i = row->used[TEXT_AREA];
|
|
|
|
|
struct glyph *glyph = row->glyphs[TEXT_AREA];
|
|
|
|
|
|
|
|
|
|
while (i < upto)
|
|
|
|
|
glyph[i++] = space_glyph;
|
|
|
|
|
|
|
|
|
|
row->used[TEXT_AREA] = i;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
Mirroring operations on frame matrices in window matrices
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Set frame being updated via frame-based redisplay to F. This
|
|
|
|
|
function must be called before updates to make explicit that we are
|
|
|
|
|
working on frame matrices or not. */
|
|
|
|
|
|
|
|
|
|
static INLINE void
|
|
|
|
|
set_frame_matrix_frame (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
frame_matrix_frame = f;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Make sure glyph row ROW in CURRENT_MATRIX is up to date.
|
|
|
|
|
DESIRED_MATRIX is the desired matrix corresponding to
|
|
|
|
|
CURRENT_MATRIX. The update is done by exchanging glyph pointers
|
|
|
|
|
between rows in CURRENT_MATRIX and DESIRED_MATRIX. If
|
|
|
|
|
frame_matrix_frame is non-null, this indicates that the exchange is
|
|
|
|
|
done in frame matrices, and that we have to perform analogous
|
|
|
|
|
operations in window matrices of frame_matrix_frame. */
|
|
|
|
|
|
|
|
|
|
static INLINE void
|
|
|
|
|
make_current (desired_matrix, current_matrix, row)
|
|
|
|
|
struct glyph_matrix *desired_matrix, *current_matrix;
|
|
|
|
|
int row;
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *current_row = MATRIX_ROW (current_matrix, row);
|
|
|
|
|
struct glyph_row *desired_row = MATRIX_ROW (desired_matrix, row);
|
2000-07-12 15:10:30 +00:00
|
|
|
|
int mouse_face_p = current_row->mouse_face_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Do current_row = desired_row. This exchanges glyph pointers
|
|
|
|
|
between both rows, and does a structure assignment otherwise. */
|
|
|
|
|
assign_row (current_row, desired_row);
|
|
|
|
|
|
|
|
|
|
/* Enable current_row to mark it as valid. */
|
|
|
|
|
current_row->enabled_p = 1;
|
2000-07-12 15:10:30 +00:00
|
|
|
|
current_row->mouse_face_p = mouse_face_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* If we are called on frame matrices, perform analogous operations
|
|
|
|
|
for window matrices. */
|
|
|
|
|
if (frame_matrix_frame)
|
|
|
|
|
mirror_make_current (XWINDOW (frame_matrix_frame->root_window), row);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* W is the root of a window tree. FRAME_ROW is the index of a row in
|
|
|
|
|
W's frame which has been made current (by swapping pointers between
|
|
|
|
|
current and desired matrix). Perform analogous operations in the
|
|
|
|
|
matrices of leaf windows in the window tree rooted at W. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
mirror_make_current (w, frame_row)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int frame_row;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
mirror_make_current (XWINDOW (w->hchild), frame_row);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
mirror_make_current (XWINDOW (w->vchild), frame_row);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* Row relative to window W. Don't use FRAME_TO_WINDOW_VPOS
|
|
|
|
|
here because the checks performed in debug mode there
|
|
|
|
|
will not allow the conversion. */
|
|
|
|
|
int row = frame_row - w->desired_matrix->matrix_y;
|
|
|
|
|
|
|
|
|
|
/* If FRAME_ROW is within W, assign the desired row to the
|
|
|
|
|
current row (exchanging glyph pointers). */
|
|
|
|
|
if (row >= 0 && row < w->desired_matrix->matrix_h)
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *current_row
|
|
|
|
|
= MATRIX_ROW (w->current_matrix, row);
|
|
|
|
|
struct glyph_row *desired_row
|
|
|
|
|
= MATRIX_ROW (w->desired_matrix, row);
|
1999-08-11 10:47:38 +00:00
|
|
|
|
|
|
|
|
|
if (desired_row->enabled_p)
|
|
|
|
|
assign_row (current_row, desired_row);
|
|
|
|
|
else
|
|
|
|
|
swap_glyph_pointers (desired_row, current_row);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
current_row->enabled_p = 1;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Perform row dance after scrolling. We are working on the range of
|
|
|
|
|
lines UNCHANGED_AT_TOP + 1 to UNCHANGED_AT_TOP + NLINES (not
|
|
|
|
|
including) in MATRIX. COPY_FROM is a vector containing, for each
|
|
|
|
|
row I in the range 0 <= I < NLINES, the index of the original line
|
|
|
|
|
to move to I. This index is relative to the row range, i.e. 0 <=
|
|
|
|
|
index < NLINES. RETAINED_P is a vector containing zero for each
|
|
|
|
|
row 0 <= I < NLINES which is empty.
|
|
|
|
|
|
|
|
|
|
This function is called from do_scrolling and do_direct_scrolling. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
mirrored_line_dance (matrix, unchanged_at_top, nlines, copy_from,
|
|
|
|
|
retained_p)
|
|
|
|
|
struct glyph_matrix *matrix;
|
|
|
|
|
int unchanged_at_top, nlines;
|
|
|
|
|
int *copy_from;
|
|
|
|
|
char *retained_p;
|
|
|
|
|
{
|
|
|
|
|
/* A copy of original rows. */
|
|
|
|
|
struct glyph_row *old_rows;
|
|
|
|
|
|
|
|
|
|
/* Rows to assign to. */
|
|
|
|
|
struct glyph_row *new_rows = MATRIX_ROW (matrix, unchanged_at_top);
|
|
|
|
|
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* Make a copy of the original rows. */
|
|
|
|
|
old_rows = (struct glyph_row *) alloca (nlines * sizeof *old_rows);
|
|
|
|
|
bcopy (new_rows, old_rows, nlines * sizeof *old_rows);
|
|
|
|
|
|
|
|
|
|
/* Assign new rows, maybe clear lines. */
|
|
|
|
|
for (i = 0; i < nlines; ++i)
|
|
|
|
|
{
|
|
|
|
|
int enabled_before_p = new_rows[i].enabled_p;
|
|
|
|
|
|
|
|
|
|
xassert (i + unchanged_at_top < matrix->nrows);
|
|
|
|
|
xassert (unchanged_at_top + copy_from[i] < matrix->nrows);
|
|
|
|
|
new_rows[i] = old_rows[copy_from[i]];
|
|
|
|
|
new_rows[i].enabled_p = enabled_before_p;
|
|
|
|
|
|
|
|
|
|
/* RETAINED_P is zero for empty lines. */
|
|
|
|
|
if (!retained_p[copy_from[i]])
|
|
|
|
|
new_rows[i].enabled_p = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Do the same for window matrices, if MATRIX Is a frame matrix. */
|
|
|
|
|
if (frame_matrix_frame)
|
|
|
|
|
mirror_line_dance (XWINDOW (frame_matrix_frame->root_window),
|
|
|
|
|
unchanged_at_top, nlines, copy_from, retained_p);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2000-02-25 13:20:36 +00:00
|
|
|
|
/* Synchronize glyph pointers in the current matrix of window W with
|
|
|
|
|
the current frame matrix. W must be full-width, and be on a tty
|
|
|
|
|
frame. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
sync_window_with_frame_matrix_rows (w)
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
struct glyph_row *window_row, *window_row_end, *frame_row;
|
|
|
|
|
|
|
|
|
|
/* Preconditions: W must be a leaf window and full-width. Its frame
|
|
|
|
|
must have a frame matrix. */
|
|
|
|
|
xassert (NILP (w->hchild) && NILP (w->vchild));
|
|
|
|
|
xassert (WINDOW_FULL_WIDTH_P (w));
|
|
|
|
|
xassert (!FRAME_WINDOW_P (f));
|
|
|
|
|
|
|
|
|
|
/* If W is a full-width window, glyph pointers in W's current matrix
|
|
|
|
|
have, by definition, to be the same as glyph pointers in the
|
|
|
|
|
corresponding frame matrix. */
|
|
|
|
|
window_row = w->current_matrix->rows;
|
|
|
|
|
window_row_end = window_row + w->current_matrix->nrows;
|
|
|
|
|
frame_row = f->current_matrix->rows + XFASTINT (w->top);
|
|
|
|
|
while (window_row < window_row_end)
|
|
|
|
|
{
|
|
|
|
|
int area;
|
|
|
|
|
|
|
|
|
|
for (area = LEFT_MARGIN_AREA; area <= LAST_AREA; ++area)
|
|
|
|
|
window_row->glyphs[area] = frame_row->glyphs[area];
|
|
|
|
|
|
|
|
|
|
++window_row, ++frame_row;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Return the window in the window tree rooted in W containing frame
|
|
|
|
|
row ROW. Value is null if none is found. */
|
|
|
|
|
|
|
|
|
|
struct window *
|
|
|
|
|
frame_row_to_window (w, row)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int row;
|
|
|
|
|
{
|
|
|
|
|
struct window *found = NULL;
|
|
|
|
|
|
|
|
|
|
while (w && !found)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
found = frame_row_to_window (XWINDOW (w->hchild), row);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
found = frame_row_to_window (XWINDOW (w->vchild), row);
|
|
|
|
|
else if (row >= XFASTINT (w->top)
|
|
|
|
|
&& row < XFASTINT (w->top) + XFASTINT (w->height))
|
|
|
|
|
found = w;
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return found;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Perform a line dance in the window tree rooted at W, after
|
|
|
|
|
scrolling a frame matrix in mirrored_line_dance.
|
|
|
|
|
|
|
|
|
|
We are working on the range of lines UNCHANGED_AT_TOP + 1 to
|
|
|
|
|
UNCHANGED_AT_TOP + NLINES (not including) in W's frame matrix.
|
|
|
|
|
COPY_FROM is a vector containing, for each row I in the range 0 <=
|
|
|
|
|
I < NLINES, the index of the original line to move to I. This
|
|
|
|
|
index is relative to the row range, i.e. 0 <= index < NLINES.
|
|
|
|
|
RETAINED_P is a vector containing zero for each row 0 <= I < NLINES
|
|
|
|
|
which is empty. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
mirror_line_dance (w, unchanged_at_top, nlines, copy_from, retained_p)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int unchanged_at_top, nlines;
|
|
|
|
|
int *copy_from;
|
|
|
|
|
char *retained_p;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
mirror_line_dance (XWINDOW (w->hchild), unchanged_at_top,
|
|
|
|
|
nlines, copy_from, retained_p);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
mirror_line_dance (XWINDOW (w->vchild), unchanged_at_top,
|
|
|
|
|
nlines, copy_from, retained_p);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* W is a leaf window, and we are working on its current
|
|
|
|
|
matrix m. */
|
|
|
|
|
struct glyph_matrix *m = w->current_matrix;
|
2000-02-25 13:20:36 +00:00
|
|
|
|
int i, sync_p = 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *old_rows;
|
|
|
|
|
|
|
|
|
|
/* Make a copy of the original rows of matrix m. */
|
|
|
|
|
old_rows = (struct glyph_row *) alloca (m->nrows * sizeof *old_rows);
|
|
|
|
|
bcopy (m->rows, old_rows, m->nrows * sizeof *old_rows);
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < nlines; ++i)
|
|
|
|
|
{
|
|
|
|
|
/* Frame relative line assigned to. */
|
|
|
|
|
int frame_to = i + unchanged_at_top;
|
|
|
|
|
|
|
|
|
|
/* Frame relative line assigned. */
|
|
|
|
|
int frame_from = copy_from[i] + unchanged_at_top;
|
|
|
|
|
|
|
|
|
|
/* Window relative line assigned to. */
|
|
|
|
|
int window_to = frame_to - m->matrix_y;
|
|
|
|
|
|
|
|
|
|
/* Window relative line assigned. */
|
|
|
|
|
int window_from = frame_from - m->matrix_y;
|
|
|
|
|
|
|
|
|
|
/* Is assigned line inside window? */
|
|
|
|
|
int from_inside_window_p
|
|
|
|
|
= window_from >= 0 && window_from < m->matrix_h;
|
|
|
|
|
|
2000-02-25 13:20:36 +00:00
|
|
|
|
/* Is assigned to line inside window? */
|
|
|
|
|
int to_inside_window_p
|
|
|
|
|
= window_to >= 0 && window_to < m->matrix_h;
|
|
|
|
|
|
|
|
|
|
if (from_inside_window_p && to_inside_window_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
/* Enabled setting before assignment. */
|
|
|
|
|
int enabled_before_p;
|
|
|
|
|
|
|
|
|
|
/* Do the assignment. The enabled_p flag is saved
|
|
|
|
|
over the assignment because the old redisplay did
|
|
|
|
|
that. */
|
|
|
|
|
enabled_before_p = m->rows[window_to].enabled_p;
|
|
|
|
|
m->rows[window_to] = old_rows[window_from];
|
|
|
|
|
m->rows[window_to].enabled_p = enabled_before_p;
|
|
|
|
|
|
|
|
|
|
/* If frame line is empty, window line is empty, too. */
|
|
|
|
|
if (!retained_p[copy_from[i]])
|
|
|
|
|
m->rows[window_to].enabled_p = 0;
|
|
|
|
|
}
|
2000-02-25 13:20:36 +00:00
|
|
|
|
else if (to_inside_window_p)
|
|
|
|
|
{
|
|
|
|
|
/* A copy between windows. This is an infrequent
|
|
|
|
|
case not worth optimizing. */
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
struct window *root = XWINDOW (FRAME_ROOT_WINDOW (f));
|
|
|
|
|
struct window *w2;
|
|
|
|
|
struct glyph_matrix *m2;
|
|
|
|
|
int m2_from;
|
|
|
|
|
|
|
|
|
|
w2 = frame_row_to_window (root, frame_to);
|
|
|
|
|
m2 = w2->current_matrix;
|
|
|
|
|
m2_from = frame_from - m2->matrix_y;
|
|
|
|
|
copy_row_except_pointers (m->rows + window_to,
|
|
|
|
|
m2->rows + m2_from);
|
|
|
|
|
|
|
|
|
|
/* If frame line is empty, window line is empty, too. */
|
|
|
|
|
if (!retained_p[copy_from[i]])
|
|
|
|
|
m->rows[window_to].enabled_p = 0;
|
|
|
|
|
sync_p = 1;
|
|
|
|
|
}
|
|
|
|
|
else if (from_inside_window_p)
|
|
|
|
|
sync_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
2000-02-25 13:20:36 +00:00
|
|
|
|
|
|
|
|
|
/* If there was a copy between windows, make sure glyph
|
|
|
|
|
pointers are in sync with the frame matrix. */
|
|
|
|
|
if (sync_p)
|
|
|
|
|
sync_window_with_frame_matrix_rows (w);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Check that no pointers are lost. */
|
|
|
|
|
CHECK_MATRIX (m);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Next window on same level. */
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
|
|
|
|
|
/* Check that window and frame matrices agree about their
|
|
|
|
|
understanding where glyphs of the rows are to find. For each
|
|
|
|
|
window in the window tree rooted at W, check that rows in the
|
|
|
|
|
matrices of leaf window agree with their frame matrices about
|
|
|
|
|
glyph pointers. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
check_window_matrix_pointers (w)
|
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
check_window_matrix_pointers (XWINDOW (w->hchild));
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
check_window_matrix_pointers (XWINDOW (w->vchild));
|
|
|
|
|
else
|
1994-03-31 23:22:23 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
check_matrix_pointers (w->desired_matrix, f->desired_matrix);
|
|
|
|
|
check_matrix_pointers (w->current_matrix, f->current_matrix);
|
1994-03-31 23:22:23 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Check that window rows are slices of frame rows. WINDOW_MATRIX is
|
|
|
|
|
a window and FRAME_MATRIX is the corresponding frame matrix. For
|
|
|
|
|
each row in WINDOW_MATRIX check that it's a slice of the
|
|
|
|
|
corresponding frame row. If it isn't, abort. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
check_matrix_pointers (window_matrix, frame_matrix)
|
|
|
|
|
struct glyph_matrix *window_matrix, *frame_matrix;
|
|
|
|
|
{
|
|
|
|
|
/* Row number in WINDOW_MATRIX. */
|
|
|
|
|
int i = 0;
|
|
|
|
|
|
|
|
|
|
/* Row number corresponding to I in FRAME_MATRIX. */
|
|
|
|
|
int j = window_matrix->matrix_y;
|
|
|
|
|
|
|
|
|
|
/* For all rows check that the row in the window matrix is a
|
|
|
|
|
slice of the row in the frame matrix. If it isn't we didn't
|
|
|
|
|
mirror an operation on the frame matrix correctly. */
|
|
|
|
|
while (i < window_matrix->nrows)
|
|
|
|
|
{
|
|
|
|
|
if (!glyph_row_slice_p (window_matrix->rows + i,
|
|
|
|
|
frame_matrix->rows + j))
|
|
|
|
|
abort ();
|
|
|
|
|
++i, ++j;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* GLYPH_DEBUG != 0 */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
VPOS and HPOS translations
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
|
|
|
|
|
/* Translate vertical position VPOS which is relative to window W to a
|
|
|
|
|
vertical position relative to W's frame. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
window_to_frame_vpos (w, vpos)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int vpos;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
|
|
|
|
|
xassert (!FRAME_WINDOW_P (f));
|
|
|
|
|
xassert (vpos >= 0 && vpos <= w->desired_matrix->nrows);
|
|
|
|
|
vpos += XFASTINT (w->top);
|
|
|
|
|
xassert (vpos >= 0 && vpos <= FRAME_HEIGHT (f));
|
|
|
|
|
return vpos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Translate horizontal position HPOS which is relative to window W to
|
|
|
|
|
a vertical position relative to W's frame. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
window_to_frame_hpos (w, hpos)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int hpos;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
|
|
|
|
|
xassert (!FRAME_WINDOW_P (f));
|
|
|
|
|
hpos += XFASTINT (w->left);
|
|
|
|
|
return hpos;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#endif /* GLYPH_DEBUG */
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/**********************************************************************
|
|
|
|
|
Redrawing Frames
|
|
|
|
|
**********************************************************************/
|
|
|
|
|
|
|
|
|
|
DEFUN ("redraw-frame", Fredraw_frame, Sredraw_frame, 1, 1, 0,
|
|
|
|
|
"Clear frame FRAME and output again what is supposed to appear on it.")
|
|
|
|
|
(frame)
|
|
|
|
|
Lisp_Object frame;
|
|
|
|
|
{
|
|
|
|
|
struct frame *f;
|
|
|
|
|
|
|
|
|
|
CHECK_LIVE_FRAME (frame, 0);
|
|
|
|
|
f = XFRAME (frame);
|
|
|
|
|
|
|
|
|
|
/* Ignore redraw requests, if frame has no glyphs yet.
|
|
|
|
|
(Implementation note: It still has to be checked why we are
|
|
|
|
|
called so early here). */
|
|
|
|
|
if (!glyphs_initialized_initially_p)
|
|
|
|
|
return Qnil;
|
|
|
|
|
|
|
|
|
|
update_begin (f);
|
|
|
|
|
if (FRAME_MSDOS_P (f))
|
|
|
|
|
set_terminal_modes ();
|
|
|
|
|
clear_frame ();
|
|
|
|
|
clear_current_matrices (f);
|
|
|
|
|
update_end (f);
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
windows_or_buffers_changed++;
|
|
|
|
|
/* Mark all windows as inaccurate, so that every window will have
|
|
|
|
|
its redisplay done. */
|
|
|
|
|
mark_window_display_accurate (FRAME_ROOT_WINDOW (f), 0);
|
|
|
|
|
set_window_update_flags (XWINDOW (FRAME_ROOT_WINDOW (f)), 1);
|
|
|
|
|
f->garbaged = 0;
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Redraw frame F. This is nothing more than a call to the Lisp
|
|
|
|
|
function redraw-frame. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
redraw_frame (f)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object frame;
|
|
|
|
|
XSETFRAME (frame, f);
|
|
|
|
|
Fredraw_frame (frame);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("redraw-display", Fredraw_display, Sredraw_display, 0, 0, "",
|
|
|
|
|
"Clear and redisplay all visible frames.")
|
|
|
|
|
()
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame;
|
|
|
|
|
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
if (FRAME_VISIBLE_P (XFRAME (frame)))
|
|
|
|
|
Fredraw_frame (frame);
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* This is used when frame_garbaged is set. Call Fredraw_frame on all
|
|
|
|
|
visible frames marked as garbaged. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
redraw_garbaged_frames ()
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame;
|
|
|
|
|
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
if (FRAME_VISIBLE_P (XFRAME (frame))
|
|
|
|
|
&& FRAME_GARBAGED_P (XFRAME (frame)))
|
|
|
|
|
Fredraw_frame (frame);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Direct Operations
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Try to update display and current glyph matrix directly.
|
|
|
|
|
|
|
|
|
|
This function is called after a character G has been inserted into
|
|
|
|
|
current_buffer. It tries to update the current glyph matrix and
|
|
|
|
|
perform appropriate screen output to reflect the insertion. If it
|
|
|
|
|
succeeds, the global flag redisplay_performed_directly_p will be
|
|
|
|
|
set to 1, and thereby prevent the more costly general redisplay
|
|
|
|
|
from running (see redisplay_internal).
|
|
|
|
|
|
|
|
|
|
This function is not called for `hairy' character insertions.
|
|
|
|
|
In particular, it is not called when after or before change
|
|
|
|
|
functions exist, like they are used by font-lock. See keyboard.c
|
|
|
|
|
for details where this function is called. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
direct_output_for_insert (g)
|
|
|
|
|
int g;
|
|
|
|
|
{
|
1999-09-13 11:13:24 +00:00
|
|
|
|
register struct frame *f = SELECTED_FRAME ();
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *w = XWINDOW (selected_window);
|
|
|
|
|
struct it it, it2;
|
|
|
|
|
struct glyph_row *glyph_row;
|
|
|
|
|
struct glyph *glyphs, *glyph, *end;
|
|
|
|
|
int n;
|
|
|
|
|
/* Non-null means that Redisplay of W is based on window matrices. */
|
|
|
|
|
int window_redisplay_p = FRAME_WINDOW_P (f);
|
|
|
|
|
/* Non-null means we are in overwrite mode. */
|
|
|
|
|
int overwrite_p = !NILP (current_buffer->overwrite_mode);
|
|
|
|
|
int added_width;
|
|
|
|
|
struct text_pos pos;
|
|
|
|
|
int delta, delta_bytes;
|
|
|
|
|
|
|
|
|
|
/* Not done directly. */
|
|
|
|
|
redisplay_performed_directly_p = 0;
|
|
|
|
|
|
|
|
|
|
/* Quickly give up for some common cases. */
|
|
|
|
|
if (cursor_in_echo_area
|
|
|
|
|
/* Give up if fonts have changed. */
|
|
|
|
|
|| fonts_changed_p
|
|
|
|
|
/* Give up if face attributes have been changed. */
|
|
|
|
|
|| face_change_count
|
|
|
|
|
/* Give up if cursor position not really known. */
|
|
|
|
|
|| !display_completed
|
|
|
|
|
/* Give up if buffer appears in two places. */
|
|
|
|
|
|| buffer_shared > 1
|
|
|
|
|
/* Give up if w is mini-buffer and a message is being displayed there */
|
1999-08-21 19:29:02 +00:00
|
|
|
|
|| (MINI_WINDOW_P (w) && !NILP (echo_area_buffer[0]))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Give up for hscrolled mini-buffer because display of the prompt
|
|
|
|
|
is handled specially there (see display_line). */
|
|
|
|
|
|| (MINI_WINDOW_P (w) && XFASTINT (w->hscroll))
|
|
|
|
|
/* Give up if overwriting in the middle of a line. */
|
|
|
|
|
|| (overwrite_p
|
|
|
|
|
&& PT != ZV
|
|
|
|
|
&& FETCH_BYTE (PT) != '\n')
|
|
|
|
|
/* Give up for tabs and line ends. */
|
|
|
|
|
|| g == '\t'
|
|
|
|
|
|| g == '\n'
|
|
|
|
|
|| g == '\r'
|
|
|
|
|
/* Give up if unable to display the cursor in the window. */
|
|
|
|
|
|| w->cursor.vpos < 0
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| (glyph_row = MATRIX_ROW (w->current_matrix, w->cursor.vpos),
|
|
|
|
|
/* Can't do it in a continued line because continuation
|
|
|
|
|
lines would change. */
|
|
|
|
|
(glyph_row->continued_p
|
|
|
|
|
/* Can't use this method if the line overlaps others or is
|
|
|
|
|
overlapped by others because these other lines would
|
|
|
|
|
have to be redisplayed. */
|
|
|
|
|
|| glyph_row->overlapping_p
|
|
|
|
|
|| glyph_row->overlapped_p))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Can't do it for partial width windows on terminal frames
|
|
|
|
|
because we can't clear to eol in such a window. */
|
|
|
|
|
|| (!window_redisplay_p && !WINDOW_FULL_WIDTH_P (w)))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Set up a display iterator structure for W. Glyphs will be
|
|
|
|
|
produced in scratch_glyph_row. Current position is W's cursor
|
|
|
|
|
position. */
|
|
|
|
|
clear_glyph_row (&scratch_glyph_row);
|
|
|
|
|
SET_TEXT_POS (pos, PT, PT_BYTE);
|
2000-03-27 16:03:29 +00:00
|
|
|
|
DEC_TEXT_POS (pos, !NILP (current_buffer->enable_multibyte_characters));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
init_iterator (&it, w, CHARPOS (pos), BYTEPOS (pos), &scratch_glyph_row,
|
|
|
|
|
DEFAULT_FACE_ID);
|
|
|
|
|
|
|
|
|
|
glyph_row = MATRIX_ROW (w->current_matrix, w->cursor.vpos);
|
2000-07-12 15:10:30 +00:00
|
|
|
|
if (glyph_row->mouse_face_p)
|
|
|
|
|
return 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Give up if highlighting trailing whitespace and we have trailing
|
|
|
|
|
whitespace in glyph_row. We would have to remove the trailing
|
|
|
|
|
whitespace face in that case. */
|
1999-08-17 00:46:01 +00:00
|
|
|
|
if (!NILP (Vshow_trailing_whitespace)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
&& glyph_row->used[TEXT_AREA])
|
|
|
|
|
{
|
|
|
|
|
struct glyph *last;
|
|
|
|
|
|
|
|
|
|
last = glyph_row->glyphs[TEXT_AREA] + glyph_row->used[TEXT_AREA] - 1;
|
|
|
|
|
if (last->type == STRETCH_GLYPH
|
|
|
|
|
|| (last->type == CHAR_GLYPH
|
1999-12-27 05:06:03 +00:00
|
|
|
|
&& last->u.ch == ' '))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Give up if there are overlay strings at pos. This would fail
|
|
|
|
|
if the overlay string has newlines in it. */
|
|
|
|
|
if (STRINGP (it.string))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
it.hpos = w->cursor.hpos;
|
|
|
|
|
it.vpos = w->cursor.vpos;
|
|
|
|
|
it.current_x = w->cursor.x + it.first_visible_x;
|
|
|
|
|
it.current_y = w->cursor.y;
|
|
|
|
|
it.end_charpos = PT;
|
|
|
|
|
it.stop_charpos = min (PT, it.stop_charpos);
|
|
|
|
|
|
|
|
|
|
/* More than one display element may be returned for PT - 1 if
|
|
|
|
|
(i) it's a control character which is translated into `\003' or
|
|
|
|
|
`^C', or (ii) it has a display table entry, or (iii) it's a
|
|
|
|
|
combination of both. */
|
|
|
|
|
delta = delta_bytes = 0;
|
|
|
|
|
while (get_next_display_element (&it))
|
|
|
|
|
{
|
|
|
|
|
PRODUCE_GLYPHS (&it);
|
|
|
|
|
|
|
|
|
|
/* Give up if glyph doesn't fit completely on the line. */
|
|
|
|
|
if (it.current_x >= it.last_visible_x)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Give up if new glyph has different ascent or descent than
|
|
|
|
|
the original row, or if it is not a character glyph. */
|
|
|
|
|
if (glyph_row->ascent != it.ascent
|
|
|
|
|
|| glyph_row->height != it.ascent + it.descent
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| glyph_row->phys_ascent != it.phys_ascent
|
|
|
|
|
|| glyph_row->phys_height != it.phys_ascent + it.phys_descent
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| it.what != IT_CHARACTER)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
delta += 1;
|
|
|
|
|
delta_bytes += it.len;
|
2000-10-13 18:27:25 +00:00
|
|
|
|
set_iterator_to_next (&it, 1);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Give up if we hit the right edge of the window. We would have
|
|
|
|
|
to insert truncation or continuation glyphs. */
|
|
|
|
|
added_width = it.current_x - (w->cursor.x + it.first_visible_x);
|
|
|
|
|
if (glyph_row->pixel_width + added_width >= it.last_visible_x)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Give up if there is a \t following in the line. */
|
|
|
|
|
it2 = it;
|
|
|
|
|
it2.end_charpos = ZV;
|
|
|
|
|
it2.stop_charpos = min (it2.stop_charpos, ZV);
|
|
|
|
|
while (get_next_display_element (&it2)
|
|
|
|
|
&& !ITERATOR_AT_END_OF_LINE_P (&it2))
|
|
|
|
|
{
|
|
|
|
|
if (it2.c == '\t')
|
|
|
|
|
return 0;
|
2000-10-13 18:27:25 +00:00
|
|
|
|
set_iterator_to_next (&it2, 1);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Number of new glyphs produced. */
|
|
|
|
|
n = it.glyph_row->used[TEXT_AREA];
|
|
|
|
|
|
|
|
|
|
/* Start and end of glyphs in original row. */
|
|
|
|
|
glyphs = glyph_row->glyphs[TEXT_AREA] + w->cursor.hpos;
|
|
|
|
|
end = glyph_row->glyphs[1 + TEXT_AREA];
|
1994-03-31 23:22:23 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Make room for new glyphs, then insert them. */
|
|
|
|
|
xassert (end - glyphs - n >= 0);
|
1999-09-20 23:18:39 +00:00
|
|
|
|
safe_bcopy ((char *) glyphs, (char *) (glyphs + n),
|
|
|
|
|
(end - glyphs - n) * sizeof (*end));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
bcopy (it.glyph_row->glyphs[TEXT_AREA], glyphs, n * sizeof *glyphs);
|
|
|
|
|
glyph_row->used[TEXT_AREA] = min (glyph_row->used[TEXT_AREA] + n,
|
|
|
|
|
end - glyph_row->glyphs[TEXT_AREA]);
|
|
|
|
|
|
|
|
|
|
/* Compute new line width. */
|
|
|
|
|
glyph = glyph_row->glyphs[TEXT_AREA];
|
|
|
|
|
end = glyph + glyph_row->used[TEXT_AREA];
|
|
|
|
|
glyph_row->pixel_width = glyph_row->x;
|
|
|
|
|
while (glyph < end)
|
|
|
|
|
{
|
|
|
|
|
glyph_row->pixel_width += glyph->pixel_width;
|
|
|
|
|
++glyph;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Increment buffer positions for glyphs following the newly
|
|
|
|
|
inserted ones. */
|
|
|
|
|
for (glyph = glyphs + n; glyph < end; ++glyph)
|
1999-11-22 17:38:21 +00:00
|
|
|
|
if (glyph->charpos > 0 && BUFFERP (glyph->object))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
glyph->charpos += delta;
|
|
|
|
|
|
|
|
|
|
if (MATRIX_ROW_END_CHARPOS (glyph_row) > 0)
|
|
|
|
|
{
|
|
|
|
|
MATRIX_ROW_END_CHARPOS (glyph_row) += delta;
|
|
|
|
|
MATRIX_ROW_END_BYTEPOS (glyph_row) += delta_bytes;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Adjust positions in lines following the one we are in. */
|
2000-04-25 19:39:59 +00:00
|
|
|
|
increment_matrix_positions (w->current_matrix,
|
|
|
|
|
w->cursor.vpos + 1,
|
|
|
|
|
w->current_matrix->nrows,
|
|
|
|
|
delta, delta_bytes);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
glyph_row->contains_overlapping_glyphs_p
|
|
|
|
|
|= it.glyph_row->contains_overlapping_glyphs_p;
|
|
|
|
|
|
1999-11-25 10:14:15 +00:00
|
|
|
|
glyph_row->displays_text_p = 1;
|
|
|
|
|
w->window_end_vpos = make_number (max (w->cursor.vpos,
|
|
|
|
|
XFASTINT (w->window_end_vpos)));
|
|
|
|
|
|
1999-08-17 00:46:01 +00:00
|
|
|
|
if (!NILP (Vshow_trailing_whitespace))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
highlight_trailing_whitespace (it.f, glyph_row);
|
|
|
|
|
|
|
|
|
|
/* Write glyphs. If at end of row, we can simply call write_glyphs.
|
|
|
|
|
In the middle, we have to insert glyphs. Note that this is now
|
|
|
|
|
implemented for X frames. The implementation uses updated_window
|
|
|
|
|
and updated_row. */
|
|
|
|
|
updated_row = glyph_row;
|
|
|
|
|
update_begin (f);
|
|
|
|
|
if (rif)
|
|
|
|
|
{
|
|
|
|
|
rif->update_window_begin_hook (w);
|
|
|
|
|
|
|
|
|
|
if (glyphs == end - n)
|
|
|
|
|
rif->write_glyphs (glyphs, n);
|
|
|
|
|
else
|
|
|
|
|
rif->insert_glyphs (glyphs, n);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
if (glyphs == end - n)
|
|
|
|
|
write_glyphs (glyphs, n);
|
|
|
|
|
else
|
|
|
|
|
insert_glyphs (glyphs, n);
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
w->cursor.hpos += n;
|
|
|
|
|
w->cursor.x = it.current_x - it.first_visible_x;
|
|
|
|
|
xassert (w->cursor.hpos >= 0
|
|
|
|
|
&& w->cursor.hpos < w->desired_matrix->matrix_w);
|
|
|
|
|
|
|
|
|
|
/* How to set the cursor differs depending on whether we are
|
|
|
|
|
using a frame matrix or a window matrix. Note that when
|
|
|
|
|
a frame matrix is used, cursor_to expects frame coordinates,
|
|
|
|
|
and the X and Y parameters are not used. */
|
|
|
|
|
if (window_redisplay_p)
|
|
|
|
|
rif->cursor_to (w->cursor.vpos, w->cursor.hpos,
|
|
|
|
|
w->cursor.y, w->cursor.x);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int x, y;
|
|
|
|
|
x = (WINDOW_TO_FRAME_HPOS (w, w->cursor.hpos)
|
|
|
|
|
+ (INTEGERP (w->left_margin_width)
|
|
|
|
|
? XFASTINT (w->left_margin_width)
|
|
|
|
|
: 0));
|
|
|
|
|
y = WINDOW_TO_FRAME_VPOS (w, w->cursor.vpos);
|
|
|
|
|
cursor_to (y, x);
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (rif)
|
2000-07-12 15:10:30 +00:00
|
|
|
|
rif->update_window_end_hook (w, 1, 0);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
update_end (f);
|
|
|
|
|
updated_row = NULL;
|
|
|
|
|
fflush (stdout);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
TRACE ((stderr, "direct output for insert\n"));
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-08-23 00:11:43 +00:00
|
|
|
|
UNCHANGED_MODIFIED = MODIFF;
|
|
|
|
|
BEG_UNCHANGED = GPT - BEG;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
XSETFASTINT (w->last_point, PT);
|
|
|
|
|
w->last_cursor = w->cursor;
|
|
|
|
|
XSETFASTINT (w->last_modified, MODIFF);
|
|
|
|
|
XSETFASTINT (w->last_overlay_modified, OVERLAY_MODIFF);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
redisplay_performed_directly_p = 1;
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
1992-10-14 21:30:21 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Perform a direct display update for moving PT by N positions
|
|
|
|
|
left or right. N < 0 means a movement backwards. This function
|
|
|
|
|
is currently only called for N == 1 or N == -1. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
direct_output_forward_char (n)
|
|
|
|
|
int n;
|
|
|
|
|
{
|
1999-09-13 11:13:24 +00:00
|
|
|
|
struct frame *f = SELECTED_FRAME ();
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *w = XWINDOW (selected_window);
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
|
1999-12-15 00:09:32 +00:00
|
|
|
|
/* Give up if point moved out of or into a composition. */
|
2000-04-05 17:29:31 +00:00
|
|
|
|
if (check_point_in_composition (current_buffer, XINT (w->last_point),
|
1999-12-15 00:09:32 +00:00
|
|
|
|
current_buffer, PT))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Give up if face attributes have been changed. */
|
|
|
|
|
if (face_change_count)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Give up if current matrix is not up to date or we are
|
|
|
|
|
displaying a message. */
|
|
|
|
|
if (!display_completed || cursor_in_echo_area)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Give up if the buffer's direction is reversed. */
|
|
|
|
|
if (!NILP (XBUFFER (w->buffer)->direction_reversed))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
/* Can't use direct output if highlighting a region. */
|
|
|
|
|
if (!NILP (Vtransient_mark_mode) && !NILP (current_buffer->mark_active))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-08-17 00:46:01 +00:00
|
|
|
|
/* Can't use direct output if highlighting trailing whitespace. */
|
|
|
|
|
if (!NILP (Vshow_trailing_whitespace))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-09-16 12:52:13 +00:00
|
|
|
|
/* Give up if we are showing a message or just cleared the message
|
|
|
|
|
because we might need to resize the echo area window. */
|
|
|
|
|
if (!NILP (echo_area_buffer[0]) || !NILP (echo_area_buffer[1]))
|
|
|
|
|
return 0;
|
|
|
|
|
|
2000-07-11 19:57:54 +00:00
|
|
|
|
/* Give up if currently displaying a message instead of the
|
|
|
|
|
minibuffer contents. */
|
|
|
|
|
if (XWINDOW (minibuf_window) == w
|
|
|
|
|
&& EQ (minibuf_window, echo_area_window))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-09-16 12:52:13 +00:00
|
|
|
|
/* Give up if we don't know where the cursor is. */
|
|
|
|
|
if (w->cursor.vpos < 0)
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
row = MATRIX_ROW (w->current_matrix, w->cursor.vpos);
|
|
|
|
|
|
2000-08-29 15:04:48 +00:00
|
|
|
|
/* Give up if PT is outside of the last known cursor row. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (PT <= MATRIX_ROW_START_BYTEPOS (row)
|
|
|
|
|
|| PT >= MATRIX_ROW_END_BYTEPOS (row))
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
set_cursor_from_row (w, row, w->current_matrix, 0, 0, 0, 0);
|
2000-08-29 15:04:48 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
w->last_cursor = w->cursor;
|
|
|
|
|
XSETFASTINT (w->last_point, PT);
|
|
|
|
|
|
|
|
|
|
xassert (w->cursor.hpos >= 0
|
|
|
|
|
&& w->cursor.hpos < w->desired_matrix->matrix_w);
|
|
|
|
|
|
|
|
|
|
if (FRAME_WINDOW_P (f))
|
|
|
|
|
rif->cursor_to (w->cursor.vpos, w->cursor.hpos,
|
|
|
|
|
w->cursor.y, w->cursor.x);
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int x, y;
|
|
|
|
|
x = (WINDOW_TO_FRAME_HPOS (w, w->cursor.hpos)
|
|
|
|
|
+ (INTEGERP (w->left_margin_width)
|
|
|
|
|
? XFASTINT (w->left_margin_width)
|
|
|
|
|
: 0));
|
|
|
|
|
y = WINDOW_TO_FRAME_VPOS (w, w->cursor.vpos);
|
|
|
|
|
cursor_to (y, x);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
redisplay_performed_directly_p = 1;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return 1;
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/***********************************************************************
|
|
|
|
|
Frame Update
|
|
|
|
|
***********************************************************************/
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Update frame F based on the data in desired matrices.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
If FORCE_P is non-zero, don't let redisplay be stopped by detecting
|
|
|
|
|
pending input. If INHIBIT_HAIRY_ID_P is non-zero, don't try
|
|
|
|
|
scrolling.
|
|
|
|
|
|
|
|
|
|
Value is non-zero if redisplay was stopped due to pending input. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int
|
|
|
|
|
update_frame (f, force_p, inhibit_hairy_id_p)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
int force_p;
|
|
|
|
|
int inhibit_hairy_id_p;
|
|
|
|
|
{
|
|
|
|
|
/* 1 means display has been paused because of pending input. */
|
|
|
|
|
int paused_p;
|
|
|
|
|
struct window *root_window = XWINDOW (f->root_window);
|
|
|
|
|
|
|
|
|
|
if (FRAME_WINDOW_P (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* We are working on window matrix basis. All windows whose
|
|
|
|
|
flag must_be_updated_p is set have to be updated. */
|
|
|
|
|
|
|
|
|
|
/* Record that we are not working on frame matrices. */
|
|
|
|
|
set_frame_matrix_frame (NULL);
|
|
|
|
|
|
|
|
|
|
/* Update all windows in the window tree of F, maybe stopping
|
|
|
|
|
when pending input is detected. */
|
|
|
|
|
update_begin (f);
|
|
|
|
|
|
|
|
|
|
/* Update the menu bar on X frames that don't have toolkit
|
|
|
|
|
support. */
|
|
|
|
|
if (WINDOWP (f->menu_bar_window))
|
|
|
|
|
update_window (XWINDOW (f->menu_bar_window), 1);
|
|
|
|
|
|
|
|
|
|
/* Update the tool-bar window, if present. */
|
1999-09-05 15:49:07 +00:00
|
|
|
|
if (WINDOWP (f->tool_bar_window))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Lisp_Object tem;
|
1999-09-05 15:49:07 +00:00
|
|
|
|
struct window *w = XWINDOW (f->tool_bar_window);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Update tool-bar window. */
|
|
|
|
|
if (w->must_be_updated_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
update_window (w, 1);
|
|
|
|
|
w->must_be_updated_p = 0;
|
|
|
|
|
|
|
|
|
|
/* Swap tool-bar strings. We swap because we want to
|
|
|
|
|
reuse strings. */
|
1999-09-05 15:49:07 +00:00
|
|
|
|
tem = f->current_tool_bar_string;
|
|
|
|
|
f->current_tool_bar_string = f->desired_tool_bar_string;
|
|
|
|
|
f->desired_tool_bar_string = tem;
|
|
|
|
|
f->n_current_tool_bar_items = f->n_desired_tool_bar_items;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Swap tool-bar items. We swap because we want to
|
|
|
|
|
reuse vectors. */
|
1999-09-05 15:49:07 +00:00
|
|
|
|
tem = f->current_tool_bar_items;
|
|
|
|
|
f->current_tool_bar_items = f->desired_tool_bar_items;
|
|
|
|
|
f->desired_tool_bar_items = tem;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Update windows. */
|
|
|
|
|
paused_p = update_window_tree (root_window, force_p);
|
|
|
|
|
update_end (f);
|
2000-08-30 15:00:17 +00:00
|
|
|
|
|
|
|
|
|
#if 0 /* This flush is a performance bottleneck under X,
|
|
|
|
|
and it doesn't seem to be necessary anyway. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
rif->flush_display (f);
|
|
|
|
|
#endif
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
/* We are working on frame matrix basis. Set the frame on whose
|
|
|
|
|
frame matrix we operate. */
|
|
|
|
|
set_frame_matrix_frame (f);
|
|
|
|
|
|
2000-08-30 15:00:17 +00:00
|
|
|
|
/* Build F's desired matrix from window matrices. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
build_frame_matrix (f);
|
|
|
|
|
|
2000-08-30 15:00:17 +00:00
|
|
|
|
/* Update the display */
|
|
|
|
|
update_begin (f);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
paused_p = update_frame_1 (f, force_p, inhibit_hairy_id_p);
|
2000-08-30 15:00:17 +00:00
|
|
|
|
update_end (f);
|
|
|
|
|
|
|
|
|
|
if (termscript)
|
|
|
|
|
fflush (termscript);
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Check window matrices for lost pointers. */
|
|
|
|
|
IF_DEBUG (check_window_matrix_pointers (root_window));
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Reset flags indicating that a window should be updated. */
|
|
|
|
|
set_window_update_flags (root_window, 0);
|
2000-08-30 15:00:17 +00:00
|
|
|
|
|
|
|
|
|
display_completed = !paused_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
return paused_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/************************************************************************
|
|
|
|
|
Window-based updates
|
|
|
|
|
************************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Perform updates in window tree rooted at W. FORCE_P non-zero means
|
|
|
|
|
don't stop updating when input is pending. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
update_window_tree (w, force_p)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int force_p;
|
|
|
|
|
{
|
|
|
|
|
int paused_p = 0;
|
|
|
|
|
|
|
|
|
|
while (w && !paused_p)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
paused_p |= update_window_tree (XWINDOW (w->hchild), force_p);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
paused_p |= update_window_tree (XWINDOW (w->vchild), force_p);
|
|
|
|
|
else if (w->must_be_updated_p)
|
|
|
|
|
paused_p |= update_window (w, force_p);
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return paused_p;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Update window W if its flag must_be_updated_p is non-zero. If
|
|
|
|
|
FORCE_P is non-zero, don't stop updating if input is pending. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
update_single_window (w, force_p)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int force_p;
|
|
|
|
|
{
|
|
|
|
|
if (w->must_be_updated_p)
|
|
|
|
|
{
|
|
|
|
|
struct frame *f = XFRAME (WINDOW_FRAME (w));
|
|
|
|
|
|
|
|
|
|
/* Record that this is not a frame-based redisplay. */
|
|
|
|
|
set_frame_matrix_frame (NULL);
|
|
|
|
|
|
|
|
|
|
/* Update W. */
|
|
|
|
|
update_begin (f);
|
|
|
|
|
update_window (w, force_p);
|
|
|
|
|
update_end (f);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Reset flag in W. */
|
|
|
|
|
w->must_be_updated_p = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* Redraw lines from the current matrix of window W that are
|
|
|
|
|
overlapped by other rows. YB is bottom-most y-position in W. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
redraw_overlapped_rows (w, yb)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int yb;
|
|
|
|
|
{
|
2000-07-12 16:00:36 +00:00
|
|
|
|
int i;
|
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* If rows overlapping others have been changed, the rows being
|
|
|
|
|
overlapped have to be redrawn. This won't draw lines that have
|
|
|
|
|
already been drawn in update_window_line because overlapped_p in
|
|
|
|
|
desired rows is 0, so after row assignment overlapped_p in
|
|
|
|
|
current rows is 0. */
|
|
|
|
|
for (i = 0; i < w->current_matrix->nrows; ++i)
|
|
|
|
|
{
|
2000-07-12 15:10:30 +00:00
|
|
|
|
struct glyph_row *row = w->current_matrix->rows + i;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|
|
|
|
|
if (!row->enabled_p)
|
|
|
|
|
break;
|
|
|
|
|
else if (row->mode_line_p)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
if (row->overlapped_p)
|
|
|
|
|
{
|
|
|
|
|
enum glyph_row_area area;
|
|
|
|
|
|
|
|
|
|
for (area = LEFT_MARGIN_AREA; area < LAST_AREA; ++area)
|
|
|
|
|
{
|
|
|
|
|
updated_row = row;
|
|
|
|
|
updated_area = area;
|
|
|
|
|
rif->cursor_to (i, 0, row->y, area == TEXT_AREA ? row->x : 0);
|
|
|
|
|
if (row->used[area])
|
|
|
|
|
rif->write_glyphs (row->glyphs[area], row->used[area]);
|
|
|
|
|
rif->clear_end_of_line (-1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
row->overlapped_p = 0;
|
|
|
|
|
}
|
|
|
|
|
|
2000-07-12 15:10:30 +00:00
|
|
|
|
if (MATRIX_ROW_BOTTOM_Y (row) >= yb)
|
1999-08-06 13:59:32 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Redraw lines from the current matrix of window W that overlap
|
|
|
|
|
others. YB is bottom-most y-position in W. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
redraw_overlapping_rows (w, yb)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int yb;
|
|
|
|
|
{
|
|
|
|
|
int i, bottom_y;
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
|
|
|
|
|
for (i = 0; i < w->current_matrix->nrows; ++i)
|
|
|
|
|
{
|
|
|
|
|
row = w->current_matrix->rows + i;
|
|
|
|
|
|
|
|
|
|
if (!row->enabled_p)
|
|
|
|
|
break;
|
|
|
|
|
else if (row->mode_line_p)
|
|
|
|
|
continue;
|
|
|
|
|
|
|
|
|
|
bottom_y = MATRIX_ROW_BOTTOM_Y (row);
|
|
|
|
|
|
|
|
|
|
if (row->overlapping_p && i > 0 && bottom_y < yb)
|
|
|
|
|
{
|
|
|
|
|
if (row->used[LEFT_MARGIN_AREA])
|
|
|
|
|
rif->fix_overlapping_area (w, row, LEFT_MARGIN_AREA);
|
|
|
|
|
|
|
|
|
|
if (row->used[TEXT_AREA])
|
|
|
|
|
rif->fix_overlapping_area (w, row, TEXT_AREA);
|
|
|
|
|
|
|
|
|
|
if (row->used[RIGHT_MARGIN_AREA])
|
|
|
|
|
rif->fix_overlapping_area (w, row, RIGHT_MARGIN_AREA);
|
|
|
|
|
|
|
|
|
|
/* Record in neighbor rows that ROW overwrites part of their
|
|
|
|
|
display. */
|
|
|
|
|
if (row->phys_ascent > row->ascent && i > 0)
|
|
|
|
|
MATRIX_ROW (w->current_matrix, i - 1)->overlapped_p = 1;
|
|
|
|
|
if ((row->phys_height - row->phys_ascent
|
|
|
|
|
> row->height - row->ascent)
|
|
|
|
|
&& bottom_y < yb)
|
|
|
|
|
MATRIX_ROW (w->current_matrix, i + 1)->overlapped_p = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (bottom_y >= yb)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Update display of window W. FORCE_P non-zero means that we should
|
|
|
|
|
not stop when detecting pending input. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
update_window (w, force_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
struct window *w;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int force_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *desired_matrix = w->desired_matrix;
|
|
|
|
|
int paused_p;
|
|
|
|
|
int preempt_count = baud_rate / 2400 + 1;
|
|
|
|
|
extern int input_pending;
|
1999-08-21 19:29:02 +00:00
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
struct frame *f = XFRAME (WINDOW_FRAME (w));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
extern struct frame *updating_frame;
|
1999-08-21 19:29:02 +00:00
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Check that W's frame doesn't have glyph matrices. */
|
|
|
|
|
xassert (FRAME_WINDOW_P (f));
|
|
|
|
|
xassert (updating_frame != NULL);
|
|
|
|
|
|
|
|
|
|
/* Check pending input the first time so that we can quickly return. */
|
|
|
|
|
if (redisplay_dont_pause)
|
|
|
|
|
force_p = 1;
|
|
|
|
|
else
|
|
|
|
|
detect_input_pending ();
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If forced to complete the update, or if no input is pending, do
|
|
|
|
|
the update. */
|
|
|
|
|
if (force_p || !input_pending)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *row, *end;
|
|
|
|
|
struct glyph_row *mode_line_row;
|
1999-09-05 16:39:19 +00:00
|
|
|
|
struct glyph_row *header_line_row = NULL;
|
2000-07-12 15:10:30 +00:00
|
|
|
|
int yb, changed_p = 0, mouse_face_overwritten_p = 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
rif->update_window_begin_hook (w);
|
|
|
|
|
yb = window_text_bottom_y (w);
|
|
|
|
|
|
|
|
|
|
/* If window has a top line, update it before everything else.
|
|
|
|
|
Adjust y-positions of other rows by the top line height. */
|
|
|
|
|
row = desired_matrix->rows;
|
|
|
|
|
end = row + desired_matrix->nrows - 1;
|
|
|
|
|
if (row->mode_line_p)
|
1999-09-05 16:39:19 +00:00
|
|
|
|
header_line_row = row++;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Update the mode line, if necessary. */
|
|
|
|
|
mode_line_row = MATRIX_MODE_LINE_ROW (desired_matrix);
|
|
|
|
|
if (mode_line_row->mode_line_p && mode_line_row->enabled_p)
|
|
|
|
|
{
|
|
|
|
|
mode_line_row->y = yb;
|
|
|
|
|
update_window_line (w, MATRIX_ROW_VPOS (mode_line_row,
|
2000-07-12 15:10:30 +00:00
|
|
|
|
desired_matrix),
|
|
|
|
|
&mouse_face_overwritten_p);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Find first enabled row. Optimizations in redisplay_internal
|
|
|
|
|
may lead to an update with only one row enabled. There may
|
|
|
|
|
be also completely empty matrices. */
|
|
|
|
|
while (row < end && !row->enabled_p)
|
|
|
|
|
++row;
|
|
|
|
|
|
2000-09-21 18:41:36 +00:00
|
|
|
|
/* Try reusing part of the display by copying. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (row < end && !desired_matrix->no_scrolling_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-09-05 16:39:19 +00:00
|
|
|
|
int rc = scrolling_window (w, header_line_row != NULL);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (rc < 0)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* All rows were found to be equal. */
|
|
|
|
|
paused_p = 0;
|
|
|
|
|
goto set_cursor;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
else if (rc > 0)
|
|
|
|
|
force_p = 1;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Update the top mode line after scrolling because a new top
|
|
|
|
|
line would otherwise overwrite lines at the top of the window
|
|
|
|
|
that can be scrolled. */
|
1999-09-05 16:39:19 +00:00
|
|
|
|
if (header_line_row && header_line_row->enabled_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-09-05 16:39:19 +00:00
|
|
|
|
header_line_row->y = 0;
|
2000-07-12 15:10:30 +00:00
|
|
|
|
update_window_line (w, 0, &mouse_face_overwritten_p);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Update the rest of the lines. */
|
|
|
|
|
for (; row < end && (force_p || !input_pending); ++row)
|
2000-09-16 13:39:36 +00:00
|
|
|
|
if (row->enabled_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
int vpos = MATRIX_ROW_VPOS (row, desired_matrix);
|
|
|
|
|
int i;
|
|
|
|
|
|
|
|
|
|
/* We'll Have to play a little bit with when to
|
|
|
|
|
detect_input_pending. If it's done too often,
|
|
|
|
|
scrolling large windows with repeated scroll-up
|
|
|
|
|
commands will too quickly pause redisplay. */
|
|
|
|
|
if (!force_p && vpos % preempt_count == 0)
|
|
|
|
|
detect_input_pending ();
|
|
|
|
|
|
2000-07-12 15:10:30 +00:00
|
|
|
|
changed_p |= update_window_line (w, vpos,
|
|
|
|
|
&mouse_face_overwritten_p);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Mark all rows below the last visible one in the current
|
|
|
|
|
matrix as invalid. This is necessary because of
|
|
|
|
|
variable line heights. Consider the case of three
|
|
|
|
|
successive redisplays, where the first displays 5
|
|
|
|
|
lines, the second 3 lines, and the third 5 lines again.
|
|
|
|
|
If the second redisplay wouldn't mark rows in the
|
|
|
|
|
current matrix invalid, the third redisplay might be
|
|
|
|
|
tempted to optimize redisplay based on lines displayed
|
|
|
|
|
in the first redisplay. */
|
|
|
|
|
if (MATRIX_ROW_BOTTOM_Y (row) >= yb)
|
|
|
|
|
for (i = vpos + 1; i < w->current_matrix->nrows - 1; ++i)
|
|
|
|
|
MATRIX_ROW (w->current_matrix, i)->enabled_p = 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Was display preempted? */
|
|
|
|
|
paused_p = row < end;
|
|
|
|
|
|
|
|
|
|
set_cursor:
|
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* Fix the appearance of overlapping(overlapped rows. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (!paused_p && !w->pseudo_window_p)
|
|
|
|
|
{
|
2000-07-12 15:10:30 +00:00
|
|
|
|
if (changed_p && rif->fix_overlapping_area)
|
|
|
|
|
{
|
|
|
|
|
redraw_overlapped_rows (w, yb);
|
|
|
|
|
redraw_overlapping_rows (w, yb);
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Make cursor visible at cursor position of W. */
|
|
|
|
|
set_window_cursor_after_update (w);
|
|
|
|
|
|
2000-07-12 15:10:30 +00:00
|
|
|
|
#if 0 /* Check that current matrix invariants are satisfied. This is
|
|
|
|
|
for debugging only. See the comment of check_matrix_invariants. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
IF_DEBUG (check_matrix_invariants (w));
|
|
|
|
|
#endif
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
#if GLYPH_DEBUG
|
|
|
|
|
/* Remember the redisplay method used to display the matrix. */
|
|
|
|
|
strcpy (w->current_matrix->method, w->desired_matrix->method);
|
|
|
|
|
#endif
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* End of update of window W. */
|
2000-07-12 15:10:30 +00:00
|
|
|
|
rif->update_window_end_hook (w, 1, mouse_face_overwritten_p);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
else
|
|
|
|
|
paused_p = 1;
|
|
|
|
|
|
|
|
|
|
clear_glyph_matrix (desired_matrix);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
return paused_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1994-04-01 02:10:14 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Update the display of area AREA in window W, row number VPOS.
|
|
|
|
|
AREA can be either LEFT_MARGIN_AREA or RIGHT_MARGIN_AREA. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
update_marginal_area (w, area, vpos)
|
1994-04-01 02:10:14 +00:00
|
|
|
|
struct window *w;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int area, vpos;
|
1994-04-01 02:10:14 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *desired_row = MATRIX_ROW (w->desired_matrix, vpos);
|
|
|
|
|
|
|
|
|
|
/* Let functions in xterm.c know what area subsequent X positions
|
|
|
|
|
will be relative to. */
|
|
|
|
|
updated_area = area;
|
|
|
|
|
|
|
|
|
|
/* Set cursor to start of glyphs, write them, and clear to the end
|
|
|
|
|
of the area. I don't think that something more sophisticated is
|
|
|
|
|
necessary here, since marginal areas will not be the default. */
|
|
|
|
|
rif->cursor_to (vpos, 0, desired_row->y, 0);
|
|
|
|
|
if (desired_row->used[area])
|
|
|
|
|
rif->write_glyphs (desired_row->glyphs[area], desired_row->used[area]);
|
|
|
|
|
rif->clear_end_of_line (-1);
|
1994-04-01 02:10:14 +00:00
|
|
|
|
}
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1994-04-05 07:39:38 +00:00
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* Update the display of the text area of row VPOS in window W.
|
|
|
|
|
Value is non-zero if display has changed. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
static int
|
1999-07-21 21:43:52 +00:00
|
|
|
|
update_text_area (w, vpos)
|
1994-04-03 01:36:57 +00:00
|
|
|
|
struct window *w;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int vpos;
|
1994-04-03 01:36:57 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_row *current_row = MATRIX_ROW (w->current_matrix, vpos);
|
|
|
|
|
struct glyph_row *desired_row = MATRIX_ROW (w->desired_matrix, vpos);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
int changed_p = 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Let functions in xterm.c know what area subsequent X positions
|
|
|
|
|
will be relative to. */
|
|
|
|
|
updated_area = TEXT_AREA;
|
|
|
|
|
|
|
|
|
|
/* If rows are at different X or Y, or rows have different height,
|
|
|
|
|
or the current row is marked invalid, write the entire line. */
|
|
|
|
|
if (!current_row->enabled_p
|
|
|
|
|
|| desired_row->y != current_row->y
|
|
|
|
|
|| desired_row->ascent != current_row->ascent
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| desired_row->phys_ascent != current_row->phys_ascent
|
|
|
|
|
|| desired_row->phys_height != current_row->phys_height
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| desired_row->visible_height != current_row->visible_height
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|| current_row->overlapped_p
|
2000-07-18 19:43:04 +00:00
|
|
|
|
|| current_row->mouse_face_p
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|| current_row->x != desired_row->x)
|
|
|
|
|
{
|
|
|
|
|
rif->cursor_to (vpos, 0, desired_row->y, desired_row->x);
|
|
|
|
|
|
|
|
|
|
if (desired_row->used[TEXT_AREA])
|
|
|
|
|
rif->write_glyphs (desired_row->glyphs[TEXT_AREA],
|
|
|
|
|
desired_row->used[TEXT_AREA]);
|
|
|
|
|
|
|
|
|
|
/* Clear to end of window. */
|
|
|
|
|
rif->clear_end_of_line (-1);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
int stop, i, x;
|
|
|
|
|
struct glyph *current_glyph = current_row->glyphs[TEXT_AREA];
|
|
|
|
|
struct glyph *desired_glyph = desired_row->glyphs[TEXT_AREA];
|
|
|
|
|
|
|
|
|
|
/* If the desired row extends its face to the text area end,
|
|
|
|
|
make sure we write at least one glyph, so that the face
|
|
|
|
|
extension actually takes place. */
|
|
|
|
|
int desired_stop_pos = (desired_row->used[TEXT_AREA]
|
|
|
|
|
- (MATRIX_ROW_EXTENDS_FACE_P (desired_row)
|
|
|
|
|
? 1 : 0));
|
|
|
|
|
|
|
|
|
|
stop = min (current_row->used[TEXT_AREA], desired_stop_pos);
|
|
|
|
|
i = 0;
|
|
|
|
|
x = desired_row->x;
|
|
|
|
|
|
|
|
|
|
while (i < stop)
|
1994-04-05 07:39:38 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Skip over glyphs that both rows have in common. These
|
|
|
|
|
don't have to be written. */
|
|
|
|
|
while (i < stop
|
|
|
|
|
&& GLYPH_EQUAL_P (desired_glyph, current_glyph))
|
1994-04-05 07:39:38 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
x += desired_glyph->pixel_width;
|
|
|
|
|
++desired_glyph, ++current_glyph, ++i;
|
1994-04-05 07:39:38 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Consider the case that the current row contains "xxx ppp
|
|
|
|
|
ggg" in italic Courier font, and the desired row is "xxx
|
|
|
|
|
ggg". The character `p' has lbearing, `g' has not. The
|
|
|
|
|
loop above will stop in front of the first `p' in the
|
|
|
|
|
current row. If we would start writing glyphs there, we
|
|
|
|
|
wouldn't erase the lbearing of the `p'. The rest of the
|
|
|
|
|
lbearing problem is then taken care of by x_draw_glyphs. */
|
|
|
|
|
if (current_row->contains_overlapping_glyphs_p
|
|
|
|
|
&& i > 0
|
|
|
|
|
&& i < current_row->used[TEXT_AREA]
|
|
|
|
|
&& current_row->used[TEXT_AREA] != desired_row->used[TEXT_AREA])
|
1994-04-05 07:39:38 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int left, right;
|
|
|
|
|
rif->get_glyph_overhangs (current_glyph, XFRAME (w->frame),
|
|
|
|
|
&left, &right);
|
|
|
|
|
while (left > 0 && i > 0)
|
|
|
|
|
{
|
|
|
|
|
--i, --desired_glyph, --current_glyph;
|
|
|
|
|
x -= desired_glyph->pixel_width;
|
|
|
|
|
left -= desired_glyph->pixel_width;
|
|
|
|
|
}
|
1994-04-05 07:39:38 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Try to avoid writing the entire rest of the desired row
|
|
|
|
|
by looking for a resync point. This mainly prevents
|
|
|
|
|
mode line flickering in the case the mode line is in
|
|
|
|
|
fixed-pitch font, which it usually will be. */
|
|
|
|
|
if (i < desired_row->used[TEXT_AREA])
|
|
|
|
|
{
|
|
|
|
|
int start_x = x, start_hpos = i;
|
|
|
|
|
struct glyph *start = desired_glyph;
|
|
|
|
|
int current_x = x;
|
|
|
|
|
|
|
|
|
|
/* Find the next glyph that's equal again. */
|
|
|
|
|
while (i < stop
|
|
|
|
|
&& !GLYPH_EQUAL_P (desired_glyph, current_glyph)
|
|
|
|
|
&& x == current_x)
|
|
|
|
|
{
|
|
|
|
|
x += desired_glyph->pixel_width;
|
|
|
|
|
current_x += current_glyph->pixel_width;
|
|
|
|
|
++desired_glyph, ++current_glyph, ++i;
|
|
|
|
|
}
|
1994-04-03 01:36:57 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (i == start_hpos || x != current_x)
|
|
|
|
|
{
|
|
|
|
|
i = start_hpos;
|
|
|
|
|
x = start_x;
|
|
|
|
|
desired_glyph = start;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
rif->cursor_to (vpos, start_hpos, desired_row->y, start_x);
|
|
|
|
|
rif->write_glyphs (start, i - start_hpos);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Write the rest. */
|
|
|
|
|
if (i < desired_row->used[TEXT_AREA])
|
|
|
|
|
{
|
|
|
|
|
rif->cursor_to (vpos, i, desired_row->y, x);
|
|
|
|
|
rif->write_glyphs (desired_glyph, desired_row->used[TEXT_AREA] - i);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Maybe clear to end of line. */
|
|
|
|
|
if (MATRIX_ROW_EXTENDS_FACE_P (desired_row))
|
|
|
|
|
{
|
|
|
|
|
/* If new row extends to the end of the text area, nothing
|
|
|
|
|
has to be cleared, if and only if we did a write_glyphs
|
|
|
|
|
above. This is made sure by setting desired_stop_pos
|
|
|
|
|
appropriately above. */
|
|
|
|
|
xassert (i < desired_row->used[TEXT_AREA]);
|
|
|
|
|
}
|
|
|
|
|
else if (MATRIX_ROW_EXTENDS_FACE_P (current_row))
|
|
|
|
|
{
|
|
|
|
|
/* If old row extends to the end of the text area, clear. */
|
|
|
|
|
if (i >= desired_row->used[TEXT_AREA])
|
|
|
|
|
rif->cursor_to (vpos, i, desired_row->y,
|
|
|
|
|
desired_row->x + desired_row->pixel_width);
|
|
|
|
|
rif->clear_end_of_line (-1);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
else if (desired_row->pixel_width < current_row->pixel_width)
|
1994-04-03 02:04:38 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Otherwise clear to the end of the old row. Everything
|
|
|
|
|
after that position should be clear already. */
|
|
|
|
|
int x;
|
|
|
|
|
|
|
|
|
|
if (i >= desired_row->used[TEXT_AREA])
|
|
|
|
|
rif->cursor_to (vpos, i, desired_row->y,
|
|
|
|
|
desired_row->x + desired_row->pixel_width);
|
|
|
|
|
|
|
|
|
|
/* If cursor is displayed at the end of the line, make sure
|
|
|
|
|
it's cleared. Nowadays we don't have a phys_cursor_glyph
|
|
|
|
|
with which to erase the cursor (because this method
|
|
|
|
|
doesn't work with lbearing/rbearing), so we must do it
|
|
|
|
|
this way. */
|
|
|
|
|
if (vpos == w->phys_cursor.vpos
|
|
|
|
|
&& w->phys_cursor.hpos >= desired_row->used[TEXT_AREA])
|
|
|
|
|
{
|
|
|
|
|
w->phys_cursor_on_p = 0;
|
|
|
|
|
x = -1;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
x = current_row->x + current_row->pixel_width;
|
|
|
|
|
rif->clear_end_of_line (x);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
changed_p = 1;
|
1994-04-03 02:04:38 +00:00
|
|
|
|
}
|
1994-04-03 01:36:57 +00:00
|
|
|
|
}
|
1999-08-06 13:59:32 +00:00
|
|
|
|
|
|
|
|
|
return changed_p;
|
1994-04-03 01:36:57 +00:00
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
/* Update row VPOS in window W. Value is non-zero if display has been
|
|
|
|
|
changed. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-08-06 13:59:32 +00:00
|
|
|
|
static int
|
2000-07-12 15:10:30 +00:00
|
|
|
|
update_window_line (w, vpos, mouse_face_overwritten_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *w;
|
2000-07-12 15:10:30 +00:00
|
|
|
|
int vpos, *mouse_face_overwritten_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
struct glyph_row *current_row = MATRIX_ROW (w->current_matrix, vpos);
|
|
|
|
|
struct glyph_row *desired_row = MATRIX_ROW (w->desired_matrix, vpos);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
int changed_p = 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Set the row being updated. This is important to let xterm.c
|
|
|
|
|
know what line height values are in effect. */
|
|
|
|
|
updated_row = desired_row;
|
|
|
|
|
|
2000-09-18 11:39:53 +00:00
|
|
|
|
/* A row can be completely invisible in case a desired matrix was
|
|
|
|
|
built with a vscroll and then make_cursor_line_fully_visible shifts
|
|
|
|
|
the matrix. Make sure to make such rows current anyway, since
|
|
|
|
|
we need the correct y-position, for example, in the current matrix. */
|
2000-09-19 15:18:07 +00:00
|
|
|
|
if (desired_row->mode_line_p
|
|
|
|
|
|| desired_row->visible_height > 0)
|
1999-08-06 13:59:32 +00:00
|
|
|
|
{
|
2000-09-18 11:39:53 +00:00
|
|
|
|
xassert (desired_row->enabled_p);
|
|
|
|
|
|
|
|
|
|
/* Update display of the left margin area, if there is one. */
|
|
|
|
|
if (!desired_row->full_width_p
|
|
|
|
|
&& !NILP (w->left_margin_width))
|
|
|
|
|
{
|
|
|
|
|
changed_p = 1;
|
|
|
|
|
update_marginal_area (w, LEFT_MARGIN_AREA, vpos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Update the display of the text area. */
|
|
|
|
|
if (update_text_area (w, vpos))
|
|
|
|
|
{
|
|
|
|
|
changed_p = 1;
|
|
|
|
|
if (current_row->mouse_face_p)
|
|
|
|
|
*mouse_face_overwritten_p = 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Update display of the right margin area, if there is one. */
|
|
|
|
|
if (!desired_row->full_width_p
|
|
|
|
|
&& !NILP (w->right_margin_width))
|
|
|
|
|
{
|
|
|
|
|
changed_p = 1;
|
|
|
|
|
update_marginal_area (w, RIGHT_MARGIN_AREA, vpos);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Draw truncation marks etc. */
|
|
|
|
|
if (!current_row->enabled_p
|
|
|
|
|
|| desired_row->y != current_row->y
|
|
|
|
|
|| desired_row->visible_height != current_row->visible_height
|
|
|
|
|
|| desired_row->overlay_arrow_p != current_row->overlay_arrow_p
|
|
|
|
|
|| desired_row->truncated_on_left_p != current_row->truncated_on_left_p
|
|
|
|
|
|| desired_row->truncated_on_right_p != current_row->truncated_on_right_p
|
|
|
|
|
|| desired_row->continued_p != current_row->continued_p
|
|
|
|
|
|| desired_row->mode_line_p != current_row->mode_line_p
|
|
|
|
|
|| (desired_row->indicate_empty_line_p
|
|
|
|
|
!= current_row->indicate_empty_line_p)
|
|
|
|
|
|| (MATRIX_ROW_CONTINUATION_LINE_P (desired_row)
|
|
|
|
|
!= MATRIX_ROW_CONTINUATION_LINE_P (current_row)))
|
|
|
|
|
rif->after_update_window_line_hook (desired_row);
|
1999-08-06 13:59:32 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Update current_row from desired_row. */
|
|
|
|
|
make_current (w->desired_matrix, w->current_matrix, vpos);
|
|
|
|
|
updated_row = NULL;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
return changed_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Set the cursor after an update of window W. This function may only
|
|
|
|
|
be called from update_window. */
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
set_window_cursor_after_update (w)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
struct window *w;
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
int cx, cy, vpos, hpos;
|
|
|
|
|
|
|
|
|
|
/* Not intended for frame matrix updates. */
|
|
|
|
|
xassert (FRAME_WINDOW_P (f));
|
|
|
|
|
|
1999-08-21 19:29:02 +00:00
|
|
|
|
if (cursor_in_echo_area
|
|
|
|
|
&& !NILP (echo_area_buffer[0])
|
|
|
|
|
/* If we are showing a message instead of the mini-buffer,
|
|
|
|
|
show the cursor for the message instead. */
|
|
|
|
|
&& XWINDOW (minibuf_window) == w
|
|
|
|
|
&& EQ (minibuf_window, echo_area_window)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* These cases apply only to the frame that contains
|
|
|
|
|
the active mini-buffer window. */
|
|
|
|
|
&& FRAME_HAS_MINIBUF_P (f)
|
|
|
|
|
&& EQ (FRAME_MINIBUF_WINDOW (f), echo_area_window))
|
|
|
|
|
{
|
|
|
|
|
cx = cy = vpos = hpos = 0;
|
|
|
|
|
|
|
|
|
|
if (cursor_in_echo_area >= 0)
|
|
|
|
|
{
|
|
|
|
|
/* If the mini-buffer is several lines high, find the last
|
|
|
|
|
line that has any text on it. Note: either all lines
|
|
|
|
|
are enabled or none. Otherwise we wouldn't be able to
|
|
|
|
|
determine Y. */
|
1999-08-06 15:05:27 +00:00
|
|
|
|
struct glyph_row *row, *last_row;
|
|
|
|
|
struct glyph *glyph;
|
|
|
|
|
int yb = window_text_bottom_y (w);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-08-06 15:05:27 +00:00
|
|
|
|
last_row = NULL;
|
1999-08-21 19:29:02 +00:00
|
|
|
|
for (row = MATRIX_ROW (w->current_matrix, 0);
|
|
|
|
|
row->enabled_p;
|
|
|
|
|
++row)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-08-06 15:05:27 +00:00
|
|
|
|
if (row->used[TEXT_AREA]
|
|
|
|
|
&& row->glyphs[TEXT_AREA][0].charpos >= 0)
|
|
|
|
|
last_row = row;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-08-06 15:05:27 +00:00
|
|
|
|
if (MATRIX_ROW_BOTTOM_Y (row) >= yb)
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (last_row)
|
|
|
|
|
{
|
1999-08-06 15:05:27 +00:00
|
|
|
|
struct glyph *start = row->glyphs[TEXT_AREA];
|
1999-08-21 19:29:02 +00:00
|
|
|
|
struct glyph *last = start + row->used[TEXT_AREA] - 1;
|
1999-08-06 15:05:27 +00:00
|
|
|
|
|
1999-08-21 19:29:02 +00:00
|
|
|
|
while (last > start && last->charpos < 0)
|
1999-08-06 15:05:27 +00:00
|
|
|
|
--last;
|
|
|
|
|
|
|
|
|
|
for (glyph = start; glyph < last; ++glyph)
|
|
|
|
|
{
|
|
|
|
|
cx += glyph->pixel_width;
|
|
|
|
|
++hpos;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
cy = last_row->y;
|
1999-08-06 15:05:27 +00:00
|
|
|
|
vpos = MATRIX_ROW_VPOS (last_row, w->current_matrix);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
cx = w->cursor.x;
|
|
|
|
|
cy = w->cursor.y;
|
|
|
|
|
hpos = w->cursor.hpos;
|
|
|
|
|
vpos = w->cursor.vpos;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Window cursor can be out of sync for horizontally split windows. */
|
|
|
|
|
hpos = max (0, hpos);
|
|
|
|
|
hpos = min (w->current_matrix->matrix_w - 1, hpos);
|
|
|
|
|
vpos = max (0, vpos);
|
|
|
|
|
vpos = min (w->current_matrix->nrows - 1, vpos);
|
|
|
|
|
rif->cursor_to (vpos, hpos, cy, cx);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
/* Set WINDOW->must_be_updated_p to ON_P for all windows in the window
|
|
|
|
|
tree rooted at W. */
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
set_window_update_flags (w, on_p)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int on_p;
|
|
|
|
|
{
|
|
|
|
|
while (w)
|
|
|
|
|
{
|
|
|
|
|
if (!NILP (w->hchild))
|
|
|
|
|
set_window_update_flags (XWINDOW (w->hchild), on_p);
|
|
|
|
|
else if (!NILP (w->vchild))
|
|
|
|
|
set_window_update_flags (XWINDOW (w->vchild), on_p);
|
|
|
|
|
else
|
|
|
|
|
w->must_be_updated_p = on_p;
|
|
|
|
|
|
|
|
|
|
w = NILP (w->next) ? 0 : XWINDOW (w->next);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Window-Based Scrolling
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Structure describing rows in scrolling_window. */
|
|
|
|
|
|
|
|
|
|
struct row_entry
|
|
|
|
|
{
|
|
|
|
|
/* Number of occurrences of this row in desired and current matrix. */
|
|
|
|
|
int old_uses, new_uses;
|
|
|
|
|
|
|
|
|
|
/* Vpos of row in new matrix. */
|
|
|
|
|
int new_line_number;
|
|
|
|
|
|
|
|
|
|
/* Bucket index of this row_entry in the hash table row_table. */
|
|
|
|
|
int bucket;
|
|
|
|
|
|
|
|
|
|
/* The row described by this entry. */
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
|
|
|
|
|
/* Hash collision chain. */
|
|
|
|
|
struct row_entry *next;
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
/* A pool to allocate row_entry structures from, and the size of the
|
|
|
|
|
pool. The pool is reallocated in scrolling_window when we find
|
|
|
|
|
that we need a larger one. */
|
|
|
|
|
|
|
|
|
|
static struct row_entry *row_entry_pool;
|
|
|
|
|
static int row_entry_pool_size;
|
|
|
|
|
|
|
|
|
|
/* Index of next free entry in row_entry_pool. */
|
|
|
|
|
|
|
|
|
|
static int row_entry_idx;
|
|
|
|
|
|
|
|
|
|
/* The hash table used during scrolling, and the table's size. This
|
|
|
|
|
table is used to quickly identify equal rows in the desired and
|
|
|
|
|
current matrix. */
|
|
|
|
|
|
|
|
|
|
static struct row_entry **row_table;
|
|
|
|
|
static int row_table_size;
|
|
|
|
|
|
|
|
|
|
/* Vectors of pointers to row_entry structures belonging to the
|
|
|
|
|
current and desired matrix, and the size of the vectors. */
|
|
|
|
|
|
|
|
|
|
static struct row_entry **old_lines, **new_lines;
|
|
|
|
|
static int old_lines_size, new_lines_size;
|
|
|
|
|
|
|
|
|
|
/* A pool to allocate run structures from, and its size. */
|
|
|
|
|
|
|
|
|
|
static struct run *run_pool;
|
|
|
|
|
static int runs_size;
|
|
|
|
|
|
|
|
|
|
/* A vector of runs of lines found during scrolling. */
|
|
|
|
|
|
|
|
|
|
static struct run **runs;
|
|
|
|
|
|
|
|
|
|
static struct row_entry *add_row_entry P_ ((struct window *,
|
|
|
|
|
struct glyph_row *));
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/* Add glyph row ROW to the scrolling hash table during the scrolling
|
|
|
|
|
of window W. */
|
|
|
|
|
|
|
|
|
|
static INLINE struct row_entry *
|
|
|
|
|
add_row_entry (w, row)
|
|
|
|
|
struct window *w;
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
{
|
|
|
|
|
struct row_entry *entry;
|
|
|
|
|
int i = row->hash % row_table_size;
|
|
|
|
|
|
|
|
|
|
entry = row_table[i];
|
2000-07-11 19:57:54 +00:00
|
|
|
|
while (entry && !row_equal_p (w, entry->row, row, 1))
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
entry = entry->next;
|
|
|
|
|
|
|
|
|
|
if (entry == NULL)
|
|
|
|
|
{
|
|
|
|
|
entry = row_entry_pool + row_entry_idx++;
|
|
|
|
|
entry->row = row;
|
|
|
|
|
entry->old_uses = entry->new_uses = 0;
|
|
|
|
|
entry->new_line_number = 0;
|
|
|
|
|
entry->bucket = i;
|
|
|
|
|
entry->next = row_table[i];
|
|
|
|
|
row_table[i] = entry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return entry;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Try to reuse part of the current display of W by scrolling lines.
|
1999-09-05 16:39:19 +00:00
|
|
|
|
HEADER_LINE_P non-zero means W has a top mode line.
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
The algorithm is taken from Communications of the ACM, Apr78 "A
|
|
|
|
|
Technique for Isolating Differences Between Files." It should take
|
|
|
|
|
O(N) time.
|
|
|
|
|
|
|
|
|
|
A short outline of the steps of the algorithm
|
|
|
|
|
|
|
|
|
|
1. Skip lines equal at the start and end of both matrices.
|
|
|
|
|
|
|
|
|
|
2. Enter rows in the current and desired matrix into a symbol
|
|
|
|
|
table, counting how often they appear in both matrices.
|
|
|
|
|
|
|
|
|
|
3. Rows that appear exactly once in both matrices serve as anchors,
|
|
|
|
|
i.e. we assume that such lines are likely to have been moved.
|
|
|
|
|
|
|
|
|
|
4. Starting from anchor lines, extend regions to be scrolled both
|
|
|
|
|
forward and backward.
|
|
|
|
|
|
|
|
|
|
Value is
|
|
|
|
|
|
|
|
|
|
-1 if all rows were found to be equal.
|
|
|
|
|
0 to indicate that we did not scroll the display, or
|
|
|
|
|
1 if we did scroll. */
|
|
|
|
|
|
|
|
|
|
static int
|
1999-09-05 16:39:19 +00:00
|
|
|
|
scrolling_window (w, header_line_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct window *w;
|
1999-09-05 16:39:19 +00:00
|
|
|
|
int header_line_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *desired_matrix = w->desired_matrix;
|
|
|
|
|
struct glyph_matrix *current_matrix = w->current_matrix;
|
|
|
|
|
int yb = window_text_bottom_y (w);
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
int i, j, first_old, first_new, last_old, last_new;
|
|
|
|
|
int nruns, nbytes, n, run_idx;
|
|
|
|
|
struct row_entry *entry;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Skip over rows equal at the start. */
|
2000-09-21 18:41:36 +00:00
|
|
|
|
for (i = header_line_p ? 1 : 0; i < current_matrix->nrows - 1; ++i)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
2000-09-21 18:41:36 +00:00
|
|
|
|
struct glyph_row *d = MATRIX_ROW (desired_matrix, i);
|
|
|
|
|
struct glyph_row *c = MATRIX_ROW (current_matrix, i);
|
|
|
|
|
|
|
|
|
|
if (c->enabled_p
|
|
|
|
|
&& d->enabled_p
|
|
|
|
|
&& c->y == d->y
|
|
|
|
|
&& MATRIX_ROW_BOTTOM_Y (c) <= yb
|
|
|
|
|
&& MATRIX_ROW_BOTTOM_Y (d) <= yb
|
|
|
|
|
&& row_equal_p (w, c, d, 1))
|
|
|
|
|
{
|
|
|
|
|
assign_row (c, d);
|
|
|
|
|
d->enabled_p = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
break;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Give up if some rows in the desired matrix are not enabled. */
|
|
|
|
|
if (!MATRIX_ROW (desired_matrix, i)->enabled_p)
|
|
|
|
|
return -1;
|
|
|
|
|
|
|
|
|
|
first_old = first_new = i;
|
|
|
|
|
|
|
|
|
|
/* Set last_new to the index + 1 of the last enabled row in the
|
|
|
|
|
desired matrix. */
|
|
|
|
|
i = first_new + 1;
|
|
|
|
|
while (i < desired_matrix->nrows - 1
|
|
|
|
|
&& MATRIX_ROW (desired_matrix, i)->enabled_p
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
&& MATRIX_ROW_BOTTOM_Y (MATRIX_ROW (desired_matrix, i)) <= yb)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
++i;
|
|
|
|
|
|
|
|
|
|
if (!MATRIX_ROW (desired_matrix, i)->enabled_p)
|
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
last_new = i;
|
|
|
|
|
|
|
|
|
|
/* Set last_old to the index + 1 of the last enabled row in the
|
|
|
|
|
current matrix. We don't look at the enabled flag here because
|
|
|
|
|
we plan to reuse part of the display even if other parts are
|
|
|
|
|
disabled. */
|
|
|
|
|
i = first_old + 1;
|
|
|
|
|
while (i < current_matrix->nrows - 1
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
&& MATRIX_ROW_BOTTOM_Y (MATRIX_ROW (current_matrix, i)) <= yb)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
++i;
|
|
|
|
|
last_old = i;
|
|
|
|
|
|
|
|
|
|
/* Skip over rows equal at the bottom. */
|
|
|
|
|
i = last_new;
|
|
|
|
|
j = last_old;
|
|
|
|
|
while (i - 1 > first_new
|
|
|
|
|
&& j - 1 > first_old
|
|
|
|
|
&& MATRIX_ROW (current_matrix, i - 1)->enabled_p
|
|
|
|
|
&& (MATRIX_ROW (current_matrix, i - 1)->y
|
|
|
|
|
== MATRIX_ROW (desired_matrix, j - 1)->y)
|
|
|
|
|
&& row_equal_p (w,
|
|
|
|
|
MATRIX_ROW (desired_matrix, i - 1),
|
2000-07-11 19:57:54 +00:00
|
|
|
|
MATRIX_ROW (current_matrix, j - 1), 1))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
--i, --j;
|
|
|
|
|
last_new = i;
|
|
|
|
|
last_old = j;
|
|
|
|
|
|
|
|
|
|
/* Nothing to do if all rows are equal. */
|
|
|
|
|
if (last_new == first_new)
|
|
|
|
|
return 0;
|
|
|
|
|
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
/* Reallocate vectors, tables etc. if necessary. */
|
|
|
|
|
|
|
|
|
|
if (current_matrix->nrows > old_lines_size)
|
|
|
|
|
{
|
|
|
|
|
old_lines_size = current_matrix->nrows;
|
|
|
|
|
nbytes = old_lines_size * sizeof *old_lines;
|
|
|
|
|
old_lines = (struct row_entry **) xrealloc (old_lines, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (desired_matrix->nrows > new_lines_size)
|
|
|
|
|
{
|
|
|
|
|
new_lines_size = desired_matrix->nrows;
|
|
|
|
|
nbytes = new_lines_size * sizeof *new_lines;
|
|
|
|
|
new_lines = (struct row_entry **) xrealloc (new_lines, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
n = desired_matrix->nrows + current_matrix->nrows;
|
|
|
|
|
if (3 * n > row_table_size)
|
|
|
|
|
{
|
|
|
|
|
row_table_size = next_almost_prime (3 * n);
|
|
|
|
|
nbytes = row_table_size * sizeof *row_table;
|
|
|
|
|
row_table = (struct row_entry **) xrealloc (row_table, nbytes);
|
|
|
|
|
bzero (row_table, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (n > row_entry_pool_size)
|
|
|
|
|
{
|
|
|
|
|
row_entry_pool_size = n;
|
|
|
|
|
nbytes = row_entry_pool_size * sizeof *row_entry_pool;
|
|
|
|
|
row_entry_pool = (struct row_entry *) xrealloc (row_entry_pool, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (desired_matrix->nrows > runs_size)
|
|
|
|
|
{
|
|
|
|
|
runs_size = desired_matrix->nrows;
|
|
|
|
|
nbytes = runs_size * sizeof *runs;
|
|
|
|
|
runs = (struct run **) xrealloc (runs, nbytes);
|
|
|
|
|
nbytes = runs_size * sizeof *run_pool;
|
|
|
|
|
run_pool = (struct run *) xrealloc (run_pool, nbytes);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
nruns = run_idx = 0;
|
|
|
|
|
row_entry_idx = 0;
|
|
|
|
|
|
|
|
|
|
/* Add rows from the current and desired matrix to the hash table
|
|
|
|
|
row_hash_table to be able to find equal ones quickly. */
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (i = first_old; i < last_old; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (MATRIX_ROW (current_matrix, i)->enabled_p)
|
|
|
|
|
{
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
entry = add_row_entry (w, MATRIX_ROW (current_matrix, i));
|
|
|
|
|
old_lines[i] = entry;
|
|
|
|
|
++entry->old_uses;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
old_lines[i] = NULL;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for (i = first_new; i < last_new; ++i)
|
|
|
|
|
{
|
|
|
|
|
xassert (MATRIX_ROW_ENABLED_P (desired_matrix, i));
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
entry = add_row_entry (w, MATRIX_ROW (desired_matrix, i));
|
|
|
|
|
++entry->new_uses;
|
|
|
|
|
entry->new_line_number = i;
|
|
|
|
|
new_lines[i] = entry;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Identify moves based on lines that are unique and equal
|
|
|
|
|
in both matrices. */
|
|
|
|
|
for (i = first_old; i < last_old;)
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
if (old_lines[i]
|
|
|
|
|
&& old_lines[i]->old_uses == 1
|
|
|
|
|
&& old_lines[i]->new_uses == 1)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
int j, k;
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
int new_line = old_lines[i]->new_line_number;
|
|
|
|
|
struct run *run = run_pool + run_idx++;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Record move. */
|
|
|
|
|
run->current_vpos = i;
|
|
|
|
|
run->current_y = MATRIX_ROW (current_matrix, i)->y;
|
|
|
|
|
run->desired_vpos = new_line;
|
|
|
|
|
run->desired_y = MATRIX_ROW (desired_matrix, new_line)->y;
|
|
|
|
|
run->nrows = 1;
|
|
|
|
|
run->height = MATRIX_ROW (current_matrix, i)->height;
|
|
|
|
|
|
|
|
|
|
/* Extend backward. */
|
|
|
|
|
j = i - 1;
|
|
|
|
|
k = new_line - 1;
|
|
|
|
|
while (j > first_old
|
|
|
|
|
&& k > first_new
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
&& old_lines[j] == new_lines[k])
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
int h = MATRIX_ROW (current_matrix, j)->height;
|
|
|
|
|
--run->current_vpos;
|
|
|
|
|
--run->desired_vpos;
|
|
|
|
|
++run->nrows;
|
|
|
|
|
run->height += h;
|
|
|
|
|
run->desired_y -= h;
|
|
|
|
|
run->current_y -= h;
|
|
|
|
|
--j, --k;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Extend forward. */
|
|
|
|
|
j = i + 1;
|
|
|
|
|
k = new_line + 1;
|
|
|
|
|
while (j < last_old
|
|
|
|
|
&& k < last_new
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
&& old_lines[j] == new_lines[k])
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
int h = MATRIX_ROW (current_matrix, j)->height;
|
|
|
|
|
++run->nrows;
|
|
|
|
|
run->height += h;
|
|
|
|
|
++j, ++k;
|
|
|
|
|
}
|
1994-07-20 19:35:41 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Insert run into list of all runs. Order runs by copied
|
|
|
|
|
pixel lines. Note that we record runs that don't have to
|
|
|
|
|
be copied because they are already in place. This is done
|
|
|
|
|
because we can avoid calling update_window_line in this
|
|
|
|
|
case. */
|
|
|
|
|
for (j = 0; j < nruns && runs[j]->height > run->height; ++j)
|
|
|
|
|
;
|
|
|
|
|
for (k = nruns; k >= j; --k)
|
|
|
|
|
runs[k] = runs[k - 1];
|
|
|
|
|
runs[j] = run;
|
|
|
|
|
++nruns;
|
|
|
|
|
|
|
|
|
|
i += run->nrows;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
++i;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Do the moves. Do it in a way that we don't overwrite something
|
|
|
|
|
we want to copy later on. This is not solvable in general
|
|
|
|
|
because there is only one display and we don't have a way to
|
|
|
|
|
exchange areas on this display. Example:
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
+-----------+ +-----------+
|
|
|
|
|
| A | | B |
|
|
|
|
|
+-----------+ --> +-----------+
|
|
|
|
|
| B | | A |
|
|
|
|
|
+-----------+ +-----------+
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Instead, prefer bigger moves, and invalidate moves that would
|
|
|
|
|
copy from where we copied to. */
|
1993-12-23 00:22:10 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (i = 0; i < nruns; ++i)
|
|
|
|
|
if (runs[i]->nrows > 0)
|
|
|
|
|
{
|
|
|
|
|
struct run *r = runs[i];
|
1997-02-20 06:35:00 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Copy on the display. */
|
|
|
|
|
if (r->current_y != r->desired_y)
|
|
|
|
|
{
|
|
|
|
|
rif->scroll_run_hook (w, r);
|
|
|
|
|
|
|
|
|
|
/* Invalidate runs that copy from where we copied to. */
|
|
|
|
|
for (j = i + 1; j < nruns; ++j)
|
|
|
|
|
{
|
|
|
|
|
struct run *p = runs[j];
|
|
|
|
|
|
|
|
|
|
if ((p->current_y >= r->desired_y
|
|
|
|
|
&& p->current_y < r->desired_y + r->height)
|
|
|
|
|
|| (p->current_y + p->height >= r->desired_y
|
|
|
|
|
&& (p->current_y + p->height
|
|
|
|
|
< r->desired_y + r->height)))
|
|
|
|
|
p->nrows = 0;
|
|
|
|
|
}
|
|
|
|
|
}
|
1993-12-23 00:22:10 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Assign matrix rows. */
|
|
|
|
|
for (j = 0; j < r->nrows; ++j)
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *from, *to;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
int to_overlapped_p;
|
2000-07-12 15:10:30 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
to = MATRIX_ROW (current_matrix, r->desired_vpos + j);
|
|
|
|
|
from = MATRIX_ROW (desired_matrix, r->desired_vpos + j);
|
2000-07-12 15:10:30 +00:00
|
|
|
|
to_overlapped_p = to->overlapped_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
assign_row (to, from);
|
|
|
|
|
to->enabled_p = 1, from->enabled_p = 0;
|
1999-08-06 13:59:32 +00:00
|
|
|
|
to->overlapped_p = to_overlapped_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1995-03-22 21:22:31 +00:00
|
|
|
|
|
(struct row_entry): New structure.
(row_entry_pool, row_entry_pool_size, row_entry_idx, row_table)
(row_table_size, old_lines, new_lines, old_lines_size)
(new_lines_size, run_pool, runs_size, runs): New variables.
(add_row_entry): New function.
(scrolling_window): Use data structures allocated with xmalloc and
held in global variables, instead of allocing them with alloca and
holding them in local variables. Use a larger hash table whose
size depends on glyph matrix sizes. Don't use bzero to clear the
hash table; instead, clear used slots only.
2000-06-28 20:28:50 +00:00
|
|
|
|
/* Clear the hash table, for the next time. */
|
|
|
|
|
for (i = 0; i < row_entry_idx; ++i)
|
|
|
|
|
row_table[row_entry_pool[i].bucket] = NULL;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Value is non-zero to indicate that we scrolled the display. */
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
1993-05-15 18:44:46 +00:00
|
|
|
|
|
1995-03-22 21:22:31 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/************************************************************************
|
|
|
|
|
Frame-Based Updates
|
|
|
|
|
************************************************************************/
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Update the desired frame matrix of frame F.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
FORCE_P non-zero means that the update should not be stopped by
|
|
|
|
|
pending input. INHIBIT_HAIRY_ID_P non-zero means that scrolling
|
|
|
|
|
should not be tried.
|
|
|
|
|
|
|
|
|
|
Value is non-zero if update was stopped due to pending input. */
|
|
|
|
|
|
|
|
|
|
static int
|
|
|
|
|
update_frame_1 (f, force_p, inhibit_id_p)
|
|
|
|
|
struct frame *f;
|
|
|
|
|
int force_p;
|
|
|
|
|
int inhibit_id_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Frame matrices to work on. */
|
|
|
|
|
struct glyph_matrix *current_matrix = f->current_matrix;
|
|
|
|
|
struct glyph_matrix *desired_matrix = f->desired_matrix;
|
|
|
|
|
int i;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
int pause;
|
|
|
|
|
int preempt_count = baud_rate / 2400 + 1;
|
1998-04-14 12:25:56 +00:00
|
|
|
|
extern int input_pending;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
xassert (current_matrix && desired_matrix);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1994-12-06 08:49:44 +00:00
|
|
|
|
if (baud_rate != FRAME_COST_BAUD_RATE (f))
|
|
|
|
|
calculate_costs (f);
|
|
|
|
|
|
1993-05-31 19:04:20 +00:00
|
|
|
|
if (preempt_count <= 0)
|
|
|
|
|
preempt_count = 1;
|
|
|
|
|
|
2000-08-09 18:39:58 +00:00
|
|
|
|
if (redisplay_dont_pause)
|
|
|
|
|
force_p = 1;
|
|
|
|
|
else if (!force_p && detect_input_pending ())
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
pause = 1;
|
|
|
|
|
goto do_pause;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If we cannot insert/delete lines, it's no use trying it. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
if (!line_ins_del_ok)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
inhibit_id_p = 1;
|
1994-04-29 04:54:39 +00:00
|
|
|
|
|
1992-01-14 08:05:08 +00:00
|
|
|
|
/* See if any of the desired lines are enabled; don't compute for
|
1999-07-21 21:43:52 +00:00
|
|
|
|
i/d line if just want cursor motion. */
|
|
|
|
|
for (i = 0; i < desired_matrix->nrows; i++)
|
|
|
|
|
if (MATRIX_ROW_ENABLED_P (desired_matrix, i))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
/* Try doing i/d line, if not yet inhibited. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (!inhibit_id_p && i < desired_matrix->nrows)
|
|
|
|
|
force_p |= scrolling (f);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* Update the individual lines as needed. Do bottom line first. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (MATRIX_ROW_ENABLED_P (desired_matrix, desired_matrix->nrows - 1))
|
|
|
|
|
update_frame_line (f, desired_matrix->nrows - 1);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Now update the rest of the lines. */
|
|
|
|
|
for (i = 0; i < desired_matrix->nrows - 1 && (force_p || !input_pending); i++)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (MATRIX_ROW_ENABLED_P (desired_matrix, i))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
if (FRAME_TERMCAP_P (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
/* Flush out every so many lines.
|
|
|
|
|
Also flush out if likely to have more than 1k buffered
|
|
|
|
|
otherwise. I'm told that some telnet connections get
|
|
|
|
|
really screwed by more than 1k output at once. */
|
|
|
|
|
int outq = PENDING_OUTPUT_COUNT (stdout);
|
|
|
|
|
if (outq > 900
|
|
|
|
|
|| (outq > 20 && ((i - 1) % preempt_count == 0)))
|
|
|
|
|
{
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
if (preempt_count == 1)
|
|
|
|
|
{
|
1992-02-20 04:15:20 +00:00
|
|
|
|
#ifdef EMACS_OUTQSIZE
|
|
|
|
|
if (EMACS_OUTQSIZE (0, &outq) < 0)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* Probably not a tty. Ignore the error and reset
|
1999-07-21 21:43:52 +00:00
|
|
|
|
* the outq count. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
outq = PENDING_OUTPUT_COUNT (stdout);
|
|
|
|
|
#endif
|
|
|
|
|
outq *= 10;
|
1994-05-17 19:38:45 +00:00
|
|
|
|
if (baud_rate <= outq && baud_rate > 0)
|
1993-05-31 19:04:20 +00:00
|
|
|
|
sleep (outq / baud_rate);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1997-01-03 23:05:58 +00:00
|
|
|
|
if ((i - 1) % preempt_count == 0)
|
|
|
|
|
detect_input_pending ();
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
update_frame_line (f, i);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
pause = (i < FRAME_HEIGHT (f) - 1) ? i : 0;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* Now just clean up termcap drivers and set cursor, etc. */
|
|
|
|
|
if (!pause)
|
|
|
|
|
{
|
1995-06-28 09:38:18 +00:00
|
|
|
|
if ((cursor_in_echo_area
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If we are showing a message instead of the mini-buffer,
|
1995-07-21 07:46:14 +00:00
|
|
|
|
show the cursor for the message instead of for the
|
1999-07-21 21:43:52 +00:00
|
|
|
|
(now hidden) mini-buffer contents. */
|
1995-07-21 07:46:14 +00:00
|
|
|
|
|| (EQ (minibuf_window, selected_window)
|
|
|
|
|
&& EQ (minibuf_window, echo_area_window)
|
1999-08-21 19:29:02 +00:00
|
|
|
|
&& !NILP (echo_area_buffer[0])))
|
1995-07-21 07:46:14 +00:00
|
|
|
|
/* These cases apply only to the frame that contains
|
1999-07-21 21:43:52 +00:00
|
|
|
|
the active mini-buffer window. */
|
1995-07-21 07:46:14 +00:00
|
|
|
|
&& FRAME_HAS_MINIBUF_P (f)
|
1996-02-01 21:30:03 +00:00
|
|
|
|
&& EQ (FRAME_MINIBUF_WINDOW (f), echo_area_window))
|
1992-06-10 03:53:16 +00:00
|
|
|
|
{
|
1992-08-29 02:55:05 +00:00
|
|
|
|
int top = XINT (XWINDOW (FRAME_MINIBUF_WINDOW (f))->top);
|
|
|
|
|
int row, col;
|
|
|
|
|
|
|
|
|
|
if (cursor_in_echo_area < 0)
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Negative value of cursor_in_echo_area means put
|
|
|
|
|
cursor at beginning of line. */
|
1992-08-29 02:55:05 +00:00
|
|
|
|
row = top;
|
|
|
|
|
col = 0;
|
|
|
|
|
}
|
1992-06-10 03:53:16 +00:00
|
|
|
|
else
|
1992-08-29 02:55:05 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Positive value of cursor_in_echo_area means put
|
|
|
|
|
cursor at the end of the prompt. If the mini-buffer
|
|
|
|
|
is several lines high, find the last line that has
|
|
|
|
|
any text on it. */
|
1992-08-29 02:55:05 +00:00
|
|
|
|
row = FRAME_HEIGHT (f);
|
|
|
|
|
do
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
--row;
|
|
|
|
|
col = 0;
|
|
|
|
|
|
|
|
|
|
if (MATRIX_ROW_ENABLED_P (current_matrix, row))
|
|
|
|
|
{
|
|
|
|
|
/* Frame rows are filled up with spaces that
|
|
|
|
|
must be ignored here. */
|
|
|
|
|
struct glyph_row *r = MATRIX_ROW (current_matrix,
|
|
|
|
|
row);
|
|
|
|
|
struct glyph *start = r->glyphs[TEXT_AREA];
|
|
|
|
|
struct glyph *last = start + r->used[TEXT_AREA];
|
|
|
|
|
|
|
|
|
|
while (last > start
|
|
|
|
|
&& (last - 1)->charpos < 0)
|
|
|
|
|
--last;
|
|
|
|
|
|
|
|
|
|
col = last - start;
|
|
|
|
|
}
|
1992-08-29 02:55:05 +00:00
|
|
|
|
}
|
|
|
|
|
while (row > top && col == 0);
|
|
|
|
|
|
1998-04-24 21:22:48 +00:00
|
|
|
|
/* Make sure COL is not out of range. */
|
1998-04-25 23:08:09 +00:00
|
|
|
|
if (col >= FRAME_CURSOR_X_LIMIT (f))
|
1992-08-29 02:55:05 +00:00
|
|
|
|
{
|
1998-04-24 21:22:48 +00:00
|
|
|
|
/* If we have another row, advance cursor into it. */
|
1992-08-29 02:55:05 +00:00
|
|
|
|
if (row < FRAME_HEIGHT (f) - 1)
|
1998-04-24 21:22:48 +00:00
|
|
|
|
{
|
|
|
|
|
col = FRAME_LEFT_SCROLL_BAR_WIDTH (f);
|
|
|
|
|
row++;
|
|
|
|
|
}
|
|
|
|
|
/* Otherwise move it back in range. */
|
|
|
|
|
else
|
1998-04-25 23:08:09 +00:00
|
|
|
|
col = FRAME_CURSOR_X_LIMIT (f) - 1;
|
1992-08-29 02:55:05 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
cursor_to (row, col);
|
1992-06-10 03:53:16 +00:00
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
else
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
/* We have only one cursor on terminal frames. Use it to
|
|
|
|
|
display the cursor of the selected window. */
|
|
|
|
|
struct window *w = XWINDOW (FRAME_SELECTED_WINDOW (f));
|
2000-06-15 22:51:37 +00:00
|
|
|
|
if (w->cursor.vpos >= 0
|
|
|
|
|
/* The cursor vpos may be temporarily out of bounds
|
|
|
|
|
in the following situation: There is one window,
|
|
|
|
|
with the cursor in the lower half of it. The window
|
|
|
|
|
is split, and a message causes a redisplay before
|
|
|
|
|
a new cursor position has been computed. */
|
|
|
|
|
&& w->cursor.vpos < XFASTINT (w->height))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
int x = WINDOW_TO_FRAME_HPOS (w, w->cursor.hpos);
|
|
|
|
|
int y = WINDOW_TO_FRAME_VPOS (w, w->cursor.vpos);
|
|
|
|
|
|
|
|
|
|
if (INTEGERP (w->left_margin_width))
|
|
|
|
|
x += XFASTINT (w->left_margin_width);
|
|
|
|
|
|
|
|
|
|
/* x = max (min (x, FRAME_WINDOW_WIDTH (f) - 1), 0); */
|
|
|
|
|
cursor_to (y, x);
|
|
|
|
|
}
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
do_pause:
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
clear_desired_matrices (f);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return pause;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Do line insertions/deletions on frame F for frame-based redisplay. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
int
|
1992-07-13 19:54:34 +00:00
|
|
|
|
scrolling (frame)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct frame *frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
int unchanged_at_top, unchanged_at_bottom;
|
|
|
|
|
int window_size;
|
|
|
|
|
int changed_lines;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
int *old_hash = (int *) alloca (FRAME_HEIGHT (frame) * sizeof (int));
|
|
|
|
|
int *new_hash = (int *) alloca (FRAME_HEIGHT (frame) * sizeof (int));
|
|
|
|
|
int *draw_cost = (int *) alloca (FRAME_HEIGHT (frame) * sizeof (int));
|
1994-12-26 15:35:48 +00:00
|
|
|
|
int *old_draw_cost = (int *) alloca (FRAME_HEIGHT (frame) * sizeof (int));
|
1991-07-03 12:10:07 +00:00
|
|
|
|
register int i;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
int free_at_end_vpos = FRAME_HEIGHT (frame);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *current_matrix = frame->current_matrix;
|
|
|
|
|
struct glyph_matrix *desired_matrix = frame->desired_matrix;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (!current_matrix)
|
|
|
|
|
abort ();
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Compute hash codes of all the lines. Also calculate number of
|
|
|
|
|
changed lines, number of unchanged lines at the beginning, and
|
|
|
|
|
number of unchanged lines at the end. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
changed_lines = 0;
|
|
|
|
|
unchanged_at_top = 0;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
unchanged_at_bottom = FRAME_HEIGHT (frame);
|
|
|
|
|
for (i = 0; i < FRAME_HEIGHT (frame); i++)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
/* Give up on this scrolling if some old lines are not enabled. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (!MATRIX_ROW_ENABLED_P (current_matrix, i))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return 0;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
old_hash[i] = line_hash_code (MATRIX_ROW (current_matrix, i));
|
|
|
|
|
if (! MATRIX_ROW_ENABLED_P (desired_matrix, i))
|
1997-05-29 03:32:17 +00:00
|
|
|
|
{
|
|
|
|
|
/* This line cannot be redrawn, so don't let scrolling mess it. */
|
|
|
|
|
new_hash[i] = old_hash[i];
|
|
|
|
|
#define INFINITY 1000000 /* Taken from scroll.c */
|
|
|
|
|
draw_cost[i] = INFINITY;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
else
|
1997-05-29 03:32:17 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
new_hash[i] = line_hash_code (MATRIX_ROW (desired_matrix, i));
|
|
|
|
|
draw_cost[i] = line_draw_cost (desired_matrix, i);
|
1997-05-29 03:32:17 +00:00
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
if (old_hash[i] != new_hash[i])
|
|
|
|
|
{
|
|
|
|
|
changed_lines++;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
unchanged_at_bottom = FRAME_HEIGHT (frame) - i - 1;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
else if (i == unchanged_at_top)
|
|
|
|
|
unchanged_at_top++;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
old_draw_cost[i] = line_draw_cost (current_matrix, i);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If changed lines are few, don't allow preemption, don't scroll. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if ((!scroll_region_ok && changed_lines < baud_rate / 2400)
|
1992-07-13 19:54:34 +00:00
|
|
|
|
|| unchanged_at_bottom == FRAME_HEIGHT (frame))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return 1;
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
window_size = (FRAME_HEIGHT (frame) - unchanged_at_top
|
1991-07-03 12:10:07 +00:00
|
|
|
|
- unchanged_at_bottom);
|
|
|
|
|
|
|
|
|
|
if (scroll_region_ok)
|
|
|
|
|
free_at_end_vpos -= unchanged_at_bottom;
|
1992-07-13 19:54:34 +00:00
|
|
|
|
else if (memory_below_frame)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
free_at_end_vpos = -1;
|
|
|
|
|
|
|
|
|
|
/* If large window, fast terminal and few lines in common between
|
1999-07-21 21:43:52 +00:00
|
|
|
|
current frame and desired frame, don't bother with i/d calc. */
|
1994-12-26 15:35:48 +00:00
|
|
|
|
if (!scroll_region_ok && window_size >= 18 && baud_rate > 2400
|
1991-07-03 12:10:07 +00:00
|
|
|
|
&& (window_size >=
|
|
|
|
|
10 * scrolling_max_lines_saved (unchanged_at_top,
|
1992-07-13 19:54:34 +00:00
|
|
|
|
FRAME_HEIGHT (frame) - unchanged_at_bottom,
|
1991-07-03 12:10:07 +00:00
|
|
|
|
old_hash, new_hash, draw_cost)))
|
|
|
|
|
return 0;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (window_size < 2)
|
|
|
|
|
return 0;
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
scrolling_1 (frame, window_size, unchanged_at_top, unchanged_at_bottom,
|
1991-07-03 12:10:07 +00:00
|
|
|
|
draw_cost + unchanged_at_top - 1,
|
1994-12-26 15:35:48 +00:00
|
|
|
|
old_draw_cost + unchanged_at_top - 1,
|
1991-07-03 12:10:07 +00:00
|
|
|
|
old_hash + unchanged_at_top - 1,
|
|
|
|
|
new_hash + unchanged_at_top - 1,
|
|
|
|
|
free_at_end_vpos - unchanged_at_top);
|
|
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Count the number of blanks at the start of the vector of glyphs R
|
|
|
|
|
which is LEN glyphs long. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
static int
|
1999-07-21 21:43:52 +00:00
|
|
|
|
count_blanks (r, len)
|
|
|
|
|
struct glyph *r;
|
|
|
|
|
int len;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int i;
|
2000-10-31 13:32:29 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
for (i = 0; i < len; ++i)
|
|
|
|
|
if (!CHAR_GLYPH_SPACE_P (r[i]))
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
|
return i;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Count the number of glyphs in common at the start of the glyph
|
|
|
|
|
vectors STR1 and STR2. END1 is the end of STR1 and END2 is the end
|
|
|
|
|
of STR2. Value is the number of equal glyphs equal at the start. */
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
static int
|
1999-07-21 21:43:52 +00:00
|
|
|
|
count_match (str1, end1, str2, end2)
|
|
|
|
|
struct glyph *str1, *end1, *str2, *end2;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph *p1 = str1;
|
|
|
|
|
struct glyph *p2 = str2;
|
|
|
|
|
|
|
|
|
|
while (p1 < end1
|
|
|
|
|
&& p2 < end2
|
1999-12-27 05:06:03 +00:00
|
|
|
|
&& GLYPH_CHAR_AND_FACE_EQUAL_P (p1, p2))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
++p1, ++p2;
|
|
|
|
|
|
|
|
|
|
return p1 - str1;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* Char insertion/deletion cost vector, from term.c */
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
extern int *char_ins_del_vector;
|
1996-09-21 04:17:30 +00:00
|
|
|
|
#define char_ins_del_cost(f) (&char_ins_del_vector[FRAME_WINDOW_WIDTH((f))])
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Perform a frame-based update on line VPOS in frame FRAME. */
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
static void
|
1999-07-21 21:43:52 +00:00
|
|
|
|
update_frame_line (frame, vpos)
|
|
|
|
|
register struct frame *frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
int vpos;
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph *obody, *nbody, *op1, *op2, *np1, *nend;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
int tem;
|
|
|
|
|
int osp, nsp, begmatch, endmatch, olen, nlen;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct glyph_matrix *current_matrix = frame->current_matrix;
|
|
|
|
|
struct glyph_matrix *desired_matrix = frame->desired_matrix;
|
|
|
|
|
struct glyph_row *current_row = MATRIX_ROW (current_matrix, vpos);
|
|
|
|
|
struct glyph_row *desired_row = MATRIX_ROW (desired_matrix, vpos);
|
|
|
|
|
int must_write_whole_line_p;
|
2000-10-31 13:32:29 +00:00
|
|
|
|
int write_spaces_p = must_write_spaces;
|
|
|
|
|
int colored_spaces_p = (FACE_FROM_ID (frame, DEFAULT_FACE_ID)->background
|
|
|
|
|
!= FACE_TTY_DEFAULT_BG_COLOR);
|
|
|
|
|
|
|
|
|
|
if (colored_spaces_p)
|
|
|
|
|
write_spaces_p = 1;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
if (desired_row->inverse_p
|
|
|
|
|
!= (current_row->enabled_p && current_row->inverse_p))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int n = current_row->enabled_p ? current_row->used[TEXT_AREA] : 0;
|
|
|
|
|
change_line_highlight (desired_row->inverse_p, vpos, vpos, n);
|
|
|
|
|
current_row->enabled_p = 0;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
1999-07-21 21:43:52 +00:00
|
|
|
|
reassert_line_highlight (desired_row->inverse_p, vpos);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* Current row not enabled means it has unknown contents. We must
|
|
|
|
|
write the whole desired line in that case. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
must_write_whole_line_p = !current_row->enabled_p;
|
|
|
|
|
if (must_write_whole_line_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
obody = 0;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
olen = 0;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
obody = MATRIX_ROW_GLYPH_START (current_matrix, vpos);
|
|
|
|
|
olen = current_row->used[TEXT_AREA];
|
|
|
|
|
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (!current_row->inverse_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* Ignore trailing spaces, if we can. */
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (!write_spaces_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (olen > 0 && CHAR_GLYPH_SPACE_P (obody[olen-1]))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
olen--;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* For an inverse-video line, make sure it's filled with
|
|
|
|
|
spaces all the way to the frame edge so that the reverse
|
|
|
|
|
video extends all the way across. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (olen < FRAME_WIDTH (frame) - 1)
|
|
|
|
|
obody[olen++] = space_glyph;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
current_row->enabled_p = 1;
|
|
|
|
|
current_row->used[TEXT_AREA] = desired_row->used[TEXT_AREA];
|
|
|
|
|
current_row->inverse_p = desired_row->inverse_p;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If desired line is empty, just clear the line. */
|
|
|
|
|
if (!desired_row->enabled_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
nlen = 0;
|
|
|
|
|
goto just_erase;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
nbody = desired_row->glyphs[TEXT_AREA];
|
|
|
|
|
nlen = desired_row->used[TEXT_AREA];
|
|
|
|
|
nend = nbody + nlen;
|
|
|
|
|
|
|
|
|
|
/* If display line has unknown contents, write the whole line. */
|
|
|
|
|
if (must_write_whole_line_p)
|
|
|
|
|
{
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* Ignore spaces at the end, if we can. */
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (!write_spaces_p)
|
1999-08-17 22:44:37 +00:00
|
|
|
|
while (nlen > 0 && CHAR_GLYPH_SPACE_P (nbody[nlen - 1]))
|
|
|
|
|
--nlen;
|
|
|
|
|
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* Write the contents of the desired line. */
|
1999-08-17 22:44:37 +00:00
|
|
|
|
if (nlen)
|
1999-09-15 12:58:31 +00:00
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, 0);
|
|
|
|
|
write_glyphs (nbody, nlen);
|
|
|
|
|
}
|
1999-08-17 22:44:37 +00:00
|
|
|
|
|
1999-09-15 12:58:31 +00:00
|
|
|
|
/* Don't call clear_end_of_line if we already wrote the whole
|
|
|
|
|
line. The cursor will not be at the right margin in that
|
|
|
|
|
case but in the line below. */
|
|
|
|
|
if (nlen < FRAME_WINDOW_WIDTH (frame))
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nlen);
|
|
|
|
|
clear_end_of_line (FRAME_WINDOW_WIDTH (frame));
|
|
|
|
|
}
|
2000-04-22 14:12:30 +00:00
|
|
|
|
else
|
|
|
|
|
/* Make sure we are in the right row, otherwise cursor movement
|
|
|
|
|
with cmgoto might use `ch' in the wrong row. */
|
|
|
|
|
cursor_to (vpos, 0);
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
make_current (desired_matrix, current_matrix, vpos);
|
|
|
|
|
return;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* Pretend trailing spaces are not there at all,
|
|
|
|
|
unless for one reason or another we must write all spaces. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (!desired_row->inverse_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (!write_spaces_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (nlen > 0 && CHAR_GLYPH_SPACE_P (nbody[nlen - 1]))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
nlen--;
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* For an inverse-video line, give it extra trailing spaces all
|
|
|
|
|
the way to the frame edge so that the reverse video extends
|
|
|
|
|
all the way across. */
|
|
|
|
|
while (nlen < FRAME_WIDTH (frame) - 1)
|
|
|
|
|
nbody[nlen++] = space_glyph;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* If there's no i/d char, quickly do the best we can without it. */
|
|
|
|
|
if (!char_ins_del_ok)
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int i, j;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Find the first glyph in desired row that doesn't agree with
|
|
|
|
|
a glyph in the current row, and write the rest from there on. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
for (i = 0; i < nlen; i++)
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (i >= olen || !GLYPH_EQUAL_P (nbody + i, obody + i))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Find the end of the run of different glyphs. */
|
|
|
|
|
j = i + 1;
|
|
|
|
|
while (j < nlen
|
|
|
|
|
&& (j >= olen
|
|
|
|
|
|| !GLYPH_EQUAL_P (nbody + j, obody + j)
|
|
|
|
|
|| CHAR_GLYPH_PADDING_P (nbody[j])))
|
|
|
|
|
++j;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* Output this run of non-matching chars. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
cursor_to (vpos, i);
|
|
|
|
|
write_glyphs (nbody + i, j - i);
|
|
|
|
|
i = j - 1;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* Now find the next non-match. */
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Clear the rest of the line, or the non-clear part of it. */
|
|
|
|
|
if (olen > nlen)
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nlen);
|
|
|
|
|
clear_end_of_line (olen);
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Make current row = desired row. */
|
|
|
|
|
make_current (desired_matrix, current_matrix, vpos);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Here when CHAR_INS_DEL_OK != 0, i.e. we can insert or delete
|
|
|
|
|
characters in a row. */
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
if (!olen)
|
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If current line is blank, skip over initial spaces, if
|
|
|
|
|
possible, and write the rest. */
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (write_spaces_p || desired_row->inverse_p)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
nsp = 0;
|
|
|
|
|
else
|
|
|
|
|
nsp = count_blanks (nbody, nlen);
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
if (nlen > nsp)
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nsp);
|
|
|
|
|
write_glyphs (nbody + nsp, nlen - nsp);
|
|
|
|
|
}
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* Exchange contents between current_frame and new_frame. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
make_current (desired_matrix, current_matrix, vpos);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Compute number of leading blanks in old and new contents. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
osp = count_blanks (obody, olen);
|
2000-10-31 13:32:29 +00:00
|
|
|
|
nsp = (desired_row->inverse_p || colored_spaces_p
|
|
|
|
|
? 0
|
|
|
|
|
: count_blanks (nbody, nlen));
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Compute number of matching chars starting with first non-blank. */
|
|
|
|
|
begmatch = count_match (obody + osp, obody + olen,
|
|
|
|
|
nbody + nsp, nbody + nlen);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
/* Spaces in new match implicit space past the end of old. */
|
|
|
|
|
/* A bug causing this to be a no-op was fixed in 18.29. */
|
2000-10-31 13:32:29 +00:00
|
|
|
|
if (!write_spaces_p && osp + begmatch == olen)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
np1 = nbody + nsp;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (np1 + begmatch < nend && CHAR_GLYPH_SPACE_P (np1[begmatch]))
|
|
|
|
|
++begmatch;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Avoid doing insert/delete char
|
|
|
|
|
just cause number of leading spaces differs
|
1999-07-21 21:43:52 +00:00
|
|
|
|
when the following text does not match. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
if (begmatch == 0 && osp != nsp)
|
|
|
|
|
osp = nsp = min (osp, nsp);
|
|
|
|
|
|
|
|
|
|
/* Find matching characters at end of line */
|
|
|
|
|
op1 = obody + olen;
|
|
|
|
|
np1 = nbody + nlen;
|
|
|
|
|
op2 = op1 + begmatch - min (olen - osp, nlen - nsp);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
while (op1 > op2
|
|
|
|
|
&& GLYPH_EQUAL_P (op1 - 1, np1 - 1))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
op1--;
|
|
|
|
|
np1--;
|
|
|
|
|
}
|
|
|
|
|
endmatch = obody + olen - op1;
|
|
|
|
|
|
|
|
|
|
/* tem gets the distance to insert or delete.
|
|
|
|
|
endmatch is how many characters we save by doing so.
|
|
|
|
|
Is it worth it? */
|
|
|
|
|
|
|
|
|
|
tem = (nlen - nsp) - (olen - osp);
|
|
|
|
|
if (endmatch && tem
|
1992-07-13 19:54:34 +00:00
|
|
|
|
&& (!char_ins_del_ok || endmatch <= char_ins_del_cost (frame)[tem]))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
endmatch = 0;
|
|
|
|
|
|
|
|
|
|
/* nsp - osp is the distance to insert or delete.
|
|
|
|
|
If that is nonzero, begmatch is known to be nonzero also.
|
|
|
|
|
begmatch + endmatch is how much we save by doing the ins/del.
|
|
|
|
|
Is it worth it? */
|
|
|
|
|
|
|
|
|
|
if (nsp != osp
|
|
|
|
|
&& (!char_ins_del_ok
|
1992-07-13 19:54:34 +00:00
|
|
|
|
|| begmatch + endmatch <= char_ins_del_cost (frame)[nsp - osp]))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
begmatch = 0;
|
|
|
|
|
endmatch = 0;
|
|
|
|
|
osp = nsp = min (osp, nsp);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Now go through the line, inserting, writing and
|
|
|
|
|
deleting as appropriate. */
|
|
|
|
|
|
|
|
|
|
if (osp > nsp)
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nsp);
|
|
|
|
|
delete_glyphs (osp - nsp);
|
|
|
|
|
}
|
|
|
|
|
else if (nsp > osp)
|
|
|
|
|
{
|
|
|
|
|
/* If going to delete chars later in line
|
|
|
|
|
and insert earlier in the line,
|
|
|
|
|
must delete first to avoid losing data in the insert */
|
|
|
|
|
if (endmatch && nlen < olen + nsp - osp)
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nlen - endmatch + osp - nsp);
|
|
|
|
|
delete_glyphs (olen + nsp - osp - nlen);
|
|
|
|
|
olen = nlen - (nsp - osp);
|
|
|
|
|
}
|
|
|
|
|
cursor_to (vpos, osp);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
insert_glyphs (0, nsp - osp);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
olen += nsp - osp;
|
|
|
|
|
|
|
|
|
|
tem = nsp + begmatch + endmatch;
|
|
|
|
|
if (nlen != tem || olen != tem)
|
|
|
|
|
{
|
|
|
|
|
cursor_to (vpos, nsp + begmatch);
|
|
|
|
|
if (!endmatch || nlen == olen)
|
|
|
|
|
{
|
|
|
|
|
/* If new text being written reaches right margin,
|
|
|
|
|
there is no need to do clear-to-eol at the end.
|
|
|
|
|
(and it would not be safe, since cursor is not
|
|
|
|
|
going to be "at the margin" after the text is done) */
|
1996-09-21 04:17:30 +00:00
|
|
|
|
if (nlen == FRAME_WINDOW_WIDTH (frame))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
olen = 0;
|
|
|
|
|
write_glyphs (nbody + nsp + begmatch, nlen - tem);
|
|
|
|
|
}
|
|
|
|
|
else if (nlen > olen)
|
|
|
|
|
{
|
1997-02-20 06:35:00 +00:00
|
|
|
|
/* Here, we used to have the following simple code:
|
|
|
|
|
----------------------------------------
|
|
|
|
|
write_glyphs (nbody + nsp + begmatch, olen - tem);
|
|
|
|
|
insert_glyphs (nbody + nsp + begmatch + olen - tem, nlen - olen);
|
|
|
|
|
----------------------------------------
|
|
|
|
|
but it doesn't work if nbody[nsp + begmatch + olen - tem]
|
|
|
|
|
is a padding glyph. */
|
|
|
|
|
int out = olen - tem; /* Columns to be overwritten originally. */
|
|
|
|
|
int del;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Calculate columns we can actually overwrite. */
|
|
|
|
|
while (CHAR_GLYPH_PADDING_P (nbody[nsp + begmatch + out])) out--;
|
|
|
|
|
write_glyphs (nbody + nsp + begmatch, out);
|
|
|
|
|
/* If we left columns to be overwritten, we must delete them. */
|
|
|
|
|
del = olen - tem - out;
|
|
|
|
|
if (del > 0) delete_glyphs (del);
|
|
|
|
|
/* At last, we insert columns not yet written out. */
|
|
|
|
|
insert_glyphs (nbody + nsp + begmatch + out, nlen - olen + del);
|
|
|
|
|
olen = nlen;
|
|
|
|
|
}
|
|
|
|
|
else if (olen > nlen)
|
|
|
|
|
{
|
|
|
|
|
write_glyphs (nbody + nsp + begmatch, nlen - tem);
|
|
|
|
|
delete_glyphs (olen - nlen);
|
|
|
|
|
olen = nlen;
|
|
|
|
|
}
|
1995-05-25 17:06:34 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
just_erase:
|
|
|
|
|
/* If any unerased characters remain after the new line, erase them. */
|
|
|
|
|
if (olen > nlen)
|
1994-06-06 05:03:41 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
cursor_to (vpos, nlen);
|
|
|
|
|
clear_end_of_line (olen);
|
1994-06-06 05:03:41 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Exchange contents between current_frame and new_frame. */
|
|
|
|
|
make_current (desired_matrix, current_matrix, vpos);
|
1994-06-06 05:03:41 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
1994-06-06 05:03:41 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/***********************************************************************
|
|
|
|
|
X/Y Position -> Buffer Position
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* Return the character position of the character at window relative
|
|
|
|
|
pixel position (*X, *Y). *X and *Y are adjusted to character
|
|
|
|
|
boundaries. */
|
|
|
|
|
|
|
|
|
|
int
|
|
|
|
|
buffer_posn_from_coords (w, x, y)
|
|
|
|
|
struct window *w;
|
|
|
|
|
int *x, *y;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct it it;
|
|
|
|
|
struct buffer *old_current_buffer = current_buffer;
|
|
|
|
|
struct text_pos startp;
|
|
|
|
|
int left_area_width;
|
|
|
|
|
|
|
|
|
|
current_buffer = XBUFFER (w->buffer);
|
|
|
|
|
SET_TEXT_POS_FROM_MARKER (startp, w->start);
|
|
|
|
|
CHARPOS (startp) = min (ZV, max (BEGV, CHARPOS (startp)));
|
|
|
|
|
BYTEPOS (startp) = min (ZV_BYTE, max (BEGV_BYTE, BYTEPOS (startp)));
|
|
|
|
|
start_display (&it, w, startp);
|
|
|
|
|
|
|
|
|
|
left_area_width = WINDOW_DISPLAY_LEFT_AREA_PIXEL_WIDTH (w);
|
|
|
|
|
move_it_to (&it, -1, *x + it.first_visible_x - left_area_width, *y, -1,
|
|
|
|
|
MOVE_TO_X | MOVE_TO_Y);
|
|
|
|
|
|
|
|
|
|
*x = it.current_x - it.first_visible_x + left_area_width;
|
|
|
|
|
*y = it.current_y;
|
|
|
|
|
current_buffer = old_current_buffer;
|
|
|
|
|
return IT_CHARPOS (it);
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Value is the string under window-relative coordinates X/Y in the
|
|
|
|
|
mode or top line of window W, or nil if none. MODE_LINE_P non-zero
|
|
|
|
|
means look at the mode line. *CHARPOS is set to the position in
|
|
|
|
|
the string returned. */
|
|
|
|
|
|
|
|
|
|
Lisp_Object
|
|
|
|
|
mode_line_string (w, x, y, mode_line_p, charpos)
|
|
|
|
|
struct window *w;
|
2000-08-09 12:29:43 +00:00
|
|
|
|
int x, y, mode_line_p;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int *charpos;
|
|
|
|
|
{
|
|
|
|
|
struct glyph_row *row;
|
|
|
|
|
struct glyph *glyph, *end;
|
|
|
|
|
struct frame *f = XFRAME (w->frame);
|
|
|
|
|
int x0;
|
|
|
|
|
Lisp_Object string = Qnil;
|
|
|
|
|
|
|
|
|
|
if (mode_line_p)
|
|
|
|
|
row = MATRIX_MODE_LINE_ROW (w->current_matrix);
|
|
|
|
|
else
|
1999-09-05 16:39:19 +00:00
|
|
|
|
row = MATRIX_HEADER_LINE_ROW (w->current_matrix);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
if (row->mode_line_p && row->enabled_p)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* The mode lines are displayed over scroll bars and bitmap
|
|
|
|
|
areas, and X is window-relative. Correct X by the scroll bar
|
|
|
|
|
and bitmap area width. */
|
|
|
|
|
if (FRAME_HAS_VERTICAL_SCROLL_BARS_ON_LEFT (f))
|
|
|
|
|
x += FRAME_SCROLL_BAR_COLS (f) * CANON_X_UNIT (f);
|
1999-08-30 23:14:23 +00:00
|
|
|
|
x += FRAME_LEFT_FLAGS_AREA_WIDTH (f);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Find the glyph under X. If we find one with a string object,
|
|
|
|
|
it's the one we were looking for. */
|
|
|
|
|
glyph = row->glyphs[TEXT_AREA];
|
|
|
|
|
end = glyph + row->used[TEXT_AREA];
|
|
|
|
|
for (x0 = 0; glyph < end; x0 += glyph->pixel_width, ++glyph)
|
|
|
|
|
if (x >= x0 && x < x0 + glyph->pixel_width)
|
|
|
|
|
{
|
|
|
|
|
string = glyph->object;
|
|
|
|
|
*charpos = glyph->charpos;
|
|
|
|
|
break;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
return string;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Changing Frame Sizes
|
|
|
|
|
***********************************************************************/
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
#ifdef SIGWINCH
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1992-01-14 08:05:08 +00:00
|
|
|
|
SIGTYPE
|
1995-02-14 14:24:43 +00:00
|
|
|
|
window_change_signal (signalnum) /* If we don't have an argument, */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
int signalnum; /* some compilers complain in signal calls. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
int width, height;
|
2000-08-22 22:14:00 +00:00
|
|
|
|
#ifndef USE_CRT_DLL
|
1991-07-03 12:10:07 +00:00
|
|
|
|
extern int errno;
|
2000-08-22 22:14:00 +00:00
|
|
|
|
#endif
|
1991-07-03 12:10:07 +00:00
|
|
|
|
int old_errno = errno;
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
get_frame_size (&width, &height);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* The frame size change obviously applies to a termcap-controlled
|
|
|
|
|
frame. Find such a frame in the list, and assume it's the only
|
1991-07-03 12:10:07 +00:00
|
|
|
|
one (since the redisplay code always writes to stdout, not a
|
1992-07-13 19:54:34 +00:00
|
|
|
|
FILE * specified in the frame structure). Record the new size,
|
1991-07-03 12:10:07 +00:00
|
|
|
|
but don't reallocate the data structures now. Let that be done
|
|
|
|
|
later outside of the signal handler. */
|
|
|
|
|
|
|
|
|
|
{
|
1993-03-18 22:50:00 +00:00
|
|
|
|
Lisp_Object tail, frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1993-03-18 22:50:00 +00:00
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1993-03-18 22:50:00 +00:00
|
|
|
|
if (FRAME_TERMCAP_P (XFRAME (frame)))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-08-21 19:29:02 +00:00
|
|
|
|
change_frame_size (XFRAME (frame), height, width, 0, 1, 0);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
signal (SIGWINCH, window_change_signal);
|
|
|
|
|
errno = old_errno;
|
|
|
|
|
}
|
|
|
|
|
#endif /* SIGWINCH */
|
|
|
|
|
|
|
|
|
|
|
1999-08-21 19:29:02 +00:00
|
|
|
|
/* Do any change in frame size that was requested by a signal. SAFE
|
|
|
|
|
non-zero means this function is called from a place where it is
|
|
|
|
|
safe to change frame sizes while a redisplay is in progress. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1999-08-21 19:29:02 +00:00
|
|
|
|
do_pending_window_change (safe)
|
|
|
|
|
int safe;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
/* If window_change_signal should have run before, run it now. */
|
1999-08-21 19:29:02 +00:00
|
|
|
|
if (redisplaying_p && !safe)
|
|
|
|
|
return;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
while (delayed_size_change)
|
|
|
|
|
{
|
1993-03-18 22:50:00 +00:00
|
|
|
|
Lisp_Object tail, frame;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
delayed_size_change = 0;
|
|
|
|
|
|
1993-03-18 22:50:00 +00:00
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
struct frame *f = XFRAME (frame);
|
1993-03-18 22:50:00 +00:00
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
int height = FRAME_NEW_HEIGHT (f);
|
|
|
|
|
int width = FRAME_NEW_WIDTH (f);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1993-06-03 06:18:18 +00:00
|
|
|
|
if (height != 0 || width != 0)
|
1999-08-21 19:29:02 +00:00
|
|
|
|
change_frame_size (f, height, width, 0, 0, safe);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* Change the frame height and/or width. Values may be given as zero to
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
indicate no change is to take place.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
If DELAY is non-zero, then assume we're being called from a signal
|
|
|
|
|
handler, and queue the change for later - perhaps the next
|
|
|
|
|
redisplay. Since this tries to resize windows, we can't call it
|
1999-08-21 19:29:02 +00:00
|
|
|
|
from a signal handler.
|
|
|
|
|
|
|
|
|
|
SAFE non-zero means this function is called from a place where it's
|
|
|
|
|
safe to change frame sizes while a redisplay is in progress. */
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1999-08-21 19:29:02 +00:00
|
|
|
|
change_frame_size (f, newheight, newwidth, pretend, delay, safe)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
register struct frame *f;
|
1999-08-21 19:29:02 +00:00
|
|
|
|
int newheight, newwidth, pretend, delay, safe;
|
1995-02-17 08:40:18 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame;
|
1997-01-19 01:34:39 +00:00
|
|
|
|
|
1996-06-10 21:13:26 +00:00
|
|
|
|
if (! FRAME_WINDOW_P (f))
|
1995-02-17 08:40:18 +00:00
|
|
|
|
{
|
1996-06-10 21:11:02 +00:00
|
|
|
|
/* When using termcap, or on MS-DOS, all frames use
|
|
|
|
|
the same screen, so a change in size affects all frames. */
|
1995-02-17 08:40:18 +00:00
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
1996-06-10 21:13:26 +00:00
|
|
|
|
if (! FRAME_WINDOW_P (XFRAME (frame)))
|
1995-02-17 08:40:18 +00:00
|
|
|
|
change_frame_size_1 (XFRAME (frame), newheight, newwidth,
|
1999-08-21 19:29:02 +00:00
|
|
|
|
pretend, delay, safe);
|
1995-02-17 08:40:18 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
1999-08-21 19:29:02 +00:00
|
|
|
|
change_frame_size_1 (f, newheight, newwidth, pretend, delay, safe);
|
1995-02-17 08:40:18 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
static void
|
1999-08-21 19:29:02 +00:00
|
|
|
|
change_frame_size_1 (f, newheight, newwidth, pretend, delay, safe)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
register struct frame *f;
|
1999-08-21 19:29:02 +00:00
|
|
|
|
int newheight, newwidth, pretend, delay, safe;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1996-09-21 01:21:18 +00:00
|
|
|
|
int new_frame_window_width;
|
1997-08-30 17:51:22 +00:00
|
|
|
|
int count = specpdl_ptr - specpdl;
|
1997-01-19 01:34:39 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* If we can't deal with the change now, queue it for later. */
|
1999-08-21 19:29:02 +00:00
|
|
|
|
if (delay || (redisplaying_p && !safe))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
FRAME_NEW_HEIGHT (f) = newheight;
|
|
|
|
|
FRAME_NEW_WIDTH (f) = newwidth;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
delayed_size_change = 1;
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* This size-change overrides any pending one for this frame. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
FRAME_NEW_HEIGHT (f) = 0;
|
|
|
|
|
FRAME_NEW_WIDTH (f) = 0;
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
|
1993-06-03 06:18:18 +00:00
|
|
|
|
/* If an argument is zero, set it to the current value. */
|
1996-08-22 08:23:30 +00:00
|
|
|
|
if (newheight == 0)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
newheight = FRAME_HEIGHT (f);
|
1996-08-22 08:23:30 +00:00
|
|
|
|
if (newwidth == 0)
|
1999-07-21 21:43:52 +00:00
|
|
|
|
newwidth = FRAME_WIDTH (f);
|
1997-01-19 01:34:39 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Compute width of windows in F.
|
|
|
|
|
This is the width of the frame without vertical scroll bars. */
|
|
|
|
|
new_frame_window_width = FRAME_WINDOW_WIDTH_ARG (f, newwidth);
|
1997-01-19 01:34:39 +00:00
|
|
|
|
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
/* Round up to the smallest acceptable size. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
check_frame_size (f, &newheight, &newwidth);
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
|
|
|
|
|
/* If we're not changing the frame size, quit now. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (newheight == FRAME_HEIGHT (f)
|
|
|
|
|
&& new_frame_window_width == FRAME_WINDOW_WIDTH (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return;
|
|
|
|
|
|
1996-04-26 15:45:46 +00:00
|
|
|
|
BLOCK_INPUT;
|
|
|
|
|
|
1996-01-24 22:33:54 +00:00
|
|
|
|
#ifdef MSDOS
|
|
|
|
|
/* We only can set screen dimensions to certain values supported
|
|
|
|
|
by our video hardware. Try to find the smallest size greater
|
|
|
|
|
or equal to the requested dimensions. */
|
|
|
|
|
dos_set_window_size (&newheight, &newwidth);
|
|
|
|
|
#endif
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (newheight != FRAME_HEIGHT (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (FRAME_HAS_MINIBUF_P (f) && !FRAME_MINIBUF_ONLY_P (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Frame has both root and mini-buffer. */
|
|
|
|
|
XSETFASTINT (XWINDOW (FRAME_ROOT_WINDOW (f))->top,
|
|
|
|
|
FRAME_TOP_MARGIN (f));
|
|
|
|
|
set_window_height (FRAME_ROOT_WINDOW (f),
|
|
|
|
|
(newheight
|
|
|
|
|
- 1
|
|
|
|
|
- FRAME_TOP_MARGIN (f)),
|
|
|
|
|
0);
|
|
|
|
|
XSETFASTINT (XWINDOW (FRAME_MINIBUF_WINDOW (f))->top,
|
1994-10-04 15:47:28 +00:00
|
|
|
|
newheight - 1);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
set_window_height (FRAME_MINIBUF_WINDOW (f), 1, 0);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
else
|
1992-07-13 19:54:34 +00:00
|
|
|
|
/* Frame has just one top-level window. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
set_window_height (FRAME_ROOT_WINDOW (f),
|
|
|
|
|
newheight - FRAME_TOP_MARGIN (f), 0);
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (FRAME_TERMCAP_P (f) && !pretend)
|
* dispnew.c (Fredraw_frame): Call clear_frame_records before
calling update_end, so that x_display_box_cursor can rely on the
contents of f->current_glyphs.
* dispnew.c (change_frame_size): Call check_frame_size here,
rather than writing out its code. Don't declare newheight and
newwidth to be register variables, since we take their address.
* dispnew.c (in_display): Variable deleted; it's only ever used as
an unofficial parameter to change_frame_size.
(change_frame_size): New argument, DELAY, which when non-zero
indicates to delay the size change until later. This should be
passed as one from signal handlers.
(window_change_signal): Call change_frame_size with a DELAY of 1.
(do_pending_window_change): Call change_frame_size with DELAY of 0.
* dispnew.c, frame.c, frame.h, keyboard.c, scroll.c, term.c,
window.c, xdisp.c, xfns.c xterm.c (FRAME_IS_TERMCAP, FRAME_IS_X,
FRAME_HAS_MINIBUF): Renamed these to FRAME_TERMCAP_P, FRAME_X_P,
and FRAME_HAS_MINIBUF_P, for consistency with the rest of the
frame macros.
1992-08-12 13:36:49 +00:00
|
|
|
|
FrameRows = newheight;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (new_frame_window_width != FRAME_WINDOW_WIDTH (f))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1999-07-21 21:43:52 +00:00
|
|
|
|
set_window_width (FRAME_ROOT_WINDOW (f), new_frame_window_width, 0);
|
|
|
|
|
if (FRAME_HAS_MINIBUF_P (f))
|
|
|
|
|
set_window_width (FRAME_MINIBUF_WINDOW (f), new_frame_window_width, 0);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
if (FRAME_TERMCAP_P (f) && !pretend)
|
1992-07-13 19:54:34 +00:00
|
|
|
|
FrameCols = newwidth;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1999-09-05 15:49:07 +00:00
|
|
|
|
if (WINDOWP (f->tool_bar_window))
|
|
|
|
|
XSETFASTINT (XWINDOW (f->tool_bar_window)->width, newwidth);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
FRAME_HEIGHT (f) = newheight;
|
|
|
|
|
SET_FRAME_WIDTH (f, newwidth);
|
1994-05-23 05:53:55 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
struct window *w = XWINDOW (FRAME_SELECTED_WINDOW (f));
|
|
|
|
|
int text_area_x, text_area_y, text_area_width, text_area_height;
|
|
|
|
|
|
|
|
|
|
window_box (w, TEXT_AREA, &text_area_x, &text_area_y, &text_area_width,
|
|
|
|
|
&text_area_height);
|
|
|
|
|
if (w->cursor.x >= text_area_x + text_area_width)
|
|
|
|
|
w->cursor.hpos = w->cursor.x = 0;
|
|
|
|
|
if (w->cursor.y >= text_area_y + text_area_height)
|
|
|
|
|
w->cursor.vpos = w->cursor.y = 0;
|
|
|
|
|
}
|
1994-05-23 05:53:55 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
adjust_glyphs (f);
|
|
|
|
|
SET_FRAME_GARBAGED (f);
|
|
|
|
|
calculate_costs (f);
|
1996-04-23 20:32:58 +00:00
|
|
|
|
|
|
|
|
|
UNBLOCK_INPUT;
|
1997-04-02 03:45:07 +00:00
|
|
|
|
|
1997-08-30 17:51:22 +00:00
|
|
|
|
record_unwind_protect (Fset_buffer, Fcurrent_buffer ());
|
|
|
|
|
|
1997-04-02 03:45:07 +00:00
|
|
|
|
/* This isn't quite a no-op: it runs window-configuration-change-hook. */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
Fset_window_buffer (FRAME_SELECTED_WINDOW (f),
|
|
|
|
|
XWINDOW (FRAME_SELECTED_WINDOW (f))->buffer);
|
1997-08-30 17:51:22 +00:00
|
|
|
|
|
|
|
|
|
unbind_to (count, Qnil);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/***********************************************************************
|
|
|
|
|
Terminal Related Lisp Functions
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
DEFUN ("open-termscript", Fopen_termscript, Sopen_termscript,
|
|
|
|
|
1, 1, "FOpen termscript file: ",
|
|
|
|
|
"Start writing all terminal output to FILE as well as the terminal.\n\
|
|
|
|
|
FILE = nil means just close any termscript file currently open.")
|
|
|
|
|
(file)
|
|
|
|
|
Lisp_Object file;
|
|
|
|
|
{
|
|
|
|
|
if (termscript != 0) fclose (termscript);
|
|
|
|
|
termscript = 0;
|
|
|
|
|
|
|
|
|
|
if (! NILP (file))
|
|
|
|
|
{
|
|
|
|
|
file = Fexpand_file_name (file, Qnil);
|
|
|
|
|
termscript = fopen (XSTRING (file)->data, "w");
|
|
|
|
|
if (termscript == 0)
|
|
|
|
|
report_file_error ("Opening termscript", Fcons (file, Qnil));
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFUN ("send-string-to-terminal", Fsend_string_to_terminal,
|
|
|
|
|
Ssend_string_to_terminal, 1, 1, 0,
|
|
|
|
|
"Send STRING to the terminal without alteration.\n\
|
|
|
|
|
Control characters in STRING will have terminal-dependent effects.")
|
1996-01-09 00:31:17 +00:00
|
|
|
|
(string)
|
|
|
|
|
Lisp_Object string;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1998-01-09 23:07:47 +00:00
|
|
|
|
/* ??? Perhaps we should do something special for multibyte strings here. */
|
1996-01-09 00:31:17 +00:00
|
|
|
|
CHECK_STRING (string, 0);
|
1998-03-21 07:06:14 +00:00
|
|
|
|
fwrite (XSTRING (string)->data, 1, STRING_BYTES (XSTRING (string)), stdout);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
fflush (stdout);
|
|
|
|
|
if (termscript)
|
|
|
|
|
{
|
1998-03-21 07:06:14 +00:00
|
|
|
|
fwrite (XSTRING (string)->data, 1, STRING_BYTES (XSTRING (string)),
|
|
|
|
|
termscript);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
fflush (termscript);
|
|
|
|
|
}
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFUN ("ding", Fding, Sding, 0, 1, 0,
|
|
|
|
|
"Beep, or flash the screen.\n\
|
|
|
|
|
Also, unless an argument is given,\n\
|
|
|
|
|
terminate any keyboard macro currently executing.")
|
|
|
|
|
(arg)
|
|
|
|
|
Lisp_Object arg;
|
|
|
|
|
{
|
1992-01-14 08:05:08 +00:00
|
|
|
|
if (!NILP (arg))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1992-05-19 02:05:47 +00:00
|
|
|
|
if (noninteractive)
|
|
|
|
|
putchar (07);
|
|
|
|
|
else
|
|
|
|
|
ring_bell ();
|
1991-07-03 12:10:07 +00:00
|
|
|
|
fflush (stdout);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
bitch_at_user ();
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1991-07-03 12:10:07 +00:00
|
|
|
|
bitch_at_user ()
|
|
|
|
|
{
|
|
|
|
|
if (noninteractive)
|
|
|
|
|
putchar (07);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
else if (!INTERACTIVE) /* Stop executing a keyboard macro. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
error ("Keyboard macro terminated by a command ringing the bell");
|
|
|
|
|
else
|
|
|
|
|
ring_bell ();
|
|
|
|
|
fflush (stdout);
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Sleeping, Waiting
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFUN ("sleep-for", Fsleep_for, Ssleep_for, 1, 2, 0,
|
1993-03-20 23:03:35 +00:00
|
|
|
|
"Pause, without updating display, for SECONDS seconds.\n\
|
1993-05-04 02:28:10 +00:00
|
|
|
|
SECONDS may be a floating-point value, meaning that you can wait for a\n\
|
|
|
|
|
fraction of a second. Optional second arg MILLISECONDS specifies an\n\
|
|
|
|
|
additional wait period, in milliseconds; this may be useful if your\n\
|
|
|
|
|
Emacs was built without floating point support.\n\
|
|
|
|
|
\(Not all operating systems support waiting for a fraction of a second.)")
|
1993-03-20 23:03:35 +00:00
|
|
|
|
(seconds, milliseconds)
|
|
|
|
|
Lisp_Object seconds, milliseconds;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
1993-03-20 23:03:35 +00:00
|
|
|
|
int sec, usec;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1993-03-20 23:03:35 +00:00
|
|
|
|
if (NILP (milliseconds))
|
1994-10-04 12:03:19 +00:00
|
|
|
|
XSETINT (milliseconds, 0);
|
1993-03-20 23:03:35 +00:00
|
|
|
|
else
|
|
|
|
|
CHECK_NUMBER (milliseconds, 1);
|
1993-05-04 02:28:10 +00:00
|
|
|
|
usec = XINT (milliseconds) * 1000;
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
double duration = extract_float (seconds);
|
|
|
|
|
sec = (int) duration;
|
|
|
|
|
usec += (duration - sec) * 1000000;
|
|
|
|
|
}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-02-20 04:15:20 +00:00
|
|
|
|
#ifndef EMACS_HAS_USECS
|
1993-03-20 23:03:35 +00:00
|
|
|
|
if (sec == 0 && usec != 0)
|
|
|
|
|
error ("millisecond `sleep-for' not supported on %s", SYSTEM_TYPE);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#endif
|
1993-03-20 23:03:35 +00:00
|
|
|
|
|
|
|
|
|
/* Assure that 0 <= usec < 1000000. */
|
|
|
|
|
if (usec < 0)
|
|
|
|
|
{
|
|
|
|
|
/* We can't rely on the rounding being correct if user is negative. */
|
|
|
|
|
if (-1000000 < usec)
|
|
|
|
|
sec--, usec += 1000000;
|
|
|
|
|
else
|
|
|
|
|
sec -= -usec / 1000000, usec = 1000000 - (-usec % 1000000);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1993-03-20 23:03:35 +00:00
|
|
|
|
else
|
|
|
|
|
sec += usec / 1000000, usec %= 1000000;
|
|
|
|
|
|
1996-02-22 05:50:00 +00:00
|
|
|
|
if (sec < 0 || (sec == 0 && usec == 0))
|
1993-03-20 23:03:35 +00:00
|
|
|
|
return Qnil;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-05-19 05:22:52 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object zero;
|
|
|
|
|
|
1994-10-04 15:47:28 +00:00
|
|
|
|
XSETFASTINT (zero, 0);
|
1992-05-19 05:22:52 +00:00
|
|
|
|
wait_reading_process_input (sec, usec, zero, 0);
|
|
|
|
|
}
|
1992-03-18 19:09:07 +00:00
|
|
|
|
|
1993-03-20 23:03:35 +00:00
|
|
|
|
/* We should always have wait_reading_process_input; we have a dummy
|
|
|
|
|
implementation for systems which don't support subprocesses. */
|
|
|
|
|
#if 0
|
|
|
|
|
/* No wait_reading_process_input */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
immediate_quit = 1;
|
|
|
|
|
QUIT;
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
|
|
|
|
sys_sleep (sec);
|
|
|
|
|
#else /* not VMS */
|
|
|
|
|
/* The reason this is done this way
|
|
|
|
|
(rather than defined (H_S) && defined (H_T))
|
|
|
|
|
is because the VMS preprocessor doesn't grok `defined' */
|
|
|
|
|
#ifdef HAVE_SELECT
|
1992-02-20 04:15:20 +00:00
|
|
|
|
EMACS_GET_TIME (end_time);
|
|
|
|
|
EMACS_SET_SECS_USECS (timeout, sec, usec);
|
1992-03-18 19:09:07 +00:00
|
|
|
|
EMACS_ADD_TIME (end_time, end_time, timeout);
|
1992-02-20 04:15:20 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
while (1)
|
|
|
|
|
{
|
1992-02-20 04:15:20 +00:00
|
|
|
|
EMACS_GET_TIME (timeout);
|
|
|
|
|
EMACS_SUB_TIME (timeout, end_time, timeout);
|
|
|
|
|
if (EMACS_TIME_NEG_P (timeout)
|
|
|
|
|
|| !select (1, 0, 0, 0, &timeout))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
#else /* not HAVE_SELECT */
|
|
|
|
|
sleep (sec);
|
|
|
|
|
#endif /* HAVE_SELECT */
|
|
|
|
|
#endif /* not VMS */
|
|
|
|
|
|
|
|
|
|
immediate_quit = 0;
|
|
|
|
|
#endif /* no subprocesses */
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1992-05-19 05:22:52 +00:00
|
|
|
|
/* This is just like wait_reading_process_input, except that
|
|
|
|
|
it does the redisplay.
|
|
|
|
|
|
1993-12-23 00:22:10 +00:00
|
|
|
|
It's also much like Fsit_for, except that it can be used for
|
1995-04-13 07:47:27 +00:00
|
|
|
|
waiting for input as well. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-05-19 05:22:52 +00:00
|
|
|
|
Lisp_Object
|
1997-05-27 01:54:42 +00:00
|
|
|
|
sit_for (sec, usec, reading, display, initial_display)
|
|
|
|
|
int sec, usec, reading, display, initial_display;
|
1992-05-19 05:22:52 +00:00
|
|
|
|
{
|
|
|
|
|
Lisp_Object read_kbd;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1996-04-09 02:41:15 +00:00
|
|
|
|
swallow_events (display);
|
|
|
|
|
|
1996-03-05 17:26:19 +00:00
|
|
|
|
if (detect_input_pending_run_timers (display))
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return Qnil;
|
|
|
|
|
|
1997-05-27 01:54:42 +00:00
|
|
|
|
if (initial_display)
|
1992-05-19 05:22:52 +00:00
|
|
|
|
redisplay_preserve_echo_area ();
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1992-06-02 05:18:06 +00:00
|
|
|
|
if (sec == 0 && usec == 0)
|
|
|
|
|
return Qt;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#ifdef SIGIO
|
1993-02-22 14:38:10 +00:00
|
|
|
|
gobble_input (0);
|
1992-05-19 05:22:52 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
1994-10-04 12:03:19 +00:00
|
|
|
|
XSETINT (read_kbd, reading ? -1 : 1);
|
1992-05-19 05:22:52 +00:00
|
|
|
|
wait_reading_process_input (sec, usec, read_kbd, display);
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return detect_input_pending () ? Qnil : Qt;
|
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1992-05-19 05:22:52 +00:00
|
|
|
|
DEFUN ("sit-for", Fsit_for, Ssit_for, 1, 3, 0,
|
1993-03-20 23:03:35 +00:00
|
|
|
|
"Perform redisplay, then wait for SECONDS seconds or until input is available.\n\
|
1993-05-04 02:28:10 +00:00
|
|
|
|
SECONDS may be a floating-point value, meaning that you can wait for a\n\
|
|
|
|
|
fraction of a second. Optional second arg MILLISECONDS specifies an\n\
|
|
|
|
|
additional wait period, in milliseconds; this may be useful if your\n\
|
|
|
|
|
Emacs was built without floating point support.\n\
|
|
|
|
|
\(Not all operating systems support waiting for a fraction of a second.)\n\
|
1996-01-09 00:31:17 +00:00
|
|
|
|
Optional third arg NODISP non-nil means don't redisplay, just wait for input.\n\
|
1992-05-19 05:22:52 +00:00
|
|
|
|
Redisplay is preempted as always if input arrives, and does not happen\n\
|
|
|
|
|
if input is available before it starts.\n\
|
|
|
|
|
Value is t if waited the full time with no input arriving.")
|
1993-03-20 23:03:35 +00:00
|
|
|
|
(seconds, milliseconds, nodisp)
|
|
|
|
|
Lisp_Object seconds, milliseconds, nodisp;
|
1992-05-19 05:22:52 +00:00
|
|
|
|
{
|
1993-03-20 23:03:35 +00:00
|
|
|
|
int sec, usec;
|
1992-05-19 05:22:52 +00:00
|
|
|
|
|
1993-03-20 23:03:35 +00:00
|
|
|
|
if (NILP (milliseconds))
|
1994-10-04 12:03:19 +00:00
|
|
|
|
XSETINT (milliseconds, 0);
|
1993-03-20 23:03:35 +00:00
|
|
|
|
else
|
|
|
|
|
CHECK_NUMBER (milliseconds, 1);
|
1993-05-04 02:28:10 +00:00
|
|
|
|
usec = XINT (milliseconds) * 1000;
|
|
|
|
|
|
|
|
|
|
{
|
|
|
|
|
double duration = extract_float (seconds);
|
|
|
|
|
sec = (int) duration;
|
|
|
|
|
usec += (duration - sec) * 1000000;
|
|
|
|
|
}
|
1992-05-19 05:22:52 +00:00
|
|
|
|
|
|
|
|
|
#ifndef EMACS_HAS_USECS
|
1993-03-20 23:03:35 +00:00
|
|
|
|
if (usec != 0 && sec == 0)
|
|
|
|
|
error ("millisecond `sit-for' not supported on %s", SYSTEM_TYPE);
|
1992-05-19 05:22:52 +00:00
|
|
|
|
#endif
|
|
|
|
|
|
1997-05-27 01:54:42 +00:00
|
|
|
|
return sit_for (sec, usec, 0, NILP (nodisp), NILP (nodisp));
|
1992-05-19 05:22:52 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Other Lisp Functions
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
|
|
|
|
/* A vector of size >= 2 * NFRAMES + 3 * NBUFFERS + 1, containing the
|
|
|
|
|
session's frames, frame names, buffers, buffer-read-only flags, and
|
|
|
|
|
buffer-modified-flags, and a trailing sentinel (so we don't need to
|
|
|
|
|
add length checks). */
|
|
|
|
|
|
|
|
|
|
static Lisp_Object frame_and_buffer_state;
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
DEFUN ("frame-or-buffer-changed-p", Fframe_or_buffer_changed_p,
|
|
|
|
|
Sframe_or_buffer_changed_p, 0, 0, 0,
|
|
|
|
|
"Return non-nil if the frame and buffer state appears to have changed.\n\
|
|
|
|
|
The state variable is an internal vector containing all frames and buffers,\n\
|
|
|
|
|
aside from buffers whose names start with space,\n\
|
|
|
|
|
along with the buffers' read-only and modified flags, which allows a fast\n\
|
|
|
|
|
check to see whether the menu bars might need to be recomputed.\n\
|
|
|
|
|
If this function returns non-nil, it updates the internal vector to reflect\n\
|
|
|
|
|
the current state.\n")
|
|
|
|
|
()
|
|
|
|
|
{
|
|
|
|
|
Lisp_Object tail, frame, buf;
|
|
|
|
|
Lisp_Object *vecp;
|
|
|
|
|
int n;
|
|
|
|
|
|
|
|
|
|
vecp = XVECTOR (frame_and_buffer_state)->contents;
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
{
|
|
|
|
|
if (!EQ (*vecp++, frame))
|
|
|
|
|
goto changed;
|
|
|
|
|
if (!EQ (*vecp++, XFRAME (frame)->name))
|
|
|
|
|
goto changed;
|
|
|
|
|
}
|
|
|
|
|
/* Check that the buffer info matches.
|
|
|
|
|
No need to test for the end of the vector
|
|
|
|
|
because the last element of the vector is lambda
|
|
|
|
|
and that will always cause a mismatch. */
|
1999-10-25 04:58:18 +00:00
|
|
|
|
for (tail = Vbuffer_alist; CONSP (tail); tail = XCDR (tail))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-10-25 04:58:18 +00:00
|
|
|
|
buf = XCDR (XCAR (tail));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Ignore buffers that aren't included in buffer lists. */
|
|
|
|
|
if (XSTRING (XBUFFER (buf)->name)->data[0] == ' ')
|
|
|
|
|
continue;
|
|
|
|
|
if (!EQ (*vecp++, buf))
|
|
|
|
|
goto changed;
|
|
|
|
|
if (!EQ (*vecp++, XBUFFER (buf)->read_only))
|
|
|
|
|
goto changed;
|
|
|
|
|
if (!EQ (*vecp++, Fbuffer_modified_p (buf)))
|
|
|
|
|
goto changed;
|
|
|
|
|
}
|
|
|
|
|
/* Detect deletion of a buffer at the end of the list. */
|
|
|
|
|
if (EQ (*vecp, Qlambda))
|
|
|
|
|
return Qnil;
|
|
|
|
|
changed:
|
|
|
|
|
/* Start with 1 so there is room for at least one lambda at the end. */
|
|
|
|
|
n = 1;
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
n += 2;
|
1999-10-25 04:58:18 +00:00
|
|
|
|
for (tail = Vbuffer_alist; CONSP (tail); tail = XCDR (tail))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
n += 3;
|
|
|
|
|
/* Reallocate the vector if it's grown, or if it's shrunk a lot. */
|
|
|
|
|
if (n > XVECTOR (frame_and_buffer_state)->size
|
|
|
|
|
|| n + 20 < XVECTOR (frame_and_buffer_state)->size / 2)
|
|
|
|
|
/* Add 20 extra so we grow it less often. */
|
|
|
|
|
frame_and_buffer_state = Fmake_vector (make_number (n + 20), Qlambda);
|
|
|
|
|
vecp = XVECTOR (frame_and_buffer_state)->contents;
|
|
|
|
|
FOR_EACH_FRAME (tail, frame)
|
|
|
|
|
{
|
|
|
|
|
*vecp++ = frame;
|
|
|
|
|
*vecp++ = XFRAME (frame)->name;
|
|
|
|
|
}
|
1999-10-25 04:58:18 +00:00
|
|
|
|
for (tail = Vbuffer_alist; CONSP (tail); tail = XCDR (tail))
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
1999-10-25 04:58:18 +00:00
|
|
|
|
buf = XCDR (XCAR (tail));
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Ignore buffers that aren't included in buffer lists. */
|
|
|
|
|
if (XSTRING (XBUFFER (buf)->name)->data[0] == ' ')
|
|
|
|
|
continue;
|
|
|
|
|
*vecp++ = buf;
|
|
|
|
|
*vecp++ = XBUFFER (buf)->read_only;
|
|
|
|
|
*vecp++ = Fbuffer_modified_p (buf);
|
|
|
|
|
}
|
|
|
|
|
/* Fill up the vector with lambdas (always at least one). */
|
|
|
|
|
*vecp++ = Qlambda;
|
|
|
|
|
while (vecp - XVECTOR (frame_and_buffer_state)->contents
|
|
|
|
|
< XVECTOR (frame_and_buffer_state)->size)
|
|
|
|
|
*vecp++ = Qlambda;
|
|
|
|
|
/* Make sure we didn't overflow the vector. */
|
|
|
|
|
if (vecp - XVECTOR (frame_and_buffer_state)->contents
|
|
|
|
|
> XVECTOR (frame_and_buffer_state)->size)
|
|
|
|
|
abort ();
|
|
|
|
|
return Qt;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/***********************************************************************
|
|
|
|
|
Initialization
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
char *terminal_type;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Initialization done when Emacs fork is started, before doing stty.
|
|
|
|
|
Determine terminal type and set terminal_driver. Then invoke its
|
|
|
|
|
decoding routine to set up variables in the terminal package. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1991-07-03 12:10:07 +00:00
|
|
|
|
init_display ()
|
|
|
|
|
{
|
|
|
|
|
#ifdef HAVE_X_WINDOWS
|
|
|
|
|
extern int display_arg;
|
|
|
|
|
#endif
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* Construct the space glyph. */
|
|
|
|
|
space_glyph.type = CHAR_GLYPH;
|
|
|
|
|
SET_CHAR_GLYPH_FROM_GLYPH (space_glyph, ' ');
|
|
|
|
|
space_glyph.charpos = -1;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
meta_key = 0;
|
|
|
|
|
inverse_video = 0;
|
|
|
|
|
cursor_in_echo_area = 0;
|
|
|
|
|
terminal_type = (char *) 0;
|
|
|
|
|
|
1993-03-24 01:51:35 +00:00
|
|
|
|
/* Now is the time to initialize this; it's used by init_sys_modes
|
|
|
|
|
during startup. */
|
|
|
|
|
Vwindow_system = Qnil;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1993-03-24 01:51:35 +00:00
|
|
|
|
/* If the user wants to use a window system, we shouldn't bother
|
|
|
|
|
initializing the terminal. This is especially important when the
|
|
|
|
|
terminal is so dumb that emacs gives up before and doesn't bother
|
|
|
|
|
using the window system.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1996-01-09 23:38:07 +00:00
|
|
|
|
If the DISPLAY environment variable is set and nonempty,
|
|
|
|
|
try to use X, and die with an error message if that doesn't work. */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
#ifdef HAVE_X_WINDOWS
|
1993-03-25 02:54:23 +00:00
|
|
|
|
if (! display_arg)
|
|
|
|
|
{
|
1996-01-09 23:38:07 +00:00
|
|
|
|
char *display;
|
1993-03-25 02:54:23 +00:00
|
|
|
|
#ifdef VMS
|
1996-01-09 23:38:07 +00:00
|
|
|
|
display = getenv ("DECW$DISPLAY");
|
1993-03-25 02:54:23 +00:00
|
|
|
|
#else
|
1996-01-09 23:38:07 +00:00
|
|
|
|
display = getenv ("DISPLAY");
|
1993-03-25 02:54:23 +00:00
|
|
|
|
#endif
|
1996-01-09 23:38:07 +00:00
|
|
|
|
|
|
|
|
|
display_arg = (display != 0 && *display != 0);
|
1993-03-25 04:08:11 +00:00
|
|
|
|
}
|
1993-03-25 02:54:23 +00:00
|
|
|
|
|
1997-07-13 23:20:41 +00:00
|
|
|
|
if (!inhibit_window_system && display_arg
|
|
|
|
|
#ifndef CANNOT_DUMP
|
|
|
|
|
&& initialized
|
|
|
|
|
#endif
|
|
|
|
|
)
|
1991-07-03 12:10:07 +00:00
|
|
|
|
{
|
|
|
|
|
Vwindow_system = intern ("x");
|
|
|
|
|
#ifdef HAVE_X11
|
|
|
|
|
Vwindow_system_version = make_number (11);
|
|
|
|
|
#else
|
|
|
|
|
Vwindow_system_version = make_number (10);
|
1996-05-21 17:32:00 +00:00
|
|
|
|
#endif
|
|
|
|
|
#if defined (LINUX) && defined (HAVE_LIBNCURSES)
|
|
|
|
|
/* In some versions of ncurses,
|
1996-05-23 01:34:07 +00:00
|
|
|
|
tputs crashes if we have not called tgetent.
|
1996-05-21 17:32:00 +00:00
|
|
|
|
So call tgetent. */
|
|
|
|
|
{ char b[2044]; tgetent (b, "xterm");}
|
1991-07-03 12:10:07 +00:00
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
adjust_frame_glyphs_initially ();
|
1991-07-03 12:10:07 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif /* HAVE_X_WINDOWS */
|
|
|
|
|
|
1995-11-07 07:14:59 +00:00
|
|
|
|
#ifdef HAVE_NTGUI
|
|
|
|
|
if (!inhibit_window_system)
|
|
|
|
|
{
|
1996-11-19 06:49:05 +00:00
|
|
|
|
Vwindow_system = intern ("w32");
|
1995-11-07 07:14:59 +00:00
|
|
|
|
Vwindow_system_version = make_number (1);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
adjust_frame_glyphs_initially ();
|
1995-11-07 07:14:59 +00:00
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif /* HAVE_NTGUI */
|
|
|
|
|
|
Initial check-in: changes for building Emacs under Mac OS.
2000-10-23 Andrew Choi <akochoi@i-cable.com>
* dispextern.h [macintosh]: Include macgui.h instead of macterm.h.
* dispnew.c [macintosh]: Include macterm.h.
(init_display) [macintosh]: initialization for window system.
* emacs.c (main) [macintosh]: Call syms_of_textprop,
syms_of_macfns, syms_of_ccl, syms_of_fontset, syms_of_xterm,
syms_of_search, x_term_init, and init_keyboard before calling
init_window_once. Also, call syms_of_xmenu.
* fontset.c (syms_of_fontset) [macintosh]: Set ASCII font of
default fontset to Monaco.
* frame.c [macintosh]: Include macterm.h. Remove declarations of
NewMacWindow and DisposeMacWindow.
(make_terminal_frame) [macintosh]: Call make_mac_terminal_frame
instead of calling NewMacWindow and setting fields of
f->output_data.mac directly. Call init_frame_faces.
(Fdelete_frame) [macintosh]: Remove unused code.
(Fmodify_frame_parameters) [macintosh]: Call
x_set_frame_parameters instead of mac_set_frame_parameters.
* frame.h [macintosh]: Define menu_bar_lines field in struct
frame. Define FRAME_EXTERNAL_MENU_BAR macro.
* keyboard.c [macintosh]: Include macterm.h.
(kbd_buffer_get_event) [macintosh]: Generate delete_window_event
and menu_bar_activate_event type events as for X and NT.
(make_lispy_event) [macintosh]: Construct lisp events of type
MENU_BAR_EVENT as for X and NT.
* sysdep.c [macintosh]: Remove declaration for sys_signal.
Include stdlib.h. Remove definition of Vx_bitmap_file_path.
(sys_subshell) [macintosh]: Remove definition entirely.
(init_sys_modes) [macintosh]: Do not initialize Vwindow_system and
Vwindow_system_version here. Remove initialization of
Vx_bitmap_file_path.
(read_input_waiting): Correct the number of parameters passed to
read_socket_hook.
Move all Macintosh functions to mac/mac.c.
* term.c [macintosh]: Include macterm.h.
* window.c [macintosh]: Include macterm.h.
* xdisp.c [macintosh]: Include macterm.h. Declare
set_frame_menubar and pending_menu_activation.
(echo_area_display) [macintosh]: Do not return if terminal frame
is the selected frame.
(update_menu_bar) [macintosh]: Check FRAME_EXTERNAL_MENU_BAR (f).
Allow only the selected frame to set menu bar.
(redisplay_window) [macintosh]: Obtain menu bar to redisplay by
calling FRAME_EXTERNAL_MENU_BAR (f).
(display_menu_bar) [macintosh]: Check FRAME_MAC_P (f).
* xfaces.c [macintosh]: Include macterm.h. Define x_display_info
and check_x. Declare XCreateGC. Define x_create_gc and
x_free_gc. Initialize font_sort_order.
(x_face_list_fonts) [macintosh]: Use the same code as WINDOWSNT,
but call x_list_fonts instead of w32_list_fonts.
(Finternal_face_x_get_resource) [macintosh]: Do not call
display_x_get_resource.
(prepare_face_for_display) [macintosh]: Set xgcv.font.
(realize_x_face) [macintosh]: Load the font if it is specified in
ATTRS.
(syms_of_xfaces) [macintosh]: Initialize Vscalable_fonts_allowed
to Qt.
* cus-edit.el (custom-button-face): Use 3D look for mac.
(custom-button-pressed-face): Likewise.
* faces.el (set-face-attributes-from-resources): Handle mac frames
in the same way as x and w32 frames.
(face-valid-attribute-values): Likewise.
(read-face-attribute): Likewise.
(defined-colors): Likewise.
(color-defined-p): Likewise.
(color-values): Likewise.
(display-grayscale-p): Likewise.
(face-set-after-frame-default): Likewise.
(mode-line): Same default face as for x and w32.
(tool-bar): Likewise.
* frame.el: Remove call to frame-notice-user-settings at end of
the file.
* info.el (Info-fontify-node): make underlines invisible for mac
as for x, pc, and w32 frame types.
* term/mac-win.el: New file.
2000-10-22 16:50:16 +00:00
|
|
|
|
#ifdef macintosh
|
|
|
|
|
if (!inhibit_window_system)
|
|
|
|
|
{
|
|
|
|
|
Vwindow_system = intern ("mac");
|
|
|
|
|
Vwindow_system_version = make_number (1);
|
|
|
|
|
adjust_frame_glyphs_initially ();
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
#endif /* macintosh */
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* If no window system has been specified, try to use the terminal. */
|
|
|
|
|
if (! isatty (0))
|
|
|
|
|
{
|
1997-01-20 02:22:25 +00:00
|
|
|
|
fatal ("standard input is not a tty");
|
1991-07-03 12:10:07 +00:00
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* Look at the TERM variable */
|
|
|
|
|
terminal_type = (char *) getenv ("TERM");
|
|
|
|
|
if (!terminal_type)
|
|
|
|
|
{
|
|
|
|
|
#ifdef VMS
|
|
|
|
|
fprintf (stderr, "Please specify your terminal type.\n\
|
|
|
|
|
For types defined in VMS, use set term /device=TYPE.\n\
|
|
|
|
|
For types not defined in VMS, use define emacs_term \"TYPE\".\n\
|
|
|
|
|
\(The quotation marks are necessary since terminal types are lower case.)\n");
|
|
|
|
|
#else
|
|
|
|
|
fprintf (stderr, "Please set the environment variable TERM; see tset(1).\n");
|
|
|
|
|
#endif
|
|
|
|
|
exit (1);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#ifdef VMS
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* VMS DCL tends to up-case things, so down-case term type.
|
1991-07-03 12:10:07 +00:00
|
|
|
|
Hardly any uppercase letters in terminal types; should be none. */
|
|
|
|
|
{
|
|
|
|
|
char *new = (char *) xmalloc (strlen (terminal_type) + 1);
|
|
|
|
|
char *p;
|
|
|
|
|
|
|
|
|
|
strcpy (new, terminal_type);
|
|
|
|
|
|
|
|
|
|
for (p = new; *p; p++)
|
|
|
|
|
if (isupper (*p))
|
|
|
|
|
*p = tolower (*p);
|
|
|
|
|
|
|
|
|
|
terminal_type = new;
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
#endif /* VMS */
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
term_init (terminal_type);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1997-01-20 01:21:07 +00:00
|
|
|
|
{
|
1999-09-13 11:13:24 +00:00
|
|
|
|
struct frame *sf = SELECTED_FRAME ();
|
|
|
|
|
int width = FRAME_WINDOW_WIDTH (sf);
|
|
|
|
|
int height = FRAME_HEIGHT (sf);
|
1997-01-20 01:21:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
unsigned int total_glyphs = height * (width + 2) * sizeof (struct glyph);
|
1997-01-20 01:21:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/* If these sizes are so big they cause overflow, just ignore the
|
|
|
|
|
change. It's not clear what better we could do. */
|
|
|
|
|
if (total_glyphs / sizeof (struct glyph) / height != width + 2)
|
1997-01-20 02:22:25 +00:00
|
|
|
|
fatal ("screen size %dx%d too big", width, height);
|
1997-01-20 01:21:07 +00:00
|
|
|
|
}
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
adjust_frame_glyphs_initially ();
|
1999-09-13 11:13:24 +00:00
|
|
|
|
calculate_costs (XFRAME (selected_frame));
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
|
|
|
|
#ifdef SIGWINCH
|
|
|
|
|
#ifndef CANNOT_DUMP
|
|
|
|
|
if (initialized)
|
|
|
|
|
#endif /* CANNOT_DUMP */
|
|
|
|
|
signal (SIGWINCH, window_change_signal);
|
|
|
|
|
#endif /* SIGWINCH */
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
/* Set up faces of the initial terminal frame of a dumped Emacs. */
|
|
|
|
|
if (initialized
|
|
|
|
|
&& !noninteractive
|
1999-07-30 08:38:10 +00:00
|
|
|
|
#ifdef MSDOS
|
|
|
|
|
/* The MSDOS terminal turns on its ``window system'' relatively
|
|
|
|
|
late into the startup, so we cannot do the frame faces'
|
|
|
|
|
initialization just yet. It will be done later by pc-win.el
|
|
|
|
|
and internal_terminal_init. */
|
|
|
|
|
&& (strcmp (terminal_type, "internal") != 0 || inhibit_window_system)
|
|
|
|
|
#endif
|
1999-07-21 21:43:52 +00:00
|
|
|
|
&& NILP (Vwindow_system))
|
Changes for automatic remapping of X colors on terminal frames:
* xfaces.c (XColor) [!HAVE_X_WINDOWS]: Provide a typedef for non-X
frames.
(Vface_tty_color_alist): Remove.
(tty_defined_color): New function.
(defined_color): Rewrite to support any type of frame.
(tty_color_name): New function.
(face_color_supported_p, Fface_color_gray_p,
Fface_color_supported_p): Support non-X frames.
(load_color): Enclose the color name in quotes, in the log
messages. Remove DOS-specific version of load_color.
(realize_tty_face): Take the supported colors from
tty-color-alist. Support translation of X colors to the closest
tty color, for both MSDOS and tty frames.
[MSDOS]: Don't invert face colors if they were taken from the
frame colors.
(Fface_register_tty_color, Fface_clear_tty_colors): Remove.
* frame.h (struct x_output) [!MSDOS, !WINDOWSNT, !HAVE_X_WINDOWS]:
Define a mostly empty surrogate.
(tty_display): Declare.
* frame.c (make_terminal_frame) [!macintosh]: Don't use
tty_display.
(Fframe_parameters): Don't invert colors of non-FRAME_WINDOW_P
frames when the frame's param_alist includes 'reverse.
(tty_display): Define.
(make_terminal_frame) [!MSDOS]: Assign &tty_display to the
output_data.x member.
(Fframe_parameters): Return foreground and background color names
on tty frames as well, in addition to MSDOS frames.
* msdos.h (DisplayWidth, DisplayHeight): Changes for Lisp_Object
selected_frame.
(struct x_output): Remove unused members; document who uses each
member.
(FRAME_PARAM_FACES, FRAME_N_PARAM_FACES, FRAME_DEFAULT_PARAM_FACE,
FRAME_MODE_LINE_PARAM_FACE, FRAME_COMPUTED_FACES,
FRAME_N_COMPUTED_FACES, FRAME_SIZE_COMPUTED_FACES,
FRAME_DEFAULT_FACE, FRAME_MODE_LINE_FACE, unload_color): Remove
unused macro definintions.
* msdos.c (IT_set_frame_parameters): Don't call
recompute_basic_faces, the next redisplay will, anyway.
(x_current_display): Remove unused variable.
Many functions: changes for Lisp_object selected_frame.
(IT_set_face): If the tty_reverse_p flag is set for the face,
reverse the foreground and background colors.
(Fmsdos_remember_default_colors): New function.
(syms_of_msdos): Defsubr it.
(IT_set_frame_parameters): Use initial_screen_colors[] when
creating a new frame. If the frame parameters include 'reverse,
swap the foreground and background colors.
(internal_terminal_init): Initialize initial_screen_colors to -1.
(syms_of_msdos): Add DEFVAR_BOOL for x-stretch-cursor, to shut up
cus-start.el.
* Makefile.in (lisp, shortlisp): Add lisp/term/tty-colors.elc.
* xfns.c (x_defined_color): Rename from defined_color. All
callers changed.
(Fxw_color_defined_p): Renamed from Fx_color_defined_p;
all callers changed.
(Fxw_color_values): Renamed from Fx_color_values; all callers
changed.
(Fxw_display_color_p): Renamed from Fx_display_color_p; all
callers changed.
(x_window_to_frame, x_any_window_to_frame,
x_non_menubar_window_to_frame, x_menubar_window_to_frame,
x_top_window_to_frame): Use !FRAME_X_P instead of
f->output_data.nothing.
* xterm.h (x_defined_color): Rename from defined_color.
* w32fns.c (x_window_to_frame): Use FRAME_W32_P instead of
f->output_data.nothing.
(Fxw_color_defined_p): Renamed from Fx_color_defined_p;
all callers changed.
(Fxw_color_values): Renamed from Fx_color_values; all callers
changed.
(Fxw_display_color_p): Renamed from Fx_display_color_p; all
callers changed.
* dispextern.h (tty_color_name): Add prototype.
* xmenu.c (menubar_id_to_frame): Use FRAME_WINDOW_P instead of
f->output_data.nothing.
* w32menu.c (menubar_id_to_frame): Likewise.
* w32term.h (w32_output): Declare.
* dosfns.c (Qmsdos_color_translate): Remove.
(msdos_stdcolor_name): Now returns a Lisp_Object.
* dosfns.h (Qmsdos_color_translate): Remove.
* s/msdos.h (INTERNAL_TERMINAL): Add entries for color support.
1999-12-06 16:54:09 +00:00
|
|
|
|
{
|
|
|
|
|
/* For the initial frame, we don't have any way of knowing what
|
|
|
|
|
are the foreground and background colors of the terminal. */
|
|
|
|
|
struct frame *sf = SELECTED_FRAME();
|
|
|
|
|
|
Changes for separate unspecified foreground and background colors
on character terminals:
* dispextern.h (FACE_TTY_DEFAULT_FG_COLOR)
(FACE_TTY_DEFAULT_BG_COLOR): New macros.
* xfaces.c (Qunspecified_fg, Qunspecified_bg): New variables.
(syms_of_xfaces): Initialize and staticpro them.
(tty_defined_color): If the color name is unspecified-fg or
unspecified-bg, return FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR, respectively, as the pixel value.
(tty_color_name): If the color pixel value is either
FACE_TTY_DEFAULT_FG_COLOR or FACE_TTY_DEFAULT_BG_COLOR, return
Qunspecified_fg or Qunspecified_bg, respectively.
(Finternal_set_lisp_face_attribute): Allow values Qunspecified_fg
and Qunspecified_bg for foreground and background colors.
(realize_default_face): If the foreground and background colors
are not specified, default to Qunspecified_fg and Qunspecified_bg.
(realize_tty_face): By default, set the face colors to
FACE_TTY_DEFAULT_FG_COLOR and FACE_TTY_DEFAULT_BG_COLOR.
[MSDOS]: Handle FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR when face colors are not defined.
Reverse the colors if the default colors were reversed.
* dispnew.c (init_display): Initialize the frame pixels of the
initial frame to FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR.
* term.c (turn_on_face): If the default fore- and background
colors are reversed, enter inverse video mode. Don't send color
escape sequences for unspecified foreground and background colors.
(turn_off_face): Handle unspecified-fg and unspecified-bg colors.
* dosfns.c (unspecified_colors): New variable.
(msdos_stdcolor_idx): Handle unspecified-fg and unspecified-bg
color names, return FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR, respectively.
(msdos_stdcolor_name): Handle FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR, return Qunspecified_fg and
Qunspecified_bg, respectively.
* msdos.c (IT_set_face): Support FACE_TTY_DEFAULT_FG_COLOR and
FACE_TTY_DEFAULT_BG_COLOR as pixel values.
* faces.el (face-read-integer, read-face-attribute)
(color-defined-p, color-values): Allow color values unspecified-fg
and unspecified-bg, handle them as unspecified.
1999-12-15 13:14:38 +00:00
|
|
|
|
FRAME_FOREGROUND_PIXEL (sf) = FACE_TTY_DEFAULT_FG_COLOR;
|
|
|
|
|
FRAME_BACKGROUND_PIXEL (sf) = FACE_TTY_DEFAULT_BG_COLOR;
|
Changes for automatic remapping of X colors on terminal frames:
* xfaces.c (XColor) [!HAVE_X_WINDOWS]: Provide a typedef for non-X
frames.
(Vface_tty_color_alist): Remove.
(tty_defined_color): New function.
(defined_color): Rewrite to support any type of frame.
(tty_color_name): New function.
(face_color_supported_p, Fface_color_gray_p,
Fface_color_supported_p): Support non-X frames.
(load_color): Enclose the color name in quotes, in the log
messages. Remove DOS-specific version of load_color.
(realize_tty_face): Take the supported colors from
tty-color-alist. Support translation of X colors to the closest
tty color, for both MSDOS and tty frames.
[MSDOS]: Don't invert face colors if they were taken from the
frame colors.
(Fface_register_tty_color, Fface_clear_tty_colors): Remove.
* frame.h (struct x_output) [!MSDOS, !WINDOWSNT, !HAVE_X_WINDOWS]:
Define a mostly empty surrogate.
(tty_display): Declare.
* frame.c (make_terminal_frame) [!macintosh]: Don't use
tty_display.
(Fframe_parameters): Don't invert colors of non-FRAME_WINDOW_P
frames when the frame's param_alist includes 'reverse.
(tty_display): Define.
(make_terminal_frame) [!MSDOS]: Assign &tty_display to the
output_data.x member.
(Fframe_parameters): Return foreground and background color names
on tty frames as well, in addition to MSDOS frames.
* msdos.h (DisplayWidth, DisplayHeight): Changes for Lisp_Object
selected_frame.
(struct x_output): Remove unused members; document who uses each
member.
(FRAME_PARAM_FACES, FRAME_N_PARAM_FACES, FRAME_DEFAULT_PARAM_FACE,
FRAME_MODE_LINE_PARAM_FACE, FRAME_COMPUTED_FACES,
FRAME_N_COMPUTED_FACES, FRAME_SIZE_COMPUTED_FACES,
FRAME_DEFAULT_FACE, FRAME_MODE_LINE_FACE, unload_color): Remove
unused macro definintions.
* msdos.c (IT_set_frame_parameters): Don't call
recompute_basic_faces, the next redisplay will, anyway.
(x_current_display): Remove unused variable.
Many functions: changes for Lisp_object selected_frame.
(IT_set_face): If the tty_reverse_p flag is set for the face,
reverse the foreground and background colors.
(Fmsdos_remember_default_colors): New function.
(syms_of_msdos): Defsubr it.
(IT_set_frame_parameters): Use initial_screen_colors[] when
creating a new frame. If the frame parameters include 'reverse,
swap the foreground and background colors.
(internal_terminal_init): Initialize initial_screen_colors to -1.
(syms_of_msdos): Add DEFVAR_BOOL for x-stretch-cursor, to shut up
cus-start.el.
* Makefile.in (lisp, shortlisp): Add lisp/term/tty-colors.elc.
* xfns.c (x_defined_color): Rename from defined_color. All
callers changed.
(Fxw_color_defined_p): Renamed from Fx_color_defined_p;
all callers changed.
(Fxw_color_values): Renamed from Fx_color_values; all callers
changed.
(Fxw_display_color_p): Renamed from Fx_display_color_p; all
callers changed.
(x_window_to_frame, x_any_window_to_frame,
x_non_menubar_window_to_frame, x_menubar_window_to_frame,
x_top_window_to_frame): Use !FRAME_X_P instead of
f->output_data.nothing.
* xterm.h (x_defined_color): Rename from defined_color.
* w32fns.c (x_window_to_frame): Use FRAME_W32_P instead of
f->output_data.nothing.
(Fxw_color_defined_p): Renamed from Fx_color_defined_p;
all callers changed.
(Fxw_color_values): Renamed from Fx_color_values; all callers
changed.
(Fxw_display_color_p): Renamed from Fx_display_color_p; all
callers changed.
* dispextern.h (tty_color_name): Add prototype.
* xmenu.c (menubar_id_to_frame): Use FRAME_WINDOW_P instead of
f->output_data.nothing.
* w32menu.c (menubar_id_to_frame): Likewise.
* w32term.h (w32_output): Declare.
* dosfns.c (Qmsdos_color_translate): Remove.
(msdos_stdcolor_name): Now returns a Lisp_Object.
* dosfns.h (Qmsdos_color_translate): Remove.
* s/msdos.h (INTERNAL_TERMINAL): Add entries for color support.
1999-12-06 16:54:09 +00:00
|
|
|
|
call0 (intern ("tty-set-up-initial-frame-faces"));
|
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
/***********************************************************************
|
|
|
|
|
Blinking cursor
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
1999-11-01 14:43:24 +00:00
|
|
|
|
DEFUN ("internal-show-cursor", Finternal_show_cursor,
|
1999-11-03 15:57:55 +00:00
|
|
|
|
Sinternal_show_cursor, 2, 2, 0,
|
|
|
|
|
"Set the cursor-visibility flag of WINDOW to SHOW.\n\
|
|
|
|
|
WINDOW nil means use the selected window. SHOW non-nil means\n\
|
|
|
|
|
show a cursor in WINDOW in the next redisplay. SHOW nil means\n\
|
|
|
|
|
don't show a cursor.")
|
|
|
|
|
(window, show)
|
|
|
|
|
Lisp_Object window, show;
|
1999-07-21 21:43:52 +00:00
|
|
|
|
{
|
|
|
|
|
/* Don't change cursor state while redisplaying. This could confuse
|
|
|
|
|
output routines. */
|
|
|
|
|
if (!redisplaying_p)
|
|
|
|
|
{
|
|
|
|
|
if (NILP (window))
|
|
|
|
|
window = selected_window;
|
|
|
|
|
else
|
|
|
|
|
CHECK_WINDOW (window, 2);
|
|
|
|
|
|
1999-11-03 15:57:55 +00:00
|
|
|
|
XWINDOW (window)->cursor_off_p = NILP (show);
|
1999-07-21 21:43:52 +00:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return Qnil;
|
1991-07-03 12:10:07 +00:00
|
|
|
|
}
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
|
|
|
|
|
1999-11-03 15:57:55 +00:00
|
|
|
|
DEFUN ("internal-show-cursor-p", Finternal_show_cursor_p,
|
|
|
|
|
Sinternal_show_cursor_p, 0, 1, 0,
|
1999-11-18 18:03:57 +00:00
|
|
|
|
"Value is non-nil if next redisplay will display a cursor in WINDOW.\n\
|
1999-11-03 15:57:55 +00:00
|
|
|
|
WINDOW nil or omitted means report on the selected window.")
|
|
|
|
|
(window)
|
|
|
|
|
Lisp_Object window;
|
|
|
|
|
{
|
|
|
|
|
struct window *w;
|
|
|
|
|
|
|
|
|
|
if (NILP (window))
|
|
|
|
|
window = selected_window;
|
|
|
|
|
else
|
|
|
|
|
CHECK_WINDOW (window, 2);
|
|
|
|
|
|
|
|
|
|
w = XWINDOW (window);
|
|
|
|
|
return w->cursor_off_p ? Qnil : Qt;
|
|
|
|
|
}
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
/***********************************************************************
|
|
|
|
|
Initialization
|
|
|
|
|
***********************************************************************/
|
|
|
|
|
|
1998-04-14 12:25:56 +00:00
|
|
|
|
void
|
1991-07-03 12:10:07 +00:00
|
|
|
|
syms_of_display ()
|
|
|
|
|
{
|
1992-07-13 19:54:34 +00:00
|
|
|
|
defsubr (&Sredraw_frame);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
defsubr (&Sredraw_display);
|
1994-06-06 05:03:41 +00:00
|
|
|
|
defsubr (&Sframe_or_buffer_changed_p);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
defsubr (&Sopen_termscript);
|
|
|
|
|
defsubr (&Sding);
|
|
|
|
|
defsubr (&Ssit_for);
|
|
|
|
|
defsubr (&Ssleep_for);
|
|
|
|
|
defsubr (&Ssend_string_to_terminal);
|
1999-11-01 14:43:24 +00:00
|
|
|
|
defsubr (&Sinternal_show_cursor);
|
1999-11-03 15:57:55 +00:00
|
|
|
|
defsubr (&Sinternal_show_cursor_p);
|
1991-07-03 12:10:07 +00:00
|
|
|
|
|
1995-06-11 20:14:36 +00:00
|
|
|
|
frame_and_buffer_state = Fmake_vector (make_number (20), Qlambda);
|
1994-06-06 05:03:41 +00:00
|
|
|
|
staticpro (&frame_and_buffer_state);
|
|
|
|
|
|
1995-10-14 05:54:16 +00:00
|
|
|
|
Qdisplay_table = intern ("display-table");
|
|
|
|
|
staticpro (&Qdisplay_table);
|
2000-08-09 18:39:58 +00:00
|
|
|
|
Qredisplay_dont_pause = intern ("redisplay-dont-pause");
|
|
|
|
|
staticpro (&Qredisplay_dont_pause);
|
1995-10-14 05:54:16 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_INT ("baud-rate", &baud_rate,
|
1994-06-16 15:14:30 +00:00
|
|
|
|
"*The output baud rate of the terminal.\n\
|
1991-07-03 12:10:07 +00:00
|
|
|
|
On most systems, changing this value will affect the amount of padding\n\
|
|
|
|
|
and the other strategic decisions made during redisplay.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_BOOL ("inverse-video", &inverse_video,
|
1992-07-13 19:54:34 +00:00
|
|
|
|
"*Non-nil means invert the entire frame display.\n\
|
1991-07-03 12:10:07 +00:00
|
|
|
|
This means everything is in inverse video which otherwise would not be.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_BOOL ("visible-bell", &visible_bell,
|
1992-07-13 19:54:34 +00:00
|
|
|
|
"*Non-nil means try to flash the frame to represent a bell.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_BOOL ("no-redraw-on-reenter", &no_redraw_on_reenter,
|
1992-07-13 19:54:34 +00:00
|
|
|
|
"*Non-nil means no need to redraw entire frame after suspending.\n\
|
1991-07-03 12:10:07 +00:00
|
|
|
|
A non-nil value is useful if the terminal can automatically preserve\n\
|
1992-07-13 19:54:34 +00:00
|
|
|
|
Emacs's frame display when you reenter Emacs.\n\
|
1991-07-03 12:10:07 +00:00
|
|
|
|
It is up to you to set this variable if your terminal can do that.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_LISP ("window-system", &Vwindow_system,
|
|
|
|
|
"A symbol naming the window-system under which Emacs is running\n\
|
|
|
|
|
\(such as `x'), or nil if emacs is running on an ordinary terminal.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_LISP ("window-system-version", &Vwindow_system_version,
|
|
|
|
|
"The version number of the window system in use.\n\
|
|
|
|
|
For X windows, this is 10 or 11.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_BOOL ("cursor-in-echo-area", &cursor_in_echo_area,
|
|
|
|
|
"Non-nil means put cursor in minibuffer, at end of any message there.");
|
1999-07-21 21:43:52 +00:00
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
DEFVAR_LISP ("glyph-table", &Vglyph_table,
|
1992-07-13 19:54:34 +00:00
|
|
|
|
"Table defining how to output a glyph code to the frame.\n\
|
1991-07-03 12:10:07 +00:00
|
|
|
|
If not nil, this is a vector indexed by glyph code to define the glyph.\n\
|
|
|
|
|
Each element can be:\n\
|
|
|
|
|
integer: a glyph code which this glyph is an alias for.\n\
|
|
|
|
|
string: output this glyph using that string (not impl. in X windows).\n\
|
|
|
|
|
nil: this glyph mod 256 is char code to output,\n\
|
1994-04-13 20:44:38 +00:00
|
|
|
|
and this glyph / 256 is face code for X windows (see `face-id').");
|
1991-07-03 12:10:07 +00:00
|
|
|
|
Vglyph_table = Qnil;
|
|
|
|
|
|
|
|
|
|
DEFVAR_LISP ("standard-display-table", &Vstandard_display_table,
|
|
|
|
|
"Display table to use for buffers that specify none.\n\
|
|
|
|
|
See `buffer-display-table' for more information.");
|
|
|
|
|
Vstandard_display_table = Qnil;
|
|
|
|
|
|
1999-07-21 21:43:52 +00:00
|
|
|
|
DEFVAR_BOOL ("redisplay-dont-pause", &redisplay_dont_pause,
|
|
|
|
|
"*Non-nil means update isn't paused when input is detected.");
|
|
|
|
|
redisplay_dont_pause = 0;
|
|
|
|
|
|
1991-07-03 12:10:07 +00:00
|
|
|
|
/* Initialize `window-system', unless init_display already decided it. */
|
|
|
|
|
#ifdef CANNOT_DUMP
|
|
|
|
|
if (noninteractive)
|
|
|
|
|
#endif
|
|
|
|
|
{
|
|
|
|
|
Vwindow_system = Qnil;
|
|
|
|
|
Vwindow_system_version = Qnil;
|
|
|
|
|
}
|
|
|
|
|
}
|