]> git.donarmstrong.com Git - lilypond.git/blobdiff - lily/slur.cc
* SConstruct: Further development.
[lilypond.git] / lily / slur.cc
index 7473197e22e5dabfe3d0843270211a9d346f3c3e..baaa56ec29d327861daff0ed376b97b6a54fdb48 100644 (file)
@@ -1,9 +1,9 @@
 /*
-  slur.cc -- implement  Slur
+  slur.cc -- implement Slur
 
   source file of the GNU LilyPond music typesetter
 
-  (c) 1996--2003 Han-Wen Nienhuys <hanwen@cs.uu.nl>
+  (c) 1996--2004 Han-Wen Nienhuys <hanwen@cs.uu.nl>
     Jan Nieuwenhuizen <janneke@gnu.org>
 */
 
 #include "group-interface.hh"
 #include "slur.hh"
 #include "lookup.hh"
-#include "paper-def.hh"
+#include "output-def.hh"
 #include "note-column.hh"
 #include "stem.hh"
 #include "paper-column.hh"
-#include "molecule.hh"
+#include "stencil.hh"
 #include "warn.hh"
 #include "slur-bezier-bow.hh"
 #include "main.hh"
@@ -42,8 +42,8 @@ void
 Slur::set_interface (Grob*me)
 {
   /* Copy to mutable list. */
-  me->set_grob_property ("attachment",
-                        ly_deep_copy (me->get_grob_property ("attachment")));
+  me->set_property ("attachment",
+                        ly_deep_copy (me->get_property ("attachment")));
 }
 
 void
@@ -56,7 +56,7 @@ Slur::add_column (Grob*me, Grob*n)
 }
 
 void
