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