]> git.donarmstrong.com Git - lilypond.git/blob - lily/extender-engraver.cc
Fix off-by-one error in constrained-breaking.
[lilypond.git] / lily / extender-engraver.cc
1 /*
2   extender-engraver.cc -- implement Extender_engraver
3
4   source file of the GNU LilyPond music typesetter
5
6   (c) 1999--2006 Glen Prideaux <glenprideaux@iname.com>,
7   Han-Wen Nienhuys <hanwen@xs4all.nl>,
8   Jan Nieuwenhuizen <janneke@gnu.org>
9 */
10
11 #include "context.hh"
12 #include "engraver.hh"
13 #include "international.hh"
14 #include "item.hh"
15 #include "lyric-extender.hh"
16 #include "note-head.hh"
17 #include "pointer-group-interface.hh"
18 #include "stream-event.hh"
19 #include "warn.hh"
20 #include "spanner.hh"
21
22 #include "translator.icc"
23
24 void completize_extender (Spanner *sp);
25
26 class Extender_engraver : public Engraver
27 {
28   Stream_event *ev_;
29   Spanner *extender_;
30   Spanner *pending_extender_;
31
32 public:
33   TRANSLATOR_DECLARATIONS (Extender_engraver);
34
35 protected:
36   DECLARE_TRANSLATOR_LISTENER (extender);
37   DECLARE_ACKNOWLEDGER (lyric_syllable);
38   virtual void finalize ();
39   void stop_translation_timestep ();
40   void process_music ();
41 };
42
43 Extender_engraver::Extender_engraver ()
44 {
45   extender_ = 0;
46   pending_extender_ = 0;
47   ev_ = 0;
48 }
49
50 IMPLEMENT_TRANSLATOR_LISTENER (Extender_engraver, extender);
51 void
52 Extender_engraver::listen_extender (Stream_event *ev)
53 {
54   ASSIGN_EVENT_ONCE (ev_, ev);
55 }
56
57 void
58 Extender_engraver::process_music ()
59 {
60   if (ev_)
61     extender_ = make_spanner ("LyricExtender", ev_->self_scm ());
62 }
63
64 void
65 Extender_engraver::acknowledge_lyric_syllable (Grob_info i)
66 {
67   Item *item = i.item ();
68   if (extender_)
69     extender_->set_bound (LEFT, item);
70
71   if (pending_extender_)
72     {
73       pending_extender_->set_object ("next", item->self_scm ());
74       completize_extender (pending_extender_);
75       pending_extender_ = 0;
76     }
77 }
78
79 void
80 Extender_engraver::stop_translation_timestep ()
81 {
82   if (extender_ || pending_extender_)
83     {
84       Context *voice = get_voice_to_lyrics (context ());
85       Grob *h = voice ? get_current_note_head (voice) : 0;
86
87       if (h)
88         {
89           if (extender_)
90             {
91               Pointer_group_interface::add_grob (extender_,
92                                                  ly_symbol2scm ("heads"), h);
93             }
94
95           if (pending_extender_)
96             {
97               Pointer_group_interface::add_grob (pending_extender_,
98                                                  ly_symbol2scm ("heads"), h);
99             }
100         }
101
102       if (extender_)
103         {
104           pending_extender_ = extender_;
105           extender_ = 0;
106         }
107     }
108
109   ev_ = 0;
110 }
111
112 void
113 completize_extender (Spanner *sp)
114 {
115   if (!sp->get_bound (RIGHT))
116     {
117       extract_item_set (sp, "heads", heads);
118       if (heads.size ())
119         sp->set_bound (RIGHT, heads.back ());
120     }
121 }
122
123 void
124 Extender_engraver::finalize ()
125 {
126   if (extender_)
127     {
128       completize_extender (extender_);
129
130       if (!extender_->get_bound (RIGHT))
131         extender_->warning (_ ("unterminated extender"));
132       extender_ = 0;
133     }
134
135   if (pending_extender_)
136     {
137       completize_extender (pending_extender_);
138
139       if (!pending_extender_->get_bound (RIGHT))
140         pending_extender_->warning (_ ("unterminated extender"));
141       pending_extender_ = 0;
142     }
143 }
144
145 ADD_ACKNOWLEDGER (Extender_engraver, lyric_syllable);
146 ADD_TRANSLATOR (Extender_engraver,
147                 /* doc */ "Create lyric extenders",
148                 /* create */ "LyricExtender",
149                 /* read */ "",
150                 /* write */ "");