| Conditions | 47 |
| Paths | > 20000 |
| Total Lines | 296 |
| Code Lines | 161 |
| Lines | 0 |
| Ratio | 0 % |
| Changes | 1 | ||
| Bugs | 0 | Features | 0 |
Small methods make your code easier to understand, in particular if combined with a good name. Besides, if your method is small, finding a good name is usually much easier.
For example, if you find yourself adding comments to a method's body, this is usually a good sign to extract the commented part to a new method, and use the comment as a starting point when coming up with a good name for this new method.
Commonly applied refactorings include:
If many parameters/temporary variables are present:
| 1 | <?php |
||
| 27 | public function Analyze() |
||
| 28 | { |
||
| 29 | $getid3 = $this->getid3; |
||
| 30 | |||
| 31 | $getid3->info['midi']['raw'] = []; |
||
| 32 | $info_midi = &$getid3->info['midi']; |
||
| 33 | $info_midi_raw = &$info_midi['raw']; |
||
| 34 | |||
| 35 | $getid3->info['fileformat'] = 'midi'; |
||
| 36 | $getid3->info['audio']['dataformat'] = 'midi'; |
||
| 37 | |||
| 38 | fseek($getid3->fp, $getid3->info['avdataoffset'], SEEK_SET); |
||
| 39 | $midi_data = fread($getid3->fp, getid3::FREAD_BUFFER_SIZE); |
||
| 40 | |||
| 41 | // Magic bytes: 'MThd' |
||
| 42 | |||
| 43 | getid3_lib::ReadSequence( |
||
| 44 | 'BigEndian2Int', |
||
| 45 | $info_midi_raw, |
||
| 46 | $midi_data, |
||
| 47 | 4, |
||
| 48 | [ |
||
| 49 | 'headersize' => 4, |
||
| 50 | 'fileformat' => 2, |
||
| 51 | 'tracks' => 2, |
||
| 52 | 'ticksperqnote' => 2 |
||
| 53 | ] |
||
| 54 | ); |
||
| 55 | |||
| 56 | $offset = 14; |
||
| 57 | |||
| 58 | for ($i = 0; $i < $info_midi_raw['tracks']; $i++) { |
||
| 59 | if ((strlen($midi_data) - $offset) < 8) { |
||
| 60 | $midi_data .= fread($getid3->fp, getid3::FREAD_BUFFER_SIZE); |
||
| 61 | } |
||
| 62 | |||
| 63 | $track_id = substr($midi_data, $offset, 4); |
||
| 64 | $offset += 4; |
||
| 65 | |||
| 66 | if ('MTrk' != $track_id) { |
||
| 67 | throw new getid3_exception('Expecting "MTrk" at ' . $offset . ', found ' . $track_id . ' instead'); |
||
| 68 | } |
||
| 69 | |||
| 70 | $track_size = getid3_lib::BigEndian2Int(substr($midi_data, $offset, 4)); |
||
| 71 | $offset += 4; |
||
| 72 | |||
| 73 | $track_data_array[$i] = substr($midi_data, $offset, $track_size); |
||
| 74 | $offset += $track_size; |
||
| 75 | } |
||
| 76 | |||
| 77 | if (!isset($track_data_array) || !is_array($track_data_array)) { |
||
|
|
|||
| 78 | throw new getid3_exception('Cannot find MIDI track information'); |
||
| 79 | } |
||
| 80 | |||
| 81 | $info_midi['totalticks'] = 0; |
||
| 82 | $getid3->info['playtime_seconds'] = 0; |
||
| 83 | $current_ms_per_beat = 500000; // 120 beats per minute; 60,000,000 microseconds per minute -> 500,000 microseconds per beat |
||
| 84 | $current_beats_per_min = 120; // 120 beats per minute; 60,000,000 microseconds per minute -> 500,000 microseconds per beat |
||
| 85 | $ms_per_quarter_note_after = []; |
||
| 86 | |||
| 87 | foreach ($track_data_array as $track_number => $track_data) { |
||
| 88 | $events_offset = $last_issued_midi_command = $last_issued_midi_channel = $cumulative_delta_time = $ticks_at_current_bpm = 0; |
||
| 89 | |||
| 90 | while ($events_offset < strlen($track_data)) { |
||
| 91 | $event_id = 0; |
||
| 92 | if (isset($midi_events[$track_number]) && is_array($midi_events[$track_number])) { |
||
| 93 | $event_id = count($midi_events[$track_number]); |
||
| 94 | } |
||
| 95 | $delta_time = 0; |
||
| 96 | for ($i = 0; $i < 4; $i++) { |
||
| 97 | $delta_time_byte = ord($track_data{$events_offset++}); |
||
| 98 | $delta_time = ($delta_time << 7) + ($delta_time_byte & 0x7F); |
||
| 99 | if ($delta_time_byte & 0x80) { |
||
| 100 | // another byte follows |
||
| 101 | } else { |
||
| 102 | break; |
||
| 103 | } |
||
| 104 | } |
||
| 105 | |||
| 106 | $cumulative_delta_time += $delta_time; |
||
| 107 | $ticks_at_current_bpm += $delta_time; |
||
| 108 | |||
| 109 | $midi_events[$track_number][$event_id]['deltatime'] = $delta_time; |
||
| 110 | |||
| 111 | $midi_event_channel = ord($track_data{$events_offset++}); |
||
| 112 | |||
| 113 | // OK, normal event - MIDI command has MSB set |
||
| 114 | if ($midi_event_channel & 0x80) { |
||
| 115 | $last_issued_midi_command = $midi_event_channel >> 4; |
||
| 116 | $last_issued_midi_channel = $midi_event_channel & 0x0F; |
||
| 117 | } // Running event - assume last command |
||
| 118 | else { |
||
| 119 | $events_offset--; |
||
| 120 | } |
||
| 121 | |||
| 122 | $midi_events[$track_number][$event_id]['eventid'] = $last_issued_midi_command; |
||
| 123 | $midi_events[$track_number][$event_id]['channel'] = $last_issued_midi_channel; |
||
| 124 | |||
| 125 | switch ($midi_events[$track_number][$event_id]['eventid']) { |
||
| 126 | case 0x8: // Note off (key is released) |
||
| 127 | case 0x9: // Note on (key is pressed) |
||
| 128 | case 0xA: // Key after-touch |
||
| 129 | |||
| 130 | //$notenumber = ord($track_data{$events_offset++}); |
||
| 131 | //$velocity = ord($track_data{$events_offset++}); |
||
| 132 | $events_offset += 2; |
||
| 133 | break; |
||
| 134 | |||
| 135 | case 0xB: // Control Change |
||
| 136 | |||
| 137 | //$controllernum = ord($track_data{$events_offset++}); |
||
| 138 | //$newvalue = ord($track_data{$events_offset++}); |
||
| 139 | $events_offset += 2; |
||
| 140 | break; |
||
| 141 | |||
| 142 | case 0xC: // Program (patch) change |
||
| 143 | |||
| 144 | $new_program_num = ord($track_data{$events_offset++}); |
||
| 145 | |||
| 146 | $info_midi_raw['track'][$track_number]['instrumentid'] = $new_program_num; |
||
| 147 | $info_midi_raw['track'][$track_number]['instrument'] = 10 == $track_number ? getid3_midi::GeneralMIDIpercussionLookup($new_program_num) : getid3_midi::GeneralMIDIinstrumentLookup($new_program_num); |
||
| 148 | break; |
||
| 149 | |||
| 150 | case 0xD: // Channel after-touch |
||
| 151 | |||
| 152 | //$channelnumber = ord($track_data{$events_offset++}); |
||
| 153 | break; |
||
| 154 | |||
| 155 | case 0xE: // Pitch wheel change (2000H is normal or no change) |
||
| 156 | |||
| 157 | //$changeLSB = ord($track_data{$events_offset++}); |
||
| 158 | //$changeMSB = ord($track_data{$events_offset++}); |
||
| 159 | //$pitchwheelchange = (($changeMSB & 0x7F) << 7) & ($changeLSB & 0x7F); |
||
| 160 | $events_offset += 2; |
||
| 161 | break; |
||
| 162 | |||
| 163 | case 0xF: |
||
| 164 | |||
| 165 | if (0xF == $midi_events[$track_number][$event_id]['channel']) { |
||
| 166 | $meta_event_command = ord($track_data{$events_offset++}); |
||
| 167 | $meta_event_length = ord($track_data{$events_offset++}); |
||
| 168 | $meta_event_data = substr($track_data, $events_offset, $meta_event_length); |
||
| 169 | $events_offset += $meta_event_length; |
||
| 170 | |||
| 171 | switch ($meta_event_command) { |
||
| 172 | case 0x00: // Set track sequence number |
||
| 173 | |||
| 174 | //$track_sequence_number = getid3_lib::BigEndian2Int(substr($meta_event_data, 0, $meta_event_length)); |
||
| 175 | //$info_midi_raw['events'][$track_number][$event_id]['seqno'] = $track_sequence_number; |
||
| 176 | break; |
||
| 177 | |||
| 178 | case 0x01: // Text: generic |
||
| 179 | |||
| 180 | $text_generic = substr($meta_event_data, 0, $meta_event_length); |
||
| 181 | //$info_midi_raw['events'][$track_number][$event_id]['text'] = $text_generic; |
||
| 182 | $info_midi['comments']['comment'][] = $text_generic; |
||
| 183 | break; |
||
| 184 | |||
| 185 | case 0x02: // Text: copyright |
||
| 186 | |||
| 187 | $text_copyright = substr($meta_event_data, 0, $meta_event_length); |
||
| 188 | //$info_midi_raw['events'][$track_number][$event_id]['copyright'] = $text_copyright; |
||
| 189 | $info_midi['comments']['copyright'][] = $text_copyright; |
||
| 190 | break; |
||
| 191 | |||
| 192 | case 0x03: // Text: track name |
||
| 193 | |||
| 194 | $text_trackname = substr($meta_event_data, 0, $meta_event_length); |
||
| 195 | $info_midi_raw['track'][$track_number]['name'] = $text_trackname; |
||
| 196 | break; |
||
| 197 | |||
| 198 | case 0x04: // Text: track instrument name |
||
| 199 | |||
| 200 | //$text_instrument = substr($meta_event_data, 0, $meta_event_length); |
||
| 201 | //$info_midi_raw['events'][$track_number][$event_id]['instrument'] = $text_instrument; |
||
| 202 | break; |
||
| 203 | |||
| 204 | case 0x05: // Text: lyrics |
||
| 205 | |||
| 206 | $text_lyrics = substr($meta_event_data, 0, $meta_event_length); |
||
| 207 | //$info_midi_raw['events'][$track_number][$event_id]['lyrics'] = $text_lyrics; |
||
| 208 | if (!isset($info_midi['lyrics'])) { |
||
| 209 | $info_midi['lyrics'] = ''; |
||
| 210 | } |
||
| 211 | $info_midi['lyrics'] .= $text_lyrics . "\n"; |
||
| 212 | break; |
||
| 213 | |||
| 214 | case 0x06: // Text: marker |
||
| 215 | |||
| 216 | //$text_marker = substr($meta_event_data, 0, $meta_event_length); |
||
| 217 | //$info_midi_raw['events'][$track_number][$event_id]['marker'] = $text_marker; |
||
| 218 | break; |
||
| 219 | |||
| 220 | case 0x07: // Text: cue point |
||
| 221 | |||
| 222 | //$text_cuepoint = substr($meta_event_data, 0, $meta_event_length); |
||
| 223 | //$info_midi_raw['events'][$track_number][$event_id]['cuepoint'] = $text_cuepoint; |
||
| 224 | break; |
||
| 225 | |||
| 226 | case 0x2F: // End Of Track |
||
| 227 | |||
| 228 | //$info_midi_raw['events'][$track_number][$event_id]['EOT'] = $cumulative_delta_time; |
||
| 229 | break; |
||
| 230 | |||
| 231 | case 0x51: // Tempo: microseconds / quarter note |
||
| 232 | |||
| 233 | $current_ms_per_beat = getid3_lib::BigEndian2Int(substr($meta_event_data, 0, $meta_event_length)); |
||
| 234 | $info_midi_raw['events'][$track_number][$cumulative_delta_time]['us_qnote'] = $current_ms_per_beat; |
||
| 235 | $current_beats_per_min = (1000000 / $current_ms_per_beat) * 60; |
||
| 236 | $ms_per_quarter_note_after[$cumulative_delta_time] = $current_ms_per_beat; |
||
| 237 | $ticks_at_current_bpm = 0; |
||
| 238 | break; |
||
| 239 | |||
| 240 | case 0x58: // Time signature |
||
| 241 | $timesig_numerator = getid3_lib::BigEndian2Int($meta_event_data[0]); |
||
| 242 | $timesig_denominator = pow(2, getid3_lib::BigEndian2Int($meta_event_data[1])); // $02 -> x/4, $03 -> x/8, etc |
||
| 243 | //$timesig_32inqnote = getid3_lib::BigEndian2Int($meta_event_data[2]); // number of 32nd notes to the quarter note |
||
| 244 | //$info_midi_raw['events'][$track_number][$event_id]['timesig_32inqnote'] = $timesig_32inqnote; |
||
| 245 | //$info_midi_raw['events'][$track_number][$event_id]['timesig_numerator'] = $timesig_numerator; |
||
| 246 | //$info_midi_raw['events'][$track_number][$event_id]['timesig_denominator'] = $timesig_denominator; |
||
| 247 | //$info_midi_raw['events'][$track_number][$event_id]['timesig_text'] = $timesig_numerator.'/'.$timesig_denominator; |
||
| 248 | $info_midi['timesignature'][] = $timesig_numerator . '/' . $timesig_denominator; |
||
| 249 | break; |
||
| 250 | |||
| 251 | case 0x59: // Keysignature |
||
| 252 | |||
| 253 | $keysig_sharpsflats = getid3_lib::BigEndian2Int($meta_event_data{0}); |
||
| 254 | if ($keysig_sharpsflats & 0x80) { |
||
| 255 | // (-7 -> 7 flats, 0 ->key of C, 7 -> 7 sharps) |
||
| 256 | $keysig_sharpsflats -= 256; |
||
| 257 | } |
||
| 258 | |||
| 259 | $keysig_majorminor = getid3_lib::BigEndian2Int($meta_event_data{1}); // 0 -> major, 1 -> minor |
||
| 260 | $keysigs = [-7 => 'Cb', -6 => 'Gb', -5 => 'Db', -4 => 'Ab', -3 => 'Eb', -2 => 'Bb', -1 => 'F', 0 => 'C', 1 => 'G', 2 => 'D', 3 => 'A', 4 => 'E', 5 => 'B', 6 => 'F#', 7 => 'C#']; |
||
| 261 | //$info_midi_raw['events'][$track_number][$event_id]['keysig_sharps'] = (($keysig_sharpsflats > 0) ? abs($keysig_sharpsflats) : 0); |
||
| 262 | //$info_midi_raw['events'][$track_number][$event_id]['keysig_flats'] = (($keysig_sharpsflats < 0) ? abs($keysig_sharpsflats) : 0); |
||
| 263 | //$info_midi_raw['events'][$track_number][$event_id]['keysig_minor'] = (bool)$keysig_majorminor; |
||
| 264 | //$info_midi_raw['events'][$track_number][$event_id]['keysig_text'] = $keysigs[$keysig_sharpsflats].' '.($info_midi_raw['events'][$track_number][$event_id]['keysig_minor'] ? 'minor' : 'major'); |
||
| 265 | |||
| 266 | // $keysigs[$keysig_sharpsflats] gets an int key (correct) - $keysigs["$keysig_sharpsflats"] gets a string key (incorrect) |
||
| 267 | $info_midi['keysignature'][] = $keysigs[$keysig_sharpsflats] . ' ' . ((bool)$keysig_majorminor ? 'minor' : 'major'); |
||
| 268 | break; |
||
| 269 | |||
| 270 | case 0x7F: // Sequencer specific information |
||
| 271 | |||
| 272 | $custom_data = substr($meta_event_data, 0, $meta_event_length); |
||
| 273 | break; |
||
| 274 | |||
| 275 | default: |
||
| 276 | |||
| 277 | $getid3->warning('Unhandled META Event Command: ' . $meta_event_command); |
||
| 278 | } |
||
| 279 | } |
||
| 280 | break; |
||
| 281 | |||
| 282 | default: |
||
| 283 | $getid3->warning('Unhandled MIDI Event ID: ' . $midi_events[$track_number][$event_id]['eventid']); |
||
| 284 | } |
||
| 285 | } |
||
| 286 | |||
| 287 | if (($track_number > 0) || (1 == count($track_data_array))) { |
||
| 288 | $info_midi['totalticks'] = max($info_midi['totalticks'], $cumulative_delta_time); |
||
| 289 | } |
||
| 290 | } |
||
| 291 | |||
| 292 | $previous_tick_offset = null; |
||
| 293 | |||
| 294 | ksort($ms_per_quarter_note_after); |
||
| 295 | foreach ($ms_per_quarter_note_after as $tick_offset => $ms_per_beat) { |
||
| 296 | if (is_null($previous_tick_offset)) { |
||
| 297 | $prev_ms_per_beat = $ms_per_beat; |
||
| 298 | $previous_tick_offset = $tick_offset; |
||
| 299 | continue; |
||
| 300 | } |
||
| 301 | |||
| 302 | if ($info_midi['totalticks'] > $tick_offset) { |
||
| 303 | $getid3->info['playtime_seconds'] += (($tick_offset - $previous_tick_offset) / $info_midi_raw['ticksperqnote']) * ($prev_ms_per_beat / 1000000); |
||
| 304 | |||
| 305 | $prev_ms_per_beat = $ms_per_beat; |
||
| 306 | $previous_tick_offset = $tick_offset; |
||
| 307 | } |
||
| 308 | } |
||
| 309 | |||
| 310 | if ($info_midi['totalticks'] > $previous_tick_offset) { |
||
| 311 | $getid3->info['playtime_seconds'] += (($info_midi['totalticks'] - $previous_tick_offset) / $info_midi_raw['ticksperqnote']) * ($ms_per_beat / 1000000); |
||
| 312 | } |
||
| 313 | |||
| 314 | if (@$getid3->info['playtime_seconds'] > 0) { |
||
| 315 | $getid3->info['bitrate'] = (($getid3->info['avdataend'] - $getid3->info['avdataoffset']) * 8) / $getid3->info['playtime_seconds']; |
||
| 316 | } |
||
| 317 | |||
| 318 | if (!empty($info_midi['lyrics'])) { |
||
| 319 | $info_midi['comments']['lyrics'][] = $info_midi['lyrics']; |
||
| 320 | } |
||
| 321 | |||
| 322 | return true; |
||
| 323 | } |
||
| 520 |