]> git.donarmstrong.com Git - lilypond.git/blob - lily/phrasing-slur-engraver.cc
* lily/include/lily-guile.hh: many new ly_ functions. Thanks to
[lilypond.git] / lily / phrasing-slur-engraver.cc
1 /*
2   phrasing-slur-engraver.cc -- implement Phrasing_slur_engraver
3
4   (c) 1997--2004 Han-Wen Nienhuys <hanwen@cs.uu.nl>
5 */
6
7 #include "event.hh"
8 #include "slur.hh"
9 #include "warn.hh"
10 #include "note-column.hh"
11 #include "context.hh"
12
13 #include "engraver.hh"
14 #include "spanner.hh"
15
16 /*
17   TODO:
18   
19   ALGRGRRGRG
20
21   Derive this from Slur_engraver. This code is completely duplicate.
22 */
23 class Phrasing_slur_engraver : public Engraver
24 {
25   Link_array<Music> eventses_;
26   Link_array<Music> new_phrasing_slur_evs_;
27   Link_array<Grob> phrasing_slurs_;
28   Link_array<Grob> end_phrasing_slurs_;
29   Moment last_start_;
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 start_translation_timestep ();
36   virtual void finalize ();
37   virtual void process_acknowledged_grobs ();
38
39 public:
40   TRANSLATOR_DECLARATIONS (Phrasing_slur_engraver);
41   
42 };
43
44 Phrasing_slur_engraver::Phrasing_slur_engraver ()
45 {
46   last_start_ = Moment (-1);
47 }
48
49 bool
50 Phrasing_slur_engraver::try_music (Music *ev)
51 {
52  if (ev->is_mus_type ("phrasing-slur-event"))
53     {
54       /*
55         Let's not start more than one phrasing slur per moment.
56       */
57       
58       Direction d = to_dir (ev->get_property ("span-direction"));
59           
60       if (d == START)
61         {
62           if (now_mom () > last_start_)
63             {
64               new_phrasing_slur_evs_.push (ev);
65               last_start_ = now_mom ();
66               return true;
67             }
68         }
69       else
70         {
71           new_phrasing_slur_evs_.push (ev);
72           return true;
73         }
74     }
75   return false;
76 }
77
78 void
79 Phrasing_slur_engraver::acknowledge_grob (Grob_info info)
80 {
81   if (Note_column::has_interface (info.grob_))
82     {
83       Grob *e =info.grob_;
84       for (int i = 0; i < phrasing_slurs_.size (); i++)
85         Slur::add_column (phrasing_slurs_[i], e);
86       for (int i = 0; i < end_phrasing_slurs_.size (); i++)
87         Slur::add_column (end_phrasing_slurs_[i], e);
88     }
89 }
90
91 void
92 Phrasing_slur_engraver::finalize ()
93 {
94   for (int i = 0; i < phrasing_slurs_.size (); i++)
95     {
96       /*
97         Let's not typeset unterminated stuff
98       */
99       phrasing_slurs_[i]->suicide ();
100     }
101   phrasing_slurs_.clear ();
102
103   for (int i=0; i < eventses_.size (); i++)
104     {
105       eventses_[i]->origin ()->warning (_ ("unterminated phrasing slur"));
106     }
107 }
108
109 void
110 Phrasing_slur_engraver::process_acknowledged_grobs ()
111 {
112   Link_array<Grob> start_phrasing_slurs;
113   for (int i=0; i< new_phrasing_slur_evs_.size (); i++)
114     {
115       Music* phrasing_slur_ev = new_phrasing_slur_evs_[i];
116       // end phrasing slur: move the phrasing slur to other array
117
118       Direction d = to_dir (phrasing_slur_ev->get_property ("span-direction"));
119       
120       if (d == STOP)
121         {
122           if (phrasing_slurs_.is_empty ())
123             phrasing_slur_ev->origin ()->warning (_f ("can't find start of phrasing slur"));
124           else
125             {
126               Grob* phrasing_slur = phrasing_slurs_.pop ();
127               end_phrasing_slurs_.push (phrasing_slur);
128               eventses_.pop ();
129             }
130         }
131       else if (d == START)
132         {
133           // push a new phrasing_slur onto stack.
134           // (use temp. array to wait for all phrasing_slur STOPs)
135           Grob* phrasing_slur = make_spanner ("PhrasingSlur");
136           Slur::set_interface (phrasing_slur); // can't remove.
137
138
139           if (Direction updown = to_dir (phrasing_slur_ev->get_property ("direction")))
140             {
141               phrasing_slur->set_property ("direction", scm_int2num (updown));
142             }
143
144           start_phrasing_slurs.push (phrasing_slur);
145           eventses_.push (phrasing_slur_ev);
146           announce_grob (phrasing_slur, phrasing_slur_ev->self_scm ());
147         }
148     }
149   for (int i=0; i < start_phrasing_slurs.size (); i++)
150     phrasing_slurs_.push (start_phrasing_slurs[i]);
151   new_phrasing_slur_evs_.clear ();
152 }
153
154 void
155 Phrasing_slur_engraver::stop_translation_timestep ()
156 {
157   for (int i = 0; i < end_phrasing_slurs_.size (); i++)
158     {
159       typeset_grob (end_phrasing_slurs_[i]);
160     }
161   end_phrasing_slurs_.clear ();
162 }
163
164 void
165 Phrasing_slur_engraver::start_translation_timestep ()
166 {
167   new_phrasing_slur_evs_.clear ();
168 }
169
170
171
172 ENTER_DESCRIPTION (Phrasing_slur_engraver,
173 /* descr */       "Print phrasing slurs. Similar to @ref{Slur_engraver}",
174 /* creats*/       "PhrasingSlur",
175 /* accepts */     "phrasing-slur-event",
176 /* acks  */       "note-column-interface",
177 /* reads */       "",
178 /* write */       "");