# HG changeset patch
# User Kim F. Storm <storm@cua.dk>
# Date 1083278272 0
# Node ID c5bad07bbb955bac5a7df49365575e0ae08d65d1
# Parent  fa2641411db1e2af7235f013f9d31b0a45122a03
(init_iterator): Handle line-spacing float value.
Initialize override_ascent member.
(append_space_for_newline): Reset override_ascent.
Remove use_default_face.
(calc_line_height_property): New function to calculate value of
line-height and line-spacing properties.  Look at overlays, too.
Set override_ascent, override_descent, override_boff members when
using another face than the current face.  Float values are now
relative to the frame default font, by default; accept a cons
of ratio and face name to specify value relative to a specific face.
(x_produce_glyphs): Use calc_line_height_property.
Use override_ascent etc. when set to handle different face heights.
A negative line-spacing property value is interpreted as a total
line height, rather than inter-line spacing.
(note_mouse_highlight): Allocate room for 40 overlays initially.

diff -r fa2641411db1 -r c5bad07bbb95 src/xdisp.c
--- a/src/xdisp.c	Thu Apr 29 22:36:34 2004 +0000
+++ b/src/xdisp.c	Thu Apr 29 22:37:52 2004 +0000
@@ -2073,6 +2073,9 @@
     {
       if (NATNUMP (current_buffer->extra_line_spacing))
 	it->extra_line_spacing = XFASTINT (current_buffer->extra_line_spacing);
+      else if (FLOATP (current_buffer->extra_line_spacing))
+	it->extra_line_spacing = (XFLOAT_DATA (current_buffer->extra_line_spacing)
+				  * FRAME_LINE_HEIGHT (it->f));
       else if (it->f->extra_line_spacing > 0)
 	it->extra_line_spacing = it->f->extra_line_spacing;
     }
@@ -2090,6 +2093,7 @@
   it->slice.x = it->slice.y = it->slice.width = it->slice.height = Qnil;
   it->space_width = Qnil;
   it->font_height = Qnil;
+  it->override_ascent = -1;
 
   /* Are control characters displayed as `^C'?  */
   it->ctl_arrow_p = !NILP (current_buffer->ctl_arrow);
@@ -14202,7 +14206,7 @@
 
 	  PRODUCE_GLYPHS (it);
 
-	  it->use_default_face = 0;
+	  it->override_ascent = -1;
 	  it->constrain_row_ascent_descent_p = 0;
 	  it->current_x = saved_x;
 	  it->object = saved_object;
@@ -18510,6 +18514,97 @@
   take_vertical_position_into_account (it);
 }
 
+/* Calculate line-height and line-spacing properties.
+   An integer value specifies explicit pixel value.
+   A float value specifies relative value to current face height.
+   A cons (float . face-name) specifies relative value to
+   height of specified face font.
+
+   Returns height in pixels, or nil.  */
+
+static Lisp_Object
+calc_line_height_property (it, prop, font, boff)
+     struct it *it;
+     Lisp_Object prop;
+     XFontStruct *font;
+     int boff;
+{
+  Lisp_Object val;
+  Lisp_Object face_name = Qnil;
+  int ascent, descent, height, override;
+
+  val = Fget_char_property (make_number (IT_CHARPOS (*it)),
+			    prop, it->object);
+
+  if (NILP (val))
+    return val;
+
+  if (INTEGERP (val))
+    return val;
+
+  if (CONSP (val))
+    {
+      face_name = XCDR (val);
+      val = XCAR (val);
+    }
+  else if (SYMBOLP (val))
+    {
+      face_name = val;
+      val = Qnil;
+    }
+
+  override = EQ (prop, Qline_height);
+
+  if (NILP (face_name))
+    {
+      font = FRAME_FONT (it->f);
+      boff = FRAME_BASELINE_OFFSET (it->f);
+    }
+  else if (EQ (face_name, Qt))
+    {
+      override = 0;
+    }
+  else
+    {
+      int face_id;
+      struct face *face;
+      struct font_info *font_info;
+
+      face_id = lookup_named_face (it->f, face_name, ' ');
+      if (face_id < 0)
+	return -1;
+
+      face = FACE_FROM_ID (it->f, face_id);
+      font = face->font;
+      if (font == NULL)
+	return -1;
+
+      font_info = FONT_INFO_FROM_ID (it->f, face->font_info_id);
+      boff = font_info->baseline_offset;
+      if (font_info->vertical_centering)
+	boff = VCENTER_BASELINE_OFFSET (font, it->f) - boff;
+    }
+
+  ascent = FONT_BASE (font) + boff;
+  descent = FONT_DESCENT (font) - boff;
+
+  if (override)
+    {
+      it->override_ascent = ascent;
+      it->override_descent = descent;
+      it->override_boff = boff;
+    }
+
+  height = ascent + descent;
+  if (FLOATP (val))
+    height = (int)(XFLOAT_DATA (val) * height);
+  else if (INTEGERP (val))
+    height *= XINT (val);
+
+  return make_number (height);
+}
+
+
 /* RIF:
    Produce glyphs/get display metrics for the display element IT is
    loaded with.  See the description of struct display_iterator in
@@ -18596,17 +18691,20 @@
 
 	  it->nglyphs = 1;
 
-	  if (it->use_default_face)
-	    {
-	      font = FRAME_FONT (it->f);
-	      boff = FRAME_BASELINE_OFFSET (it->f);
-	    }
-
 	  pcm = rif->per_char_metric (font, &char2b,
 				      FONT_TYPE_FOR_UNIBYTE (font, it->char_to_display));
 
-	  it->ascent = FONT_BASE (font) + boff;
-	  it->descent = FONT_DESCENT (font) - boff;
+	  if (it->override_ascent >= 0)
+	    {
+	      it->ascent = it->override_ascent;
+	      it->descent = it->override_descent;
+	      boff = it->override_boff;
+	    }
+	  else
+	    {
+	      it->ascent = FONT_BASE (font) + boff;
+	      it->descent = FONT_DESCENT (font) - boff;
+	    }
 
 	  if (pcm)
 	    {
@@ -18709,26 +18807,27 @@
 	     But if previous part of the line set a height, don't
 	     increase that height */
 
