]> git.donarmstrong.com Git - lilypond.git/blob - lily/text-spanner-engraver.cc
Fix some bugs in the dynamic engraver and PostScript backend
[lilypond.git] / lily / text-spanner-engraver.cc
1 /*
2   text-spanner-engraver.cc -- implement Text_spanner_engraver
3
4   source file of the GNU LilyPond music typesetter
5
6   (c) 2000--2006 Jan Nieuwenhuizen <janneke@gnu.org>
7 */
8
9 #include "engraver.hh"
10
11 #include "international.hh"
12 #include "note-column.hh"
13 #include "side-position-interface.hh"
14
15 class Text_spanner_engraver : public Engraver
16 {
17 public:
18   TRANSLATOR_DECLARATIONS (Text_spanner_engraver);
19 protected:
20   virtual void finalize ();
21   DECLARE_ACKNOWLEDGER (note_column);
22   virtual bool try_music (Music *);
23   void stop_translation_timestep ();
24   void process_music ();
25
26 private:
27   Spanner *span_;
28   Spanner *finished_;
29   Music *current_event_;
30   Drul_array<Music *> event_drul_;
31   void typeset_all ();
32 };
33
34 Text_spanner_engraver::Text_spanner_engraver ()
35 {
36   finished_ = 0;
37   current_event_ = 0;
38   span_ = 0;
39   event_drul_[START] = 0;
40   event_drul_[STOP] = 0;
41 }
42
43 bool
44 Text_spanner_engraver::try_music (Music *m)
45 {
46   if (m->is_mus_type ("text-span-event"))
47     {
48       Direction d = to_dir (m->get_property ("span-direction"));
49       event_drul_[d] = m;
50       return true;
51     }
52
53   return false;
54 }
55
56 void
57 Text_spanner_engraver::process_music ()
58 {
59   if (event_drul_[STOP])
60     {
61       if (!span_)
62         event_drul_[STOP]->origin ()->warning (_ ("can't find start of text spanner"));
63       else
64         {
65           finished_ = span_;
66           span_ = 0;
67           current_event_ = 0;
68         }
69     }
70
71   if (event_drul_[START])
72     {
73       if (current_event_)
74         event_drul_[START]->origin ()->warning (_ ("already have a text spanner"));
75       else
76         {
77           current_event_ = event_drul_[START];
78           span_ = make_spanner ("TextSpanner", event_drul_[START]->self_scm ());
79
80           Side_position_interface::set_axis (span_, Y_AXIS);
81           event_drul_[START] = 0;
82         }
83     }
84 }
85
86 void
87 Text_spanner_engraver::acknowledge_note_column (Grob_info info)
88 {
89   Spanner *spans[2] ={span_, finished_};
90   for (int i = 0; i < 2; i++)
91     {
92       if (spans[i])
93         {
94           Side_position_interface::add_support (spans[i], info.grob ());
95           add_bound_item (spans[i], dynamic_cast<Item *> (info.grob ()));
96         }
97     }
98 }
99
100 void
101 Text_spanner_engraver::typeset_all ()
102 {
103   if (finished_)
104     {
105       if (!finished_->get_bound (RIGHT))
106         {
107           Grob *e = unsmob_grob (get_property ("currentMusicalColumn"));
108           finished_->set_bound (RIGHT, e);
109         }
110       finished_ = 0;
111     }
112 }
113
114 void
115 Text_spanner_engraver::stop_translation_timestep ()
116 {
117   if (span_ && !span_->get_bound (LEFT))
118     {
119       Grob *e = unsmob_grob (get_property ("currentMusicalColumn"));
120       span_->set_bound (LEFT, e);
121     }
122
123   typeset_all ();
124   event_drul_[START] = 0;
125   event_drul_[STOP] = 0;
126 }
127
128 void
129 Text_spanner_engraver::finalize ()
130 {
131   typeset_all ();
132   if (span_)
133     {
134       current_event_->origin ()->warning (_ ("unterminated text spanner"));
135       span_->suicide ();
136       span_ = 0;
137     }
138 }
139
140 #include "translator.icc"
141 ADD_ACKNOWLEDGER (Text_spanner_engraver, note_column);
142 ADD_TRANSLATOR (Text_spanner_engraver,
143                 /* doc */ "Create text spanner from a Music.",
144                 /* create */ "TextSpanner",
145                 /* accept */ "text-span-event",
146                 /* read */ "",
147                 /* write */ "");