Nitpick: ly:spanner-bound grob name slur -> spanner.
[lilypond.git] / lily / key-engraver.cc
blobc396d73351e080aa6eb8c78f8142a64312418a36
1 /*
2 key-engraver.cc -- implement Key_engraver
4 source file of the GNU LilyPond music typesetter
6 (c) 1997--2009 Han-Wen Nienhuys <hanwen@xs4all.nl>
7 */
9 #include "bar-line.hh"
10 #include "clef.hh"
11 #include "context.hh"
12 #include "engraver.hh"
13 #include "item.hh"
14 #include "pitch.hh"
15 #include "protected-scm.hh"
16 #include "staff-symbol-referencer.hh"
17 #include "stream-event.hh"
19 #include "translator.icc"
21 class Key_engraver : public Engraver
23 void create_key (bool);
24 void read_event (Stream_event const *r);
26 Stream_event *key_event_;
27 Item *item_;
28 Item *cancellation_;
29 public:
30 TRANSLATOR_DECLARATIONS (Key_engraver);
32 protected:
33 virtual void initialize ();
34 virtual void finalize ();
35 void stop_translation_timestep ();
36 void process_music ();
38 DECLARE_TRANSLATOR_LISTENER (key_change);
39 DECLARE_ACKNOWLEDGER (clef);
40 DECLARE_ACKNOWLEDGER (bar_line);
43 void
44 Key_engraver::finalize ()
48 Key_engraver::Key_engraver ()
50 key_event_ = 0;
51 item_ = 0;
52 cancellation_ = 0;
56 void
57 Key_engraver::create_key (bool is_default)
59 if (!item_)
61 item_ = make_item ("KeySignature",
62 key_event_ ? key_event_->self_scm () : SCM_EOL);
64 item_->set_property ("c0-position",
65 get_property ("middleCPosition"));
67 SCM last = get_property ("lastKeySignature");
68 SCM key = get_property ("keySignature");
69 bool extranatural = to_boolean (get_property ("extraNatural"));
71 if ((to_boolean (get_property ("printKeyCancellation"))
72 || key == SCM_EOL)
73 && !scm_is_eq (last, key))
75 SCM restore = SCM_EOL;
76 SCM *tail = &restore;
77 for (SCM s = last; scm_is_pair (s); s = scm_cdr (s))
79 SCM new_alter_pair = scm_assoc (scm_caar (s), key);
80 Rational old_alter = robust_scm2rational (scm_cdar (s), 0);
81 if (new_alter_pair == SCM_BOOL_F
82 || (extranatural
83 && (ly_scm2rational (scm_cdr (new_alter_pair)) - old_alter)*old_alter
84 < Rational (0)))
86 *tail = scm_cons (scm_car (s), *tail);
87 tail = SCM_CDRLOC (*tail);
91 if (scm_is_pair (restore))
93 cancellation_ = make_item ("KeyCancellation",
94 key_event_
95 ? key_event_->self_scm () : SCM_EOL);
97 cancellation_->set_property ("alteration-alist", scm_reverse (restore));
98 cancellation_->set_property ("c0-position",
99 get_property ("middleCPosition"));
103 item_->set_property ("alteration-alist", scm_reverse (key));
106 if (!is_default)
108 SCM visibility = get_property ("explicitKeySignatureVisibility");
109 item_->set_property ("break-visibility", visibility);
113 IMPLEMENT_TRANSLATOR_LISTENER (Key_engraver, key_change);
114 void
115 Key_engraver::listen_key_change (Stream_event *ev)
117 /* do this only once, just to be on the safe side. */
118 if (ASSIGN_EVENT_ONCE (key_event_, ev))
119 read_event (key_event_);
122 void
123 Key_engraver::acknowledge_clef (Grob_info /* info */)
125 SCM c = get_property ("createKeyOnClefChange");
126 if (to_boolean (c))
127 create_key (false);
130 void
131 Key_engraver::acknowledge_bar_line (Grob_info /* info */)
133 if (scm_is_pair (get_property ("keySignature")))
134 create_key (true);
137 void
138 Key_engraver::process_music ()
140 if (key_event_
141 || get_property ("lastKeySignature") != get_property ("keySignature"))
142 create_key (false);
145 void
146 Key_engraver::stop_translation_timestep ()
148 item_ = 0;
149 context ()->set_property ("lastKeySignature", get_property ("keySignature"));
150 cancellation_ = 0;
151 key_event_ = 0;
154 void
155 Key_engraver::read_event (Stream_event const *r)
157 SCM p = r->get_property ("pitch-alist");
158 if (!scm_is_pair (p))
159 return;
161 SCM accs = SCM_EOL;
163 SCM alist = scm_list_copy (p);
164 SCM order = get_property ("keyAlterationOrder");
165 for (SCM s = order;
166 scm_is_pair (s) && scm_is_pair (alist); s = scm_cdr (s))
168 SCM head = scm_member (scm_car (s), alist);
170 if (scm_is_pair (head))
172 accs = scm_cons (scm_car (head), accs);
173 alist = scm_delete_x (scm_car (head), alist);
177 if (scm_is_pair (alist))
179 bool warn = false;
180 for (SCM s = alist; scm_is_pair (s); s = scm_cdr (s))
181 if (ly_scm2rational (scm_cdar (s)))
183 warn = true;
184 accs = scm_cons (scm_car (s), accs);
187 if (warn)
188 r->origin ()->warning ("No ordering for key signature alterations");
191 context ()->set_property ("keySignature", scm_reverse (accs));
192 context ()->set_property ("tonic",
193 r->get_property ("tonic"));
196 void
197 Key_engraver::initialize ()
199 context ()->set_property ("keySignature", SCM_EOL);
200 context ()->set_property ("lastKeySignature", SCM_EOL);
202 Pitch p (0, 0, 0);
203 context ()->set_property ("tonic", p.smobbed_copy ());
206 ADD_ACKNOWLEDGER (Key_engraver, clef);
207 ADD_ACKNOWLEDGER (Key_engraver, bar_line);
209 ADD_TRANSLATOR (Key_engraver,
210 /* doc */
211 "Engrave a key signature.",
213 /* create */
214 "KeyCancellation "
215 "KeySignature ",
217 /* read */
218 "createKeyOnClefChange "
219 "explicitKeySignatureVisibility "
220 "extraNatural "
221 "keyAlterationOrder "
222 "keySignature "
223 "lastKeySignature "
224 "printKeyCancellation ",
226 /* write */
227 "keySignature "
228 "lastKeySignature "
229 "tonic "