mirror of
				https://github.com/cookiengineer/audacity
				synced 2025-10-31 14:13:50 +01:00 
			
		
		
		
	
		
			
				
	
	
		
			775 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			775 lines
		
	
	
		
			26 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| #include "assert.h"
 | |
| #include "stdlib.h"
 | |
| #include "string.h"
 | |
| #include "ctype.h"
 | |
| #include "trace.h"
 | |
| #include <string>
 | |
| #include <fstream>
 | |
| #include <algorithm>
 | |
| #include "strparse.h"
 | |
| #include "allegro.h"
 | |
| #include "algrd_internal.h"
 | |
| 
 | |
| using namespace std;
 | |
| 
 | |
| #define streql(s1, s2) (strcmp(s1, s2) == 0)
 | |
| #define field_max 80
 | |
| 
 | |
| class Alg_reader {
 | |
| public:
 | |
|     istream *file;
 | |
|     string input_line;
 | |
|     int line_no;
 | |
|     String_parse line_parser;
 | |
|     bool line_parser_flag;
 | |
|     string field;
 | |
|     bool error_flag;
 | |
|     Alg_seq_ptr seq;
 | |
|     double tsnum;
 | |
|     double tsden;
 | |
|     double offset;
 | |
|     bool offset_found;
 | |
| 
 | |
|     Alg_reader(istream *a_file, Alg_seq_ptr new_seq);
 | |
|     void readline();
 | |
|     Alg_parameters_ptr process_attributes(Alg_parameters_ptr attributes,
 | |
|                                           double time);
 | |
|     bool parse();
 | |
|     long parse_chan(string &field);
 | |
|     long parse_int(string &field);
 | |
|     int find_real_in(string &field, int n);
 | |
|     double parse_real(string &field);
 | |
|     void parse_error(string &field, long offset, const char *message);
 | |
|     double parse_dur(string &field, double base);
 | |
|     double parse_after_dur(double dur, string &field, int n, double base);
 | |
|     double parse_loud(string &field);
 | |
|     long parse_key(string &field);
 | |
|     double parse_pitch(string &field);
 | |
|     long parse_after_key(int key, string &field, int n);
 | |
|     long find_int_in(string &field, int n);
 | |
|     bool parse_attribute(string &field, Alg_parameter_ptr parm);
 | |
|     bool parse_val(Alg_parameter_ptr param, string &s, int i);
 | |
|     bool check_type(char type_char, Alg_parameter_ptr param);
 | |
| };
 | |
| 
 | |
| 
 | |
| double Alg_reader::parse_pitch(string &field)
 | |
