]> git.donarmstrong.com Git - lilypond.git/blob - lily/fingering-engraver.cc
(Paper_column): copy rank_. This fixes
[lilypond.git] / lily / fingering-engraver.cc
1 /*   
2   fingering-engraver.cc --  implement Fingering_engraver
3   
4   source file of the GNU LilyPond music typesetter
5   
6   (c) 1998--2004 Han-Wen Nienhuys <hanwen@cs.uu.nl>
7   
8  */
9
10 #include "engraver.hh"
11 #include "side-position-interface.hh"
12 #include "stem.hh"
13 #include "rhythmic-head.hh"
14 #include "self-alignment-interface.hh"
15
16 class Fingering_engraver : public Engraver
17 {
18   Link_array<Music> reqs_;
19   Link_array<Item> fingerings_;
20
21 public:
22   TRANSLATOR_DECLARATIONS (Fingering_engraver);
23 protected:
24   virtual bool try_music (Music* m);
25   virtual void stop_translation_timestep ();
26   virtual void start_translation_timestep ();
27   virtual void process_music ();
28   virtual void acknowledge_grob (Grob_info);
29
30 private:
31   void make_script (Direction, Music*, int);
32 };
33
34 bool
35 Fingering_engraver::try_music (Music *m)
36 {
37   if (m->is_mus_type ("fingering-event"))
38     {
39       reqs_.push (m);
40       return true;
41     }
42   return false;
43 }
44
45 void
46 Fingering_engraver::acknowledge_grob (Grob_info inf)
47 {
48   if (Stem::has_interface (inf.grob_))
49     {
50       for (int i= 0; i < fingerings_.size (); i++)
51         {
52           Side_position_interface::add_support (fingerings_[i], inf.grob_);
53         }
54     }
55   else if (Rhythmic_head::has_interface (inf.grob_))
56     {
57       for (int i= 0; i < fingerings_.size (); i++)
58         {
59           Grob*t = fingerings_[i];
60           Side_position_interface::add_support (t,inf.grob_);
61           if (!t->get_parent (X_AXIS))
62             t->set_parent (inf.grob_, X_AXIS);
63         }
64     }
65 }
66
67 void
68 Fingering_engraver::process_music ()
69 {
70   for (int i= reqs_.size (); i--;)
71     {
72       SCM dir = reqs_[i]->get_property ("direction");
73       make_script (to_dir (dir), reqs_[i], i);
74     }
75 }
76
77
78 void
79 Fingering_engraver::make_script (Direction d, Music *r, int i)
80 {
81   Item *fingering = make_item ("Fingering", r->self_scm ());
82   Axis a = Y_AXIS;
83   Axis other = other_axis (a);
84
85   /*
86     Huh, what's this for? --hwn.
87
88     junkme.
89    */
90   SCM pitch = r->get_property ("pitch");
91   if (unsmob_pitch (pitch))
92     fingering->set_property ("pitch", pitch);
93
94   /*
95     We can't fold these definitions into define-grobs since
96     fingerings for chords need different settings. 
97   */
98   Side_position_interface::set_axis (fingering, a);
99   fingering->add_offset_callback (Self_alignment_interface::aligned_on_self_proc, other);
100   fingering->add_offset_callback (Self_alignment_interface::centered_on_parent_proc, other);
101
102   // Hmm
103   int priority = 200;
104   SCM s = fingering->get_property ("script-priority");
105   if (scm_is_number (s))
106     priority = scm_to_int (s);
107   
108   /* See script-engraver.cc */
109   priority += i;
110
111   fingering->set_property ("script-priority", scm_int2num (priority));
112
113
114   if (!is_direction (fingering->get_property ("direction")))
115     {
116       if (d)
117         fingering->set_property ("direction", scm_int2num (d));
118       else
119         fingering->set_property ("direction",  scm_int2num (RIGHT));
120     }
121
122   SCM dig =  r->get_property ("digit");
123   fingering->set_property ("text", scm_number_to_string (dig, scm_int2num (10)));
124
125   fingerings_.push (fingering);
126 }
127
128 void
129 Fingering_engraver::stop_translation_timestep ()
130 {
131   if (!fingerings_.size ())
132     return;
133   
134   fingerings_.clear ();
135 }
136
137 void
138 Fingering_engraver::start_translation_timestep ()
139 {
140   reqs_.clear ();
141 }
142
143 Fingering_engraver::Fingering_engraver ()
144 {
145
146 }
147
148 ENTER_DESCRIPTION (Fingering_engraver,
149 /* descr */       "Create fingering-scripts",
150 /* creats*/       "Fingering",
151 /* accepts */     "fingering-event",
152 /* acks  */      "rhythmic-head-interface stem-interface",
153 /* reads */       "",
154 /* write */       "");