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