cmScoreMatchGraphic.h/c : Added cmScoreMatchGraphicUpdateMidiFromScore() to set the MIDI note velocities based on the score velocities.

This commit is contained in:
kevin 2016-06-16 12:55:34 -04:00
parent 670edcaec7
commit 3acdeb61a0
2 changed files with 71 additions and 4 deletions

View File

@ -90,6 +90,7 @@ typedef struct
{ {
cmErr_t err; cmErr_t err;
cmChar_t* scFn;
cmSmgSc_t* scV; // scV[scN] score bars and notes cmSmgSc_t* scV; // scV[scN] score bars and notes
unsigned scN; unsigned scN;
@ -100,6 +101,7 @@ typedef struct
// (Each match after the first gets a line from the box representing the midi event // (Each match after the first gets a line from the box representing the midi event
// to the matching score event) // to the matching score event)
cmChar_t* mfFn; // MIDI file name
cmSmgMidi_t* mV; // mV[mN] midi note-on events cmSmgMidi_t* mV; // mV[mN] midi note-on events
unsigned mN; unsigned mN;
double mfDurSecs; // midi file duration in seconds double mfDurSecs; // midi file duration in seconds
@ -121,6 +123,7 @@ cmSmgRC_t _cmSmgFree( cmSmg_t* p )
{ {
unsigned i; unsigned i;
for(i=0; i<p->mN; ++i) for(i=0; i<p->mN; ++i)
{ {
cmSmgMatch_t* m0 = p->mV[i].matchV; cmSmgMatch_t* m0 = p->mV[i].matchV;
@ -156,6 +159,8 @@ cmSmgRC_t _cmSmgFree( cmSmg_t* p )
l0 = l1; l0 = l1;
} }
cmMemFree(p->scFn);
cmMemFree(p->mfFn);
cmMemFree(p->scV); cmMemFree(p->scV);
cmMemFree(p->mV); cmMemFree(p->mV);
cmMemFree(p->locV); cmMemFree(p->locV);
@ -198,6 +203,7 @@ cmSmgRC_t _cmSmgInitFromScore( cmCtx_t* ctx, cmSmg_t* p, const cmChar_t* scoreFn
if( cmScoreInitialize(ctx,&scH,scoreFn,44100.0, NULL, 0, NULL, NULL, cmSymTblNullHandle ) != kOkScRC ) if( cmScoreInitialize(ctx,&scH,scoreFn,44100.0, NULL, 0, NULL, NULL, cmSymTblNullHandle ) != kOkScRC )
return cmErrMsg(&p->err,kScoreFailSmgRC,"Score initializatio failed on '%s'.",cmStringNullGuard(scoreFn)); return cmErrMsg(&p->err,kScoreFailSmgRC,"Score initializatio failed on '%s'.",cmStringNullGuard(scoreFn));
p->scFn = cmMemAllocStr(scoreFn);
p->scN = cmScoreEvtCount(scH); p->scN = cmScoreEvtCount(scH);
p->scV = cmMemAllocZ(cmSmgSc_t,p->scN); p->scV = cmMemAllocZ(cmSmgSc_t,p->scN);
@ -269,6 +275,7 @@ cmSmgRC_t _cmSmgInitFromMidi( cmCtx_t* ctx, cmSmg_t* p, const cmChar_t* midiFn )
p->mV = cmMemAllocZ(cmSmgMidi_t,mN); p->mV = cmMemAllocZ(cmSmgMidi_t,mN);
p->mN = mN; p->mN = mN;
p->mfDurSecs = cmMidiFileDurSecs(mfH); p->mfDurSecs = cmMidiFileDurSecs(mfH);
p->mfFn = cmMemAllocStr(midiFn);
for(i=0,j=0; i<mN; ++i) for(i=0,j=0; i<mN; ++i)
if( (mV[i]!=NULL) && cmMidiIsChStatus(mV[i]->status) && cmMidiIsNoteOn(mV[i]->status) && (mV[i]->u.chMsgPtr->d1>0) ) if( (mV[i]!=NULL) && cmMidiIsChStatus(mV[i]->status) && cmMidiIsNoteOn(mV[i]->status) && (mV[i]->u.chMsgPtr->d1>0) )
@ -606,3 +613,57 @@ cmSmgRC_t cmScoreMatchGraphicGenTimeLineBars( cmSmgH_t h, const cmChar_t* fn, un
return rc; return rc;
} }
cmSmgRC_t cmScoreMatchGraphicUpdateMidiFromScore( cmCtx_t* ctx, cmSmgH_t h, const cmChar_t* newMidiFn )
{
cmSmgRC_t rc = kOkSmgRC;
cmSmg_t* p = _cmSmgHandleToPtr(h);
unsigned i = 0;
cmMidiFileH_t mfH = cmMidiFileNullHandle;
cmScH_t scH = cmScNullHandle;
if( cmMidiFileOpen(ctx, &mfH, p->mfFn ) != kOkMfRC )
return cmErrMsg(&p->err,kMidiFileFailSmgRC,"MIDI file open failed on '%s'.",cmStringNullGuard(p->mfFn));
if( cmScoreInitialize(ctx,&scH,p->scFn,44100.0, NULL, 0, NULL, NULL, cmSymTblNullHandle ) != kOkScRC )
{
rc = cmErrMsg(&p->err,kScoreFailSmgRC,"Score initializatio failed on '%s'.",cmStringNullGuard(p->scFn));
goto errLabel;
}
for(i=0; i<p->mN; ++i)
{
cmSmgMidi_t* mr = p->mV + i;
// only update midi events which were matched exactly once
if( mr->matchV==NULL || mr->matchV->link!=NULL )
continue;
// locate the matched score event
const cmScoreEvt_t* s= cmScoreIdToEvt( scH, mr->matchV->score->csvEventId );
assert( s!=NULL );
// assign the score velocity to the MIDI file
if(cmMidiFileSetVelocity( mfH, mr->uid, s->vel ) != kOkMfRC )
{
rc = cmErrMsg(&p->err,kOpFailSmgRC,"Set velocify operation failed.");
goto errLabel;
}
}
// write the updated MIDI file
if( cmMidiFileWrite( mfH, newMidiFn ) != kOkMfRC )
{
rc = cmErrMsg(&p->err,kMidiFileFailSmgRC,"MIDI file write failed on '%s'.",cmStringNullGuard(newMidiFn));
goto errLabel;
}
errLabel:
cmMidiFileClose(&mfH);
cmScoreFinalize(&scH);
return rc;
}

View File

@ -10,7 +10,8 @@ extern "C" {
kOkSmgRC = cmOkRC, kOkSmgRC = cmOkRC,
kFileSmgRC, kFileSmgRC,
kScoreFailSmgRC, kScoreFailSmgRC,
kMidiFileFailSmgRC kMidiFileFailSmgRC,
kOpFailSmgRC
}; };
typedef cmRC_t cmSmgRC_t; typedef cmRC_t cmSmgRC_t;
@ -28,6 +29,11 @@ extern "C" {
// beginning at each bar line and ends at the end of the file. // beginning at each bar line and ends at the end of the file.
cmSmgRC_t cmScoreMatchGraphicGenTimeLineBars( cmSmgH_t h, const cmChar_t* fn, unsigned srate ); cmSmgRC_t cmScoreMatchGraphicGenTimeLineBars( cmSmgH_t h, const cmChar_t* fn, unsigned srate );
// Update the MIDI file velocity values and insert pedal events
// from from score into MIDI file and then write the updated MIDI
// file to 'newMidiFn'.
cmSmgRC_t cmScoreMatchGraphicUpdateMidiFromScore( cmCtx_t* ctx, cmSmgH_t h, const cmChar_t* newMidiFn );
#ifdef __cplusplus #ifdef __cplusplus
} }
#endif #endif