-	  Lisp_Object lsp, lh;
-
+	  Lisp_Object height, spacing;
+
+	  it->override_ascent = -1;
 	  it->pixel_width = 0;
 	  it->nglyphs = 0;
 
-	  lh = Fget_text_property (make_number (IT_CHARPOS (*it)),
-				   Qline_height, it->object);
-
-	  if (EQ (lh, Qt))
-	    {
-	      it->use_default_face = 1;
-	      font = FRAME_FONT (it->f);
-	      boff = FRAME_BASELINE_OFFSET (it->f);
-	      font_info = NULL;
-	    }
-
-	  it->ascent = FONT_BASE (font) + boff;
-	  it->descent = FONT_DESCENT (font) - boff;
-
-	  if (EQ (lh, make_number (0)))
+	  height = calc_line_height_property(it, Qline_height, font, boff);
+
+	  if (it->override_ascent >= 0)
+	    {
+	      it->ascent = it->override_ascent;
+	      it->descent = it->override_descent;
+	      boff = it->override_boff;
+	    }
+	  else
+	    {
+	      it->ascent = FONT_BASE (font) + boff;
+	      it->descent = FONT_DESCENT (font) - boff;
+	    }
+
+	  if (EQ (height, make_number(0)))
 	    {
 	      if (it->descent > it->max_descent)
 		{
@@ -18747,7 +18846,6 @@
 	    }
 	  else
 	    {
-	      int explicit_height = -1;
 	      it->phys_ascent = it->ascent;
 	      it->phys_descent = it->descent;
 
@@ -18758,23 +18856,20 @@
 		  it->ascent += face->box_line_width;
 		  it->descent += face->box_line_width;
 		}
-	      if (INTEGERP (lh))
-		explicit_height = XINT (lh);
-	      else if (FLOATP (lh))
-		explicit_height = (it->phys_ascent + it->phys_descent)
-		  * XFLOAT_DATA (lh);
-
-	      if (explicit_height > it->ascent + it->descent)
-		it->ascent = explicit_height - it->descent;
-	    }
-
-	  lsp = Fget_text_property (make_number (IT_CHARPOS (*it)),
-				    Qline_spacing, it->object);
-	  if (INTEGERP (lsp))
-	    extra_line_spacing = XINT (lsp);
-	  else if (FLOATP (lsp))
-	    extra_line_spacing = (it->phys_ascent + it->phys_descent)
-	      * XFLOAT_DATA (lsp);
+	      if (!NILP (height)
+		  && XINT (height) > it->ascent + it->descent)
+		it->ascent = XINT (height) - it->descent;
+	    }
+
+	  spacing = calc_line_height_property(it, Qline_spacing, font, boff);
+	  if (!NILP (spacing))
+	    {
+	      int sp = XINT (spacing);
+	      if (sp < 0)
+		extra_line_spacing = (-sp) - (it->phys_ascent + it->phys_descent);
+	      else
+		extra_line_spacing = sp;
+	    }
 	}
       else if (it->char_to_display == '\t')
 	{
@@ -19151,7 +19246,8 @@
   if (it->area == TEXT_AREA)
     it->current_x += it->pixel_width;
 
-  it->descent += extra_line_spacing;
+  if (extra_line_spacing > 0)
+    it->descent += extra_line_spacing;
 
   it->max_ascent = max (it->max_ascent, it->ascent);
   it->max_descent = max (it->max_descent, it->descent);
@@ -20815,9 +20911,9 @@
       if (BUFFERP (object))
 	{
 	  /* Put all the overlays we want in a vector in overlay_vec.
-	     Store the length in len.  If there are more than 10, make
+	     Store the length in len.  If there are more than 40, make
 	     enough space for all, and try again.  */
-	  len = 10;
+	  len = 40;
 	  overlay_vec = (Lisp_Object *) alloca (len * sizeof (Lisp_Object));
 	  noverlays = overlays_at (pos, 0, &overlay_vec, &len, NULL, NULL, 0);
 	  if (noverlays > len)