]> git.donarmstrong.com Git - lilypond.git/blob - lily/performance.cc
patch::: 1.3.136.jcn3
[lilypond.git] / lily / performance.cc
1 /*
2   performance.cc -- implement Performance
3
4   source file of the GNU LilyPond music typesetter
5
6   (c)  1997--2001 Jan Nieuwenhuizen <janneke@gnu.org>
7 */
8
9 #include <time.h>
10 #include "debug.hh"
11 #include "string.hh"
12 #include "string-convert.hh"
13 #include "main.hh"
14 #include "midi-def.hh"
15 #include "midi-item.hh"
16 #include "midi-stream.hh"
17 #include "audio-column.hh"
18 #include "audio-item.hh"
19 #include "audio-staff.hh"
20 #include "performance.hh"
21 #include "score.hh"
22 #include "file-results.hh"
23 #include "file-path.hh"
24 #include "lily-version.hh"
25
26 #include "killing-cons.tcc"
27
28 Performance::Performance ()
29 {
30   midi_l_ =0;
31   audio_elem_p_list_ = 0;
32 }
33
34
35 Performance::~Performance ()
36 {
37   delete audio_elem_p_list_;
38 }
39
40 void
41 Performance::output (Midi_stream& midi_stream)
42 {
43   int tracks_i = audio_staff_l_arr_.size () + 1;
44
45   // ugh
46   int clocks_per_4_i = 384;
47
48   midi_stream << Midi_header (1, tracks_i, clocks_per_4_i);
49   output_header_track (midi_stream);
50   progress_indication ("\n");
51   progress_indication (_ ("Track ... "));
52   int channel = 1;
53   for (int i =0; i < audio_staff_l_arr_.size (); i++)
54     {
55       Audio_staff *s = audio_staff_l_arr_[i];
56       if (verbose_global_b)
57         progress_indication ("[" + to_str (i)) ;
58
59       /*
60         Aargh, let's hear it for the MIDI standard.
61         MIDI players tend to ignore instrument settings on
62         channel 10, the percussion channel by default.
63        */
64       if (channel == 9)
65         channel++;
66       s->output (midi_stream, channel++);
67       if (verbose_global_b)
68         progress_indication ("]");
69     }
70 }
71
72 void
73 Performance::output_header_track (Midi_stream& midi_stream)
74 {
75   Midi_track midi_track;
76
77   // perhaps multiple text events?
78   String id_str;
79   String str = String (_ ("Creator: "));
80   if (no_timestamps_global_b)
81     id_str = gnu_lilypond_str ();
82   else
83     id_str = gnu_lilypond_version_str ();
84   str += id_str;
85   str += "\n";
86
87   /*
88     This seems silly, but in fact the audio elements should
89     be generated elsewhere: not midi-specific.
90    */
91   Audio_text creator_a (Audio_text::TEXT, str);
92   Midi_text creator (&creator_a);
93   midi_track.add (Moment (0), &creator);
94
95   /* Better not translate this */
96   str = "Generated automatically by: ";
97   str += id_str;
98   if (no_timestamps_global_b)
99     str += ".\n";
100   else
101     {
102       str += _ (", at ");
103       time_t t (time (0));
104       str += ctime (&t);
105       str = str.left_str (str.length_i () - 1);
106     }
107
108   /*
109     Pad out time stamps to 120 chars.  */
110   str = str + to_str (' ' , (120 - str.length_i ()) >? 0);
111   
112   Audio_text generate_a (Audio_text::TEXT, str);
113   Midi_text generate (&generate_a);
114   midi_track.add (Moment (0), &generate);
115
116   str = _f ("from musical definition: %s", origin_str_);
117
118   Audio_text from_a (Audio_text::TEXT, str);
119   Midi_text from (&from_a);
120   midi_track.add (Moment (0), &from);
121
122   Audio_text track_name_a (Audio_text::TRACK_NAME, "Track "
123                            + String_convert::i2dec_str (0, 0, '0'));
124   Midi_text track_name (&track_name_a);
125                         
126   midi_track.add (Moment (0), &track_name);
127
128   // Some sequencers read track 0 last.
129   //  Audio_tempo tempo_a (midi_l_->get_tempo_i (Moment (1, 4)));
130   //  Midi_tempo tempo (&tempo_a);
131   //  midi_track.add (Moment (0), &tempo);
132
133   midi_stream << midi_track;
134 }
135
136 void
137 Performance::add_element (Audio_element *p)
138 {
139   if (Audio_staff*s=dynamic_cast<Audio_staff *> (p)) 
140     {
141       audio_staff_l_arr_.push (s);
142     }
143   else if (Audio_column *c = dynamic_cast<Audio_column*> (p))
144     {
145       c->performance_l_ = this;
146     }
147   audio_elem_p_list_ = new Killing_cons<Audio_element> (p, audio_elem_p_list_);
148 }
149
150
151 void
152 Performance::process ()
153 {
154   String out = output_name_global;
155   if (out == "-")
156     out = "lelie.midi";
157   int def = midi_l_->get_next_score_count ();
158   if (def)
159     {
160       Path p = split_path (out);
161       p.base += "-" + to_str (def);
162       out = p.str ();
163     }
164
165   /* Maybe a bit crude, but we had this before */
166   Path p = split_path (out);
167   p.ext = "midi";
168   out = p.str ();
169   
170   Midi_stream midi_stream (out);
171   progress_indication (_f ("MIDI output to %s...", out));
172   target_str_global_array.push (out);
173
174   output (midi_stream);
175   progress_indication ("\n");
176 }