X-Git-Url: https://git.donarmstrong.com/?a=blobdiff_plain;f=lily%2Fgrob-pq-engraver.cc;h=e4e8b944f968aefd5647185bc9f2ef21c87a560f;hb=fa938d4b51ccb3c18e489b395838f2587b531ec2;hp=2cd150969328cf40b44ac9a964ed99ee908f7974;hpb=ba858880848d6aca1de4401d185860eb2017a01c;p=lilypond.git diff --git a/lily/grob-pq-engraver.cc b/lily/grob-pq-engraver.cc index 2cd1509693..e4e8b944f9 100644 --- a/lily/grob-pq-engraver.cc +++ b/lily/grob-pq-engraver.cc @@ -1,9 +1,9 @@ -/* - grob-pq-engraver.cc -- implement Grob_pq_engraver +/* + grob-pq-engraver.cc -- implement Grob_pq_engraver source file of the GNU LilyPond music typesetter - (c) 2001--2004 Han-Wen Nienhuys + (c) 2001--2006 Han-Wen Nienhuys */ #include "context.hh" @@ -11,17 +11,29 @@ #include "grob.hh" #include "warn.hh" -class Grob_pq_engraver: public Engraver +struct Grob_pq_entry +{ + Grob *grob_; + Moment end_; + static int compare (Grob_pq_entry const &a, + Grob_pq_entry const &b) + { + return Moment::compare (a.end_, b.end_); + } +}; + +class Grob_pq_engraver : public Engraver { public: TRANSLATOR_DECLARATIONS (Grob_pq_engraver); protected: virtual void initialize (); - virtual void acknowledge_grob (Grob_info); - virtual void start_translation_timestep (); - virtual void stop_translation_timestep (); -}; + DECLARE_ACKNOWLEDGER (grob); + void start_translation_timestep (); + void stop_translation_timestep (); + vector started_now_; +}; Grob_pq_engraver::Grob_pq_engraver () { @@ -30,34 +42,34 @@ Grob_pq_engraver::Grob_pq_engraver () void Grob_pq_engraver::initialize () { - get_parent_context ()->set_property ("busyGrobs", SCM_EOL); + context ()->set_property ("busyGrobs", SCM_EOL); } LY_DEFINE (ly_grob_pq_less_p, "ly:grob-pq-less?", - 2, 0 ,0, (SCM a, SCM b), - "Compare 2 grob priority queue entries. Internal") + 2, 0, 0, (SCM a, SCM b), + "Compare 2 grob priority queue entries. Internal") { - if (Moment::compare (*unsmob_moment (ly_car (a)), - *unsmob_moment (ly_car (b))) < 0) + if (Moment::compare (*unsmob_moment (scm_car (a)), + *unsmob_moment (scm_car (b))) < 0) return SCM_BOOL_T; else return SCM_BOOL_F; } - + void Grob_pq_engraver::acknowledge_grob (Grob_info gi) { - Music * m = gi.music_cause (); + Music *m = gi.music_cause (); if (m - && !gi.grob_->internal_has_interface (ly_symbol2scm ("multi-measure-interface"))) + && !gi.grob ()->internal_has_interface (ly_symbol2scm ("multi-measure-interface"))) { Moment n = now_mom (); Moment l = m->get_length (); if (!l.to_bool ()) - return ; - + return; + if (n.grace_part_) { l.grace_part_ = l.main_part_; @@ -65,30 +77,39 @@ Grob_pq_engraver::acknowledge_grob (Grob_info gi) } Moment end = n + l; - SCM lst = scm_acons (end.smobbed_copy (), - gi.grob_->self_scm (), - SCM_EOL); - SCM busy= get_property ("busyGrobs"); - busy = scm_merge_x (lst, busy, ly_grob_pq_less_p_proc); - get_parent_context ()->set_property ("busyGrobs", busy); + Grob_pq_entry e; + e.grob_ = gi.grob (); + e.end_ = end; + + started_now_.push_back (e); } } - void Grob_pq_engraver::stop_translation_timestep () { Moment now = now_mom (); SCM start_busy = get_property ("busyGrobs"); SCM busy = start_busy; - while (ly_c_pair_p (busy) && *unsmob_moment (ly_caar (busy)) == now) + while (scm_is_pair (busy) && *unsmob_moment (scm_caar (busy)) == now) + busy = scm_cdr (busy); + + vector_sort (started_now_, Grob_pq_entry::compare); + SCM lst = SCM_EOL; + SCM *tail = &lst; + for (vsize i = 0; i < started_now_.size (); i++) { - busy = ly_cdr (busy); + *tail = scm_acons (started_now_[i].end_.smobbed_copy (), + started_now_[i].grob_->self_scm (), + SCM_EOL); + tail = SCM_CDRLOC (*tail); } - if (start_busy != busy) - get_parent_context ()->set_property ("busyGrobs", busy); + busy = scm_merge_x (lst, busy, ly_grob_pq_less_p_proc); + context ()->set_property ("busyGrobs", busy); + + started_now_.clear (); } void @@ -98,29 +119,25 @@ Grob_pq_engraver::start_translation_timestep () SCM start_busy = get_property ("busyGrobs"); SCM busy = start_busy; - while (ly_c_pair_p (busy) && *unsmob_moment (ly_caar (busy)) < now) + while (scm_is_pair (busy) && *unsmob_moment (scm_caar (busy)) < now) { /* The grob-pq-engraver is not water tight, and stuff like tupletSpannerDuration confuses it. - */ - busy = ly_cdr (busy); + */ + busy = scm_cdr (busy); } if (start_busy != busy) - get_parent_context ()->set_property ("busyGrobs", busy); + context ()->set_property ("busyGrobs", busy); } +#include "translator.icc" +ADD_ACKNOWLEDGER (Grob_pq_engraver, grob); +ADD_TRANSLATOR (Grob_pq_engraver, -ENTER_DESCRIPTION (Grob_pq_engraver, - -/* descr */ "Administrate when certain grobs (eg. note heads) stop playing; this \ -engraver is a sort-of a failure, since it doesn't handle all sorts of \ -borderline cases very well. \ -", \ - -/* creats*/ "", \ -/* accepts */ "", \ -/* acks */ "grob-interface", \ -/* reads */ "busyGrobs", \ -/* write */ "busyGrobs"); + /* doc */ "Administrate when certain grobs (eg. note heads) stop playing", + /* create */ "", + /* accept */ "", + /* read */ "busyGrobs", + /* write */ "busyGrobs");