/* Display generation from window structure and buffer text. Copyright (C) 1994, 1995, 1996 Board of Trustees, University of Illinois. Copyright (C) 1995 Free Software Foundation, Inc. Copyright (C) 1995, 1996, 2000, 2001, 2002, 2003, 2005, 2010 Ben Wing. Copyright (C) 1995 Sun Microsystems, Inc. Copyright (C) 1996 Chuck Thompson. This file is part of XEmacs. XEmacs is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. XEmacs 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 XEmacs. If not, see . */ /* Synched up with: Not in FSF. */ /* Author: Chuck Thompson */ /* Fixed up by Ben Wing for Mule */ /* This file has been Mule-ized. */ /***************************************************************************** The Golden Rules of Redisplay First: It Is Better To Be Correct Than Fast Second: Thou Shalt Use Due Caution When Running Elisp From Within Redisplay Third: It Is Better To Be Fast Than Not To Be ****************************************************************************/ /* Note: The second rule used to prohibit running Elisp from within redisplay, but that's not correct any more -- use call*_trapping_problems() or call_with_suspended_errors() instead. --ben */ #include #include "lisp.h" #include "buffer.h" #include "commands.h" #include "debug.h" #include "device-impl.h" #include "elhash.h" #include "events.h" #include "extents-impl.h" #include "faces.h" #include "frame-impl.h" #include "glyphs.h" #include "gutter.h" #include "insdel.h" #include "menubar.h" #include "fontcolor-impl.h" #include "opaque.h" #include "process.h" #include "profile.h" #include "redisplay.h" #include "toolbar.h" #include "window-impl.h" #include "line-number.h" #include "file-coding.h" #include "sysfile.h" #ifdef HAVE_TTY #include "console-tty.h" #endif /* HAVE_TTY */ /* Note: We have to be careful throughout this code to properly handle and differentiate between Ibytes and Ichars. Since strings are generally composed of Ibytes, I've taken the tack that any contiguous set of Ibytes is called a "string", while any contiguous set of Ichars is called an "array". */ /* Return value to indicate a failure by an add_*_rune routine to add a rune, but no propagation information needs to be returned. */ #define ADD_FAILED (prop_block_dynarr *) 1 #define BEGIN_GLYPHS 0 #define END_GLYPHS 1 #define LEFT_GLYPHS 2 #define RIGHT_GLYPHS 3 #define VERTICAL_CLIP(w, display) \ ((WINDOW_TTY_P (w) | (!display && scroll_on_clipped_lines)) \ ? INT_MAX \ : vertical_clip) /* The following structures are completely private to redisplay.c so we put them here instead of in a header file, for modularity. */ /* NOTE: Bytexpos's not Charxpos's in this structure. */ typedef struct position_redisplay_data_type { /* This information is normally filled in by the create_*_block routines and is used by the add_*_rune routines. */ Lisp_Object window; /* if we are working with strings rather than buffers we need a handle to the string */ Lisp_Object string; struct device *d; struct display_block *db; struct display_line *dl; Ichar ch; /* Character that is to be added. This is used to communicate this information to add_ichar_rune(). */ Lisp_Object last_charset; /* The charset of the previous character. Used to optimize some lookups -- we only have to do some things when the charset changes. */ face_index last_findex; /* The face index of the previous character. Needed to ensure the validity of the last_charset optimization. */ int last_char_width; /* The width of the previous character. */ int font_is_bogus; /* If true, it means we couldn't instantiate the font for this charset, so we substitute ~'s from the ASCII charset. */ Bytexpos byte_charpos; /* Position of character we are processing. This is a Bytexpos, meaning it refers to bytes (not chars) and can refer to either buffers (1-based) or strings (0-based). We need to be careful when doing anything that references the text in the buffer or string. */ Bytexpos byte_endpos; /* ????? Unknown, under-used. */ int pixpos; int max_pixpos; int blank_width; /* Width of the blank that is to be added. This is used to communicate this information to add_blank_rune(). This is also used rather cheesily to communicate the width of the eol-cursor-size blank that exists at the end of the line. add_ichar_rune() is called cheesily with the non-printing char '\n', which is stuck in the output routines with its width being BLANK_WIDTH. */ Bytexpos byte_cursor_charpos; /* This stores the buffer position of the cursor. */ unsigned int cursor_type :3; int cursor_x; /* rune block cursor is at */ int start_col; /* Number of character columns (each column has a width of the default char width) that still need to be skipped. This is used for horizontal scrolling, where a certain number of columns (those off the left side of the screen) need to be skipped before anything is displayed. */ Bytexpos byte_start_col_enabled; int start_col_xoffset; /* Number of pixels that still need to be skipped. This is used for horizontal scrolling of glyphs, where we want to be able to scroll over part of the glyph. */ int hscroll_glyph_width_adjust; /* how much the width of the hscroll glyph differs from space_width (w). 0 if no hscroll glyph was used, i.e. the window is not scrolled horizontally. Used in tab calculations. */ /* Information about the face the text should be displayed in and any begin-glyphs and end-glyphs. */ struct extent_fragment *ef; face_index findex; /* The height of a pixmap may either be predetermined if the user has set a baseline value, or it may be dependent on whatever the line ascent and descent values end up being, based just on font and pixmap-ascent information. In the first case we can immediately update the values, thus their inclusion here. In the last case we cannot determine the actual contribution to the line height until we have finished laying out all text on the line. Thus we propagate the max height of such pixmaps and do a final calculation (in calculate_baseline()) after all text has been added to the line. */ int new_ascent; int new_descent; int max_pixmap_height; int need_baseline_computation; int end_glyph_width; /* Well, it is the kitchen sink after all ... */ Lisp_Object result_str; /* String where we put the result of generating a formatted string in the modeline. */ int is_modeline; /* Non-zero if we're generating the modeline. */ Charcount modeline_charpos; /* Number of chars used in result_str so far; corresponds to bytepos. */ Bytecount bytepos; /* Number of bytes used in result_str so far. We don't actually copy the bytes into result_str until the end because we don't know how big the string is going to be until then. */ } pos_data; enum prop_type { PROP_STRING, PROP_CHAR, PROP_MINIBUF_PROMPT, PROP_BLANK, PROP_GLYPH }; /* Data that should be propagated to the next line. Either a single Ichar, a string of Ibyte's or a glyph. The actual data that is propagated ends up as a Dynarr of these blocks. prop_blocks are used to indicate that data that was supposed to go on the previous line couldn't actually be displayed. Generally this shouldn't happen if we are clipping the end of lines. If we are wrapping then we need to display the propagation data before moving on. Its questionable whether we should wrap or clip glyphs in this instance. Most e-lisp relies on clipping so we preserve this behavior. #### It's unclean that both Ichars and Ibytes are here. */ typedef struct prop_block prop_block; struct prop_block { enum prop_type type; union data { struct { Ibyte *str; Bytecount len; /* length of the string. */ } p_string; struct { Ichar ch; Bytebpos byte_cursor_charpos; /* NOTE: is in Bytebpos's */ unsigned int cursor_type :3; } p_char; struct { int width; face_index findex; } p_blank; struct { /* Not used as yet, but could be used to wrap rather than clip glyphs. */ int width; Lisp_Object glyph; } p_glyph; struct { Lisp_Object preprompt; Lisp_Object prompt; } p_minibuf_prompt; } data; }; typedef struct { Dynarr_declare (prop_block); } prop_block_dynarr; static Charcount generate_fstring_runes (struct window *w, pos_data *data, Charcount pos, Charcount min_pos, Charcount max_pos, Lisp_Object elt, int depth, int max_pixsize, face_index findex, int type, Charcount *offset, Lisp_Object cur_ext); static prop_block_dynarr *add_glyph_rune (pos_data *data, struct glyph_block *gb, int pos_type, int allow_cursor, struct glyph_cachel *cachel); static Bytebpos create_text_block (struct window *w, struct display_line *dl, Bytebpos byte_start_pos, prop_block_dynarr **prop, int type); static int create_overlay_glyph_block (struct window *w, struct display_line *dl); static void create_left_glyph_block (struct window *w, struct display_line *dl, int overlay_width); static void create_right_glyph_block (struct window *w, struct display_line *dl); static void redisplay_windows (Lisp_Object window, int skip_selected); static void decode_mode_spec (struct window *w, Ichar spec, int type); static void free_display_line (struct display_line *dl); static void update_line_start_cache (struct window *w, Charbpos from, Charbpos to, Charbpos point, int no_regen); static int point_visible (struct window *w, Charbpos point, int type); static void calculate_yoffset (struct display_line *dl, struct display_block *fixup); static void calculate_baseline (pos_data *data); #ifdef ERROR_CHECK_DISPLAY static void sledgehammer_check_redisplay_structs (void); #endif /* ERROR_CHECK_DISPLAY */ /* This used to be 10 but 30 seems to give much better performance. */ #define INIT_MAX_PREEMPTS 30 static Fixnum max_preempts; #define QUEUED_EVENTS_REQUIRED_FOR_PREEMPTION 4 /* Note that doing this can call Lisp. */ #define REDISPLAY_PREEMPTION_CHECK \ ((void) \ (preempted = \ (!disable_preemption && \ ((preemption_count < max_preempts) || !NILP (Vexecuting_macro)) && \ (!INTERACTIVE || \ detect_input_pending (QUEUED_EVENTS_REQUIRED_FOR_PREEMPTION))))) /* * Redisplay global variables. */ /* We need a third set of display structures for the cursor motion routines. We used to just give each window a third set. However, we always fully regenerate the structures when needed so there isn't any reason we need more than a single set. */ display_line_dynarr *cmotion_display_lines; /* We store the extents that we need to generate in a Dynarr and then frob them all on at the end of generating the string. We do it this way rather than adding them as we generate the string because we don't store the text into the resulting string until we're done (to avoid having to resize the string multiple times), and we don't want to go around adding extents to a string when the extents might stretch off the end of the string. */ static EXTENT_dynarr *formatted_string_extent_dynarr; static Bytecount_dynarr *formatted_string_extent_start_dynarr; static Bytecount_dynarr *formatted_string_extent_end_dynarr; /* #### probably temporary */ Fixnum cache_adjustment; /* This holds a string representing the text corresponding to a single modeline % spec. */ static Ibyte_dynarr *mode_spec_ibyte_string; int in_display; /* 1 if in redisplay. */ /* Whether we should delay size changes. Broken out of enter_redisplay_critical_section(). */ int hold_frame_size_changes; int disable_preemption; /* Used for debugging redisplay and for force-redisplay. */ /* We only allow max_preempts preemptions before we force a redisplay. */ static int preemption_count; /* Minimum pixel height of clipped bottom display line. */ Fixnum vertical_clip; /* Minimum visible pixel width of clipped glyphs at right margin. */ Fixnum horizontal_clip; /* Nonzero means reading single-character input with prompt so put cursor on minibuffer after the prompt. */ int cursor_in_echo_area; Lisp_Object Qcursor_in_echo_area; /* Nonzero means truncate lines in all windows less wide than the frame */ int truncate_partial_width_windows; /* non-nil if a buffer has changed since the last time redisplay completed */ int buffers_changed; int buffers_changed_set; /* non-nil if hscroll has changed somewhere or a buffer has been narrowed or widened */ int clip_changed; int clip_changed_set; /* non-nil if any extent has changed since the last time redisplay completed */ int extents_changed; int extents_changed_set; /* non-nil if any face has changed since the last time redisplay completed */ int faces_changed; /* Nonzero means some frames have been marked as garbaged */ int frame_changed; /* non-zero if any of the builtin display glyphs (continuation, hscroll, control-arrow, etc) is in need of updating somewhere. */ int glyphs_changed; int glyphs_changed_set; /* non-zero if any subwindow has been deleted. */ int subwindows_changed; int subwindows_changed_set; /* non-zero if any displayed subwindow is in need of updating somewhere. */ int subwindows_state_changed; int subwindows_state_changed_set; /* This variable is 1 if the icon has to be updated. It is set to 1 when `frame-icon-glyph' changes. */ int icon_changed; int icon_changed_set; /* This variable is 1 if the menubar widget has to be updated. It is set to 1 by set-menubar-dirty-flag and cleared when the widget has been updated. */ int menubar_changed; int menubar_changed_set; /* true iff we should redraw the modelines on the next redisplay */ int modeline_changed; int modeline_changed_set; /* non-nil if point has changed in some buffer since the last time redisplay completed */ int point_changed; int point_changed_set; /* non-nil if some frame has changed its size */ int size_changed; /* non-nil if some device has signaled that it wants to change size */ int asynch_device_change_pending; /* non-nil if any toolbar has changed */ int toolbar_changed; int toolbar_changed_set; /* Nonzero if some frame has changed the layout of internal elements (gutters or toolbars). */ int frame_layout_changed; /* non-nil if any gutter has changed */ int gutter_changed; int gutter_changed_set; /* non-nil if any window has changed since the last time redisplay completed */ int windows_changed; /* non-nil if any frame's window structure has changed since the last time redisplay completed */ int windows_structure_changed; /* If non-nil, use vertical bar cursor. */ Lisp_Object Vbar_cursor; Lisp_Object Qbar_cursor; Lisp_Object Vvisible_bell; /* If true and the terminal will support it then the frame will flash instead of beeping when an error occurs */ /* Nonzero means no need to redraw the entire frame on resuming a suspended Emacs. This is useful on terminals with multiple pages, where one page is used for Emacs and another for all else. */ int no_redraw_on_reenter; Lisp_Object Vwindow_system; /* #### this variable is deprecated nil or a symbol naming the window system under which emacs is running (`x', `gtk', `mswindows', and `tty' are supported -- yes, TTYs are window systems for this purpose. */ Lisp_Object Vinitial_device_type; Lisp_Object Vglobal_mode_string; /* The number of lines scroll a window by when point leaves the window; if it is <=0 then point is centered in the window */ Fixnum scroll_step; /* Scroll up to this many lines, to bring point back on screen. */ Fixnum scroll_conservatively; /* Marker for where to display an arrow on top of the buffer text. */ Lisp_Object Voverlay_arrow_position; /* String to display for the arrow. */ Lisp_Object Voverlay_arrow_string; Lisp_Object Qbuffer_list_changed_hook, Vbuffer_list_changed_hook; static Fixnum last_display_warning_tick; static Fixnum display_warning_tick; Lisp_Object Qdisplay_warning_buffer; int inhibit_warning_display; Lisp_Object Vleft_margin_width, Vright_margin_width; Lisp_Object Vminimum_line_ascent, Vminimum_line_descent; Lisp_Object Vuse_left_overflow, Vuse_right_overflow; Lisp_Object Vtext_cursor_visible_p; static Lisp_Object QSin_redisplay; static Lisp_Object Vpost_redisplay_actions; int column_number_start_at_one; Lisp_Object Qtop_bottom; #define WINDOW_SCROLLED(w) ((w)->hscroll > 0 || (w)->left_xoffset) static const struct memory_description rune_dglyph_description_1[] = { { XD_LISP_OBJECT, offsetof (struct rune_dglyph, glyph) }, { XD_LISP_OBJECT, offsetof (struct rune_dglyph, extent) }, { XD_END } }; static const struct sized_memory_description rune_dglyph_description = { sizeof (struct rune_dglyph), rune_dglyph_description_1 }; static const struct memory_description rune_object_description_1[] = { { XD_BLOCK_ARRAY, RUNE_DGLYPH, 1, { &rune_dglyph_description } }, { XD_END } }; static const struct sized_memory_description rune_object_description = { 0, rune_object_description_1 }; static const struct memory_description rune_description_1[] = { { XD_INT, offsetof (rune, type) }, { XD_UNION, offsetof (rune, object), XD_INDIRECT (0, 0), { &rune_object_description } }, { XD_END } }; static const struct sized_memory_description rune_description = { sizeof (rune), rune_description_1 }; static const struct memory_description rune_dynarr_description_1[] = { XD_DYNARR_DESC (rune_dynarr, &rune_description), { XD_END } }; static const struct sized_memory_description rune_dynarr_description = { sizeof (rune_dynarr), rune_dynarr_description_1 }; static const struct memory_description display_block_description_1[] = { { XD_BLOCK_PTR, offsetof (display_block, runes), 1, { &rune_dynarr_description } }, { XD_END } }; static const struct sized_memory_description display_block_description = { sizeof (display_block), display_block_description_1 }; static const struct memory_description display_block_dynarr_description_1[] = { XD_DYNARR_DESC (display_block_dynarr, &display_block_description), { XD_END } }; static const struct sized_memory_description display_block_dynarr_description = { sizeof (display_block_dynarr), display_block_dynarr_description_1 }; static const struct memory_description glyph_block_description_1[] = { { XD_LISP_OBJECT, offsetof (glyph_block, glyph) }, { XD_LISP_OBJECT, offsetof (glyph_block, extent) }, { XD_END } }; static const struct sized_memory_description glyph_block_description = { sizeof (glyph_block), glyph_block_description_1 }; static const struct memory_description glyph_block_dynarr_description_1[] = { XD_DYNARR_DESC (glyph_block_dynarr, &glyph_block_description), { XD_END } }; static const struct sized_memory_description glyph_block_dynarr_description = { sizeof (glyph_block_dynarr), glyph_block_dynarr_description_1 }; static const struct memory_description display_line_description_1[] = { { XD_BLOCK_PTR, offsetof (display_line, display_blocks), 1, { &display_block_dynarr_description } }, { XD_BLOCK_PTR, offsetof (display_line, left_glyphs), 1, { &glyph_block_dynarr_description } }, { XD_BLOCK_PTR, offsetof (display_line, right_glyphs), 1, { &glyph_block_dynarr_description } }, { XD_END } }; static const struct sized_memory_description display_line_description = { sizeof (display_line), display_line_description_1 }; static const struct memory_description display_line_dynarr_description_1[] = { XD_DYNARR_DESC (display_line_dynarr, &display_line_description), { XD_END } }; const struct sized_memory_description display_line_dynarr_description = { sizeof (display_line_dynarr), display_line_dynarr_description_1 }; /***************************************************************************/ /* */ /* low-level interfaces onto device routines */ /* */ /***************************************************************************/ static int redisplay_window_text_width_ichar_string (struct window *w, int findex, Ichar *str, Charcount len) { unsigned char charsets[NUM_LEADING_BYTES]; Lisp_Object window; find_charsets_in_ichar_string (charsets, str, len); window = wrap_window (w); ensure_face_cachel_complete (WINDOW_FACE_CACHEL (w, findex), window, charsets); return DEVMETH (WINDOW_XDEVICE (w), text_width, (WINDOW_XFRAME (w), WINDOW_FACE_CACHEL (w, findex), str, len)); } static Ichar_dynarr *rtw_ichar_dynarr; static int redisplay_window_text_width_string (struct window *w, int findex, Ibyte *nonreloc, Lisp_Object reloc, Bytecount offset, Bytecount len) { if (!rtw_ichar_dynarr) rtw_ichar_dynarr = Dynarr_new (Ichar); Dynarr_reset (rtw_ichar_dynarr); fixup_internal_substring (nonreloc, reloc, offset, &len); if (STRINGP (reloc)) nonreloc = XSTRING_DATA (reloc); convert_ibyte_string_into_ichar_dynarr (nonreloc, len, rtw_ichar_dynarr); return redisplay_window_text_width_ichar_string (w, findex, Dynarr_begin (rtw_ichar_dynarr), Dynarr_length (rtw_ichar_dynarr)); } int redisplay_text_width_string (Lisp_Object domain, Lisp_Object face, Ibyte *nonreloc, Lisp_Object reloc, Bytecount offset, Bytecount len) { Lisp_Object window = DOMAIN_WINDOW (domain); Lisp_Object frame = DOMAIN_FRAME (domain); unsigned char charsets[NUM_LEADING_BYTES]; struct face_cachel cachel; if (!rtw_ichar_dynarr) rtw_ichar_dynarr = Dynarr_new (Ichar); Dynarr_reset (rtw_ichar_dynarr); fixup_internal_substring (nonreloc, reloc, offset, &len); if (STRINGP (reloc)) nonreloc = XSTRING_DATA (reloc); convert_ibyte_string_into_ichar_dynarr (nonreloc, len, rtw_ichar_dynarr); find_charsets_in_ibyte_string (charsets, nonreloc, len); reset_face_cachel (&cachel); cachel.face = face; ensure_face_cachel_complete (&cachel, NILP (window) ? frame : window, charsets); return DEVMETH (FRAME_XDEVICE (XFRAME (frame)), text_width, (XFRAME (frame), &cachel, Dynarr_begin (rtw_ichar_dynarr), Dynarr_length (rtw_ichar_dynarr))); } /* Return the display block from DL of the given TYPE. A display line can have only one display block of each possible type. If DL does not have a block of type TYPE, one will be created and added to DL. */ struct display_block * get_display_block_from_line (struct display_line *dl, enum display_type type) { int elt; struct display_block db; /* Check if this display line already has a block of the desired type and if so, return it. */ if (dl->display_blocks) { for (elt = 0; elt < Dynarr_length (dl->display_blocks); elt++) { if (Dynarr_at (dl->display_blocks, elt).type == type) return Dynarr_atp (dl->display_blocks, elt); } /* There isn't an active block of the desired type, but there might still be allocated blocks we need to reuse. */ if (elt < Dynarr_largest (dl->display_blocks)) { struct display_block *dbp = Dynarr_atp (dl->display_blocks, elt); /* "add" the block to the list */ Dynarr_incrementr (dl->display_blocks); /* initialize and return */ dbp->type = type; return dbp; } } else { /* This line doesn't have any display blocks, so initialize the display bock array. */ dl->display_blocks = Dynarr_new (display_block); } /* The line doesn't have a block of the desired type so go ahead and create one and add it to the line. */ xzero (db); db.type = type; db.runes = Dynarr_new (rune); Dynarr_add (dl->display_blocks, db); /* Return the newly added display block. */ elt = Dynarr_length (dl->display_blocks) - 1; return Dynarr_atp (dl->display_blocks, elt); } static int tab_char_width (struct window *w) { struct buffer *b = XBUFFER (w->buffer); int char_tab_width = XFIXNUM (b->tab_width); if (char_tab_width <= 0 || char_tab_width > 1000) char_tab_width = 8; return char_tab_width; } static int space_width (struct window *w) { /* While tabs are traditionally composed of spaces, for variable-width fonts the space character tends to give too narrow a value. So we use 'n' instead. Except that we don't. We use the default character width for the default face. If this is actually defined by the font then it is probably the best thing to actually use. If it isn't, we have assumed it is 'n' and have already calculated its width. Thus we can avoid a call to XTextWidth on X frames by just querying the default width. */ return XFONT_INSTANCE (WINDOW_FACE_CACHEL_FONT (w, DEFAULT_INDEX, Vcharset_ascii))->width; } static int tab_pix_width (struct window *w) { return space_width (w) * tab_char_width (w); } /* Given a pixel position in a window, return the pixel location of the next tabstop. Tabs are calculated from the left window edge in terms of spaces displayed in the default face. Formerly the space width was determined using the currently active face. That method leads to tabstops which do not line up. */ static int next_tab_position (struct window *w, int start_pixpos, int left_pixpos) { int n_pos = left_pixpos; int pix_tab_width = tab_pix_width (w); /* Adjust n_pos for any hscrolling which has happened. */ if (WINDOW_SCROLLED (w)) n_pos -= space_width (w) * (w->hscroll - 1) + w->left_xoffset; while (n_pos <= start_pixpos) n_pos += pix_tab_width; return n_pos; } /* For the given window, calculate the outside and margin boundaries for a display line. The whitespace boundaries must be calculated by the text layout routines. */ layout_bounds calculate_display_line_boundaries (struct window *w, int modeline) { layout_bounds bounds; /* Set the outermost boundaries which are the boundaries of the window itself minus the gutters (and minus the scrollbars if this is for the modeline). */ if (!modeline) { bounds.left_out = WINDOW_TEXT_LEFT (w); bounds.right_out = WINDOW_TEXT_RIGHT (w); } else { bounds.left_out = WINDOW_MODELINE_LEFT (w); bounds.right_out = WINDOW_MODELINE_RIGHT (w); } /* The inner boundaries mark where the glyph margins are located. */ bounds.left_in = bounds.left_out + window_left_margin_width (w); bounds.right_in = bounds.right_out - window_right_margin_width (w); /* We cannot fully calculate the whitespace boundaries as they depend on the contents of the line being displayed. */ bounds.left_white = bounds.left_in; bounds.right_white = bounds.right_in; return bounds; } /* This takes a display_block and its containing line and corrects the yoffset of each glyph in the block to cater for the ascent of the line as a whole. Must be called *after* the line-ascent is known! */ static void calculate_yoffset (struct display_line *dl, struct display_block *fixup) { int i; for (i=0; irunes); i++) { struct rune *r = Dynarr_atp (fixup->runes,i); if (r->type == RUNE_DGLYPH) { if (r->object.dglyph.ascent < dl->ascent) r->object.dglyph.yoffset = dl->ascent - r->object.dglyph.ascent + r->object.dglyph.descent; } } } /* Calculate the textual baseline (the ascent and descent values for the display_line as a whole). If the baseline is completely blank, or contains no manually positioned glyphs, then the textual baseline is simply the baseline of the default font. (The `contains no manually positioned glyphs' part is actually done for us by `add_ichar_rune'.) If the baseline contains pixmaps, and they're all manually positioned, then the textual baseline location is constrained that way, and we need do no work. If the baseline contains pixmaps, and at least one is automatically positioned, then the textual ascent is the largest ascent on the line, and the textual descent is the largest descent (which is how things are set up at entry to this function anyway): except that if the max_ascent + max_descent is too small for the height of the line (say you've adjusted the baseline of a short glyph, and there's a tall one next to it), then take the ascent and descent for the line individually from the largest of the explicitly set ascent/descent, and the rescaled ascent/descent of the default font, scaled such that the largest glyph will fit. This means that if you have a short glyph (but taller than the default font's descent) forced right under the baseline, and a really tall automatically positioned glyph, that the descent for the line is just big enough for the manually positioned short glyph, and the tall one uses as much of that space as the default font would were it as tall as the tall glyph; but that the ascent is big enough for the tall glyph to fit. This behaviour means that under no circumstances will changing the baseline of a short glyph cause a tall glyph to move around; nor will it move the textual baseline more than necessary. (Changing a tall glyph's baseline might move the text's baseline arbitrarily, of course.) */ static void calculate_baseline (pos_data *data) { /* Blank line: baseline is default font's baseline. */ if (!data->new_ascent && !data->new_descent) { /* We've got a blank line so initialize these values from the default face. */ default_face_font_info (data->window, &data->new_ascent, &data->new_descent, 0, 0, 0); } /* No automatically positioned glyphs? Return at once. */ if (!data->need_baseline_computation) return; /* Is the tallest glyph on the line automatically positioned? If it's manually positioned, or it's automatically positioned and there's enough room for it anyway, we need do no more work. */ if (data->max_pixmap_height > data->new_ascent + data->new_descent) { int default_font_ascent, default_font_descent, default_font_height; int scaled_default_font_ascent, scaled_default_font_descent; default_face_font_info (data->window, &default_font_ascent, &default_font_descent, 0, &default_font_height, 0); scaled_default_font_ascent = data->max_pixmap_height * default_font_ascent / default_font_height; data->new_ascent = max (data->new_ascent, scaled_default_font_ascent); /* The ascent may have expanded now. Do we still need to grow the descent, or are things big enough? The +1 caters for the baseline row itself. */ if (data->max_pixmap_height > data->new_ascent + data->new_descent) { scaled_default_font_descent = (data->max_pixmap_height * default_font_descent / default_font_height) + 1; data->new_descent = max (data->new_descent, scaled_default_font_descent); } } } /* Given a display line and a starting position, ensure that the contents of the display line accurately represent the visual representation of the buffer contents starting from the given position when displayed in the given window. The display line ends when the contents of the line reach the right boundary of the given window. */ static Charbpos generate_display_line (struct window *w, struct display_line *dl, int bounds, Charbpos start_pos, prop_block_dynarr **prop, int type) { Charbpos ret_charpos; int overlay_width; struct buffer *b = XBUFFER (WINDOW_BUFFER (w)); /* If our caller hasn't already set the boundaries, then do so now. */ if (!bounds) dl->bounds = calculate_display_line_boundaries (w, 0); /* Reset what this line is using. */ if (dl->display_blocks) Dynarr_reset (dl->display_blocks); if (dl->left_glyphs) { Dynarr_free (dl->left_glyphs); dl->left_glyphs = 0; } if (dl->right_glyphs) { Dynarr_free (dl->right_glyphs); dl->right_glyphs = 0; } /* We aren't generating a modeline at the moment. */ dl->modeline = 0; /* Create a display block for the text region of the line. */ { /* #### urk urk urk!!! Chuck fix this shit! */ Bytebpos hacked_up_bytebpos = create_text_block (w, dl, charbpos_to_bytebpos (b, start_pos), prop, type); if (hacked_up_bytebpos > BYTE_BUF_ZV (b)) ret_charpos = BUF_ZV (b) + 1; else ret_charpos = bytebpos_to_charbpos (b, hacked_up_bytebpos); } dl->charpos = start_pos; if (dl->end_charpos < dl->charpos) dl->end_charpos = dl->charpos; if (MARKERP (Voverlay_arrow_position) && EQ (w->buffer, Fmarker_buffer (Voverlay_arrow_position)) && start_pos == marker_position (Voverlay_arrow_position) && (STRINGP (Voverlay_arrow_string) || GLYPHP (Voverlay_arrow_string))) { overlay_width = create_overlay_glyph_block (w, dl); } else overlay_width = 0; /* If there are left glyphs associated with any character in the text block, then create a display block to handle them. */ if (dl->left_glyphs != NULL && Dynarr_length (dl->left_glyphs)) create_left_glyph_block (w, dl, overlay_width); /* If there are right glyphs associated with any character in the text block, then create a display block to handle them. */ if (dl->right_glyphs != NULL && Dynarr_length (dl->right_glyphs)) create_right_glyph_block (w, dl); /* In the future additional types of display blocks may be generated here. */ w->last_redisplay_pos = ret_charpos; return ret_charpos; } /* Adds an hscroll glyph to a display block. If this is called, then the block had better be empty. Yes, there are multiple places where this function is called but that is the way it has to be. Each calling function has to deal with byte_start_col_enabled a little differently depending on the object being worked with. */ static prop_block_dynarr * add_hscroll_rune (pos_data *data) { struct glyph_block gb; prop_block_dynarr *retval; Bytebpos byte_old_cursor_charpos = data->byte_cursor_charpos; int old_cursor_type = data->cursor_type; Bytebpos byte_old_charpos = data->byte_charpos; if (data->cursor_type == CURSOR_ON && data->byte_cursor_charpos >= data->byte_start_col_enabled && data->byte_cursor_charpos <= data->byte_charpos) { data->byte_cursor_charpos = data->byte_start_col_enabled; } else { data->cursor_type = NO_CURSOR; } data->byte_endpos = data->byte_charpos; data->byte_charpos = data->byte_start_col_enabled; gb.extent = Qnil; gb.glyph = Vhscroll_glyph; { int oldpixpos = data->pixpos; retval = add_glyph_rune (data, &gb, BEGIN_GLYPHS, 0, GLYPH_CACHEL (XWINDOW (data->window), HSCROLL_GLYPH_INDEX)); data->hscroll_glyph_width_adjust = data->pixpos - oldpixpos - space_width (XWINDOW (data->window)); } data->byte_endpos = 0; data->byte_cursor_charpos = byte_old_cursor_charpos; data->cursor_type = old_cursor_type; data->byte_charpos = byte_old_charpos; data->byte_start_col_enabled = 0; return retval; } /* Adds a character rune to a display block. If there is not enough room to fit the rune on the display block (as determined by the MAX_PIXPOS) then it adds nothing and returns ADD_FAILED. If NO_CONTRIBUTE_TO_LINE_HEIGHT is non-zero, don't allow the char's height to affect the total line height. (See add_ibyte_string_runes()). */ static prop_block_dynarr * add_ichar_rune_1 (pos_data *data, int no_contribute_to_line_height) { struct rune rb, *crb; int width, local; if (data->start_col) { data->start_col--; if (data->start_col) return NULL; } if (data->byte_start_col_enabled) { return add_hscroll_rune (data); } if (data->ch == '\n') { data->font_is_bogus = 0; /* Cheesy end-of-line pseudo-character. */ width = data->blank_width; } else { Lisp_Object charset = ichar_charset (data->ch); if (!EQ (charset, data->last_charset) || data->findex != data->last_findex) { /* OK, we need to do things the hard way. */ struct window *w = XWINDOW (data->window); struct face_cachel *cachel = WINDOW_FACE_CACHEL (w, data->findex); Lisp_Object font_instance = ensure_face_cachel_contains_charset (cachel, data->window, charset); Lisp_Font_Instance *fi; if (EQ (font_instance, Vthe_null_font_instance)) { font_instance = FACE_CACHEL_FONT (cachel, Vcharset_ascii); data->font_is_bogus = 1; } else data->font_is_bogus = 0; fi = XFONT_INSTANCE (font_instance); if (!fi->proportional_p || data->font_is_bogus) { Ichar ch = data->font_is_bogus ? '~' : data->ch; data->last_char_width = redisplay_window_text_width_ichar_string (XWINDOW (data->window), data->findex, &ch, 1); } else data->last_char_width = -1; if (!no_contribute_to_line_height) { data->new_ascent = max (data->new_ascent, (int) fi->ascent); data->new_descent = max (data->new_descent, (int) fi->descent); } data->last_charset = charset; data->last_findex = data->findex; } width = data->last_char_width; if (width < 0) /* proportional fonts */ width = redisplay_window_text_width_ichar_string (XWINDOW (data->window), data->findex, &data->ch, 1); } if (data->max_pixpos != -1 && (data->pixpos + width > data->max_pixpos)) { return ADD_FAILED; } if (Dynarr_length (data->db->runes) < Dynarr_largest (data->db->runes)) { crb = Dynarr_past_lastp (data->db->runes); local = 0; } else { crb = &rb; local = 1; } crb->findex = data->findex; crb->xpos = data->pixpos; crb->width = width; if (data->byte_charpos) { if (NILP (data->string)) crb->charpos = bytebpos_to_charbpos (XBUFFER (WINDOW_BUFFER (XWINDOW (data->window))), data->byte_charpos); else crb->charpos = string_index_byte_to_char (data->string, data->byte_charpos); } else if (data->is_modeline) crb->charpos = data->modeline_charpos; else /* Text but not in buffer */ crb->charpos = 0; crb->type = RUNE_CHAR; crb->object.chr.ch = data->font_is_bogus ? '~' : data->ch; crb->endpos = 0; if (data->cursor_type == CURSOR_ON) { if (data->byte_charpos == data->byte_cursor_charpos) { crb->cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); } else crb->cursor_type = CURSOR_OFF; } else if (data->cursor_type == NEXT_CURSOR) { crb->cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); data->cursor_type = NO_CURSOR; } else if (data->cursor_type == IGNORE_CURSOR) crb->cursor_type = IGNORE_CURSOR; else crb->cursor_type = CURSOR_OFF; if (local) Dynarr_add (data->db->runes, *crb); else Dynarr_incrementr (data->db->runes); data->pixpos += width; return NULL; } static prop_block_dynarr * add_ichar_rune (pos_data *data) { return add_ichar_rune_1 (data, 0); } /* Given a string C_STRING of length C_LENGTH, call add_ichar_rune for each character in the string. Propagate any left-over data unless NO_PROP is non-zero. If NO_CONTRIBUTE_TO_LINE_HEIGHT is non-zero, don't allow this character to increase the total height of the line. (This is used when the character is part of a text glyph. In that case, the glyph code itself adjusts the line height as necessary, depending on whether glyph-contrib-p is true.) */ static prop_block_dynarr * add_ibyte_string_runes (pos_data *data, Ibyte *c_string, Bytecount c_length, int no_prop, int no_contribute_to_line_height) { Ibyte *pos, *end = c_string + c_length; prop_block_dynarr *prop; /* #### This function is too simplistic. It needs to do the same sort of character interpretation (display-table lookup, ctl-arrow checking), etc. that create_text_block() does. The functionality to do this in that routine needs to be modularized. */ for (pos = c_string; pos < end;) { Ibyte *old_pos = pos; data->ch = itext_ichar (pos); prop = add_ichar_rune_1 (data, no_contribute_to_line_height); if (prop) { if (no_prop) return ADD_FAILED; else { struct prop_block pb; Bytecount len = end - pos; prop = Dynarr_new (prop_block); pb.type = PROP_STRING; pb.data.p_string.str = xnew_array (Ibyte, len); qxestrncpy (pb.data.p_string.str, pos, len); pb.data.p_string.len = len; Dynarr_add (prop, pb); return prop; } } INC_IBYTEPTR (pos); assert (pos <= end); /* #### Duplicate code from add_string_to_fstring_db_runes should we do more?*/ data->bytepos += pos - old_pos; } return NULL; } /* Add a single rune of the specified width. The area covered by this rune will be displayed in the foreground color of the associated face. */ static prop_block_dynarr * add_blank_rune (pos_data *data, struct window *w, int char_tab_width) { struct rune rb; /* If data->start_col is not 0 then this call to add_blank_rune must have been to add it as a tab. */ if (data->start_col) { /* assert (w != NULL) */ prop_block_dynarr *retval; /* If we have still not fully scrolled horizontally, subtract the width of this tab and return. */ if (char_tab_width < data->start_col) { data->start_col -= char_tab_width; return NULL; } else if (char_tab_width == data->start_col) data->blank_width = 0; else { int spcwid = space_width (w); if (spcwid >= data->blank_width) data->blank_width = 0; else data->blank_width -= spcwid; } data->start_col = 0; retval = add_hscroll_rune (data); /* Could be caused by the handling of the hscroll rune. */ if (retval != NULL || !data->blank_width) return retval; } /* Blank runes are always calculated to fit. */ assert (data->pixpos + data->blank_width <= data->max_pixpos); rb.findex = data->findex; rb.xpos = data->pixpos; rb.width = data->blank_width; if (data->byte_charpos) rb.charpos = bytebpos_to_charbpos (XBUFFER (WINDOW_BUFFER (XWINDOW (data->window))), data->byte_charpos); else /* #### and this is really correct too? */ rb.charpos = 0; rb.endpos = 0; rb.type = RUNE_BLANK; if (data->cursor_type == CURSOR_ON) { if (data->byte_charpos == data->byte_cursor_charpos) { rb.cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); } else rb.cursor_type = CURSOR_OFF; } else if (data->cursor_type == NEXT_CURSOR) { rb.cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); data->cursor_type = NO_CURSOR; } else rb.cursor_type = CURSOR_OFF; Dynarr_add (data->db->runes, rb); data->pixpos += data->blank_width; return NULL; } /* Add runes representing a character in octal. */ #define ADD_NEXT_OCTAL_RUNE_CHAR do \ { \ if (add_failed || (add_failed = add_ichar_rune (data))) \ { \ struct prop_block pb; \ if (!prop) \ prop = Dynarr_new (prop_block); \ \ pb.type = PROP_CHAR; \ pb.data.p_char.ch = data->ch; \ pb.data.p_char.cursor_type = data->cursor_type; \ Dynarr_add (prop, pb); \ } \ } while (0) static prop_block_dynarr * add_octal_runes (pos_data *data) { prop_block_dynarr *add_failed, *prop = 0; Ichar orig_char = data->ch; int orig_cursor_type = data->cursor_type; /* Initialize */ add_failed = NULL; if (data->start_col) data->start_col--; if (!data->start_col) { if (data->byte_start_col_enabled) { add_failed = add_hscroll_rune (data); } else { struct glyph_block gb; struct window *w = XWINDOW (data->window); gb.extent = Qnil; gb.glyph = Voctal_escape_glyph; add_failed = add_glyph_rune (data, &gb, BEGIN_GLYPHS, 1, GLYPH_CACHEL (w, OCT_ESC_GLYPH_INDEX)); } } /* We only propagate information if the glyph was partially added. */ if (add_failed) return add_failed; data->cursor_type = IGNORE_CURSOR; if (data->ch >= 0x100) { /* If the character is an extended Mule character, it could have up to 21 bits. For the moment, we treat it as a seven-digit octal number. This is not that pretty, but whatever. */ data->ch = (7 & (orig_char >> 18)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->ch = (7 & (orig_char >> 15)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->ch = (7 & (orig_char >> 12)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->ch = (7 & (orig_char >> 9)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; } data->ch = (7 & (orig_char >> 6)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->ch = (7 & (orig_char >> 3)) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->ch = (7 & orig_char) + '0'; ADD_NEXT_OCTAL_RUNE_CHAR; data->cursor_type = orig_cursor_type; return NULL; } #undef ADD_NEXT_OCTAL_RUNE_CHAR /* Add runes representing a control character to a display block. */ static prop_block_dynarr * add_control_char_runes (pos_data *data, struct buffer *b) { if (!NILP (b->ctl_arrow)) { prop_block_dynarr *prop; Ichar orig_char = data->ch; int old_cursor_type = data->cursor_type; /* Initialize */ prop = NULL; if (data->start_col) data->start_col--; if (!data->start_col) { if (data->byte_start_col_enabled) { prop_block_dynarr *retval; retval = add_hscroll_rune (data); if (retval) return retval; } else { struct glyph_block gb; struct window *w = XWINDOW (data->window); gb.extent = Qnil; gb.glyph = Vcontrol_arrow_glyph; /* We only propagate information if the glyph was partially added. */ if (add_glyph_rune (data, &gb, BEGIN_GLYPHS, 1, GLYPH_CACHEL (w, CONTROL_GLYPH_INDEX))) return ADD_FAILED; } } if (orig_char == 0177) data->ch = '?'; else data->ch = orig_char ^ 0100; data->cursor_type = IGNORE_CURSOR; if (add_ichar_rune (data)) { struct prop_block pb; if (!prop) prop = Dynarr_new (prop_block); pb.type = PROP_CHAR; pb.data.p_char.ch = data->ch; pb.data.p_char.cursor_type = data->cursor_type; Dynarr_add (prop, pb); } data->cursor_type = old_cursor_type; return prop; } else { return add_octal_runes (data); } } static prop_block_dynarr * add_disp_table_entry_runes_1 (pos_data *data, Lisp_Object entry) { prop_block_dynarr *prop = NULL; if (STRINGP (entry)) { prop = add_ibyte_string_runes (data, XSTRING_DATA (entry), XSTRING_LENGTH (entry), 0, 0); } else if (GLYPHP (entry)) { if (data->start_col) data->start_col--; if (!data->start_col && data->byte_start_col_enabled) { prop = add_hscroll_rune (data); } else { struct glyph_block gb; gb.glyph = entry; gb.extent = Qnil; prop = add_glyph_rune (data, &gb, BEGIN_GLYPHS, 0, 0); } } else if (CHAR_OR_CHAR_INTP (entry)) { data->ch = XCHAR_OR_CHAR_INT (entry); prop = add_ichar_rune (data); } else if (CONSP (entry)) { if (EQ (XCAR (entry), Qformat) && CONSP (XCDR (entry)) && STRINGP (XCAR (XCDR (entry)))) { Lisp_Object format = XCAR (XCDR (entry)); Bytebpos len = XSTRING_LENGTH (format); Ibyte *src = XSTRING_DATA (format), *end = src + len; Ibyte *result = alloca_ibytes (len); Ibyte *dst = result; while (src < end) { Ichar c = itext_ichar (src); INC_IBYTEPTR (src); if (c != '%' || src == end) dst += set_itext_ichar (dst, c); else { c = itext_ichar (src); INC_IBYTEPTR (src); switch (c) { /*case 'x': dst += long_to_string_base ((char *)dst, data->ch, 16); break;*/ case '%': dst += set_itext_ichar (dst, '%'); break; /* #### unimplemented */ } } } prop = add_ibyte_string_runes (data, result, dst - result, 0, 0); } } /* Else blow it off because someone added a bad entry and we don't have any safe way of signaling an error. */ return prop; } /* Given a display table entry, call the appropriate functions to display each element of the entry. */ static prop_block_dynarr * add_disp_table_entry_runes (pos_data *data, Lisp_Object entry) { prop_block_dynarr *prop = NULL; if (VECTORP (entry)) { Lisp_Vector *de = XVECTOR (entry); EMACS_INT len = vector_length (de); int elt; for (elt = 0; elt < len; elt++) { if (NILP (vector_data (de)[elt])) continue; else prop = add_disp_table_entry_runes_1 (data, vector_data (de)[elt]); /* Else blow it off because someone added a bad entry and we don't have any safe way of signaling an error. Hey, this comment sounds familiar. */ /* #### Still need to add any remaining elements to the propagation information. */ if (prop) return prop; } } else prop = add_disp_table_entry_runes_1 (data, entry); return prop; } /* Add runes which were propagated from the previous line. */ static prop_block_dynarr * add_propagation_runes (prop_block_dynarr **prop, pos_data *data) { /* #### Remember to handle start_col parameter of data when the rest of this is finished. */ /* #### Chuck -- I've redone this function a bit. It looked like the case of not all the propagation blocks being added was not handled well. */ /* #### Chuck -- I also think the double indirection of PROP is kind of bogus. A cleaner solution is just to check for Dynarr_length (prop) > 0. */ /* #### This function also doesn't even pay attention to ADD_FAILED! This is seriously fucked! Seven ####'s in 130 lines -- is that a record? */ int elt; prop_block_dynarr *add_failed; Bytebpos byte_old_cursor_charpos = data->byte_cursor_charpos; int old_cursor_type = data->cursor_type; for (elt = 0; elt < Dynarr_length (*prop); elt++) { struct prop_block *pb = Dynarr_atp (*prop, elt); switch (pb->type) { case PROP_CHAR: data->ch = pb->data.p_char.ch; data->byte_cursor_charpos = pb->data.p_char.byte_cursor_charpos; data->cursor_type = pb->data.p_char.cursor_type; add_failed = add_ichar_rune (data); if (add_failed) goto oops_no_more_space; break; case PROP_STRING: if (pb->data.p_string.str) { xfree (pb->data.p_string.str); pb->data.p_string.str = 0; } /* #### bogus bogus -- this doesn't do anything! Should probably call add_ibyte_string_runes(), once that function is fixed. */ break; case PROP_MINIBUF_PROMPT: { face_index old_findex = data->findex; Bytebpos byte_old_charpos = data->byte_charpos; Boolint stop_after = NILP (pb->data.p_minibuf_prompt.preprompt); Lisp_Object str = stop_after ? pb->data.p_minibuf_prompt.prompt : pb->data.p_minibuf_prompt.preprompt; struct window *w = XWINDOW (data->window); data->byte_charpos = 0; data->cursor_type = NO_CURSOR; /* This doesn't handle begin-glyphs and end-glyphs and so on. It may be reasonable not to, given that we're a "propagation glyph", but it's not intuitively clear either way. It is clear that it should handle the face and the display table. */ while (STRINGP (str)) { Ibyte *pstart = XSTRING_DATA (str), *pp = pstart, *pend = pstart + XSTRING_LENGTH (str); struct extent_fragment *ef = extent_fragment_new (str, XFRAME (w->frame)); while (pp < pend) { Lisp_Object face_dt, window_dt, entry = Qnil; face_index new_findex = data->findex = extent_fragment_update (w, ef, pp - pstart, Qnil); data->ch = itext_ichar (pp); get_display_tables (w, new_findex, &face_dt, &window_dt); if (!NILP (face_dt) || !NILP (window_dt)) { entry = display_table_entry (data->ch, face_dt, window_dt); } /* If there is a display table entry for it, hand it off to add_disp_table_entry_runes and let it worry about it. */ if (!NILP (entry) && !EQ (entry, make_char (data->ch))) { add_failed = add_disp_table_entry_runes (data, entry); } else { add_failed = add_ichar_rune (data); } if (add_failed) { data->findex = old_findex; data->byte_charpos = byte_old_charpos; extent_fragment_delete (ef); goto oops_no_more_space; } INC_IBYTEPTR (pp); } extent_fragment_delete (ef); if (stop_after) { break; } str = pb->data.p_minibuf_prompt.prompt; stop_after = 1; } data->findex = old_findex; /* ##### FIXME FIXME FIXME -- Upon successful return from this function, data->byte_charpos is automatically incremented. However, we don't want that to happen if we were adding the minibuffer prompt. */ { struct buffer *buf = XBUFFER (WINDOW_BUFFER (XWINDOW (data->window))); /* #### Chuck fix this shit or I'm gonna scream! */ if (byte_old_charpos > BYTE_BUF_BEGV (buf)) data->byte_charpos = prev_bytebpos (buf, byte_old_charpos); else /* #### is this correct? Does anyone know? Does anyone care? Is this a cheesy hack or what? */ data->byte_charpos = BYTE_BUF_BEGV (buf) - 1; } } break; case PROP_BLANK: { /* #### I think it's unnecessary and misleading to preserve the blank_width, as it implies that the value carries over from one rune to the next, which is wrong. */ int old_width = data->blank_width; face_index old_findex = data->findex; data->findex = pb->data.p_blank.findex; data->blank_width = pb->data.p_blank.width; data->byte_cursor_charpos = 0; data->cursor_type = IGNORE_CURSOR; if (data->pixpos + data->blank_width > data->max_pixpos) data->blank_width = data->max_pixpos - data->pixpos; /* We pass a bogus value of char_tab_width. It shouldn't matter because unless something is really screwed up this call won't cause that arg to be used. */ add_failed = add_blank_rune (data, XWINDOW (data->window), 0); /* This can happen in the case where we have a tab which is wider than the window. */ if (data->blank_width != pb->data.p_blank.width) { pb->data.p_blank.width -= data->blank_width; add_failed = ADD_FAILED; } data->findex = old_findex; data->blank_width = old_width; if (add_failed) goto oops_no_more_space; } break; default: ABORT (); } } oops_no_more_space: data->byte_cursor_charpos = byte_old_cursor_charpos; data->cursor_type = old_cursor_type; if (elt < Dynarr_length (*prop)) { Dynarr_delete_many (*prop, 0, elt); return *prop; } else { Dynarr_free (*prop); return NULL; } } /* Add `text' layout glyphs at position POS_TYPE that are contained to the display block, but add all other types to the appropriate list of the display line. They will be added later by different routines. */ static prop_block_dynarr * add_glyph_rune (pos_data *data, struct glyph_block *gb, int pos_type, int allow_cursor, struct glyph_cachel *cachel) { struct window *w = XWINDOW (data->window); /* If window faces changed, and glyph instance is text, then glyph sizes might have changed too */ invalidate_glyph_geometry_maybe (gb->glyph, w); /* This makes sure the glyph is in the cachels. #### We do this to make sure the glyph is in the glyph cachels, so that the dirty flag can be reset after redisplay has finished. We should do this some other way, maybe by iterating over the window cache of subwindows. */ get_glyph_cachel_index (w, gb->glyph); /* A nil extent indicates a special glyph (ex. truncator). */ if (NILP (gb->extent) || (pos_type == BEGIN_GLYPHS && extent_begin_glyph_layout (XEXTENT (gb->extent)) == GL_TEXT) || (pos_type == END_GLYPHS && extent_end_glyph_layout (XEXTENT (gb->extent)) == GL_TEXT) || pos_type == LEFT_GLYPHS || pos_type == RIGHT_GLYPHS) { struct rune rb; int width; int xoffset = 0; int ascent, descent; Lisp_Object baseline; Lisp_Object face; Lisp_Object instance; face_index findex; prop_block_dynarr *retval = 0; if (cachel) width = cachel->width; else width = glyph_width (gb->glyph, data->window); if (!width) return NULL; if (data->start_col || data->start_col_xoffset) { int glyph_char_width = width / space_width (w); /* If we still have not fully scrolled horizontally after taking into account the width of the glyph, subtract its width and return. */ if (glyph_char_width < data->start_col) { data->start_col -= glyph_char_width; return NULL; } else if (glyph_char_width == data->start_col) width = 0; else { xoffset = space_width (w) * data->start_col; width -= xoffset; /* #### Can this happen? */ if (width < 0) width = 0; } data->start_col = 0; retval = add_hscroll_rune (data); /* Could be caused by the handling of the hscroll rune. */ if (retval != NULL || !width) return retval; } else xoffset = 0; if (data->pixpos + width > data->max_pixpos) { /* If this is the first object we are attempting to add to the line then we ignore the horizontal_clip threshold. Otherwise we will loop until the bottom of the window continually failing to add this glyph because it is wider than the window. We could alternatively just completely ignore the glyph and proceed from there but I think that this is a better solution. This does, however, create a different problem in that we can end up adding the object to every single line, never getting any further - for instance an extent with a long start-glyph that covers multitple following characters. */ if (Dynarr_length (data->db->runes) && data->max_pixpos - data->pixpos < horizontal_clip) return ADD_FAILED; else { struct prop_block pb; /* We need to account for the width of the end-of-line glyph if there is nothing more in the line to display, since we will not display it in this instance. It seems kind of gross doing it here, but otherwise we have to search the runes in create_text_block(). */ if (data->ch == '\n') data->max_pixpos += data->end_glyph_width; width = data->max_pixpos - data->pixpos; /* Add the glyph we are displaying, but clipping, to the propagation data so that we don't try and do it again. */ retval = Dynarr_new (prop_block); pb.type = PROP_GLYPH; pb.data.p_glyph.glyph = gb->glyph; pb.data.p_glyph.width = width; Dynarr_add (retval, pb); } } if (cachel) { ascent = cachel->ascent; descent = cachel->descent; } else { ascent = glyph_ascent (gb->glyph, data->window); descent = glyph_descent (gb->glyph, data->window); } baseline = glyph_baseline (gb->glyph, data->window); rb.object.dglyph.descent = 0; /* Gets reset lower down, if it is known. */ if (glyph_contrib_p (gb->glyph, data->window)) { /* A pixmap that has not had a baseline explicitly set. Its contribution will be determined later. */ if (NILP (baseline)) { int height = ascent + descent; data->need_baseline_computation = 1; data->max_pixmap_height = max (data->max_pixmap_height, height); } /* A string so determine contribution normally. */ else if (EQ (baseline, Qt)) { data->new_ascent = max (data->new_ascent, ascent); data->new_descent = max (data->new_descent, descent); } /* A pixmap with an explicitly set baseline. We determine the contribution here. */ else if (FIXNUMP (baseline)) { int height = ascent + descent; int pix_ascent, pix_descent; pix_ascent = height * XFIXNUM (baseline) / 100; pix_descent = height - pix_ascent; data->new_ascent = max (data->new_ascent, pix_ascent); data->new_descent = max (data->new_descent, pix_descent); data->max_pixmap_height = max (data->max_pixmap_height, height); rb.object.dglyph.descent = pix_descent; } /* Otherwise something is screwed up. */ else ABORT (); } face = glyph_face (gb->glyph, data->window); if (NILP (face)) findex = data->findex; else findex = get_builtin_face_cache_index (w, face); instance = glyph_image_instance (gb->glyph, data->window, ERROR_ME_DEBUG_WARN, 1); if (TEXT_IMAGE_INSTANCEP (instance)) { Lisp_Object string = XIMAGE_INSTANCE_TEXT_STRING (instance); face_index orig_findex = data->findex; Bytebpos orig_charpos = data->byte_charpos; Bytebpos orig_start_col_enabled = data->byte_start_col_enabled; data->findex = findex; data->byte_start_col_enabled = 0; if (!allow_cursor) data->byte_charpos = 0; add_ibyte_string_runes (data, XSTRING_DATA (string), XSTRING_LENGTH (string), 0, 1); data->findex = orig_findex; data->byte_charpos = orig_charpos; data->byte_start_col_enabled = orig_start_col_enabled; return retval; } rb.findex = findex; rb.xpos = data->pixpos; rb.width = width; rb.charpos = 0; /* glyphs are never "at" anywhere */ if (data->byte_endpos) /* #### is this necessary at all? */ rb.endpos = bytebpos_to_charbpos (XBUFFER (WINDOW_BUFFER (w)), data->byte_endpos); else rb.endpos = 0; rb.type = RUNE_DGLYPH; rb.object.dglyph.glyph = gb->glyph; rb.object.dglyph.extent = gb->extent; rb.object.dglyph.xoffset = xoffset; rb.object.dglyph.ascent = ascent; rb.object.dglyph.yoffset = 0; /* Until we know better, assume that it has a normal (textual) baseline. */ if (allow_cursor) { rb.charpos = bytebpos_to_charbpos (XBUFFER (WINDOW_BUFFER (w)), data->byte_charpos); if (data->cursor_type == CURSOR_ON) { if (data->byte_charpos == data->byte_cursor_charpos) { rb.cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); } else rb.cursor_type = CURSOR_OFF; } else if (data->cursor_type == NEXT_CURSOR) { rb.cursor_type = CURSOR_ON; data->cursor_x = Dynarr_length (data->db->runes); data->cursor_type = NO_CURSOR; } else if (data->cursor_type == IGNORE_CURSOR) rb.cursor_type = IGNORE_CURSOR; else if (data->cursor_type == NO_CURSOR) rb.cursor_type = NO_CURSOR; else rb.cursor_type = CURSOR_OFF; } else rb.cursor_type = CURSOR_OFF; Dynarr_add (data->db->runes, rb); data->pixpos += width; return retval; } else { if (!NILP (glyph_face (gb->glyph, data->window))) gb->findex = get_builtin_face_cache_index (w, glyph_face (gb->glyph, data->window)); else gb->findex = data->findex; if (pos_type == BEGIN_GLYPHS) { if (!data->dl->left_glyphs) data->dl->left_glyphs = Dynarr_new (glyph_block); Dynarr_add (data->dl->left_glyphs, *gb); return NULL; } else if (pos_type == END_GLYPHS) { if (!data->dl->right_glyphs) data->dl->right_glyphs = Dynarr_new (glyph_block); Dynarr_add (data->dl->right_glyphs, *gb); return NULL; } else ABORT (); /* there are no unknown types */ } return NULL; } /* Add all glyphs at position POS_TYPE that are contained in the given data. */ static prop_block_dynarr * add_glyph_runes (pos_data *data, int pos_type) { /* #### This still needs to handle the start_col parameter. Duh, Chuck, why didn't you just modify add_glyph_rune in the first place? */ int elt; glyph_block_dynarr *glyph_arr = (pos_type == BEGIN_GLYPHS ? data->ef->begin_glyphs : data->ef->end_glyphs); prop_block_dynarr *prop; for (elt = 0; elt < Dynarr_length (glyph_arr); elt++) { prop = add_glyph_rune (data, Dynarr_atp (glyph_arr, elt), pos_type, 0, 0); if (prop) { /* #### Add some propagation information. */ return prop; } } Dynarr_reset (glyph_arr); return NULL; } /* Given a position for a buffer in a window, ensure that the given display line DL accurately represents the text on a line starting at the given position. NOTE NOTE NOTE NOTE: This function works with and returns Bytebpos's. You must do appropriate conversion. */ static Bytebpos create_text_block (struct window *w, struct display_line *dl, Bytebpos byte_start_pos, prop_block_dynarr **prop, int type) { struct frame *f = XFRAME (w->frame); struct buffer *b = XBUFFER (w->buffer); struct device *d = XDEVICE (f->device); pos_data data; /* Don't display anything in the minibuffer if this window is not on a selected frame. We consider all other windows to be active minibuffers as it simplifies the coding. */ int active_minibuffer = (!MINI_WINDOW_P (w) || (f == device_selected_frame (d)) || is_surrogate_for_selected_frame (f)); int truncate_win = window_truncation_on (w); /* If the buffer's value of selective_display is an integer then only lines that start with less than selective_display columns of space will be displayed. If selective_display is t then all text after a ^M is invisible. */ int selective = (FIXNUMP (b->selective_display) ? XFIXNUM (b->selective_display) : (!NILP (b->selective_display) ? -1 : 0)); /* The variable ctl-arrow allows the user to specify what characters can actually be displayed and which octal should be used for. #### This variable should probably have some rethought done to it. See also (Info-goto-node "(internals)Future Work -- Display Tables") */ Ichar printable_min = (CHAR_OR_CHAR_INTP (b->ctl_arrow) ? XCHAR_OR_CHAR_INT (b->ctl_arrow) : ((EQ (b->ctl_arrow, Qt) || EQ (b->ctl_arrow, Qnil)) ? 255 : 160)); Lisp_Object face_dt, window_dt; /* The text display block for this display line. */ struct display_block *db = get_display_block_from_line (dl, TEXT); /* The first time through the main loop we need to force the glyph data to be updated. */ int initial = 1; /* Apparently the new extent_fragment_update returns an end position equal to the position passed in if there are no more runs to be displayed. */ int no_more_frags = 0; Lisp_Object synch_minibuffers_value = symbol_value_in_buffer (Qsynchronize_minibuffers, w->buffer); dl->used_prop_data = 0; dl->num_chars = 0; dl->line_continuation = 0; dl->clear_findex = DEFAULT_INDEX; xzero (data); data.ef = extent_fragment_new (w->buffer, f); /* These values are used by all of the rune addition routines. We add them to this structure for ease of passing. */ data.d = d; data.window = wrap_window (w); data.string = Qnil; data.db = db; data.dl = dl; data.byte_charpos = byte_start_pos; data.pixpos = dl->bounds.left_in; data.last_charset = Qunbound; data.last_findex = DEFAULT_INDEX; data.result_str = Qnil; /* Set the right boundary adjusting it to take into account any end glyph. Save the width of the end glyph for later use. */ data.max_pixpos = dl->bounds.right_in; if (truncate_win) data.end_glyph_width = GLYPH_CACHEL_WIDTH (w, TRUN_GLYPH_INDEX); else data.end_glyph_width = GLYPH_CACHEL_WIDTH (w, CONT_GLYPH_INDEX); data.max_pixpos -= data.end_glyph_width; if (cursor_in_echo_area && MINI_WINDOW_P (w) && echo_area_active (f)) { data.byte_cursor_charpos = BYTE_BUF_ZV (b); data.cursor_type = CURSOR_ON; } else if (MINI_WINDOW_P (w) && !active_minibuffer) data.cursor_type = NO_CURSOR; else if (w == XWINDOW (FRAME_SELECTED_WINDOW (f)) && EQ (DEVICE_CONSOLE (d), Vselected_console) && d == XDEVICE (CONSOLE_SELECTED_DEVICE (XCONSOLE (DEVICE_CONSOLE (d))))&& f == XFRAME (DEVICE_SELECTED_FRAME (d))) { data.byte_cursor_charpos = BYTE_BUF_PT (b); data.cursor_type = CURSOR_ON; } else if (w == XWINDOW (FRAME_SELECTED_WINDOW (f))) { data.byte_cursor_charpos = byte_marker_position (w->pointm[type]); data.cursor_type = CURSOR_ON; } else data.cursor_type = NO_CURSOR; data.cursor_x = -1; data.start_col = w->hscroll; data.start_col_xoffset = w->left_xoffset; data.byte_start_col_enabled = (w->hscroll ? byte_start_pos : 0); data.hscroll_glyph_width_adjust = 0; /* We regenerate the line from the very beginning. */ Dynarr_reset (db->runes); /* Why is this less than or equal and not just less than? If the starting position is already equal to the maximum we can't add anything else, right? Wrong. We might still have a newline to add. A newline can use the room allocated for an end glyph since if we add it we know we aren't going to be adding any end glyph. */ /* #### Chuck -- I think this condition should be while (1). Otherwise if (e.g.) there is one begin-glyph and one end-glyph and the begin-glyph ends exactly at the end of the window, the end-glyph and text might not be displayed. while (1) ensures that the loop terminates only when either (a) there is propagation data or (b) the end-of-line or end-of-buffer is hit. #### Also I think you need to ensure that the operation "add begin glyphs; add end glyphs; add text" is atomic and can't get interrupted in the middle. If you run off the end of the line during that operation, then you keep accumulating propagation data until you're done. Otherwise, if the (e.g.) there's a begin glyph at a particular position and attempting to display that glyph results in window-end being hit and propagation data being generated, then the character at that position won't be displayed. #### See also the comment after the end of this loop, below. */ while (data.pixpos <= data.max_pixpos && (active_minibuffer || !NILP (synch_minibuffers_value))) { /* #### This check probably should not be necessary. */ if (data.byte_charpos > BYTE_BUF_ZV (b)) { /* #### urk! More of this lossage! */ data.byte_charpos--; goto done; } /* If selective display was an integer and we aren't working on a continuation line then find the next line we are actually supposed to display. */ if (selective > 0 && (data.byte_charpos == BYTE_BUF_BEGV (b) || BUF_FETCH_CHAR (b, prev_bytebpos (b, data.byte_charpos)) == '\n')) { while (byte_spaces_at_point (b, data.byte_charpos) >= selective) { data.byte_charpos = byte_find_next_newline_no_quit (b, data.byte_charpos, 1); if (data.byte_charpos >= BYTE_BUF_ZV (b)) { data.byte_charpos = BYTE_BUF_ZV (b); goto done; } } } /* Check for face changes. */ if (initial || (!no_more_frags && data.byte_charpos == data.ef->end)) { Lisp_Object last_glyph = Qnil; /* Deal with glyphs that we have already displayed. The theory is that if we end up with a PROP_GLYPH in the propagation data then we are clipping the glyph and there can be no propagation data before that point. The theory works because we always recalculate the extent-fragments for propagated data, we never actually propagate the fragments that still need to be displayed. */ if (*prop && Dynarr_begin (*prop)->type == PROP_GLYPH) { last_glyph = Dynarr_begin (*prop)->data.p_glyph.glyph; Dynarr_free (*prop); *prop = 0; } /* Now compute the face and begin/end-glyph information. */ data.findex = /* Remember that the extent-fragment routines deal in Bytebpos's. */ extent_fragment_update (w, data.ef, data.byte_charpos, last_glyph); get_display_tables (w, data.findex, &face_dt, &window_dt); if (data.byte_charpos == data.ef->end) no_more_frags = 1; } initial = 0; /* Determine what is next to be displayed. We first handle any glyphs returned by glyphs_at_charbpos. If there are no glyphs to display then we determine what to do based on the character at the current buffer position. */ /* If the current position is covered by an invisible extent, do nothing (except maybe add some ellipses). #### The behavior of begin and end-glyphs at the edge of an invisible extent should be investigated further. This is fairly low priority though. */ if (data.ef->invisible) { /* #### Chuck, perhaps you could look at this code? I don't really know what I'm doing. */ if (*prop) { Dynarr_free (*prop); *prop = 0; } /* The extent fragment code only sets this when we should really display the ellipses. It makes sure the ellipses don't get displayed more than once in a row. */ if (data.ef->invisible_ellipses) { struct glyph_block gb; data.ef->invisible_ellipses_already_displayed = 1; data.ef->invisible_ellipses = 0; gb.extent = Qnil; gb.glyph = Vinvisible_text_glyph; *prop = add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, GLYPH_CACHEL (w, INVIS_GLYPH_INDEX)); /* Perhaps they shouldn't propagate if the very next thing is to display a newline (for compatibility with selective-display-ellipses)? Maybe that's too abstruse. */ if (*prop) goto done; } /* If point is in an invisible region we place it on the next visible character. */ if (data.cursor_type == CURSOR_ON && data.byte_charpos == data.byte_cursor_charpos) { data.cursor_type = NEXT_CURSOR; } /* #### What if we we're dealing with a display table? */ if (data.start_col) data.start_col--; if (data.byte_charpos == BYTE_BUF_ZV (b)) goto done; else INC_BYTEBPOS (b, data.byte_charpos); } /* If there is propagation data, then it represents the current buffer position being displayed. Add them and advance the position counter. This might also add the minibuffer prompt. */ else if (*prop) { dl->used_prop_data = 1; *prop = add_propagation_runes (prop, &data); if (*prop) goto done; /* gee, a really narrow window */ else if (data.byte_charpos == BYTE_BUF_ZV (b)) goto done; else if (data.byte_charpos < BYTE_BUF_BEGV (b)) /* #### urk urk urk! Aborts are not very fun! Fix this please! */ data.byte_charpos = BYTE_BUF_BEGV (b); else INC_BYTEBPOS (b, data.byte_charpos); } /* If there are end glyphs, add them to the line. These are the end glyphs for the previous run of text. We add them here rather than doing them at the end of handling the previous run so that glyphs at the beginning and end of a line are handled correctly. */ else if (Dynarr_length (data.ef->end_glyphs) > 0 || Dynarr_length (data.ef->begin_glyphs) > 0) { glyph_block_dynarr* tmpglyphs = 0; /* #### I think this is safe, but could be wrong. */ data.ch = BYTE_BUF_FETCH_CHAR (b, data.byte_charpos); if (Dynarr_length (data.ef->end_glyphs) > 0) { *prop = add_glyph_runes (&data, END_GLYPHS); tmpglyphs = data.ef->end_glyphs; } /* If there are begin glyphs, add them to the line. */ if (!*prop && Dynarr_length (data.ef->begin_glyphs) > 0) { *prop = add_glyph_runes (&data, BEGIN_GLYPHS); tmpglyphs = data.ef->begin_glyphs; } if (*prop) { /* If we just clipped a glyph and we are at the end of a line and there are more glyphs to display then do appropriate processing to not get a continuation glyph. */ if (*prop != ADD_FAILED && Dynarr_begin (*prop)->type == PROP_GLYPH && data.ch == '\n') { /* If there are no more glyphs then do the normal processing. #### This doesn't actually work if the same glyph is present more than once in the block. To solve this we would have to carry the index around which might be problematic since the fragment is recalculated for each line. */ if (EQ (Dynarr_lastp (tmpglyphs)->glyph, Dynarr_begin (*prop)->data.p_glyph.glyph)) { Dynarr_free (*prop); *prop = 0; } else { data.blank_width = DEVMETH (d, eol_cursor_width, ()); add_ichar_rune (&data); /* discard prop data. */ goto done; } } else goto done; } } /* If at end-of-buffer, we've already processed begin and end-glyphs at this point and there's no text to process, so we're done. */ else if (data.byte_charpos == BYTE_BUF_ZV (b)) goto done; else { Lisp_Object entry = Qnil; /* Get the character at the current buffer position. */ data.ch = BYTE_BUF_FETCH_CHAR (b, data.byte_charpos); if (!NILP (face_dt) || !NILP (window_dt)) entry = display_table_entry (data.ch, face_dt, window_dt); /* If there is a display table entry for it, hand it off to add_disp_table_entry_runes and let it worry about it. */ if (!NILP (entry) && !EQ (entry, make_char (data.ch))) { *prop = add_disp_table_entry_runes (&data, entry); if (*prop) goto done; } /* Check if we have hit a newline character. If so, add a marker to the line and end this loop. */ else if (data.ch == '\n') { /* Update the clearing face index unless the shrink property is set. -- dvl */ if ((data.findex > DEFAULT_INDEX) && ! WINDOW_FACE_CACHEL_SHRINK_P (w, data.findex)) dl->clear_findex = data.findex; /* We aren't going to be adding an end glyph so give its space back in order to make sure that the cursor can fit. */ data.max_pixpos += data.end_glyph_width; if (selective > 0 && (byte_spaces_at_point (b, next_bytebpos (b, data.byte_charpos)) >= selective)) { if (!NILP (b->selective_display_ellipses)) { struct glyph_block gb; gb.extent = Qnil; gb.glyph = Vinvisible_text_glyph; add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, GLYPH_CACHEL (w, INVIS_GLYPH_INDEX)); } else { /* Cheesy, cheesy, cheesy. We mark the end of the line with a special "character rune" whose width is the EOL cursor width and whose character is the non-printing character '\n'. */ data.blank_width = DEVMETH (d, eol_cursor_width, ()); *prop = add_ichar_rune (&data); } /* We need to set data.byte_charpos to the start of the next visible region in order to make this line appear to contain all of the invisible area. Otherwise, the line cache won't work correctly. */ INC_BYTEBPOS (b, data.byte_charpos); while (byte_spaces_at_point (b, data.byte_charpos) >= selective) { data.byte_charpos = byte_find_next_newline_no_quit (b, data.byte_charpos, 1); if (data.byte_charpos >= BYTE_BUF_ZV (b)) { data.byte_charpos = BYTE_BUF_ZV (b); break; } } if (BYTE_BUF_FETCH_CHAR (b, prev_bytebpos (b, data.byte_charpos)) == '\n') DEC_BYTEBPOS (b, data.byte_charpos); } else { data.blank_width = DEVMETH (d, eol_cursor_width, ()); *prop = add_ichar_rune (&data); } goto done; } /* If the current character is ^M, and selective display is enabled, then add the invisible-text-glyph if selective-display-ellipses is set. In any case, this line is done. */ else if (data.ch == (('M' & 037)) && selective == -1) { Bytebpos byte_next_charpos; /* Find the buffer position at the end of the line. */ byte_next_charpos = byte_find_next_newline_no_quit (b, data.byte_charpos, 1); if (BYTE_BUF_FETCH_CHAR (b, prev_bytebpos (b, byte_next_charpos)) == '\n') DEC_BYTEBPOS (b, byte_next_charpos); /* If the cursor is somewhere in the elided text make sure that the cursor gets drawn appropriately. */ if (data.cursor_type == CURSOR_ON && (data.byte_cursor_charpos >= data.byte_charpos && data.byte_cursor_charpos < byte_next_charpos)) { data.cursor_type = NEXT_CURSOR; } /* We won't be adding a truncation or continuation glyph so give up the room allocated for them. */ data.max_pixpos += data.end_glyph_width; if (!NILP (b->selective_display_ellipses)) { /* We don't propagate anything from the invisible text glyph if it fails to fit. This is intentional. */ struct glyph_block gb; gb.extent = Qnil; gb.glyph = Vinvisible_text_glyph; add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 1, GLYPH_CACHEL (w, INVIS_GLYPH_INDEX)); } /* Set the buffer position to the end of the line. We need to do this before potentially adding a newline so that the cursor flag will get set correctly (if needed). */ data.byte_charpos = byte_next_charpos; if (NILP (b->selective_display_ellipses) || data.byte_cursor_charpos == byte_next_charpos) { /* We have to at least add a newline character so that the cursor shows up properly. */ data.ch = '\n'; data.blank_width = DEVMETH (d, eol_cursor_width, ()); data.findex = DEFAULT_INDEX; data.start_col = 0; data.start_col_xoffset = 0; data.byte_start_col_enabled = 0; add_ichar_rune (&data); } /* This had better be a newline but doing it this way we'll see obvious incorrect results if it isn't. No need to abort here. */ data.ch = BYTE_BUF_FETCH_CHAR (b, data.byte_charpos); goto done; } /* If the current character is considered to be printable, then just add it. */ else if (data.ch >= printable_min) { *prop = add_ichar_rune (&data); if (*prop) goto done; } /* If the current character is a tab, determine the next tab starting position and add a blank rune which extends from the current pixel position to that starting position. */ else if (data.ch == '\t') { int tab_start_pixpos = data.pixpos; int next_tab_start; int char_tab_width; int prop_width = 0; if (data.start_col > 1) tab_start_pixpos -= (space_width (w) * (data.start_col - 1)) + data.start_col_xoffset; next_tab_start = next_tab_position (w, tab_start_pixpos, dl->bounds.left_in + data.hscroll_glyph_width_adjust); if (next_tab_start > data.max_pixpos) { prop_width = next_tab_start - data.max_pixpos; next_tab_start = data.max_pixpos; } data.blank_width = next_tab_start - data.pixpos; char_tab_width = (next_tab_start - tab_start_pixpos) / space_width (w); *prop = add_blank_rune (&data, w, char_tab_width); /* add_blank_rune is only supposed to be called with sizes guaranteed to fit in the available space. */ assert (!(*prop)); if (prop_width) { struct prop_block pb; *prop = Dynarr_new (prop_block); pb.type = PROP_BLANK; pb.data.p_blank.width = prop_width; pb.data.p_blank.findex = data.findex; Dynarr_add (*prop, pb); goto done; } } /* If character is a control character, pass it off to add_control_char_runes. The is_*() routines have undefined results on arguments outside of the range [-1, 255]. (This often bites people who carelessly use `char' instead of `unsigned char'.) */ else if (data.ch < 0x100 && iscntrl ((Ibyte) data.ch)) { *prop = add_control_char_runes (&data, b); if (*prop) goto done; } /* If the character is above the ASCII range and we have not already handled it, then print it as an octal number. */ else if (data.ch >= 0200) { *prop = add_octal_runes (&data); if (*prop) goto done; } /* Assume the current character is considered to be printable, then just add it. */ else { *prop = add_ichar_rune (&data); if (*prop) goto done; } INC_BYTEBPOS (b, data.byte_charpos); } } done: /* Determine the starting point of the next line if we did not hit the end of the buffer. */ if (data.byte_charpos < BYTE_BUF_ZV (b) && (active_minibuffer || !NILP (synch_minibuffers_value))) { /* #### This check is not correct. If the line terminated due to a begin-glyph or end-glyph hitting window-end, then data.ch will not point to the character at data.byte_charpos. If you make the two changes mentioned at the top of this loop, you should be able to say '(if (*prop))'. That should also make it possible to eliminate the data.byte_charpos < BYTE_BUF_ZV (b) check. */ /* The common case is that the line ended because we hit a newline. In that case, the next character is just the next buffer position. */ if (data.ch == '\n') { /* If data.start_col_enabled is still true, then the window is scrolled far enough so that nothing on this line is visible. We need to stick a truncation glyph at the beginning of the line in that case unless the line is completely blank. */ if (data.byte_start_col_enabled) { if (data.cursor_type == CURSOR_ON) { if (data.byte_cursor_charpos >= byte_start_pos && data.byte_cursor_charpos <= data.byte_charpos) data.byte_cursor_charpos = data.byte_charpos; } data.findex = DEFAULT_INDEX; data.start_col = 0; data.byte_start_col_enabled = 0; if (data.byte_charpos != byte_start_pos) { struct glyph_block gb; gb.extent = Qnil; gb.glyph = Vhscroll_glyph; add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, GLYPH_CACHEL (w, HSCROLL_GLYPH_INDEX)); } else { /* This duplicates code down below to add a newline to the end of an otherwise empty line.*/ data.ch = '\n'; data.blank_width = DEVMETH (d, eol_cursor_width, ()); add_ichar_rune (&data); } } INC_BYTEBPOS (b, data.byte_charpos); } /* Otherwise we have a buffer line which cannot fit on one display line. */ else { struct glyph_block gb; struct glyph_cachel *cachel; /* If the line is to be truncated then we actually have to look for the next newline. We also add the end-of-line glyph which we know will fit because we adjusted the right border before we starting laying out the line. */ data.max_pixpos += data.end_glyph_width; data.findex = DEFAULT_INDEX; gb.extent = Qnil; if (truncate_win) { Bytebpos byte_pos; /* Now find the start of the next line. */ byte_pos = byte_find_next_newline_no_quit (b, data.byte_charpos, 1); /* If the cursor is past the truncation line then we make it appear on the truncation glyph. If we've hit the end of the buffer then we also make the cursor appear unless eob is immediately preceded by a newline. In that case the cursor should actually appear on the next line. */ if (data.cursor_type == CURSOR_ON && data.byte_cursor_charpos >= data.byte_charpos && (data.byte_cursor_charpos < byte_pos || (byte_pos == BYTE_BUF_ZV (b) && (byte_pos == BYTE_BUF_BEGV (b) || (BYTE_BUF_FETCH_CHAR (b, prev_bytebpos (b, byte_pos)) != '\n'))))) data.byte_cursor_charpos = byte_pos; else data.cursor_type = NO_CURSOR; data.byte_charpos = byte_pos; gb.glyph = Vtruncation_glyph; cachel = GLYPH_CACHEL (w, TRUN_GLYPH_INDEX); } else { /* The cursor can never be on the continuation glyph. */ data.cursor_type = NO_CURSOR; /* data.byte_charpos is already at the start of the next line. */ dl->line_continuation = 1; gb.glyph = Vcontinuation_glyph; cachel = GLYPH_CACHEL (w, CONT_GLYPH_INDEX); } add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, cachel); if (truncate_win && data.byte_charpos == BYTE_BUF_ZV (b) && BYTE_BUF_FETCH_CHAR (b, prev_bytebpos (b, BYTE_BUF_ZV (b))) != '\n') /* #### Damn this losing shit. */ data.byte_charpos++; } } else if ((active_minibuffer || !NILP (synch_minibuffers_value)) && (!echo_area_active (f) || data.byte_charpos == BYTE_BUF_ZV (b))) { /* We need to add a marker to the end of the line since there is no newline character in order for the cursor to get drawn. We label it as a newline so that it gets handled correctly by the whitespace routines below. */ data.ch = '\n'; data.blank_width = DEVMETH (d, eol_cursor_width, ()); data.findex = DEFAULT_INDEX; data.start_col = 0; data.start_col_xoffset = 0; data.byte_start_col_enabled = 0; data.max_pixpos += data.blank_width; add_ichar_rune (&data); data.max_pixpos -= data.blank_width; /* #### urk! Chuck, this shit is bad news. Going around manipulating invalid positions is guaranteed to result in trouble sooner or later. */ data.byte_charpos = BYTE_BUF_ZV (b) + 1; } /* Calculate left whitespace boundary. */ { int elt = 0; /* Whitespace past a newline is considered right whitespace. */ while (elt < Dynarr_length (db->runes)) { struct rune *rb = Dynarr_atp (db->runes, elt); if ((rb->type == RUNE_CHAR && rb->object.chr.ch == ' ') || rb->type == RUNE_BLANK) { dl->bounds.left_white += rb->width; elt++; } else elt = Dynarr_length (db->runes); } } /* Calculate right whitespace boundary. */ { int elt = Dynarr_length (db->runes) - 1; int done = 0; while (!done && elt >= 0) { struct rune *rb = Dynarr_atp (db->runes, elt); if (!(rb->type == RUNE_CHAR && rb->object.chr.ch < 0x100 && isspace (rb->object.chr.ch)) && !rb->type == RUNE_BLANK) { dl->bounds.right_white = rb->xpos + rb->width; done = 1; } elt--; } /* The line is blank so everything is considered to be right whitespace. */ if (!done) dl->bounds.right_white = dl->bounds.left_in; } /* Set the display blocks bounds. */ db->start_pos = dl->bounds.left_in; if (Dynarr_length (db->runes)) { struct rune *rb = Dynarr_lastp (db->runes); db->end_pos = rb->xpos + rb->width; } else db->end_pos = dl->bounds.right_white; calculate_baseline (&data); dl->ascent = data.new_ascent; dl->descent = data.new_descent; { unsigned short ascent = (unsigned short) XFIXNUM (w->minimum_line_ascent); if (dl->ascent < ascent) dl->ascent = ascent; } { unsigned short descent = (unsigned short) XFIXNUM (w->minimum_line_descent); if (dl->descent < descent) dl->descent = descent; } calculate_yoffset (dl, db); dl->cursor_elt = data.cursor_x; /* #### lossage lossage lossage! Fix this shit! */ if (data.byte_charpos > BYTE_BUF_ZV (b)) dl->end_charpos = BUF_ZV (b); else dl->end_charpos = bytebpos_to_charbpos (b, data.byte_charpos) - 1; if (truncate_win) data.dl->num_chars = column_at_point (b, dl->end_charpos, 0); else /* This doesn't correctly take into account tabs and control characters but if the window isn't being truncated then this value isn't going to end up being used anyhow. */ data.dl->num_chars = dl->end_charpos - dl->charpos; /* #### handle horizontally scrolled line with text none of which was actually laid out. */ /* #### handle any remainder of overlay arrow */ if (*prop == ADD_FAILED) *prop = NULL; if (truncate_win && *prop) { Dynarr_free (*prop); *prop = NULL; } extent_fragment_delete (data.ef); /* #### If we started at EOB, then make sure we return a value past it so that regenerate_window will exit properly. This is bogus. The main loop should get fixed so that it isn't necessary to call this function if we are already at EOB. */ if (data.byte_charpos == BYTE_BUF_ZV (b) && byte_start_pos == BYTE_BUF_ZV (b)) return data.byte_charpos + 1; /* Yuck! */ else return data.byte_charpos; } /* Display the overlay arrow at the beginning of the given line. */ static int create_overlay_glyph_block (struct window *w, struct display_line *dl) { struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); pos_data data; /* If Voverlay_arrow_string isn't valid then just fail silently. */ if (!STRINGP (Voverlay_arrow_string) && !GLYPHP (Voverlay_arrow_string)) return 0; xzero (data); data.ef = NULL; data.d = d; data.window = wrap_window (w); data.db = get_display_block_from_line (dl, OVERWRITE); data.dl = dl; data.pixpos = dl->bounds.left_in; data.max_pixpos = dl->bounds.right_in; data.cursor_type = NO_CURSOR; data.cursor_x = -1; data.findex = DEFAULT_INDEX; data.last_charset = Qunbound; data.last_findex = DEFAULT_INDEX; data.result_str = Qnil; data.string = Qnil; Dynarr_reset (data.db->runes); if (STRINGP (Voverlay_arrow_string)) { add_ibyte_string_runes (&data, XSTRING_DATA (Voverlay_arrow_string), XSTRING_LENGTH (Voverlay_arrow_string), 1, 0); } else if (GLYPHP (Voverlay_arrow_string)) { struct glyph_block gb; gb.glyph = Voverlay_arrow_string; gb.extent = Qnil; add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, 0); } if (data.max_pixmap_height) { int height = data.new_ascent + data.new_descent; int pix_ascent, pix_descent; pix_descent = data.max_pixmap_height * data.new_descent / height; pix_ascent = data.max_pixmap_height - pix_descent; calculate_baseline (&data); data.new_ascent = max (data.new_ascent, pix_ascent); data.new_descent = max (data.new_descent, pix_descent); } dl->ascent = data.new_ascent; dl->descent = data.new_descent; data.db->start_pos = dl->bounds.left_in; data.db->end_pos = data.pixpos; calculate_yoffset (dl, data.db); return data.pixpos - dl->bounds.left_in; } /* Add a type of glyph to a margin display block. */ static int add_margin_runes (struct display_line *dl, struct display_block *db, int start, int count, enum glyph_layout layout, int side, Lisp_Object window) { glyph_block_dynarr *gbd = (side == LEFT_GLYPHS ? dl->left_glyphs : dl->right_glyphs); int elt, end; int reverse; struct window *w = XWINDOW (window); struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); pos_data data; xzero (data); data.d = d; data.window = window; data.db = db; data.dl = dl; data.pixpos = start; data.cursor_type = NO_CURSOR; data.cursor_x = -1; data.last_charset = Qunbound; data.last_findex = DEFAULT_INDEX; data.result_str = Qnil; data.string = Qnil; data.new_ascent = dl->ascent; data.new_descent = dl->descent; if ((layout == GL_WHITESPACE && side == LEFT_GLYPHS) || (layout == GL_INSIDE_MARGIN && side == RIGHT_GLYPHS)) { reverse = 1; elt = Dynarr_length (gbd) - 1; end = 0; } else { reverse = 0; elt = 0; end = Dynarr_length (gbd); } while (count && ((!reverse && elt < end) || (reverse && elt >= end))) { struct glyph_block *gb = Dynarr_atp (gbd, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (gb->active && ((side == LEFT_GLYPHS && extent_begin_glyph_layout (XEXTENT (gb->extent)) == layout) || (side == RIGHT_GLYPHS && extent_end_glyph_layout (XEXTENT (gb->extent)) == layout))) { data.findex = gb->findex; data.max_pixpos = data.pixpos + gb->width; add_glyph_rune (&data, gb, side, 0, NULL); count--; gb->active = 0; } (reverse ? elt-- : elt++); } calculate_baseline (&data); dl->ascent = data.new_ascent; dl->descent = data.new_descent; calculate_yoffset (dl, data.db); return data.pixpos; } /* Add a blank to a margin display block. */ static void add_margin_blank (struct display_line *UNUSED (dl), struct display_block *db, struct window *w, int xpos, int width, int side) { struct rune rb; rb.findex = (side == LEFT_GLYPHS ? get_builtin_face_cache_index (w, Vleft_margin_face) : get_builtin_face_cache_index (w, Vright_margin_face)); rb.xpos = xpos; rb.width = width; rb.charpos = -1; rb.endpos = 0; rb.type = RUNE_BLANK; rb.cursor_type = CURSOR_OFF; Dynarr_add (db->runes, rb); } /* Display glyphs in the left outside margin, left inside margin and left whitespace area. */ static void create_left_glyph_block (struct window *w, struct display_line *dl, int overlay_width) { Lisp_Object window; int use_overflow = (NILP (w->use_left_overflow) ? 0 : 1); int elt, end_xpos; int out_end, in_out_start, in_in_end, white_out_start, white_in_start; int out_cnt, in_out_cnt, in_in_cnt, white_out_cnt, white_in_cnt; int left_in_start = dl->bounds.left_in; int left_in_end = dl->bounds.left_in + overlay_width; struct display_block *odb, *idb; window = wrap_window (w); /* We have to add the glyphs to the line in the order outside, inside, whitespace. However the precedence dictates that we determine how many will fit in the reverse order. */ /* Determine how many whitespace glyphs we can display and where they should start. */ white_in_start = dl->bounds.left_white; white_out_start = left_in_start; white_out_cnt = white_in_cnt = 0; elt = 0; while (elt < Dynarr_length (dl->left_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->left_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_begin_glyph_layout (XEXTENT (gb->extent)) == GL_WHITESPACE) { int width; width = glyph_width (gb->glyph, window); if (white_in_start - width >= left_in_end) { white_in_cnt++; white_in_start -= width; gb->width = width; gb->active = 1; } else if (use_overflow && (white_out_start - width > dl->bounds.left_out)) { white_out_cnt++; white_out_start -= width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } /* Determine how many inside margin glyphs we can display and where they should start. The inside margin glyphs get whatever space is left after the whitespace glyphs have been displayed. These are tricky to calculate since if we decide to use the overflow area we basically have to start over. So for these we build up a list of just the inside margin glyphs and manipulate it to determine the needed info. */ { glyph_block_dynarr *ib; int avail_in, avail_out; int done = 0; int marker = 0; int used_in, used_out; elt = 0; used_in = used_out = 0; ib = Dynarr_new (glyph_block); while (elt < Dynarr_length (dl->left_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->left_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_begin_glyph_layout (XEXTENT (gb->extent)) == GL_INSIDE_MARGIN) { gb->width = glyph_width (gb->glyph, window); used_in += gb->width; Dynarr_add (ib, *gb); } elt++; } if (white_out_cnt) avail_in = 0; else { avail_in = white_in_start - left_in_end; if (avail_in < 0) avail_in = 0; } if (!use_overflow) avail_out = 0; else avail_out = white_out_start - dl->bounds.left_out; marker = 0; while (!done && marker < Dynarr_length (ib)) { int width = Dynarr_atp (ib, marker)->width; /* If everything now fits in the available inside margin space, we're done. */ if (used_in <= avail_in) done = 1; else { /* Otherwise see if we have room to move a glyph to the outside. */ if (used_out + width <= avail_out) { used_out += width; used_in -= width; } else done = 1; } if (!done) marker++; } /* At this point we now know that everything from marker on goes in the inside margin and everything before it goes in the outside margin. The stuff going into the outside margin is guaranteed to fit, but we may have to trim some stuff from the inside. */ in_in_end = left_in_end; in_out_start = white_out_start; in_out_cnt = in_in_cnt = 0; Dynarr_free (ib); elt = 0; while (elt < Dynarr_length (dl->left_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->left_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_begin_glyph_layout (XEXTENT (gb->extent)) == GL_INSIDE_MARGIN) { int width = glyph_width (gb->glyph, window); if (used_out) { in_out_cnt++; in_out_start -= width; gb->width = width; gb->active = 1; used_out -= width; } else if (in_in_end + width < white_in_start) { in_in_cnt++; in_in_end += width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } } /* Determine how many outside margin glyphs we can display. They always start at the left outside margin and can only use the outside margin space. */ out_end = dl->bounds.left_out; out_cnt = 0; elt = 0; while (elt < Dynarr_length (dl->left_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->left_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_begin_glyph_layout (XEXTENT (gb->extent)) == GL_OUTSIDE_MARGIN) { int width = glyph_width (gb->glyph, window); if (out_end + width <= in_out_start) { out_cnt++; out_end += width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } /* Now that we know where everything goes, we add the glyphs as runes to the appropriate display blocks. */ if (out_cnt || in_out_cnt || white_out_cnt) { odb = get_display_block_from_line (dl, LEFT_OUTSIDE_MARGIN); odb->start_pos = dl->bounds.left_out; /* #### We should stop adding a blank to account for the space between the end of the glyphs and the margin and instead set this accordingly. */ odb->end_pos = dl->bounds.left_in; Dynarr_reset (odb->runes); } else odb = 0; if (in_in_cnt || white_in_cnt) { idb = get_display_block_from_line (dl, LEFT_INSIDE_MARGIN); idb->start_pos = dl->bounds.left_in; /* #### See above comment for odb->end_pos */ idb->end_pos = dl->bounds.left_white; Dynarr_reset (idb->runes); } else idb = 0; /* First add the outside margin glyphs. */ if (out_cnt) end_xpos = add_margin_runes (dl, odb, dl->bounds.left_out, out_cnt, GL_OUTSIDE_MARGIN, LEFT_GLYPHS, window); else end_xpos = dl->bounds.left_out; /* There may be blank space between the outside margin glyphs and the inside margin glyphs. If so, add a blank. */ if (in_out_cnt && (in_out_start - end_xpos)) { add_margin_blank (dl, odb, w, end_xpos, in_out_start - end_xpos, LEFT_GLYPHS); } /* Next add the inside margin glyphs which are actually in the outside margin. */ if (in_out_cnt) { end_xpos = add_margin_runes (dl, odb, in_out_start, in_out_cnt, GL_INSIDE_MARGIN, LEFT_GLYPHS, window); } /* If we didn't add any inside margin glyphs to the outside margin, but are adding whitespace glyphs, then we need to add a blank here. */ if (!in_out_cnt && white_out_cnt && (white_out_start - end_xpos)) { add_margin_blank (dl, odb, w, end_xpos, white_out_start - end_xpos, LEFT_GLYPHS); } /* Next add the whitespace margin glyphs which are actually in the outside margin. */ if (white_out_cnt) { end_xpos = add_margin_runes (dl, odb, white_out_start, white_out_cnt, GL_WHITESPACE, LEFT_GLYPHS, window); } /* We take care of clearing between the end of the glyphs and the start of the inside margin for lines which have glyphs. */ if (odb && (left_in_start - end_xpos)) { add_margin_blank (dl, odb, w, end_xpos, left_in_start - end_xpos, LEFT_GLYPHS); } /* Next add the inside margin glyphs which are actually in the inside margin. */ if (in_in_cnt) { end_xpos = add_margin_runes (dl, idb, left_in_end, in_in_cnt, GL_INSIDE_MARGIN, LEFT_GLYPHS, window); } else end_xpos = left_in_end; /* Make sure that the area between the end of the inside margin glyphs and the whitespace glyphs is cleared. */ if (idb && (white_in_start - end_xpos > 0)) { add_margin_blank (dl, idb, w, end_xpos, white_in_start - end_xpos, LEFT_GLYPHS); } /* Next add the whitespace margin glyphs which are actually in the inside margin. */ if (white_in_cnt) { add_margin_runes (dl, idb, white_in_start, white_in_cnt, GL_WHITESPACE, LEFT_GLYPHS, window); } /* Whitespace glyphs always end right next to the text block so there is nothing we have to make sure is cleared after them. */ } /* Display glyphs in the right outside margin, right inside margin and right whitespace area. */ static void create_right_glyph_block (struct window *w, struct display_line *dl) { Lisp_Object window; int use_overflow = (NILP (w->use_right_overflow) ? 0 : 1); int elt, end_xpos; int out_start, in_out_end, in_in_start, white_out_end, white_in_end; int out_cnt, in_out_cnt, in_in_cnt, white_out_cnt, white_in_cnt; struct display_block *odb, *idb; window = wrap_window (w); /* We have to add the glyphs to the line in the order outside, inside, whitespace. However the precedence dictates that we determine how many will fit in the reverse order. */ /* Determine how many whitespace glyphs we can display and where they should start. */ white_in_end = dl->bounds.right_white; white_out_end = dl->bounds.right_in; white_out_cnt = white_in_cnt = 0; elt = 0; while (elt < Dynarr_length (dl->right_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->right_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_end_glyph_layout (XEXTENT (gb->extent)) == GL_WHITESPACE) { int width = glyph_width (gb->glyph, window); if (white_in_end + width <= dl->bounds.right_in) { white_in_cnt++; white_in_end += width; gb->width = width; gb->active = 1; } else if (use_overflow && (white_out_end + width <= dl->bounds.right_out)) { white_out_cnt++; white_out_end += width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } /* Determine how many inside margin glyphs we can display and where they should start. The inside margin glyphs get whatever space is left after the whitespace glyphs have been displayed. These are tricky to calculate since if we decide to use the overflow area we basically have to start over. So for these we build up a list of just the inside margin glyphs and manipulate it to determine the needed info. */ { glyph_block_dynarr *ib; int avail_in, avail_out; int done = 0; int marker = 0; int used_in, used_out; elt = 0; used_in = used_out = 0; ib = Dynarr_new (glyph_block); while (elt < Dynarr_length (dl->right_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->right_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_end_glyph_layout (XEXTENT (gb->extent)) == GL_INSIDE_MARGIN) { gb->width = glyph_width (gb->glyph, window); used_in += gb->width; Dynarr_add (ib, *gb); } elt++; } if (white_out_cnt) avail_in = 0; else avail_in = dl->bounds.right_in - white_in_end; if (!use_overflow) avail_out = 0; else avail_out = dl->bounds.right_out - white_out_end; marker = 0; while (!done && marker < Dynarr_length (ib)) { int width = Dynarr_atp (ib, marker)->width; /* If everything now fits in the available inside margin space, we're done. */ if (used_in <= avail_in) done = 1; else { /* Otherwise see if we have room to move a glyph to the outside. */ if (used_out + width <= avail_out) { used_out += width; used_in -= width; } else done = 1; } if (!done) marker++; } /* At this point we now know that everything from marker on goes in the inside margin and everything before it goes in the outside margin. The stuff going into the outside margin is guaranteed to fit, but we may have to trim some stuff from the inside. */ in_in_start = dl->bounds.right_in; in_out_end = dl->bounds.right_in; in_out_cnt = in_in_cnt = 0; Dynarr_free (ib); elt = 0; while (elt < Dynarr_length (dl->right_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->right_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_end_glyph_layout (XEXTENT (gb->extent)) == GL_INSIDE_MARGIN) { int width = glyph_width (gb->glyph, window); if (used_out) { in_out_cnt++; in_out_end += width; gb->width = width; gb->active = 1; used_out -= width; } else if (in_in_start - width >= white_in_end) { in_in_cnt++; in_in_start -= width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } } /* Determine how many outside margin glyphs we can display. They always start at the right outside margin and can only use the outside margin space. */ out_start = dl->bounds.right_out; out_cnt = 0; elt = 0; while (elt < Dynarr_length (dl->right_glyphs)) { struct glyph_block *gb = Dynarr_atp (dl->right_glyphs, elt); if (NILP (gb->extent)) ABORT (); /* these should have been handled in add_glyph_rune */ if (extent_end_glyph_layout (XEXTENT (gb->extent)) == GL_OUTSIDE_MARGIN) { int width = glyph_width (gb->glyph, window); if (out_start - width >= in_out_end) { out_cnt++; out_start -= width; gb->width = width; gb->active = 1; } else gb->active = 0; } elt++; } /* Now that we now where everything goes, we add the glyphs as runes to the appropriate display blocks. */ if (out_cnt || in_out_cnt || white_out_cnt) { odb = get_display_block_from_line (dl, RIGHT_OUTSIDE_MARGIN); /* #### See comments before odb->start_pos init in create_left_glyph_block */ odb->start_pos = dl->bounds.right_in; odb->end_pos = dl->bounds.right_out; Dynarr_reset (odb->runes); } else odb = 0; if (in_in_cnt || white_in_cnt) { idb = get_display_block_from_line (dl, RIGHT_INSIDE_MARGIN); idb->start_pos = dl->bounds.right_white; /* #### See comments before odb->start_pos init in create_left_glyph_block */ idb->end_pos = dl->bounds.right_in; Dynarr_reset (idb->runes); } else idb = 0; /* First add the whitespace margin glyphs which are actually in the inside margin. */ if (white_in_cnt) { end_xpos = add_margin_runes (dl, idb, dl->bounds.right_white, white_in_cnt, GL_WHITESPACE, RIGHT_GLYPHS, window); } else end_xpos = dl->bounds.right_white; /* Make sure that the area between the end of the whitespace glyphs and the inside margin glyphs is cleared. */ if (in_in_cnt && (in_in_start - end_xpos)) { add_margin_blank (dl, idb, w, end_xpos, in_in_start - end_xpos, RIGHT_GLYPHS); } /* Next add the inside margin glyphs which are actually in the inside margin. */ if (in_in_cnt) { end_xpos = add_margin_runes (dl, idb, in_in_start, in_in_cnt, GL_INSIDE_MARGIN, RIGHT_GLYPHS, window); } /* If we didn't add any inside margin glyphs then make sure the rest of the inside margin area gets cleared. */ if (idb && (dl->bounds.right_in - end_xpos)) { add_margin_blank (dl, idb, w, end_xpos, dl->bounds.right_in - end_xpos, RIGHT_GLYPHS); } /* Next add any whitespace glyphs in the outside margin. */ if (white_out_cnt) { end_xpos = add_margin_runes (dl, odb, dl->bounds.right_in, white_out_cnt, GL_WHITESPACE, RIGHT_GLYPHS, window); } else end_xpos = dl->bounds.right_in; /* Next add any inside margin glyphs in the outside margin. */ if (in_out_cnt) { end_xpos = add_margin_runes (dl, odb, end_xpos, in_out_cnt, GL_INSIDE_MARGIN, RIGHT_GLYPHS, window); } /* There may be space between any whitespace or inside margin glyphs in the outside margin and the actual outside margin glyphs. */ if (odb && (out_start - end_xpos)) { add_margin_blank (dl, odb, w, end_xpos, out_start - end_xpos, RIGHT_GLYPHS); } /* Finally, add the outside margin glyphs. */ if (out_cnt) { add_margin_runes (dl, odb, out_start, out_cnt, GL_OUTSIDE_MARGIN, RIGHT_GLYPHS, window); } } /***************************************************************************/ /* */ /* modeline routines */ /* */ /***************************************************************************/ /* This function is also used in frame.c by `generate_title_string' */ void generate_formatted_string_db (Lisp_Object format_str, Lisp_Object result_str, struct window *w, struct display_line *dl, struct display_block *db, face_index findex, int min_pixpos, int max_pixpos, int type) { struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); pos_data data; int c_pixpos; Charcount offset = 0; xzero (data); data.d = d; data.db = db; data.dl = dl; data.findex = findex; data.pixpos = min_pixpos; data.max_pixpos = max_pixpos; data.cursor_type = NO_CURSOR; data.last_charset = Qunbound; data.last_findex = DEFAULT_INDEX; data.result_str = result_str; data.is_modeline = 1; data.string = Qnil; data.window = wrap_window (w); Dynarr_reset (formatted_string_extent_dynarr); Dynarr_reset (formatted_string_extent_start_dynarr); Dynarr_reset (formatted_string_extent_end_dynarr); /* result_str is nil when we're building a frame or icon title. Otherwise, we're building a modeline, so the offset starts at the modeline horizontal scrolling amount */ if (! NILP (result_str)) offset = w->modeline_hscroll; generate_fstring_runes (w, &data, 0, 0, -1, format_str, 0, max_pixpos - min_pixpos, findex, type, &offset, Qnil); if (Dynarr_length (db->runes)) { struct rune *rb = Dynarr_lastp (db->runes); c_pixpos = rb->xpos + rb->width; } else c_pixpos = min_pixpos; /* If we don't reach the right side of the window, add a blank rune to make up the difference. This usually only occurs if the modeline face is using a proportional width font or a fixed width font of a different size from the default face font. */ if (c_pixpos < max_pixpos) { data.pixpos = c_pixpos; data.blank_width = max_pixpos - data.pixpos; add_blank_rune (&data, NULL, 0); } /* Now create the result string and frob the extents into it. */ if (!NILP (result_str)) { int elt; Bytecount len; Ibyte *strdata; struct buffer *buf = XBUFFER (WINDOW_BUFFER (w)); in_modeline_generation = 1; sledgehammer_check_ascii_begin (result_str); detach_all_extents (result_str); resize_string (result_str, -1, data.bytepos - XSTRING_LENGTH (result_str)); strdata = XSTRING_DATA (result_str); for (elt = 0, len = 0; elt < Dynarr_length (db->runes); elt++) { if (Dynarr_atp (db->runes, elt)->type == RUNE_CHAR) { len += (set_itext_ichar (strdata + len, Dynarr_atp (db->runes, elt)->object.chr.ch)); } } init_string_ascii_begin (result_str); bump_string_modiff (result_str); sledgehammer_check_ascii_begin (result_str); for (elt = 0; elt < Dynarr_length (formatted_string_extent_dynarr); elt++) { Lisp_Object extent = Qnil; Lisp_Object child; extent = wrap_extent (Dynarr_at (formatted_string_extent_dynarr, elt)); child = Fgethash (extent, buf->modeline_extent_table, Qnil); if (NILP (child)) { child = Fmake_extent (Qnil, Qnil, result_str); Fputhash (extent, child, buf->modeline_extent_table); } Fset_extent_parent (child, extent); set_extent_endpoints (XEXTENT (child), Dynarr_at (formatted_string_extent_start_dynarr, elt), Dynarr_at (formatted_string_extent_end_dynarr, elt), result_str); } in_modeline_generation = 0; } } /* Ensure that the given display line DL accurately represents the modeline for the given window. */ static void generate_modeline (struct window *w, struct display_line *dl, int type) { struct buffer *b = XBUFFER (w->buffer); struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); /* Unlike display line and rune pointers, this one can't change underneath our feet. */ struct display_block *db = get_display_block_from_line (dl, TEXT); int max_pixpos, min_pixpos, ypos_adj; Lisp_Object font_inst; /* This will actually determine incorrect inside boundaries for the modeline since it ignores the margins. However being aware of this fact we never use those values anywhere so it doesn't matter. */ dl->bounds = calculate_display_line_boundaries (w, 1); /* We are generating a modeline. */ dl->modeline = 1; dl->cursor_elt = -1; /* Reset the runes on the modeline. */ Dynarr_reset (db->runes); if (!WINDOW_HAS_MODELINE_P (w)) { struct rune rb; /* If there is a horizontal scrollbar, don't add anything. */ if (window_scrollbar_height (w)) return; dl->ascent = DEVMETH (d, divider_height, ()); dl->descent = 0; /* The modeline is at the bottom of the gutters. */ dl->ypos = WINDOW_BOTTOM (w); rb.findex = MODELINE_INDEX; rb.xpos = dl->bounds.left_out; rb.width = dl->bounds.right_out - dl->bounds.left_out; rb.charpos = 0; rb.endpos = 0; rb.type = RUNE_HLINE; rb.object.hline.thickness = 1; rb.object.hline.yoffset = 0; rb.cursor_type = NO_CURSOR; if (!EQ (Qzero, w->modeline_shadow_thickness) && FRAME_WIN_P (f)) { int shadow_thickness = MODELINE_SHADOW_THICKNESS (w); dl->ypos -= shadow_thickness; rb.xpos += shadow_thickness; rb.width -= 2 * shadow_thickness; } Dynarr_add (db->runes, rb); return; } /* !!#### not right; needs to compute the max height of all the charsets */ font_inst = WINDOW_FACE_CACHEL_FONT (w, MODELINE_INDEX, Vcharset_ascii); dl->ascent = XFONT_INSTANCE (font_inst)->ascent; dl->descent = XFONT_INSTANCE (font_inst)->descent; min_pixpos = dl->bounds.left_out; max_pixpos = dl->bounds.right_out; if (!EQ (Qzero, w->modeline_shadow_thickness) && FRAME_WIN_P (f)) { int shadow_thickness = MODELINE_SHADOW_THICKNESS (w); ypos_adj = shadow_thickness; min_pixpos += shadow_thickness; max_pixpos -= shadow_thickness; } else ypos_adj = 0; generate_formatted_string_db (b->modeline_format, b->generated_modeline_string, w, dl, db, MODELINE_INDEX, min_pixpos, max_pixpos, type); /* The modeline is at the bottom of the gutters. We have to wait to set this until we've generated the modeline in order to account for any embedded faces. */ dl->ypos = WINDOW_BOTTOM (w) - dl->descent - ypos_adj; } static Charcount add_string_to_fstring_db_runes (pos_data *data, const Ibyte *str, Charcount pos, Charcount min_pos, Charcount max_pos) { /* This function has been Mule-ized. */ Charcount end; const Ibyte *cur_pos = str; struct display_block *db = data->db; data->blank_width = space_width (XWINDOW (data->window)); while (Dynarr_length (db->runes) < pos) add_blank_rune (data, NULL, 0); end = (Dynarr_length (db->runes) + bytecount_to_charcount (str, qxestrlen (str))); if (max_pos != -1) end = min (max_pos, end); while (pos < end && *cur_pos) { const Ibyte *old_cur_pos = cur_pos; int succeeded; data->ch = itext_ichar (cur_pos); succeeded = (add_ichar_rune (data) != ADD_FAILED); INC_IBYTEPTR (cur_pos); if (succeeded) { pos++; data->modeline_charpos++; data->bytepos += cur_pos - old_cur_pos; } } while (Dynarr_length (db->runes) < min_pos && (data->pixpos + data->blank_width <= data->max_pixpos)) add_blank_rune (data, NULL, 0); return Dynarr_length (db->runes); } /* #### Urk! Should also handle begin-glyphs and end-glyphs in modeline extents. */ static Charcount add_glyph_to_fstring_db_runes (pos_data *data, Lisp_Object glyph, Charcount pos, Charcount UNUSED (min_pos), Charcount max_pos, Lisp_Object extent) { /* This function has been Mule-ized. */ Charcount end; struct display_block *db = data->db; struct glyph_block gb; data->blank_width = space_width (XWINDOW (data->window)); while (Dynarr_length (db->runes) < pos) add_blank_rune (data, NULL, 0); end = Dynarr_length (db->runes) + 1; if (max_pos != -1) end = min (max_pos, end); gb.glyph = glyph; gb.extent = extent; add_glyph_rune (data, &gb, BEGIN_GLYPHS, 0, 0); pos++; while (Dynarr_length (db->runes) < pos && (data->pixpos + data->blank_width <= data->max_pixpos)) add_blank_rune (data, NULL, 0); return Dynarr_length (db->runes); } /* If max_pos is == -1, it is considered to be infinite. The same is true of max_pixsize. */ #define SET_CURRENT_MODE_CHARS_PIXSIZE \ if (Dynarr_length (data->db->runes)) \ cur_pixsize = data->pixpos - Dynarr_begin (data->db->runes)->xpos; \ else \ cur_pixsize = 0; /* Note that this function does "positions" in terms of characters and not in terms of columns. This is necessary to make the formatting work correctly when proportional width fonts are used in the modeline. */ static Charcount generate_fstring_runes (struct window *w, pos_data *data, Charcount pos, Charcount min_pos, Charcount max_pos, Lisp_Object elt, int depth, int max_pixsize, face_index findex, int type, Charcount *offset, Lisp_Object cur_ext) { /* This function has been Mule-ized. */ /* #### The other losing things in this function are: -- C zero-terminated-string lossage. -- Non-printable characters should be converted into something appropriate (e.g. ^F) instead of blindly being printed anyway. */ tail_recurse: if (depth > 10) goto invalid; depth++; if (STRINGP (elt)) { /* A string. Add to the display line and check for %-constructs within it. */ Ibyte *this_str = XSTRING_DATA (elt); while ((pos < max_pos || max_pos == -1) && *this_str) { Ibyte *last = this_str; while (*this_str && *this_str != '%') this_str++; if (this_str != last) { /* No %-construct */ Charcount size = bytecount_to_charcount (last, this_str - last); if (size <= *offset) *offset -= size; else { Charcount tmp_max = (max_pos == -1 ? pos + size - *offset : min (pos + size - *offset, max_pos)); const Ibyte *tmp_last = itext_n_addr (last, *offset); pos = add_string_to_fstring_db_runes (data, tmp_last, pos, pos, tmp_max); *offset = 0; } } else /* *this_str == '%' */ { Charcount spec_width = 0; this_str++; /* skip over '%' */ /* We can't allow -ve args due to the "%-" construct. * Argument specifies minwidth but not maxwidth * (maxwidth can be specified by * ( . ) modeline elements) */ while (isdigit (*this_str)) { spec_width = spec_width * 10 + (*this_str - '0'); this_str++; } spec_width += pos; if (*this_str == 'M') { pos = generate_fstring_runes (w, data, pos, spec_width, max_pos, Vglobal_mode_string, depth, max_pixsize, findex, type, offset, cur_ext); } else if (*this_str == '-') { Charcount num_to_add; if (max_pixsize < 0) num_to_add = 0; else if (max_pos != -1) num_to_add = max_pos - pos; else { int cur_pixsize; int dash_pixsize; Ibyte ch = '-'; SET_CURRENT_MODE_CHARS_PIXSIZE; dash_pixsize = redisplay_window_text_width_string (w, findex, &ch, Qnil, 0, 1); if (dash_pixsize == 0) num_to_add = 0; else { num_to_add = (max_pixsize - cur_pixsize) / dash_pixsize; num_to_add++; } } while (num_to_add--) pos = add_string_to_fstring_db_runes (data, (const Ibyte *) "-", pos, pos, max_pos); } else if (*this_str != 0) { Ichar ch = itext_ichar (this_str); Ibyte *str; Charcount size; decode_mode_spec (w, ch, type); str = Dynarr_begin (mode_spec_ibyte_string); size = bytecount_to_charcount /* Skip the null character added by `decode_mode_spec' */ (str, Dynarr_length (mode_spec_ibyte_string)) - 1; if (size <= *offset) *offset -= size; else { const Ibyte *tmp_str = itext_n_addr (str, *offset); /* #### NOTE: I don't understand why a tmp_max is not computed and used here as in the plain string case above. -- dv */ pos = add_string_to_fstring_db_runes (data, tmp_str, pos, pos, max_pos); *offset = 0; } } /* NOT this_str++. There could be any sort of character at the current position. */ INC_IBYTEPTR (this_str); } if (max_pixsize > 0) { int cur_pixsize; SET_CURRENT_MODE_CHARS_PIXSIZE; if (cur_pixsize >= max_pixsize) break; } } } else if (SYMBOLP (elt)) { /* A symbol: process the value of the symbol recursively as if it appeared here directly. */ Lisp_Object tem = symbol_value_in_buffer (elt, w->buffer); if (!UNBOUNDP (tem)) { /* If value is a string, output that string literally: don't check for % within it. */ if (STRINGP (tem)) { Ibyte *str = XSTRING_DATA (tem); Charcount size = string_char_length (tem); if (size <= *offset) *offset -= size; else { const Ibyte *tmp_str = itext_n_addr (str, *offset); /* #### NOTE: I don't understand why a tmp_max is not computed and used here as in the plain string case above. -- dv */ pos = add_string_to_fstring_db_runes (data, tmp_str, pos, min_pos, max_pos); *offset = 0; } } /* Give up right away for nil or t. */ else if (!EQ (tem, elt)) { elt = tem; goto tail_recurse; } } } else if (GENERIC_SPECIFIERP (elt)) { Lisp_Object window, tem; window = wrap_window (w); tem = specifier_instance_no_quit (elt, Qunbound, window, ERROR_ME_DEBUG_WARN, 0, Qzero); if (!UNBOUNDP (tem)) { elt = tem; goto tail_recurse; } } else if (CONSP (elt)) { /* A cons cell: four distinct cases. * - If first element is a string or a cons, process all the elements * and effectively concatenate them. * - If first element is a negative number, truncate displaying cdr to * at most that many characters. If positive, pad (with spaces) * to at least that many characters. * - If first element is another symbol or a boolean specifier, process * the cadr or caddr recursively according to whether the symbol's * value or specifier's instance is non-nil or nil. * - If first element is , process the cadr or caddr * recursively according to whether the instance of the specifier in * the modeline's window is non-nil or nil. * - If first element is an extent, process the cdr recursively * and handle the extent's face. */ Lisp_Object car, tem; car = XCAR (elt); if (SYMBOLP (car) || BOOLEAN_SPECIFIERP (car)) { elt = XCDR (elt); if (!CONSP (elt)) goto invalid; if (SYMBOLP (car)) tem = symbol_value_in_buffer (car, w->buffer); else tem = specifier_instance_no_quit (car, Qunbound, wrap_window (w), ERROR_ME_DEBUG_WARN, 0, Qzero); /* elt is now the cdr, and we know it is a cons cell. Use its car if CAR has a non-nil value. */ if (!UNBOUNDP (tem) && !NILP (tem)) { elt = XCAR (elt); goto tail_recurse; } /* Symbol's value or specifier's instance is nil or unbound * Get the cddr of the original list * and if possible find the caddr and use that. */ elt = XCDR (elt); if (NILP (elt)) ; else if (!CONSP (elt)) goto invalid; else { elt = XCAR (elt); goto tail_recurse; } } else if (FIXNUMP (car)) { Charcount lim = XFIXNUM (car); elt = XCDR (elt); if (lim < 0) { /* Negative int means reduce maximum width. * DO NOT change MIN_PIXPOS here! * (20 -10 . foo) should truncate foo to 10 col * and then pad to 20. */ if (max_pos == -1) max_pos = pos - lim; else max_pos = min (max_pos, pos - lim); } else if (lim > 0) { /* Padding specified. Don't let it be more than * current maximum. */ lim += pos; if (max_pos != -1 && lim > max_pos) lim = max_pos; /* If that's more padding than already wanted, queue it. * But don't reduce padding already specified even if * that is beyond the current truncation point. */ if (lim > min_pos) min_pos = lim; } goto tail_recurse; } else if (STRINGP (car) || CONSP (car)) { int limit = 50; /* LIMIT is to protect against circular lists. */ while (CONSP (elt) && --limit > 0 && (pos < max_pos || max_pos == -1)) { pos = generate_fstring_runes (w, data, pos, pos, max_pos, XCAR (elt), depth, max_pixsize, findex, type, offset, cur_ext); elt = XCDR (elt); } } else if (EXTENTP (car)) { struct extent *ext = XEXTENT (car); if (EXTENT_LIVE_P (ext)) { face_index old_findex = data->findex; Lisp_Object face; Lisp_Object font_inst; face_index new_findex; Bytecount start = data->bytepos; face = extent_face (ext); if (FACEP (face)) { /* #### needs to merge faces, sigh */ /* #### needs to handle list of faces */ new_findex = get_builtin_face_cache_index (w, face); /* !!#### not right; needs to compute the max height of all the charsets */ font_inst = WINDOW_FACE_CACHEL_FONT (w, new_findex, Vcharset_ascii); data->dl->ascent = max (data->dl->ascent, XFONT_INSTANCE (font_inst)->ascent); data->dl->descent = max (data->dl->descent, XFONT_INSTANCE (font_inst)-> descent); } else new_findex = old_findex; data->findex = new_findex; pos = generate_fstring_runes (w, data, pos, pos, max_pos, XCDR (elt), depth - 1, max_pixsize, new_findex, type, offset, car); data->findex = old_findex; Dynarr_add (formatted_string_extent_dynarr, ext); Dynarr_add (formatted_string_extent_start_dynarr, start); Dynarr_add (formatted_string_extent_end_dynarr, data->bytepos); } } } else if (GLYPHP (elt)) { /* Glyphs are considered as one character with respect to the modeline horizontal scrolling facility. -- dv */ if (*offset > 0) *offset -= 1; else pos = add_glyph_to_fstring_db_runes (data, elt, pos, pos, max_pos, cur_ext); } else { invalid: { const Ascbyte *str = GETTEXT ("*invalid*"); Charcount size = (Charcount) strlen (str); /* is this ok ?? -- dv */ if (size <= *offset) *offset -= size; else { const Ibyte *tmp_str = itext_n_addr ((const Ibyte *) str, *offset); /* #### NOTE: I don't understand why a tmp_max is not computed and used here as in the plain string case above. -- dv */ pos = add_string_to_fstring_db_runes (data, tmp_str, pos, min_pos, max_pos); *offset = 0; } } } if (min_pos > pos) { add_string_to_fstring_db_runes (data, (const Ibyte *) "", pos, min_pos, -1); } return pos; } /* Update just the modeline. Assumes the desired display structs. If they do not have a modeline block, it does nothing. */ static void regenerate_modeline (struct window *w) { display_line_dynarr *dla = window_display_lines (w, DESIRED_DISP); if (!Dynarr_length (dla) || !Dynarr_begin (dla)->modeline) return; else { generate_modeline (w, Dynarr_begin (dla), DESIRED_DISP); redisplay_update_line (w, 0, 0, 0); } } /* Make sure that modeline display line is present in the given display structs if the window has a modeline and update that line. Returns true if a modeline was needed. */ static int ensure_modeline_generated (struct window *w, int type) { int need_modeline; /* minibuffer windows don't have modelines */ if (MINI_WINDOW_P (w)) need_modeline = 0; /* windows which haven't had it turned off do */ else if (WINDOW_HAS_MODELINE_P (w)) need_modeline = 1; /* windows which have it turned off don't have a divider if there is a horizontal scrollbar */ else if (window_scrollbar_height (w)) need_modeline = 0; /* and in this case there is none */ else need_modeline = 1; if (need_modeline) { display_line_dynarr *dla; dla = window_display_lines (w, type); /* We don't care if there is a display line which is not currently a modeline because it is definitely going to become one if we have gotten to this point. */ if (Dynarr_length (dla) == 0) { if (Dynarr_largest (dla) > 0) Dynarr_incrementr (dla); else { struct display_line modeline; DISPLAY_LINE_INIT (modeline); Dynarr_add (dla, modeline); } } /* If we're adding a new place marker go ahead and generate the modeline so that it is available for use by window_modeline_height. */ generate_modeline (w, Dynarr_begin (dla), type); } return need_modeline; } /* #### Kludge or not a kludge. I tend towards the former. */ int real_current_modeline_height (struct window *w) { Fset_marker (w->start[CMOTION_DISP], w->start[CURRENT_DISP], w->buffer); Fset_marker (w->pointm[CMOTION_DISP], w->pointm[CURRENT_DISP], w->buffer); if (ensure_modeline_generated (w, CMOTION_DISP)) { display_line_dynarr *dla = window_display_lines (w, CMOTION_DISP); if (Dynarr_length (dla)) { if (Dynarr_begin (dla)->modeline) return (Dynarr_begin (dla)->ascent + Dynarr_begin (dla)->descent); } } return 0; } /***************************************************************************/ /* */ /* displayable string routines */ /* */ /***************************************************************************/ /* Given a position for a string in a window, ensure that the given display line DL accurately represents the text on a line starting at the given position. Yes, this is duplicating the code of create_text_block, but it looked just too hard to change create_text_block to handle strings *and* buffers. We already make a distinction between the two elsewhere in the code so I think unifying them would require a complete MULE rewrite. Besides, the other distinction is that these functions cover text that the user *cannot edit* so we can remove everything to do with cursors, minibuffers etc. Eventually the modeline routines should be modified to use this code as it copes with many more types of display situation. */ static Charbpos create_string_text_block (struct window *w, Lisp_Object disp_string, struct display_line *dl, Charcount start_pos, prop_block_dynarr **prop, face_index default_face) { struct frame *f = XFRAME (w->frame); /* Note that a lot of the buffer controlled stuff has been left in because you might well want to make use of it (selective display etc), its just the buffer text that we do not use. However, it seems to be possible for buffer to be nil sometimes so protect against this case. */ struct buffer *b = BUFFERP (w->buffer) ? XBUFFER (w->buffer) : 0; struct device *d = XDEVICE (f->device); /* we're working with these a lot so precalculate them */ Bytecount slen = XSTRING_LENGTH (disp_string); Bytecount byte_string_zv = slen; Bytecount byte_start_pos = string_index_char_to_byte (disp_string, start_pos); pos_data data; int truncate_win = b ? window_truncation_on (w) : 0; /* We're going to ditch selective display for static text, it's an FSF thing and invisible extents are the way to go here. Implementing it also relies on a number of buffer-specific functions that we don't have the luxury of being able to use here. */ /* The variable ctl-arrow allows the user to specify what characters can actually be displayed and which octal should be used for. #### This variable should probably have some rethought done to it. #### It would also be really nice if you could specify that the characters come out in hex instead of in octal. Mule does that by adding a ctl-hexa variable similar to ctl-arrow, but that's bogus -- we need a more general solution. I think you need to extend the concept of display tables into a more general conversion mechanism. Ideally you could specify a Lisp function that converts characters, but this violates the Second Golden Rule and besides would make things way way way way slow. So instead, we extend the display-table concept, which was historically limited to 256-byte vectors, to one of the following: a) A 256-entry vector, for backward compatibility; b) char-table, mapping characters to values; c) range-table, mapping ranges of characters to values; d) a list of the above. The (d) option allows you to specify multiple display tables instead of just one. Each display table can specify conversions for some characters and leave others unchanged. The way the character gets displayed is determined by the first display table with a binding for that character. This way, you could call a function `enable-hex-display' that adds a hex display-table to the list of display tables for the current buffer. #### ...not yet implemented... Also, we extend the concept of "mapping" to include a printf-like spec. Thus you can make all extended characters show up as hex with a display table like this: #s(range-table data ((256 524288) (format "%x"))) Since more than one display table is possible, you have great flexibility in mapping ranges of characters. */ Ichar printable_min = b ? (CHAR_OR_CHAR_INTP (b->ctl_arrow) ? XCHAR_OR_CHAR_INT (b->ctl_arrow) : ((EQ (b->ctl_arrow, Qt) || EQ (b->ctl_arrow, Qnil)) ? 255 : 160)) : 255; Lisp_Object face_dt, window_dt; /* The text display block for this display line. */ struct display_block *db = get_display_block_from_line (dl, TEXT); /* The first time through the main loop we need to force the glyph data to be updated. */ int initial = 1; /* Apparently the new extent_fragment_update returns an end position equal to the position passed in if there are no more runs to be displayed. */ int no_more_frags = 0; dl->used_prop_data = 0; dl->num_chars = 0; dl->line_continuation = 0; /* Set up faces to use for clearing areas, used by output_display_line. */ dl->clear_findex = default_face; if (default_face > DEFAULT_INDEX) { dl->left_margin_findex = default_face; dl->right_margin_findex = default_face; } else { dl->left_margin_findex = get_builtin_face_cache_index (w, Vleft_margin_face); dl->right_margin_findex = get_builtin_face_cache_index (w, Vright_margin_face); } xzero (data); data.ef = extent_fragment_new (disp_string, f); /* These values are used by all of the rune addition routines. We add them to this structure for ease of passing. */ data.d = d; data.window = wrap_window (w); data.db = db; data.dl = dl; data.byte_charpos = byte_start_pos; data.pixpos = dl->bounds.left_in; data.last_charset = Qunbound; data.last_findex = default_face; data.result_str = Qnil; data.string = disp_string; /* Set the right boundary adjusting it to take into account any end glyph. Save the width of the end glyph for later use. */ data.max_pixpos = dl->bounds.right_in; data.max_pixpos -= data.end_glyph_width; data.cursor_type = NO_CURSOR; data.cursor_x = -1; data.start_col = 0; /* I don't think we want this, string areas should not scroll with the window data.start_col = w->hscroll; data.byte_start_col_enabled = (w->hscroll ? byte_start_pos : 0); */ data.byte_start_col_enabled = 0; data.hscroll_glyph_width_adjust = 0; /* We regenerate the line from the very beginning. */ Dynarr_reset (db->runes); /* Why is this less than or equal and not just less than? If the starting position is already equal to the maximum we can't add anything else, right? Wrong. We might still have a newline to add. A newline can use the room allocated for an end glyph since if we add it we know we aren't going to be adding any end glyph. */ /* #### Chuck -- I think this condition should be while (1). Otherwise if (e.g.) there is one begin-glyph and one end-glyph and the begin-glyph ends exactly at the end of the window, the end-glyph and text might not be displayed. while (1) ensures that the loop terminates only when either (a) there is propagation data or (b) the end-of-line or end-of-buffer is hit. #### Also I think you need to ensure that the operation "add begin glyphs; add end glyphs; add text" is atomic and can't get interrupted in the middle. If you run off the end of the line during that operation, then you keep accumulating propagation data until you're done. Otherwise, if the (e.g.) there's a begin glyph at a particular position and attempting to display that glyph results in window-end being hit and propagation data being generated, then the character at that position won't be displayed. #### See also the comment after the end of this loop, below. */ while (data.pixpos <= data.max_pixpos) { /* #### This check probably should not be necessary. */ if (data.byte_charpos > byte_string_zv) { /* #### urk! More of this lossage! */ data.byte_charpos--; goto done; } /* Check for face changes. */ if (initial || (!no_more_frags && data.byte_charpos == data.ef->end)) { Lisp_Object last_glyph = Qnil; /* Deal with clipped glyphs that we have already displayed. */ if (*prop && Dynarr_begin (*prop)->type == PROP_GLYPH) { last_glyph = Dynarr_begin (*prop)->data.p_glyph.glyph; Dynarr_free (*prop); *prop = 0; } /* Now compute the face and begin/end-glyph information. */ data.findex = /* Remember that the extent-fragment routines deal in Bytexpos's. */ extent_fragment_update (w, data.ef, data.byte_charpos, last_glyph); /* This is somewhat cheesy but the alternative is to propagate default_face into extent_fragment_update. */ if (data.findex == DEFAULT_INDEX) data.findex = default_face; get_display_tables (w, data.findex, &face_dt, &window_dt); if (data.byte_charpos == data.ef->end) no_more_frags = 1; } initial = 0; /* Determine what is next to be displayed. We first handle any glyphs returned by glyphs_at_charbpos. If there are no glyphs to display then we determine what to do based on the character at the current buffer position. */ /* If the current position is covered by an invisible extent, do nothing (except maybe add some ellipses). #### The behavior of begin and end-glyphs at the edge of an invisible extent should be investigated further. This is fairly low priority though. */ if (data.ef->invisible) { /* #### Chuck, perhaps you could look at this code? I don't really know what I'm doing. */ if (*prop) { Dynarr_free (*prop); *prop = 0; } /* The extent fragment code only sets this when we should really display the ellipses. It makes sure the ellipses don't get displayed more than once in a row. */ if (data.ef->invisible_ellipses) { struct glyph_block gb; data.ef->invisible_ellipses_already_displayed = 1; data.ef->invisible_ellipses = 0; gb.extent = Qnil; gb.glyph = Vinvisible_text_glyph; *prop = add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, GLYPH_CACHEL (w, INVIS_GLYPH_INDEX)); /* Perhaps they shouldn't propagate if the very next thing is to display a newline (for compatibility with selective-display-ellipses)? Maybe that's too abstruse. */ if (*prop) goto done; } /* #### What if we're dealing with a display table? */ if (data.start_col) data.start_col--; if (data.byte_charpos == byte_string_zv) goto done; else INC_BYTECOUNT (XSTRING_DATA (disp_string), data.byte_charpos); } /* If there is propagation data, then it represents the current buffer position being displayed. Add them and advance the position counter. This might also add the minibuffer prompt. */ else if (*prop) { dl->used_prop_data = 1; *prop = add_propagation_runes (prop, &data); if (*prop) goto done; /* gee, a really narrow window */ else if (data.byte_charpos == byte_string_zv) goto done; else if (data.byte_charpos < 0) /* #### urk urk urk! Aborts are not very fun! Fix this please! */ data.byte_charpos = 0; else INC_BYTECOUNT (XSTRING_DATA (disp_string), data.byte_charpos); } /* If there are end glyphs, add them to the line. These are the end glyphs for the previous run of text. We add them here rather than doing them at the end of handling the previous run so that glyphs at the beginning and end of a line are handled correctly. */ else if (Dynarr_length (data.ef->end_glyphs) > 0) { data.ch = string_ichar (disp_string, data.byte_charpos); *prop = add_glyph_runes (&data, END_GLYPHS); if (*prop) { goto done; } } /* If there are begin glyphs, add them to the line. */ else if (Dynarr_length (data.ef->begin_glyphs) > 0) { data.ch = string_ichar (disp_string, data.byte_charpos); *prop = add_glyph_runes (&data, BEGIN_GLYPHS); if (*prop) { goto done; } } /* If at end-of-buffer, we've already processed begin and end-glyphs at this point and there's no text to process, so we're done. */ else if (data.byte_charpos == byte_string_zv) goto done; else { Lisp_Object entry = Qnil; /* Get the character at the current buffer position. */ data.ch = string_ichar (disp_string, data.byte_charpos); if (!NILP (face_dt) || !NILP (window_dt)) entry = display_table_entry (data.ch, face_dt, window_dt); /* If there is a display table entry for it, hand it off to add_disp_table_entry_runes and let it worry about it. */ if (!NILP (entry) && !EQ (entry, make_char (data.ch))) { *prop = add_disp_table_entry_runes (&data, entry); if (*prop) goto done; } /* Check if we have hit a newline character. If so, add a marker to the line and end this loop. */ else if (data.ch == '\n') { /* Update the clearing face index unless the shrink property is set. -- dvl */ if ((data.findex > DEFAULT_INDEX) && ! WINDOW_FACE_CACHEL_SHRINK_P (w, data.findex)) dl->clear_findex = data.findex; /* We aren't going to be adding an end glyph so give its space back in order to make sure that the cursor can fit. */ data.max_pixpos += data.end_glyph_width; goto done; } /* If the current character is considered to be printable, then just add it. */ else if (data.ch >= printable_min) { *prop = add_ichar_rune (&data); if (*prop) goto done; } /* If the current character is a tab, determine the next tab starting position and add a blank rune which extends from the current pixel position to that starting position. */ else if (data.ch == '\t') { int tab_start_pixpos = data.pixpos; int next_tab_start; int char_tab_width; int prop_width = 0; if (data.start_col > 1) tab_start_pixpos -= (space_width (w) * (data.start_col - 1)); next_tab_start = next_tab_position (w, tab_start_pixpos, dl->bounds.left_in + data.hscroll_glyph_width_adjust); if (next_tab_start > data.max_pixpos) { prop_width = next_tab_start - data.max_pixpos; next_tab_start = data.max_pixpos; } data.blank_width = next_tab_start - data.pixpos; char_tab_width = (next_tab_start - tab_start_pixpos) / space_width (w); *prop = add_blank_rune (&data, w, char_tab_width); /* add_blank_rune is only supposed to be called with sizes guaranteed to fit in the available space. */ assert (!(*prop)); if (prop_width) { struct prop_block pb; *prop = Dynarr_new (prop_block); pb.type = PROP_BLANK; pb.data.p_blank.width = prop_width; pb.data.p_blank.findex = data.findex; Dynarr_add (*prop, pb); goto done; } } /* If character is a control character, pass it off to add_control_char_runes. The is_*() routines have undefined results on arguments outside of the range [-1, 255]. (This often bites people who carelessly use `char' instead of `unsigned char'.) */ else if (data.ch < 0x100 && iscntrl ((Ibyte) data.ch)) { *prop = add_control_char_runes (&data, b); if (*prop) goto done; } /* If the character is above the ASCII range and we have not already handled it, then print it as an octal number. */ else if (data.ch >= 0200) { *prop = add_octal_runes (&data); if (*prop) goto done; } /* Assume the current character is considered to be printable, then just add it. */ else { *prop = add_ichar_rune (&data); if (*prop) goto done; } INC_BYTECOUNT (XSTRING_DATA (disp_string), data.byte_charpos); } } done: /* Determine the starting point of the next line if we did not hit the end of the buffer. */ if (data.byte_charpos < byte_string_zv) { /* #### This check is not correct. If the line terminated due to a begin-glyph or end-glyph hitting window-end, then data.ch will not point to the character at data.byte_charpos. If you make the two changes mentioned at the top of this loop, you should be able to say '(if (*prop))'. That should also make it possible to eliminate the data.byte_charpos < BYTE_BUF_ZV (b) check. */ /* The common case is that the line ended because we hit a newline. In that case, the next character is just the next buffer position. */ if (data.ch == '\n') { INC_BYTECOUNT (XSTRING_DATA (disp_string), data.byte_charpos); } /* Otherwise we have a buffer line which cannot fit on one display line. */ else { struct glyph_block gb; struct glyph_cachel *cachel; /* If the line is to be truncated then we actually have to look for the next newline. We also add the end-of-line glyph which we know will fit because we adjusted the right border before we starting laying out the line. */ data.max_pixpos += data.end_glyph_width; data.findex = default_face; gb.extent = Qnil; if (truncate_win) { Bytecount byte_pos; /* Now find the start of the next line. */ byte_pos = byte_find_next_ichar_in_string (disp_string, '\n', data.byte_charpos, 1); data.cursor_type = NO_CURSOR; data.byte_charpos = byte_pos; gb.glyph = Vtruncation_glyph; cachel = GLYPH_CACHEL (w, TRUN_GLYPH_INDEX); } else { /* The cursor can never be on the continuation glyph. */ data.cursor_type = NO_CURSOR; /* data.byte_charpos is already at the start of the next line. */ dl->line_continuation = 1; gb.glyph = Vcontinuation_glyph; cachel = GLYPH_CACHEL (w, CONT_GLYPH_INDEX); } if (data.end_glyph_width) add_glyph_rune (&data, &gb, BEGIN_GLYPHS, 0, cachel); if (truncate_win && data.byte_charpos == byte_string_zv) { const Ibyte *endb = itext_n_addr (XSTRING_DATA (disp_string), byte_string_zv); DEC_IBYTEPTR (endb); if (itext_ichar (endb) != '\n') { /* #### Damn this losing shit. */ data.byte_charpos++; } } } } else if (data.byte_charpos == byte_string_zv) { /* create_text_block () adds a bogus \n marker here which screws up subwindow display. Since we never have a cursor in the gutter we can safely ignore it. */ } /* Calculate left whitespace boundary. */ { int elt = 0; /* Whitespace past a newline is considered right whitespace. */ while (elt < Dynarr_length (db->runes)) { struct rune *rb = Dynarr_atp (db->runes, elt); if ((rb->type == RUNE_CHAR && rb->object.chr.ch == ' ') || rb->type == RUNE_BLANK) { dl->bounds.left_white += rb->width; elt++; } else elt = Dynarr_length (db->runes); } } /* Calculate right whitespace boundary. */ { int elt = Dynarr_length (db->runes) - 1; int done = 0; while (!done && elt >= 0) { struct rune *rb = Dynarr_atp (db->runes, elt); if (!(rb->type == RUNE_CHAR && rb->object.chr.ch < 0x100 && isspace (rb->object.chr.ch)) && !rb->type == RUNE_BLANK) { dl->bounds.right_white = rb->xpos + rb->width; done = 1; } elt--; } /* The line is blank so everything is considered to be right whitespace. */ if (!done) dl->bounds.right_white = dl->bounds.left_in; } /* Set the display blocks bounds. */ db->start_pos = dl->bounds.left_in; if (Dynarr_length (db->runes)) { struct rune *rb = Dynarr_lastp (db->runes); db->end_pos = rb->xpos + rb->width; } else db->end_pos = dl->bounds.right_white; calculate_baseline (&data); dl->ascent = data.new_ascent; dl->descent = data.new_descent; { unsigned short ascent = (unsigned short) XFIXNUM (w->minimum_line_ascent); if (dl->ascent < ascent) dl->ascent = ascent; } { unsigned short descent = (unsigned short) XFIXNUM (w->minimum_line_descent); if (dl->descent < descent) dl->descent = descent; } calculate_yoffset (dl, db); dl->cursor_elt = data.cursor_x; /* #### lossage lossage lossage! Fix this shit! */ if (data.byte_charpos > byte_string_zv) dl->end_charpos = buffer_or_string_bytexpos_to_charxpos (disp_string, byte_string_zv); else dl->end_charpos = buffer_or_string_bytexpos_to_charxpos (disp_string, data.byte_charpos) - 1; if (truncate_win) data.dl->num_chars = string_column_at_point (disp_string, dl->end_charpos, b ? XFIXNUM (b->tab_width) : 8); else /* This doesn't correctly take into account tabs and control characters but if the window isn't being truncated then this value isn't going to end up being used anyhow. */ data.dl->num_chars = dl->end_charpos - dl->charpos; /* #### handle horizontally scrolled line with text none of which was actually laid out. */ /* #### handle any remainder of overlay arrow */ if (*prop == ADD_FAILED) *prop = NULL; if (truncate_win && *prop) { Dynarr_free (*prop); *prop = NULL; } extent_fragment_delete (data.ef); /* #### If we started at EOB, then make sure we return a value past it so that regenerate_window will exit properly. This is bogus. The main loop should get fixed so that it isn't necessary to call this function if we are already at EOB. */ if (data.byte_charpos == byte_string_zv && byte_start_pos == byte_string_zv) return string_index_byte_to_char (disp_string, data.byte_charpos) + 1; /* Yuck! */ else return string_index_byte_to_char (disp_string, data.byte_charpos); } /* Given a display line and a starting position, ensure that the contents of the display line accurately represent the visual representation of the buffer contents starting from the given position when displayed in the given window. The display line ends when the contents of the line reach the right boundary of the given window. This is very similar to generate_display_line but with the same limitations as create_string_text_block. I have taken the liberty of fixing the bytebpos stuff though.*/ static Charbpos generate_string_display_line (struct window *w, Lisp_Object disp_string, struct display_line *dl, Charcount start_pos, prop_block_dynarr **prop, face_index default_face) { Charcount ret_charcount; /* you must set bounds before calling this. */ /* Reset what this line is using. */ if (dl->display_blocks) Dynarr_reset (dl->display_blocks); if (dl->left_glyphs) { Dynarr_free (dl->left_glyphs); dl->left_glyphs = 0; } if (dl->right_glyphs) { Dynarr_free (dl->right_glyphs); dl->right_glyphs = 0; } /* We aren't generating a modeline at the moment. */ dl->modeline = 0; /* Create a display block for the text region of the line. */ ret_charcount = create_string_text_block (w, disp_string, dl, start_pos, prop, default_face); dl->charpos = start_pos; if (dl->end_charpos < dl->charpos) dl->end_charpos = dl->charpos; /* If there are left glyphs associated with any character in the text block, then create a display block to handle them. */ if (dl->left_glyphs != NULL && Dynarr_length (dl->left_glyphs)) create_left_glyph_block (w, dl, 0); /* If there are right glyphs associated with any character in the text block, then create a display block to handle them. */ if (dl->right_glyphs != NULL && Dynarr_length (dl->right_glyphs)) create_right_glyph_block (w, dl); return ret_charcount; } /* Info on reentrancy crashes, with backtraces given: (Info-goto-node "(internals)Critical Redisplay Sections") */ /* This is ripped off from regenerate_window. All we want to do is loop through elements in the string creating display lines until we have covered the provided area. Simple really. */ void generate_displayable_area (struct window *w, Lisp_Object disp_string, int xpos, int ypos, int width, int height, display_line_dynarr* dla, Charcount start_pos, face_index default_face) { int yend = ypos + height; Charcount s_zv; prop_block_dynarr *prop = 0; layout_bounds bounds; int depth = -1; /* if there's nothing to do then do nothing. code after this assumes there is something to do. */ if (NILP (disp_string)) return; /* See comment in regenerate_window() */ if (!in_display) depth = enter_redisplay_critical_section (); assert (dla); Dynarr_reset (dla); s_zv = string_char_length (disp_string); bounds.left_out = xpos; bounds.right_out = xpos + width; /* The inner boundaries mark where the glyph margins are located. */ bounds.left_in = bounds.left_out + window_left_margin_width (w); bounds.right_in = bounds.right_out - window_right_margin_width (w); /* We cannot fully calculate the whitespace boundaries as they depend on the contents of the line being displayed. */ bounds.left_white = bounds.left_in; bounds.right_white = bounds.right_in; while (ypos < yend) { struct display_line dl; struct display_line *dlp; Charcount next_pos; int local; int pos_of_dlp = -1; if (Dynarr_length (dla) < Dynarr_largest (dla)) { pos_of_dlp = Dynarr_length (dla); dlp = Dynarr_atp (dla, pos_of_dlp); local = 0; } else { DISPLAY_LINE_INIT (dl); dlp = &dl; local = 1; } dlp->bounds = bounds; dlp->offset = 0; Dynarr_lock (dla); next_pos = generate_string_display_line (w, disp_string, dlp, start_pos, &prop, default_face); Dynarr_unlock (dla); /* we need to make sure that we continue along the line if there is more left to display otherwise we just end up redisplaying the same chunk over and over again. */ if (next_pos == start_pos && next_pos < s_zv) start_pos++; else start_pos = next_pos; dlp->ypos = ypos + dlp->ascent; ypos = dlp->ypos + dlp->descent; if (ypos > yend) { int visible_height = dlp->ascent + dlp->descent; dlp->clip = (ypos - yend); visible_height -= dlp->clip; if (visible_height < VERTICAL_CLIP (w, 1)) { if (local) free_display_line (dlp); break; } } else dlp->clip = 0; if (pos_of_dlp < 0) Dynarr_add (dla, *dlp); else if (pos_of_dlp == Dynarr_length (dla)) Dynarr_incrementr (dla); else ABORT (); /* #### This type of check needs to be done down in the generate_display_line call. */ if (start_pos >= s_zv) break; } if (prop) Dynarr_free (prop); if (depth >= 0) exit_redisplay_critical_section (depth); } /***************************************************************************/ /* */ /* window-regeneration routines */ /* */ /***************************************************************************/ /* For a given window and starting position in the buffer it contains, ensure that the TYPE display lines accurately represent the presentation of the window. We pass the buffer instead of getting it from the window since redisplay_window may have temporarily changed it to the echo area buffer. */ static void regenerate_window (struct window *w, Charbpos start_pos, Charbpos point, int type) { struct frame *f = XFRAME (w->frame); struct buffer *b = XBUFFER (w->buffer); int ypos = WINDOW_TEXT_TOP (w); int yend; /* set farther down */ int yclip = WINDOW_TEXT_TOP_CLIP (w); int force; int depth = -1; prop_block_dynarr *prop; layout_bounds bounds; display_line_dynarr *dla; int need_modeline; /* The lines had better exist by this point. */ if (!(dla = window_display_lines (w, type))) ABORT (); if (!in_display) depth = enter_redisplay_critical_section (); /* This is one spot where a reentrancy crash will occur, due to a check in the dynarr to make sure it isn't "locked" */ /* Info on reentrancy crashes, with backtraces given: (Info-goto-node "(internals)Critical Redisplay Sections") */ Dynarr_reset (dla); w->max_line_len = 0; /* Added 2-1-10 -- we should never have empty face or glyph cachels because we initialized them at startup and the only way to reduce their number is through calling reset_face_cachels() or reset_glyph_cachels(), which as a side effect sets up a number of standard entries */ assert (Dynarr_length (w->face_cachels)); assert (Dynarr_length (w->glyph_cachels)); #if 0 /* #### Delete this code sometime later than 2-1-10 when we're sure it's not needed */ /* Normally these get updated in redisplay_window but it is possible for this function to get called from some other points where that update may not have occurred. This acts as a safety check. */ if (!Dynarr_length (w->face_cachels)) reset_face_cachels (w); if (!Dynarr_length (w->glyph_cachels)) reset_glyph_cachels (w); #endif Fset_marker (w->start[type], make_fixnum (start_pos), w->buffer); Fset_marker (w->pointm[type], make_fixnum (point), w->buffer); w->last_point_x[type] = -1; w->last_point_y[type] = -1; /* Make sure a modeline is in the structs if needed. */ need_modeline = ensure_modeline_generated (w, type); /* Wait until here to set this so that the structs have a modeline generated in the case where one didn't exist. */ yend = WINDOW_TEXT_BOTTOM (w); bounds = calculate_display_line_boundaries (w, 0); /* 97/3/14 jhod: stuff added here to support pre-prompts (used for input systems) */ if (MINI_WINDOW_P (w) && (!NILP (Vminibuf_prompt) || !NILP (Vminibuf_preprompt)) && !echo_area_active (f) && start_pos == BUF_BEGV (b)) { struct prop_block pb; prop = Dynarr_new (prop_block); pb.type = PROP_MINIBUF_PROMPT; pb.data.p_minibuf_prompt.preprompt = Vminibuf_preprompt; pb.data.p_minibuf_prompt.prompt = Vminibuf_prompt; Dynarr_add (prop, pb); } else prop = 0; /* When we are computing things for scrolling purposes, make sure at least one line is always generated */ force = (type == CMOTION_DISP); /* Make sure this is set always */ /* Note the conversion at end */ w->window_end_pos[type] = start_pos; while (ypos < yend || force) { struct display_line dl; struct display_line *dlp; int local; int pos_of_dlp = -1; if (Dynarr_length (dla) < Dynarr_largest (dla)) { pos_of_dlp = Dynarr_length (dla); dlp = Dynarr_atp (dla, pos_of_dlp); local = 0; } else { DISPLAY_LINE_INIT (dl); dlp = &dl; local = 1; } dlp->bounds = bounds; dlp->offset = 0; Dynarr_lock (dla); start_pos = generate_display_line (w, dlp, 1, start_pos, &prop, type); Dynarr_unlock (dla); if (yclip > dlp->ascent) { /* this should never happen, but if it does just display the whole line */ yclip = 0; } dlp->ypos = (ypos + dlp->ascent) - yclip; ypos = dlp->ypos + dlp->descent; /* See if we've been asked to start midway through a line, for partial display line scrolling. */ if (yclip) { dlp->top_clip = yclip; yclip = 0; } else dlp->top_clip = 0; if (ypos > yend) { int visible_height = dlp->ascent + dlp->descent; dlp->clip = (ypos - yend); /* Although this seems strange we could have a single very tall line visible for which we need to account for both the top clip and the bottom clip. */ visible_height -= (dlp->clip + dlp->top_clip); if (visible_height < VERTICAL_CLIP (w, 1) && !force) { if (local) free_display_line (dlp); break; } } else dlp->clip = 0; if (dlp->cursor_elt != -1) { /* #### This check is steaming crap. Have to get things fixed so when create_text_block hits EOB, we're done, period. */ if (w->last_point_x[type] == -1) { w->last_point_x[type] = dlp->cursor_elt; w->last_point_y[type] = Dynarr_length (dla); } else { /* #### This means that we've added a cursor at EOB twice. Yuck oh yuck. */ struct display_block *db; Dynarr_lock (dla); db = get_display_block_from_line (dlp, TEXT); Dynarr_unlock (dla); Dynarr_atp (db->runes, dlp->cursor_elt)->cursor_type = NO_CURSOR; dlp->cursor_elt = -1; } } if (dlp->num_chars > w->max_line_len) w->max_line_len = dlp->num_chars; if (pos_of_dlp < 0) Dynarr_add (dla, *dlp); else if (pos_of_dlp == Dynarr_length (dla)) Dynarr_incrementr (dla); else ABORT (); /* #### This isn't right, but it is close enough for now. */ w->window_end_pos[type] = start_pos; /* #### This type of check needs to be done down in the generate_display_line call. */ if (start_pos > BUF_ZV (b)) break; force = 0; } if (prop) Dynarr_free (prop); /* #### More not quite right, but close enough. */ /* Ben sez: apparently window_end_pos[] is measured as the number of characters between the window end and the end of the buffer? This seems rather weirdo. What's the justification for this? JV sez: Because BUF_Z (b) would be a good initial value, however that can change. This representation allows initalizing with 0. */ w->window_end_pos[type] = BUF_Z (b) - w->window_end_pos[type]; if (need_modeline) { /* We know that this is the right thing to use because we put it there when we first started working in this function. */ generate_modeline (w, Dynarr_begin (dla), type); } if (depth >= 0) exit_redisplay_critical_section (depth); } #define REGEN_INC_FIND_START_END \ do { \ /* Determine start and end of lines. */ \ if (!Dynarr_length (cdla)) \ return 0; \ else \ { \ if (Dynarr_begin (cdla)->modeline && Dynarr_begin (ddla)->modeline) \ { \ dla_start = 1; \ } \ else if (!Dynarr_begin (cdla)->modeline \ && !Dynarr_begin (ddla)->modeline) \ { \ dla_start = 0; \ } \ else \ ABORT (); /* structs differ */ \ \ dla_end = Dynarr_length (cdla) - 1; \ } \ \ start_pos = (Dynarr_atp (cdla, dla_start)->charpos \ + Dynarr_atp (cdla, dla_start)->offset); \ /* If this isn't true, then startp has changed and we need to do a \ full regen. */ \ if (startp != start_pos) \ return 0; \ \ /* Point is outside the visible region so give up. */ \ if (pointm < start_pos) \ return 0; \ \ } while (0) /* This attempts to incrementally update the display structures. It returns a boolean indicating success or failure. This function is very similar to regenerate_window_incrementally and is in fact only called from that function. However, because of the nature of the changes it deals with it sometimes makes different assumptions which can lead to success which are much more difficult to make when dealing with buffer changes. */ static int regenerate_window_extents_only_changed (struct window *w, Charbpos startp, Charbpos pointm, Charcount beg_unchanged, Charcount end_unchanged) { struct buffer *b = XBUFFER (w->buffer); display_line_dynarr *cdla = window_display_lines (w, CURRENT_DISP); display_line_dynarr *ddla = window_display_lines (w, DESIRED_DISP); int dla_start = 0; int dla_end, line; int first_line, last_line; Charbpos start_pos; /* Don't define this in the loop where it is used because we definitely want its value to survive between passes. */ prop_block_dynarr *prop = NULL; /* If we don't have any buffer change recorded but the modiff flag has been incremented, then fail. I'm not sure of the exact circumstances under which this can happen, but I believe that it is probably a reasonable happening. */ if (!point_visible (w, pointm, CURRENT_DISP) || XFIXNUM (w->last_modified[CURRENT_DISP]) < BUF_MODIFF (b)) return 0; /* If the cursor is moved we attempt to update it. If we succeed we go ahead and proceed with the optimization attempt. */ if (!EQ (Fmarker_buffer (w->last_point[CURRENT_DISP]), w->buffer) || pointm != marker_position (w->last_point[CURRENT_DISP])) { struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); struct frame *sel_f = device_selected_frame (d); int success = 0; if (w->last_point_x[CURRENT_DISP] != -1 && w->last_point_y[CURRENT_DISP] != -1) { if (redisplay_move_cursor (w, pointm, WINDOW_TTY_P (w))) { /* Always regenerate the modeline in case it is displaying the current line or column. */ regenerate_modeline (w); success = 1; } } else if (w != XWINDOW (FRAME_SELECTED_WINDOW (sel_f))) { if (f->modeline_changed) regenerate_modeline (w); success = 1; } if (!success) return 0; } if (beg_unchanged == -1 && end_unchanged == -1) return 1; /* assert: There are no buffer modifications or they are all below the visible region. We assume that regenerate_window_incrementally has not called us unless this is true. */ REGEN_INC_FIND_START_END; /* If the changed are starts before the visible area, give up. */ if (beg_unchanged < startp) return 0; /* Find what display line the extent changes first affect. */ line = dla_start; while (line <= dla_end) { struct display_line *dl = Dynarr_atp (cdla, line); Charbpos lstart = dl->charpos + dl->offset; Charbpos lend = dl->end_charpos + dl->offset; if (beg_unchanged >= lstart && beg_unchanged <= lend) break; line++; } /* If the changes are below the visible area then if point hasn't moved return success otherwise fail in order to be safe. */ if (line > dla_end) { if (EQ (Fmarker_buffer (w->last_point[CURRENT_DISP]), w->buffer) && pointm == marker_position (w->last_point[CURRENT_DISP])) return 1; else return 0; } /* At this point we know what line the changes first affect. We now begin redrawing lines as long as we are still in the affected region and the line's size and positioning don't change. Otherwise we fail. If we fail we will have altered the desired structs which could lead to an assertion failure. However, if we fail the next thing that is going to happen is a full regen so we will actually end up being safe. */ w->last_modified[DESIRED_DISP] = make_fixnum (BUF_MODIFF (b)); w->last_facechange[DESIRED_DISP] = make_fixnum (BUF_FACECHANGE (b)); Fset_marker (w->last_start[DESIRED_DISP], make_fixnum (startp), w->buffer); Fset_marker (w->last_point[DESIRED_DISP], make_fixnum (pointm), w->buffer); first_line = last_line = line; while (line <= dla_end) { Charbpos old_start, old_end; struct display_line *cdl = Dynarr_atp (cdla, line); struct display_line *ddl = Dynarr_atp (ddla, line); struct display_block *db; int initial_size; assert (cdl->charpos == ddl->charpos); assert (cdl->end_charpos == ddl->end_charpos); assert (cdl->offset == ddl->offset); db = get_display_block_from_line (ddl, TEXT); initial_size = Dynarr_length (db->runes); old_start = ddl->charpos + ddl->offset; old_end = ddl->end_charpos + ddl->offset; /* If this is the first line being updated and it used propagation data, fail. Otherwise we'll be okay because we'll have the necessary propagation data. */ if (line == first_line && ddl->used_prop_data) return 0; generate_display_line (w, ddl, 0, ddl->charpos + ddl->offset, &prop, DESIRED_DISP); ddl->offset = 0; /* #### If there is propagated stuff the fail. We could probably actually deal with this if the line had propagated information when originally created by a full regeneration. */ if (prop) { Dynarr_free (prop); return 0; } /* If any line position parameters have changed or a cursor has disappeared or disappeared, fail. */ db = get_display_block_from_line (ddl, TEXT); if (cdl->ypos != ddl->ypos || cdl->ascent != ddl->ascent || cdl->descent != ddl->descent || cdl->top_clip != ddl->top_clip || (cdl->cursor_elt != -1 && ddl->cursor_elt == -1) || (cdl->cursor_elt == -1 && ddl->cursor_elt != -1) || old_start != ddl->charpos || old_end != ddl->end_charpos || initial_size != Dynarr_length (db->runes) || cdl->clear_findex != ddl->clear_findex) { return 0; } if (ddl->cursor_elt != -1) { w->last_point_x[DESIRED_DISP] = ddl->cursor_elt; w->last_point_y[DESIRED_DISP] = line; } last_line = line; /* If the extent changes end on the line we just updated then we're done. Otherwise go on to the next line. */ if (end_unchanged <= ddl->end_charpos) break; else line++; } redisplay_update_line (w, first_line, last_line, 1); return 1; } /* Attempt to update the display data structures based on knowledge of the changed region in the buffer. Returns a boolean indicating success or failure. If this function returns a failure then a regenerate_window _must_ be performed next in order to maintain invariants located here. */ static int regenerate_window_incrementally (struct window *w, Charbpos startp, Charbpos pointm) { struct buffer *b = XBUFFER (w->buffer); display_line_dynarr *cdla = window_display_lines (w, CURRENT_DISP); display_line_dynarr *ddla = window_display_lines (w, DESIRED_DISP); Charcount beg_unchanged, end_unchanged; Charcount extent_beg_unchanged, extent_end_unchanged; int dla_start = 0; int dla_end, line; Charbpos start_pos; /* If this function is called, the current and desired structures had better be identical. If they are not, then that is a bug. */ assert (Dynarr_length (cdla) == Dynarr_length (ddla)); /* We don't handle minibuffer windows yet. The minibuffer prompt screws us up. */ if (MINI_WINDOW_P (w)) return 0; extent_beg_unchanged = BUF_EXTENT_BEGIN_UNCHANGED (b); extent_end_unchanged = (BUF_EXTENT_END_UNCHANGED (b) == -1 ? -1 : BUF_Z (b) - BUF_EXTENT_END_UNCHANGED (b)); /* If nothing has changed in the buffer, then make sure point is ok and succeed. */ if (BUF_BEGIN_UNCHANGED (b) == -1 && BUF_END_UNCHANGED (b) == -1) return regenerate_window_extents_only_changed (w, startp, pointm, extent_beg_unchanged, extent_end_unchanged); /* We can't deal with deleted newlines. */ if (BUF_NEWLINE_WAS_DELETED (b)) return 0; beg_unchanged = BUF_BEGIN_UNCHANGED (b); end_unchanged = (BUF_END_UNCHANGED (b) == -1 ? -1 : BUF_Z (b) - BUF_END_UNCHANGED (b)); REGEN_INC_FIND_START_END; /* If the changed area starts before the visible area, give up. */ if (beg_unchanged < startp) return 0; /* Find what display line the buffer changes first affect. */ line = dla_start; while (line <= dla_end) { struct display_line *dl = Dynarr_atp (cdla, line); Charbpos lstart = dl->charpos + dl->offset; Charbpos lend = dl->end_charpos + dl->offset; if (beg_unchanged >= lstart && beg_unchanged <= lend) break; line++; } /* If the changes are below the visible area then if point hasn't moved return success otherwise fail in order to be safe. */ if (line > dla_end) return regenerate_window_extents_only_changed (w, startp, pointm, extent_beg_unchanged, extent_end_unchanged); else /* At this point we know what line the changes first affect. We now redraw that line. If the changes are contained within it we are going to succeed and can update just that one line. Otherwise we fail. If we fail we will have altered the desired structs which could lead to an assertion failure. However, if we fail the next thing that is going to happen is a full regen so we will actually end up being safe. */ { prop_block_dynarr *prop = NULL; struct display_line *cdl = Dynarr_atp (cdla, line); struct display_line *ddl = Dynarr_atp (ddla, line); assert (cdl->charpos == ddl->charpos); assert (cdl->end_charpos == ddl->end_charpos); assert (cdl->offset == ddl->offset); /* If the line continues to next display line, fail. */ if (ddl->line_continuation) return 0; /* If the line was generated using propagation data, fail. */ if (ddl->used_prop_data) return 0; generate_display_line (w, ddl, 0, ddl->charpos + ddl->offset, &prop, DESIRED_DISP); ddl->offset = 0; /* If there is propagated stuff then it is pretty much a guarantee that more than just the one line is affected. */ if (prop) { Dynarr_free (prop); return 0; } /* If the line continues to next display line, fail. */ if (ddl->line_continuation) return 0; /* If any line position parameters have changed or a cursor has disappeared or disappeared, fail. */ if (cdl->ypos != ddl->ypos || cdl->ascent != ddl->ascent || cdl->descent != ddl->descent || cdl->top_clip != ddl->top_clip || (cdl->cursor_elt != -1 && ddl->cursor_elt == -1) || (cdl->cursor_elt == -1 && ddl->cursor_elt != -1) || cdl->clear_findex != ddl->clear_findex) { return 0; } /* If the changed area also ends on this line, then we may be in business. Update everything and return success. */ if (end_unchanged >= ddl->charpos && end_unchanged <= ddl->end_charpos) { w->last_modified[DESIRED_DISP] = make_fixnum (BUF_MODIFF (b)); w->last_facechange[DESIRED_DISP] = make_fixnum (BUF_FACECHANGE (b)); Fset_marker (w->last_start[DESIRED_DISP], make_fixnum (startp), w->buffer); Fset_marker (w->last_point[DESIRED_DISP], make_fixnum (pointm), w->buffer); if (ddl->cursor_elt != -1) { w->last_point_x[DESIRED_DISP] = ddl->cursor_elt; w->last_point_y[DESIRED_DISP] = line; } redisplay_update_line (w, line, line, 1); regenerate_modeline (w); /* #### For now we just flush the cache until this has been tested. After that is done, this should correct the cache directly. */ Dynarr_reset (w->line_start_cache); /* Adjust the extent changed boundaries to remove any overlap with the buffer changes since we've just successfully updated that area. */ if (extent_beg_unchanged != -1 && extent_beg_unchanged >= beg_unchanged && extent_beg_unchanged < end_unchanged) extent_beg_unchanged = end_unchanged; if (extent_end_unchanged != -1 && extent_end_unchanged >= beg_unchanged && extent_end_unchanged < end_unchanged) extent_end_unchanged = beg_unchanged - 1; if (extent_end_unchanged <= extent_beg_unchanged) extent_beg_unchanged = extent_end_unchanged = -1; /* This could lead to odd results if it fails, but since the buffer changes update succeeded this probably will to. We already know that the extent changes start at or after the line because we checked before entering the loop. */ if (extent_beg_unchanged != -1 && extent_end_unchanged != -1 && ((extent_beg_unchanged < ddl->charpos) || (extent_end_unchanged > ddl->end_charpos))) return regenerate_window_extents_only_changed (w, startp, pointm, extent_beg_unchanged, extent_end_unchanged); else return 1; } } /* Oh, well. */ return 0; } /* Given a window and a point, update the given display lines such that point is displayed in the middle of the window. Return the window's new start position. */ static Charbpos regenerate_window_point_center (struct window *w, Charbpos point, int type) { Charbpos startp; /* We need to make sure that the modeline is generated so that the window height can be calculated correctly. */ ensure_modeline_generated (w, type); startp = start_with_line_at_pixpos (w, point, window_half_pixpos (w)); regenerate_window (w, startp, point, type); Fset_marker (w->start[type], make_fixnum (startp), w->buffer); return startp; } /* Given a window and a set of display lines, return a boolean indicating whether the given point is contained within. */ static int point_visible (struct window *w, Charbpos point, int type) { struct buffer *b = XBUFFER (w->buffer); display_line_dynarr *dla = window_display_lines (w, type); int first_line; if (Dynarr_length (dla) && Dynarr_begin (dla)->modeline) first_line = 1; else first_line = 0; if (Dynarr_length (dla) > first_line) { Charbpos start, end; struct display_line *dl = Dynarr_atp (dla, first_line); start = dl->charpos; end = BUF_Z (b) - w->window_end_pos[type] - 1; if (point >= start && point <= end) { if (!MINI_WINDOW_P (w) && scroll_on_clipped_lines) { dl = Dynarr_lastp (dla); if (point >= (dl->charpos + dl->offset) && point <= (dl->end_charpos + dl->offset)) return !dl->clip; else return 1; } else return 1; } else return 0; } else return 0; } /* Return pixel position the middle of the window, not including the modeline and any potential horizontal scrollbar. */ int window_half_pixpos (struct window *w) { return WINDOW_TEXT_TOP (w) + (WINDOW_TEXT_HEIGHT (w) >> 1); } /* Return the display line which is currently in the middle of the window W for display lines TYPE. */ int line_at_center (struct window *w, int type, Charbpos start, Charbpos point) { display_line_dynarr *dla; int half; int elt; int first_elt = (MINI_WINDOW_P (w) ? 0 : 1); if (type == CMOTION_DISP) regenerate_window (w, start, point, type); dla = window_display_lines (w, type); half = window_half_pixpos (w); for (elt = first_elt; elt < Dynarr_length (dla); elt++) { struct display_line *dl = Dynarr_atp (dla, elt); int line_bot = dl->ypos + dl->descent; if (line_bot > half) return elt; } /* We may not have a line at the middle if the end of the buffer is being displayed. */ return -1; } /* Return a value for point that would place it at the beginning of the line which is in the middle of the window. */ Charbpos point_at_center (struct window *w, int type, Charbpos start, Charbpos point) { /* line_at_center will regenerate the display structures, if necessary. */ int line = line_at_center (w, type, start, point); if (line == -1) return BUF_ZV (XBUFFER (w->buffer)); else { display_line_dynarr *dla = window_display_lines (w, type); struct display_line *dl = Dynarr_atp (dla, line); return dl->charpos; } } /* For a given window, ensure that the current visual representation is accurate. */ static void redisplay_window (Lisp_Object window, int skip_selected) { struct window *w = XWINDOW (window); struct frame *f = XFRAME (w->frame); struct device *d = XDEVICE (f->device); Lisp_Object old_buffer = w->buffer; Lisp_Object the_buffer = w->buffer; struct buffer *b; int echo_active = 0; int startp = 1; int pointm; int old_startp = 1; int old_pointm = 1; int selected_in_its_frame; int selected_globally; int skip_output = 0; int truncation_changed; int inactive_minibuffer = (MINI_WINDOW_P (w) && (f != device_selected_frame (d)) && !is_surrogate_for_selected_frame (f)); /* #### In the new world this function actually does a bunch of optimizations such as buffer-based scrolling, but none of that is implemented yet. */ /* If this is a combination window, do its children; that's all. The selected window is always a leaf so we don't check for skip_selected here. */ if (!NILP (w->vchild)) { redisplay_windows (w->vchild, skip_selected); return; } if (!NILP (w->hchild)) { redisplay_windows (w->hchild, skip_selected); return; } /* Is this window the selected window on its frame? */ selected_in_its_frame = (w == XWINDOW (FRAME_SELECTED_WINDOW (f))); selected_globally = selected_in_its_frame && EQ (DEVICE_CONSOLE (d), Vselected_console) && XDEVICE (CONSOLE_SELECTED_DEVICE (XCONSOLE (DEVICE_CONSOLE (d)))) == d && XFRAME (DEVICE_SELECTED_FRAME (d)) == f; if (skip_selected && selected_in_its_frame) return; /* It is possible that the window is not fully initialized yet. */ if (NILP (w->buffer)) return; if (MINI_WINDOW_P (w) && echo_area_active (f)) { w->buffer = the_buffer = Vecho_area_buffer; echo_active = 1; } b = XBUFFER (w->buffer); if (echo_active) { old_pointm = selected_globally ? BUF_PT (b) : marker_position (w->pointm[CURRENT_DISP]); pointm = 1; } else { if (selected_globally) { pointm = BUF_PT (b); } else { pointm = marker_position (w->pointm[CURRENT_DISP]); if (pointm < BUF_BEGV (b)) pointm = BUF_BEGV (b); else if (pointm > BUF_ZV (b)) pointm = BUF_ZV (b); } } Fset_marker (w->pointm[DESIRED_DISP], make_fixnum (pointm), the_buffer); /* Added 2-1-10 -- we should never have empty face or glyph cachels because we initialized them at startup and the only way to reduce their number is through calling reset_face_cachels() or reset_glyph_cachels(), which as a side effect sets up a number of standard entries 2011-10-29 -- We were managing to hit the glyph_cachels assert in certain contexts where VM was creating a lot of frames. I don't have a full analysis, but I suspect that we were failing to setup the glyph_cachels at about l. 961 of frame.c, and a message was being sent to the echo area before the initialization was complete. This triggered a redisplay of the minibuffer window (this part is confirmed), and thus this assert. */ assert (Dynarr_length (w->face_cachels)); assert (Dynarr_length (w->glyph_cachels)); /* If the buffer has changed we have to invalidate all of our face cache elements. */ if ((!echo_active && b != window_display_buffer (w)) || f->faces_changed) reset_face_cachels (w); else mark_face_cachels_as_not_updated (w); /* Ditto the glyph cache elements, although we do *not* invalidate the cache purely because glyphs have changed - this is now handled by the dirty flag.*/ if ((!echo_active && b != window_display_buffer (w)) || f->faces_changed) reset_glyph_cachels (w); else mark_glyph_cachels_as_not_updated (w); /* If the marker's buffer is not the window's buffer, then we need to find a new starting position. */ if (!MINI_WINDOW_P (w) && !EQ (Fmarker_buffer (w->start[CURRENT_DISP]), w->buffer)) { startp = regenerate_window_point_center (w, pointm, DESIRED_DISP); goto regeneration_done; } if (echo_active) { old_startp = marker_position (w->start[CURRENT_DISP]); startp = 1; } else { startp = marker_position (w->start[CURRENT_DISP]); if (startp < BUF_BEGV (b)) startp = BUF_BEGV (b); else if (startp > BUF_ZV (b)) startp = BUF_ZV (b); } Fset_marker (w->start[DESIRED_DISP], make_fixnum (startp), the_buffer); truncation_changed = (find_window_mirror (w)->truncate_win != (unsigned int) window_truncation_on (w)); /* If w->force_start is set, then some function set w->start and we should display from there and change point, if necessary, to ensure that it is visible. */ if (w->force_start || inactive_minibuffer) { w->force_start = 0; w->last_modified[DESIRED_DISP] = Qzero; w->last_facechange[DESIRED_DISP] = Qzero; regenerate_window (w, startp, pointm, DESIRED_DISP); if (!point_visible (w, pointm, DESIRED_DISP) && !inactive_minibuffer) { pointm = point_at_center (w, DESIRED_DISP, 0, 0); if (selected_globally) BUF_SET_PT (b, pointm); Fset_marker (w->pointm[DESIRED_DISP], make_fixnum (pointm), the_buffer); /* #### BUFU amounts of overkill just to get the cursor location marked properly. FIX ME FIX ME FIX ME */ regenerate_window (w, startp, pointm, DESIRED_DISP); } goto regeneration_done; } /* If nothing has changed since the last redisplay, then we just need to make sure that point is still visible. */ if (XFIXNUM (w->last_modified[CURRENT_DISP]) >= BUF_MODIFF (b) && XFIXNUM (w->last_facechange[CURRENT_DISP]) >= BUF_FACECHANGE (b) && pointm >= startp /* This check is to make sure we restore the minibuffer after a temporary change to the echo area. */ && !(MINI_WINDOW_P (w) && f->buffers_changed) && !f->frame_changed && !truncation_changed /* check whether start is really at the beginning of a line GE */ && (!w->start_at_line_beg || beginning_of_line_p (b, startp)) ) { /* Check if the cursor has actually moved. */ if (EQ (Fmarker_buffer (w->last_point[CURRENT_DISP]), w->buffer) && pointm == marker_position (w->last_point[CURRENT_DISP]) && selected_globally && !w->windows_changed && !f->clip_changed && !f->extents_changed && !f->faces_changed && !f->glyphs_changed && !f->subwindows_changed /* && !f->subwindows_state_changed*/ && !f->point_changed && !f->windows_structure_changed) { /* If not, we're done. */ if (f->modeline_changed) regenerate_modeline (w); skip_output = 1; goto regeneration_done; } else { /* If the new point is visible in the redisplay structures, then let the output update routines handle it, otherwise do things the hard way. */ if (!w->windows_changed && !f->clip_changed && !f->extents_changed && !f->faces_changed && !f->glyphs_changed && !f->subwindows_changed /* && !f->subwindows_state_changed*/ && !f->windows_structure_changed) { if (point_visible (w, pointm, CURRENT_DISP) && w->last_point_x[CURRENT_DISP] != -1 && w->last_point_y[CURRENT_DISP] != -1) { if (redisplay_move_cursor (w, pointm, FRAME_TTY_P (f))) { /* Always regenerate in case it is displaying the current line or column. */ regenerate_modeline (w); skip_output = 1; goto regeneration_done; } } else if (!selected_in_its_frame && !f->point_changed) { if (f->modeline_changed) regenerate_modeline (w); skip_output = 1; goto regeneration_done; } } /* If we weren't able to take the shortcut method, then use the brute force method. */ regenerate_window (w, startp, pointm, DESIRED_DISP); if (point_visible (w, pointm, DESIRED_DISP)) goto regeneration_done; } } /* Check if the starting point is no longer at the beginning of a line, in which case find a new starting point. We also recenter if our start position is equal to point-max. Otherwise we'll end up with a blank window. */ else if (((w->start_at_line_beg || MINI_WINDOW_P (w)) && !(startp == BUF_BEGV (b) || BUF_FETCH_CHAR (b, startp - 1) == '\n')) || (pointm == startp && EQ (Fmarker_buffer (w->last_start[CURRENT_DISP]), w->buffer) && startp < marker_position (w->last_start[CURRENT_DISP])) || (startp == BUF_ZV (b))) { startp = regenerate_window_point_center (w, pointm, DESIRED_DISP); goto regeneration_done; } /* See if we can update the data structures locally based on knowledge of what changed in the buffer. */ else if (!w->windows_changed && !f->clip_changed && !f->faces_changed && !f->glyphs_changed && !f->subwindows_changed /* && !f->subwindows_state_changed*/ && !f->windows_structure_changed && !f->frame_changed && !truncation_changed && pointm >= startp && regenerate_window_incrementally (w, startp, pointm)) { if (f->modeline_changed || XFIXNUM (w->last_modified[CURRENT_DISP]) < BUF_MODIFF (b) || XFIXNUM (w->last_facechange[CURRENT_DISP]) < BUF_FACECHANGE (b)) regenerate_modeline (w); skip_output = 1; goto regeneration_done; } /* #### This is where a check for structure based scrolling would go. */ /* If all else fails, try just regenerating and see what happens. */ else { regenerate_window (w, startp, pointm, DESIRED_DISP); if (point_visible (w, pointm, DESIRED_DISP)) goto regeneration_done; } /* We still haven't gotten the window regenerated with point visible. Next we try scrolling a little and see if point comes back onto the screen. */ if (scroll_step > 0) { int scrolled = scroll_conservatively; for (; scrolled >= 0; scrolled -= scroll_step) { startp = vmotion (w, startp, (pointm < startp) ? -scroll_step : scroll_step, 0); regenerate_window (w, startp, pointm, DESIRED_DISP); if (point_visible (w, pointm, DESIRED_DISP)) goto regeneration_done; } } /* We still haven't managed to get the screen drawn with point on the screen, so just center it and be done with it. */ startp = regenerate_window_point_center (w, pointm, DESIRED_DISP); regeneration_done: /* If the window's frame is changed then reset the current display lines in order to force a full repaint. */ if (f->frame_changed) { display_line_dynarr *cla = window_display_lines (w, CURRENT_DISP); Dynarr_reset (cla); } /* Must do this before calling redisplay_output_window because it sets some markers on the window. */ if (echo_active) { w->buffer = old_buffer; Fset_marker (w->pointm[DESIRED_DISP], make_fixnum (old_pointm), old_buffer); Fset_marker (w->start[DESIRED_DISP], make_fixnum (old_startp), old_buffer); } /* These also have to be set before calling redisplay_output_window since it sets the CURRENT_DISP values based on them. */ w->last_modified[DESIRED_DISP] = make_fixnum (BUF_MODIFF (b)); w->last_facechange[DESIRED_DISP] = make_fixnum (BUF_FACECHANGE (b)); Fset_marker (w->last_start[DESIRED_DISP], make_fixnum (startp), w->buffer); Fset_marker (w->last_point[DESIRED_DISP], make_fixnum (pointm), w->buffer); if (!skip_output) { Charbpos start = marker_position (w->start[DESIRED_DISP]); Charbpos end = (w->window_end_pos[DESIRED_DISP] == -1 ? BUF_ZV (b) : BUF_Z (b) - w->window_end_pos[DESIRED_DISP] - 1); /* Don't pollute the cache if not sure if we are correct */ if (w->start_at_line_beg) update_line_start_cache (w, start, end, pointm, 1); redisplay_output_window (w); /* * If we just displayed the echo area, the line start cache is * no longer valid, because the minibuffer window is associated * with the window now. */ if (echo_active) w->line_cache_last_updated = make_fixnum (-1); } /* #### This should be dependent on face changes and will need to be somewhere else once tty updates occur on a per-frame basis. */ mark_face_cachels_as_clean (w); /* The glyph cachels only get dirty if someone changed something. Since redisplay has now effectively ended we can reset the dirty flag since everything must be up-to-date. */ if (glyphs_changed) mark_glyph_cachels_as_clean (w); w->windows_changed = 0; } /* Call buffer_reset_changes for all buffers present in any window currently visible in all frames on all devices. #### There has to be a better way to do this. */ static int reset_buffer_changes_mapfun (struct window *w, void *UNUSED (closure)) { buffer_reset_changes (XBUFFER (w->buffer)); return 0; } static void reset_buffer_changes (void) { Lisp_Object frmcons, devcons, concons; FRAME_LOOP_NO_BREAK (frmcons, devcons, concons) { struct frame *f = XFRAME (XCAR (frmcons)); if (FRAME_REPAINT_P (f)) map_windows (f, reset_buffer_changes_mapfun, 0); } } /* Ensure that all windows underneath the given window in the window hierarchy are correctly displayed. */ static void redisplay_windows (Lisp_Object window, int skip_selected) { for (; !NILP (window) ; window = XWINDOW (window)->next) { redisplay_window (window, skip_selected); } } /* Register an action to be called at the end of redisplay. in_display is 0 when this is called. This is used when it is discovered that an action needs to be taken, but it's during redisplay, so it's not safe. (Typically, it's an action that needs to enter redisplay, which can't happen reentrantly.) NEVER signal an error in these functions. */ void register_post_redisplay_action (void (*fun) (Lisp_Object), Lisp_Object arg) { Vpost_redisplay_actions = nconc2 (Vpost_redisplay_actions, list1 (Fcons (make_opaque_ptr ((void *) fun), arg))); } static int running_post_redisplay_actions; static void run_post_redisplay_actions (void) { int depth; if (running_post_redisplay_actions) return; depth = internal_bind_int (&running_post_redisplay_actions, 1); /* If the function pushes further actions, they will be tacked onto the end of the list, and we'll run them when we're done with the current ones. */ while (!NILP (Vpost_redisplay_actions)) { Lisp_Object car = XCAR (Vpost_redisplay_actions); void (*fun) (Lisp_Object) = (void (*)(Lisp_Object)) get_opaque_ptr (XCAR (car)); (*fun) (XCDR (car)); free_opaque_ptr (XCAR (car)); free_cons (car); Vpost_redisplay_actions = XCDR (Vpost_redisplay_actions); } unbind_to (depth); } static int the_ritual_suicide_has_been_cancelled = 0; void redisplay_cancel_ritual_suicide(void) { the_ritual_suicide_has_been_cancelled = 1; } #ifdef ERROR_CHECK_TRAPPING_PROBLEMS static Lisp_Object commit_ritual_suicide (Lisp_Object UNUSED (ceci_nest_pas_une_pipe)) { if (!the_ritual_suicide_has_been_cancelled) { assert (!in_display); } else the_ritual_suicide_has_been_cancelled = 0; return Qnil; } #endif /* Within the guts of redisplay, we are defenseless and cannot allow any of the following to happen: 1) garbage collection 2) QUIT 3) any non-local exits 4) frame size changes 5) deletion of any buffers, windows, frames, etc. 6) modification of buffer text 7) reentrant entry of redisplay (see the stack trace above generate_displayable_area()) The general reason is that the redisplay code is written to assume that it is the only code running, and thus (a) cannot tolerate structures changed out from under it (hence 1, 4, 5, 6, 7) and (b) at various points within redisplay the redisplay structures may be in an inconsistent state and there are no unwind-protects to clean the structures up in case of non-local exit (hence 2, 3). Fixing redisplay to address these issues is hard and perhaps not worth it (and might slow things down a fair amount). We address 1, 4, 5 and 6 ourselves inside of enter_redisplay_critical_section() by simply inhibiting them, but we cannot handle 2 and 3, which must be handled at the actual point where they may occur (especially, internal_equal() or any place that may call Lisp), by wrapping the code in call_trapping_problems() or call_with_suspended_errors(). [[ NOTE: We could address QUIT by inhibiting it but this would be anti-social because it would prevent the user from interrupting any Lisp code called within the critical section. With the call_*() wrapping, C-g will interrupt the Lisp code and throw back to the innermost wrapping. ]] In fact we do turn off QUIT handling, since it's just too dangerous otherwise. See below. Code calling enter_redisplay_critical_section() must check for reentrancy (#7) and take appropriate corrective action. To help debug potential problems, we arrange (when ERROR_CHECK_TRAPPING_PROBLEMS is set) to crash automatically every time we execute QUIT or call Lisp code unless proper wrapping is in place, as well as further checks when we actually Fsignal(), Fthrow(), garbage_collect_1(). #### If a frame-size change does occur we should probably actually be preempting redisplay. */ /* Count of number of recursive times we call enter_redisplay_critical_section() or enter_redisplay_critical_section_maybe(). enter_redisplay_critical_section() cannot occur reentrantly but we have to know in the *maybe() version whether to exit the section when we're done. */ static int in_display_nesting; static Lisp_Object end_hold_frame_size_changes (Lisp_Object UNUSED (obj)) { if (!hold_frame_size_changes) { /* we used to have a function to do this for only one frame, and it was typical to call it at the end of a critical section (which occurs once per frame); but what then happens if multiple frames have frame changes held up? This means we are O(N^2) over frames. I seriously doubt it matters. --ben */ Lisp_Object frmcons, devcons, concons; FRAME_LOOP_NO_BREAK (frmcons, devcons, concons) { struct frame *f = XFRAME (XCAR (frmcons)); if (f->size_change_pending) change_frame_size (f, f->new_width, f->new_height, 0); } } return Qnil; } /* Call this to temporarily prevent frame-size changes from being processed. To undo, use unbind_to(), passing it the value returned by this function. */ int begin_hold_frame_size_changes (void) { int depth = specpdl_depth (); record_unwind_protect (end_hold_frame_size_changes, Qnil); internal_bind_int (&hold_frame_size_changes, 1 + hold_frame_size_changes); return depth; } int enter_redisplay_critical_section (void) { int depth = specpdl_depth (); /* Reentrant entry is deadly. The calling function must check for this. */ assert (!in_display); begin_hold_frame_size_changes (); /* Make sure in_display gets reset, but don't set it yet so that commit_ritual_suicide() can be used. */ internal_bind_int (&in_display, 0); internal_bind_int (&in_display_nesting, 1 + in_display_nesting); #ifdef ERROR_CHECK_TRAPPING_PROBLEMS /* Force every call to QUIT to check for in_displayness. This will verify proper wrapping, as in the previous comment, aborting if not. */ something_happened++; /* Verify that no nonlocal exits blow past us. */ record_unwind_protect (commit_ritual_suicide, Qnil); #endif in_display++; set_trapping_problems_flags (INHIBIT_ANY_CHANGE_AFFECTING_REDISPLAY); /* Even checking for QUIT can cause arbitrary Lisp code to be executed, e.g. through a menu handler. We really don't want that happening inside of redisplay. Code that we `eval' is at least written with the expectation that it's inside of redisplay, and shouldn't try anything weird; but that's not the case for menu code (e.g. custom loads huge amounts of LISP FILES from a menu handler! FMH!). Safest just to turn this off. We could turn it on using UNINHIBIT_QUIT or begin_do_check_for_quit() in certain places if we want, if we know it's not in an especially tricky place. */ begin_dont_check_for_quit (); return depth; } void exit_redisplay_critical_section (int depth) { in_display--; assert (!in_display); unbind_to (depth); run_post_redisplay_actions (); } /* Enter the redisplay critical section if we're not already in it. This is for code that needs frame changes held up and other protections from being inside, but doesn't modify the redisplay structures, and doesn't look at them in a way that they will be confused by inconsistencies. */ int enter_redisplay_critical_section_maybe (void) { if (!in_display) return enter_redisplay_critical_section (); else return internal_bind_int (&in_display_nesting, 1 + in_display_nesting); } void exit_redisplay_critical_section_maybe (int depth) { if (in_display_nesting == 1) exit_redisplay_critical_section (depth); else unbind_to (depth); } /* Ensure that all windows on the given frame are correctly displayed. Return non-zero if pre-empted. */ int redisplay_frame (struct frame *f, int preemption_check) { struct device *d = XDEVICE (f->device); int depth; assert (f->init_finished); /* NOTE: Without sufficient checks for stream frames, we got weird crashes in pdump. These came and went very easily -- adding the critical-section code for redisplay was enough to trigger them. Perhaps I should have debugged them but there didn't seem to be any point. --ben */ if (FRAME_STREAM_P (f)) /* nothing to do */ return 0; /* Reentrancy into redisplay can be deadly. See stack trace above generate_displayable_area(). */ if (in_display) return 1; if (preemption_check && !DEVICE_IMPL_FLAG (d, XDEVIMPF_DONT_PREEMPT_REDISPLAY)) { /* The preemption check itself takes a lot of time, so normally don't do it here. We do it if called from Lisp, though (`redisplay-frame'). */ int preempted; REDISPLAY_PREEMPTION_CHECK; if (preempted) return 1; } if (!internal_equal (f->old_buffer_alist, f->buffer_alist, 0)) { Lisp_Object frame; f->old_buffer_alist = Freplace_list (f->old_buffer_alist, f->buffer_alist); frame = wrap_frame (f); va_run_hook_with_args (Qbuffer_list_changed_hook, 1, frame); } /* Before we put a hold on frame size changes, attempt to process any which are already pending. */ if (f->size_change_pending) change_frame_size (f, f->new_width, f->new_height, 0); /* If frame size might need to be changed, due to changed size of toolbars, scrollbars etc, change it now */ if (f->size_slipped) { adjust_frame_size (f); assert (!f->size_slipped); } /* The menubar, toolbar, and icon updates should be done before enter_redisplay_critical_section is called and we are officially `in_display'. They is because they tend to eval Lisp code, which needs to be carefully wrapped within the critical section (and hence is difficult to debug). */ #ifdef HAVE_MENUBARS /* Update the menubar. It is done first since it could change the menubar's visibility. This way we avoid having flashing caused by an Expose event generated by the visibility change being handled. */ update_frame_menubars (f); #endif /* HAVE_MENUBARS */ #ifdef HAVE_TOOLBARS /* Update the toolbars geometry. We don't update the toolbars themselves at this point since the space they are trying to occupy may currently by occupied by gutter elements. Instead we update the geometry, then update the gutter geometry, then update the gutters - which will cause mapped windows to be repositioned - and finally update the toolbars. */ update_frame_toolbars_geometry (f); #endif /* HAVE_TOOLBARS */ /* Gutter update proper has to be done inside display when no frame size changes can occur, thus we separately update the gutter geometry here if it needs it. */ update_frame_gutter_geometry (f); /* If we clear the frame we have to force its contents to be redrawn. */ if (f->clear) f->frame_changed = 1; /* Invalidate the subwindow caches. We use subwindows_changed here to cause subwindows to get instantiated. This is because subwindows_state_changed is less strict - dealing with things like the clicked state of button. We have to do this before redisplaying the gutters as subwindows get unmapped in the process.*/ if (f->frame_changed) reset_frame_subwindow_instance_cache (f); if (f->frame_changed || f->subwindows_changed) { /* we have to do this so the gutter gets regenerated. */ reset_gutter_display_lines (f); } depth = enter_redisplay_critical_section (); /* ----------------- BEGIN CRITICAL REDISPLAY SECTION ---------------- */ /* See comments in enter_redisplay_critical_section() */ MAYBE_DEVMETH (d, frame_output_begin, (f)); /* We can now update the gutters, safe in the knowledge that our efforts won't get undone. */ /* This can call lisp, but redisplay is protected by binding inhibit_quit. More importantly the code involving display lines *assumes* that GC will not happen and so does not GCPRO anything. Since we use this code the whole time with the gutters we cannot allow GC to happen when manipulating the gutters. This must be inside of the critical section for various reasons. For example, it messes with display structures, which be left in an inconsistent state. */ update_frame_gutters (f); /* Erase the frame before outputting its contents. */ if (f->clear) { MAYBE_DEVMETH (d, clear_frame, (f)); } /* Do the selected window first. */ redisplay_window (FRAME_SELECTED_WINDOW (f), 0); /* Then do the rest. */ redisplay_windows (f->root_window, 1); MAYBE_DEVMETH (d, frame_output_end, (f)); update_frame_title (f); #ifdef HAVE_TOOLBARS /* Finally update the toolbars. It seems its possible to get in a cycle between updating the gutter and the toolbars. Basically we want to end up with both being up-to-date and this doesn't seem possible in a single pass. */ update_frame_toolbars (f); #endif /* HAVE_TOOLBARS */ CLASS_RESET_CHANGED_FLAGS (f); f->window_face_cache_reset = 0; f->echo_area_garbaged = 0; f->clear = 0; if (!f->size_change_pending) f->size_changed = 0; /* ----------------- END CRITICAL REDISPLAY SECTION ---------------- */ /* Allow frame size changes to occur again. */ exit_redisplay_critical_section (depth); return 0; } /* Ensure that all frames on the given device are correctly displayed. If AUTOMATIC is non-zero, and the device implementation indicates no automatic redisplay, as printers do, then the device is not redisplayed. AUTOMATIC is set to zero when called from lisp functions (redraw-device) and (redisplay-device), and to non-zero when called from "lazy" redisplay(); */ static int redisplay_device (struct device *d, int automatic) { Lisp_Object frame, frmcons; int size_change_failed = 0; struct frame *f; if (automatic && DEVICE_IMPL_FLAG (d, XDEVIMPF_NO_AUTO_REDISPLAY)) return 0; if (DEVICE_STREAM_P (d)) /* nothing to do */ return 0; /* It is possible that redisplay has been called before the device is fully initialized, or that the console implementation allows frameless devices. If so then continue with the next device. */ if (NILP (DEVICE_SELECTED_FRAME (d))) return 0; if (!DEVICE_IMPL_FLAG (d, XDEVIMPF_DONT_PREEMPT_REDISPLAY)) { int preempted; REDISPLAY_PREEMPTION_CHECK; if (preempted) return 1; } /* Always do the selected frame first. */ frame = DEVICE_SELECTED_FRAME (d); f = XFRAME (frame); if (f->icon_changed || f->windows_changed) update_frame_icon (f); if (FRAME_REPAINT_P (f)) { if (CLASS_REDISPLAY_FLAGS_CHANGEDP (f)) { int preempted = redisplay_frame (f, 1); if (preempted) return 1; } /* If the frame redisplay did not get preempted, then this flag should have gotten set to 0. It might be possible for that not to happen if a size change event were to occur at an odd time. To make sure we don't miss anything we simply don't reset the top level flags until the condition ends up being in the right state. */ if (f->size_changed) size_change_failed = 1; } DEVICE_FRAME_LOOP (frmcons, d) { f = XFRAME (XCAR (frmcons)); if (f == XFRAME (DEVICE_SELECTED_FRAME (d))) continue; if (f->icon_changed || f->windows_changed) update_frame_icon (f); if (FRAME_REPAINT_P (f)) { if (CLASS_REDISPLAY_FLAGS_CHANGEDP (f)) { int preempted = redisplay_frame (f, 1); if (preempted) return 1; } if (f->size_change_pending) size_change_failed = 1; } } /* If we get here then we redisplayed all of our frames without getting preempted so mark ourselves as clean. */ CLASS_RESET_CHANGED_FLAGS (d); if (!size_change_failed) d->size_changed = 0; return 0; } /* Ensure that all windows on all frames on all devices are displaying the current contents of their respective buffers. */ static void redisplay_without_hooks (void) { Lisp_Object devcons, concons; int size_change_failed = 0; PROFILE_DECLARE (); PROFILE_RECORD_ENTERING_SECTION (QSin_redisplay); if (asynch_device_change_pending) handle_asynch_device_change (); if (!GLOBAL_REDISPLAY_FLAGS_CHANGEDP && !disable_preemption && preemption_count < max_preempts) goto done; DEVICE_LOOP_NO_BREAK (devcons, concons) { struct device *d = XDEVICE (XCAR (devcons)); int preempted; if (CLASS_REDISPLAY_FLAGS_CHANGEDP (d)) { preempted = redisplay_device (d, 1); if (preempted) { preemption_count++; RESET_CHANGED_SET_FLAGS; goto done; } /* See comment in redisplay_device. */ if (d->size_changed) size_change_failed = 1; } } preemption_count = 0; /* Mark redisplay as accurate */ GLOBAL_RESET_CHANGED_FLAGS; RESET_CHANGED_SET_FLAGS; if (faces_changed) { mark_all_faces_as_clean (); faces_changed = 0; } if (!size_change_failed) size_changed = 0; reset_buffer_changes (); done: #ifdef ERROR_CHECK_DISPLAY sledgehammer_check_redisplay_structs (); #endif /* ERROR_CHECK_DISPLAY */ PROFILE_RECORD_EXITING_SECTION (QSin_redisplay); } /* Note: All places in the C code that call redisplay() are prepared to handle GCing, which can happen from run_pre_idle_hook(). However, we can't currently handle GC inside the guts of redisplay; see enter_redisplay_critical_section(). (#### What about other external entry points to the redisplay code? Someone should go through and make sure that all callers can handle GC there, too.) */ void redisplay (void) { run_pre_idle_hook (); redisplay_no_pre_idle_hook (); } void redisplay_no_pre_idle_hook (void) { if (last_display_warning_tick != display_warning_tick && !inhibit_warning_display) { /* If an error occurs during this function, oh well. If we report another warning, we could get stuck in an infinite loop reporting warnings. */ call0_trapping_problems (0, Qdisplay_warning_buffer, INHIBIT_EXISTING_PERMANENT_DISPLAY_OBJECT_DELETION); last_display_warning_tick = display_warning_tick; } redisplay_without_hooks (); } Lisp_Object eval_within_redisplay (Lisp_Object dont_trust_this_damn_sucker) { return eval_in_buffer_trapping_problems ("Error calling function within redisplay", current_buffer, dont_trust_this_damn_sucker, 0); } /* Efficiently determine the window line number, and return a pointer to its printed representation. Do this regardless of whether line-number-mode is on. The first line in the buffer is counted as 1. If narrowing is in effect, the lines are counted from the beginning of the visible portion of the buffer. */ static Ascbyte * window_line_number (struct window *w, int type) { struct device *d = XDEVICE (XFRAME (w->frame)->device); struct buffer *b = XBUFFER (w->buffer); /* Be careful in the order of these tests. The first clause will fail if DEVICE_SELECTED_FRAME == Qnil (since w->frame cannot be). This can occur when the frame title is computed really early */ Charbpos pos = ((EQ (DEVICE_SELECTED_FRAME (d), w->frame) && (w == XWINDOW (FRAME_SELECTED_WINDOW (device_selected_frame (d)))) && EQ (DEVICE_CONSOLE (d), Vselected_console) && XDEVICE (CONSOLE_SELECTED_DEVICE (XCONSOLE (DEVICE_CONSOLE (d)))) == d ) ? BUF_PT (b) : marker_position (w->pointm[type])); EMACS_INT line; line = buffer_line_number (b, pos, 1); { static Ascbyte window_line_number_buf[DECIMAL_PRINT_SIZE (long)]; long_to_string (window_line_number_buf, line + 1); return window_line_number_buf; } } /* Given a character representing an object in a modeline specification, return a string (stored into the global array `mode_spec_ibyte_string') with the information that object represents. This function is largely unchanged from previous versions of the redisplay engine. Warning! This code is also used for frame titles and can be called very early in the device/frame update process! JV */ static void decode_mode_spec (struct window *w, Ichar spec, int type) { Lisp_Object obj = Qnil; const Ascbyte *str = NULL; struct buffer *b = XBUFFER (w->buffer); Dynarr_reset (mode_spec_ibyte_string); switch (spec) { /* print buffer name */ case 'b': obj = b->name; break; /* print visited file name */ case 'f': obj = b->filename; break; /* print the current column */ case 'c': { Charbpos pt = (w == XWINDOW (Fselected_window (Qnil))) ? BUF_PT (b) : marker_position (w->pointm[type]); int col = column_at_point (b, pt, 1) + !!column_number_start_at_one; Ascbyte buf[DECIMAL_PRINT_SIZE (long)]; long_to_string (buf, col); Dynarr_add_many (mode_spec_ibyte_string, (const Ibyte *) buf, strlen (buf)); goto decode_mode_spec_done; } /* print the file coding system */ case 'C': { Lisp_Object codesys = b->buffer_file_coding_system; /* Be very careful here not to get an error. */ if (NILP (codesys) || SYMBOLP (codesys) || CODING_SYSTEMP (codesys)) { codesys = find_coding_system_for_text_file (codesys, 0); if (CODING_SYSTEMP (codesys)) obj = XCODING_SYSTEM_MNEMONIC (codesys); } } break; /* print the current line number */ case 'l': str = window_line_number (w, type); break; /* print value of mode-name (obsolete) */ case 'm': obj = b->mode_name; break; /* print hyphen and frame number, if != 1 */ case 'N': #ifdef HAVE_TTY { struct frame *f = XFRAME (w->frame); if (FRAME_TTY_P (f) && f->order_count > 1 && f->order_count <= 99999999) { /* Naughty, naughty */ Ascbyte *writable_str = alloca_array (Ascbyte, 10); sprintf (writable_str, "-%d", f->order_count); str = writable_str; } } #endif /* HAVE_TTY */ break; /* print Narrow if appropriate */ case 'n': if (BUF_BEGV (b) > BUF_BEG (b) || BUF_ZV (b) < BUF_Z (b)) str = " Narrow"; break; /* print %, * or hyphen, if buffer is read-only, modified or neither */ case '*': str = (!NILP (b->read_only) ? "%" : ((BUF_MODIFF (b) > BUF_SAVE_MODIFF (b)) ? "*" : "-")); break; /* print * or hyphen -- XEmacs change to allow a buffer to be read-only but still indicate whether it is modified. */ case '+': str = ((BUF_MODIFF (b) > BUF_SAVE_MODIFF (b)) ? "*" : (!NILP (b->read_only) ? "%" : "-")); break; /* #### defined in 19.29 decode_mode_spec, but not in modeline-format doc string. */ /* This differs from %* in that it ignores read-only-ness. */ case '&': str = ((BUF_MODIFF (b) > BUF_SAVE_MODIFF (b)) ? "*" : "-"); break; /* print process status */ case 's': obj = Fget_buffer_process (w->buffer); if (NILP (obj)) str = GETTEXT ("no process"); else obj = Fsymbol_name (Fprocess_status (obj)); break; /* Print name of selected frame. */ case 'S': obj = XFRAME (w->frame)->name; break; /* indicate TEXT or BINARY */ case 't': /* #### NT does not use this any more. Now what? */ str = "T"; break; /* print percent of buffer above top of window, or Top, Bot or All */ case 'p': { Charbpos pos = marker_position (w->start[type]); /* This had better be while the desired lines are being done. */ if (w->window_end_pos[type] <= BUF_Z (b) - BUF_ZV (b)) { if (pos <= BUF_BEGV (b)) str = "All"; else str = "Bottom"; } else if (pos <= BUF_BEGV (b)) str = "Top"; else { /* This hard limit is ok since the string it will hold has a fixed maximum length of 3. But just to be safe... */ Ascbyte buf[10]; Charcount chars = pos - BUF_BEGV (b); Charcount total = BUF_ZV (b) - BUF_BEGV (b); /* Avoid overflow on big buffers */ int percent = total > LONG_MAX/200 ? (chars + total/200) / (total / 100) : (chars * 100 + total/2) / total; /* We can't normally display a 3-digit number, so get us a 2-digit number that is close. */ if (percent == 100) percent = 99; sprintf (buf, "%d%%", percent); Dynarr_add_many (mode_spec_ibyte_string, (Ibyte *) buf, strlen (buf)); goto decode_mode_spec_done; } break; } /* print percent of buffer above bottom of window, perhaps plus Top, or print Bottom or All */ case 'P': { Charbpos toppos = marker_position (w->start[type]); Charbpos botpos = BUF_Z (b) - w->window_end_pos[type]; /* botpos is only accurate as of the last redisplay, so we can only treat it as a hint. In particular, after erase-buffer, botpos may be negative. */ if (botpos < toppos) botpos = toppos; if (botpos >= BUF_ZV (b)) { if (toppos <= BUF_BEGV (b)) str = "All"; else str = "Bottom"; } else { /* This hard limit is ok since the string it will hold has a fixed maximum length of around 6. But just to be safe... */ Ascbyte buf[10]; Charcount chars = botpos - BUF_BEGV (b); Charcount total = BUF_ZV (b) - BUF_BEGV (b); /* Avoid overflow on big buffers */ int percent = total > LONG_MAX/200 ? (chars + total/200) / (total / 100) : (chars * 100 + total/2) / max (total, 1); /* We can't normally display a 3-digit number, so get us a 2-digit number that is close. */ if (percent == 100) percent = 99; if (toppos <= BUF_BEGV (b)) sprintf (buf, "Top%d%%", percent); else sprintf (buf, "%d%%", percent); Dynarr_add_many (mode_spec_ibyte_string, (Ibyte *) buf, strlen (buf)); goto decode_mode_spec_done; } break; } /* print % */ case '%': str = "%"; break; /* print one [ for each recursive editing level. */ case '[': { int i; if (command_loop_level > 5) { str = "[[[... "; break; } for (i = 0; i < command_loop_level; i++) Dynarr_add (mode_spec_ibyte_string, '['); goto decode_mode_spec_done; } /* print one ] for each recursive editing level. */ case ']': { int i; if (command_loop_level > 5) { str = "...]]]"; break; } for (i = 0; i < command_loop_level; i++) Dynarr_add (mode_spec_ibyte_string, ']'); goto decode_mode_spec_done; } /* print infinitely many dashes -- handle at top level now */ case '-': break; } if (STRINGP (obj)) Dynarr_add_many (mode_spec_ibyte_string, XSTRING_DATA (obj), XSTRING_LENGTH (obj)); else if (str) Dynarr_add_many (mode_spec_ibyte_string, (Ibyte *) str, strlen (str)); decode_mode_spec_done: Dynarr_add (mode_spec_ibyte_string, '\0'); } /* Given a display line, free all of its data structures. */ static void free_display_line (struct display_line *dl) { int block; if (dl->display_blocks) { for (block = 0; block < Dynarr_largest (dl->display_blocks); block++) { struct display_block *db = Dynarr_atp (dl->display_blocks, block); Dynarr_free (db->runes); } Dynarr_free (dl->display_blocks); dl->display_blocks = NULL; } if (dl->left_glyphs) { Dynarr_free (dl->left_glyphs); dl->left_glyphs = NULL; } if (dl->right_glyphs) { Dynarr_free (dl->right_glyphs); dl->right_glyphs = NULL; } } /* Given an array of display lines, free them and all data structures contained within them. */ void free_display_lines (display_line_dynarr *dla) { int line; for (line = 0; line < Dynarr_largest (dla); line++) { free_display_line (Dynarr_atp (dla, line)); } Dynarr_free (dla); } /* Call internal free routine for each set of display lines. */ void free_display_structs (struct window_mirror *mir) { if (mir->current_display_lines) { free_display_lines (mir->current_display_lines); mir->current_display_lines = 0; } if (mir->desired_display_lines) { free_display_lines (mir->desired_display_lines); mir->desired_display_lines = 0; } } static void mark_glyph_block_dynarr (glyph_block_dynarr *gba) { if (gba) { glyph_block *gb = Dynarr_begin (gba); glyph_block *gb_last = Dynarr_past_lastp (gba); for (; gb < gb_last; gb++) { if (!NILP (gb->glyph)) mark_object (gb->glyph); if (!NILP (gb->extent)) mark_object (gb->extent); } } } /* See the comment in image_instantiate_cache_result as to why marking the glyph will also mark the image_instance. */ void mark_redisplay_structs (display_line_dynarr *dla) { display_line *dl = Dynarr_begin (dla); display_line *dl_last = Dynarr_past_lastp (dla); for (; dl < dl_last; dl++) { display_block_dynarr *dba = dl->display_blocks; display_block *db = Dynarr_begin (dba); display_block *db_last = Dynarr_past_lastp (dba); for (; db < db_last; db++) { rune_dynarr *ra = db->runes; rune *r = Dynarr_begin (ra); rune *r_last = Dynarr_past_lastp (ra); for (; r < r_last; r++) { if (r->type == RUNE_DGLYPH) { if (!NILP (r->object.dglyph.glyph)) mark_object (r->object.dglyph.glyph); if (!NILP (r->object.dglyph.extent)) mark_object (r->object.dglyph.extent); } } } mark_glyph_block_dynarr (dl->left_glyphs); mark_glyph_block_dynarr (dl->right_glyphs); } } /* Info on line-start cache: (Info-goto-node "(internals)Line Start Cache") */ /* This will get used quite a bit so we don't want to be constantly allocating and freeing it. */ static line_start_cache_dynarr *internal_cache; /* Makes internal_cache represent the TYPE display structs and only the TYPE display structs. */ static void update_internal_cache_list (struct window *w, int type) { int line; display_line_dynarr *dla = window_display_lines (w, type); Dynarr_reset (internal_cache); for (line = 0; line < Dynarr_length (dla); line++) { struct display_line *dl = Dynarr_atp (dla, line); if (dl->modeline) continue; else { struct line_start_cache lsc; lsc.start = dl->charpos; lsc.end = dl->end_charpos; lsc.height = dl->ascent + dl->descent; Dynarr_add (internal_cache, lsc); } } } /* Reset the line cache if necessary. This should be run at the beginning of any function which access the cache. */ static void validate_line_start_cache (struct window *w) { struct buffer *b = XBUFFER (w->buffer); struct frame *f = XFRAME (w->frame); if (!w->line_cache_validation_override) { /* f->extents_changed used to be in here because extent face and size changes can cause text shifting. However, the extent covering the region is constantly having its face set and priority altered by the mouse code. This means that the line start cache is constantly being invalidated. This is bad since the mouse code also triggers heavy usage of the cache. Since it is an unlikely that f->extents being changed indicates that the cache really needs to be updated and if it does redisplay will catch it pretty quickly we no longer invalidate the cache if it is set. This greatly speeds up dragging out regions with the mouse. */ if (XFIXNUM (w->line_cache_last_updated) < BUF_MODIFF (b) || f->faces_changed || f->clip_changed) { Dynarr_reset (w->line_start_cache); } } } /* Return the very first buffer position contained in the given window's cache, or -1 if the cache is empty. Assumes that the cache is valid. */ static Charbpos line_start_cache_start (struct window *w) { line_start_cache_dynarr *cache = w->line_start_cache; if (!Dynarr_length (cache)) return -1; else return Dynarr_begin (cache)->start; } /* Return the very last buffer position contained in the given window's cache, or -1 if the cache is empty. Assumes that the cache is valid. */ static Charbpos line_start_cache_end (struct window *w) { line_start_cache_dynarr *cache = w->line_start_cache; if (!Dynarr_length (cache)) return -1; else return Dynarr_lastp (cache)->end; } /* Return the index of the line POINT is contained within in window W's line start cache. It will enlarge the cache or move the cache window in order to have POINT be present in the cache. MIN_PAST is a guarantee of the number of entries in the cache present on either side of POINT (unless a buffer boundary is hit). If MIN_PAST is -1 then it will be treated as 0, but the cache window will not be allowed to shift. Returns -1 if POINT cannot be found in the cache for any reason. */ int point_in_line_start_cache (struct window *w, Charbpos point, int min_past) { struct buffer *b = XBUFFER (w->buffer); line_start_cache_dynarr *cache = w->line_start_cache; int top, bottom, pos; validate_line_start_cache (w); w->line_cache_validation_override++; /* Let functions pass in negative values, but we still treat -1 specially. */ /* #### bogosity alert */ if (min_past < 0 && min_past != -1) min_past = -min_past; if (!Dynarr_length (cache) || line_start_cache_start (w) > point || line_start_cache_end (w) < point) { int loop; int win_char_height = window_char_height (w, 1); /* Occasionally we get here with a 0 height window. find_next_newline_no_quit will abort if we pass it a count of 0 so handle that case. */ if (!win_char_height) win_char_height = 1; if (!Dynarr_length (cache)) { Charbpos from = find_next_newline_no_quit (b, point, -1); Charbpos to = find_next_newline_no_quit (b, from, win_char_height); update_line_start_cache (w, from, to, point, 0); if (!Dynarr_length (cache)) { w->line_cache_validation_override--; return -1; } } assert (Dynarr_length (cache)); loop = 0; while (line_start_cache_start (w) > point && (loop < cache_adjustment || min_past == -1)) { Charbpos from, to; from = line_start_cache_start (w); if (from <= BUF_BEGV (b)) break; from = find_next_newline_no_quit (b, from, -win_char_height); to = line_start_cache_end (w); update_line_start_cache (w, from, to, point, 0); loop++; } if (line_start_cache_start (w) > point) { Charbpos from, to; from = find_next_newline_no_quit (b, point, -1); if (from >= BUF_ZV (b)) { to = find_next_newline_no_quit (b, from, -win_char_height); from = to; to = BUF_ZV (b); } else to = find_next_newline_no_quit (b, from, win_char_height); update_line_start_cache (w, from, to, point, 0); } loop = 0; while (line_start_cache_end (w) < point && (loop < cache_adjustment || min_past == -1)) { Charbpos from, to; to = line_start_cache_end (w); if (to >= BUF_ZV (b)) break; from = line_start_cache_end (w); to = find_next_newline_no_quit (b, from, win_char_height); update_line_start_cache (w, from, to, point, 0); loop++; } if (line_start_cache_end (w) < point) { Charbpos from, to; from = find_next_newline_no_quit (b, point, -1); if (from >= BUF_ZV (b)) { to = find_next_newline_no_quit (b, from, -win_char_height); from = to; to = BUF_ZV (b); } else to = find_next_newline_no_quit (b, from, win_char_height); update_line_start_cache (w, from, to, point, 0); } } assert (Dynarr_length (cache)); if (min_past == -1) min_past = 0; /* This could happen if the buffer is narrowed. */ if (line_start_cache_start (w) > point || line_start_cache_end (w) < point) { w->line_cache_validation_override--; return -1; } find_point_loop: top = Dynarr_length (cache) - 1; bottom = 0; while (1) { int new_pos; Charbpos start, end; pos = (bottom + top + 1) >> 1; start = Dynarr_atp (cache, pos)->start; end = Dynarr_atp (cache, pos)->end; if (point >= start && point <= end) { if (pos < min_past && line_start_cache_start (w) > BUF_BEGV (b)) { Charbpos from = find_next_newline_no_quit (b, line_start_cache_start (w), -min_past - 1); Charbpos to = line_start_cache_end (w); update_line_start_cache (w, from, to, point, 0); goto find_point_loop; } else if ((Dynarr_length (cache) - pos - 1) < min_past && line_start_cache_end (w) < BUF_ZV (b)) { Charbpos from = line_start_cache_end (w); Charbpos to = find_next_newline_no_quit (b, from, (min_past ? min_past : 1)); update_line_start_cache (w, from, to, point, 0); goto find_point_loop; } else { w->line_cache_validation_override--; return pos; } } else if (point > end) bottom = pos + 1; else if (point < start) top = pos - 1; else ABORT (); new_pos = (bottom + top + 1) >> 1; if (pos == new_pos) { w->line_cache_validation_override--; return -1; } } } /* Return a boolean indicating if POINT would be visible in window W if display of the window was to begin at STARTP. If PARTIALLY is zero, then if POINT has fewer visible pixels than the window clip, 0 is returned; otherwise, 1 is returned if POINT has any visible pixels. */ int point_would_be_visible (struct window *w, Charbpos startp, Charbpos point, int partially) { struct buffer *b = XBUFFER (w->buffer); int pixpos = -WINDOW_TEXT_TOP_CLIP (w); int bottom = WINDOW_TEXT_HEIGHT (w); int start_elt; /* If point is before the intended start it obviously can't be visible. */ if (point < startp) return 0; /* If point or start are not in the accessible buffer range, then fail. */ if (startp < BUF_BEGV (b) || startp > BUF_ZV (b) || point < BUF_BEGV (b) || point > BUF_ZV (b)) return 0; validate_line_start_cache (w); w->line_cache_validation_override++; start_elt = point_in_line_start_cache (w, startp, 0); if (start_elt == -1) { w->line_cache_validation_override--; return 0; } assert (line_start_cache_start (w) <= startp && line_start_cache_end (w) >= startp); while (1) { int height; /* Expand the cache if necessary. */ if (start_elt == Dynarr_length (w->line_start_cache)) { Charbpos old_startp = Dynarr_atp (w->line_start_cache, start_elt - 1)->start; start_elt = point_in_line_start_cache (w, old_startp, window_char_height (w, 0)); /* We've already actually processed old_startp, so increment immediately. */ start_elt++; /* If this happens we didn't add any extra elements. Bummer. */ if (start_elt == Dynarr_length (w->line_start_cache)) { w->line_cache_validation_override--; return 0; } } height = Dynarr_atp (w->line_start_cache, start_elt)->height; if (pixpos + height > bottom) { if (bottom - pixpos < (partially ? 0 : VERTICAL_CLIP (w, 0))) { w->line_cache_validation_override--; return 0; } } pixpos += height; if (point <= Dynarr_atp (w->line_start_cache, start_elt)->end) { w->line_cache_validation_override--; return 1; } start_elt++; } } /* For the given window W, if display starts at STARTP, what will be the buffer position at the beginning or end of the last line displayed. The end of the last line is also know as the window end position. WARNING: It is possible that redisplay failed to layout any lines for the windows. Under normal circumstances this is rare. However it seems that it does occur in the following situation: A mouse event has come in and we need to compute its location in a window. That code (in pixel_to_glyph_translation) already can handle 0 as an error return value. #### With a little work this could probably be reworked as just a call to start_with_line_at_pixpos. */ static Charbpos start_end_of_last_line (struct window *w, Charbpos startp, int end, int may_error) { struct buffer *b = XBUFFER (w->buffer); line_start_cache_dynarr *cache = w->line_start_cache; int pixpos = 0; int bottom = WINDOW_TEXT_HEIGHT (w); Charbpos cur_start; int start_elt; validate_line_start_cache (w); w->line_cache_validation_override++; if (startp < BUF_BEGV (b)) startp = BUF_BEGV (b); else if (startp > BUF_ZV (b)) startp = BUF_ZV (b); cur_start = startp; start_elt = point_in_line_start_cache (w, cur_start, 0); if (start_elt == -1) return may_error ? 0 : startp; while (1) { int height = Dynarr_atp (cache, start_elt)->height; cur_start = Dynarr_atp (cache, start_elt)->start; if (pixpos + height > bottom) { /* Adjust for any possible clip. */ if (bottom - pixpos < VERTICAL_CLIP (w, 0)) start_elt--; if (start_elt < 0) { w->line_cache_validation_override--; if (end) return BUF_ZV (b); else return BUF_BEGV (b); } else { w->line_cache_validation_override--; if (end) return Dynarr_atp (cache, start_elt)->end; else return Dynarr_atp (cache, start_elt)->start; } } pixpos += height; start_elt++; if (start_elt == Dynarr_length (cache)) { Charbpos from = line_start_cache_end (w); int win_char_height = window_char_height (w, 0); Charbpos to = find_next_newline_no_quit (b, from, (win_char_height ? win_char_height : 1)); /* We've hit the end of the bottom so that's what it is. */ if (from >= BUF_ZV (b)) { w->line_cache_validation_override--; return BUF_ZV (b); } update_line_start_cache (w, from, to, BUF_PT (b), 0); /* Updating the cache invalidates any current indexes. */ start_elt = point_in_line_start_cache (w, cur_start, -1) + 1; } } } /* For the given window W, if display starts at STARTP, what will be the buffer position at the beginning of the last line displayed. */ Charbpos start_of_last_line (struct window *w, Charbpos startp) { return start_end_of_last_line (w, startp, 0 , 0); } /* For the given window W, if display starts at STARTP, what will be the buffer position at the end of the last line displayed. This is also know as the window end position. */ Charbpos end_of_last_line (struct window *w, Charbpos startp) { return start_end_of_last_line (w, startp, 1, 0); } static Charbpos end_of_last_line_may_error (struct window *w, Charbpos startp) { return start_end_of_last_line (w, startp, 1, 1); } /* For window W, what does the starting position have to be so that the line containing POINT will cover pixel position PIXPOS. */ Charbpos start_with_line_at_pixpos (struct window *w, Charbpos point, int pixpos) { struct buffer *b = XBUFFER (w->buffer); int cur_elt; Charbpos cur_pos, prev_pos = point; int point_line_height; int pixheight = pixpos - WINDOW_TEXT_TOP (w); validate_line_start_cache (w); w->line_cache_validation_override++; cur_elt = point_in_line_start_cache (w, point, 0); /* #### See comment in update_line_start_cache about big minibuffers. */ if (cur_elt < 0) { w->line_cache_validation_override--; return point; } point_line_height = Dynarr_atp (w->line_start_cache, cur_elt)->height; while (1) { cur_pos = Dynarr_atp (w->line_start_cache, cur_elt)->start; pixheight -= Dynarr_atp (w->line_start_cache, cur_elt)->height; /* Do not take into account the value of vertical_clip here. That is the responsibility of the calling functions. */ if (pixheight < 0) { w->line_cache_validation_override--; if (-pixheight > point_line_height) /* We can't make the target line cover pixpos, so put it above pixpos. That way it will at least be visible. */ return prev_pos; else return cur_pos; } cur_elt--; while (cur_elt < 0) { Charbpos from, to; int win_char_height; if (cur_pos <= BUF_BEGV (b)) { w->line_cache_validation_override--; return BUF_BEGV (b); } win_char_height = window_char_height (w, 0); if (!win_char_height) win_char_height = 1; from = find_next_newline_no_quit (b, cur_pos, -win_char_height); to = line_start_cache_end (w); update_line_start_cache (w, from, to, point, 0); cur_elt = point_in_line_start_cache (w, cur_pos, 2) - 1; assert (cur_elt >= -1); /* This used to be cur_elt>=0 under the assumption that if point is in the top line and not at BUF_BEGV, then setting the window_start to a newline before the start of the first line will always cause scrolling. However in my (jv) opinion this is wrong. That new line can be hidden in various ways: invisible extents, an explicit window-start not at a newline character etc. The existence of those are indeed known to create crashes on that assert. So we have no option but to continue the search if we found point at the top of the line_start_cache again. */ cur_pos = Dynarr_begin (w->line_start_cache)->start; } prev_pos = cur_pos; } } /* For window W, what does the starting position have to be so that the line containing point is on display line LINE. If LINE is positive it is considered to be the number of lines from the top of the window (0 is the top line). If it is negative the number is considered to be the number of lines from the bottom (-1 is the bottom line). */ Charbpos start_with_point_on_display_line (struct window *w, Charbpos point, int line) { validate_line_start_cache (w); w->line_cache_validation_override++; if (line >= 0) { int cur_elt = point_in_line_start_cache (w, point, line); if (cur_elt - line < 0) cur_elt = 0; /* Hit the top */ else cur_elt -= line; w->line_cache_validation_override--; return Dynarr_atp (w->line_start_cache, cur_elt)->start; } else { /* The calculated value of pixpos is correct for the bottom line or what we want when line is -1. Therefore we subtract one because we have already handled one line. */ int new_line = -line - 1; int cur_elt = point_in_line_start_cache (w, point, new_line); int pixpos = WINDOW_TEXT_BOTTOM (w); Charbpos retval, search_point; /* If scroll_on_clipped_lines is false, the last "visible" line of the window covers the pixel at WINDOW_TEXT_BOTTOM (w) - 1. If s_o_c_l is true, then we don't want to count a clipped line, so back up from the bottom by the height of the line containing point. */ if (scroll_on_clipped_lines) pixpos -= Dynarr_atp (w->line_start_cache, cur_elt)->height; else pixpos -= 1; if (cur_elt + new_line >= Dynarr_length (w->line_start_cache)) { /* Hit the bottom of the buffer. */ int adjustment = (cur_elt + new_line) - Dynarr_length (w->line_start_cache) + 1; Lisp_Object window; int defheight; window = wrap_window (w); default_face_width_and_height (window, 0, &defheight); cur_elt = Dynarr_length (w->line_start_cache) - 1; pixpos -= (adjustment * defheight); if (pixpos < WINDOW_TEXT_TOP (w)) pixpos = WINDOW_TEXT_TOP (w); } else cur_elt = cur_elt + new_line; search_point = Dynarr_atp (w->line_start_cache, cur_elt)->start; retval = start_with_line_at_pixpos (w, search_point, pixpos); w->line_cache_validation_override--; return retval; } } /* This is used to speed up vertical scrolling by caching the known buffer starting positions for display lines. This allows the scrolling routines to avoid costly calls to regenerate_window. If NO_REGEN is true then it will only add the values in the DESIRED display structs which are in the given range. Note also that the FROM/TO values are minimums. It is possible that this function will actually add information outside of the lines containing those positions. This can't hurt but it could possibly help. #### We currently force the cache to have only 1 contiguous region. It might help to make the cache a dynarr of caches so that we can cover more areas. This might, however, turn out to be a lot of overhead for too little gain. */ static void update_line_start_cache (struct window *w, Charbpos from, Charbpos to, Charbpos point, int no_regen) { struct buffer *b = XBUFFER (w->buffer); line_start_cache_dynarr *cache = w->line_start_cache; Charbpos low_bound, high_bound; validate_line_start_cache (w); w->line_cache_validation_override++; if (from < BUF_BEGV (b)) from = BUF_BEGV (b); if (to > BUF_ZV (b)) to = BUF_ZV (b); if (from > to) { w->line_cache_validation_override--; return; } if (Dynarr_length (cache)) { low_bound = line_start_cache_start (w); high_bound = line_start_cache_end (w); /* Check to see if the desired range is already in the cache. */ if (from >= low_bound && to <= high_bound) { w->line_cache_validation_override--; return; } /* Check to make sure that the desired range is adjacent to the current cache. If not, invalidate the cache. */ if (to < low_bound || from > high_bound) { Dynarr_reset (cache); low_bound = high_bound = -1; } } else { low_bound = high_bound = -1; } w->line_cache_last_updated = make_fixnum (BUF_MODIFF (b)); /* This could be integrated into the next two sections, but it is easier to follow what's going on by having it separate. */ if (no_regen) { Charbpos start, end; update_internal_cache_list (w, DESIRED_DISP); if (!Dynarr_length (internal_cache)) { w->line_cache_validation_override--; return; } start = Dynarr_begin (internal_cache)->start; end = Dynarr_lastp (internal_cache)->end; /* We aren't allowed to generate additional information to fill in gaps, so if the DESIRED structs don't overlap the cache, reset the cache. */ if (Dynarr_length (cache)) { if (end < low_bound || start > high_bound) Dynarr_reset (cache); /* #### What should really happen if what we are doing is extending a line (the last line)? */ if (Dynarr_length (cache) == 1 && Dynarr_length (internal_cache) == 1) Dynarr_reset (cache); } if (!Dynarr_length (cache)) { Dynarr_add_many (cache, Dynarr_begin (internal_cache), Dynarr_length (internal_cache)); w->line_cache_validation_override--; return; } /* An extra check just in case the calling function didn't pass in the bounds of the DESIRED structs in the first place. */ if (start >= low_bound && end <= high_bound) { w->line_cache_validation_override--; return; } /* At this point we know that the internal cache partially overlaps the main cache. */ if (start < low_bound) { int ic_elt = Dynarr_length (internal_cache) - 1; while (ic_elt >= 0) { if (Dynarr_atp (internal_cache, ic_elt)->start < low_bound) break; else ic_elt--; } if (!(ic_elt >= 0)) { Dynarr_reset (cache); Dynarr_add_many (cache, Dynarr_begin (internal_cache), Dynarr_length (internal_cache)); w->line_cache_validation_override--; return; } Dynarr_prepend_many (cache, Dynarr_begin (internal_cache), ic_elt + 1); } if (end > high_bound) { int ic_elt = 0; while (ic_elt < Dynarr_length (internal_cache)) { if (Dynarr_atp (internal_cache, ic_elt)->start > high_bound) break; else ic_elt++; } if (!(ic_elt < Dynarr_length (internal_cache))) { Dynarr_reset (cache); Dynarr_add_many (cache, Dynarr_begin (internal_cache), Dynarr_length (internal_cache)); w->line_cache_validation_override--; return; } Dynarr_add_many (cache, Dynarr_atp (internal_cache, ic_elt), Dynarr_length (internal_cache) - ic_elt); } w->line_cache_validation_override--; return; } if (!Dynarr_length (cache) || from < low_bound) { Charbpos startp = find_next_newline_no_quit (b, from, -1); int marker = 0; int old_lb = low_bound; while (startp < old_lb || low_bound == -1) { int ic_elt; Charbpos new_startp; regenerate_window (w, startp, point, CMOTION_DISP); update_internal_cache_list (w, CMOTION_DISP); /* If this assert is triggered then regenerate_window failed to layout a single line. This is not possible since we force at least a single line to be layout for CMOTION_DISP */ assert (Dynarr_length (internal_cache)); assert (startp == Dynarr_begin (internal_cache)->start); ic_elt = Dynarr_length (internal_cache) - 1; if (low_bound != -1) { while (ic_elt >= 0) { if (Dynarr_atp (internal_cache, ic_elt)->start < old_lb) break; else ic_elt--; } } assert (ic_elt >= 0); new_startp = Dynarr_atp (internal_cache, ic_elt)->end + 1; /* * Handle invisible text properly: * If the last line we're inserting has the same end as the * line before which it will be added, merge the two lines. */ if (Dynarr_length (cache) && Dynarr_atp (internal_cache, ic_elt)->end == Dynarr_atp (cache, marker)->end) { Dynarr_atp (cache, marker)->start = Dynarr_atp (internal_cache, ic_elt)->start; Dynarr_atp (cache, marker)->height = Dynarr_atp (internal_cache, ic_elt)->height; ic_elt--; } if (ic_elt >= 0) /* we still have lines to add.. */ { Dynarr_insert_many (cache, Dynarr_begin (internal_cache), ic_elt + 1, marker); marker += (ic_elt + 1); } if (startp < low_bound || low_bound == -1) low_bound = startp; startp = new_startp; if (startp > BUF_ZV (b)) { w->line_cache_validation_override--; return; } } } assert (Dynarr_length (cache)); assert (from >= low_bound); /* Readjust the high_bound to account for any changes made while correcting the low_bound. */ high_bound = Dynarr_lastp (cache)->end; if (to > high_bound) { Charbpos startp = Dynarr_lastp (cache)->end + 1; do { regenerate_window (w, startp, point, CMOTION_DISP); update_internal_cache_list (w, CMOTION_DISP); /* See comment above about regenerate_window failing. */ assert (Dynarr_length (internal_cache)); Dynarr_add_many (cache, Dynarr_begin (internal_cache), Dynarr_length (internal_cache)); high_bound = Dynarr_lastp (cache)->end; startp = high_bound + 1; } while (to > high_bound); } w->line_cache_validation_override--; assert (to <= high_bound); } /* Given x and y coordinates in characters, relative to a window, return the pixel location corresponding to those coordinates. The pixel location returned is the center of the given character position. The pixel values are generated relative to the window, not the frame. The modeline is considered to be part of the window. */ void glyph_to_pixel_translation (struct window *w, int char_x, int char_y, int *pix_x, int *pix_y) { display_line_dynarr *dla = window_display_lines (w, CURRENT_DISP); int num_disp_lines, modeline; Lisp_Object window; int defheight, defwidth; window = wrap_window (w); default_face_width_and_height (window, &defwidth, &defheight); /* If we get a bogus value indicating somewhere above or to the left of the window, use the first window line or character position instead. */ if (char_y < 0) char_y = 0; if (char_x < 0) char_x = 0; num_disp_lines = Dynarr_length (dla); modeline = 0; if (num_disp_lines) { if (Dynarr_begin (dla)->modeline) { num_disp_lines--; modeline = 1; } } /* First check if the y position intersects the display lines. */ if (char_y < num_disp_lines) { struct display_line *dl = Dynarr_atp (dla, char_y + modeline); struct display_block *db = get_display_block_from_line (dl, TEXT); *pix_y = (dl->ypos - dl->ascent + ((dl->ascent + dl->descent - dl->clip) >> 1)); if (char_x < Dynarr_length (db->runes)) { struct rune *rb = Dynarr_atp (db->runes, char_x); *pix_x = rb->xpos + (rb->width >> 1); } else { int last_rune = Dynarr_length (db->runes) - 1; struct rune *rb = Dynarr_atp (db->runes, last_rune); char_x -= last_rune; *pix_x = rb->xpos + rb->width; *pix_x += ((char_x - 1) * defwidth); *pix_x += (defwidth >> 1); } } else { /* It didn't intersect, so extrapolate. #### For now, we include the modeline in this since we don't have true character positions in it. */ if (!Dynarr_length (w->face_cachels)) reset_face_cachels (w); char_y -= num_disp_lines; if (Dynarr_length (dla)) { struct display_line *dl = Dynarr_lastp (dla); *pix_y = dl->ypos + dl->descent - dl->clip; } else *pix_y = WINDOW_TEXT_TOP (w); *pix_y += (char_y * defheight); *pix_y += (defheight >> 1); *pix_x = WINDOW_TEXT_LEFT (w); /* Don't adjust by one because this is still the unadjusted value. */ *pix_x += (char_x * defwidth); *pix_x += (defwidth >> 1); } if (*pix_x > w->pixel_left + w->pixel_width) *pix_x = w->pixel_left + w->pixel_width; if (*pix_y > w->pixel_top + w->pixel_height) *pix_y = w->pixel_top + w->pixel_height; *pix_x -= w->pixel_left; *pix_y -= w->pixel_top; } /* Given a display line and a position, determine if there is a glyph there and return information about it if there is. */ static void get_position_object (struct display_line *dl, Lisp_Object *obj1, Lisp_Object *obj2, int x_coord, int *low_x_coord, int *high_x_coord) { struct display_block *db; int elt; int block = get_next_display_block (dl->bounds, dl->display_blocks, x_coord, 0); /* We use get_next_display_block to get the actual display block that would be displayed at x_coord. */ if (block == NO_BLOCK) return; else db = Dynarr_atp (dl->display_blocks, block); for (elt = 0; elt < Dynarr_length (db->runes); elt++) { struct rune *rb = Dynarr_atp (db->runes, elt); if (rb->xpos <= x_coord && x_coord < (rb->xpos + rb->width)) { if (rb->type == RUNE_DGLYPH) { *obj1 = rb->object.dglyph.glyph; *obj2 = rb->object.dglyph.extent; } else { *obj1 = Qnil; *obj2 = Qnil; } if (low_x_coord) *low_x_coord = rb->xpos; if (high_x_coord) *high_x_coord = rb->xpos + rb->width; return; } } } #define UPDATE_CACHE_RETURN \ do { \ d->pixel_to_glyph_cache.valid = 1; \ d->pixel_to_glyph_cache.low_x_coord = low_x_coord; \ d->pixel_to_glyph_cache.high_x_coord = high_x_coord; \ d->pixel_to_glyph_cache.low_y_coord = low_y_coord; \ d->pixel_to_glyph_cache.high_y_coord = high_y_coord; \ d->pixel_to_glyph_cache.frame = f; \ d->pixel_to_glyph_cache.col = *col; \ d->pixel_to_glyph_cache.row = *row; \ d->pixel_to_glyph_cache.obj_x = *obj_x; \ d->pixel_to_glyph_cache.obj_y = *obj_y; \ d->pixel_to_glyph_cache.w = *w; \ d->pixel_to_glyph_cache.charpos = *charpos; \ d->pixel_to_glyph_cache.closest = *closest; \ d->pixel_to_glyph_cache.modeline_closest = *modeline_closest; \ d->pixel_to_glyph_cache.obj1 = *obj1; \ d->pixel_to_glyph_cache.obj2 = *obj2; \ d->pixel_to_glyph_cache.retval = position; \ RETURN_SANS_WARNINGS position; \ } while (0) /* Given x and y coordinates in pixels relative to a frame, return information about what is located under those coordinates. The return value will be one of: OVER_TOOLBAR: over one of the 4 frame toolbars OVER_MODELINE: over a modeline OVER_BORDER: over an internal border OVER_V_DIVIDER: over a vertical divider between windows (used as a grab bar for resizing) OVER_NOTHING: over the text area, but not over text OVER_OUTSIDE: outside of the frame border OVER_TEXT: over text in the text area #### GEOM! We need to also have an OVER_GUTTER, OVER_SCROLLBAR and OVER_DEAD_BOX. OBJ1 is one of -- a toolbar button -- a glyph -- nil if the coordinates are not over a glyph or a toolbar button. OBJ2 is one of -- an extent, if the coordinates are over a glyph in the text area -- nil otherwise. If the coordinates are over a glyph, OBJ_X and OBJ_Y give the equivalent coordinates relative to the upper-left corner of the glyph. If the coordinates are over a character, OBJ_X and OBJ_Y give the equivalent coordinates relative to the upper-left corner of the character. Otherwise, OBJ_X and OBJ_Y are undefined. */ int pixel_to_glyph_translation (struct frame *f, int x_coord, int y_coord, int *col, int *row, int *obj_x, int *obj_y, struct window **w, Charbpos *charpos, Charbpos *closest, Charcount *modeline_closest, Lisp_Object *obj1, Lisp_Object *obj2) { struct device *d; struct pixel_to_glyph_translation_cache *cache; Lisp_Object window; int frm_left, frm_right, frm_top, frm_bottom; int low_x_coord, high_x_coord, low_y_coord, high_y_coord; int position = OVER_NOTHING; int device_check_failed = 0; display_line_dynarr *dla; /* This is a safety valve in case this got called with a frame in the middle of being deleted. */ if (!DEVICEP (f->device) || !DEVICE_LIVE_P (XDEVICE (f->device))) { device_check_failed = 1; d = NULL, cache = NULL; /* Warning suppression */ } else { d = XDEVICE (f->device); cache = &d->pixel_to_glyph_cache; } if (!device_check_failed && cache->valid && cache->frame == f && cache->low_x_coord <= x_coord && cache->high_x_coord > x_coord && cache->low_y_coord <= y_coord && cache->high_y_coord > y_coord) { *col = cache->col; *row = cache->row; *obj_x = cache->obj_x; *obj_y = cache->obj_y; *w = cache->w; *charpos = cache->charpos; *closest = cache->closest; *modeline_closest = cache->modeline_closest; *obj1 = cache->obj1; *obj2 = cache->obj2; return cache->retval; } else { *col = 0; *row = 0; *obj_x = 0; *obj_y = 0; *w = 0; *charpos = 0; *closest = 0; *modeline_closest = -1; *obj1 = Qnil; *obj2 = Qnil; low_x_coord = x_coord; high_x_coord = x_coord + 1; low_y_coord = y_coord; high_y_coord = y_coord + 1; } if (device_check_failed) return OVER_NOTHING; /* #### GEOM! The gutter is just inside of this. We should also have an OVER_GUTTER return value to indicate that we're over a gutter. See above. */ frm_left = FRAME_LEFT_INTERNAL_BORDER_END (f); frm_right = FRAME_RIGHT_INTERNAL_BORDER_START (f); frm_top = FRAME_TOP_INTERNAL_BORDER_END (f); frm_bottom = FRAME_BOTTOM_INTERNAL_BORDER_START (f); /* Check if the mouse is outside of the text area actually used by redisplay. */ if (y_coord < frm_top) { if (y_coord >= FRAME_TOP_INTERNAL_BORDER_START (f)) { low_y_coord = FRAME_TOP_INTERNAL_BORDER_START (f); high_y_coord = frm_top; position = OVER_BORDER; } else if (y_coord >= 0) { low_y_coord = 0; high_y_coord = FRAME_TOP_INTERNAL_BORDER_START (f); position = OVER_TOOLBAR; } else { low_y_coord = y_coord; high_y_coord = 0; position = OVER_OUTSIDE; } } else if (y_coord >= frm_bottom) { if (y_coord < FRAME_BOTTOM_INTERNAL_BORDER_END (f)) { low_y_coord = frm_bottom; high_y_coord = FRAME_BOTTOM_INTERNAL_BORDER_END (f); position = OVER_BORDER; } else if (y_coord < FRAME_PIXHEIGHT (f)) { low_y_coord = FRAME_BOTTOM_INTERNAL_BORDER_END (f); high_y_coord = FRAME_PIXHEIGHT (f); position = OVER_TOOLBAR; } else { low_y_coord = FRAME_PIXHEIGHT (f); high_y_coord = y_coord; position = OVER_OUTSIDE; } } if (position != OVER_TOOLBAR && position != OVER_BORDER) { if (x_coord < frm_left) { if (x_coord >= FRAME_LEFT_INTERNAL_BORDER_START (f)) { low_x_coord = FRAME_LEFT_INTERNAL_BORDER_START (f); high_x_coord = frm_left; position = OVER_BORDER; } else if (x_coord >= 0) { low_x_coord = 0; high_x_coord = FRAME_LEFT_INTERNAL_BORDER_START (f); position = OVER_TOOLBAR; } else { low_x_coord = x_coord; high_x_coord = 0; position = OVER_OUTSIDE; } } else if (x_coord >= frm_right) { if (x_coord < FRAME_RIGHT_INTERNAL_BORDER_END (f)) { low_x_coord = frm_right; high_x_coord = FRAME_RIGHT_INTERNAL_BORDER_END (f); position = OVER_BORDER; } else if (x_coord < FRAME_PIXWIDTH (f)) { low_x_coord = FRAME_RIGHT_INTERNAL_BORDER_END (f); high_x_coord = FRAME_PIXWIDTH (f); position = OVER_TOOLBAR; } else { low_x_coord = FRAME_PIXWIDTH (f); high_x_coord = x_coord; position = OVER_OUTSIDE; } } } #ifdef HAVE_TOOLBARS if (position == OVER_TOOLBAR) { *obj1 = toolbar_button_at_pixpos (f, x_coord, y_coord); *obj2 = Qnil; *w = 0; UPDATE_CACHE_RETURN; } #endif /* HAVE_TOOLBARS */ /* We still have to return the window the pointer is next to and its relative y position even if it is outside the x boundary. */ if (x_coord < frm_left) x_coord = frm_left; else if (x_coord > frm_right) x_coord = frm_right; /* Same in reverse. */ if (y_coord < frm_top) y_coord = frm_top; else if (y_coord > frm_bottom) y_coord = frm_bottom; /* Find what window the given coordinates are actually in. */ window = f->root_window; *w = find_window_by_pixel_pos (x_coord, y_coord, window); /* If we didn't find a window, we're done. */ if (!*w) { UPDATE_CACHE_RETURN; } else if (position != OVER_NOTHING) { *closest = 0; *modeline_closest = -1; if (high_y_coord <= frm_top || high_y_coord >= frm_bottom) { *w = 0; UPDATE_CACHE_RETURN; } } /* Check if the window is a minibuffer but isn't active. */ if (MINI_WINDOW_P (*w) && !minibuf_level) { /* Must reset the window value since some callers will ignore the return value if it is set. */ *w = 0; UPDATE_CACHE_RETURN; } /* See if the point is over window vertical divider */ if (window_needs_vertical_divider (*w)) { int div_x_high = WINDOW_RIGHT (*w); int div_x_low = div_x_high - window_divider_width (*w); int div_y_high = WINDOW_BOTTOM (*w); int div_y_low = WINDOW_TOP (*w); if (div_x_low < x_coord && x_coord <= div_x_high && div_y_low < y_coord && y_coord <= div_y_high) { low_x_coord = div_x_low; high_x_coord = div_x_high; low_y_coord = div_y_low; high_y_coord = div_y_high; position = OVER_V_DIVIDER; UPDATE_CACHE_RETURN; } } dla = window_display_lines (*w, CURRENT_DISP); for (*row = 0; *row < Dynarr_length (dla); (*row)++) { int really_over_nothing = 0; struct display_line *dl = Dynarr_atp (dla, *row); if ((int) (dl->ypos - dl->ascent) <= y_coord && y_coord <= (int) (dl->ypos + dl->descent)) { int check_margin_glyphs = 0; struct display_block *db = get_display_block_from_line (dl, TEXT); struct rune *rb = 0; if (x_coord < dl->bounds.left_white || x_coord >= dl->bounds.right_white) check_margin_glyphs = 1; low_y_coord = dl->ypos - dl->ascent; high_y_coord = dl->ypos + dl->descent + 1; if (position == OVER_BORDER || position == OVER_OUTSIDE || check_margin_glyphs) { int x_check, left_bound; if (check_margin_glyphs) { x_check = x_coord; left_bound = dl->bounds.left_white; } else { x_check = high_x_coord; left_bound = frm_left; } if (Dynarr_length (db->runes)) { if (x_check <= left_bound) { if (dl->modeline) *modeline_closest = Dynarr_begin (db->runes)->charpos; else *closest = Dynarr_begin (db->runes)->charpos; } else { if (dl->modeline) *modeline_closest = Dynarr_lastp (db->runes)->charpos; else *closest = Dynarr_lastp (db->runes)->charpos; } if (dl->modeline) *modeline_closest += dl->offset; else *closest += dl->offset; } else { /* #### What should be here. */ if (dl->modeline) *modeline_closest = 0; else *closest = 0; } if (check_margin_glyphs) { if (x_coord < dl->bounds.left_in || x_coord >= dl->bounds.right_in) { /* If we are over the outside margins then we know the loop over the text block isn't going to accomplish anything. So we go ahead and set what information we can right here and return. */ (*row)--; *obj_y = y_coord - (dl->ypos - dl->ascent); get_position_object (dl, obj1, obj2, x_coord, &low_x_coord, &high_x_coord); UPDATE_CACHE_RETURN; } } else UPDATE_CACHE_RETURN; } for (*col = 0; *col <= Dynarr_length (db->runes); (*col)++) { if (*col == Dynarr_length (db->runes)) { /* We've run out of runes to look at. Treat the same as the case below where we failed to find a non-glyph character. */ if (dl->modeline) *modeline_closest = dl->end_charpos + dl->offset; else *closest = dl->end_charpos + dl->offset; if (check_margin_glyphs) get_position_object (dl, obj1, obj2, x_coord, &low_x_coord, &high_x_coord); UPDATE_CACHE_RETURN; } rb = Dynarr_atp (db->runes, *col); if (rb->xpos <= x_coord && x_coord < rb->xpos + rb->width) { *charpos = rb->charpos + dl->offset; low_x_coord = rb->xpos; high_x_coord = rb->xpos + rb->width; if (rb->type == RUNE_DGLYPH) { int elt = *col + 1; /* Find the first character after the glyph. */ while (elt < Dynarr_length (db->runes)) { if (Dynarr_atp (db->runes, elt)->type != RUNE_DGLYPH) { if (dl->modeline) *modeline_closest = (Dynarr_atp (db->runes, elt)->charpos + dl->offset); else *closest = (Dynarr_atp (db->runes, elt)->charpos + dl->offset); break; } elt++; } /* In this case we failed to find a non-glyph character so we return the last position displayed on the line. */ if (elt == Dynarr_length (db->runes)) { if (dl->modeline) *modeline_closest = dl->end_charpos + dl->offset; else *closest = dl->end_charpos + dl->offset; really_over_nothing = 1; } } else { if (dl->modeline) *modeline_closest = rb->charpos + dl->offset; else *closest = rb->charpos + dl->offset; } if (dl->modeline) { *row = window_displayed_height (*w); if (position == OVER_NOTHING) position = OVER_MODELINE; if (rb->type == RUNE_DGLYPH) { *obj1 = rb->object.dglyph.glyph; *obj2 = rb->object.dglyph.extent; } else if (rb->type == RUNE_CHAR) { *obj1 = Qnil; *obj2 = Qnil; } else { *obj1 = Qnil; *obj2 = Qnil; } UPDATE_CACHE_RETURN; } else if (rb->type == RUNE_CHAR && rb->object.chr.ch == '\n') { (*row)--; /* At this point we may have glyphs in the right inside margin. */ if (check_margin_glyphs) get_position_object (dl, obj1, obj2, x_coord, &low_x_coord, &high_x_coord); UPDATE_CACHE_RETURN; } else { (*row)--; if (rb->type == RUNE_DGLYPH) { *obj1 = rb->object.dglyph.glyph; *obj2 = rb->object.dglyph.extent; } else if (rb->type == RUNE_CHAR) { *obj1 = Qnil; *obj2 = Qnil; } else { *obj1 = Qnil; *obj2 = Qnil; } *obj_x = x_coord - rb->xpos; *obj_y = y_coord - (dl->ypos - dl->ascent); /* At this point we may have glyphs in the left inside margin. */ if (check_margin_glyphs) get_position_object (dl, obj1, obj2, x_coord, 0, 0); if (position == OVER_NOTHING && !really_over_nothing) position = OVER_TEXT; UPDATE_CACHE_RETURN; } } } } } *row = Dynarr_length (dla) - 1; if (FRAME_WIN_P (f)) { int bot_elt = Dynarr_length (dla) - 1; if (bot_elt >= 0) { struct display_line *dl = Dynarr_atp (dla, bot_elt); int adj_area = y_coord - (dl->ypos + dl->descent); Lisp_Object lwin; int defheight; lwin = wrap_window (*w); default_face_width_and_height (lwin, 0, &defheight); *row += (adj_area / defheight); } } /* #### This should be checked out some more to determine what should really be going on. */ if (!MARKERP ((*w)->start[CURRENT_DISP])) *closest = 0; else *closest = end_of_last_line_may_error (*w, marker_position ((*w)->start[CURRENT_DISP])); *col = 0; UPDATE_CACHE_RETURN; } #undef UPDATE_CACHE_RETURN /***************************************************************************/ /* */ /* Lisp functions */ /* */ /***************************************************************************/ DEFUN ("redisplay-echo-area", Fredisplay_echo_area, 0, 0, 0, /* Ensure that all minibuffers are correctly showing the echo area. */ ()) { Lisp_Object devcons, concons; if (in_display) return Qnil; DEVICE_LOOP_NO_BREAK (devcons, concons) { struct device *d = XDEVICE (XCAR (devcons)); Lisp_Object frmcons; if (DEVICE_STREAM_P (d)) continue; DEVICE_FRAME_LOOP (frmcons, d) { struct frame *f = XFRAME (XCAR (frmcons)); int depth; if (FRAME_REPAINT_P (f) && FRAME_HAS_MINIBUF_P (f)) { Lisp_Object window = FRAME_MINIBUF_WINDOW (f); MAYBE_DEVMETH (d, frame_output_begin, (f)); /* * If the frame size has changed, there may be random * chud on the screen left from previous messages * because redisplay_frame hasn't been called yet. * Clear the screen to get rid of the potential mess. */ if (f->echo_area_garbaged) { MAYBE_DEVMETH (d, clear_frame, (f)); f->echo_area_garbaged = 0; } depth = enter_redisplay_critical_section (); redisplay_window (window, 0); exit_redisplay_critical_section (depth); MAYBE_DEVMETH (d, frame_output_end, (f)); } } } return Qnil; } DEFUN ("redraw-frame", Fredraw_frame, 0, 2, 0, /* Clear frame FRAME and output again what is supposed to appear on it. FRAME defaults to the selected frame if omitted. Normally, redisplay is preempted as normal if input arrives. However, if optional second arg NO-PREEMPT is non-nil, redisplay will not stop for input and is guaranteed to proceed to completion. */ (frame, no_preempt)) { struct frame *f = decode_frame (frame); int count = specpdl_depth (); if (!NILP (no_preempt)) internal_bind_int (&disable_preemption, 1 + disable_preemption); f->clear = 1; redisplay_frame (f, 1); /* See the comment in Fredisplay_frame. */ RESET_CHANGED_SET_FLAGS; return unbind_to (count); } DEFUN ("redisplay-frame", Fredisplay_frame, 0, 2, 0, /* Ensure that FRAME's contents are correctly displayed. This differs from `redraw-frame' in that it only redraws what needs to be updated, as opposed to unconditionally clearing and redrawing the frame. FRAME defaults to the selected frame if omitted. Normally, redisplay is preempted as normal if input arrives. However, if optional second arg NO-PREEMPT is non-nil, redisplay will not stop for input and is guaranteed to proceed to completion. */ (frame, no_preempt)) { struct frame *f = decode_frame (frame); int count = specpdl_depth (); if (!NILP (no_preempt)) internal_bind_int (&disable_preemption, 1 + disable_preemption); redisplay_frame (f, 1); /* If we don't reset the global redisplay flags here, subsequent changes to the display will not get registered by redisplay because it thinks it already has registered changes. If you really knew what you were doing you could confuse redisplay by calling Fredisplay_frame while updating another frame. We assume that if you know what you are doing you will not be that stupid. */ RESET_CHANGED_SET_FLAGS; return unbind_to (count); } DEFUN ("redraw-device", Fredraw_device, 0, 2, 0, /* Clear device DEVICE and output again what is supposed to appear on it. DEVICE defaults to the selected device if omitted. Normally, redisplay is preempted as normal if input arrives. However, if optional second arg NO-PREEMPT is non-nil, redisplay will not stop for input and is guaranteed to proceed to completion. */ (device, no_preempt)) { struct device *d = decode_device (device); Lisp_Object frmcons; int count = specpdl_depth (); if (!NILP (no_preempt)) internal_bind_int (&disable_preemption, 1 + disable_preemption); DEVICE_FRAME_LOOP (frmcons, d) { XFRAME (XCAR (frmcons))->clear = 1; } redisplay_device (d, 0); /* See the comment in Fredisplay_frame. */ RESET_CHANGED_SET_FLAGS; return unbind_to (count); } DEFUN ("redisplay-device", Fredisplay_device, 0, 2, 0, /* Ensure that DEVICE's contents are correctly displayed. This differs from `redraw-device' in that it only redraws what needs to be updated, as opposed to unconditionally clearing and redrawing the device. DEVICE defaults to the selected device if omitted. Normally, redisplay is preempted as normal if input arrives. However, if optional second arg NO-PREEMPT is non-nil, redisplay will not stop for input and is guaranteed to proceed to completion. Note: If you simply want everything redisplayed, the current idiom is `(sit-for 0)'. */ (device, no_preempt)) { struct device *d = decode_device (device); int count = specpdl_depth (); if (!NILP (no_preempt)) internal_bind_int (&disable_preemption, 1 + disable_preemption); redisplay_device (d, 0); /* See the comment in Fredisplay_frame. */ RESET_CHANGED_SET_FLAGS; return unbind_to (count); } /* Big lie. Big lie. This will force all modelines to be updated regardless if the all flag is set or not. It remains in existence solely for backwards compatibility. */ DEFUN ("redraw-modeline", Fredraw_modeline, 0, 1, 0, /* Force the modeline of the current buffer to be redisplayed. With optional non-nil ALL, force redisplay of all modelines. */ (UNUSED (all))) { MARK_MODELINE_CHANGED; return Qnil; } DEFUN ("force-cursor-redisplay", Fforce_cursor_redisplay, 0, 1, 0, /* Force an immediate update of the cursor on FRAME. FRAME defaults to the selected frame if omitted. */ (frame)) { struct frame *f = decode_frame (frame); if (!FRAME_STREAM_P (f)) redisplay_redraw_cursor (f, 1); return Qnil; } /***************************************************************************/ /* */ /* Change flags */ /* */ /***************************************************************************/ static void margin_width_changed_in_frame (Lisp_Object UNUSED (specifier), struct frame *UNUSED (f), Lisp_Object UNUSED (oldval)) { /* Nothing to be done? */ } int redisplay_variable_changed (Lisp_Object UNUSED (sym), Lisp_Object *UNUSED (val), Lisp_Object UNUSED (in_object), int UNUSED (flags)) { /* #### clip_changed should really be renamed something like global_redisplay_change. */ MARK_CLIP_CHANGED; return 0; } /* This is called if the built-in glyphs have their properties changed. */ void redisplay_glyph_changed (Lisp_Object UNUSED (glyph), Lisp_Object UNUSED (property), Lisp_Object locale) { if (WINDOWP (locale)) { MARK_FRAME_GLYPHS_CHANGED (XFRAME (WINDOW_FRAME (XWINDOW (locale)))); } else if (FRAMEP (locale)) { MARK_FRAME_GLYPHS_CHANGED (XFRAME (locale)); } else if (DEVICEP (locale)) { Lisp_Object frmcons; DEVICE_FRAME_LOOP (frmcons, XDEVICE (locale)) MARK_FRAME_GLYPHS_CHANGED (XFRAME (XCAR (frmcons))); } else if (CONSOLEP (locale)) { Lisp_Object frmcons, devcons; CONSOLE_FRAME_LOOP_NO_BREAK (frmcons, devcons, XCONSOLE (locale)) MARK_FRAME_GLYPHS_CHANGED (XFRAME (XCAR (frmcons))); } else /* global or buffer */ { Lisp_Object frmcons, devcons, concons; FRAME_LOOP_NO_BREAK (frmcons, devcons, concons) MARK_FRAME_GLYPHS_CHANGED (XFRAME (XCAR (frmcons))); } } static void text_cursor_visible_p_changed (Lisp_Object UNUSED (specifier), struct window *w, Lisp_Object UNUSED (oldval)) { if (XFRAME (w->frame)->init_finished) Fforce_cursor_redisplay (w->frame); } void mark_buffers_changed (void) { MARK_TYPE_CHANGED (buffers); } void mark_clip_changed (void) { MARK_TYPE_CHANGED (clip); } void mark_extents_changed (void) { MARK_TYPE_CHANGED (extents); } void mark_icon_changed (void) { MARK_TYPE_CHANGED (icon); } void mark_menubar_changed (void) { MARK_TYPE_CHANGED (menubar); } void mark_modeline_changed (void) { MARK_TYPE_CHANGED (modeline); } void mark_point_changed (void) { MARK_TYPE_CHANGED (point); } void mark_toolbar_changed (void) { MARK_TYPE_CHANGED (toolbar); } void mark_gutter_changed (void) { MARK_TYPE_CHANGED (gutter); } void mark_glyphs_changed (void) { MARK_TYPE_CHANGED (glyphs); } void mark_subwindows_changed (void) { MARK_TYPE_CHANGED (subwindows); } void mark_subwindows_state_changed (void) { MARK_TYPE_CHANGED (subwindows_state); } #ifdef MEMORY_USAGE_STATS /***************************************************************************/ /* */ /* memory usage computation */ /* */ /***************************************************************************/ static int compute_rune_dynarr_usage (rune_dynarr *dyn, struct usage_stats *ustats) { return dyn ? Dynarr_memory_usage (dyn, ustats) : 0; } static int compute_display_block_dynarr_usage (display_block_dynarr *dyn, struct usage_stats *ustats) { int total, i; if (!dyn) return 0; total = Dynarr_memory_usage (dyn, ustats); for (i = 0; i < Dynarr_largest (dyn); i++) total += compute_rune_dynarr_usage (Dynarr_at (dyn, i).runes, ustats); return total; } static int compute_glyph_block_dynarr_usage (glyph_block_dynarr *dyn, struct usage_stats *ustats) { return dyn ? Dynarr_memory_usage (dyn, ustats) : 0; } int compute_display_line_dynarr_usage (display_line_dynarr *dyn, struct usage_stats *ustats) { int total, i; if (!dyn) return 0; total = Dynarr_memory_usage (dyn, ustats); for (i = 0; i < Dynarr_largest (dyn); i++) { struct display_line *dl = &Dynarr_at (dyn, i); total += compute_display_block_dynarr_usage (dl->display_blocks, ustats); total += compute_glyph_block_dynarr_usage (dl->left_glyphs, ustats); total += compute_glyph_block_dynarr_usage (dl->right_glyphs, ustats); } return total; } int compute_line_start_cache_dynarr_usage (line_start_cache_dynarr *dyn, struct usage_stats *ustats) { return dyn ? Dynarr_memory_usage (dyn, ustats) : 0; } #endif /* MEMORY_USAGE_STATS */ #ifdef ERROR_CHECK_DISPLAY static int sledgehammer_check_redisplay_structs_1 (struct window *w, void *UNUSED (closure)) { int i, j; display_line_dynarr *dl; dl = window_display_lines (w, CURRENT_DISP); for (i = 0; i < Dynarr_largest (dl); i++) for (j = i + 1; j < Dynarr_largest (dl); j++) assert (Dynarr_atp (dl, i)->display_blocks != Dynarr_atp (dl, j)->display_blocks); dl = window_display_lines (w, DESIRED_DISP); for (i = 0; i < Dynarr_largest (dl); i++) for (j = i + 1; j < Dynarr_largest (dl); j++) assert (Dynarr_atp (dl, i)->display_blocks != Dynarr_atp (dl, j)->display_blocks); return 0; } static void sledgehammer_check_redisplay_structs (void) { map_windows (0, sledgehammer_check_redisplay_structs_1, NULL); } #endif /* ERROR_CHECK_DISPLAY */ /***************************************************************************/ /* */ /* initialization */ /* */ /***************************************************************************/ void init_redisplay (void) { disable_preemption = 0; preemption_count = 0; #ifndef PDUMP if (!initialized) #endif { if (!cmotion_display_lines) cmotion_display_lines = Dynarr_new (display_line); if (!mode_spec_ibyte_string) mode_spec_ibyte_string = Dynarr_new (Ibyte); if (!formatted_string_extent_dynarr) formatted_string_extent_dynarr = Dynarr_new (EXTENT); if (!formatted_string_extent_start_dynarr) formatted_string_extent_start_dynarr = Dynarr_new (Bytecount); if (!formatted_string_extent_end_dynarr) formatted_string_extent_end_dynarr = Dynarr_new (Bytecount); if (!internal_cache) internal_cache = Dynarr_new (line_start_cache); } if (!initialized) return; if (noninteractive) { Vinitial_device_type = Qstream; return; } /* 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. If the DISPLAY environment variable is set, try to use X, and die with an error message if that doesn't work. */ #ifdef HAVE_X_WINDOWS if (!strcmp (display_use, "x")) { /* Some stuff checks this way early. */ Vwindow_system = Qx; Vinitial_device_type = Qx; return; } #endif /* HAVE_X_WINDOWS */ #ifdef HAVE_GTK if (!strcmp (display_use, "gtk")) { Vwindow_system = Qgtk; Vinitial_device_type = Qgtk; return; } #endif #ifdef HAVE_MS_WINDOWS if (!strcmp (display_use, "mswindows")) { /* Some stuff checks this way early. */ Vwindow_system = Qmswindows; Vinitial_device_type = Qmswindows; return; } #endif /* HAVE_MS_WINDOWS */ #ifdef HAVE_TTY /* If no window system has been specified, try to use the terminal. */ if (!isatty (0)) { stderr_out ("XEmacs: standard input is not a tty\n"); exit (1); } /* Look at the TERM variable */ if (!egetenv ("TERM")) { stderr_out ("Please set the environment variable TERM; see tset(1).\n"); exit (1); } Vinitial_device_type = Qtty; return; #else /* not HAVE_TTY */ /* No DISPLAY specified, and no TTY support. */ stderr_out ("XEmacs: Cannot open display.\n\ Please set the environmental variable DISPLAY to an appropriate value.\n"); exit (1); #endif /* Unreached. */ } void syms_of_redisplay (void) { DEFSYMBOL (Qcursor_in_echo_area); DEFSYMBOL (Qdisplay_warning_buffer); DEFSYMBOL (Qbar_cursor); DEFSYMBOL (Qtop_bottom); DEFSYMBOL (Qbuffer_list_changed_hook); DEFSUBR (Fredisplay_echo_area); DEFSUBR (Fredraw_frame); DEFSUBR (Fredisplay_frame); DEFSUBR (Fredraw_device); DEFSUBR (Fredisplay_device); DEFSUBR (Fredraw_modeline); DEFSUBR (Fforce_cursor_redisplay); } void vars_of_redisplay (void) { QSin_redisplay = build_defer_string ("(in redisplay)"); staticpro (&QSin_redisplay); Vpost_redisplay_actions = Qnil; staticpro (&Vpost_redisplay_actions); #if 0 staticpro (&last_arrow_position); staticpro (&last_arrow_string); last_arrow_position = Qnil; last_arrow_string = Qnil; #endif /* 0 */ /* #### Probably temporary */ DEFVAR_INT ("redisplay-cache-adjustment", &cache_adjustment /* \(Temporary) Setting this will impact the performance of the internal line start cache. */ ); cache_adjustment = 2; DEFVAR_INT ("maximum-preempts", &max_preempts /* Maximum number of times redisplay can be preempted by user input. */ ); max_preempts = INIT_MAX_PREEMPTS; DEFVAR_INT_MAGIC ("pixel-vertical-clip-threshold", &vertical_clip /* Minimum pixel height for clipped bottom display line. A clipped line shorter than this won't be displayed. */ , redisplay_variable_changed); vertical_clip = 5; DEFVAR_INT_MAGIC ("pixel-horizontal-clip-threshold", &horizontal_clip /* Minimum visible area for clipped glyphs at right boundary. Clipped glyphs shorter than this won't be displayed. Only pixmap glyph instances are currently allowed to be clipped. */ , redisplay_variable_changed); horizontal_clip = 5; DEFVAR_LISP ("global-mode-string", &Vglobal_mode_string /* String displayed by modeline-format's "%m" specification. */ ); Vglobal_mode_string = Qnil; DEFVAR_LISP_MAGIC ("overlay-arrow-position", &Voverlay_arrow_position /* Marker for where to display an arrow on top of the buffer text. This must be the beginning of a line in order to work. See also `overlay-arrow-string'. */ , redisplay_variable_changed); Voverlay_arrow_position = Qnil; DEFVAR_LISP_MAGIC ("overlay-arrow-string", &Voverlay_arrow_string /* String or glyph to display as an arrow. See also `overlay-arrow-position'. \(Note that despite the name of this variable, it can be set to a glyph as well as a string.) */ , redisplay_variable_changed); Voverlay_arrow_string = Qnil; DEFVAR_INT ("scroll-step", &scroll_step /* *The number of lines to try scrolling a window by when point moves out. If that fails to bring point back on frame, point is centered instead. If this is zero, point is always centered after it moves off screen. */ ); scroll_step = 0; DEFVAR_INT ("scroll-conservatively", &scroll_conservatively /* *Scroll up to this many lines, to bring point back on screen. */ ); scroll_conservatively = 0; DEFVAR_BOOL_MAGIC ("truncate-partial-width-windows", &truncate_partial_width_windows /* *Non-nil means truncate lines in all windows less than full frame wide. */ , redisplay_variable_changed); truncate_partial_width_windows = 1; DEFVAR_LISP ("visible-bell", &Vvisible_bell /* *Non-nil substitutes a visual signal for the audible bell. Default behavior is to flash the whole screen. On some platforms, special effects are available using the following values: `display' Flash the whole screen (ie, the default behavior). `top-bottom' Flash only the top and bottom lines of the selected frame. When effects are unavailable on a platform, the visual bell is the default, whole screen. (Currently only X supports any special effects.) */ ); Vvisible_bell = Qnil; DEFVAR_BOOL ("no-redraw-on-reenter", &no_redraw_on_reenter /* *Non-nil means no need to redraw entire frame after suspending. A non-nil value is useful if the terminal can automatically preserve Emacs's frame display when you reenter Emacs. It is up to you to set this variable if your terminal can do that. */ ); no_redraw_on_reenter = 0; DEFVAR_LISP ("window-system", &Vwindow_system /* A symbol naming the window-system under which Emacs is running, such as `x', or nil if emacs is running on an ordinary terminal. Do not use this variable, except for GNU Emacs compatibility, as it gives wrong values in a multi-device environment. Use `console-type' instead. */ ); Vwindow_system = Qnil; DEFVAR_CONST_LISP ("initial-device-type", &Vinitial_device_type /* The type of the first XEmacs device to be created. This is constant; it's used by the command line handling code to communicate to Lisp what type the initial device to be created should be. */ ); Vinitial_device_type = Qnil; DEFVAR_BOOL ("cursor-in-echo-area", &cursor_in_echo_area /* Non-nil means put cursor in minibuffer, at end of any message there. */ ); cursor_in_echo_area = 0; /* #### Shouldn't this be generalized as follows: if nil, use block cursor. if a number, use a bar cursor of that width. Otherwise, use a 1-pixel bar cursor. #### Or better yet, this variable should be trashed entirely (use a Lisp-magic variable to maintain compatibility) and a specifier `cursor-shape' added, which allows a block cursor, a bar cursor, a flashing block or bar cursor, maybe a caret cursor, etc. */ DEFVAR_LISP ("bar-cursor", &Vbar_cursor /* *Use vertical bar cursor if non-nil. If t width is 1 pixel, otherwise 2. */ ); Vbar_cursor = Qnil; DEFVAR_LISP ("buffer-list-changed-hook", &Vbuffer_list_changed_hook /* Function or functions to call when a frame's buffer list has changed. This is called during redisplay, before redisplaying each frame. Functions on this hook are called with one argument, the frame. */ ); Vbuffer_list_changed_hook = Qnil; DEFVAR_INT ("display-warning-tick", &display_warning_tick /* Bump this to tell the C code to call `display-warning-buffer' at next redisplay. You should not normally change this; the function `display-warning' automatically does this at appropriate times. */ ); display_warning_tick = 0; DEFVAR_BOOL ("inhibit-warning-display", &inhibit_warning_display /* Non-nil means inhibit display of warning messages. You should *bind* this, not set it. Any pending warning messages will be displayed when the binding no longer applies. */ ); /* reset to 0 by startup.el after the splash screen has displayed. This way, the warnings don't obliterate the splash screen. */ inhibit_warning_display = 1; DEFVAR_BOOL ("column-number-start-at-one", &column_number_start_at_one /* *Non-nil means column display number starts at 1. */ ); column_number_start_at_one = 0; } void specifier_vars_of_redisplay (void) { DEFVAR_SPECIFIER ("left-margin-width", &Vleft_margin_width /* *Width of left margin. This is a specifier; use `set-specifier' to change it. */ ); Vleft_margin_width = Fmake_specifier (Qnatnum); set_specifier_fallback (Vleft_margin_width, list1 (Fcons (Qnil, Qzero))); set_specifier_caching (Vleft_margin_width, offsetof (struct window, left_margin_width), some_window_value_changed, offsetof (struct frame, left_margin_width), margin_width_changed_in_frame, 0); DEFVAR_SPECIFIER ("right-margin-width", &Vright_margin_width /* *Width of right margin. This is a specifier; use `set-specifier' to change it. */ ); Vright_margin_width = Fmake_specifier (Qnatnum); set_specifier_fallback (Vright_margin_width, list1 (Fcons (Qnil, Qzero))); set_specifier_caching (Vright_margin_width, offsetof (struct window, right_margin_width), some_window_value_changed, offsetof (struct frame, right_margin_width), margin_width_changed_in_frame, 0); DEFVAR_SPECIFIER ("minimum-line-ascent", &Vminimum_line_ascent /* *Minimum ascent height of lines. This is a specifier; use `set-specifier' to change it. */ ); Vminimum_line_ascent = Fmake_specifier (Qnatnum); set_specifier_fallback (Vminimum_line_ascent, list1 (Fcons (Qnil, Qzero))); set_specifier_caching (Vminimum_line_ascent, offsetof (struct window, minimum_line_ascent), some_window_value_changed, 0, 0, 0); DEFVAR_SPECIFIER ("minimum-line-descent", &Vminimum_line_descent /* *Minimum descent height of lines. This is a specifier; use `set-specifier' to change it. */ ); Vminimum_line_descent = Fmake_specifier (Qnatnum); set_specifier_fallback (Vminimum_line_descent, list1 (Fcons (Qnil, Qzero))); set_specifier_caching (Vminimum_line_descent, offsetof (struct window, minimum_line_descent), some_window_value_changed, 0, 0, 0); DEFVAR_SPECIFIER ("use-left-overflow", &Vuse_left_overflow /* *Non-nil means use the left outside margin as extra whitespace when displaying `whitespace' or `inside-margin' glyphs. This is a specifier; use `set-specifier' to change it. */ ); Vuse_left_overflow = Fmake_specifier (Qboolean); set_specifier_fallback (Vuse_left_overflow, list1 (Fcons (Qnil, Qnil))); set_specifier_caching (Vuse_left_overflow, offsetof (struct window, use_left_overflow), some_window_value_changed, 0, 0, 0); DEFVAR_SPECIFIER ("use-right-overflow", &Vuse_right_overflow /* *Non-nil means use the right outside margin as extra whitespace when displaying `whitespace' or `inside-margin' glyphs. This is a specifier; use `set-specifier' to change it. */ ); Vuse_right_overflow = Fmake_specifier (Qboolean); set_specifier_fallback (Vuse_right_overflow, list1 (Fcons (Qnil, Qnil))); set_specifier_caching (Vuse_right_overflow, offsetof (struct window, use_right_overflow), some_window_value_changed, 0, 0, 0); DEFVAR_SPECIFIER ("text-cursor-visible-p", &Vtext_cursor_visible_p /* *Non-nil means the text cursor is visible (this is usually the case). This is a specifier; use `set-specifier' to change it. */ ); Vtext_cursor_visible_p = Fmake_specifier (Qboolean); set_specifier_fallback (Vtext_cursor_visible_p, list1 (Fcons (Qnil, Qt))); set_specifier_caching (Vtext_cursor_visible_p, offsetof (struct window, text_cursor_visible_p), text_cursor_visible_p_changed, 0, 0, 0); }