-Slur::de_uglyfy (Grob*me, Slur_bezier_bow* bb, Real default_height)
+Slur::de_uglify (Grob*me, Slur_bezier_bow* bb, Real default_height)
 {
   Real length = bb->curve_.control_[3][X_AXIS] ; 
   Real ff = bb->fit_factor ();
@@ -67,11 +67,11 @@ Slur::de_uglyfy (Grob*me, Slur_bezier_bow* bb, Real default_height)
       Real h = bb->curve_.control_[i][Y_AXIS] * ff / length;
 
       Real f = default_height / length;
-      SCM up = me->get_grob_property ("de-uglify-parameters");
+      SCM up = me->get_property ("de-uglify-parameters");
       
-      Real c1 = gh_scm2double (ly_car (up));
-      Real c2 = gh_scm2double (ly_cadr (up));
-      Real c3 = gh_scm2double (ly_caddr (up)); 
+      Real c1 = ly_scm2double (ly_car (up));
+      Real c2 = ly_scm2double (ly_cadr (up));
+      Real c3 = ly_scm2double (ly_caddr (up)); 
       
       if (h > c1 * f)
        {
@@ -114,13 +114,14 @@ SCM
 Slur::after_line_breaking (SCM smob)
 {
   Grob *me = unsmob_grob (smob);
-  if (!scm_ilength (me->get_grob_property ("note-columns")))
+  if (!scm_ilength (me->get_property ("note-columns")))
     {
       me->suicide ();
       return SCM_UNSPECIFIED;
     }
   set_extremities (me);
-  set_control_points (me);
+  if (!ly_c_pair_p (me->get_property ("control-points")))
+    set_control_points (me);
   return SCM_UNSPECIFIED;
 } 
 
@@ -128,12 +129,11 @@ Slur::after_line_breaking (SCM smob)
 void
 Slur::check_slope (Grob *me)
 {
-
   /*
     Avoid too steep slurs.
    */
-  SCM s = me->get_grob_property ("slope-limit");
-  if (gh_number_p (s))
+  SCM s = me->get_property ("slope-limit");
+  if (ly_c_number_p (s))
     {
       Array<Offset> encompass = get_encompass_offsets (me);
       Drul_array<Offset> attachment;
@@ -147,23 +147,23 @@ Slur::check_slope (Grob *me)
       
       Real slope = slope = abs (dy / dx);
 
-      Real limit = gh_scm2double (s);
+      Real limit = ly_scm2double (s);
 
       if (slope > limit)
        {
          Real staff_space = Staff_symbol_referencer::staff_space ((Grob*)me);
-         Direction dir = (Direction)gh_scm2int (me->get_grob_property ("direction"));
+         Direction dir = (Direction)ly_scm2int (me->get_property ("direction"));
          Direction d = (Direction) (- dir * (sign (dy)));
-         SCM a = me->get_grob_property ("attachment-offset");
+         SCM a = me->get_property ("attachment-offset");
          Drul_array<Offset> o;
          o[LEFT] = ly_scm2offset (index_get_cell (a, LEFT));
          o[RIGHT] = ly_scm2offset (index_get_cell (a, RIGHT));
          o[d][Y_AXIS] -= (limit - slope) * dx * dir / staff_space;
 
-         o[d][Y_AXIS] *= Directional_element_interface::get (me);
+         o[d][Y_AXIS] *= get_grob_direction (me);
 
-         me->set_grob_property ("attachment-offset",
-                               gh_cons (ly_offset2scm (o[LEFT]),
+         me->set_property ("attachment-offset",
+                               scm_cons (ly_offset2scm (o[LEFT]),
                                         ly_offset2scm (o[RIGHT])));
        }
     }
@@ -176,37 +176,33 @@ Slur::check_slope (Grob *me)
 SCM
 Slur::set_extremities (Grob *me)
 {
-  if (!Directional_element_interface::get (me))
-    Directional_element_interface::set (me, get_default_dir (me));
+  if (!get_grob_direction (me))
+    set_grob_direction (me, get_default_dir (me));
 
-  SCM att = me->get_grob_property ("attachment");
+  SCM att = me->get_property ("attachment");
       /*
        */
-      if (!gh_pair_p (att))
+      if (!ly_c_pair_p (att))
        {
          programming_error ("attachment is not a cons?!");
-         att = gh_cons (SCM_EOL, SCM_EOL);
-         me->set_grob_property ("attachment", att);
+         att = scm_cons (SCM_EOL, SCM_EOL);
+         me->set_property ("attachment", att);
        }
       
   Direction dir = LEFT;
   do 
     {
     
-      if (!gh_symbol_p (index_get_cell (att, dir)))
+      if (!ly_c_symbol_p (index_get_cell (att, dir)))
        {
-         for (SCM s = me->get_grob_property ("extremity-rules");
-              s != SCM_EOL; s = ly_cdr (s))
-           {
-             SCM r = gh_call2 (ly_caar (s), me->self_scm (),
-                                gh_int2scm ((int)dir));
-             if (r != SCM_BOOL_F)
-               {
-                 index_set_cell (att, dir,
-                                 ly_cdar (s));
-                 break;
-               }
-           }
+         SCM p = me->get_property ("extremity-function");
+         SCM res = ly_symbol2scm ("head");
+         
+         if (ly_c_procedure_p (p))
+           res =  scm_call_2 (p, me->self_scm (), scm_int2num (dir));
+
+         if (ly_c_symbol_p (res))
+           index_set_cell (att, dir, res);
        }
     }
   while (flip (&dir) != LEFT);
@@ -220,9 +216,9 @@ Slur::set_extremities (Grob *me)
 Real
 Slur::get_boundary_notecolumn_y (Grob *me, Direction dir)
 {
-  SCM cols = me->get_grob_property ("note-columns");
+  SCM cols = me->get_property ("note-columns");
 
-  if(!gh_pair_p (cols))
+  if (!ly_c_pair_p (cols))
     {
       programming_error ("No note-columns in slur?");
       me->suicide ();
@@ -230,7 +226,7 @@ Slur::get_boundary_notecolumn_y (Grob *me, Direction dir)
     }
   
   if (dir == LEFT)
-    cols = scm_reverse(cols);
+    cols = scm_reverse (cols);
   
   Grob *col = unsmob_grob (ly_car (cols));
   Grob *common[] =
@@ -262,14 +258,18 @@ Slur::broken_trend_offset (Grob *me, Direction dir)
   if (Spanner *mother =  dynamic_cast<Spanner*> (me->original_))
     {
       int k = broken_spanner_index (dynamic_cast<Spanner*> (me));
-      Grob *neighbour = mother->broken_intos_[k + dir];      
+      int j = k + dir;
+      if (j < 0 || j >= mother->broken_intos_.size ())
+       return o;
+      
+      Grob *neighbour = mother->broken_intos_[j];      
       if (dir == RIGHT)
-       neighbour->set_grob_property ("direction",
-                                     me->get_grob_property ("direction"));
+       neighbour->set_property ("direction",
+                                     me->get_property ("direction"));
       Real neighbour_y = get_boundary_notecolumn_y (neighbour, dir);
       Real y = get_boundary_notecolumn_y (me, -dir);
-      int neighbour_cols = scm_ilength (neighbour->get_grob_property ("note-columns"));
-      int cols = scm_ilength (me->get_grob_property ("note-columns"));
+      int neighbour_cols = scm_ilength (neighbour->get_property ("note-columns"));
+      int cols = scm_ilength (me->get_property ("note-columns"));
       o = Offset (0, (y*neighbour_cols + neighbour_y*cols) /
                  (cols + neighbour_cols));
     }
@@ -287,43 +287,50 @@ Offset
 Slur::get_attachment (Grob *me, Direction dir,
                      Grob **common) 
 {
-  SCM s = me->get_grob_property ("attachment");
-  if (!gh_symbol_p (index_get_cell (s, dir)))
+  SCM s = me->get_property ("attachment");
+  if (!ly_c_pair_p (s) || !ly_c_symbol_p (index_get_cell (s, dir)))
     {
       s = set_extremities (me);
     }
   
-  SCM a = dir == LEFT ? ly_car (s) : ly_cdr (s);
+  SCM a = (dir == LEFT) ? ly_car (s) : ly_cdr (s);
   Spanner*sp = dynamic_cast<Spanner*> (me);
   String str = ly_symbol2string (a);
+  
   Real staff_space = Staff_symbol_referencer::staff_space ((Grob*)me);
   Real hs = staff_space / 2.0;
   Offset o;
   
-  int slurdir = gh_scm2int (me->get_grob_property ("direction"));
+  Direction slurdir = to_dir (me->get_property ("direction"));
   
   Grob *stem = 0;
   if (Note_column::has_interface (sp->get_bound (dir)))
     {
       Grob * n =sp->get_bound (dir);
-      if ((stem = Note_column::get_stem (n)))
+      stem = Note_column::get_stem (n);
+      if (stem)
        {
          Real x_extent;
-         if (Grob *head = Note_column::first_head (n))
+         Grob *head = Note_column::first_head (n);
+         if (head)
            x_extent = head->extent (head, X_AXIS).length ();
          else
            x_extent = n->extent (n, X_AXIS).length ();
 
-         if (str == "head")
+         if (!head)
+           {
+             o = Offset (0, n->extent (n, Y_AXIS)[slurdir]);
+           }
+         else if (str == "head")
            {
              o = Offset (0, Stem::head_positions (stem)
-                         [Directional_element_interface::get (me)] * hs);
+                         [slurdir] * hs);
              /*
                Default position is centered in X, on outer side of head Y
               */
              o += Offset (0.5 * x_extent,
                           0.5 * staff_space
-                          * Directional_element_interface::get (me));
+                          * slurdir);
            }
          else if (str == "alongside-stem")
            {
@@ -333,7 +340,7 @@ Slur::get_attachment (Grob *me, Direction dir,
               */
              o += Offset (x_extent * (1 + Stem::get_direction (stem)),
                           0.5 * staff_space
-                          * Directional_element_interface::get (me));
+                          * slurdir);
            }
          else if (str == "stem")
            {
@@ -341,9 +348,7 @@ Slur::get_attachment (Grob *me, Direction dir,
              /*
                Default position is on stem X, at stem end Y
               */
-             Real stem_thickness =
-               gh_scm2double (stem->get_grob_property ("thickness"))
-               * stem->get_paper ()->get_var ("linethickness");
+             Real stem_thickness = Stem::thickness (stem);
              o += Offset (0.5 *
                           x_extent * (1 + Stem::get_direction (stem))
                           - ((dir + 1)/2) * stem_thickness
@@ -365,19 +370,19 @@ Slur::get_attachment (Grob *me, Direction dir,
        o = broken_trend_offset (me, dir);
     }
 
-  SCM alist = me->get_grob_property ("extremity-offset-alist");
+  SCM alist = me->get_property ("extremity-offset-alist");
   int stemdir = stem ? Stem::get_direction (stem) : 1;
   SCM l = scm_assoc
     (scm_list_n (a,
-                 gh_int2scm (stemdir * dir),
-                 gh_int2scm (slurdir * dir),
+                 scm_int2num (stemdir * dir),
+                 scm_int2num (slurdir * dir),
                   SCM_UNDEFINED), alist);
 
   if (l != SCM_BOOL_F)
     {
       Offset off = ly_scm2offset (ly_cdr (l)) * staff_space;
       off[X_AXIS] *= dir;
-      off[Y_AXIS] *= Directional_element_interface::get (me);
+      off[Y_AXIS] *= slurdir;
       o += off;
     }
 
@@ -391,11 +396,11 @@ Slur::get_attachment (Grob *me, Direction dir,
        - me->relative_coordinate (common[Y_AXIS], Y_AXIS);
     }
 
-  Offset off = ly_scm2offset (index_get_cell (me->get_grob_property
+  Offset off = ly_scm2offset (index_get_cell (me->get_property
                                          ("attachment-offset"),
                                          dir)) * staff_space;
 
-  off[Y_AXIS] *= Directional_element_interface::get (me);
+  off[Y_AXIS] *= slurdir;
   o += off;
   return o;
 }
@@ -406,9 +411,9 @@ Slur::encompass_offset (Grob*me,
                        Grob **common) 
 {
   Offset o;
-  Grob* stem = unsmob_grob (col->get_grob_property ("stem"));
+  Grob* stem = unsmob_grob (col->get_property ("stem"));
   
-  Direction dir = Directional_element_interface::get (me);
+  Direction dir = get_grob_direction (me);
   
   if (!stem)
     {
@@ -417,7 +422,7 @@ Slur::encompass_offset (Grob*me,
       o[Y_AXIS] = col->relative_coordinate (common[Y_AXIS], Y_AXIS);
       return o;  
     }
-  Direction stem_dir = Directional_element_interface::get (stem);
+  Direction stem_dir = get_grob_direction (stem);
   o[X_AXIS] = stem->relative_coordinate (0, X_AXIS);
 
   /*
@@ -431,7 +436,7 @@ Slur::encompass_offset (Grob*me,
   o[X_AXIS] -= 0.5 * stem_dir * x_extent;
 
   if ((stem_dir == dir)
-      && !stem->extent (stem, Y_AXIS).empty_b ())
+      && !stem->extent (stem, Y_AXIS).is_empty ())
     {
       o[Y_AXIS] = stem->extent (common[Y_AXIS], Y_AXIS)[dir];
     }
@@ -443,7 +448,7 @@ Slur::encompass_offset (Grob*me,
   /*
    leave a gap: slur mustn't touch head/stem
    */
-  o[Y_AXIS] += dir * gh_scm2double (me->get_grob_property ("y-free")) *
+  o[Y_AXIS] += dir * robust_scm2double (me->get_property ("y-free"), 0) *
     1.0;
   return o;
 }
@@ -452,7 +457,7 @@ Array<Offset>
 Slur::get_encompass_offsets (Grob *me)
 {
   Spanner*sp = dynamic_cast<Spanner*> (me);
-  SCM eltlist = me->get_grob_property ("note-columns");
+  SCM eltlist = me->get_property ("note-columns");
   Grob *common[] = {common_refpoint_of_list (eltlist, me, X_AXIS),
                    common_refpoint_of_list (eltlist, me, Y_AXIS)};
 
@@ -519,42 +524,52 @@ MAKE_SCHEME_CALLBACK (Slur, height, 2);
 SCM
 Slur::height (SCM smob, SCM ax)
 {
-  Axis a = (Axis)gh_scm2int (ax);
+  Axis a = (Axis)ly_scm2int (ax);
   Grob * me = unsmob_grob (smob);
   assert (a == Y_AXIS);
 
-  SCM mol = me->get_uncached_molecule ();
-  return ly_interval2scm (unsmob_molecule (mol)->extent (a));
+  SCM mol = me->get_uncached_stencil ();
+  Interval ext;
+  if (Stencil * m = unsmob_stencil (mol))
+    ext = m->extent (a);
+  return ly_interval2scm (ext);
 }
 
 /*
   Ugh should have dash-length + dash-period
  */
-MAKE_SCHEME_CALLBACK (Slur, brew_molecule,1);
+MAKE_SCHEME_CALLBACK (Slur, print,1);
 SCM
-Slur::brew_molecule (SCM smob)
+Slur::print (SCM smob)
 {
   Grob * me = unsmob_grob (smob);
-  if (!scm_ilength (me->get_grob_property ("note-columns")))
+  if (!scm_ilength (me->get_property ("note-columns")))
     {
       me->suicide ();
       return SCM_EOL;
     }
 
-  Real thick = me->get_paper ()->get_var ("linethickness") *
-    gh_scm2double (me->get_grob_property ("thickness"));
+  Real base_thick = robust_scm2double (me->get_property ("thickness"), 1);
+  Real thick = base_thick * Staff_symbol_referencer::line_thickness (me);
+
+  Real ss = Staff_symbol_referencer::staff_space (me);
   Bezier one = get_curve (me);
 
   // get_curve may suicide
-  if (!scm_ilength (me->get_grob_property ("note-columns")))
+  if (!scm_ilength (me->get_property ("note-columns")))
     return SCM_EOL;
 
-  Molecule a;
-  SCM d =  me->get_grob_property ("dashed");
-  if (gh_number_p (d))
-    a = Lookup::dashed_slur (one, thick, thick * gh_scm2double (d));
+  Stencil a;
+
+  /*
+    TODO: replace dashed with generic property.
+   */
+  SCM d =  me->get_property ("dashed");
+  if (ly_c_number_p (d))
+    a = Lookup::dashed_slur (one, thick, thick * robust_scm2double (d, 0));
   else
-    a = Lookup::slur (one, Directional_element_interface::get (me) * thick, thick);
+    a = Lookup::slur (one, get_grob_direction (me) * base_thick * ss / 10.0,
+                     thick);
 
   return a.smobbed_copy ();
 }
@@ -564,15 +579,15 @@ Slur::set_control_points (Grob*me)
 {
   Real staff_space = Staff_symbol_referencer::staff_space ((Grob*)me);
 
-  SCM details = me->get_grob_property ("details");
-  SCM h_inf_scm = me->get_grob_property ("height-limit");
-  SCM r_0_scm = me->get_grob_property ("ratio");
+  SCM details = me->get_property ("details");
+  SCM h_inf_scm = me->get_property ("height-limit");
+  SCM r_0_scm = me->get_property ("ratio");
 
-  Real r_0 = gh_scm2double (r_0_scm);
-  Real h_inf = staff_space * gh_scm2double (h_inf_scm);
+  Real r_0 = robust_scm2double (r_0_scm, 1);
+  Real h_inf = staff_space * ly_scm2double (h_inf_scm);
   
   Slur_bezier_bow bb (get_encompass_offsets (me),
-                     Directional_element_interface::get (me),
+                     get_grob_direction (me),
                      h_inf, r_0);
 
 
@@ -581,16 +596,16 @@ Slur::set_control_points (Grob*me)
       Real length = bb.curve_.control_[3][X_AXIS]; 
       Real default_height = slur_height (length, h_inf, r_0);
 
-      SCM ssb = me->get_grob_property ("beautiful");
+      SCM ssb = me->get_property ("beautiful");
       Real sb = 0;
-      if (gh_number_p (ssb))
-       sb = gh_scm2double (ssb);
+      if (ly_c_number_p (ssb))
+       sb = ly_scm2double (ssb);
 
       bb.minimise_enclosed_area (sb, details);
       SCM sbf = scm_assq (ly_symbol2scm ("force-blowfit"), details);
       Real bff = 1.0;
-      if (gh_pair_p (sbf) && gh_number_p (ly_cdr (sbf)))
-         bff = gh_scm2double (ly_cdr (sbf));
+      if (ly_c_pair_p (sbf) && ly_c_number_p (ly_cdr (sbf)))
+         bff = ly_scm2double (ly_cdr (sbf));
 
       bb.curve_.control_[1][Y_AXIS] *= bff;
       bb.curve_.control_[2][Y_AXIS] *= bff;
@@ -604,7 +619,7 @@ Slur::set_control_points (Grob*me)
        Slurs that fit beautifully are not ugly
       */
       if (area > beautiful)
-       de_uglyfy (me, &bb, default_height);
+       de_uglify (me, &bb, default_height);
     }
 
   Bezier b = bb.get_bezier ();
@@ -613,7 +628,7 @@ Slur::set_control_points (Grob*me)
   SCM controls = SCM_EOL;
   for (int i= 4; i--;)
     {
-      controls = gh_cons (ly_offset2scm (b.control_[i]), controls);
+      controls = scm_cons (ly_offset2scm (b.control_[i]), controls);
       /*
        BRRR WHURG.
        All these null control-points, where do they all come from?
@@ -625,7 +640,7 @@ Slur::set_control_points (Grob*me)
        }
     }
 
-  me->set_grob_property ("control-points", controls);
+  me->set_property ("control-points", controls);
 }
   
 Bezier
@@ -634,31 +649,31 @@ Slur::get_curve (Grob*me)
   Bezier b;
   int i = 0;
 
-  SCM attach = me->get_grob_property ("attachment");
-  if (!gh_pair_p (attach))
-    attach = set_extremities(me);
+  SCM attach = me->get_property ("attachment");
+  if (!ly_c_pair_p (attach))
+    attach = set_extremities (me);
 
   
-  if (!Directional_element_interface::get (me)
-      || ! gh_symbol_p (index_get_cell (attach, LEFT))
-      || ! gh_symbol_p (index_get_cell (attach, RIGHT)))
+  if (!get_grob_direction (me)
+      || ! ly_c_symbol_p (index_get_cell (attach, LEFT))
+      || ! ly_c_symbol_p (index_get_cell (attach, RIGHT)))
     set_extremities (me);
   
-  if (!gh_pair_p (me->get_grob_property ("control-points")))
+  if (!ly_c_pair_p (me->get_property ("control-points")))
     set_control_points (me);
 
   // set_control_points may suicide
-  if (!scm_ilength (me->get_grob_property ("note-columns")))
+  if (!scm_ilength (me->get_property ("note-columns")))
     return b;
 
-  for (SCM s= me->get_grob_property ("control-points"); s != SCM_EOL; s = ly_cdr (s))
+  for (SCM s= me->get_property ("control-points"); s != SCM_EOL; s = ly_cdr (s))
     {
       b.control_[i] = ly_scm2offset (ly_car (s));
       i++;
     }
 
   Array<Offset> enc (get_encompass_offsets (me));
-  Direction dir = Directional_element_interface::get (me);
+  Direction dir = get_grob_direction (me);
   
   Real x1 = enc[0][X_AXIS];
   Real x2 = enc.top ()[X_AXIS];
@@ -682,5 +697,5 @@ Slur::get_curve (Grob*me)
 
 ADD_INTERFACE (Slur,"slur-interface",
   "A slur",
-  "attachment attachment-offset beautiful control-points dashed details de-uglify-parameters direction extremity-rules extremity-offset-alist height-limit ratio slope-limit thickness y-free");
+  "attachment attachment-offset beautiful control-points dashed details de-uglify-parameters direction extremity-function extremity-offset-alist height-limit note-columns ratio slope-limit thickness y-free");