| {
 | |
|     if (isdigit(field[1])) {
 | |
|         int last = find_real_in(field, 1);
 | |
|         string real_string = field.substr(1, last - 1);
 | |
|         return atof(real_string.c_str());
 | |
|     } else {
 | |
|         return (double) parse_key(field);
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| // it is the responsibility of the caller to delete
 | |
| // the seq
 | |
| Alg_reader::Alg_reader(istream *a_file, Alg_seq_ptr new_seq)
 | |
| {
 | |
|     file = a_file; // save the file
 | |
|     line_parser_flag = false;
 | |
|     line_no = 0;
 | |
|     tsnum = 4; // default time signature
 | |
|     tsden = 4;
 | |
|     seq = new_seq;
 | |
|     offset = 0.0;
 | |
|     offset_found = false;
 | |
| }
 | |
| 
 | |
| 
 | |
| Alg_error alg_read(istream &file, Alg_seq_ptr new_seq, double *offset_ptr)
 | |
|     // read a sequence from allegro file
 | |
| {
 | |
|     assert(new_seq);
 | |
|     Alg_reader alg_reader(&file, new_seq);
 | |
|     bool err = alg_reader.parse();
 | |
|     if (!err && offset_ptr) {
 | |
|         *offset_ptr = alg_reader.offset;
 | |
|     }
 | |
|     return (err ? alg_error_syntax : alg_no_error);
 | |
| }
 | |
| 
 | |
| 
 | |
| void Alg_reader::readline()
 | |
| {
 | |
|     // a word about memory management: this Alg_reader has a
 | |
|     // member variable input_line that holds a line of input
 | |
|     // it is reused for each line. input_line is parsed by
 | |
|     // line_parser, which holds a reference to input_line
 | |
|     line_parser_flag = false;
 | |
|     if (getline(*file, input_line)) {
 | |
|         line_parser.init(&input_line);
 | |
|         line_parser_flag = true;
 | |
|         error_flag = false;
 | |
|     }
 | |
| }
 | |
| 
 | |
| 
 | |
| Alg_parameters_ptr Alg_reader::process_attributes(
 | |
|         Alg_parameters_ptr attributes, double time)
 | |
| {
 | |
|     // print "process_attributes:", attributes
 | |
|     bool ts_flag = false;
 | |
|     if (attributes) {
 | |
|         Alg_parameters_ptr a;
 | |
|         bool in_seconds = seq->get_units_are_seconds();
 | |
|         if ((a = Alg_parameters::remove_key(&attributes, "tempor"))) {
 | |
|             double tempo = a->parm.r;
 | |
|             seq->insert_tempo(tempo, seq->get_time_map()->time_to_beat(time));
 | |
|         }
 | |
|         if ((a = Alg_parameters::remove_key(&attributes, "beatr"))) {
 | |
|             double beat = a->parm.r;
 | |
|             seq->insert_beat(time, beat);
 | |
|         }
 | |
|         if ((a = Alg_parameters::remove_key(&attributes, "timesig_numr"))) {
 | |
|             tsnum = a->parm.r;
 | |
|             ts_flag = true;
 | |
|         }
 | |
|         if ((a = Alg_parameters::remove_key(&attributes, "timesig_denr"))) {
 | |
|             tsden = a->parm.r;
 | |
|             ts_flag = true;
 | |
|         }
 | |
|         if (ts_flag) {
 | |
|             seq->set_time_sig(seq->get_time_map()->time_to_beat(time),
 | |
|             tsnum, tsden);
 | |
|         }
 | |
|         if (in_seconds) seq->convert_to_seconds();
 | |
|     }
 | |
|     return attributes; // in case it was modified
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Alg_reader::parse()
 | |
| {
 | |
|     int voice = 0;
 | |
|     int key = 60;
 | |
|     double loud = 100.0;
 | |
|     double pitch = 60.0;
 | |
|     double dur = 1.0;
 | |
|     double time = 0.0;
 | |
|     int track_num = 0;
 | |
|     seq->convert_to_seconds();
 | |
|     //seq->set_real_dur(0.0); // just in case it's not initialized already
 | |
|     readline();
 | |
|     bool valid = false; // ignore blank lines
 | |
|     while (line_parser_flag) {
 | |
|         bool time_flag = false;
 | |
|         bool next_flag = false;
 | |
|         double next;
 | |
|         bool voice_flag = false;
 | |
|         bool loud_flag = false;
 | |
|         bool dur_flag = false;
 | |
|         bool new_pitch_flag = false; // "P" syntax or "A"-"G" syntax
 | |
|         double new_pitch = 0.0;
 | |
|         bool new_key_flag = false;   // "K" syntax
 | |
|         int new_key = 0;
 | |
|         Alg_parameters_ptr attributes = NULL;
 | |
|         if (line_parser.peek() == '#') {
 | |
|             // look for #track
 | |
|             line_parser.get_nonspace_quoted(field);
 | |
|             if (streql(field.c_str(), "#track")) {
 | |
|                 line_parser.get_nonspace_quoted(field); // number
 | |
|                 field.insert(0, " "); // need char at beginning because
 | |
|                 // parse_int ignores the first character of the argument
 | |
|                 track_num = parse_int(field);
 | |
|                 seq->add_track(track_num);
 | |
| 
 | |
|                 // maybe we have a sequence or track name
 | |
|                 line_parser.get_remainder(field);
 | |
|                 // if there is a non-space character after #track n then
 | |
|                 // use it as sequence or track name. Note that because we
 | |
|                 // skip over spaces, a sequence or track name cannot begin
 | |
|                 // with leading blanks. Another decision is that the name
 | |
|                 // must be at time zero
 | |
|                 if (field.length() > 0) {
 | |
|                     // insert the field as sequence name or track name
 | |
|                     Alg_update_ptr update = new Alg_update;
 | |
|                     update->chan = -1;
 | |
|                     update->time = 0;
 | |
|                     update->set_identifier(-1);
 | |
|                     // sequence name is whatever is on track 0
 | |
|                     // other tracks have track names
 | |
|                     const char *attr =
 | |
|                             (track_num == 0 ? "seqnames" : "tracknames");
 | |
|                     update->parameter.set_attr(
 | |
|                             symbol_table.insert_string(attr));
 | |
|                     update->parameter.s = heapify(field.c_str());
 | |
|                     seq->add_event(update, track_num);
 | |
|                 }
 | |
|             } else if (streql(field.c_str(), "#offset")) {
 | |
|                 if (offset_found) {
 | |
|                     parse_error(field, 0, "#offset specified twice");
 | |
|                 }
 | |
|                 offset_found = true;
 | |
|                 line_parser.get_nonspace_quoted(field); // number
 | |
|                 field.insert(0, " "); // need char at beginning because
 | |
|                 // parse_real ignores first character in the argument
 | |
|                 offset = parse_real(field);
 | |
|             }
 | |
|         } else {
 | |
|             // we must have a track to insert into
 | |
|             if (seq->tracks() == 0) seq->add_track(0);
 | |
|             line_parser.get_nonspace_quoted(field);
 | |
|             char pk = line_parser.peek();
 | |
|             // attributes are parsed as two adjacent nonspace_quoted tokens
 | |
|             // so we have to conditionally call get_nonspace_quoted() again
 | |
|             if (pk && !isspace(pk)) {
 | |
|                 string field2;
 | |
|                 line_parser.get_nonspace_quoted(field2);
 | |
|                 field.append(field2);
 | |
|             }
 | |
|             while (field[0]) {
 | |
|                 char first = toupper(field[0]);
 | |
|                 if (strchr("ABCDEFGKLPUSIQHW-", first)) {
 | |
|                     valid = true; // it's a note or event
 | |
|                 }
 | |
|                 if (first == 'V') {
 | |
|                     if (voice_flag) {
 | |
|                         parse_error(field, 0, "Voice specified twice");
 | |
|                     } else {
 | |
|                         voice = parse_chan(field);
 | |
|                     }
 | |
|                     voice_flag = true;
 | |
|                 } else if (first == 'T') {
 | |
|                     if (time_flag) {
 | |
|                         parse_error(field, 0, "Time specified twice");
 | |
|                     } else {
 | |
|                         time = parse_dur(field, 0.0);
 | |
|                     }
 | |
|                     time_flag = true;
 | |
|                 } else if (first == 'N') {
 | |
|                     if (next_flag) {
 | |
|                         parse_error(field, 0, "Next specified twice");
 | |
|                     } else {
 | |
|                         next = parse_dur(field, time);
 | |
|                     }
 | |
|                     next_flag = true;
 | |
|                 } else if (first == 'K') {
 | |
|                     if (new_key_flag) {
 | |
|                         parse_error(field, 0, "Key specified twice");
 | |
|                     } else {
 | |
|                         new_key = parse_key(field);
 | |
|                         new_key_flag = true;
 | |
|                     }
 | |
|                 } else if (first == 'L') {
 | |
|                     if (loud_flag) {
 | |
|                         parse_error(field, 0, "Loudness specified twice");
 | |
|                     } else {
 | |
|                         loud = parse_loud(field);
 | |
|                     }
 | |
|                     loud_flag = true;
 | |
|                 } else if (first == 'P') {
 | |
|                     if (new_pitch_flag) {
 | |
|                         parse_error(field, 0, "Pitch specified twice");
 | |
|                     } else {
 | |
|                         new_pitch = parse_pitch(field);
 | |
|                         new_pitch_flag = true;
 | |
|                     }
 | |
|                 } else if (first == 'U') {
 | |
|                     if (dur_flag) {
 | |
|                         parse_error(field, 0, "Dur specified twice");
 | |
|                     } else {
 | |
|                         dur = parse_dur(field, time);
 | |
|                         dur_flag = true;
 | |
|                     }
 | |
|                 } else if (strchr("SIQHW", first)) {
 | |
|                     if (dur_flag) {
 | |
|                         parse_error(field, 0, "Dur specified twice");
 | |
|                     } else {
 | |
|                         // prepend 'U' to field, copy EOS too
 | |
|                         field.insert((unsigned int) 0, 1, 'U');
 | |
|                         dur = parse_dur(field, time);
 | |
|                         dur_flag = true;
 | |
|                     }
 | |
|                 } else if (strchr("ABCDEFG", first)) {
 | |
|                     if (new_pitch_flag) {
 | |
|                         parse_error(field, 0, "Pitch specified twice");
 | |
|                     } else {
 | |
|                         // prepend 'P' to field
 | |
|                         field.insert((unsigned int) 0, 1, 'P');
 | |
|                         new_pitch = parse_pitch(field);
 | |
|                         new_pitch_flag = true;
 | |
|                     }
 | |
|                 } else if (first == '-') {
 | |
|                     Alg_parameter parm;
 | |
|                     if (parse_attribute(field, &parm)) { // enter attribute-value pair
 | |
|                         attributes = new Alg_parameters(attributes);
 | |
|                         attributes->parm = parm;
 | |
|                         parm.s = NULL; // protect string from deletion by destructor
 | |
|                     }
 | |
|                 } else {
 | |
|                     parse_error(field, 0, "Unknown field");
 | |
|                 }
 | |
| 
 | |
|                 if (error_flag) {
 | |
|                     field[0] = 0; // exit the loop
 | |
|                 } else {
 | |
|                     line_parser.get_nonspace_quoted(field);
 | |
|                     pk = line_parser.peek();
 | |
|                     // attributes are parsed as two adjacent nonspace_quoted 
 | |
|                     // tokens so we have to conditionally call 
 | |
|                     // get_nonspace_quoted() again
 | |
|                     if (pk && !isspace(pk)) {
 | |
|                         string field2;
 | |
|                         line_parser.get_nonspace_quoted(field2);
 | |
|                         field.append(field2);
 | |
|                     }
 | |
|                 }
 | |
|             }
 | |
|             // a case analysis:
 | |
|             // Key < 128 implies pitch unless pitch is explicitly given
 | |
|             // Pitch implies Key unless key is explicitly given,
 | |
|             // Pitch is rounded to nearest integer to determine the Key
 | |
|             //    if necessary, so MIDI files will lose the pitch fraction
 | |
|             // A-G is a Pitch specification (therefore it implies Key)
 | |
|             //   K60 P60 -- both are specified, use 'em
 | |
|             //   K60 P60 C4 -- overconstrained, an error
 | |
|             //   K60 C4 -- OK, but K60 is already implied by C4
 | |
|             //   K60 -- OK, pitch is 60
 | |
|             //   C4 P60 -- over constrained
 | |
|             //   P60 -- OK, key is 60
 | |
|             //   P60.1 -- OK, key is 60
 | |
|             //   C4 -- OK, key is 60, pitch is 60
 | |
|             //   <nothing> -- OK, key and pitch from before
 | |
|             //   K200 P60 -- ok, pitch is 60
 | |
|             //   K200 with neither P60 nor C4 uses 
 | |
|             //       pitch from before
 | |
| 
 | |
|             // figure out what the key/instance is:
 | |
|             if (new_key_flag) { // it was directly specified
 | |
|                 key = new_key;
 | |
|             } else if (new_pitch_flag) {
 | |
|                 // pitch was specified, but key was not; get key from pitch
 | |
|                 key = (int) (new_pitch + 0.5); // round to integer key number
 | |
|             }
 | |
|             if (new_pitch_flag) {
 | |
|                 pitch = new_pitch;
 | |
|             } else if (key < 128 && new_key_flag) {
 | |
|                 // no explicit pitch, but key < 128, so it implies pitch
 | |
|                 pitch = key;
 | |
|                 new_pitch_flag = true;
 | |
|             }
 | |
|             // now we've acquired new parameters
 | |
|             // if (it is a note, then enter the note
 | |
|             if (valid) {
 | |
|                 // change tempo or beat
 | |
|                 attributes = process_attributes(attributes, time);
 | |
|                 // if there's a duration or pitch, make a note:
 | |
|                 if (new_pitch_flag || dur_flag) {
 | |
|                     Alg_note_ptr note_ptr = new Alg_note;
 | |
|                     note_ptr->chan = voice;
 | |
|                     note_ptr->time = time;
 | |
|                     note_ptr->dur = dur;
 | |
|                     note_ptr->set_identifier(key);
 | |
|                     note_ptr->pitch = (float) pitch;
 | |
|                     note_ptr->loud = (float) loud;
 | |
|                     note_ptr->parameters = attributes;
 | |
|                     seq->add_event(note_ptr, track_num); // sort later
 | |
|                     if (seq->get_real_dur() < (time + dur)) seq->set_real_dur(time + dur);
 | |
|                 } else {
 | |
|                     int update_key = -1;
 | |
|                     // key must appear explicitly; otherwise
 | |
|                     //    update applies to channel
 | |
|                     if (new_key_flag) {
 | |
|                         update_key = key;
 | |
|                     }
 | |
|                     if (loud_flag) {
 | |
|                         Alg_update_ptr new_upd = new Alg_update;
 | |
|                         new_upd->chan = voice;
 | |
|                         new_upd->time = time;
 | |
|                         new_upd->set_identifier(update_key);
 | |
|                         new_upd->parameter.set_attr(symbol_table.insert_string("loudr"));
 | |
|                         new_upd->parameter.r = pitch;
 | |
|                         seq->add_event(new_upd, track_num);
 | |
|                         if (seq->get_real_dur() < time) seq->set_real_dur(time);
 | |
|                     }
 | |
|                     if (attributes) {
 | |
|                         while (attributes) {
 | |
|                             Alg_update_ptr new_upd = new Alg_update;
 | |
|                             new_upd->chan = voice;
 | |
|                             new_upd->time = time;
 | |
|                             new_upd->set_identifier(update_key);
 | |
|                             new_upd->parameter = attributes->parm;
 | |
|                             seq->add_event(new_upd, track_num);
 | |
|                             Alg_parameters_ptr p = attributes;
 | |
|                             attributes = attributes->next;
 | |
|                             p->parm.s = NULL; // so we don't delete the string
 | |
|                             delete p;
 | |
|                         }
 | |
|                     }
 | |
|                 }
 | |
|                 if (next_flag) {
 | |
|                     time = time + next;
 | |
|                 } else if (dur_flag || new_pitch_flag) { // a note: incr by dur
 | |
|                     time = time + dur;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
|         readline();
 | |
|     }
 | |
|     if (!error_flag) { // why not convert even if there was an error? -RBD
 | |
|         seq->convert_to_seconds(); // make sure format is correct
 | |
|     }
 | |
|     // real_dur is valid, translate to beat_dur
 | |
|     seq->set_beat_dur((seq->get_time_map())->time_to_beat(seq->get_real_dur()));
 | |
|     return error_flag;
 | |
| }
 | |
| 
 | |
| 
 | |
| long Alg_reader::parse_chan(string &field)
 | |
| {
 | |
|     const char *int_string = field.c_str() + 1;
 | |
|     const char *msg = "Integer or - expected";
 | |
|     const char *p = int_string;
 | |
|     char c;
 | |
|     // check that all chars in int_string are digits or '-':
 | |
|     while ((c = *p++)) {
 | |
|         if (!isdigit(c) && c != '-') {
 | |
|             parse_error(field, p - field.c_str() - 1, msg);
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
|     p--; // p now points to end-of-string character
 | |
|     if (p - int_string == 0) {
 | |
|         // bad: string length is zero
 | |
|         parse_error(field, 1, msg);
 | |
|         return 0;
 | |
|     }
 | |
|     if (p - int_string == 1 && int_string[0] == '-') {
 | |
|         // special case: entire string is "-", interpret as -1
 | |
|         return -1;
 | |
|     }
 | |
|     return atoi(int_string);
 | |
| }
 | |
| 
 | |
| 
 | |
| long Alg_reader::parse_int(string &field)
 | |
| {
 | |
|     const char *int_string = field.c_str() + 1;
 | |
|     const char *msg = "Integer expected";
 | |
|     const char *p = int_string;
 | |
|     char c;
 | |
|     // check that all chars in int_string are digits:
 | |
|     while ((c = *p++)) {
 | |
|         if (!isdigit(c)) {
 | |
|             parse_error(field, p - field.c_str() - 1, msg);
 | |
|             return 0;
 | |
|         }
 | |
|     }
 | |
|     p--; // p now points to end-of-string character
 | |
|     if (p - int_string == 0) {
 | |
|         // bad: string length is zero
 | |
|         parse_error(field, 1, msg);
 | |
|         return 0;
 | |
|     }
 | |
|     return atoi(int_string);
 | |
| }
 | |
| 
 | |
| 
 | |
| int Alg_reader::find_real_in(string &field, int n)
 | |
| {
 | |
|     // scans from offset n to the end of a real constant
 | |
|     bool decimal = false;
 | |
|     int len = field.length();
 | |
|     if (n < len && field[n] == '-') n += 1; // parse one minus sign
 | |
|     for (int i = n; i < len; i++) {
 | |
|         char c = field[i];
 | |
|         if (!isdigit(c)) {
 | |
|             if (c == '.' && !decimal) {
 | |
|                 decimal = true;
 | |
|             } else {
 | |
|                 return i;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     return len;
 | |
| }
 | |
| 
 | |
| 
 | |
| double Alg_reader::parse_real(string &field)
 | |
| {
 | |
|     const char *msg = "Real expected";
 | |
|     int last = find_real_in(field, 1);
 | |
|     string real_string = field.substr(1, last - 1);
 | |
|     if (last <= 1 || last < (int) field.length()) {
 | |
|        parse_error(field, 1, msg);
 | |
|        return 0;
 | |
|     }
 | |
|     return atof(real_string.c_str());
 | |
| }
 | |
| 
 | |
| 
 | |
| void Alg_reader::parse_error(string &field, long offset, const char *message)
 | |
| {
 | |
|     int position = line_parser.pos - field.length() + offset;
 | |
|     error_flag = true;
 | |
|     puts(line_parser.str->c_str());
 | |
|     for (int i = 0; i < position; i++) {
 | |
|         putc(' ', stdout);
 | |
|     }
 | |
|     putc('^', stdout);
 | |
|     printf("    %s\n", message);
 | |
| }
 | |
| 
 | |
| 
 | |
| double duration_lookup[] = { 0.25, 0.5, 1.0, 2.0, 4.0 };
 | |
| 
 | |
| 
 | |
| double Alg_reader::parse_dur(string &field, double base)
 | |
| {
 | |
|     const char *msg = "Duration expected";
 | |
|     const char *durs = "SIQHW";
 | |
|     const char *p;
 | |
|     int last;
 | |
|     double dur;
 | |
|     if (field.length() < 2) {
 | |
|         // fall through to error message
 | |
|         return -1;
 | |
|     } else if (isdigit(field[1])) {
 | |
|         last = find_real_in(field, 1);
 | |
|         string real_string = field.substr(1, last - 1);
 | |
|         dur = atof(real_string.c_str());
 | |
|         // convert dur from seconds to beats
 | |
|         dur = seq->get_time_map()->time_to_beat(base + dur) - 
 | |
|               seq->get_time_map()->time_to_beat(base);
 | |
|     } else if ((p = strchr(durs, toupper(field[1])))) {
 | |
|         dur = duration_lookup[p - durs];
 | |
|         last = 2;
 | |
|     } else {
 | |
|         parse_error(field, 1, msg);
 | |
|         return 0;
 | |
|     }
 | |
|     dur = parse_after_dur(dur, field, last, base);
 | |
|     dur = seq->get_time_map()->beat_to_time(
 | |
|               seq->get_time_map()->time_to_beat(base) + dur) - base;
 | |
|     return dur;
 | |
| }
 | |
| 
 | |
| 
 | |
| double Alg_reader::parse_after_dur(double dur, string &field, 
 | |
|                                    int n, double base)
 | |
| {
 | |
|     if ((int) field.length() == n) {
 | |
|         return dur;
 | |
|     }
 | |
|     if (toupper(field[n]) == 'T') {
 | |
|         return parse_after_dur(dur * 2/3, field, n + 1, base);
 | |
|     }
 | |
|     if (field[n] == '.') {
 | |
|         return parse_after_dur(dur * 1.5, field, n + 1, base);
 | |
|     }
 | |
|     if (isdigit(field[n])) {
 | |
|         int last = find_real_in(field, n);
 | |
|         string a_string = field.substr(n, last - n);
 | |
|         double f = atof(a_string.c_str());
 | |
|         return parse_after_dur(dur * f, field, last, base);
 | |
|     }
 | |
|     if (field[n] == '+') {
 | |
|         string a_string = field.substr(n + 1);
 | |
|         return dur + parse_dur(
 | |
|                 a_string, seq->get_time_map()->beat_to_time(
 | |
|                         seq->get_time_map()->time_to_beat(base) + dur));
 | |
|     }
 | |
|     parse_error(field, n, "Unexpected character in duration");
 | |
|     return dur;
 | |
| }
 | |
| 
 | |
| struct loud_lookup_struct {
 | |
|     const char *str;
 | |
|     int val;
 | |
| } loud_lookup[] = { {"FFF", 127}, {"FF", 120}, {"F", 110}, {"MF", 100}, 
 | |
|                     {"MP", 90}, {"P", 80}, {"PP", 70}, {"PPP", 60}, 
 | |
|                     {NULL, 0} };
 | |
| 
 | |
| 
 | |
| double Alg_reader::parse_loud(string &field)
 | |
| {
 | |
|     const char *msg = "Loudness expected";
 | |
|     if (isdigit(field[1])) {
 | |
|         return parse_int(field);
 | |
|     } else {
 | |
|         string dyn = field.substr(1);
 | |
|         transform(dyn.begin(), dyn.end(), dyn.begin(), ::toupper);
 | |
|         for (int i = 0; loud_lookup[i].str; i++) {
 | |
|             if (streql(loud_lookup[i].str, dyn.c_str())) {
 | |
|                 return (double) loud_lookup[i].val;
 | |
|             }
 | |
|         }
 | |
|     }
 | |
|     parse_error(field, 1, msg);
 | |
|     return 100.0;
 | |
| }
 | |
| 
 | |
| 
 | |
| int key_lookup[] = {21, 23, 12, 14, 16, 17, 19};
 | |
| 
 | |
| 
 | |
| // the field can be K<number> or K[A-G]<number> or P[A-G]<number>
 | |
| // (this can be called from parse_pitch() to handle [A-G])
 | |
| // Notice that the routine ignores the first character: K or P
 | |
| //
 | |
| long Alg_reader::parse_key(string &field)
 | |
| {
 | |
|     const char *msg = "Pitch expected";
 | |
|     const char *pitches = "ABCDEFG";
 | |
|     const char *p;
 | |
|     if (isdigit(field[1])) {
 | |
|         // This routine would not have been called if field = "P<number>"
 | |
|         // so it must be "K<number>" so <number> must be an integer.
 | |
|         return parse_int(field);
 | |
|     } else if ((p = strchr(pitches, toupper(field[1])))) {
 | |
|         long key = key_lookup[p - pitches];
 | |
|         key = parse_after_key(key, field, 2);
 | |
|         return key;
 | |
|     }
 | |
|     parse_error(field, 1, msg);
 | |
|     return 0;
 | |
| }
 | |
| 
 | |
| 
 | |
| long Alg_reader::parse_after_key(int key, string &field, int n)
 | |
| {
 | |
|     if ((int) field.length() == n) {
 | |
|         return key;
 | |
|     }
 | |
|     char c = toupper(field[n]);
 | |
|     if (c == 'S') {
 | |
|         return parse_after_key(key + 1, field, n + 1);
 | |
|     }
 | |
|     if (c == 'F') {
 | |
|         return parse_after_key(key - 1, field, n + 1);
 | |
|     }
 | |
|     if (isdigit(field[n])) {
 | |
|         int last = find_int_in(field, n);
 | |
|         string octave = field.substr(n, last - n);
 | |
|         int oct = atoi(octave.c_str());
 | |
|         return parse_after_key(key + oct * 12, field, last);
 | |
|     }
 | |
|     parse_error(field, n, "Unexpected character in pitch");
 | |
|     return key;
 | |
| }
 | |
| 
 | |
| 
 | |
| long Alg_reader::find_int_in(string &field, int n)
 | |
| {
 | |
|     while ((int) field.length() > n && isdigit(field[n])) {
 | |
|         n = n + 1;
 | |
|     }
 | |
|     return n;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Alg_reader::parse_attribute(string &field, Alg_parameter_ptr param)
 | |
| {
 | |
|     int i = 1;
 | |
|     while (i < (int) field.length()) {
 | |
|         if (field[i] == ':') {
 | |
|             string attr = field.substr(1, i - 1);
 | |
|             char type_char = field[i - 1];
 | |
|             if (strchr("iarsl", type_char)) {
 | |
|                 param->set_attr(symbol_table.insert_string(attr.c_str()));
 | |
|                 parse_val(param, field, i + 1);
 | |
|             } else {
 | |
|                 parse_error(field, 0, "attribute needs to end with typecode: i,a,r,s, or l");
 | |
|             }
 | |
|             return !error_flag;
 | |
|         }
 | |
|         i = i + 1;
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Alg_reader::parse_val(Alg_parameter_ptr param, string &s, int i)
 | |
| {
 | |
|     int len = (int) s.length();
 | |
|     if (i >= len) {
 | |
|         return false;
 | |
|     }
 | |
|     if (s[i] == '"') {
 | |
|         if (!check_type('s', param)) {
 | |
|             return false;
 | |
|         }
 | |
|         // note: (len - i) includes 2 quote characters but no EOS character
 | |
|         // so total memory to allocate is (len - i) - 1
 | |
|         char *r = new char[(len - i) - 1];
 | |
|         strncpy(r, s.c_str() + i + 1, (len - i) - 2);
 | |
|         r[(len - i) - 2] = 0; // terminate the string
 | |
|         param->s = r;
 | |
|     } else if (s[i] == '\'') {
 | |
|         if (!check_type('a', param)) {
 | |
|             return false;
 | |
|         }
 | |
|         string r = s.substr(i + 1, len - i - 2);
 | |
|         param->a = symbol_table.insert_string(r.c_str());
 | |
|     } else if (param->attr_type() == 'l') {
 | |
|         if (streql(s.c_str() + i, "true") || 
 | |
|             streql(s.c_str() + i, "t")) {
 | |
|             param->l = true;
 | |
|         } else if (streql(s.c_str() + i, "false") || 
 | |
|                    streql(s.c_str() + i, "nil")) {
 | |
|             param->l = false;
 | |
|         } else return false;
 | |
|     } else if (isdigit(s[i]) || s[i] == '-' || s[i] == '.') {
 | |
|         int pos = i;
 | |
|         bool period = false;
 | |
|         int sign = 1;
 | |
|         if (s[pos] == '-') {
 | |
|             sign = -1;
 | |
|             pos++;
 | |
|         }
 | |
|         while (pos < len) {
 | |
|             if (isdigit(s[pos])) {
 | |
|                 ;
 | |
|             } else if (!period && s[pos] == '.') {
 | |
|                 period = true;
 | |
|             } else {
 | |
|                 parse_error(s, pos, "Unexpected char in number");
 | |
|                 return false;
 | |
|             }
 | |
|             pos = pos + 1;
 | |
|         }
 | |
|         string r = s.substr(i, len - i);
 | |
|         if (period) {
 | |
|             if (!check_type('r', param)) {
 | |
|                 return false;
 | |
|             }
 | |
|             param->r = atof(r.c_str());
 | |
|         } else {
 | |
|             if (param->attr_type() == 'r') {
 | |
|                 param->r = atoi(r.c_str());
 | |
|             } else if (!check_type('i', param)) {
 | |
|                 return false;
 | |
|             } else {
 | |
|                 param->i = atoi(r.c_str());
 | |
|             }
 | |
|         }
 | |
|     } else {
 | |
|         parse_error(s, i, "invalid value");
 | |
|         return false;
 | |
|     }
 | |
|     return true;
 | |
| }
 | |
| 
 | |
| 
 | |
| bool Alg_reader::check_type(char type_char, Alg_parameter_ptr param)
 | |
| {
 | |
|     return param->attr_type() == type_char;
 | |
| }
 | |
| 
 | |
| 
 | |
| //duration_lookup = {"S": 0.5, "I": 0.5, "Q": 1, "H": 2, "W": 4}
 | |
| //key_lookup = {"C": 12, "D": 14, "E": 16, "F": 17, "G": 19, "A": 21, "B": 23}
 | |
| 
 | |
| /*
 | |
| def test():
 | |
|     reader = Alg_reader(open("data\\test.gro", "r"))
 | |
|     reader.parse()
 | |
|     score = reader->seq.notes
 | |
|     print "score:", score
 | |
|     reader = nil
 | |
| */
 |