]> git.donarmstrong.com Git - lilypond.git/blob - lily/phrasing-slur-engraver.cc
Update.
[lilypond.git] / lily / phrasing-slur-engraver.cc
1 /*
2   phrasing-slur-engraver.cc -- implement Phrasing_slur_engraver
3
4   (c) 1997--2005 Han-Wen Nienhuys <hanwen@cs.uu.nl>
5 */
6
7 #include "context.hh"
8 #include "directional-element-interface.hh"
9 #include "engraver.hh"
10 #include "slur.hh"
11 #include "note-column.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 Phrasing_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 protected:
29   virtual bool try_music (Music *);
30   virtual void acknowledge_grob (Grob_info);
31   virtual void stop_translation_timestep ();
32   virtual void finalize ();
33   virtual void process_music ();
34
35 public:
36   TRANSLATOR_DECLARATIONS (Phrasing_slur_engraver);
37 };
38
39 Phrasing_slur_engraver::Phrasing_slur_engraver ()
40 {
41   events_[START] = events_[STOP] = 0;
42 }
43
44 bool
45 Phrasing_slur_engraver::try_music (Music *m)
46 {
47   if (m->is_mus_type ("phrasing-slur-event"))
48     {
49       /*
50         Let's not start more than one slur per moment.
51       */
52       Direction d = to_dir (m->get_property ("span-direction"));
53       if (d == START)
54         {
55           events_[START] = m;
56           return true;
57         }
58       else if (d == STOP)
59         {
60           if (slurs_.is_empty ())
61             return false;
62
63           events_[STOP] = m;
64           return true;
65         }
66     }
67   return false;
68 }
69
70 void
71 Phrasing_slur_engraver::acknowledge_grob (Grob_info info)
72 {
73   Grob *e = info.grob_;
74   if (Note_column::has_interface (info.grob_))
75     {
76       for (int i = slurs_.size (); i--;)
77         Slur::add_column (slurs_[i], e);
78       for (int i = end_slurs_.size (); i--;)
79         Slur::add_column (end_slurs_[i], e);
80     }
81   else
82     {
83       /*
84         ugh. cut & paste from slur-engraver.cc
85       */
86       SCM inside = e->get_property ("inside-slur");
87       if (Tie::has_interface (e)
88           || Slur::has_interface (e)
89           || to_boolean (inside))
90         {
91           for (int i = slurs_.size (); i--;)
92             Slur::add_extra_encompass (slurs_[i], e);
93           for (int i = end_slurs_.size (); i--;)
94             Slur::add_extra_encompass (end_slurs_[i], e);
95         }
96       else if (inside == SCM_BOOL_F)
97         {
98           Grob *slur = slurs_.size () ? slurs_[0] : 0;
99           slur = (end_slurs_.size () && !slur)
100             ? end_slurs_[0] : slur;
101
102           if (slur)
103             {
104               e->add_offset_callback (Slur::outside_slur_callback_proc, Y_AXIS);
105               e->set_property ("slur", slur->self_scm ());
106             }
107         }
108     }
109 }
110
111 void
112 Phrasing_slur_engraver::finalize ()
113 {
114   if (slurs_.size ())
115     slurs_[0]->warning (_ ("unterminated phrasing slur"));
116 }
117
118 void
119 Phrasing_slur_engraver::process_music ()
120 {
121   if (events_[STOP])
122     {
123       end_slurs_ = slurs_;
124       slurs_.clear ();
125     }
126
127   if (events_[START] && slurs_.is_empty ())
128     {
129       Music *ev = events_[START];
130
131       Grob *slur = make_spanner ("PhrasingSlur", events_[START]->self_scm ());
132       Direction updown = to_dir (ev->get_property ("direction"));
133       if (updown)
134         set_grob_direction (slur, updown);
135
136       slurs_.push (slur);
137     }
138 }
139
140 void
141 Phrasing_slur_engraver::stop_translation_timestep ()
142 {
143   end_slurs_.clear ();
144   events_[START] = events_[STOP] = 0;
145 }
146
147 ADD_TRANSLATOR (Phrasing_slur_engraver,
148                 /* descr */ "Print phrasing slurs. Similar to @ref{Slur_engraver}",
149                 /* creats*/ "PhrasingSlur",
150                 /* accepts */ "phrasing-slur-event",
151                 /* acks  */ "note-column-interface tie-interface fingering-interface script-interface slur-interface",
152                 /* reads */ "",
153                 /* write */ "");