]> git.donarmstrong.com Git - lilypond.git/blob - lily/rhythmic-column-engraver.cc
(process_acknowledged_grobs):
[lilypond.git] / lily / rhythmic-column-engraver.cc
1 /*
2   rhythmic-column-grav.cc -- implement Rhythmic_column_engraver
3
4   source file of the GNU LilyPond music typesetter
5
6   (c)  1997--2003 Han-Wen Nienhuys <hanwen@cs.uu.nl>
7 */
8
9
10 #include "slur.hh"
11 #include "engraver.hh"
12 #include "rhythmic-head.hh"
13 #include "stem.hh"
14 #include "note-column.hh"
15 #include "dot-column.hh"
16 #include "event.hh"
17 #include "item.hh"
18 #include "group-interface.hh"
19
20
21
22 /*
23   this engraver  glues together stems, rests and note heads into a NoteColumn
24   grob.
25
26   It also generates spacing objects.  Originally, we have tried to
27   have the spacing functionality at different levels.
28   
29   - by simply using the sequence of Separation-item as
30   spacing-sequences (at staff level). Unfortunately, this fucks up if
31   there are different kinds of tuplets in different voices (8th and
32   8ths triplets combined made the program believe there were 1/12 th
33   notes.).
34
35   Doing it in a separate engraver using timing info is generally
36   complicated (start/end time management), and fucks up if a voice
37   changes staff.
38
39   Now we do it from here again. This has the problem that voices can
40   appear and disappear at will, leaving lots of loose ends (the note
41   spacing engraver don't know where to connect the last note of the
42   voice on the right with), but we don't complain about those, and let
43   the default spacing do its work.
44
45  */
46
47
48 class Rhythmic_column_engraver :public Engraver
49 {
50   Link_array<Grob> rheads_;
51   Grob * stem_;
52   Grob * note_column_;
53   Grob * dotcol_;
54
55   Grob * last_spacing_;
56   Grob * spacing_;
57   
58   TRANSLATOR_DECLARATIONS(Rhythmic_column_engraver);
59 protected:
60
61   virtual void acknowledge_grob (Grob_info);
62   virtual void process_acknowledged_grobs ();
63   virtual void stop_translation_timestep ();
64   virtual void start_translation_timestep ();
65 };
66
67
68
69 Rhythmic_column_engraver::Rhythmic_column_engraver ()
70 {
71   spacing_ =0 ;
72   last_spacing_ = 0;
73   
74   stem_ =0;
75   note_column_=0;
76   dotcol_ =0;
77 }
78
79
80 void
81 Rhythmic_column_engraver::process_acknowledged_grobs ()
82 {
83   if (rheads_.size ())
84     {
85       if (!note_column_)
86         {
87           note_column_ = new Item (get_property ("NoteColumn"));
88           announce_grob(note_column_, rheads_[0]->self_scm ());
89
90           spacing_ = new Item (get_property ("NoteSpacing"));
91           spacing_->set_grob_property ("left-items", gh_cons (note_column_->self_scm (), SCM_EOL));
92
93
94           announce_grob (spacing_, SCM_EOL);
95
96           if (last_spacing_)
97             {
98               Pointer_group_interface::add_grob (last_spacing_,
99                                                  ly_symbol2scm ("right-items" ),
100                                                  note_column_);
101             }
102
103         }
104
105       for (int i=0; i < rheads_.size (); i++)
106         {
107           if (!rheads_[i]->get_parent (X_AXIS))
108             Note_column::add_head (note_column_, rheads_[i]);
109         }
110       rheads_.set_size (0);
111     }
112
113   
114   if (note_column_)
115     {
116       if (dotcol_
117           && !dotcol_->get_parent (X_AXIS))
118         {
119           Note_column::set_dotcol (note_column_, dotcol_);
120         }
121
122       if (stem_
123           && !stem_->get_parent (X_AXIS))
124         {
125           Note_column::set_stem (note_column_, stem_);
126           stem_ = 0;
127         }
128
129     }
130 }
131
132 void
133 Rhythmic_column_engraver::acknowledge_grob (Grob_info i)
134 {
135   Item * item =  dynamic_cast <Item *> (i.grob_);
136   if (!item || item->get_parent (X_AXIS))
137     return ; 
138   if (Stem::has_interface (item))
139     {
140       stem_ = item;
141     }
142   else if (Rhythmic_head::has_interface (item))
143     {
144       rheads_.push (item);
145     }
146   else if (Dot_column::has_interface (item))
147     {
148       dotcol_ = item;
149     }
150 }
151
152 void
153 Rhythmic_column_engraver::stop_translation_timestep ()
154 {
155   if (note_column_) 
156     {
157       typeset_grob (note_column_);
158       note_column_ =0;
159     }
160
161   if (spacing_)
162     {
163       typeset_grob (spacing_);
164       last_spacing_ = spacing_;
165       spacing_ =0;
166     }
167 }
168
169 void
170 Rhythmic_column_engraver::start_translation_timestep ()
171 {
172   dotcol_ =0;
173   stem_ =0;
174 }
175
176
177
178 ENTER_DESCRIPTION(Rhythmic_column_engraver,
179 /* descr */       "Generates NoteColumn, an objects that groups stems, noteheads and rests.",
180 /* creats*/       "NoteColumn NoteSpacing",
181 /* accepts */     "",
182 /* acks  */      "stem-interface rhythmic-head-interface dot-column-interface",
183 /* reads */       "",
184 /* write */       "");