Fixes Issue 1504, allowing feather beam line breaking.
[lilypond/patrick.git] / lily / tuplet-iterator.cc
blob97737db0f2b8582f19f24c6213119886d20a57b6
1 /*
2 This file is part of LilyPond, the GNU music typesetter.
4 Copyright (C) 1998--2011 Han-Wen Nienhuys <hanwen@xs4all.nl>,
5 Erik Sandberg <mandolaerik@gmail.com>
7 LilyPond is free software: you can redistribute it and/or modify
8 it under the terms of the GNU General Public License as published by
9 the Free Software Foundation, either version 3 of the License, or
10 (at your option) any later version.
12 LilyPond is distributed in the hope that it will be useful,
13 but WITHOUT ANY WARRANTY; without even the implied warranty of
14 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
15 GNU General Public License for more details.
17 You should have received a copy of the GNU General Public License
18 along with LilyPond. If not, see <http://www.gnu.org/licenses/>.
21 #include "context.hh"
22 #include "input.hh"
23 #include "international.hh"
24 #include "music.hh"
25 #include "music-wrapper-iterator.hh"
26 #include "stream-event.hh"
29 Iterates \times, by sending TupletSpanEvents at the start/end of each
30 tuplet bracket. Extra stop/start events are sent at regular
31 intervals if tupletSpannerDuration is set.
33 class Tuplet_iterator : public Music_wrapper_iterator
35 public:
36 DECLARE_SCHEME_CALLBACK (constructor, ());
37 /* construction */
38 DECLARE_CLASSNAME (Tuplet_iterator);
39 Tuplet_iterator ();
40 protected:
41 virtual void process (Moment m);
42 virtual void construct_children ();
43 virtual void derived_mark () const;
44 virtual Moment pending_moment () const;
46 Music *create_event (Direction d);
48 private:
50 /* tupletSpannerDuration */
51 Moment spanner_duration_;
53 /* next time to add a stop/start pair */
54 Moment next_split_mom_;
56 /* Recycle start/stop events if tupletSpannerDuration is set. */
57 SCM synthesized_events_;
59 Context_handle tuplet_handler_;
62 Music *
63 Tuplet_iterator::create_event (Direction d)
65 SCM ev_scm = scm_call_2 (ly_lily_module_constant ("make-span-event"),
66 ly_symbol2scm ("TupletSpanEvent"),
67 scm_from_int (d));
69 Music *mus = get_music ();
71 Music *ev = unsmob_music (ev_scm);
72 ev->set_spot (*mus->origin ());
73 if (d == START)
75 ev->set_property ("numerator", mus->get_property ("numerator"));
76 ev->set_property ("denominator", mus->get_property ("denominator"));
77 ev->set_property ("tweaks", mus->get_property ("tweaks"));
78 ev->set_property ("length", spanner_duration_.smobbed_copy ());
81 synthesized_events_ = scm_cons (ev_scm, synthesized_events_);
82 return ev;
85 Tuplet_iterator::Tuplet_iterator ()
87 spanner_duration_ = next_split_mom_ = 0;
88 synthesized_events_ = SCM_EOL;
91 Moment
92 Tuplet_iterator::pending_moment () const
94 Moment next_mom = Music_wrapper_iterator::pending_moment ();
95 next_mom = min (next_mom, next_split_mom_);
97 return next_mom;
100 void
101 Tuplet_iterator::process (Moment m)
103 if (spanner_duration_.to_bool ()
104 && m.main_part_ == next_split_mom_)
106 descend_to_bottom_context ();
107 if (tuplet_handler_.get_outlet ())
108 create_event (STOP)->send_to_context (tuplet_handler_.get_outlet ());
110 if (m.main_part_ < music_get_length ().main_part_)
112 tuplet_handler_.set_context (get_outlet ());
113 report_event (create_event (START));
115 next_split_mom_ += spanner_duration_;
117 else
118 tuplet_handler_.set_context (0);
120 Music_wrapper_iterator::process (m);
121 if (child_iter_ && child_iter_->ok ())
122 descend_to_child (child_iter_->get_outlet ());
126 void
127 Tuplet_iterator::construct_children ()
129 spanner_duration_ = music_get_length ();
131 Moment *mp
132 = unsmob_moment (get_outlet ()->get_property ("tupletSpannerDuration"));
133 if (mp)
134 spanner_duration_ = min (mp->main_part_, spanner_duration_);
136 Music_wrapper_iterator::construct_children ();
138 if (child_iter_ && child_iter_->ok ())
139 descend_to_child (child_iter_->get_outlet ());
142 void
143 Tuplet_iterator::derived_mark () const
145 scm_gc_mark (synthesized_events_);
146 Music_wrapper_iterator::derived_mark ();
149 IMPLEMENT_CTOR_CALLBACK (Tuplet_iterator);