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