//| Copyright: (C) 2020-2024 Kevin Larke //| License: GNU GPL version 3.0 or above. See the accompanying LICENSE file. #ifndef cwIoMidiRecordPlay_h #define cwIoMidiRecordPlay_h namespace cw { namespace midi_record_play { typedef handle< struct midi_record_play_str > handle_t; typedef struct midi_msg_str { unsigned id; time::spec_t timestamp; unsigned loc; const void* arg; uint8_t ch; uint8_t status; uint8_t d0; uint8_t d1; } midi_msg_t; enum { kMidiEventActionId, kPlayerStoppedActionId }; typedef void (*event_callback_t)( void* arg, unsigned actionId, unsigned id, const time::spec_t timestamp, unsigned loc, const void* msg_arg, uint8_t ch, uint8_t status, uint8_t d0, uint8_t d1 ); enum { kSampler_MRP_DevIdx = 0, kPiano_MRP_DevIdx = 1 }; rc_t create( handle_t& hRef, io::handle_t ioH, const object_t& cfg, const char* velTableFname=nullptr, event_callback_t cb=nullptr, void* cb_arg=nullptr ); rc_t destroy( handle_t& hRef ); // Set rewindFl to play from start, otherwise play from current output location. rc_t start( handle_t h, bool rewindFl=true, const time::spec_t* end_play_event_timestamp=nullptr ); rc_t stop( handle_t h ); bool is_started( handle_t h ); rc_t rewind( handle_t h ); rc_t clear( handle_t h ); rc_t set_record_state( handle_t h, bool record_fl ); bool record_state( handle_t h ); rc_t set_mute_state( handle_t h, bool record_fl ); bool mute_state( handle_t h ); rc_t set_thru_state( handle_t h, bool record_thru ); bool thru_state( handle_t h ); rc_t save( handle_t h, const char* fn ); rc_t save_csv( handle_t h, const char* fn ); rc_t save_synced_csv( handle_t h, const char* fn, const score_follower::ssf_note_on_t* syncA, unsigned syncN ); rc_t write_svg( handle_t h, const char* fn ); rc_t open( handle_t h, const char* fn ); // Load the playback buffer with messages to output. rc_t load( handle_t h, const midi_msg_t* msg, unsigned msg_count ); rc_t seek( handle_t h, time::spec_t timestamp ); unsigned elapsed_micros( handle_t h ); unsigned event_count( handle_t h ); // Current count of stored messages. unsigned event_index( handle_t h ); // record mode: index of next event to store play mode:index of next event to play unsigned event_loc( handle_t h ); // play mode: loc of next event to play record mode:kInvalidId unsigned last_store_index( handle_t h ); rc_t exec( handle_t h, const io::msg_t& msg ); unsigned label_to_device_index( handle_t h, const char* devLabel ); const char* device_index_to_label( handle_t h, unsigned devIdx ); unsigned device_count( handle_t h ); bool is_device_enabled( handle_t h, unsigned devIdx ); void enable_device( handle_t h, unsigned devIdx, bool enableFl ); rc_t send_midi_msg( handle_t h, unsigned devIdx, uint8_t ch, uint8_t status, uint8_t d0, uint8_t d1 ); void half_pedal_params( handle_t h, unsigned noteDelayMs, unsigned pitch, unsigned vel, unsigned pedal_vel, unsigned noteDurMs, unsigned downDelayMs ); // Convert an audio-midi file to a MIDI file and/or a CSV file rc_t am_to_midi_file( const char* am_filename, const char* midi_filename = nullptr, const char* csv_filename = nullptr ); rc_t am_to_midi_dir( const char* player_name, const char* prefix_label, unsigned sess_numb, unsigned beg_loc, unsigned end_loc, const char* inDir ); rc_t am_to_midi_file( const object_t* cfg ); // See main.cfg: 'am_to_midi_file' unsigned dev_count( handle_t h ); // Set devIdx to kInvalididx to disable vel tables on all devices rc_t vel_table_disable( handle_t h, unsigned devIdx=kInvalidIdx ); unsigned vel_table_count( handle_t h, unsigned devIdx ); const uint8_t* vel_table( handle_t h, unsigned devIdx ); rc_t vel_table_set( handle_t h, unsigned devIdx, const uint8_t* tbl, unsigned tblN ); void report( handle_t h ); } } #endif