]> git.donarmstrong.com Git - lilypond.git/blob - lily/slur-engraver.cc
* lily/slur-quanting.cc (score_extra_encompass): avoid other slurs
[lilypond.git] / lily / slur-engraver.cc
1 /*
2   slur-engraver.cc -- implement Slur_engraver
3
4   (c) 1997--2004 Han-Wen Nienhuys <hanwen@cs.uu.nl>
5 */
6
7 #include "event.hh"
8 #include "new-slur.hh"
9 #include "note-column.hh"
10 #include "context.hh"
11 #include "directional-element-interface.hh"
12 #include "engraver.hh"
13 #include "spanner.hh"
14 #include "tie.hh"
15
16 /*
17   It is possible that a slur starts and ends on the same note.  At
18   least, it is for phrasing slurs: a note can be both beginning and
19   ending of a phrase.
20 */
21
22 class Slur_engraver : public Engraver
23 {
24   Drul_array<Music *> events_;
25   Music * running_slur_start_;
26   Link_array<Grob> slurs_;
27   Link_array<Grob> end_slurs_;
28
29   void set_melisma (bool);
30
31 protected:
32   virtual bool try_music (Music*);
33   virtual void acknowledge_grob (Grob_info);
34   virtual void stop_translation_timestep ();
35   virtual void finalize ();
36   virtual void process_music ();
37
38 public:
39   TRANSLATOR_DECLARATIONS (Slur_engraver);
40 };
41
42 Slur_engraver::Slur_engraver ()
43 {
44   events_[START] =events_[STOP] = 0;
45 }
46
47 bool
48 Slur_engraver::try_music (Music *m)
49 {
50   if (m->is_mus_type ("slur-event"))
51     {
52       /*
53         Let's not start more than one slur per moment.
54       */
55       Direction d = to_dir (m->get_property ("span-direction"));
56       if (d == START)
57         {
58           events_[START] = m;
59           return true;
60         }
61       else if (d == STOP)
62         {
63           if (slurs_.is_empty ())
64             return false;
65           
66           events_[STOP] = m;
67           return true;
68         }
69     }
70   return false;
71 }
72
73 void
74 Slur_engraver::set_melisma (bool m)
75 {
76   context ()->set_property ("slurMelismaBusy", m ? SCM_BOOL_T :SCM_BOOL_F);
77 }
78
79 void
80 Slur_engraver::acknowledge_grob (Grob_info info)
81 {
82   Grob *e =info.grob_;
83   if (Note_column::has_interface (info.grob_))
84     {
85       for (int i = slurs_.size (); i--; )
86         New_slur::add_column (slurs_[i], e);
87       for (int i = end_slurs_.size (); i-- ; )
88         New_slur::add_column (end_slurs_[i], e);
89     }
90   else
91     {
92       SCM inside = e->get_property ("inside-slur");
93       if (Tie::has_interface (e)
94           || to_boolean (inside))
95         {
96           for (int i = slurs_.size (); i--; )
97             New_slur::add_extra_encompass (slurs_[i], e);
98           for (int i = end_slurs_.size (); i--; )
99             New_slur::add_extra_encompass (end_slurs_[i], e);
100         }
101       else if (inside == SCM_BOOL_F)
102         {
103           Grob *slur = slurs_.size()?slurs_[0] : 0;
104           slur =  (end_slurs_.size () && !slur)
105             ? end_slurs_[0] : slur;
106
107           if (slur)
108             {
109               e->add_offset_callback (New_slur::outside_slur_callback_proc, Y_AXIS);
110               e->set_property ("slur", slur->self_scm());
111             }
112         }
113     }
114 }
115
116 void
117 Slur_engraver::finalize ()
118 {
119   if (slurs_.size ())
120     slurs_[0]->warning (_("unterminated slur"));
121 }
122
123 void
124 Slur_engraver::process_music ()
125 {
126   if (events_[STOP])
127     {
128       end_slurs_ = slurs_;
129       slurs_.clear ();
130     }
131   
132   if (events_[START] && slurs_.is_empty ())
133     {
134       Music *ev = events_[START];
135
136       bool double_slurs = to_boolean (get_property ("doubleSlurs"));
137
138       Grob * slur = make_spanner ("Slur", events_[START]->self_scm ());
139       Direction updown = to_dir (ev->get_property ("direction"));
140       if (updown && !double_slurs)
141         set_grob_direction (slur, updown);
142
143       slurs_.push (slur);
144
145       if (double_slurs)
146         {
147           set_grob_direction (slur, DOWN);
148           slur = make_spanner ("Slur", events_[START]->self_scm ());
149           set_grob_direction (slur, UP);
150           slurs_.push (slur); 
151         }
152     }
153   set_melisma (slurs_.size ());
154 }
155
156 void
157 Slur_engraver::stop_translation_timestep ()
158 {
159   end_slurs_.clear ();
160   events_[START] = events_[STOP] = 0;
161 }
162
163 ENTER_DESCRIPTION (Slur_engraver,
164   /* descr */       "Build slurs grobs from slur events",
165   /* creats*/       "Slur",
166   /* accepts */     "slur-event",
167   /* acks  */      "note-column-interface accidental-interface fingering-interface script-interface tie-interface",
168   /* reads */       "slurMelismaBusy doubleSlurs",
169   /* write */       "");