libcm is a C development framework with an emphasis on audio signal processing applications.
Du kannst nicht mehr als 25 Themen auswählen Themen müssen mit entweder einem Buchstaben oder einer Ziffer beginnen. Sie können Bindestriche („-“) enthalten und bis zu 35 Zeichen lang sein.

cmXScore.c 130KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401140214031404140514061407140814091410141114121413141414151416141714181419142014211422142314241425142614271428142914301431143214331434143514361437143814391440144114421443144414451446144714481449145014511452145314541455145614571458145914601461146214631464146514661467146814691470147114721473147414751476147714781479148014811482148314841485148614871488148914901491149214931494149514961497149814991500150115021503150415051506150715081509151015111512151315141515151615171518151915201521152215231524152515261527152815291530153115321533153415351536153715381539154015411542154315441545154615471548154915501551155215531554155515561557155815591560156115621563156415651566156715681569157015711572157315741575157615771578157915801581158215831584158515861587158815891590159115921593159415951596159715981599160016011602160316041605160616071608160916101611161216131614161516161617161816191620162116221623162416251626162716281629163016311632163316341635163616371638163916401641164216431644164516461647164816491650165116521653165416551656165716581659166016611662166316641665166616671668166916701671167216731674167516761677167816791680168116821683168416851686168716881689169016911692169316941695169616971698169917001701170217031704170517061707170817091710171117121713171417151716171717181719172017211722172317241725172617271728172917301731173217331734173517361737173817391740174117421743174417451746174717481749175017511752175317541755175617571758175917601761176217631764176517661767176817691770177117721773177417751776177717781779178017811782178317841785178617871788178917901791179217931794179517961797179817991800180118021803180418051806180718081809181018111812181318141815181618171818181918201821182218231824182518261827182818291830183118321833183418351836183718381839184018411842184318441845184618471848184918501851185218531854185518561857185818591860186118621863186418651866186718681869187018711872187318741875187618771878187918801881188218831884188518861887188818891890189118921893189418951896189718981899190019011902190319041905190619071908190919101911191219131914191519161917191819191920192119221923192419251926192719281929193019311932193319341935193619371938193919401941194219431944194519461947194819491950195119521953195419551956195719581959196019611962196319641965196619671968196919701971197219731974197519761977197819791980198119821983198419851986198719881989199019911992199319941995199619971998199920002001200220032004200520062007200820092010201120122013201420152016201720182019202020212022202320242025202620272028202920302031203220332034203520362037203820392040204120422043204420452046204720482049205020512052205320542055205620572058205920602061206220632064206520662067206820692070207120722073207420752076207720782079208020812082208320842085208620872088208920902091209220932094209520962097209820992100210121022103210421052106210721082109211021112112211321142115211621172118211921202121212221232124212521262127212821292130213121322133213421352136213721382139214021412142214321442145214621472148214921502151215221532154215521562157215821592160216121622163216421652166216721682169217021712172217321742175217621772178217921802181218221832184218521862187218821892190219121922193219421952196219721982199220022012202220322042205220622072208220922102211221222132214221522162217221822192220222122222223222422252226222722282229223022312232223322342235223622372238223922402241224222432244224522462247224822492250225122522253225422552256225722582259226022612262226322642265226622672268226922702271227222732274227522762277227822792280228122822283228422852286228722882289229022912292229322942295229622972298229923002301230223032304230523062307230823092310231123122313231423152316231723182319232023212322232323242325232623272328232923302331233223332334233523362337233823392340234123422343234423452346234723482349235023512352235323542355235623572358235923602361236223632364236523662367236823692370237123722373237423752376237723782379238023812382238323842385238623872388238923902391239223932394239523962397239823992400240124022403240424052406240724082409241024112412241324142415241624172418241924202421242224232424242524262427242824292430243124322433243424352436243724382439244024412442244324442445244624472448244924502451245224532454245524562457245824592460246124622463246424652466246724682469247024712472247324742475247624772478247924802481248224832484248524862487248824892490249124922493249424952496249724982499250025012502250325042505250625072508250925102511251225132514251525162517251825192520252125222523252425252526252725282529253025312532253325342535253625372538253925402541254225432544254525462547254825492550255125522553255425552556255725582559256025612562256325642565256625672568256925702571257225732574257525762577257825792580258125822583258425852586258725882589259025912592259325942595259625972598259926002601260226032604260526062607260826092610261126122613261426152616261726182619262026212622262326242625262626272628262926302631263226332634263526362637263826392640264126422643264426452646264726482649265026512652265326542655265626572658265926602661266226632664266526662667266826692670267126722673267426752676267726782679268026812682268326842685268626872688268926902691269226932694269526962697269826992700270127022703270427052706270727082709271027112712271327142715271627172718271927202721272227232724272527262727272827292730273127322733273427352736273727382739274027412742274327442745274627472748274927502751275227532754275527562757275827592760276127622763276427652766276727682769277027712772277327742775277627772778277927802781278227832784278527862787278827892790279127922793279427952796279727982799280028012802280328042805280628072808280928102811281228132814281528162817281828192820282128222823282428252826282728282829283028312832283328342835283628372838283928402841284228432844284528462847284828492850285128522853285428552856285728582859286028612862286328642865286628672868286928702871287228732874287528762877287828792880288128822883288428852886288728882889289028912892289328942895289628972898289929002901290229032904290529062907290829092910291129122913291429152916291729182919292029212922292329242925292629272928292929302931293229332934293529362937293829392940294129422943294429452946294729482949295029512952295329542955295629572958295929602961296229632964296529662967296829692970297129722973297429752976297729782979298029812982298329842985298629872988298929902991299229932994299529962997299829993000300130023003300430053006300730083009301030113012301330143015301630173018301930203021302230233024302530263027302830293030303130323033303430353036303730383039304030413042304330443045304630473048304930503051305230533054305530563057305830593060306130623063306430653066306730683069307030713072307330743075307630773078307930803081308230833084308530863087308830893090309130923093309430953096309730983099310031013102310331043105310631073108310931103111311231133114311531163117311831193120312131223123312431253126312731283129313031313132313331343135313631373138313931403141314231433144314531463147314831493150315131523153315431553156315731583159316031613162316331643165316631673168316931703171317231733174317531763177317831793180318131823183318431853186318731883189319031913192319331943195319631973198319932003201320232033204320532063207320832093210321132123213321432153216321732183219322032213222322332243225322632273228322932303231323232333234323532363237323832393240324132423243324432453246324732483249325032513252325332543255325632573258325932603261326232633264326532663267326832693270327132723273327432753276327732783279328032813282328332843285328632873288328932903291329232933294329532963297329832993300330133023303330433053306330733083309331033113312331333143315331633173318331933203321332233233324332533263327332833293330333133323333333433353336333733383339334033413342334333443345334633473348334933503351335233533354335533563357335833593360336133623363336433653366336733683369337033713372337333743375337633773378337933803381338233833384338533863387338833893390339133923393339433953396339733983399340034013402340334043405340634073408340934103411341234133414341534163417341834193420342134223423342434253426342734283429343034313432343334343435343634373438343934403441344234433444344534463447344834493450345134523453345434553456345734583459346034613462346334643465346634673468346934703471347234733474347534763477347834793480348134823483348434853486348734883489349034913492349334943495349634973498349935003501350235033504350535063507350835093510351135123513351435153516351735183519352035213522352335243525352635273528352935303531353235333534353535363537353835393540354135423543354435453546354735483549355035513552355335543555355635573558355935603561356235633564356535663567356835693570357135723573357435753576357735783579358035813582358335843585358635873588358935903591359235933594359535963597359835993600360136023603360436053606360736083609361036113612361336143615361636173618361936203621362236233624362536263627362836293630363136323633363436353636363736383639364036413642364336443645364636473648364936503651365236533654365536563657365836593660366136623663366436653666366736683669367036713672367336743675367636773678367936803681368236833684368536863687368836893690369136923693369436953696369736983699370037013702370337043705370637073708370937103711371237133714371537163717371837193720372137223723372437253726372737283729373037313732373337343735373637373738373937403741374237433744374537463747374837493750375137523753375437553756375737583759376037613762376337643765376637673768376937703771377237733774377537763777377837793780378137823783378437853786378737883789379037913792379337943795379637973798379938003801380238033804380538063807380838093810381138123813381438153816381738183819382038213822382338243825382638273828382938303831383238333834383538363837383838393840384138423843384438453846384738483849385038513852385338543855385638573858385938603861386238633864386538663867386838693870387138723873387438753876387738783879388038813882388338843885388638873888388938903891389238933894389538963897389838993900390139023903390439053906390739083909391039113912391339143915391639173918391939203921392239233924392539263927392839293930393139323933393439353936393739383939394039413942394339443945394639473948394939503951395239533954395539563957395839593960396139623963396439653966396739683969397039713972397339743975397639773978397939803981398239833984398539863987398839893990399139923993399439953996399739983999400040014002400340044005400640074008400940104011401240134014401540164017401840194020402140224023402440254026402740284029403040314032403340344035403640374038403940404041404240434044404540464047404840494050405140524053405440554056405740584059406040614062406340644065406640674068406940704071407240734074407540764077407840794080408140824083408440854086408740884089409040914092409340944095409640974098409941004101410241034104410541064107410841094110411141124113411441154116411741184119412041214122412341244125412641274128412941304131413241334134413541364137413841394140414141424143414441454146414741484149415041514152415341544155415641574158415941604161416241634164416541664167416841694170417141724173417441754176417741784179418041814182418341844185418641874188418941904191419241934194419541964197419841994200420142024203420442054206420742084209421042114212421342144215421642174218421942204221422242234224422542264227422842294230423142324233423442354236423742384239424042414242424342444245424642474248424942504251425242534254425542564257425842594260426142624263426442654266426742684269427042714272427342744275427642774278427942804281428242834284428542864287428842894290429142924293429442954296429742984299430043014302430343044305430643074308430943104311431243134314431543164317431843194320432143224323432443254326432743284329433043314332433343344335433643374338433943404341434243434344434543464347434843494350435143524353435443554356435743584359436043614362436343644365436643674368436943704371437243734374437543764377437843794380438143824383438443854386438743884389439043914392439343944395439643974398439944004401440244034404440544064407440844094410441144124413441444154416441744184419442044214422442344244425442644274428442944304431443244334434443544364437443844394440444144424443444444454446444744484449445044514452445344544455445644574458445944604461446244634464446544664467446844694470447144724473447444754476447744784479448044814482448344844485448644874488448944904491449244934494449544964497449844994500450145024503450445054506450745084509451045114512451345144515451645174518451945204521452245234524452545264527452845294530453145324533453445354536453745384539454045414542454345444545454645474548454945504551455245534554455545564557455845594560456145624563456445654566456745684569457045714572457345744575457645774578457945804581458245834584458545864587458845894590459145924593459445954596459745984599460046014602460346044605460646074608460946104611461246134614461546164617461846194620462146224623462446254626462746284629463046314632463346344635463646374638463946404641464246434644464546464647464846494650465146524653465446554656465746584659466046614662
  1. #include "cmPrefix.h"
  2. #include "cmGlobal.h"
  3. #include "cmFloatTypes.h"
  4. #include "cmComplexTypes.h"
  5. #include "cmRpt.h"
  6. #include "cmErr.h"
  7. #include "cmCtx.h"
  8. #include "cmMem.h"
  9. #include "cmMallocDebug.h"
  10. #include "cmLinkedHeap.h"
  11. #include "cmXml.h"
  12. #include "cmText.h"
  13. #include "cmFileSys.h"
  14. #include "cmXScore.h"
  15. #include "cmTime.h"
  16. #include "cmMidi.h"
  17. #include "cmMidiFile.h"
  18. #include "cmLex.h"
  19. #include "cmCsv.h"
  20. #include "cmSymTbl.h"
  21. #include "cmScore.h"
  22. #include "cmFile.h"
  23. #include "cmSymTbl.h"
  24. #include "cmAudioFile.h"
  25. #include "cmAudioFile.h"
  26. #include "cmProcObj.h"
  27. #include "cmProcTemplate.h"
  28. #include "cmProc.h"
  29. #include "cmProc2.h"
  30. #include "cmProc5.h"
  31. #include "cmSvgWriter.h"
  32. cmXsH_t cmXsNullHandle = cmSTATIC_NULL_HANDLE;
  33. enum
  34. {
  35. kSectionXsFl = 0x00000001, // rvalue holds section number
  36. kBarXsFl = 0x00000002,
  37. kRestXsFl = 0x00000004,
  38. kGraceXsFl = 0x00000008,
  39. kDotXsFl = 0x00000010,
  40. kChordXsFl = 0x00000020,
  41. kDynXsFl = 0x00000040,
  42. kEvenXsFl = 0x00000080,
  43. kTempoXsFl = 0x00000100,
  44. kHeelXsFl = 0x00000200,
  45. kTieBegXsFl = 0x00000400,
  46. kTieEndXsFl = 0x00000800,
  47. kTieProcXsFl = 0x00001000,
  48. kDampDnXsFl = 0x00002000,
  49. kDampUpXsFl = 0x00004000,
  50. kDampUpDnXsFl = 0x00008000,
  51. kSostDnXsFl = 0x00010000,
  52. kSostUpXsFl = 0x00020000,
  53. kMetronomeXsFl = 0x00040000, // duration holds BPM
  54. kOnsetXsFl = 0x00080000, // this is a sounding note
  55. kBegGroupXsFl = 0x00100000,
  56. kEndGroupXsFl = 0x00200000,
  57. kBegGraceXsFl = 0x00400000, // (b) beg grace note group
  58. kEndGraceXsFl = 0x00800000, // end grace note group
  59. kAddGraceXsFl = 0x01000000, // (a) end grace note group operator flag - add time
  60. kSubGraceXsFl = 0x02000000, // (s) " " " " " " - subtract time
  61. kAFirstGraceXsFl = 0x04000000, // (A) add time after first note
  62. kNFirstGraceXsFl = 0x08000000, // (n) grace notes start as soon as possible after first note and add time
  63. kDeleteXsFl = 0x10000000,
  64. kDynBegForkXsFl = 0x20000000,
  65. kDynEndForkXsFl = 0x40000000,
  66. kDynEndXsFl = 0x100000000,
  67. kEvenEndXsFl = 0x200000000,
  68. kTempoEndXsFl = 0x400000000
  69. };
  70. struct cmXsMeas_str;
  71. struct cmXsVoice_str;
  72. // Values measured for each sounding note in the preceding time window....
  73. typedef struct cmXsComplexity_str
  74. {
  75. unsigned sum_d_vel; // sum of first order difference of cmXsNote_t.dynamics
  76. unsigned sum_d_rym; // sum of first order difference of cmXsNote_t.rvalue
  77. unsigned sum_d_lpch; // sum of first order difference of cmXsNote_t.pitch of note assigned to the bass cleff
  78. unsigned sum_n_lpch; // count of notes assigned to the bass cleff
  79. unsigned sum_d_rpch; // sum of first order difference of cmXsNote_t.pitch of note assigned to the treble cleff
  80. unsigned sum_n_rpch; // count of notes assigned to the treble cleff
  81. } cmXsComplexity_t;
  82. typedef struct cmXsNote_str
  83. {
  84. unsigned uid; // unique id of this note record
  85. unsigned long long flags; // See k???XsFl
  86. unsigned pitch; // midi pitch
  87. unsigned dynamics; // dynamic level 1=pppp 9=fff
  88. unsigned vel; // score specified MIDI velocity
  89. cmChar_t step; // A-G
  90. unsigned octave; // sci pitch octave
  91. int alter; // +n=sharps,-n=flats
  92. unsigned staff; // 1=treble 2=bass
  93. unsigned tick; //
  94. unsigned duration; // duration in ticks
  95. unsigned tied_dur; // duration in ticks (including all tied notes)
  96. double secs; // absolute time in seconds
  97. double dsecs; // delta time in seconds since previous event
  98. unsigned locIdx; // location index (chords share the same location index)
  99. double rvalue; // 1/rvalue = rythmic value (1/0.5 double whole 1/1 whole 1/2 half 1/4=quarter note, 1/8=eighth note, ...)
  100. const cmChar_t* tvalue; // text value
  101. const cmChar_t* editStr; // merged manual edit string
  102. unsigned evenGroupId; // eveness group id
  103. unsigned dynGroupId; // dynamics group id
  104. unsigned tempoGroupId; // tempo group id
  105. unsigned graceGroupId; // grace note group id
  106. struct cmXsVoice_str* voice; // voice this note belongs to
  107. struct cmXsMeas_str* meas; // measure this note belongs to
  108. const cmXmlNode_t* xmlNode; // note xml ptr
  109. struct cmXsNote_str* tied; // subsequent note tied to this note
  110. struct cmXsNote_str* grace; // grace note groups link backward in time from the anchor note
  111. struct cmXsNote_str* mlink; // measure note list
  112. struct cmXsNote_str* slink; // time sorted event list
  113. cmXsComplexity_t cplx;
  114. } cmXsNote_t;
  115. typedef struct cmXsVoice_str
  116. {
  117. unsigned id; // Voice id
  118. cmXsNote_t* noteL; // List of notes in this voice
  119. struct cmXsVoice_str* link; // Link to other voices in this measure
  120. } cmXsVoice_t;
  121. typedef struct cmXsSpan_str
  122. {
  123. unsigned staff;
  124. unsigned number;
  125. struct cmXsMeas_str* meas;
  126. unsigned tick0;
  127. unsigned tick1;
  128. int pitch_offset;
  129. struct cmXsSpan_str* link;
  130. } cmXsSpan_t;
  131. typedef struct cmXsMeas_str
  132. {
  133. unsigned number; // Measure number
  134. unsigned divisions; // ticks-per-quarter-note
  135. unsigned beats; // beats per measure
  136. unsigned beat_type; // whole/half/quarter/eighth ...
  137. cmXsVoice_t* voiceL; // List of voices in this measure
  138. cmXsNote_t* noteL; // List of time sorted notes in this measure
  139. struct cmXsMeas_str* link; // Link to other measures in this part.
  140. } cmXsMeas_t;
  141. typedef struct cmXsPart_str
  142. {
  143. const cmChar_t* idStr; // Id of this part
  144. cmXsMeas_t* measL; // List of measures in this part.
  145. struct cmXsPart_str* link; // Link to other parts in this score
  146. } cmXsPart_t;
  147. typedef struct
  148. {
  149. cmErr_t err;
  150. cmXmlH_t xmlH;
  151. cmLHeapH_t lhH;
  152. cmXsPart_t* partL;
  153. cmCsvH_t csvH;
  154. cmXsSpan_t* spanL;
  155. unsigned nextUid;
  156. cmXsComplexity_t cplx_max;
  157. } cmXScore_t;
  158. void _cmXScoreReport( cmXScore_t* p, cmRpt_t* rpt, bool sortFl );
  159. cmXScore_t* _cmXScoreHandleToPtr( cmXsH_t h )
  160. {
  161. cmXScore_t* p = (cmXScore_t*)h.h;
  162. assert( p != NULL );
  163. return p;
  164. }
  165. cmXsRC_t _cmXScoreFinalize( cmXScore_t* p )
  166. {
  167. cmXsRC_t rc = kOkXsRC;
  168. // release the XML file
  169. cmXmlFree( &p->xmlH );
  170. // release the local linked heap memory
  171. cmLHeapDestroy(&p->lhH);
  172. // release the CSV output object
  173. cmCsvFinalize(&p->csvH);
  174. cmMemFree(p);
  175. return rc;
  176. }
  177. cmXsRC_t _cmXScoreMissingNode( cmXScore_t* p, const cmXmlNode_t* parent, const cmChar_t* label )
  178. {
  179. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Missing XML node '%s'. Parent line:%i",label,parent->line);
  180. }
  181. cmXsRC_t _cmXScoreMissingAttribute( cmXScore_t* p, const cmXmlNode_t* np, const cmChar_t* attrLabel )
  182. {
  183. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Missing XML attribute '%s' from node '%s'.",attrLabel,np->label);
  184. }
  185. cmXsVoice_t* _cmXScoreIdToVoice( cmXsMeas_t* meas, unsigned voiceId )
  186. {
  187. cmXsVoice_t* v = meas->voiceL;
  188. for(; v!=NULL; v=v->link)
  189. if( v->id == voiceId )
  190. return v;
  191. return NULL;
  192. }
  193. cmXsMeas_t* _cmXsFindMeas( cmXsPart_t* part, unsigned measNumb )
  194. {
  195. cmXsMeas_t* m = part->measL;
  196. for(; m!=NULL; m=m->link)
  197. if( m->number == measNumb )
  198. return m;
  199. return NULL;
  200. }
  201. cmXsNote_t* _cmXsFindNote( cmXsMeas_t* m, unsigned idx, unsigned midi, double rval, unsigned durtn, unsigned* idxRef )
  202. {
  203. unsigned i;
  204. cmXsNote_t* np = m->noteL;
  205. for(i=0; np!=NULL; np=np->slink,++i)
  206. {
  207. //printf("idx:%i %i midi:%i %i rval:%f %f durtn:%i %i\n", i,idx, np->pitch,midi, np->rvalue,rval, np->tied_dur,durtn);
  208. if( i>=idx && np->pitch==midi && np->rvalue==rval && np->tied_dur==durtn )
  209. {
  210. *idxRef = i;
  211. return np;
  212. }
  213. }
  214. return NULL;
  215. }
  216. cmXsRC_t _cmXScorePushNote( cmXScore_t* p, cmXsMeas_t* meas, unsigned voiceId, cmXsNote_t* note )
  217. {
  218. cmXsVoice_t* v;
  219. // get the voice recd
  220. if((v = _cmXScoreIdToVoice(meas,voiceId)) == NULL)
  221. {
  222. // the voice recd doesn't exist for this voiceId - allocate one
  223. v = cmLhAllocZ(p->lhH,cmXsVoice_t,1);
  224. v->id = voiceId;
  225. // add the voice record to the meas->voiceL
  226. if( meas->voiceL == NULL )
  227. meas->voiceL = v;
  228. else
  229. {
  230. cmXsVoice_t* vp = meas->voiceL;
  231. while( vp->link!=NULL )
  232. vp = vp->link;
  233. vp->link = v;
  234. }
  235. }
  236. // add the note record to the end of meas->voiceL
  237. if( v->noteL == NULL )
  238. v->noteL = note;
  239. else
  240. {
  241. cmXsNote_t* n = v->noteL;
  242. while( n->mlink != NULL )
  243. n = n->mlink;
  244. n->mlink = note;
  245. }
  246. note->voice = v;
  247. note->uid = p->nextUid++;
  248. note->tied_dur = note->duration;
  249. return kOkXsRC;
  250. }
  251. cmXsRC_t _cmXScoreRemoveNote( cmXsNote_t* note )
  252. {
  253. cmXsNote_t* n0 = NULL;
  254. cmXsNote_t* n1 = note->voice->noteL;
  255. unsigned cnt = 0;
  256. for(; n1!=NULL; n1=n1->mlink)
  257. {
  258. if( n1->uid == note->uid )
  259. {
  260. if( n0 == NULL )
  261. note->voice->noteL = NULL;
  262. else
  263. n0->mlink = n1->mlink;
  264. cnt = 1;
  265. break;
  266. }
  267. n0 = n1;
  268. }
  269. n0 = NULL;
  270. n1 = note->meas->noteL;
  271. for(; n1!=NULL; n1=n1->slink)
  272. {
  273. if( n1->uid == note->uid )
  274. {
  275. if( n0 == NULL )
  276. note->voice->noteL = NULL;
  277. else
  278. n0->slink = n1->slink;
  279. cnt = 2;
  280. break;
  281. }
  282. n0 = n1;
  283. }
  284. return cnt == 2 ? kOkXsRC : kSyntaxErrorXsRC;
  285. }
  286. void _cmXScoreInsertNoteBefore( cmXsNote_t* note, cmXsNote_t* nn )
  287. {
  288. assert( note != NULL );
  289. // insert the new note into the voice list before 'note'
  290. cmXsNote_t* n0 = NULL;
  291. cmXsNote_t* n1 = note->voice->noteL;
  292. for(; n1 != NULL; n1=n1->mlink )
  293. {
  294. if( n1->uid == note->uid )
  295. {
  296. if( n0 == NULL )
  297. note->voice->noteL = nn;
  298. else
  299. n0->mlink = nn;
  300. nn->mlink = note;
  301. break;
  302. }
  303. n0 = n1;
  304. }
  305. assert( n1 != NULL );
  306. // insert the new note into the time sorted note list before 'note'
  307. n0 = NULL;
  308. n1 = note->meas->noteL;
  309. for(; n1!=NULL; n1=n1->slink)
  310. {
  311. if( n1->tick >= nn->tick )
  312. {
  313. if( n0 == NULL )
  314. note->meas->noteL = nn;
  315. else
  316. n0->slink = nn;
  317. nn->slink = n1;
  318. break;
  319. }
  320. n0 = n1;
  321. }
  322. assert( n1 != NULL );
  323. }
  324. void _cmXScoreInsertNoteAfter( cmXsNote_t* note, cmXsNote_t* nn )
  325. {
  326. // insert the new note into the voice list after 'note'
  327. cmXsNote_t* n0 = note->voice->noteL;
  328. for(; n0 != NULL; n0=n0->mlink )
  329. if( n0->uid == note->uid )
  330. {
  331. nn->mlink = note->mlink;
  332. note->mlink = nn;
  333. break;
  334. }
  335. assert(n0 != NULL );
  336. // insert the new note into the time sorted note list after 'note'
  337. n0 = note->meas->noteL;
  338. for(; n0!=NULL; n0=n0->slink)
  339. {
  340. if( n0->tick >= nn->tick )
  341. {
  342. nn->slink = n0->slink;
  343. n0->slink = nn;
  344. break;
  345. }
  346. // if n0 is the last ele in the list
  347. if( n0->slink == NULL )
  348. {
  349. n0->slink = nn;
  350. nn->slink = NULL;
  351. break;
  352. }
  353. }
  354. assert(n0 != NULL);
  355. }
  356. cmXsRC_t _cmXScoreParsePartList( cmXScore_t* p )
  357. {
  358. cmXsRC_t rc = kOkXsRC;
  359. cmXsPart_t* lastPartPtr = NULL;
  360. const cmXmlNode_t* xnp;
  361. // find the 'part-list'
  362. if((xnp = cmXmlSearch( cmXmlRoot(p->xmlH), "part-list", NULL, 0)) == NULL )
  363. return _cmXScoreMissingNode(p,cmXmlRoot(p->xmlH),"part-list");
  364. const cmXmlNode_t* cnp = xnp->children;
  365. // for each child of the 'part-list'
  366. for(; cnp!=NULL; cnp=cnp->sibling)
  367. if( cmTextCmp( cnp->label, "score-part" ) == 0 )
  368. {
  369. const cmXmlAttr_t* a;
  370. // find the 'score-part' id
  371. if((a = cmXmlFindAttrib(cnp,"id")) == NULL )
  372. return _cmXScoreMissingAttribute(p,cnp,"id");
  373. // allocate a new part record
  374. cmXsPart_t* pp = cmLhAllocZ(p->lhH,cmXsPart_t,1);
  375. pp->idStr = a->value; // set the part id
  376. // link the new part record to the end of the part list
  377. if(lastPartPtr == NULL)
  378. p->partL = pp;
  379. else
  380. lastPartPtr->link = pp;
  381. lastPartPtr = pp;
  382. }
  383. return rc;
  384. }
  385. cmXsRC_t _cmXScoreParsePitch( cmXScore_t* p, const cmXmlNode_t* nnp, cmXsNote_t* np )
  386. {
  387. cmXsRC_t rc = kOkXsRC;
  388. unsigned octave = 0;
  389. double alter = 0;
  390. const cmChar_t* step = NULL;
  391. if((step = cmXmlNodeValue(nnp,"pitch","step",NULL)) == NULL )
  392. return _cmXScoreMissingNode(p,nnp,"step");
  393. if((rc = cmXmlNodeUInt( nnp,&octave,"pitch","octave",NULL)) != kOkXmlRC )
  394. return _cmXScoreMissingNode(p,nnp,"octave");
  395. cmXmlNodeDouble( nnp,&alter,"pitch","alter",NULL);
  396. int acc = alter;
  397. unsigned midi = cmSciPitchToMidiPitch(*step,acc,octave);
  398. np->pitch = midi;
  399. np->step = *step;
  400. np->octave = octave;
  401. np->alter = alter;
  402. np->flags |= kOnsetXsFl;
  403. return rc;
  404. }
  405. cmXsRC_t _cmXScoreParseNoteRValue( cmXScore_t* p, const cmXmlNode_t* nnp, const cmChar_t* label, double* rvalueRef )
  406. {
  407. typedef struct map_str
  408. {
  409. double rvalue;
  410. const cmChar_t* label;
  411. } map_t;
  412. map_t mapV[] =
  413. {
  414. {-1.0, "measure" }, // whole measure rest
  415. { 0.5, "breve" }, // double whole
  416. { 1.0, "whole" },
  417. { 2.0, "half" },
  418. { 4.0, "quarter" },
  419. { 8.0, "eighth" },
  420. {16.0, "16th" },
  421. {32.0, "32nd" },
  422. { 64.0, "64th" },
  423. {128.0, "128th" },
  424. { 0.0, "" }
  425. };
  426. const cmChar_t* str;
  427. // get the XML rvalue label
  428. if((str = cmXmlNodeValue(nnp,label,NULL)) == NULL)
  429. {
  430. if((nnp = cmXmlSearch(nnp,"rest",NULL,0)) != NULL )
  431. {
  432. const cmXmlAttr_t* a;
  433. if((a = cmXmlFindAttrib(nnp,"measure")) != NULL && cmTextCmp(a->value,"yes")==0)
  434. {
  435. *rvalueRef = -1;
  436. return kOkXsRC;
  437. }
  438. }
  439. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"The 'beat-unit' metronome value is missing on line %i.",nnp->line);
  440. }
  441. unsigned i;
  442. // lookup the rvalue numeric value from the mapV[] table
  443. for(i=0; mapV[i].rvalue!=0; ++i)
  444. if( cmTextCmp(mapV[i].label,str) == 0 )
  445. {
  446. *rvalueRef = mapV[i].rvalue;
  447. return kOkXsRC;
  448. }
  449. // the rvalue label was not found
  450. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unknown rvalue type='%s'.",str);
  451. }
  452. cmXsRC_t _cmXScoreParseColor( cmXScore_t* p, const cmXmlNode_t* nnp, cmXsNote_t* note )
  453. {
  454. cmXsRC_t rc = kOkXsRC;
  455. const cmXmlAttr_t* a;
  456. typedef struct map_str
  457. {
  458. unsigned long long value;
  459. const cmChar_t* label;
  460. } map_t;
  461. map_t mapV[] =
  462. {
  463. { kEvenXsFl, "#0000FF" }, // blue (even)
  464. { kEvenXsFl | kEvenEndXsFl, "#0000FE" }, // blue (even end)
  465. { kEvenXsFl | kEvenEndXsFl, "#0000FD" }, // blue (even end)
  466. { kTempoXsFl, "#00FF00" }, // green (tempo)
  467. { kTempoXsFl | kTempoEndXsFl, "#00FE00" }, // green (tempo end)
  468. { kDynXsFl, "#FF0000" }, // red (dynamics)
  469. { kDynXsFl | kDynEndXsFl, "#FE0000" }, // red (dynamics end)
  470. { kDynXsFl | kDynEndXsFl, "#FD0000" }, // red (dynamics end)
  471. { kTempoXsFl | kEvenXsFl, "#00FFFF" }, // green + blue (turquoise)
  472. { kTempoXsFl | kEvenXsFl | kEvenEndXsFl, "#00FFFE" }, // green + blue (turquoise) (end)
  473. { kDynXsFl | kEvenXsFl, "#FF00FF" }, // red + blue
  474. { kDynXsFl | kEvenXsFl | kEvenEndXsFl, "#FF00FE" }, // red + blue (end)
  475. { kDynXsFl | kEvenXsFl | kEvenEndXsFl, "#FF00FD" }, // red + blue (end)
  476. { kDynXsFl | kEvenXsFl, "#FF0CF7" }, // magenta (even+dyn)
  477. { kDynXsFl | kTempoXsFl, "#FF7F00" }, // red + green (brown)
  478. { kDynXsFl | kTempoXsFl, "#FE7F00" }, // red + green (brown) (end)
  479. { kDynXsFl | kTempoXsFl, "#FD7F00" }, // red + green (brown) (end)
  480. { kDynXsFl | kTempoXsFl | kTempoEndXsFl, "#FF7E00" }, //
  481. { kDynXsFl | kDynEndXsFl | kEvenXsFl | kEvenEndXsFl, "#FE00FE" }, //
  482. { kDynXsFl | kDynEndXsFl | kEvenXsFl, "#FE00FF" },
  483. { kTempoXsFl | kEvenXsFl | kDynXsFl, "#996633" }, // (purple)
  484. { kTempoXsFl | kEvenXsFl | kDynXsFl | kDynEndXsFl, "#996632" }, // (purple)
  485. { kDynXsFl, "#FF6A03" }, // 176 orange (dynamics)
  486. { kEvenXsFl, "#2F00E8" }, // 1001 blue (even)
  487. { kTempoXsFl, "#01CD1F" }, // 1196 green (tempo)
  488. { kEvenXsFl, "#3600E8" }, // 1627 blue (even)
  489. { kDynXsFl | kTempoXsFl, "#9E8F15" }, // 8827 brown (dyn + tempo)
  490. { kEvenXsFl, "#2E00E6" }, // 5393 blue (even)
  491. { kEvenXsFl, "#2C00DD" }, // 5895 blue (even)
  492. { kDynXsFl, "#FF5B03" }, // 6498 orange (dyn)
  493. { kDynXsFl, "#FF6104" }, // 6896 orange
  494. { kEvenXsFl, "#2A00E6" }, // 7781 blue
  495. { kEvenXsFl, "#2300DD" }, // 8300 blue (even)
  496. { kTempoXsFl, "#03CD22" }, // 10820 green (tempo)
  497. { kEvenXsFl, "#3400DB" }, // 11627 blue (dyn)
  498. { -1, "" }
  499. };
  500. /*
  501. orange #FF6A03
  502. magenta #FF0CF7
  503. blue #2F00E8
  504. green #01CD1F
  505. gold #9E8F15
  506. green #03CD22
  507. */
  508. if((a = cmXmlFindAttrib(nnp, "color" )) != NULL )
  509. {
  510. unsigned i;
  511. for(i=0; mapV[i].value != -1; ++i)
  512. if( cmTextCmp(a->value,mapV[i].label) == 0 )
  513. {
  514. note->flags += mapV[i].value;
  515. break;
  516. }
  517. if( mapV[i].value == -1 )
  518. cmErrMsg(&p->err,kSyntaxErrorXsRC,"The note color '%s' was not found on line %i.",a->value,nnp->line);
  519. }
  520. return rc;
  521. }
  522. // On input tick0Ref is set to the tick of the previous event.
  523. // On input tickRef is set to the tick of this event.
  524. // On output tick0Ref is set to the tick of this event.
  525. // On output tickRef is set to the tick of the next event.
  526. cmXsRC_t _cmXScoreParseNote(cmXScore_t* p, cmXsMeas_t* meas, const cmXmlNode_t* nnp, unsigned* tick0Ref, unsigned* tickRef )
  527. {
  528. cmXsRC_t rc = kOkXsRC;
  529. cmXsNote_t* note = cmLhAllocZ(p->lhH,cmXsNote_t,1);
  530. unsigned voiceId;
  531. note->pitch = kInvalidMidiPitch;
  532. note->meas = meas;
  533. note->xmlNode = nnp;
  534. // get the voice id for this node
  535. if( cmXmlNodeUInt(nnp,&voiceId,"voice",NULL) != kOkXmlRC )
  536. return _cmXScoreMissingNode(p,nnp,"voice");
  537. // if this note has a pitch
  538. if( cmXmlNodeHasChild(nnp,"pitch",NULL) )
  539. if((rc = _cmXScoreParsePitch(p,nnp,note)) != kOkXsRC )
  540. return rc;
  541. cmXmlNodeUInt(nnp,&note->duration,"duration",NULL); // get the note duration
  542. cmXmlNodeUInt(nnp,&note->staff,"staff",NULL); // get th staff number
  543. // is 'rest'
  544. if( cmXmlNodeHasChild(nnp,"rest",NULL) )
  545. note->flags |= kRestXsFl;
  546. // is 'grace'
  547. if( cmXmlNodeHasChild(nnp,"grace",NULL) )
  548. note->flags |= kGraceXsFl;
  549. // is 'dot'
  550. if( cmXmlNodeHasChild(nnp,"dot",NULL) )
  551. note->flags |= kDotXsFl;
  552. // is 'chord'
  553. if( cmXmlNodeHasChild(nnp,"chord",NULL) )
  554. note->flags |= kChordXsFl;
  555. // is this is first note in a tied pair
  556. if( cmXmlNodeHasChildWithAttrAndValue(nnp,"tie","type","start",NULL) )
  557. note->flags |= kTieBegXsFl;
  558. // is this is second note in a tied pair
  559. if( cmXmlNodeHasChildWithAttrAndValue(nnp,"tie","type","stop",NULL) )
  560. note->flags |= kTieEndXsFl;
  561. // has 'heel' mark
  562. if( cmXmlNodeHasChild(nnp,"notations","technical","heel",NULL) )
  563. note->flags |= kHeelXsFl;
  564. // set color coded flags
  565. if((rc = _cmXScoreParseColor(p, nnp, note )) != kOkXsRC )
  566. return rc;
  567. // get the note's rythmic value
  568. if((rc = _cmXScoreParseNoteRValue(p,nnp,"type",&note->rvalue)) != kOkXsRC )
  569. return rc;
  570. // if this is a chord note
  571. if( cmIsFlag(note->flags,kChordXsFl) )
  572. {
  573. note->tick = *tick0Ref; // then use the onset time from the previous note and do not advance time
  574. }
  575. else
  576. {
  577. *tick0Ref = *tickRef;
  578. note->tick = *tickRef;
  579. *tickRef += note->duration;
  580. }
  581. return _cmXScorePushNote(p, meas, voiceId, note );
  582. }
  583. cmXsRC_t _cmXScorePushNonNote( cmXScore_t* p, cmXsMeas_t* meas, const cmXmlNode_t* noteXmlNode, unsigned tick, unsigned duration, unsigned staff, double rvalue, const cmChar_t* tvalue, unsigned flags )
  584. {
  585. cmXsNote_t* note = cmLhAllocZ(p->lhH,cmXsNote_t,1);
  586. unsigned voiceId = 0; // non-note's are always assigned to voiceId=0;
  587. note->pitch = kInvalidMidiPitch;
  588. note->tick = tick;
  589. note->staff = staff;
  590. note->flags = flags;
  591. note->rvalue = rvalue;
  592. note->tvalue = tvalue;
  593. note->duration = duration;
  594. note->tied_dur = duration;
  595. note->meas = meas;
  596. note->xmlNode = noteXmlNode;
  597. return _cmXScorePushNote(p, meas, voiceId, note );
  598. }
  599. cmXsSpan_t* _cmXScoreFindOpenOctaveShift( cmXScore_t* p, unsigned staff, unsigned number )
  600. {
  601. cmXsSpan_t* s = p->spanL;
  602. for(; s!=NULL; s=s->link)
  603. if( s->tick1 == -1 && s->staff == staff && s->number == number )
  604. return s;
  605. return NULL;
  606. }
  607. cmXsRC_t _cmXScorePushOctaveShift(cmXScore_t* p, cmXsMeas_t* meas, unsigned staff, unsigned span_number, const cmChar_t* type_str, unsigned tick)
  608. {
  609. assert( meas != NULL);
  610. cmXsSpan_t* s;
  611. if( cmTextCmp(type_str,"stop") == 0 )
  612. {
  613. if((s = _cmXScoreFindOpenOctaveShift(p,staff,span_number)) == NULL )
  614. return cmErrWarnMsg(&p->err,kUnterminatedOctaveShiftXsrRC,"An illegal octave shift was encounted in meas %i.\n",meas->number);
  615. s->tick1 = tick;
  616. }
  617. else
  618. {
  619. s = cmLhAllocZ(p->lhH,cmXsSpan_t,1);
  620. s->staff = staff;
  621. s->meas = meas;
  622. s->number = span_number;
  623. s->tick0 = tick;
  624. s->tick1 = -1;
  625. s->pitch_offset = cmTextCmp(type_str,"up")==0 ? -12 : 12;
  626. s->link = p->spanL;
  627. p->spanL = s;
  628. }
  629. return kOkXsRC;
  630. }
  631. cmXsRC_t _cmXScoreParseDirection(cmXScore_t* p, cmXsMeas_t* meas, const cmXmlNode_t* dnp, unsigned tick)
  632. {
  633. cmXsRC_t rc = kOkXsRC;
  634. const cmXmlNode_t* np = NULL;
  635. const cmXmlAttr_t* a = NULL;
  636. unsigned flags = 0;
  637. int offset = 0;
  638. double rvalue = 0;
  639. const cmChar_t* tvalue = NULL;
  640. unsigned duration = 0;
  641. bool pushFl = true;
  642. unsigned staff = 0;
  643. cmXmlNodeInt( dnp, &offset, "offset", NULL );
  644. cmXmlNodeUInt(dnp, &staff, "staff", NULL );
  645. // if this is a metronome direction
  646. if((np = cmXmlSearch( dnp, "metronome", NULL, 0)) != NULL )
  647. {
  648. if( cmXmlNodeUInt(np,&duration,"per-minute",NULL) != kOkXmlRC )
  649. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"The 'per-minute' metronome value is missing on line %i.",np->line);
  650. if((rc = _cmXScoreParseNoteRValue(p,np,"beat-unit",&rvalue)) != kOkXsRC )
  651. return rc;
  652. flags = kMetronomeXsFl;
  653. }
  654. else
  655. // if this is a pedal direction
  656. if((np = cmXmlSearch( dnp, "pedal",NULL,0)) != NULL )
  657. {
  658. if((a = cmXmlFindAttrib(np,"type")) == NULL )
  659. return _cmXScoreMissingAttribute(p, np, "type" );
  660. if( cmTextCmp(a->value,"start") == 0 )
  661. flags = kDampDnXsFl;
  662. else
  663. if( cmTextCmp(a->value,"change") == 0 )
  664. flags = kDampUpDnXsFl;
  665. else
  666. if( cmTextCmp(a->value,"stop") == 0 )
  667. flags = kDampUpXsFl;
  668. else
  669. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unrecognized pedal type:'%s'.",cmStringNullGuard(a->value));
  670. }
  671. else
  672. // if this is a 'words' direction
  673. if((np = cmXmlSearch( dnp, "words", NULL, 0)) != NULL )
  674. {
  675. if((a = cmXmlFindAttrib(np,"enclosure")) != NULL && cmTextCmp(a->value,"rectangle")==0 )
  676. {
  677. if( cmTextIsEmpty( tvalue = np->dataStr ) )
  678. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Section number is blank or missing on line %i.",np->line);
  679. flags = kSectionXsFl;
  680. }
  681. else
  682. {
  683. // we only care about 'words' in 'enclosures'
  684. pushFl = false;
  685. }
  686. }
  687. else
  688. // if this is an 'octave-shift' direction
  689. if((np = cmXmlSearch( dnp, "octave-shift", NULL, 0)) != NULL )
  690. {
  691. unsigned span_number = -1;
  692. if( cmXmlAttrUInt(np,"number",&span_number) != kOkXmlRC )
  693. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Octave-shift is missing a 'number' attribute.");
  694. if((a = cmXmlFindAttrib(np,"type")) == NULL)
  695. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Octave-shift is missing a 'type' attribute.");
  696. rc = _cmXScorePushOctaveShift(p,meas,staff,span_number,a->value,tick+offset);
  697. pushFl = false;
  698. }
  699. else
  700. {
  701. pushFl = false;
  702. }
  703. if( pushFl )
  704. rc = _cmXScorePushNonNote(p,meas,dnp,tick+offset,duration,staff,rvalue,tvalue,flags);
  705. return rc;
  706. }
  707. // On input tickRef is set to the absolute tick of the bar line and on output it is set
  708. // to the absolute tick of the next bar line.
  709. cmXsRC_t _cmXScoreParseMeasure(cmXScore_t* p, cmXsPart_t* pp, const cmXmlNode_t* mnp, unsigned* tickRef)
  710. {
  711. cmXsRC_t rc = kOkXsRC;
  712. const cmXmlNode_t* np = NULL;
  713. unsigned tick = *tickRef;
  714. unsigned tick0= 0;
  715. cmXsMeas_t* m = NULL;
  716. // allocate the 'measure' record
  717. cmXsMeas_t* meas = cmLhAllocZ(p->lhH,cmXsMeas_t,1);
  718. // get measure number
  719. if( cmXmlAttrUInt(mnp,"number", &meas->number) != kOkXmlRC )
  720. return _cmXScoreMissingAttribute(p,mnp,"number");
  721. if( pp->measL == NULL )
  722. pp->measL = meas;
  723. else
  724. {
  725. m = pp->measL;
  726. while( m->link != NULL )
  727. m = m->link;
  728. m->link = meas;
  729. meas->divisions = m->divisions;
  730. meas->beats = m->beats;
  731. meas->beat_type = m->beat_type;
  732. }
  733. // get measure attributes node
  734. if((np = cmXmlSearch(mnp,"attributes",NULL,0)) != NULL)
  735. {
  736. cmXmlNodeUInt(np,&meas->divisions,"divisions",NULL);
  737. cmXmlNodeUInt(np,&meas->beats, "time","beats",NULL);
  738. cmXmlNodeUInt(np,&meas->beat_type,"time","beat-type",NULL);
  739. }
  740. // store the bar line
  741. if((rc = _cmXScorePushNonNote(p,meas,mnp,tick,0,0,0,NULL,kBarXsFl)) != kOkXsRC )
  742. return rc;
  743. np = mnp->children;
  744. // for each child of the 'meas' XML node
  745. for(; rc==kOkXsRC && np!=NULL; np=np->sibling)
  746. {
  747. // if this is a 'note' node
  748. if( cmTextCmp(np->label,"note") == 0 )
  749. {
  750. rc = _cmXScoreParseNote(p,meas,np,&tick0,&tick);
  751. }
  752. else
  753. // if this is a 'backup' node
  754. if( cmTextCmp(np->label,"backup") == 0 )
  755. {
  756. unsigned backup;
  757. cmXmlNodeUInt(np,&backup,"duration",NULL);
  758. if( backup > tick )
  759. tick = 0;
  760. else
  761. tick -= backup;
  762. tick0 = tick;
  763. }
  764. else
  765. // if this is a 'direction' node
  766. if( cmTextCmp(np->label,"direction") == 0 )
  767. {
  768. rc = _cmXScoreParseDirection(p,meas,np,tick);
  769. }
  770. }
  771. *tickRef = tick;
  772. return rc;
  773. }
  774. cmXsRC_t _cmXScoreParsePart( cmXScore_t* p, cmXsPart_t* pp )
  775. {
  776. cmXsRC_t rc = kOkXsRC;
  777. const cmXmlNode_t* xnp;
  778. cmXmlAttr_t partAttr = { "id", pp->idStr };
  779. unsigned barTick = 0;
  780. // find the 'part'
  781. if((xnp = cmXmlSearch( cmXmlRoot(p->xmlH), "part", &partAttr, 1)) == NULL )
  782. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"The part '%s' was not found.",pp->idStr);
  783. // for each child of this part - find each measure
  784. const cmXmlNode_t* cnp = xnp->children;
  785. for(; cnp!=NULL; cnp=cnp->sibling)
  786. if( cmTextCmp(cnp->label,"measure") == 0 )
  787. if((rc = _cmXScoreParseMeasure(p,pp,cnp,&barTick)) != kOkXsRC )
  788. return rc;
  789. return rc;
  790. }
  791. // Insert note 'np' into the sorted note list based at 's0'.
  792. // Return a pointer to the base of the list after the insertion.
  793. cmXsNote_t* _cmXScoreInsertSortedNote( cmXsNote_t* s0, cmXsNote_t* np )
  794. {
  795. assert( np != NULL );
  796. // np->slink is not NULL if the list is being resorted
  797. np->slink = NULL;
  798. // this list is empty so np is the first element on the list
  799. if( s0 == NULL )
  800. return np;
  801. // np is before the first element on the list
  802. if( np->tick < s0->tick )
  803. {
  804. np->slink = s0;
  805. return np;
  806. }
  807. cmXsNote_t* s1 = s0;
  808. cmXsNote_t* s2 = s0->slink;
  809. while( s2 != NULL )
  810. {
  811. if( s2->tick > np->tick )
  812. {
  813. s1->slink = np;
  814. np->slink = s2;
  815. return s0;
  816. }
  817. s1 = s2;
  818. s2 = s2->slink;
  819. }
  820. s1->slink = np;
  821. return s0;
  822. }
  823. // Set the cmXsNode_t.secs and dsecs.
  824. void _cmXScoreSetAbsoluteTime( cmXScore_t* p )
  825. {
  826. double tpqn = 0; // ticks per quarter note
  827. double tps = 0; // ticks per second
  828. unsigned metro_tick = 0;
  829. double metro_sec = 0;
  830. double sec0 = 0;
  831. cmXsPart_t* pp = p->partL;
  832. for(; pp!=NULL; pp=pp->link)
  833. {
  834. cmXsMeas_t* mp = pp->measL;
  835. for(; mp!=NULL; mp=mp->link)
  836. {
  837. if( mp->divisions != 0 )
  838. tpqn = mp->divisions;
  839. cmXsNote_t* np = mp->noteL;
  840. for(; np!=NULL; np=np->slink)
  841. {
  842. // Seconds are calculated as:
  843. // dticks = np->tick - metro_tick; // where metro_tick is the absolute tick of the last metro event
  844. // secs = (dticks/tps) + metro_secs; // where metro_secs is the absoute time of the last metro event
  845. unsigned dticks = np->tick - metro_tick;
  846. double secs = tps==0 ? 0 : (dticks/tps) + metro_sec;
  847. double dsecs = secs - sec0;
  848. //
  849. if( cmIsFlag(np->flags,kMetronomeXsFl) )
  850. {
  851. double bpm = np->duration; // beats per minute
  852. double bps = bpm / 60.0; // beats per second
  853. tps = bps * tpqn; // ticks per second
  854. metro_tick = np->tick; // store tick of the last metronome marker
  855. metro_sec = secs; // store time of the last metronome marker
  856. }
  857. if( cmIsFlag(np->flags,kBarXsFl|kDampDnXsFl|kDampUpXsFl|kDampUpDnXsFl|kSostDnXsFl|kSostUpXsFl|kOnsetXsFl|kSectionXsFl) )
  858. {
  859. np->secs = secs;
  860. np->dsecs = dsecs;
  861. sec0 = secs;
  862. }
  863. }
  864. }
  865. }
  866. }
  867. void _cmXScoreSort( cmXScore_t* p )
  868. {
  869. // for each part
  870. cmXsPart_t* pp = p->partL;
  871. for(; pp!=NULL; pp=pp->link)
  872. {
  873. // for each measure in this part
  874. cmXsMeas_t* mp = pp->measL;
  875. for(; mp!=NULL; mp=mp->link)
  876. {
  877. // explicitely set noteL to NULL to in case we are re-sorting
  878. mp->noteL = NULL;
  879. // for each voice in this measure
  880. cmXsVoice_t* vp = mp->voiceL;
  881. for(; vp!=NULL; vp=vp->link)
  882. {
  883. // for each note in this measure
  884. cmXsNote_t* np = vp->noteL;
  885. for(; np!=NULL; np=np->mlink)
  886. mp->noteL = _cmXScoreInsertSortedNote(mp->noteL,np);
  887. }
  888. }
  889. }
  890. // The order of events may have changed update the absolute time.
  891. _cmXScoreSetAbsoluteTime( p );
  892. }
  893. // All notes in a[aN] are on the same tick
  894. unsigned _cmXsSpreadGraceNotes( cmXsNote_t** a, unsigned aN )
  895. {
  896. unsigned i;
  897. bool barFl = false;
  898. // set barFl to true if a bar marker is included in the notes
  899. for(i=0; i<aN; ++i)
  900. if( cmIsFlag(a[i]->flags,kBarXsFl) )
  901. barFl = true;
  902. // spread any grace notes by one tick
  903. unsigned nextGraceTick = UINT_MAX;
  904. for(i=0; i<aN; ++i)
  905. if( cmIsFlag(a[i]->flags,kGraceXsFl) )
  906. {
  907. if( nextGraceTick == UINT_MAX )
  908. nextGraceTick = a[i]->tick + 1;
  909. else
  910. {
  911. a[i]->tick = nextGraceTick;
  912. nextGraceTick += 1;
  913. }
  914. }
  915. // if this tick group includes the bar ...
  916. if( barFl && nextGraceTick != UINT_MAX )
  917. {
  918. // ... then move all non-grace note events (except the bar marker) after
  919. // the grace notes
  920. for(i=0; i<aN; ++i)
  921. if( cmIsNotFlag(a[i]->flags,kGraceXsFl) && cmIsNotFlag(a[i]->flags,kBarXsFl) )
  922. a[i]->tick = nextGraceTick;
  923. }
  924. return nextGraceTick==UINT_MAX ? 0 : nextGraceTick;
  925. }
  926. void _cmXScoreSpreadGraceNotes( cmXScore_t* p )
  927. {
  928. cmXsPart_t* pp = p->partL;
  929. for(; pp!=NULL; pp=pp->link)
  930. {
  931. // tick1 is the location of the minimum current tick
  932. // (or 0 if it should be ignored)
  933. unsigned tick1 = 0;
  934. cmXsMeas_t* mp = pp->measL;
  935. for(; mp!=NULL; mp=mp->link)
  936. {
  937. cmXsNote_t* np = mp->noteL;
  938. unsigned aN = 128;
  939. cmXsNote_t* a[ aN ];
  940. unsigned ai = 0;
  941. // The first event in a measure may have been forced ahead
  942. // by spreading at the end of the previous measure
  943. if( tick1 > np->tick )
  944. np->tick = tick1;
  945. else
  946. tick1 = 0;
  947. // tick0 is the tick of the current tick group we are examining
  948. // A tick group is a group of events that share the same tick.
  949. unsigned tick0 = np->tick;
  950. // for each note
  951. for(; np!=NULL; np=np->slink)
  952. {
  953. // if this event is the first of a new tick group (then a[] holds a group completed on the previous note)
  954. if( np->tick != tick0 )
  955. {
  956. // if there is more than one event in the completed tick group ...
  957. if( ai > 1 )
  958. tick1 = _cmXsSpreadGraceNotes(a,ai); // ... then process the group
  959. ai = 0; // empty the tick group array
  960. tick0 = np->tick; // update the current group's common tick
  961. }
  962. // if the min. current tick is ahead of this event then move the event ahead
  963. if( tick1 > np->tick )
  964. np->tick = tick1;
  965. else
  966. tick1 = 0; // otherwise disable tick1
  967. // add this event to the tick group
  968. assert(ai<aN);
  969. a[ai++] = np;
  970. }
  971. // if there are events in the group array then process them
  972. if( ai > 1 )
  973. tick1 = _cmXsSpreadGraceNotes(a,ai);
  974. }
  975. }
  976. }
  977. bool _cmXScoreFindTiedNote( cmXScore_t* p, cmXsMeas_t* mp, cmXsNote_t* n0p, bool rptFl )
  978. {
  979. cmXsNote_t* nbp = n0p;
  980. cmXsNote_t* nnp = n0p->slink; // begin w/ note following np
  981. unsigned measNumb = mp->number;
  982. cmChar_t acc = n0p->alter==-1?'b' : (n0p->alter==1?'#':' ');
  983. if( rptFl )
  984. printf("%i %i %s ",n0p->meas->number,n0p->tick,cmMidiToSciPitch(n0p->pitch,NULL,0));
  985. while(1)
  986. {
  987. // if we are at the end of a measure advance to the next measure
  988. if( nnp == NULL )
  989. {
  990. mp = mp->link;
  991. nnp = mp->noteL;
  992. // if a measure was completed and no end note was found ... then the tie is unterminated
  993. // (a tie must be continued in every measure which it passes through)
  994. if( mp->number > measNumb + 1 )
  995. break;
  996. }
  997. // for each note starting at nnp
  998. for(; nnp!=NULL; nnp=nnp->slink)
  999. {
  1000. // if this note is tied to the originating note (np)
  1001. if( nnp->voice->id == n0p->voice->id && nnp->step == n0p->step && nnp->octave == n0p->octave )
  1002. {
  1003. nnp->flags |= kTieProcXsFl;
  1004. nnp->flags = cmClrFlag(nnp->flags,kOnsetXsFl);
  1005. n0p->tied = nnp;
  1006. nbp->tied_dur += nnp->duration;
  1007. nnp->tied_dur = 0;
  1008. if( rptFl )
  1009. printf("---> %i %i %s ",nnp->meas->number,nnp->tick,cmMidiToSciPitch(nnp->pitch,NULL,0));
  1010. // if this note is not tied to a subsequent note
  1011. if( cmIsNotFlag(nnp->flags,kTieBegXsFl) )
  1012. return true;
  1013. n0p = nnp;
  1014. // record the measure number of the last note with a tie-start
  1015. measNumb = mp->number;
  1016. }
  1017. }
  1018. }
  1019. cmErrWarnMsg(&p->err,kUnterminatedTieXsRC,"The tied %c%c%i in measure %i (tick:%i) was not terminated.",n0p->step,acc,n0p->octave,measNumb,n0p->tick);
  1020. return false;
  1021. }
  1022. void _cmXScoreResolveTiesAndLoc( cmXScore_t* p )
  1023. {
  1024. unsigned n = 0; // count of notes which begin a tie
  1025. unsigned m = 0; // count of tied notes that are correctly terminated.
  1026. bool rptFl = false;
  1027. cmXsPart_t* pp = p->partL;
  1028. // for each part
  1029. for(; pp!=NULL; pp=pp->link)
  1030. {
  1031. unsigned locIdx = 1;
  1032. cmXsMeas_t* mp = pp->measL;
  1033. // for each measure
  1034. for(; mp!=NULL; mp=mp->link)
  1035. {
  1036. cmXsNote_t* n0 = NULL;
  1037. cmXsNote_t* np = mp->noteL;
  1038. // for each note in this measure
  1039. for(; np!=NULL; np=np->slink)
  1040. {
  1041. // if this note begins a tie and has not yet been processed
  1042. // (A note that continues a tie and therefore has a kTieBegXsFl set
  1043. // may have already been processed by an earlier tied note.)
  1044. if( cmIsFlag(np->flags,kTieBegXsFl) && cmIsNotFlag(np->flags,kTieProcXsFl))
  1045. {
  1046. if( _cmXScoreFindTiedNote(p,mp,np,rptFl) )
  1047. m += 1;
  1048. if( rptFl )
  1049. printf("\n");
  1050. n += 1;
  1051. }
  1052. // Validate the tie state of the current note.
  1053. if( cmIsFlag(np->flags,kTieEndXsFl) && cmIsFlag(np->flags,kOnsetXsFl) )
  1054. {
  1055. cmChar_t acc = np->alter==-1?'b' : (np->alter==1?'#':' ');
  1056. cmErrWarnMsg(&p->err,kUnterminatedTieXsRC,"The tied %c%c%i in measure %i marked as a tied note but is also marked to sound.",np->step,acc,np->octave,mp->number);
  1057. }
  1058. //
  1059. // Set the score location of notes marked for onset and bar lines.
  1060. //
  1061. if( cmIsFlag(np->flags,kOnsetXsFl|kBarXsFl) )
  1062. {
  1063. // if this note does not share the same location as the previous 'located' note then increment the 'loc' index
  1064. if( cmIsFlag(np->flags,kBarXsFl) || (n0!=NULL && n0->tick!=np->tick))
  1065. locIdx += 1;
  1066. np->locIdx = locIdx;
  1067. n0 = np;
  1068. }
  1069. }
  1070. }
  1071. }
  1072. printf("Tied notes found:%i Not found:%i\n",m,n-m);
  1073. }
  1074. cmXsRC_t _cmXScoreResolveOctaveShift( cmXScore_t* p )
  1075. {
  1076. const cmXsSpan_t* s;
  1077. for(s=p->spanL; s!=NULL; s=s->link)
  1078. {
  1079. if( s->tick1 == -1)
  1080. {
  1081. cmErrWarnMsg(&p->err,kSyntaxErrorXsRC,"An unterminated octave shift span was encountered in measure %i staff=%i.",s->meas->number,s->staff);
  1082. }
  1083. else
  1084. {
  1085. cmXsMeas_t* m = p->partL->measL;
  1086. for(; m!=NULL; m=m->link)
  1087. if( m->number == s->meas->number )
  1088. break;
  1089. assert( m != NULL );
  1090. cmXsNote_t* note = m->noteL;
  1091. for(; note!=NULL; note=note->slink)
  1092. if( note->staff==s->staff && s->tick0 <= note->tick && note->tick < s->tick1 )
  1093. note->pitch += s->pitch_offset;
  1094. }
  1095. }
  1096. return kOkXsRC;
  1097. }
  1098. cmXsNote_t* _cmXScoreFindOverlappingNote( cmXScore_t* p, const cmXsNote_t* knp )
  1099. {
  1100. cmXsPart_t* pp = p->partL;
  1101. // for each part
  1102. for(; pp!=NULL; pp=pp->link)
  1103. {
  1104. cmXsMeas_t* mp = pp->measL;
  1105. // for each measure
  1106. for(; mp!=NULL; mp=mp->link)
  1107. {
  1108. cmXsNote_t* np = mp->noteL;
  1109. // for each note in this measure
  1110. for(; np!=NULL; np=np->slink)
  1111. if( np->uid != knp->uid
  1112. && cmIsFlag(np->flags,kOnsetXsFl)
  1113. && knp->pitch == np->pitch
  1114. && knp->tick >= np->tick
  1115. && knp->tick < (np->tick + np->tied_dur) )
  1116. {
  1117. return np;
  1118. }
  1119. }
  1120. }
  1121. return NULL;
  1122. }
  1123. void _cmXScoreProcessOverlappingNotes( cmXScore_t* p )
  1124. {
  1125. cmXsPart_t* pp = p->partL;
  1126. // for each part
  1127. for(; pp!=NULL; pp=pp->link)
  1128. {
  1129. cmXsMeas_t* mp = pp->measL;
  1130. // for each measure
  1131. for(; mp!=NULL; mp=mp->link)
  1132. {
  1133. cmXsNote_t* np = mp->noteL;
  1134. cmXsNote_t* fnp;
  1135. // for each note in this measure
  1136. for(; np!=NULL; np=np->slink)
  1137. if( cmIsFlag(np->flags,kOnsetXsFl) && (fnp = _cmXScoreFindOverlappingNote(p,np)) != NULL)
  1138. {
  1139. // is np entirely contained inside fnp
  1140. bool embeddedFl = fnp->tick + fnp->tied_dur > np->tick + np->tied_dur;
  1141. //printf("bar=%3i %4s voice:%2i %2i : %7i %7i : %7i %7i : %7i : %c \n",np->meas->number,cmMidiToSciPitch(np->pitch,NULL,0),np->voice->id,fnp->voice->id,fnp->tick,fnp->tick+fnp->duration,np->tick,np->tick+np->duration, (fnp->tick+fnp->duration) - np->tick, embeddedFl ? 'E' : 'O');
  1142. // turn off embedded notes
  1143. if( embeddedFl )
  1144. {
  1145. if( np->voice->id == fnp->voice->id )
  1146. cmErrWarnMsg(&p->err,kOverlapWarnXsRC,"A time embedded note (bar=%i %s) was removed even though it overlapped with a note in the same voice.",np->meas->number,cmMidiToSciPitch(np->pitch,NULL,0));
  1147. np->flags = cmClrFlag(np->flags,kOnsetXsFl);
  1148. }
  1149. else
  1150. {
  1151. int d = (fnp->tick+fnp->tied_dur) - np->tick;
  1152. // shorten the first note
  1153. if( d > 0 && d < fnp->tied_dur )
  1154. fnp->tied_dur -= d;
  1155. // move the second note just past it
  1156. np->tick = fnp->tick + fnp->tied_dur;
  1157. cmErrWarnMsg(&p->err,kOverlapWarnXsRC,"A shorten/shift operation was done to reconcile two overlapped %s (ticks:%i %i) notes in measure %i.",cmMidiToSciPitch(np->pitch,NULL,0),fnp->tick,np->tick,np->meas->number);
  1158. }
  1159. }
  1160. }
  1161. }
  1162. }
  1163. // The identical pitch may be notated to play simultaneously on different voices.
  1164. // As performed on the piano this will equate to a single sounding note.
  1165. // This function clears the onset flag on all except one of the duplicated notes.
  1166. void _cmXScoreRemoveDuplicateNotes( cmXScore_t* p )
  1167. {
  1168. cmXsPart_t* pp = p->partL;
  1169. // for each part
  1170. for(; pp!=NULL; pp=pp->link)
  1171. {
  1172. cmXsMeas_t* mp = pp->measL;
  1173. // for each measure
  1174. for(; mp!=NULL; mp=mp->link)
  1175. {
  1176. cmXsNote_t* np = mp->noteL;
  1177. // for each note in this measure
  1178. for(; np!=NULL; np=np->slink)
  1179. if( cmIsFlag(np->flags,kOnsetXsFl) )
  1180. {
  1181. cmXsNote_t* n0p = mp->noteL;
  1182. for(; n0p!=NULL; n0p=n0p->slink)
  1183. if( n0p!=np && cmIsFlag(n0p->flags,kOnsetXsFl) && np->locIdx==n0p->locIdx && np->pitch==n0p->pitch )
  1184. n0p->flags = cmClrFlag(n0p->flags,kOnsetXsFl);
  1185. }
  1186. }
  1187. }
  1188. }
  1189. void _cmXScoreSetMeasGroups( cmXScore_t* p, unsigned flag )
  1190. {
  1191. unsigned sectionId = 0;
  1192. cmXsNote_t* n0 = NULL;
  1193. cmXsPart_t* pp = p->partL;
  1194. // for each part
  1195. for(; pp!=NULL; pp=pp->link)
  1196. {
  1197. cmXsMeas_t* mp = pp->measL;
  1198. // for each measure
  1199. for(; mp!=NULL; mp=mp->link)
  1200. {
  1201. cmXsNote_t* np = mp->noteL;
  1202. // for each note in this measure
  1203. for(; np!=NULL; np=np->slink)
  1204. {
  1205. // if this note has a heel marker and we are looking for evenness events
  1206. if( cmIsFlag(flag,kEvenXsFl) && cmIsFlag(np->flags,kHeelXsFl) )
  1207. {
  1208. np->flags = cmSetFlag(np->flags,kBegGroupXsFl | kEndGroupXsFl );
  1209. np->evenGroupId = sectionId + 1;
  1210. }
  1211. // if this note is of the type we are looking for
  1212. if( cmIsFlag(np->flags,flag) )
  1213. {
  1214. if( n0 == NULL )
  1215. np->flags = cmSetFlag(np->flags,kBegGroupXsFl);
  1216. n0 = np;
  1217. }
  1218. // if this is a section marker
  1219. if( cmIsFlag(np->flags,kSectionXsFl) )
  1220. {
  1221. if( n0 != NULL )
  1222. {
  1223. np->flags = cmSetFlag(np->flags,kEndGroupXsFl);
  1224. switch( flag )
  1225. {
  1226. case kEvenXsFl: n0->evenGroupId = sectionId+1; break;
  1227. case kDynXsFl: n0->dynGroupId = sectionId+1; break;
  1228. case kTempoXsFl: n0->tempoGroupId= sectionId+1; break;
  1229. }
  1230. }
  1231. if( cmIsFlag(np->flags,kSectionXsFl) )
  1232. {
  1233. sectionId = 0;
  1234. if( np->tvalue == NULL )
  1235. cmErrWarnMsg(&p->err,kSyntaxErrorXsRC,"An apparent section label in measure '%i' is blank.",np->meas->number);
  1236. else
  1237. if( cmTextToUInt( np->tvalue, &sectionId, NULL ) != kOkTxRC )
  1238. cmErrWarnMsg(&p->err,kSyntaxErrorXsRC,"The section label '%s' is not an integer.",np->tvalue);
  1239. //sectionId = np->tvalue==NULL ? 0 : strtol(np->tvalue,NULL,10);
  1240. }
  1241. n0 = NULL;
  1242. }
  1243. }
  1244. }
  1245. }
  1246. }
  1247. cmXsRC_t _cmXScoreWriteScorePlotFile( cmXScore_t* p, const cmChar_t* fn )
  1248. {
  1249. cmXsRC_t rc = kOkXsRC;
  1250. cmFileH_t fH = cmFileNullHandle;
  1251. double ticks_per_sec = 0;
  1252. double onset_secs = 0;
  1253. if( cmFileOpen(&fH,fn,kWriteFileFl,p->err.rpt) != kOkFileRC )
  1254. return cmErrMsg(&p->err,kFileFailXsRC,"Unable to create the file '%s'.",cmStringNullGuard(fn));
  1255. cmXsPart_t* pp = p->partL;
  1256. for(; pp!=NULL; pp=pp->link)
  1257. {
  1258. cmXsMeas_t* mp = pp->measL;
  1259. for(; mp!=NULL; mp=mp->link)
  1260. {
  1261. cmFilePrintf(fH,"b %f %i %s B\n",onset_secs,mp->number,"bar");
  1262. cmXsNote_t* np = mp->noteL;
  1263. unsigned tick0 = 0;
  1264. for(; np!=NULL; np=np->slink)
  1265. {
  1266. if( cmIsFlag(np->flags,kMetronomeXsFl) )
  1267. {
  1268. double bps = np->tied_dur / 60.0;
  1269. // t b t
  1270. // - = - -
  1271. // s s b
  1272. ticks_per_sec = bps * mp->divisions;
  1273. }
  1274. else
  1275. {
  1276. if( cmIsFlag(np->flags,kOnsetXsFl) )
  1277. {
  1278. onset_secs += (np->tick - tick0) / ticks_per_sec;
  1279. tick0 = np->tick;
  1280. cmFilePrintf(fH,"n %f %f %i %s %s\n",onset_secs,np->tied_dur/ticks_per_sec,np->uid,cmMidiToSciPitch(np->pitch,NULL,0),cmIsFlag(np->flags,kGraceXsFl)?"G":"N");
  1281. }
  1282. }
  1283. }
  1284. onset_secs += (mp->divisions * mp->beats - tick0) / ticks_per_sec;
  1285. }
  1286. }
  1287. cmFileClose(&fH);
  1288. return rc;
  1289. }
  1290. // Force the bar event to be the first event in the measure.
  1291. void _cmXScoreFixBarLines( cmXScore_t* p )
  1292. {
  1293. cmXsPart_t* pp = p->partL;
  1294. for(; pp!=NULL; pp=pp->link)
  1295. {
  1296. cmXsMeas_t* mp = pp->measL;
  1297. for(; mp!=NULL; mp=mp->link)
  1298. {
  1299. cmXsNote_t* np = mp->noteL;
  1300. cmXsNote_t* ep = NULL;
  1301. for(; np!=NULL; np=np->slink )
  1302. {
  1303. if( cmIsFlag(np->flags,kBarXsFl) )
  1304. {
  1305. if( ep != NULL )
  1306. np->tick = ep->tick;
  1307. break;
  1308. }
  1309. if( ep == NULL )
  1310. ep = np;
  1311. }
  1312. }
  1313. }
  1314. }
  1315. // Assign pedal down durations to pedal down events.
  1316. cmXsRC_t _cmXScoreProcessPedals( cmXScore_t* p, bool reportFl )
  1317. {
  1318. cmXsRC_t rc = kOkXsRC;
  1319. cmXsPart_t* pp = p->partL;
  1320. for(; pp!=NULL; pp=pp->link)
  1321. {
  1322. cmXsNote_t* dnp = NULL; // pointer to last damper down event
  1323. cmXsNote_t* snp = NULL; // pointer to last sostenuto down event
  1324. cmXsMeas_t* mp = pp->measL;
  1325. for(; mp!=NULL; mp=mp->link)
  1326. {
  1327. cmXsNote_t* np = mp->noteL;
  1328. for(; np!=NULL; np=np->slink )
  1329. {
  1330. unsigned x = np->flags & (kDampDnXsFl|kDampUpXsFl|kDampUpDnXsFl|kSostUpXsFl|kSostDnXsFl);
  1331. switch( x )
  1332. {
  1333. case 0:
  1334. break;
  1335. case kDampDnXsFl:
  1336. if( dnp != NULL )
  1337. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Damper down not preceded by damper up in measure:%i tick:%i.",mp->number,np->tick);
  1338. else
  1339. dnp = np;
  1340. if( reportFl )
  1341. cmRptPrintf(p->err.rpt,"Damp Down : staff:%i meas:%i tick:%i\n", np->staff, mp->number, np->tick);
  1342. break;
  1343. case kDampUpXsFl:
  1344. if( dnp == NULL )
  1345. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Damper up not preceded by damper down in measure:%i tick:%i.",mp->number,np->tick);
  1346. else
  1347. {
  1348. dnp->duration = np->tick - dnp->tick;
  1349. dnp = NULL;
  1350. }
  1351. if( reportFl )
  1352. cmRptPrintf(p->err.rpt,"Damp Up : staff:%i meas:%i tick:%i\n", np->staff, mp->number, np->tick);
  1353. break;
  1354. case kDampUpDnXsFl:
  1355. if( dnp == NULL )
  1356. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Damper up/down not preceded by damper down in measure:%i tick:%i.",mp->number,np->tick);
  1357. else
  1358. {
  1359. dnp->duration = np->tick - dnp->tick;
  1360. dnp = np;
  1361. }
  1362. if( reportFl )
  1363. cmRptPrintf(p->err.rpt,"Damp UpDn : staff:%i meas:%i tick:%i\n", np->staff, mp->number, np->tick);
  1364. break;
  1365. case kSostDnXsFl:
  1366. if( snp != NULL )
  1367. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Sostenuto down not preceded by sostenuto up in measure:%i tick:%i.",mp->number, np->tick);
  1368. else
  1369. snp = np;
  1370. if( reportFl )
  1371. cmRptPrintf(p->err.rpt,"Sost Down : staff:%i meas:%i tick:%i\n", np->staff, mp->number, np->tick);
  1372. break;
  1373. case kSostUpXsFl:
  1374. if( snp == NULL )
  1375. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Sostenuto up not preceded by sostenuto down in measure:%i tick:%i.",mp->number,np->tick);
  1376. else
  1377. {
  1378. snp->duration = np->tick - snp->tick;
  1379. snp = NULL;
  1380. }
  1381. if( reportFl )
  1382. cmRptPrintf(p->err.rpt,"Sost Up : staff:%i meas:%i tick:%i\n", np->staff, mp->number, np->tick);
  1383. break;
  1384. default:
  1385. {
  1386. assert(0);
  1387. }
  1388. } // switch
  1389. } // for notes
  1390. } // for measures
  1391. if( dnp != NULL )
  1392. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Damper left down at the end of a part.");
  1393. if( snp != NULL )
  1394. cmErrWarnMsg(&p->err,kPedalStateErrorXsRc,"Sostenuto left down at the end of a part.");
  1395. }
  1396. _cmXScoreSort(p);
  1397. return rc;
  1398. }
  1399. void _cmXScoreInsertTime( cmXScore_t* p, cmXsMeas_t* mp, cmXsNote_t* np, unsigned expand_ticks )
  1400. {
  1401. for(; mp!=NULL; mp=mp->link)
  1402. {
  1403. if( np == NULL )
  1404. np = mp->noteL;
  1405. for(; np!=NULL; np=np->slink)
  1406. np->tick += expand_ticks;
  1407. }
  1408. }
  1409. void _cmXScoreGraceInsertTimeBase( cmXScore_t* p, unsigned graceGroupId, cmXsNote_t* aV[], unsigned aN, unsigned initTick )
  1410. {
  1411. cmXsNote_t* np = NULL;
  1412. unsigned expand_ticks = 0;
  1413. unsigned ticks = initTick;
  1414. unsigned t0 = 0;
  1415. unsigned i;
  1416. for(i=0; i<aN; ++i)
  1417. if( cmIsFlag(aV[i]->flags,kGraceXsFl) && aV[i]->graceGroupId == graceGroupId )
  1418. {
  1419. // if this grace note falls on the same tick as the previous grace note
  1420. if( np != NULL && aV[i]->tick == t0 )
  1421. aV[i]->tick = np->tick;
  1422. else
  1423. {
  1424. t0 = aV[i]->tick; // store the unmodified tick value of this note
  1425. aV[i]->tick = ticks; // set the new tick value
  1426. ticks += aV[i]->duration; // calc the next grace not location
  1427. expand_ticks += aV[i]->duration; // track how much we are expanding time by
  1428. }
  1429. np = aV[i];
  1430. }
  1431. np = np->slink;
  1432. if( np != NULL )
  1433. _cmXScoreInsertTime(p,np->meas,np,expand_ticks);
  1434. }
  1435. // (a) Insert the grace notes in between the first and last note in the group
  1436. // by inserting time between the first and last note.
  1437. // Note that in effect his means that the last note is pushed back
  1438. // in time by the total duration of the grace notes.
  1439. void _cmXScoreGraceInsertTime( cmXScore_t* p, unsigned graceGroupId, cmXsNote_t* aV[], unsigned aN )
  1440. {
  1441. _cmXScoreGraceInsertTimeBase( p, graceGroupId,aV,aN, aV[aN-1]->tick );
  1442. }
  1443. // (s) Insert the grace notes in between the first and last note in the group
  1444. // but do not insert any additional time betwee the first and last note.
  1445. // In effect time is removed from the first note and taken by the grace notes.
  1446. // The time position of the last note is therefore unchanged.
  1447. void _cmXScoreGraceOverlayTime( cmXScore_t* p, unsigned graceGroupId, cmXsNote_t* aV[], unsigned aN )
  1448. {
  1449. assert(aN >= 3 );
  1450. int i = (int)aN-2;
  1451. cmXsNote_t* np = aV[aN-1];
  1452. unsigned t0 = -1;
  1453. for(; i>0; --i)
  1454. if( cmIsFlag(aV[i]->flags,kGraceXsFl) && aV[i]->graceGroupId == graceGroupId )
  1455. {
  1456. if( aV[i]->tick == t0)
  1457. aV[i]->tick = np->tick;
  1458. else
  1459. {
  1460. t0 = aV[i]->tick;
  1461. aV[i]->tick = np->tick - aV[i]->duration;
  1462. }
  1463. np = aV[i];
  1464. }
  1465. }
  1466. // (A) Play the first grace at the time of the first note in the group (which is a non-grace note)
  1467. // and then expand time while inserting the other grace notes.
  1468. void _cmXScoreGraceInsertAfterFirst( cmXScore_t* p, unsigned graceGroupId, cmXsNote_t* aV[], unsigned aN )
  1469. {
  1470. _cmXScoreGraceInsertTimeBase( p, graceGroupId,aV,aN, aV[0]->tick );
  1471. }
  1472. // (n) Play the first grace not shortly (one grace note duration) after the first note
  1473. // in the group (which is a non-grace note) and then expand time while inserting the other
  1474. // grace notes.
  1475. void _cmXScoreGraceInsertSoonAfterFirst( cmXScore_t* p, unsigned graceGroupId, cmXsNote_t* aV[], unsigned aN )
  1476. {
  1477. _cmXScoreGraceInsertTimeBase( p, graceGroupId,aV,aN, aV[0]->tick + aV[1]->duration );
  1478. }
  1479. // Adjust the locations of grace notes. Note that this must be done
  1480. // after reordering so that we can be sure that the order in time of
  1481. // the notes in each group has been set prior to building the
  1482. // grace note groups - which must be in reverse time order.
  1483. cmXsRC_t _cmXScoreProcessGraceNotes( cmXScore_t* p, unsigned nextGraceGroupId )
  1484. {
  1485. cmXsRC_t rc = kOkXsRC;
  1486. unsigned graceGroupId = 1;
  1487. double graceDurSec = 1.0/15.0; // duration of all grace notes in seconds
  1488. for(; graceGroupId<nextGraceGroupId; ++graceGroupId)
  1489. {
  1490. cmXsNote_t* gn0p = NULL; // first note in the grace group
  1491. cmXsNote_t* gn1p = NULL; // last note in the grace group
  1492. unsigned gN = 0;
  1493. cmXsPart_t* pp = p->partL;
  1494. double ticksPerSec = 0;
  1495. // Build a note chain, using cmXsNote_t.grace, between gn0p and
  1496. // gn1p containing all the grace notes with
  1497. // cmXsNote_t.graceGroupId == graceGroupId.
  1498. for(; pp!=NULL; pp=pp->link)
  1499. {
  1500. cmXsMeas_t* mp = pp->measL;
  1501. for(; mp!=NULL; mp=mp->link)
  1502. {
  1503. cmXsNote_t* np = mp->noteL;
  1504. for(; np!=NULL; np=np->slink )
  1505. {
  1506. // notice change of tempo
  1507. if( cmIsFlag(np->flags,kMetronomeXsFl) )
  1508. {
  1509. // ticks/sec = ticks/qn * qn/sec
  1510. ticksPerSec = mp->divisions * np->duration / 60.0;
  1511. }
  1512. // if this note is part of the grace note group we are searching for
  1513. if( np->graceGroupId == graceGroupId )
  1514. {
  1515. // track the first note in the grace note list
  1516. if( gn0p == NULL )
  1517. gn0p = np;
  1518. // add the note to the end of the grace note list
  1519. if( gn1p != NULL )
  1520. gn1p->grace = np;
  1521. // track the last note in the grace note list
  1522. gn1p = np;
  1523. // set each grace note to have 1/20 of a second duration
  1524. if( cmIsFlag(np->flags,kGraceXsFl) )
  1525. np->duration = np->tied_dur = floor(ticksPerSec * graceDurSec);
  1526. gN += 1;
  1527. }
  1528. } // for each note in this meassure
  1529. } // for each measure
  1530. } // for each part
  1531. // no records were found for this grace id - this grace group id was not used
  1532. if( gn0p == NULL )
  1533. continue;
  1534. // grace note groups must have at least 3 members
  1535. if( gN < 3 )
  1536. {
  1537. rc = cmErrMsg(&p->err,kSyntaxErrorXsRC,"The grace note group (groupid=%i) ending in meas %i has fewer than 3 (%i) members.", gn1p->graceGroupId, gn1p->meas->number, gN );
  1538. break;
  1539. }
  1540. // gn0p is now set to the first note in th group
  1541. // gn1p is now set to the last note in the group
  1542. // verify that the first note is marked with kBegGraceXsFl
  1543. if( cmIsNotFlag(gn0p->flags,kBegGraceXsFl) )
  1544. {
  1545. rc = cmErrMsg(&p->err,kSyntaxErrorXsRC,"The first note in a grace note group in meas %i tick %i groupid:%i is not marked with a 'b'.", gn0p->meas->number, gn0p->tick, graceGroupId );
  1546. break;
  1547. }
  1548. // verify that the last note is marked with kEndGraceXsFl
  1549. if( cmIsNotFlag(gn1p->flags,kEndGraceXsFl) )
  1550. {
  1551. rc = cmErrMsg(&p->err,kSyntaxErrorXsRC,"The last note in a grace note group in meas %i is not marked with a valid operator character.", gn1p->meas->number );
  1552. break;
  1553. }
  1554. // Count the total number of events between gn0p and gn1p
  1555. cmXsNote_t* n0p = NULL;
  1556. cmXsNote_t* n1p = gn0p;
  1557. cmXsMeas_t* mp = gn0p->meas;
  1558. unsigned aN = 0;
  1559. for(; n0p != gn1p; n1p=n1p->slink )
  1560. {
  1561. // if we are crossing a measure boundary
  1562. if( n1p == NULL )
  1563. {
  1564. mp = mp->link;
  1565. assert(mp!=NULL);
  1566. n1p = mp->noteL;
  1567. }
  1568. if(0)
  1569. {
  1570. bool fl = n0p != NULL && n0p->tick < n1p->tick;
  1571. unsigned type = n1p->flags & (kBegGraceXsFl|kEndGraceXsFl|kAddGraceXsFl|kSubGraceXsFl|kAFirstGraceXsFl|kNFirstGraceXsFl);
  1572. printf("%3i 0x%08x %i %3i %5i %i\n",n1p->graceGroupId,type,n1p->meas->number,n1p->tick,n1p->duration,fl);
  1573. }
  1574. ++aN;
  1575. n0p = n1p;
  1576. }
  1577. // create a vector of pointers to all events between gn0p and gn1p
  1578. cmXsNote_t* aV[ aN ];
  1579. unsigned i;
  1580. n1p = gn0p;
  1581. n0p = NULL;
  1582. mp = gn0p->meas;
  1583. for(i=0; n0p != gn1p; n1p=n1p->slink )
  1584. {
  1585. // if we are crossing a measure boundary
  1586. if( n1p == NULL )
  1587. {
  1588. mp = mp->link;
  1589. assert(mp!=NULL);
  1590. n1p = mp->noteL;
  1591. }
  1592. assert(i<aN);
  1593. aV[i++] = n1p;
  1594. n0p = n1p;
  1595. }
  1596. switch( gn1p->flags & (kAddGraceXsFl | kSubGraceXsFl | kAFirstGraceXsFl | kNFirstGraceXsFl ) )
  1597. {
  1598. case kAddGraceXsFl:
  1599. _cmXScoreGraceInsertTime(p, graceGroupId, aV, aN );
  1600. break;
  1601. case kSubGraceXsFl:
  1602. _cmXScoreGraceOverlayTime(p, graceGroupId, aV, aN );
  1603. break;
  1604. case kAFirstGraceXsFl:
  1605. _cmXScoreGraceInsertAfterFirst(p,graceGroupId,aV,aN);
  1606. break;
  1607. case kNFirstGraceXsFl:
  1608. _cmXScoreGraceInsertSoonAfterFirst(p,graceGroupId,aV,aN);
  1609. break;
  1610. default:
  1611. { assert(0); }
  1612. }
  1613. }
  1614. return rc;
  1615. }
  1616. cmXsNote_t* _cmXScoreDynamicForkEndNote( cmXScore_t* p, cmXsNote_t* bnp )
  1617. {
  1618. assert( cmIsFlag(bnp->flags,kDynBegForkXsFl) );
  1619. cmXsMeas_t* mp = bnp->meas;
  1620. cmXsNote_t* np = bnp->slink;
  1621. for(; mp!=NULL; mp=mp->link,np = mp==NULL?NULL : mp->noteL)
  1622. {
  1623. for(; np!=NULL; np=np->slink)
  1624. {
  1625. if( cmIsFlag(np->flags,kDynBegForkXsFl) )
  1626. {
  1627. cmErrMsg(&p->err,kSyntaxErrorXsRC,"The a dynamic fork begin (meas:%i loc:%i) was found prior to a matched end.",np->meas->number,np->locIdx);
  1628. return NULL;
  1629. }
  1630. if( cmIsFlag(np->flags,kDynEndForkXsFl) /*&& np->voice->id == bnp->voice->id*/ )
  1631. return np;
  1632. }
  1633. }
  1634. return NULL;
  1635. }
  1636. cmXsRC_t _cmXScoreProcessDynamicFork( cmXScore_t* p, cmXsNote_t* bnp )
  1637. {
  1638. cmXsNote_t* enp;
  1639. if((enp = _cmXScoreDynamicForkEndNote(p,bnp)) == NULL)
  1640. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"The dynamic fork beginning in measure %i at tick %i voice %i loc:%i was not terminated.", bnp->meas->number,bnp->tick,bnp->voice->id,bnp->locIdx);
  1641. //cmRptPrintf( p->err.rpt, "Dynamic Fork: meas:%i tick:%i loc:%i to meas:%i tick:%i loc:%i\n", bnp->meas->number, bnp->tick, bnp->locIdx, enp->meas->number, enp->tick, enp->locIdx );
  1642. double begDynFrac = bnp->dynamics;
  1643. double begVelFrac = bnp->vel;
  1644. double endDynFrac = enp->dynamics;
  1645. double endVelFrac = enp->vel;
  1646. cmXsMeas_t* mp = bnp->meas;
  1647. cmXsNote_t* np = bnp->slink;
  1648. while( mp != NULL )
  1649. {
  1650. for(; np!=NULL; np=np->slink)
  1651. {
  1652. //Conditioning this on np->dynamics==0 will may would cause 'silent' notes to be assigned
  1653. // a non-zero dynamics value - so we condition on np->vel instead.
  1654. if( cmIsFlag(np->flags,kOnsetXsFl) && np->vel == 0 )
  1655. {
  1656. double frac = ((double)np->tick - bnp->tick) / (enp->tick - bnp->tick);
  1657. np->dynamics = lround(begDynFrac + ((endDynFrac - begDynFrac) * frac));
  1658. np->vel = lround(begVelFrac + ((endVelFrac - begVelFrac) * frac));
  1659. }
  1660. if( np == enp )
  1661. break;
  1662. }
  1663. if( np == enp)
  1664. break;
  1665. if( (mp=mp->link) != NULL )
  1666. np=mp->noteL;
  1667. }
  1668. return kOkXsRC;
  1669. }
  1670. cmXsRC_t _cmXScoreProcessDynamicForks( cmXScore_t* p )
  1671. {
  1672. cmXsRC_t rc = kOkXsRC;
  1673. cmXsPart_t* pp = p->partL;
  1674. for(; pp!=NULL; pp=pp->link)
  1675. {
  1676. cmXsMeas_t* mp = pp->measL;
  1677. for(; mp!=NULL; mp=mp->link)
  1678. {
  1679. cmXsNote_t* np = mp->noteL;
  1680. for(; np!=NULL; np=np->slink )
  1681. {
  1682. if( cmIsFlag(np->flags,kDynBegForkXsFl) )
  1683. if((rc = _cmXScoreProcessDynamicFork(p,np)) != kOkXsRC )
  1684. return rc;
  1685. }
  1686. }
  1687. }
  1688. return rc;
  1689. }
  1690. //-------------------------------------------------------------------------------------------
  1691. typedef struct _cmXsReorder_str
  1692. {
  1693. unsigned idx; // Fields from the reordering input file which are
  1694. unsigned voice; // used to match the reorder record to
  1695. unsigned locIdx; // the associated a cmXsNode_t record.
  1696. unsigned tick; //
  1697. unsigned durtn; //
  1698. float rval; //
  1699. unsigned midi; //
  1700. cmXsNote_t* note; // The cmXsNote_t* associated with this cmXsReorder_t record
  1701. unsigned dynIdx; // cmInvalidIdx=ignore otherwise index into _cmXScoreDynMarkArray[]
  1702. unsigned newFlags; // 0=ignore | kSostUp/DnXsFl | kDampUp/DnXsFl | kTieEndXsFl
  1703. unsigned newTick; // 0=ignore >0 new tick value
  1704. unsigned graceFlags; // 0=ignore See kXXXGraceXsFl
  1705. unsigned graceGroupId; // 0=ignore >0=grace note group id
  1706. unsigned pitch; // 0=ignore >0 new pitch
  1707. const char* editStr;
  1708. struct _cmXsReorder_str* link;
  1709. } cmXsReorder_t;
  1710. typedef struct _cmXsReorderMeas_str
  1711. {
  1712. unsigned measNumb;
  1713. cmXsReorder_t* beg;
  1714. cmXsReorder_t* end;
  1715. struct _cmXsReorderMeas_str* link;
  1716. } cmXsReorderMeas_t;
  1717. typedef struct
  1718. {
  1719. cmXsReorderMeas_t* beg;
  1720. cmXsReorderMeas_t* end;
  1721. } cmXsReorderFile_t;
  1722. typedef struct _cmXScoreDynMark_str
  1723. {
  1724. const cmChar_t* mark; //
  1725. unsigned id; // (1-17) maps to velocity
  1726. unsigned dyn; // pppp - fff (1-9) as used by cmScore
  1727. int adjust; // {-1,0,+1}
  1728. unsigned vel; // associated MIDI velocity
  1729. } _cmXScoreDynMark_t;
  1730. _cmXScoreDynMark_t _cmXScoreDynMarkArray[] =
  1731. {
  1732. {"s", 1, 0, 0, 1}, // silent note
  1733. {"pppp-", 2, 1, -1, 3},
  1734. {"pppp", 3, 1, 0, 10},
  1735. {"pppp+", 4, 1, 1, 22},
  1736. {"ppp-", 4, 2, -1, 22},
  1737. {"ppp", 5, 2, 0, 29},
  1738. {"ppp+", 6, 2, 1, 36},
  1739. {"pp-", 6, 3, -1, 36},
  1740. {"pp", 7, 3, 0, 43},
  1741. {"pp+", 8, 3, 1, 50},
  1742. {"p-", 8, 4, -1, 50},
  1743. {"p", 9, 4, 0, 57},
  1744. {"p+", 10, 4, 1, 64},
  1745. {"mp-", 10, 5, -1, 64},
  1746. {"mp", 11, 5, 0, 71},
  1747. {"mp+", 12, 5, 1, 78},
  1748. {"mf-", 12, 6, -1, 78},
  1749. {"mf", 13, 6, 0, 85},
  1750. {"mf+", 14, 6, 1, 92},
  1751. {"f-", 14, 7, -1, 92},
  1752. {"f", 15, 7, 0, 99},
  1753. {"f+", 16, 7, 1, 106},
  1754. {"ff", 17, 8, 0, 113},
  1755. {"ff+", 18, 8, 1, 120},
  1756. {"fff", 19, 9, 0, 127},
  1757. {NULL,0,0,0,0}
  1758. };
  1759. cmXsReorderMeas_t* _cmXsReorderFileAllocMeas( cmXScore_t* p, cmXsReorderFile_t* rfp, unsigned measNumb )
  1760. {
  1761. cmXsReorderMeas_t* m = cmLhAllocZ(p->lhH,cmXsReorderMeas_t,1);
  1762. m->measNumb = measNumb;
  1763. if( rfp->end == NULL )
  1764. {
  1765. rfp->beg = m;
  1766. rfp->end = m;
  1767. }
  1768. else
  1769. {
  1770. rfp->end->link = m;
  1771. rfp->end = m;
  1772. }
  1773. return m;
  1774. }
  1775. cmXsReorderMeas_t* _cmXsReorderFileFindMeas( cmXsReorderFile_t* rfp, unsigned measNumb )
  1776. {
  1777. cmXsReorderMeas_t* m = rfp->beg;
  1778. for(; m!=NULL; m=m->link)
  1779. if( m->measNumb == measNumb )
  1780. return m;
  1781. return NULL;
  1782. }
  1783. cmXsReorder_t* _cmXsReorderMeasAllocEvent( cmXScore_t* p, cmXsReorderMeas_t* m )
  1784. {
  1785. cmXsReorder_t* r = cmLhAllocZ(p->lhH, cmXsReorder_t,1);
  1786. r->midi = kInvalidMidiPitch;
  1787. if( m->end == NULL )
  1788. {
  1789. m->beg = r;
  1790. m->end = r;
  1791. }
  1792. else
  1793. {
  1794. m->end->link = r;
  1795. m->end = r;
  1796. }
  1797. return r;
  1798. }
  1799. // find key in meas (m) by searching after event idx0.
  1800. cmXsReorder_t* _cmXsReorderFindEvent( cmXsReorderMeas_t* m, unsigned idx0, cmXsReorder_t* key )
  1801. {
  1802. cmXsReorder_t* r;
  1803. for(r=m->beg; r!=NULL; r=r->link)
  1804. if( r->idx >= idx0 && r->midi==key->midi && r->rval==key->rval && r->durtn==key->durtn )
  1805. return r;
  1806. return NULL;
  1807. }
  1808. cmXsReorder_t* _cmXsReorderMeasPop( cmXsReorderMeas_t* m )
  1809. {
  1810. cmXsReorder_t* r0 = NULL;
  1811. cmXsReorder_t* r = NULL;
  1812. for(r=m->beg; r!=NULL; r=r->link)
  1813. {
  1814. if( r == m->end )
  1815. break;
  1816. r0 = r;
  1817. }
  1818. if( r0 == NULL )
  1819. {
  1820. m->beg = NULL;
  1821. m->end = NULL;
  1822. }
  1823. else
  1824. {
  1825. m->end = r0;
  1826. m->end->link = NULL;
  1827. }
  1828. return r;
  1829. }
  1830. cmXsNote_t* _cmXsReorderFindNote( cmXScore_t* p, unsigned measNumb, const cmXsReorder_t* r, unsigned iii )
  1831. {
  1832. cmXsPart_t* pp = p->partL;
  1833. for(; pp!=NULL; pp=pp->link)
  1834. {
  1835. cmXsMeas_t* mp = pp->measL;
  1836. for(; mp!=NULL; mp=mp->link)
  1837. if( mp->number == measNumb)
  1838. {
  1839. cmXsNote_t* np = mp->noteL;
  1840. int index = 0;
  1841. for(; np!=NULL; np=np->slink,++index)
  1842. {
  1843. if( 0 /*mp->number==27*/ )
  1844. printf("voice: %i %i loc:%i %i tick:%i %i pitch:%i %i rval:%f %f idx:%i %i \n",
  1845. np->voice->id, r->voice,
  1846. np->locIdx , r->locIdx ,
  1847. np->tick , r->tick ,
  1848. np->pitch , r->midi ,
  1849. np->rvalue, r->rval,
  1850. index , r->idx);
  1851. if( np->voice->id == r->voice &&
  1852. np->locIdx == r->locIdx &&
  1853. np->tick == r->tick &&
  1854. //np->duration == r->durtn &&
  1855. np->rvalue == r->rval &&
  1856. np->pitch == r->midi &&
  1857. index == r->idx )
  1858. {
  1859. //printf("Found: %i \n", r->midi);
  1860. return np;
  1861. }
  1862. }
  1863. }
  1864. }
  1865. cmErrMsg(&p->err,kSyntaxErrorXsRC,"Reorder note not found meas:%i index:%i.",measNumb,iii);
  1866. return NULL;
  1867. }
  1868. void _cmXScoreInsertPedalEvent( cmXScore_t* p, const cmXsReorder_t* r, unsigned flags )
  1869. {
  1870. // Create a new score event record
  1871. cmXsNote_t* nn = cmLhAllocZ(p->lhH,cmXsNote_t,1);
  1872. nn->pitch = kInvalidMidiPitch;
  1873. nn->uid = p->nextUid++;
  1874. nn->voice = r->note->voice;
  1875. nn->meas = r->note->meas;
  1876. nn->flags = flags;
  1877. // Pedal down events occur after the event they are attached to
  1878. if( cmIsFlag(flags,kSostDnXsFl | kDampDnXsFl ) )
  1879. {
  1880. nn->tick = r->note->tick + 1;
  1881. _cmXScoreInsertNoteAfter(r->note,nn);
  1882. }
  1883. else
  1884. {
  1885. // Pedal up events occur before the event they are attached to
  1886. if( cmIsFlag(flags,kSostUpXsFl | kDampUpXsFl ) )
  1887. {
  1888. nn->tick = r->note->tick==0 ? 0 : r->note->tick - 1;
  1889. _cmXScoreInsertNoteBefore(r->note,nn);
  1890. }
  1891. else
  1892. { assert(0); }
  1893. }
  1894. }
  1895. cmXsRC_t _cmXScoreReorderMeas( cmXScore_t* p, cmXsReorderMeas_t* m )
  1896. {
  1897. unsigned i;
  1898. cmXsReorder_t* r;
  1899. if( m->beg == NULL )
  1900. return kOkXsRC;
  1901. // set the 'note' field on each cmXsReorder_t record
  1902. for(r=m->beg,i=0; r!=NULL; r=r->link,++i)
  1903. if((r->note = _cmXsReorderFindNote(p,m->measNumb,r,i)) == NULL )
  1904. return kSyntaxErrorXsRC;
  1905. // remove deleted notes
  1906. for(r=m->beg; r!=NULL; r=r->link)
  1907. if( cmIsFlag(r->newFlags,kDeleteXsFl) )
  1908. if( _cmXScoreRemoveNote( r->note ) != kOkXsRC )
  1909. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Event marked to skip was not found in measure: %i",m->measNumb);
  1910. cmXsMeas_t* mp = m->beg->note->meas;
  1911. cmXsNote_t* n0p = NULL;
  1912. assert( mp->number == m->measNumb );
  1913. // Reassign the slink of the cmXsNote_t records in this measure
  1914. // according to their order in rV[].
  1915. for(r=m->beg; r!=NULL; r=r->link)
  1916. {
  1917. if( cmIsFlag(r->newFlags,kDeleteXsFl) )
  1918. continue;
  1919. if( n0p == NULL )
  1920. mp->noteL = r->note;
  1921. else
  1922. n0p->slink = r->note;
  1923. // if a new tick was specified
  1924. if( r->newTick != 0 )
  1925. r->note->tick = r->newTick;
  1926. // if a dynamic or velocity mark was included
  1927. if( r->dynIdx != cmInvalidIdx )
  1928. {
  1929. r->note->dynamics = _cmXScoreDynMarkArray[ r->dynIdx ].dyn;
  1930. r->note->vel = _cmXScoreDynMarkArray[ r->dynIdx ].vel;
  1931. }
  1932. // Set the dynamic fork begin/end flags for later _cmXScoreProcessDynamicForks()
  1933. if( cmIsFlag(r->newFlags,kDynBegForkXsFl) )
  1934. r->note->flags = cmSetFlag(r->note->flags,kDynBegForkXsFl);
  1935. if( cmIsFlag(r->newFlags,kDynEndForkXsFl) )
  1936. r->note->flags = cmSetFlag(r->note->flags,kDynEndForkXsFl);
  1937. // if the tie end flag was set
  1938. if( cmIsFlag(r->newFlags,kTieEndXsFl) )
  1939. {
  1940. r->note->flags |= kTieEndXsFl;
  1941. r->note->flags = cmClrFlag( r->note->flags, kOnsetXsFl );
  1942. r->newFlags = cmClrFlag( r->newFlags, kTieEndXsFl);
  1943. }
  1944. // if a new note value was specified
  1945. if( r->pitch != 0 )
  1946. r->note->pitch = r->pitch;
  1947. r->note->flags |= r->graceFlags;
  1948. r->note->graceGroupId = r->graceGroupId;
  1949. n0p = r->note;
  1950. n0p->slink = NULL;
  1951. }
  1952. // Insert new note records for pedal up/dn events.
  1953. for(r=m->beg; r!=NULL; r=r->link)
  1954. {
  1955. if( r->newFlags != 0 )
  1956. {
  1957. if( cmIsFlag(r->newFlags,kDampDnXsFl ) )
  1958. _cmXScoreInsertPedalEvent(p,r,kDampDnXsFl);
  1959. if( cmIsFlag(r->newFlags,kSostDnXsFl ) )
  1960. _cmXScoreInsertPedalEvent(p,r,kSostDnXsFl);
  1961. if( cmIsFlag(r->newFlags,kDampUpXsFl ) )
  1962. _cmXScoreInsertPedalEvent(p,r,kDampUpXsFl);
  1963. if( cmIsFlag(r->newFlags,kSostUpXsFl ) )
  1964. _cmXScoreInsertPedalEvent(p,r,kSostUpXsFl);
  1965. }
  1966. }
  1967. return kOkXsRC;
  1968. }
  1969. cmXsRC_t _cmXScoreReorderMeas0( cmXScore_t* p, unsigned measNumb, cmXsReorder_t* rV, unsigned rN )
  1970. {
  1971. unsigned i;
  1972. if( rN == 0 )
  1973. return kOkXsRC;
  1974. // set the 'note' field on each cmXsReorder_t record
  1975. for(i=0; i<rN; ++i)
  1976. if((rV[i].note = _cmXsReorderFindNote(p,measNumb,rV+i,i)) == NULL )
  1977. return kSyntaxErrorXsRC;
  1978. // remove deleted notes
  1979. for(i=0; i<rN; ++i)
  1980. if( cmIsFlag(rV[i].newFlags,kDeleteXsFl) )
  1981. if( _cmXScoreRemoveNote( rV[i].note ) != kOkXsRC )
  1982. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Event marked to skip was not found in measure: %i",measNumb);
  1983. cmXsMeas_t* mp = rV[0].note->meas;
  1984. cmXsNote_t* n0p = NULL;
  1985. assert( mp->number == measNumb );
  1986. // Reassign the slink of the cmXsNote_t records in this measure
  1987. // according to their order in rV[].
  1988. for(i=0; i<rN; ++i)
  1989. {
  1990. if( cmIsFlag(rV[i].newFlags,kDeleteXsFl) )
  1991. continue;
  1992. if( n0p == NULL )
  1993. mp->noteL = rV[i].note;
  1994. else
  1995. n0p->slink = rV[i].note;
  1996. // if a new tick was specified
  1997. if( rV[i].newTick != 0 )
  1998. rV[i].note->tick = rV[i].newTick;
  1999. // if a dynamic or velocity mark was included
  2000. if( rV[i].dynIdx != cmInvalidIdx )
  2001. {
  2002. rV[i].note->dynamics = _cmXScoreDynMarkArray[ rV[i].dynIdx ].dyn;
  2003. rV[i].note->vel = _cmXScoreDynMarkArray[ rV[i].dynIdx ].vel;
  2004. }
  2005. // Set the dynamic fork begin/end flags for later _cmXScoreProcessDynamicForks()
  2006. if( cmIsFlag(rV[i].newFlags,kDynBegForkXsFl) )
  2007. rV[i].note->flags = cmSetFlag(rV[i].note->flags,kDynBegForkXsFl);
  2008. if( cmIsFlag(rV[i].newFlags,kDynEndForkXsFl) )
  2009. rV[i].note->flags = cmSetFlag(rV[i].note->flags,kDynEndForkXsFl);
  2010. // if the tie end flag was set
  2011. if( cmIsFlag(rV[i].newFlags,kTieEndXsFl) )
  2012. {
  2013. rV[i].note->flags |= kTieEndXsFl;
  2014. rV[i].note->flags = cmClrFlag( rV[i].note->flags, kOnsetXsFl );
  2015. rV[i].newFlags = cmClrFlag( rV[i].newFlags, kTieEndXsFl);
  2016. }
  2017. // if a new note value was specified
  2018. if( rV[i].pitch != 0 )
  2019. rV[i].note->pitch = rV[i].pitch;
  2020. rV[i].note->flags |= rV[i].graceFlags;
  2021. rV[i].note->graceGroupId = rV[i].graceGroupId;
  2022. n0p = rV[i].note;
  2023. n0p->slink = NULL;
  2024. }
  2025. // Insert new note records for pedal up/dn events.
  2026. for(i=0; i<rN; ++i)
  2027. {
  2028. if( rV[i].newFlags != 0 )
  2029. {
  2030. if( cmIsFlag(rV[i].newFlags,kDampDnXsFl ) )
  2031. _cmXScoreInsertPedalEvent(p,rV + i,kDampDnXsFl);
  2032. if( cmIsFlag(rV[i].newFlags,kSostDnXsFl ) )
  2033. _cmXScoreInsertPedalEvent(p,rV + i,kSostDnXsFl);
  2034. if( cmIsFlag(rV[i].newFlags,kDampUpXsFl ) )
  2035. _cmXScoreInsertPedalEvent(p,rV + i,kDampUpXsFl);
  2036. if( cmIsFlag(rV[i].newFlags,kSostUpXsFl ) )
  2037. _cmXScoreInsertPedalEvent(p,rV + i,kSostUpXsFl);
  2038. }
  2039. }
  2040. return kOkXsRC;
  2041. }
  2042. cmXsRC_t _cmXScoreReorderParseDyn(cmXScore_t* p, const cmChar_t* b, unsigned lineNumb, char** s0, unsigned* dynIdxRef, unsigned* flagsRef, int measNumb )
  2043. {
  2044. cmXsRC_t rc = kOkXsRC;
  2045. cmChar_t* s = NULL;
  2046. bool begForkFl = false;
  2047. bool endForkFl = false;
  2048. *dynIdxRef = cmInvalidIdx;
  2049. // locate the '!' which indicates the start of a dynamic marking
  2050. if( (s = strchr(b,'!')) == NULL )
  2051. return rc;
  2052. if( *s0==NULL || s<*s0 )
  2053. *s0 = s;
  2054. ++s; // increment past the '!'
  2055. if( *s == 0 )
  2056. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unexpected end-of-line on dynamics parsing on line:%i.",lineNumb);
  2057. // some dynamic markings are surrounded by parenthesis (to indicate a dynamic level with greater uncertainty)
  2058. if( *s == '(' )
  2059. ++s; // skip the paren.
  2060. if( *s == '!')
  2061. {
  2062. //printf("E %3i %5i %s\n",measNumb,lineNumb,b);
  2063. endForkFl = true;
  2064. ++s;
  2065. }
  2066. if( *s == 0 )
  2067. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unexpected end-of-line on dynamics parsing on line:%i.",lineNumb);
  2068. // check for beg/end fork dynamic
  2069. if( isupper(*s) )
  2070. {
  2071. if( !endForkFl)
  2072. {
  2073. begForkFl=true;
  2074. //printf("B %3i %5i %s\n",measNumb,lineNumb,b);
  2075. }
  2076. }
  2077. else
  2078. {
  2079. if( endForkFl )
  2080. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"A double exclaimation point (end-of-dynamic fork) must always be followed by an upper case M,P, or F. See line:%i",lineNumb);
  2081. }
  2082. unsigned i = 0;
  2083. unsigned j = 0;
  2084. unsigned n = 6;
  2085. bool doneFl = false;
  2086. cmChar_t mark[n+1];
  2087. memset(mark,0,n+1);
  2088. for(i=0; j<n && doneFl==false; ++i)
  2089. {
  2090. char c = tolower(s[i]);
  2091. switch(c)
  2092. {
  2093. case 's':
  2094. case 'm':
  2095. case 'p':
  2096. case 'f':
  2097. case '+':
  2098. case '-':
  2099. mark[j++] = c;
  2100. break;
  2101. case ')': // ending paren.
  2102. case 0: // end of string
  2103. case ' ': // end of mark
  2104. case '\n': // end of line
  2105. default: // anything else
  2106. doneFl = true;
  2107. break;
  2108. }
  2109. }
  2110. if( !doneFl )
  2111. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Illegal dynamic mark (%s) syntax on line:%i.",mark,lineNumb);
  2112. // look up the dynamic mark in the reference table
  2113. for(j=0; _cmXScoreDynMarkArray[j].mark!=NULL; ++j)
  2114. if( strcmp(mark,_cmXScoreDynMarkArray[j].mark) == 0 )
  2115. break;
  2116. if( _cmXScoreDynMarkArray[j].mark == NULL )
  2117. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"The dynamic mark '%s' is not legal on line:%i.",mark,lineNumb);
  2118. *dynIdxRef = j;
  2119. if( begForkFl )
  2120. *flagsRef = cmSetFlag(*flagsRef,kDynBegForkXsFl);
  2121. if( endForkFl )
  2122. *flagsRef = cmSetFlag(*flagsRef,kDynEndForkXsFl);
  2123. return rc;
  2124. }
  2125. cmXsRC_t _cmXScoreReorderParseFlags(cmXScore_t* p, const cmChar_t* b, unsigned line, char** s0, unsigned* newFlagsRef )
  2126. {
  2127. cmXsRC_t rc = kOkXsRC;
  2128. cmChar_t* s;
  2129. bool doneFl = false;
  2130. unsigned i = 0;
  2131. *newFlagsRef = 0;
  2132. // tilde indicates a pedal event
  2133. if((s = strchr(b,'~')) == NULL )
  2134. return rc;
  2135. if( *s0==NULL || s<*s0)
  2136. *s0 = s;
  2137. do
  2138. {
  2139. ++s;
  2140. switch( *s )
  2141. {
  2142. case 'd':
  2143. *newFlagsRef |= kSostDnXsFl; // sostenuto pedal down just after this note onset
  2144. break;
  2145. case 'u':
  2146. *newFlagsRef |= kSostUpXsFl; // sostenuto pedal up just before this event
  2147. break;
  2148. case 'x':
  2149. *newFlagsRef |= (kSostUpXsFl | kSostDnXsFl); // sostenuto pedal up just before this event and sost down just after it.
  2150. break;
  2151. case 'D':
  2152. *newFlagsRef |= kDampDnXsFl; // damper pedal down
  2153. break;
  2154. case 'U':
  2155. *newFlagsRef |= kDampUpXsFl; // damper pedal up
  2156. break;
  2157. case '_':
  2158. *newFlagsRef |= kTieEndXsFl; // set tie end flag
  2159. break;
  2160. case '&':
  2161. *newFlagsRef |= kDeleteXsFl; // delete this evetn
  2162. break;
  2163. default:
  2164. if( i == 0 )
  2165. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unexpected flag marking '%c' on line %i.",*s,line);
  2166. doneFl = true;
  2167. }
  2168. ++i;
  2169. }while(!doneFl);
  2170. return rc;
  2171. }
  2172. cmXsRC_t _cmXScoreReorderParseTick(cmXScore_t* p, const cmChar_t* b, unsigned line, char** s0, unsigned* tickRef )
  2173. {
  2174. cmXsRC_t rc = kOkXsRC;
  2175. cmChar_t* s;
  2176. if((s = strchr(b,'@')) == NULL )
  2177. return rc;
  2178. if( *s0 == NULL || s<*s0 )
  2179. *s0 = s;
  2180. ++s;
  2181. if(!isdigit(*s))
  2182. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unexpected tick reorder value '%c' on line %i.",*s,line);
  2183. if(sscanf(s,"%i",tickRef) != 1 )
  2184. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"tick reorder parse failed on line %i.",line);
  2185. return rc;
  2186. }
  2187. cmXsRC_t _cmXScoreReorderParseGrace(cmXScore_t* p, const cmChar_t* b, unsigned line, char** s0, cmXsReorder_t* r, unsigned* graceGroupIdRef )
  2188. {
  2189. cmXsRC_t rc = kOkXsRC;
  2190. cmChar_t* s;
  2191. if((s = strchr(b,'%')) == NULL )
  2192. return rc;
  2193. if( *s0==NULL || s<*s0 )
  2194. *s0 = s;
  2195. ++s;
  2196. r->graceGroupId = *graceGroupIdRef;
  2197. while(1)
  2198. {
  2199. switch(*s)
  2200. {
  2201. case 'b': r->graceFlags |= kBegGraceXsFl; break;
  2202. case 'a': r->graceFlags |= kAddGraceXsFl | kEndGraceXsFl; break;
  2203. case 's': r->graceFlags |= kSubGraceXsFl | kEndGraceXsFl; break;
  2204. case 'A': r->graceFlags |= kAFirstGraceXsFl| kEndGraceXsFl; break;
  2205. case 'n': r->graceFlags |= kNFirstGraceXsFl| kEndGraceXsFl; break;
  2206. case 'g': break;
  2207. case '1':
  2208. r->graceGroupId += 1;
  2209. ++s;
  2210. continue;
  2211. case '%':
  2212. *graceGroupIdRef += 1;
  2213. ++s;
  2214. continue;
  2215. default:
  2216. {
  2217. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Unexpected grace note reorder character code %c on line %i.",*s,line);
  2218. assert(0);
  2219. }
  2220. }
  2221. break;
  2222. }
  2223. return rc;
  2224. }
  2225. cmXsRC_t _cmXScoreReorderParsePitch(cmXScore_t* p, const cmChar_t* b, unsigned line, char** s0, unsigned* pitchRef )
  2226. {
  2227. cmXsRC_t rc = kOkXsRC;
  2228. cmChar_t* s;
  2229. cmChar_t buf[4];
  2230. unsigned i,j;
  2231. memset(buf,0,sizeof(buf));
  2232. *pitchRef = 0;
  2233. if((s = strchr(b,'$')) == NULL )
  2234. return rc;
  2235. if( *s0==NULL || s<*s0)
  2236. *s0 = s;
  2237. ++s;
  2238. j=2;
  2239. for(i=0; i<j && *s; ++i,++s)
  2240. {
  2241. buf[i] = *s;
  2242. if( i==1 && (*s=='#' || *s=='b') )
  2243. j = 3;
  2244. if( i==0 && strchr("ABCDEFG",*s)==NULL )
  2245. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Illegal pitch letter ('%c')specification line %i.",*s,line);
  2246. if( i==1 && !isdigit(*s) && *s!='#' && *s!='b' )
  2247. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Illegal pitch level ('%c') specification line %i.",*s,line);
  2248. if( i==2 && !isdigit(*s) )
  2249. return cmErrMsg(&p->err,kSyntaxErrorXsRC,"Illegal pitch octave ('%c') specification line %i.",*s,line);
  2250. }
  2251. unsigned pitch = cmSciPitchToMidi(buf);
  2252. if( pitch<kInvalidMidiByte)
  2253. *pitchRef = pitch;
  2254. else
  2255. {
  2256. rc = cmErrMsg(&p->err,kSyntaxErrorXsRC,"Pitch conversion from '%s' failed on line %i.",buf,line);
  2257. }
  2258. return rc;
  2259. }
  2260. cmXsRC_t _cmXsReadEditFile( cmXScore_t* p, const cmChar_t* fn, unsigned* graceGroupIdPtr, cmXsReorderFile_t* rfp )
  2261. {
  2262. typedef enum { kFindMeasStId, kFindEventStId, kReadEventStId } stateId_t;
  2263. cmXsRC_t rc = kOkXsRC;
  2264. cmFileH_t fH = cmFileNullHandle;
  2265. cmChar_t* b = NULL;
  2266. unsigned bN = 0;
  2267. unsigned ln = 0;
  2268. stateId_t stateId = kFindMeasStId;
  2269. cmXsReorderMeas_t* curMeas = NULL;
  2270. *graceGroupIdPtr = 1;
  2271. if( cmFileOpen(&fH,fn,kReadFileFl,p->err.rpt) != kOkFileRC )
  2272. {
  2273. rc = cmErrMsg(&p->err,kFileFailXsRC,"The reordering file '%s' could not be opened.",cmStringNullGuard(fn));
  2274. return rc;
  2275. }
  2276. for(; cmFileGetLineAuto(fH,&b,&bN) == kOkFileRC; ++ln)
  2277. {
  2278. switch( stateId )
  2279. {
  2280. case kFindEventStId: // scanning past labels to an event line
  2281. {
  2282. unsigned voice,loc;
  2283. if( sscanf(b,"%i %i",&voice,&loc) != 2 )
  2284. continue;
  2285. stateId = kReadEventStId;
  2286. }
  2287. // fall through
  2288. case kReadEventStId:
  2289. {
  2290. cmXsReorder_t* r = _cmXsReorderMeasAllocEvent(p, curMeas );
  2291. char pitchStr[4];
  2292. char* s0 = NULL;
  2293. // parse an event line
  2294. if( sscanf(b,"%i %i %i %i %i %f",&r->idx,&r->voice,&r->locIdx,&r->tick,&r->durtn,&r->rval) == 6 )
  2295. {
  2296. assert( strlen(b)>=52);
  2297. int PC = 39; // text file column where first pitch char occurs
  2298. if( b[PC] == ' ')
  2299. r->midi = kInvalidMidiPitch;
  2300. else
  2301. {
  2302. pitchStr[0] = b[PC+0];
  2303. pitchStr[1] = b[PC+1];
  2304. pitchStr[2] = b[PC+2];
  2305. pitchStr[3] = 0;
  2306. if( !isdigit(pitchStr[2]) )
  2307. r->midi = kInvalidMidiPitch;
  2308. else
  2309. {
  2310. if( pitchStr[1] == ' ')
  2311. {
  2312. pitchStr[1] = pitchStr[2];
  2313. pitchStr[2] = 0;
  2314. }
  2315. r->midi = cmSciPitchToMidi(pitchStr);
  2316. //printf("%i %i %s %s\n",curMeas->measNumb,r->midi,pitchStr,fn);
  2317. }
  2318. }
  2319. // parse the flag edits following a '~'
  2320. if((rc = _cmXScoreReorderParseFlags(p,b,ln+1, &s0, &r->newFlags)) != kOkXsRC )
  2321. goto errLabel;
  2322. // parse the dynamic marking following a '!'
  2323. if((rc = _cmXScoreReorderParseDyn(p,b,ln+1, &s0, &r->dynIdx, &r->newFlags, curMeas->measNumb)) != kOkXsRC )
  2324. goto errLabel;
  2325. // parse the @newtick marker
  2326. if((rc = _cmXScoreReorderParseTick(p, b, ln+1, &s0, &r->newTick)) != kOkXsRC )
  2327. goto errLabel;
  2328. // parse the %grace note marker
  2329. if((rc = _cmXScoreReorderParseGrace(p, b, ln+1, &s0, r, graceGroupIdPtr)) != kOkXsRC )
  2330. goto errLabel;
  2331. // parse the $pitch marker
  2332. if((rc = _cmXScoreReorderParsePitch(p, b, ln+1, &s0, &r->pitch )) != kOkXsRC )
  2333. goto errLabel;
  2334. if( s0 != NULL )
  2335. r->editStr = cmTextTrimEnd(cmLhAllocStrN( p->lhH, s0, strlen(s0)+1 ));
  2336. continue;
  2337. }
  2338. // remove the last reorder record because it was not filled
  2339. _cmXsReorderMeasPop(curMeas);
  2340. stateId = kFindMeasStId;
  2341. // fall through
  2342. }
  2343. case kFindMeasStId: // scanning for a bar-line
  2344. {
  2345. char colon;
  2346. unsigned measNumb = 0;
  2347. if( sscanf(b,"%i %c",&measNumb,&colon) == 2 && colon == ':' )
  2348. {
  2349. curMeas = _cmXsReorderFileAllocMeas( p, rfp, measNumb );
  2350. stateId = kFindEventStId;
  2351. }
  2352. }
  2353. break;
  2354. }
  2355. }
  2356. errLabel:
  2357. cmMemFree(b);
  2358. cmFileClose(&fH);
  2359. return rc;
  2360. }
  2361. cmXsRC_t _cmXsApplyEditFile( cmXScore_t* p, const cmChar_t* fn )
  2362. {
  2363. cmXsRC_t rc = kOkXsRC;
  2364. unsigned graceGroupId = 1;
  2365. cmXsReorderFile_t rf;
  2366. cmXsReorderMeas_t* m;
  2367. memset(&rf,0,sizeof(rf));
  2368. if((rc = _cmXsReadEditFile( p, fn, &graceGroupId, &rf )) != kOkXsRC )
  2369. return rc;
  2370. // reorder each measure
  2371. for(m=rf.beg; m!=NULL; m=m->link)
  2372. if((rc = _cmXScoreReorderMeas(p, m)) != kOkXsRC )
  2373. goto errLabel;
  2374. // the ticks may have changed so the 'secs' and 'dsecs' must be updated
  2375. _cmXScoreSetAbsoluteTime( p );
  2376. // the bar lines should be the first event in the measure
  2377. _cmXScoreFixBarLines(p);
  2378. // resort to force the links to be correct
  2379. _cmXScoreSort(p);
  2380. // process the grace notes.
  2381. _cmXScoreProcessGraceNotes( p, graceGroupId );
  2382. // inserting grace notes may have left the score unsorted
  2383. _cmXScoreSort(p);
  2384. // process the dynamic forks
  2385. _cmXScoreProcessDynamicForks(p);
  2386. //_cmXScoreReport(p, NULL, true );
  2387. errLabel:
  2388. return rc;
  2389. }
  2390. cmXsRC_t _cmXsMergeEditFiles( cmXScore_t* p, unsigned measNumb0, const cmChar_t* keyEditFn, unsigned keyMeasNumb, const cmChar_t* outFn )
  2391. {
  2392. cmXsRC_t rc = kOkXsRC;
  2393. unsigned graceGroup1Id = 1;
  2394. unsigned measNumb1 = keyMeasNumb;
  2395. cmXsReorderFile_t rf1;
  2396. memset(&rf1,0,sizeof(rf1));
  2397. if((rc = _cmXsReadEditFile( p, keyEditFn, &graceGroup1Id, &rf1 )) != kOkXsRC )
  2398. return rc;
  2399. while(1)
  2400. {
  2401. cmXsMeas_t* m0 = _cmXsFindMeas( p->partL, measNumb0 );
  2402. cmXsReorderMeas_t* m1 = _cmXsReorderFileFindMeas( &rf1, measNumb1 );
  2403. cmXsReorder_t* key = NULL;
  2404. unsigned idx0 = 0;
  2405. if( m1==NULL )
  2406. {
  2407. rc = cmErrMsg(&p->err,kEventNotFoundXsRC,"The measure %i was not found in the key edit file '%s'.",keyMeasNumb,cmStringNullGuard(keyEditFn));
  2408. break;
  2409. }
  2410. key = m1->beg;
  2411. for(; key!=NULL; key=key->link)
  2412. {
  2413. unsigned idx1 = cmInvalidIdx;
  2414. cmXsNote_t* np = _cmXsFindNote( m0, idx0, key->midi, key->rval, key->durtn, &idx1);
  2415. if( np==NULL )
  2416. {
  2417. if( key->editStr != NULL )
  2418. {
  2419. const char* sciPitch = key->midi!=kInvalidMidiPitch ? cmMidiToSciPitch(key->midi,NULL,0) : "<non-pitch>";
  2420. cmErrWarnMsg(&p->err,kEventNotFoundXsRC,"Sync error: meas: ref:%i key:%i index:%i %s (midi:%i) edit:%s did not match to the reference edit file.", measNumb0,m1->measNumb,key->idx, sciPitch, key->midi, key->editStr);
  2421. }
  2422. }
  2423. else
  2424. {
  2425. np->editStr = key->editStr;
  2426. if( key->editStr != NULL )
  2427. idx0 = idx1;
  2428. }
  2429. }
  2430. ++measNumb0;
  2431. ++measNumb1;
  2432. }
  2433. return rc;
  2434. }
  2435. cmXsRC_t _cmXsApplyEditFile0( cmXScore_t* p, const cmChar_t* fn )
  2436. {
  2437. typedef enum { kFindMeasStId, kFindEventStId, kReadEventStId } stateId_t;
  2438. cmXsRC_t rc = kOkXsRC;
  2439. cmFileH_t fH = cmFileNullHandle;
  2440. cmChar_t* b = NULL;
  2441. unsigned bN = 0;
  2442. unsigned ln = 0;
  2443. stateId_t stateId = kFindMeasStId;
  2444. unsigned rN = 1024;
  2445. unsigned ri = 0;
  2446. unsigned measNumb = 0;
  2447. unsigned graceGroupId = 1;
  2448. cmXsReorder_t rV[ rN ];
  2449. if( cmFileOpen(&fH,fn,kReadFileFl,p->err.rpt) != kOkFileRC )
  2450. {
  2451. rc = cmErrMsg(&p->err,kFileFailXsRC,"The reordering file '%s' could not be opened.",cmStringNullGuard(fn));
  2452. return rc;
  2453. }
  2454. for(; cmFileGetLineAuto(fH,&b,&bN)==kOkFileRC; ++ln)
  2455. {
  2456. switch( stateId )
  2457. {
  2458. case kFindEventStId: // scanning past labels to an event line
  2459. {
  2460. unsigned voice,loc;
  2461. if( sscanf(b,"%i %i",&voice,&loc) != 2 )
  2462. continue;
  2463. stateId = kReadEventStId;
  2464. }
  2465. // fall through
  2466. case kReadEventStId:
  2467. {
  2468. cmXsReorder_t r;
  2469. char pitchStr[4];
  2470. char* s0 = NULL;
  2471. memset(&r,0,sizeof(r));
  2472. // parse an event line
  2473. if( sscanf(b,"%i %i %i %i %i %f",&r.idx,&r.voice,&r.locIdx,&r.tick,&r.durtn,&r.rval) == 6 )
  2474. {
  2475. assert( strlen(b)>=52);
  2476. int PC = 39; // text file column where first pitch char occurs
  2477. if( b[PC] == ' ')
  2478. r.midi = kInvalidMidiPitch;
  2479. else
  2480. {
  2481. pitchStr[0] = b[PC+0];
  2482. pitchStr[1] = b[PC+1];
  2483. pitchStr[2] = b[PC+2];
  2484. pitchStr[3] = 0;
  2485. if( !isdigit(pitchStr[2]) )
  2486. r.midi = kInvalidMidiPitch;
  2487. else
  2488. {
  2489. if( pitchStr[1] == ' ')
  2490. {
  2491. pitchStr[1] = pitchStr[2];
  2492. pitchStr[2] = 0;
  2493. }
  2494. r.midi = cmSciPitchToMidi(pitchStr);
  2495. }
  2496. }
  2497. // parse the flag edits following a '~'
  2498. if((rc = _cmXScoreReorderParseFlags(p,b,ln+1, &s0, &r.newFlags)) != kOkXsRC )
  2499. goto errLabel;
  2500. // parse the dynamic marking following a '!'
  2501. if((rc = _cmXScoreReorderParseDyn(p,b,ln+1,&s0, &r.dynIdx, &r.newFlags, measNumb)) != kOkXsRC )
  2502. goto errLabel;
  2503. // parse the @newtick marker
  2504. if((rc = _cmXScoreReorderParseTick(p, b, ln+1, &s0, &r.newTick)) != kOkXsRC )
  2505. goto errLabel;
  2506. // parse the %grace note marker
  2507. if((rc = _cmXScoreReorderParseGrace(p, b, ln+1, &s0, &r, &graceGroupId)) != kOkXsRC )
  2508. goto errLabel;
  2509. // parse the $pitch marker
  2510. if((rc = _cmXScoreReorderParsePitch(p, b, ln+1, &s0, &r.pitch )) != kOkXsRC )
  2511. goto errLabel;
  2512. // store the record
  2513. assert( ri < rN );
  2514. rV[ri++] = r;
  2515. continue;
  2516. }
  2517. // the end of the measure was encountered -
  2518. // reorder the measure based on the cmXsReorder_t in rV[ri]
  2519. if((rc = _cmXScoreReorderMeas0(p, measNumb, rV, ri )) != kOkXsRC )
  2520. goto errLabel;
  2521. ri = 0;
  2522. stateId = kFindMeasStId;
  2523. // fall through
  2524. }
  2525. case kFindMeasStId: // scanning for a bar-line
  2526. {
  2527. char colon;
  2528. if( sscanf(b,"%i %c",&measNumb,&colon) == 2 && colon == ':' )
  2529. {
  2530. //printf("meas: %i \n",measNumb);
  2531. stateId = kFindEventStId;
  2532. }
  2533. }
  2534. break;
  2535. }
  2536. }
  2537. // If reorder records remain to be processed
  2538. if( ri > 0 )
  2539. if((rc = _cmXScoreReorderMeas0(p, measNumb, rV, ri )) != kOkXsRC )
  2540. goto errLabel;
  2541. // the ticks may have changed so the 'secs' and 'dsecs' must be updated
  2542. _cmXScoreSetAbsoluteTime( p );
  2543. // the bar lines should be the first event in the measure
  2544. _cmXScoreFixBarLines(p);
  2545. // resort to force the links to be correct
  2546. _cmXScoreSort(p);
  2547. // process the grace notes.
  2548. _cmXScoreProcessGraceNotes( p, graceGroupId );
  2549. // inserting grace notes may have left the score unsorted
  2550. _cmXScoreSort(p);
  2551. // process the dynamic forks
  2552. _cmXScoreProcessDynamicForks(p);
  2553. //_cmXScoreReport(p, NULL, true );
  2554. errLabel:
  2555. cmFileClose(&fH);
  2556. cmMemFree(b);
  2557. return rc;
  2558. }
  2559. cmXsRC_t cmXScoreAlloc( cmCtx_t* ctx, cmXsH_t* hp )
  2560. {
  2561. cmXsRC_t rc = kOkXsRC;
  2562. if((rc = cmXScoreFinalize(hp)) != kOkXsRC )
  2563. return rc;
  2564. cmXScore_t* p = cmMemAllocZ(cmXScore_t,1);
  2565. cmErrSetup(&p->err,&ctx->rpt,"XScore");
  2566. // create a local linked heap
  2567. if( cmLHeapIsValid( p->lhH = cmLHeapCreate(8196,ctx)) == false )
  2568. return cmErrMsg(&p->err,kLHeapFailXsRC,"Lheap create failed.");
  2569. hp->h = p;
  2570. return rc;
  2571. }
  2572. cmXsRC_t cmXScoreInitialize( cmCtx_t* ctx, cmXsH_t* hp, const cmChar_t* xmlFn, const cmChar_t* editFn, bool damperRptFl )
  2573. {
  2574. cmXsRC_t rc;
  2575. cmXScore_t* p = NULL;
  2576. if((rc = cmXScoreAlloc(ctx,hp)) != kOkXsRC )
  2577. goto errLabel;
  2578. p = _cmXScoreHandleToPtr(*hp);
  2579. // open the music xml file
  2580. if( cmXmlAlloc(ctx, &p->xmlH, xmlFn) != kOkXmlRC )
  2581. {
  2582. rc = cmErrMsg(&p->err,kXmlFailXsRC,"Unable to open the MusicXML file '%s'.",cmStringNullGuard(xmlFn));
  2583. goto errLabel;
  2584. }
  2585. //cmXmlPrint(p->xmlH,&ctx->rpt);
  2586. // parse the part-list
  2587. if((rc = _cmXScoreParsePartList( p )) != kOkXsRC )
  2588. goto errLabel;
  2589. // parse each score 'part'
  2590. cmXsPart_t* pp = p->partL;
  2591. for(; pp!=NULL; pp=pp->link)
  2592. if((rc = _cmXScoreParsePart(p,pp)) != kOkXsRC )
  2593. goto errLabel;
  2594. // fill in the note->slink chain to link the notes in each measure in time order
  2595. _cmXScoreSort(p);
  2596. // kpl: 4/19/17 fix problem where notes ended up out of order by one tick _cmXScoreSpreadGraceNotes(p);
  2597. _cmXScoreSort(p);
  2598. _cmXScoreResolveTiesAndLoc(p);
  2599. _cmXScoreRemoveDuplicateNotes(p);
  2600. _cmXScoreSetMeasGroups(p,kEvenXsFl);
  2601. _cmXScoreSetMeasGroups(p,kDynXsFl);
  2602. _cmXScoreSetMeasGroups(p,kTempoXsFl);
  2603. //_cmXScoreResolveOctaveShift(p);
  2604. // CSV output initialize failed.
  2605. if( cmCsvInitialize(&p->csvH,ctx) != kOkCsvRC )
  2606. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV output object create failed.");
  2607. if( editFn != NULL )
  2608. {
  2609. if((rc = _cmXsApplyEditFile(p,editFn)) != kOkXsRC )
  2610. {
  2611. cmErrMsg(&ctx->err,rc,"XScore reorder failed.");
  2612. goto errLabel;
  2613. }
  2614. }
  2615. // assign durations to pedal down events
  2616. _cmXScoreProcessPedals(p,damperRptFl);
  2617. // remove some notes which share a pitch and are overlapped or embedded within another note.
  2618. _cmXScoreProcessOverlappingNotes(p);
  2619. errLabel:
  2620. if( rc != kOkXsRC )
  2621. cmXScoreFinalize(hp);
  2622. return rc;
  2623. }
  2624. cmXsRC_t cmXScoreFinalize( cmXsH_t* hp )
  2625. {
  2626. cmXsRC_t rc = kOkXsRC;
  2627. if( hp == NULL || cmXScoreIsValid(*hp)==false )
  2628. return kOkXsRC;
  2629. cmXScore_t* p = _cmXScoreHandleToPtr(*hp);
  2630. if((rc = _cmXScoreFinalize(p)) != kOkXsRC )
  2631. return rc;
  2632. hp->h = NULL;
  2633. return rc;
  2634. }
  2635. bool cmXScoreIsValid( cmXsH_t h )
  2636. { return h.h != NULL; }
  2637. /* CSV score columns
  2638. kMidiFileIdColScIdx= 0,
  2639. kTypeLabelColScIdx = 3,
  2640. kDSecsColScIdx = 4,
  2641. kSecsColScIdx = 5,
  2642. kD0ColScIdx = 9,
  2643. kD1ColScIdx = 10,
  2644. kPitchColScIdx = 11,
  2645. kBarColScIdx = 13,
  2646. kSkipColScIdx = 14,
  2647. kEvenColScIdx = 15,
  2648. kGraceColScIdx = 16,
  2649. kTempoColScIdx = 17,
  2650. kFracColScIdx = 18,
  2651. kDynColScIdx = 19,
  2652. kSectionColScIdx = 20,
  2653. kRecdPlayColScIdx = 21,
  2654. kRemarkColScIdx = 22
  2655. */
  2656. cmXsRC_t _cmXScoreWriteCsvHdr( cmXScore_t* p )
  2657. {
  2658. const cmChar_t* s[] =
  2659. {
  2660. "id","trk","evt","opcode","dticks","micros","status",
  2661. "meta","ch","d0","d1","arg0","arg1","bar","skip",
  2662. "even","grace","tempo","t_frac","dyn","section","play_recd","remark",NULL
  2663. };
  2664. cmCsvCell_t* lcp = NULL;
  2665. if( cmCsvAppendRow( p->csvH, &lcp, cmCsvInsertSymText(p->csvH,s[0]), 0, 0 ) != kOkCsvRC )
  2666. return cmErrMsg(&p->err,kCsvFailXsRC,"CSV append row failed.");
  2667. unsigned i;
  2668. for(i=1; s[i]!=NULL; ++i)
  2669. {
  2670. if( cmCsvInsertIdentColAfter(p->csvH, lcp, &lcp, s[i], 0 ) != kOkCsvRC )
  2671. return cmErrMsg(&p->err,kCsvFailXsRC,"CSV error inserting CSV title %i.\n",i);
  2672. }
  2673. return kOkXsRC;
  2674. }
  2675. cmXsRC_t _cmXScoreWriteCsvBlankCols( cmXScore_t* p, unsigned cnt, cmCsvCell_t** leftCellPtrPtr )
  2676. {
  2677. unsigned i;
  2678. for(i=0; i<cnt; ++i)
  2679. if( cmCsvInsertIdentColAfter(p->csvH,*leftCellPtrPtr,leftCellPtrPtr,"",0) != kOkCsvRC )
  2680. return cmErrMsg(&p->err,kCsvFailXsRC,"CSV output failed on blank column.");
  2681. return kOkCsvRC;
  2682. }
  2683. const cmChar_t* _cmXScoreTranslateDynamics( cmXScore_t* p, const cmXsNote_t* np, cmChar_t* buf, unsigned bufN )
  2684. {
  2685. if( cmIsFlag(np->flags,kDynXsFl) && np->dynamics != 0 )
  2686. {
  2687. const cmChar_t* dynStr = NULL;
  2688. switch(np->dynamics)
  2689. {
  2690. case 1: dynStr = "pppp"; break;
  2691. case 2: dynStr = "ppp"; break;
  2692. case 3: dynStr = "pp"; break;
  2693. case 4: dynStr = "p"; break;
  2694. case 5: dynStr = "mp"; break;
  2695. case 6: dynStr = "mf"; break;
  2696. case 7: dynStr = "f"; break;
  2697. case 8: dynStr = "ff"; break;
  2698. case 9: dynStr = "fff"; break;
  2699. default:
  2700. cmErrMsg(&p->err,kSyntaxErrorXsRC,"An invalid dynamic value (%i) was encountered.",np->dynamics);
  2701. goto errLabel;
  2702. }
  2703. if( np->dynGroupId == 0 )
  2704. snprintf(buf,bufN,"%s",dynStr);
  2705. else
  2706. snprintf(buf,bufN,"%s %i",dynStr,np->dynGroupId);
  2707. return buf;
  2708. }
  2709. errLabel:
  2710. return "";
  2711. }
  2712. const cmChar_t* cmXsFormatMeasurementCsvField( unsigned flags, unsigned fl, char abbrev, unsigned sectionId, char* buf, unsigned bufN )
  2713. {
  2714. assert( bufN > 1 );
  2715. buf[0] = ' ';
  2716. buf[1] = 0;
  2717. if( cmIsFlag(flags,fl) )
  2718. {
  2719. if( sectionId != 0 )
  2720. snprintf(buf,bufN-1,"%c %i%c",abbrev,sectionId, cmIsFlag(flags,kHeelXsFl)?'*':' ');
  2721. else
  2722. buf[0] = abbrev;
  2723. }
  2724. return buf;
  2725. }
  2726. cmXsRC_t _cmXScoreWriteCsvRow(
  2727. cmXScore_t* p,
  2728. unsigned rowIdx,
  2729. unsigned uid,
  2730. unsigned bar,
  2731. const cmChar_t* sectionStr,
  2732. const cmChar_t* opCodeStr,
  2733. double dsecs,
  2734. double secs,
  2735. unsigned d0,
  2736. unsigned d1,
  2737. unsigned pitch, // set to -1 if the pitch is not valid
  2738. double frac,
  2739. const cmChar_t* dynStr,
  2740. unsigned flags,
  2741. const cmChar_t* evenStr,
  2742. const cmChar_t* tempoStr)
  2743. {
  2744. cmXsRC_t rc = kOkXsRC;
  2745. cmCsvCell_t* lcp = NULL;
  2746. // append an empty row to the CSV object
  2747. if( cmCsvAppendRow( p->csvH, &lcp, cmCsvInsertSymUInt(p->csvH, rowIdx ), 0, 0 ) != kOkCsvRC )
  2748. {
  2749. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV append row failed.");
  2750. goto errLabel;
  2751. }
  2752. /*
  2753. // col 0 : blanks
  2754. if( cmCsvInsertUIntColAfter(p->csvH, lcp, &lcp, rowIdx, 0 ) != kOkCsvRC )
  2755. {
  2756. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV output row index failed.");
  2757. goto errLabel;
  2758. }
  2759. */
  2760. // col 1 : track (always 1)
  2761. if( cmCsvInsertUIntColAfter(p->csvH,lcp,&lcp,1,0) != kOkCsvRC )
  2762. {
  2763. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'd0'.");
  2764. goto errLabel;
  2765. }
  2766. // col 2 : evt (set to event uid, or blank if uid == -1)
  2767. if( uid == -1 )
  2768. {
  2769. if((rc = _cmXScoreWriteCsvBlankCols(p,1,&lcp)) != kOkXsRC )
  2770. goto errLabel;
  2771. }
  2772. else
  2773. if( cmCsvInsertUIntColAfter(p->csvH,lcp,&lcp,uid,0) != kOkCsvRC )
  2774. {
  2775. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'd0'.");
  2776. goto errLabel;
  2777. }
  2778. // col 3 : output the opcode
  2779. if( cmCsvInsertIdentColAfter(p->csvH,lcp,&lcp,opCodeStr,0) != kOkCsvRC )
  2780. {
  2781. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on opcode label.");
  2782. goto errLabel;
  2783. }
  2784. // col 4 : dsecs
  2785. if( cmCsvInsertDoubleColAfter(p->csvH,lcp,&lcp,dsecs,0) != kOkCsvRC )
  2786. {
  2787. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'dsecs'.");
  2788. goto errLabel;
  2789. }
  2790. // col 5 : secs
  2791. if( cmCsvInsertDoubleColAfter(p->csvH,lcp,&lcp,secs,0) != kOkCsvRC )
  2792. {
  2793. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'secs'.");
  2794. goto errLabel;
  2795. }
  2796. // cols 6,7,8 blanks
  2797. if((rc = _cmXScoreWriteCsvBlankCols(p,3,&lcp)) != kOkXsRC )
  2798. goto errLabel;
  2799. // col 9 : d0
  2800. if( cmCsvInsertUIntColAfter(p->csvH,lcp,&lcp,d0,0) != kOkCsvRC )
  2801. {
  2802. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'd0'.");
  2803. goto errLabel;
  2804. }
  2805. // col 10 : d1
  2806. if( cmCsvInsertUIntColAfter(p->csvH,lcp,&lcp,d1,0) != kOkCsvRC )
  2807. {
  2808. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'd1'.");
  2809. goto errLabel;
  2810. }
  2811. // col 11 : pitch
  2812. if( pitch == -1 )
  2813. {
  2814. if((rc = _cmXScoreWriteCsvBlankCols(p,1,&lcp)) != kOkXsRC )
  2815. goto errLabel;
  2816. }
  2817. else
  2818. {
  2819. if( cmCsvInsertIdentColAfter(p->csvH,lcp,&lcp,cmMidiToSciPitch(pitch,NULL,0),0) != kOkCsvRC )
  2820. {
  2821. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'pitch'.");
  2822. goto errLabel;
  2823. }
  2824. }
  2825. // col 12 : blanks
  2826. if((rc = _cmXScoreWriteCsvBlankCols(p,1 + (cmIsFlag(flags,kBarXsFl) ? 0 : 1), &lcp)) != kOkXsRC )
  2827. goto errLabel;
  2828. // col 13 : bar number
  2829. if( cmIsFlag(flags,kBarXsFl) )
  2830. {
  2831. if( cmCsvInsertUIntColAfter(p->csvH,lcp,&lcp,bar,0) != kOkCsvRC )
  2832. {
  2833. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'pitch'.");
  2834. goto errLabel;
  2835. }
  2836. }
  2837. // col 14 : skip (blank for now)
  2838. if((rc = _cmXScoreWriteCsvBlankCols(p,1,&lcp)) != kOkXsRC )
  2839. goto errLabel;
  2840. // col 15: even (all grace notes are 'even' notes
  2841. if( cmCsvInsertQTextColAfter(p->csvH,lcp,&lcp, evenStr, 0) != kOkCsvRC )
  2842. {
  2843. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on eveness flag label.");
  2844. goto errLabel;
  2845. }
  2846. // col 16: grace
  2847. if( cmCsvInsertIdentColAfter(p->csvH,lcp,&lcp,cmIsFlag(flags,kGraceXsFl) ? "g" : "",0) != kOkCsvRC )
  2848. {
  2849. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on eveness flag label.");
  2850. goto errLabel;
  2851. }
  2852. // col 17: tempo
  2853. if( cmCsvInsertQTextColAfter(p->csvH,lcp,&lcp,tempoStr,0) != kOkCsvRC )
  2854. {
  2855. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on eveness flag label.");
  2856. goto errLabel;
  2857. }
  2858. // col 18: frac
  2859. if( frac == 0 )
  2860. {
  2861. if((rc = _cmXScoreWriteCsvBlankCols(p,1,&lcp)) != kOkXsRC )
  2862. goto errLabel;
  2863. }
  2864. else
  2865. {
  2866. if( cmCsvInsertDoubleColAfter(p->csvH,lcp,&lcp,1.0/frac,0) != kOkCsvRC )
  2867. {
  2868. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 't frac'.");
  2869. goto errLabel;
  2870. }
  2871. }
  2872. // col 19: dynamic marking
  2873. if( cmCsvInsertQTextColAfter(p->csvH, lcp, &lcp, dynStr, 0 ) != kOkCsvRC )
  2874. {
  2875. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on 'dynamics'.");
  2876. goto errLabel;
  2877. }
  2878. //if((rc = _cmXScoreWriteCsvBlankCols(p,1,&lcp)) != kOkXsRC )
  2879. // goto errLabel;
  2880. // col 20: section
  2881. if( cmCsvInsertIdentColAfter(p->csvH,lcp,&lcp,sectionStr!=NULL ? sectionStr : "",0) != kOkCsvRC )
  2882. {
  2883. rc = cmErrMsg(&p->err,kCsvFailXsRC,"CSV insert failed on eveness flag label.");
  2884. goto errLabel;
  2885. }
  2886. // col 21, 22 : recd-play, remark (blank for now)
  2887. if((rc = _cmXScoreWriteCsvBlankCols(p,2,&lcp)) != kOkXsRC )
  2888. goto errLabel;
  2889. errLabel:
  2890. return rc;
  2891. }
  2892. cmXsRC_t cmXScoreWriteCsv( cmXsH_t h, int begMeasNumb, const cmChar_t* csvFn )
  2893. {
  2894. cmXsRC_t rc = kOkXsRC;
  2895. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  2896. unsigned rowIdx = 1;
  2897. const cmChar_t* sectionIdStr = NULL;
  2898. double baseSecs = -1;
  2899. if( !cmCsvIsValid(p->csvH) )
  2900. return cmErrMsg(&p->err,kCsvFailXsRC,"The CSV output object is not initialized.");
  2901. if((rc = _cmXScoreWriteCsvHdr( p )) != kOkXsRC )
  2902. goto errLabel;
  2903. cmXsPart_t* pp = p->partL;
  2904. for(; pp!=NULL; pp=pp->link)
  2905. {
  2906. cmXsMeas_t* mp = pp->measL;
  2907. for(; mp!=NULL; mp=mp->link)
  2908. {
  2909. if( mp->number < begMeasNumb)
  2910. continue;
  2911. cmXsNote_t* np = mp->noteL;
  2912. if( baseSecs == -1 )
  2913. baseSecs = np->secs;
  2914. for(; np!=NULL; np=np->slink)
  2915. {
  2916. double thisSecs = np->secs - baseSecs;
  2917. // if this is a section event
  2918. if( cmIsFlag(np->flags,kSectionXsFl) )
  2919. sectionIdStr = np->tvalue;
  2920. // if this is a bar event
  2921. if( cmIsFlag(np->flags,kBarXsFl) )
  2922. {
  2923. _cmXScoreWriteCsvRow(p,rowIdx,-1,mp->number,sectionIdStr,"bar",np->dsecs,thisSecs,0,0,-1,0,"",np->flags,"","");
  2924. sectionIdStr = NULL;
  2925. }
  2926. else
  2927. {
  2928. // if this is a pedal event
  2929. if( cmIsFlag(np->flags,kDampDnXsFl|kDampUpXsFl|kDampUpDnXsFl|kSostDnXsFl|kSostUpXsFl) )
  2930. {
  2931. unsigned d0 = cmIsFlag(np->flags,kSostDnXsFl |kSostUpXsFl) ? 66 : 64; // pedal MIDI ctl id
  2932. unsigned d1 = cmIsFlag(np->flags,kDampDnXsFl|kSostDnXsFl) ? 64 : 0; // pedal-dn: d1>=64 pedal-up:<64
  2933. _cmXScoreWriteCsvRow(p,rowIdx,-1,mp->number,sectionIdStr,"ctl",np->dsecs,thisSecs,d0,d1,-1,0,"",np->flags,"","");
  2934. sectionIdStr = NULL;
  2935. if( cmIsFlag(np->flags,kDampUpDnXsFl) )
  2936. {
  2937. rowIdx += 1;
  2938. double millisecond = 0.0;
  2939. _cmXScoreWriteCsvRow(p,rowIdx,-1,mp->number,sectionIdStr,"ctl",millisecond,thisSecs+millisecond,d0,64,-1,0,"",np->flags,"","");
  2940. }
  2941. }
  2942. else
  2943. {
  2944. // if this is a sounding note event
  2945. if( cmIsFlag(np->flags,kOnsetXsFl) )
  2946. {
  2947. unsigned bufN = 128;
  2948. cmChar_t ebuf[ bufN+1]; ebuf[bufN] = 0;
  2949. cmChar_t dbuf[ bufN+1]; dbuf[bufN] = 0;
  2950. cmChar_t tbuf[ bufN+1]; tbuf[bufN] = 0;
  2951. double frac = np->rvalue + (cmIsFlag(np->flags,kDotXsFl) ? (np->rvalue/2) : 0);
  2952. const cmChar_t* dyn = _cmXScoreTranslateDynamics( p, np, dbuf, bufN );
  2953. unsigned vel = np->vel==0 ? 60 : np->vel;
  2954. //
  2955. _cmXScoreWriteCsvRow(p,rowIdx,np->uid,mp->number,sectionIdStr,"non",np->dsecs,thisSecs,np->pitch,vel,np->pitch,frac,dyn,np->flags,
  2956. cmXsFormatMeasurementCsvField(np->flags, kEvenXsFl, 'e', np->evenGroupId, ebuf, bufN ),
  2957. cmXsFormatMeasurementCsvField(np->flags, kTempoXsFl,'t', np->tempoGroupId, tbuf, bufN ));
  2958. sectionIdStr = NULL;
  2959. }
  2960. }
  2961. }
  2962. rowIdx += 1;
  2963. }
  2964. }
  2965. }
  2966. // Section labels are output on the next bar/pedal/note
  2967. // but what if there is no bar pedal note-on after the section label
  2968. if( sectionIdStr != NULL )
  2969. cmErrMsg(&p->err,kSyntaxErrorXsRC,"The section label '%s' was ignored because it was not followed by a score event.",sectionIdStr);
  2970. if( cmCsvWrite( p->csvH, csvFn ) != kOkCsvRC )
  2971. rc = cmErrMsg(&p->err,kCsvFailXsRC,"The CSV output write failed on file '%s'.",csvFn);
  2972. errLabel:
  2973. return rc;
  2974. }
  2975. bool _cmXsIsCsvValid(cmCtx_t* ctx, cmXsH_t h, const cmChar_t* outFn)
  2976. {
  2977. bool retFl = true;
  2978. cmScH_t scH = cmScNullHandle;
  2979. double srate = 44100.0;
  2980. cmSymTblH_t stH = cmSymTblCreate(cmSymTblNullHandle, 0, ctx );
  2981. if( cmScoreInitialize( ctx, &scH, outFn, srate, NULL, 0, NULL, NULL, stH) != kOkScRC )
  2982. {
  2983. cmErrMsg(&ctx->err,kFileFailXsRC,"The generated CSV file (%s) could not be parsed.",cmStringNullGuard(outFn));
  2984. retFl = false;
  2985. }
  2986. else
  2987. {
  2988. //cmScorePrintSets(scH,&ctx->rpt);
  2989. //cmScorePrint(scH,&ctx->rpt);
  2990. cmScoreFinalize(&scH);
  2991. }
  2992. cmSymTblDestroy(&stH);
  2993. return retFl;
  2994. }
  2995. void _cmXScoreReportTitle( cmRpt_t* rpt )
  2996. {
  2997. cmRptPrintf(rpt," idx voc loc tick durtn rval flags\n");
  2998. cmRptPrintf(rpt," --- --- ----- ------- ----- ---- --- ---------------\n");
  2999. }
  3000. void _cmXScoreReportNote( cmRpt_t* rpt, const cmXsNote_t* note,unsigned index )
  3001. {
  3002. const cmChar_t* B = cmIsFlag(note->flags,kBarXsFl) ? "|" : "-";
  3003. const cmChar_t* R = cmIsFlag(note->flags,kRestXsFl) ? "R" : "-";
  3004. const cmChar_t* G = cmIsFlag(note->flags,kGraceXsFl) ? "G" : "-";
  3005. const cmChar_t* D = cmIsFlag(note->flags,kDotXsFl) ? "." : "-";
  3006. const cmChar_t* C = cmIsFlag(note->flags,kChordXsFl) ? "C" : "-";
  3007. const cmChar_t* P = cmIsFlag(note->flags,kDampDnXsFl) ? "V" : "-";
  3008. const cmChar_t* s = cmIsFlag(note->flags,kSostDnXsFl) ? "{" : "-";
  3009. const cmChar_t* S = cmIsFlag(note->flags,kSectionXsFl) ? "S" : "-";
  3010. const cmChar_t* H = cmIsFlag(note->flags,kHeelXsFl) ? "H" : "-";
  3011. const cmChar_t* T0 = cmIsFlag(note->flags,kTieBegXsFl) ? "T" : "-";
  3012. const cmChar_t* T1 = cmIsFlag(note->flags,kTieEndXsFl) ? "_" : "-";
  3013. const cmChar_t* O = cmIsFlag(note->flags,kOnsetXsFl) ? "*" : "-";
  3014. const cmChar_t* e = cmIsFlag(note->flags,kEvenXsFl) ? "e" : "-";
  3015. const cmChar_t* d = cmIsFlag(note->flags,kDynXsFl) ? "d" : "-";
  3016. const cmChar_t* t = cmIsFlag(note->flags,kTempoXsFl) ? "t" : "-";
  3017. if( cmIsFlag(note->flags,kEvenEndXsFl) ) e="E";
  3018. if( cmIsFlag(note->flags,kDynEndXsFl) ) d="D";
  3019. if( cmIsFlag(note->flags,kTempoEndXsFl) ) t="T";
  3020. P = cmIsFlag(note->flags,kDampUpXsFl) ? "^" : P;
  3021. P = cmIsFlag(note->flags,kDampUpDnXsFl) ? "X" : P;
  3022. s = cmIsFlag(note->flags,kSostUpXsFl) ? "}" : s;
  3023. //const cmChar_t* N = note->pitch==0 ? " " : cmMidiToSciPitch( note->pitch, NULL, 0 );
  3024. cmChar_t N[] = {'\0','\0','\0','\0'};
  3025. cmChar_t acc = note->alter==-1?'b':(note->alter==1?'#':' ');
  3026. snprintf(N,4,"%c%c%1i",note->step,acc,note->octave);
  3027. cmRptPrintf(rpt," %3i %3i %5i %7i %5i %4.1f %3s %s%s%s%s%s%s%s%s%s%s%s%s%s%s%s",
  3028. index,
  3029. note->voice->id,
  3030. note->locIdx,
  3031. note->tick,
  3032. note->tied_dur,
  3033. note->rvalue,
  3034. N,B,R,G,D,C,e,d,t,P,s,S,H,T0,T1,O);
  3035. if( cmIsFlag(note->flags,kSectionXsFl) )
  3036. cmRptPrintf(rpt," %s",cmStringNullGuard(note->tvalue));
  3037. if( cmIsFlag(note->flags,kMetronomeXsFl) )
  3038. cmRptPrintf(rpt," %i bpm",note->duration);
  3039. if( note->evenGroupId != 0 )
  3040. cmRptPrintf(rpt," e=%i",note->evenGroupId);
  3041. if( note->dynGroupId != 0 )
  3042. cmRptPrintf(rpt," d=%i",note->dynGroupId);
  3043. if( note->tempoGroupId != 0 )
  3044. cmRptPrintf(rpt," t=%i",note->tempoGroupId);
  3045. if( note->graceGroupId != 0)
  3046. cmRptPrintf(rpt," g=%i",note->graceGroupId);
  3047. if( note->dynamics != 0)
  3048. cmRptPrintf(rpt," dyn=%i %i",note->dynamics,note->vel);
  3049. if( note->editStr != NULL )
  3050. cmRptPrintf(rpt, " %s", note->editStr );
  3051. /*
  3052. if( cmIsFlag(note->flags,kBegGraceXsFl) )
  3053. cmRptPrintf(rpt," B");
  3054. if( cmIsFlag(note->flags,kEndGraceXsFl) )
  3055. cmRptPrintf(rpt," E");
  3056. if( cmIsFlag(note->flags,kDynBegForkXsFl) )
  3057. cmRptPrintf(rpt," B");
  3058. if( cmIsFlag(note->flags,kDynEndForkXsFl) )
  3059. cmRptPrintf(rpt," E");
  3060. */
  3061. }
  3062. void _cmXScoreReport( cmXScore_t* p, cmRpt_t* rpt, bool sortFl )
  3063. {
  3064. if( rpt == NULL )
  3065. rpt = p->err.rpt;
  3066. cmXsPart_t* pp = p->partL;
  3067. for(; pp!=NULL; pp=pp->link)
  3068. {
  3069. cmRptPrintf(rpt,"Part:%s\n",pp->idStr);
  3070. const cmXsMeas_t* meas = pp->measL;
  3071. for(; meas!=NULL; meas=meas->link)
  3072. {
  3073. unsigned idx = 0;
  3074. cmRptPrintf(rpt," %i : div:%i beat:%i beat-type:%i (%i)\n",meas->number,meas->divisions,meas->beats,meas->beat_type,meas->divisions*meas->beats);
  3075. _cmXScoreReportTitle(rpt);
  3076. if( sortFl )
  3077. {
  3078. const cmXsNote_t* note = meas->noteL;
  3079. unsigned t0 = 0;
  3080. unsigned t1 = 0;
  3081. for(; note!=NULL; note=note->slink,++idx)
  3082. {
  3083. _cmXScoreReportNote(rpt,note,idx);
  3084. t1 = note->slink==NULL ? note->tick : note->slink->tick;
  3085. // check that this note is in tick order
  3086. if( !(t0 <= note->tick && note->tick <= t1) )
  3087. {
  3088. cmRptPrintf(rpt," +");
  3089. }
  3090. t0 = note->tick;
  3091. if( note->slink!=NULL || note->voice->id==0)
  3092. cmRptPrintf(rpt,"\n");
  3093. else
  3094. cmRptPrintf(rpt," %i\n", note->tick + note->duration);
  3095. }
  3096. }
  3097. else
  3098. {
  3099. const cmXsVoice_t* v = meas->voiceL;
  3100. for(; v!=NULL; v=v->link)
  3101. {
  3102. const cmXsNote_t* note = v->noteL;
  3103. cmRptPrintf(rpt," voice:%i\n",v->id);
  3104. for(; note!=NULL; note=note->mlink)
  3105. {
  3106. _cmXScoreReportNote(rpt,note,idx);
  3107. if( note->mlink!=NULL || note->voice->id==0)
  3108. cmRptPrintf(rpt,"\n");
  3109. else
  3110. cmRptPrintf(rpt," %i\n", note->tick + note->duration);
  3111. }
  3112. }
  3113. }
  3114. }
  3115. }
  3116. }
  3117. void cmXScoreReport( cmXsH_t h, cmRpt_t* rpt, bool sortFl )
  3118. {
  3119. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3120. return _cmXScoreReport(p,rpt,sortFl);
  3121. }
  3122. void _cmXScoreGenEditFileWrite( void* arg, const cmChar_t* text )
  3123. {
  3124. if( text != NULL && arg != NULL )
  3125. {
  3126. cmFileH_t* hp = (cmFileH_t*)arg;
  3127. cmFilePrint(*hp,text);
  3128. }
  3129. }
  3130. cmXsRC_t _cmXScoreEditFileRpt( cmCtx_t* ctx, cmXScore_t* p, const cmChar_t* outFn, bool damperRptFl )
  3131. {
  3132. cmXsRC_t rc = kOkXsRC;
  3133. cmErr_t err;
  3134. cmRpt_t rpt;
  3135. cmFileH_t fH = cmFileNullHandle;
  3136. cmErrSetup(&err,&ctx->rpt,"cmXScoreGenEditFile");
  3137. cmRptSetup(&rpt,_cmXScoreGenEditFileWrite,_cmXScoreGenEditFileWrite,&fH);
  3138. if( cmFileOpen(&fH,outFn,kWriteFileFl,&ctx->rpt) != kOkFileRC )
  3139. {
  3140. cmErrMsg(&err,kFileFailXsRC,"Unable to open the output file '%s'.",cmStringNullGuard(outFn));
  3141. goto errLabel;
  3142. }
  3143. _cmXScoreReport(p,&rpt,true);
  3144. errLabel:
  3145. if( cmFileClose(&fH) != kOkFileRC )
  3146. rc = cmErrMsg(&err,kFileFailXsRC,"File close failed on '%s'.",cmStringNullGuard(outFn));
  3147. return rc;
  3148. }
  3149. cmXsRC_t cmXScoreGenEditFile( cmCtx_t* ctx, const cmChar_t* xmlFn, const cmChar_t* outFn, bool damperRptFl )
  3150. {
  3151. cmXsH_t xsH = cmXsNullHandle;
  3152. cmXsRC_t rc = kOkXsRC;
  3153. if((rc = cmXScoreInitialize(ctx,&xsH,xmlFn,NULL,damperRptFl)) != kOkXsRC )
  3154. return rc;
  3155. rc = _cmXScoreEditFileRpt( ctx, _cmXScoreHandleToPtr(xsH), outFn, damperRptFl );
  3156. cmXScoreFinalize(&xsH);
  3157. return rc;
  3158. }
  3159. typedef struct
  3160. {
  3161. unsigned ival;
  3162. double fval;
  3163. unsigned cnt;
  3164. } cmXsHist_t;
  3165. void _cmXsHistUpdateI( cmXsHist_t* hist, unsigned histN, unsigned ival )
  3166. {
  3167. unsigned i;
  3168. for(i=0; i<histN && hist[i].cnt!=0; ++i)
  3169. if( hist[i].ival == ival )
  3170. break;
  3171. if( i==histN )
  3172. return;
  3173. hist[i].ival = ival;
  3174. hist[i].cnt += 1;
  3175. }
  3176. void _cmXsHistUpdateF( cmXsHist_t* hist, unsigned histN, double fval )
  3177. {
  3178. unsigned i;
  3179. for(i=0; i<histN && hist[i].cnt!=0; ++i)
  3180. if( hist[i].fval == fval )
  3181. break;
  3182. if( i==histN )
  3183. return;
  3184. hist[i].fval = fval;
  3185. hist[i].cnt += 1;
  3186. }
  3187. unsigned _cmXsHistValue( cmXsHist_t* hist, unsigned histN )
  3188. {
  3189. unsigned i,n;
  3190. for(i=0,n=0; i<histN && hist[i].cnt>0; ++i)
  3191. n += 1;
  3192. return n;
  3193. }
  3194. // Measure the score complexity for the the time window 'wndSecs' seconds
  3195. // prior to the note n1 and following n0.
  3196. const cmXsNote_t* _cmXsMeasComplexityInWindow( const cmXsNote_t* n0, cmXsNote_t* n1, double wndSecs )
  3197. {
  3198. const cmXsNote_t* n2 = NULL;
  3199. unsigned l_pch_0 = 0;
  3200. unsigned l_pch_value = 0;
  3201. unsigned l_pch_cnt = n1->staff==1 ? 0 : 1;
  3202. unsigned r_pch_0 = n1->staff==1 ? 1 : 0;
  3203. unsigned r_pch_value = 0;
  3204. unsigned r_pch_cnt = 0;
  3205. unsigned i = 0;
  3206. unsigned histN = 100;
  3207. cmXsHist_t velHist[ histN ];
  3208. cmXsHist_t rymHist[ histN ];
  3209. memset(velHist,0,sizeof(velHist));
  3210. memset(rymHist,0,sizeof(rymHist));
  3211. const cmXsNote_t* n = n0;
  3212. while(n!=NULL && n != n1)
  3213. {
  3214. // if this event is less than wndSecs behind 'n1' and is not a sounding note ...
  3215. if( n1->secs - n->secs <= wndSecs && cmIsFlag(n->flags,kOnsetXsFl) )
  3216. {
  3217. _cmXsHistUpdateI( velHist, histN, n->dynamics );
  3218. _cmXsHistUpdateF( rymHist, histN, n->rvalue );
  3219. switch( n->staff )
  3220. {
  3221. case 1: // treble cleff
  3222. if( i > 0 )
  3223. {
  3224. r_pch_value += r_pch_0 > n->pitch ? r_pch_0-n->pitch : n->pitch-r_pch_0;
  3225. r_pch_cnt += 1;
  3226. }
  3227. r_pch_0 = n->pitch;
  3228. break;
  3229. case 2: // bass cleff
  3230. if( i > 0 )
  3231. {
  3232. l_pch_value += l_pch_0 > n->pitch ? l_pch_0-n->pitch : n->pitch-l_pch_0;
  3233. l_pch_cnt += 1;
  3234. }
  3235. l_pch_0 = n->pitch;
  3236. break;
  3237. default:
  3238. { assert(0); }
  3239. }
  3240. // track the first note that is inside the window
  3241. if( i == 0 )
  3242. n2 = n;
  3243. // count the number of notes in the window
  3244. i += 1;
  3245. }
  3246. cmXsMeas_t* m = n->meas;
  3247. // advance th note pointer
  3248. n = n->slink;
  3249. // if we have reached the end of a measure
  3250. if( n == NULL )
  3251. {
  3252. if( m != NULL )
  3253. {
  3254. m = m->link;
  3255. if( m != NULL )
  3256. n = m->noteL;
  3257. }
  3258. }
  3259. }
  3260. // update the cplx record in n1 with the results of this window analysis
  3261. n1->cplx.sum_d_vel = _cmXsHistValue( velHist, histN );
  3262. n1->cplx.sum_d_rym = _cmXsHistValue( rymHist, histN );
  3263. n1->cplx.sum_d_lpch = l_pch_value;
  3264. n1->cplx.sum_n_lpch = l_pch_cnt;
  3265. n1->cplx.sum_d_rpch = r_pch_value;
  3266. n1->cplx.sum_n_rpch = r_pch_cnt;
  3267. return n2;
  3268. }
  3269. // Measure the score complexity and fill in the cmXsComplexity_t record associated
  3270. // with the cmXsNote_t record of each sounding note.
  3271. cmXsRC_t _cmXsMeasComplexity( cmXsH_t h, double wndSecs )
  3272. {
  3273. cmXsRC_t rc = kOkXsRC;
  3274. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3275. cmXsPart_t* pp = p->partL;
  3276. memset(&p->cplx_max,0,sizeof(p->cplx_max));
  3277. const cmXsNote_t* n0 = NULL;
  3278. // for each part
  3279. for(; pp!=NULL; pp=pp->link)
  3280. {
  3281. cmXsMeas_t* mp = pp->measL;
  3282. // for each measure
  3283. for(; mp!=NULL; mp=mp->link)
  3284. {
  3285. cmXsNote_t* n1 = mp->noteL;
  3286. // for each note in this measure
  3287. for(; n1!=NULL; n1=n1->slink)
  3288. if( cmIsFlag(n1->flags,kOnsetXsFl) )
  3289. {
  3290. if( n0 == NULL )
  3291. n0 = n1;
  3292. else
  3293. if((n0 = _cmXsMeasComplexityInWindow(n0,n1,wndSecs)) == NULL )
  3294. n0 = n1;
  3295. // track the max value for all complexity values to allow
  3296. // eventual normalization of the complexity values
  3297. p->cplx_max.sum_d_vel = cmMax(p->cplx_max.sum_d_vel, n1->cplx.sum_d_vel);
  3298. p->cplx_max.sum_d_rym = cmMax(p->cplx_max.sum_d_rym, n1->cplx.sum_d_rym);
  3299. p->cplx_max.sum_d_lpch = cmMax(p->cplx_max.sum_d_lpch,n1->cplx.sum_d_lpch);
  3300. p->cplx_max.sum_n_lpch = cmMax(p->cplx_max.sum_n_lpch,n1->cplx.sum_n_lpch);
  3301. p->cplx_max.sum_d_rpch = cmMax(p->cplx_max.sum_d_rpch,n1->cplx.sum_d_rpch);
  3302. p->cplx_max.sum_n_rpch = cmMax(p->cplx_max.sum_n_rpch,n1->cplx.sum_n_rpch);
  3303. }
  3304. }
  3305. }
  3306. return rc;
  3307. }
  3308. cmXsRC_t _cmXsWriteMidiFile( cmCtx_t* ctx, cmXsH_t h, int beginMeasNumb, int beginBPM, const cmChar_t* dir, const cmChar_t* fn )
  3309. {
  3310. cmXsRC_t rc = kOkXsRC;
  3311. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3312. if( p->partL==NULL || p->partL->measL == NULL )
  3313. return rc;
  3314. cmMidiFileH_t mfH = cmMidiFileNullHandle;
  3315. unsigned trkN = 2;
  3316. unsigned ticksPerQN = p->partL->measL->divisions;
  3317. const cmChar_t* outFn = cmFsMakeFn(dir,fn,"mid",NULL);
  3318. unsigned baseTick = -1;
  3319. unsigned bpm = beginBPM==0 ? 60 : beginBPM;
  3320. //if( cmMidiFileCreate( ctx, &mfH, trkN, ticksPerQN ) != kOkMfRC )
  3321. // return cmErrMsg(&p->err,kMidiFailXsRC,"Unable to create the MIDI file object.");
  3322. cmXsPart_t* pp = p->partL;
  3323. // for each part
  3324. for(; pp!=NULL; pp=pp->link)
  3325. {
  3326. cmXsMeas_t* mp = pp->measL;
  3327. // for each measure
  3328. for(; mp!=NULL; mp=mp->link)
  3329. {
  3330. // skip all measures until we reach the first measure to output
  3331. if(mp->number < beginMeasNumb)
  3332. continue;
  3333. // if the MIDI file has not yet been created
  3334. if( !cmMidiFileIsValid(mfH) )
  3335. {
  3336. ticksPerQN = mp->divisions;
  3337. // create the MIDI file
  3338. if( cmMidiFileCreate( ctx, &mfH, trkN, ticksPerQN ) != kOkMfRC )
  3339. {
  3340. rc = cmErrMsg(&p->err,kMidiFailXsRC,"Unable to create the MIDI file object.");
  3341. goto errLabel;
  3342. }
  3343. // set the starting tempo
  3344. cmMidFileInsertTrackTempoMsg(mfH, 0, 0, bpm );
  3345. }
  3346. cmXsNote_t* np = mp->noteL;
  3347. if( baseTick == -1 )
  3348. baseTick = np->tick;
  3349. if( mp->divisions != ticksPerQN )
  3350. cmErrWarnMsg(&p->err,kMidiFailXsRC,"The 'tick per quarter note' (divisions) field in measure %i does not match the value in the first measure (%i).",mp->divisions,ticksPerQN);
  3351. unsigned ni = 0;
  3352. // for each note in this measure
  3353. for(; np!=NULL; np=np->slink,++ni)
  3354. {
  3355. unsigned thisTick = np->tick - baseTick;
  3356. switch( np->flags & (kOnsetXsFl|kMetronomeXsFl|kDampDnXsFl|kDampUpDnXsFl|kSostDnXsFl) )
  3357. {
  3358. case kOnsetXsFl:
  3359. if( cmMidiFileIsValid(mfH) )
  3360. {
  3361. if( np->tied_dur <= 0 )
  3362. cmErrWarnMsg(&p->err,kOkXsRC,"A zero length note was encountered bar:%i tick:%i (%i) %s",np->meas->number,np->tick,thisTick,cmMidiToSciPitch(np->pitch,NULL,0));
  3363. /*
  3364. if( mp->number == 20 )
  3365. {
  3366. _cmXScoreReportNote(ctx->err.rpt,np,ni);
  3367. cmRptPrintf(ctx->err.rpt,"\n");
  3368. }
  3369. */
  3370. if( np->vel == 0 )
  3371. cmErrWarnMsg(&p->err,kOkXsRC,"A sounding note with zero velocity was encountered at bar:%i tick:%i pitch:%s.",np->meas->number,np->tick,cmMidiToSciPitch(np->pitch,NULL,0));
  3372. if( cmMidiFileInsertTrackChMsg(mfH, 1, thisTick, kNoteOnMdId, np->pitch, np->vel ) != kOkMfRC
  3373. ||cmMidiFileInsertTrackChMsg(mfH, 1, thisTick + np->tied_dur, kNoteOffMdId, np->pitch, 0 ) != kOkMfRC )
  3374. {
  3375. rc = kMidiFailXsRC;
  3376. }
  3377. }
  3378. break;
  3379. case kDampDnXsFl:
  3380. case kDampUpDnXsFl:
  3381. case kSostDnXsFl:
  3382. if( cmMidiFileIsValid(mfH) )
  3383. {
  3384. if( np->duration <= 0 )
  3385. cmErrWarnMsg(&p->err,kOkXsRC,"A zero length pedal event was encountered bar:%i tick:%i (%i)",np->meas->number,np->tick,thisTick);
  3386. cmMidiByte_t d0 = cmIsFlag(np->flags,kSostDnXsFl) ? kSostenutoCtlMdId : kSustainCtlMdId;
  3387. if( (cmMidiFileInsertTrackChMsg(mfH, 1, thisTick, kCtlMdId, d0, 127 ) != kOkMfRC )
  3388. ||(cmMidiFileInsertTrackChMsg(mfH, 1, thisTick + np->duration, kCtlMdId, d0, 0 ) != kOkMfRC ) )
  3389. {
  3390. rc = kMidiFailXsRC;
  3391. }
  3392. }
  3393. break;
  3394. case kMetronomeXsFl:
  3395. bpm = np->duration;
  3396. if( cmMidiFileIsValid(mfH) )
  3397. if( cmMidFileInsertTrackTempoMsg(mfH, 0, thisTick, bpm ) != kOkMfRC )
  3398. rc = kMidiFailXsRC;
  3399. break;
  3400. case 0:
  3401. break;
  3402. default:
  3403. { assert(0); }
  3404. }
  3405. if( rc != kOkXsRC )
  3406. {
  3407. rc = cmErrMsg(&p->err,rc,"MIDI message insert failed on '%s'.",cmStringNullGuard(outFn));
  3408. goto errLabel;
  3409. }
  3410. }
  3411. }
  3412. }
  3413. if( cmMidiFileIsValid(mfH) )
  3414. if( cmMidiFileWrite(mfH,outFn) != kOkMfRC )
  3415. {
  3416. rc = cmErrMsg(&p->err,kMidiFailXsRC,"MIDI file write failed on '%s'.",cmStringNullGuard(outFn));
  3417. goto errLabel;
  3418. }
  3419. errLabel:
  3420. cmFsFreeFn(outFn);
  3421. if( cmMidiFileClose(&mfH) != kOkMfRC )
  3422. {
  3423. rc = cmErrMsg(&p->err,kMidiFailXsRC,"Unable to create the MIDI file object.");
  3424. goto errLabel;
  3425. }
  3426. return rc;
  3427. }
  3428. bool _cmXsIsMidiFileValid( cmCtx_t* ctx, cmXsH_t h, const cmChar_t* dir, const cmChar_t* fn )
  3429. {
  3430. const cmChar_t* midiFn = cmFsMakeFn(dir,fn,"mid",NULL);
  3431. cmMidiFileH_t mfH = cmMidiFileNullHandle;
  3432. if( cmMidiFileOpen( ctx, &mfH, midiFn ) == kOkMfRC )
  3433. {
  3434. cmMidiFileClose(&mfH);
  3435. return true;
  3436. }
  3437. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3438. cmErrMsg(&p->err,kMidiFailXsRC,"The generated MIDI file '%s' is not valid.", cmStringNullGuard(midiFn));
  3439. return false;
  3440. }
  3441. typedef struct cmXsSvgEvt_str
  3442. {
  3443. unsigned flags; // k???XsFl
  3444. unsigned tick; // start tick
  3445. unsigned durTicks; // dur-ticks
  3446. unsigned voice; // score voice number
  3447. unsigned d0; // MIDI d0 (barNumb)
  3448. unsigned d1; // MIDI d1
  3449. cmXsComplexity_t cplx;
  3450. struct cmXsSvgEvt_str* link;
  3451. } cmXsSvgEvt_t;
  3452. typedef struct cmXsMidiFile_str
  3453. {
  3454. cmXsSvgEvt_t* elist;
  3455. cmXsSvgEvt_t* eol;
  3456. unsigned pitch_min;
  3457. unsigned pitch_max;
  3458. } cmXsMidiFile_t;
  3459. void _cmXsWriteMidiSvgLegend( cmSvgH_t svgH, unsigned index, const cmChar_t* label, const cmChar_t* classStr )
  3460. {
  3461. double x = 100;
  3462. double y = 120*10 - 20*index;
  3463. cmSvgWriterText( svgH, x, y, label, "legend" );
  3464. x += 75;
  3465. cmSvgWriterLine( svgH, x, y, x+125, y, classStr );
  3466. }
  3467. void _cmXsWriteMidiSvgLinePoint( cmSvgH_t svgH, double x0, double y0, double x1, double y1, double y_max, const cmChar_t* classStr, const cmChar_t* label )
  3468. {
  3469. int bn = 255;
  3470. char b[bn+1];
  3471. double y_scale = 10;
  3472. double y_label = y1;
  3473. b[0] = 0;
  3474. y0 = (y0/y_max) * 127.0 * y_scale;
  3475. y1 = (y1/y_max) * 127.0 * y_scale;
  3476. cmSvgWriterLine(svgH, x0, y0, x1, y1, classStr );
  3477. if( y0 != y1 )
  3478. snprintf(b,bn,"%5.0f %s",y_label,label==NULL?"":label);
  3479. else
  3480. {
  3481. if( label != NULL )
  3482. snprintf(b,bn,"%s",label);
  3483. }
  3484. if( strlen(b) )
  3485. cmSvgWriterText(svgH, x1, y1, b, "pt_text");
  3486. }
  3487. cmXsRC_t _cmXsWriteMidiSvg( cmCtx_t* ctx, cmXScore_t* p, cmXsMidiFile_t* mf, const cmChar_t* svgFn, bool standAloneFl, bool panZoomFl )
  3488. {
  3489. cmXsRC_t rc = kOkXsRC;
  3490. cmSvgH_t svgH = cmSvgNullHandle;
  3491. cmXsSvgEvt_t* e = mf->elist;
  3492. unsigned noteHeight = 10;
  3493. //cmChar_t* fn0 = cmMemAllocStr( fn );
  3494. //const cmChar_t* svgFn = cmFsMakeFn(dir,fn0 = cmTextAppendSS(fn0,"_midi_svg"),"html",NULL);
  3495. const cmChar_t* cssFn = cmFsMakeFn(NULL,"score_midi_svg","css",NULL);
  3496. cmChar_t* t0 = NULL; // temporary dynamic string
  3497. unsigned i = 0;
  3498. const cmXsSvgEvt_t* e0 = NULL;
  3499. //cmMemFree(fn0);
  3500. // create the SVG writer
  3501. if( cmSvgWriterAlloc(ctx,&svgH) != kOkSvgRC )
  3502. {
  3503. rc = cmErrMsg(&p->err,kSvgFailXsRC,"Unable to create the MIDI SVG output file '%s'.",cmStringNullGuard(svgFn));
  3504. goto errLabel;
  3505. }
  3506. _cmXsWriteMidiSvgLegend( svgH, 0, "Velocity", "cplx_vel" );
  3507. _cmXsWriteMidiSvgLegend( svgH, 1, "Duration", "cplx_rym" );
  3508. _cmXsWriteMidiSvgLegend( svgH, 2, "Left Pitch", "cplx_lpch" );
  3509. _cmXsWriteMidiSvgLegend( svgH, 3, "Right Pitch", "cplx_rpch" );
  3510. _cmXsWriteMidiSvgLegend( svgH, 4, "Density", "cplx_density" );
  3511. // for each MIDI file element
  3512. for(; e!=NULL && rc==kOkXsRC; e=e->link)
  3513. {
  3514. switch( e->flags & (kOnsetXsFl|kBarXsFl|kDampDnXsFl|kDampUpDnXsFl|kSostDnXsFl))
  3515. {
  3516. // if this is a note
  3517. case kOnsetXsFl:
  3518. {
  3519. //const cmChar_t* classLabel = "note";
  3520. t0 = cmTsPrintfP(t0,"note_%i%s",e->voice, cmIsFlag(e->flags,kGraceXsFl) ? "_g":"");
  3521. //if( cmIsFlag(e->flags,kGraceXsFl) )
  3522. // classLabel = "grace";
  3523. if( cmSvgWriterRect(svgH, e->tick, e->d0 * noteHeight, e->durTicks, noteHeight-1, t0 ) != kOkSvgRC )
  3524. rc = kSvgFailXsRC;
  3525. else
  3526. {
  3527. t0 = cmTsPrintfP(t0,"%s",cmMidiToSciPitch( e->d0, NULL, 0));
  3528. if( cmSvgWriterText(svgH, e->tick + e->durTicks/2, e->d0 * noteHeight + noteHeight/2, t0, "pitch") != kOkSvgRC )
  3529. rc = kSvgFailXsRC;
  3530. else
  3531. {
  3532. if( e0 != NULL )
  3533. {
  3534. bool fl = (i % 10) == 0;
  3535. _cmXsWriteMidiSvgLinePoint(svgH, e0->tick, e0->cplx.sum_d_vel, e->tick, e->cplx.sum_d_vel, p->cplx_max.sum_d_vel, "cplx_vel",fl?"V":NULL);
  3536. _cmXsWriteMidiSvgLinePoint(svgH, e0->tick, e0->cplx.sum_d_rym, e->tick, e->cplx.sum_d_rym, p->cplx_max.sum_d_rym, "cplx_rym",fl?"D":NULL);
  3537. _cmXsWriteMidiSvgLinePoint(svgH, e0->tick, e0->cplx.sum_d_lpch, e->tick, e->cplx.sum_d_lpch, p->cplx_max.sum_d_lpch, "cplx_lpch",fl?"L":NULL);
  3538. _cmXsWriteMidiSvgLinePoint(svgH, e0->tick, e0->cplx.sum_d_rpch, e->tick, e->cplx.sum_d_rpch, p->cplx_max.sum_d_rpch, "cplx_rpch",fl?"R":NULL);
  3539. _cmXsWriteMidiSvgLinePoint(svgH, e0->tick, e0->cplx.sum_n_lpch + e0->cplx.sum_n_rpch, e->tick, e->cplx.sum_n_lpch + e->cplx.sum_n_rpch, p->cplx_max.sum_n_lpch + p->cplx_max.sum_n_rpch, "cplx_density",fl?"N":NULL);
  3540. }
  3541. e0 = e;
  3542. }
  3543. }
  3544. i+=1;
  3545. }
  3546. break;
  3547. // if this is a bar
  3548. case kBarXsFl:
  3549. {
  3550. if( cmSvgWriterLine(svgH, e->tick, 0, e->tick, 127*noteHeight, "bar") != kOkSvgRC )
  3551. rc = kSvgFailXsRC;
  3552. else
  3553. {
  3554. if( cmSvgWriterText(svgH, e->tick, 10, t0 = cmTsPrintfP(t0,"%i",e->d0), "text" ) != kOkSvgRC )
  3555. rc = kSvgFailXsRC;
  3556. }
  3557. }
  3558. break;
  3559. // if this is a pedal event
  3560. case kDampDnXsFl:
  3561. case kDampUpDnXsFl:
  3562. case kSostDnXsFl:
  3563. {
  3564. const cmChar_t* classLabel = cmIsFlag(e->flags,kSostDnXsFl) ? "sost" : "damp";
  3565. unsigned y = (128 + cmIsFlag(e->flags,kSostDnXsFl)?1:0) * noteHeight;
  3566. cmSvgWriterRect(svgH, e->tick, y, e->durTicks, noteHeight-1, classLabel);
  3567. }
  3568. break;
  3569. }
  3570. }
  3571. if( rc != kOkXsRC )
  3572. cmErrMsg(&p->err,kSvgFailXsRC,"SVG element insert failed.");
  3573. if( rc == kOkXsRC )
  3574. if( cmSvgWriterWrite(svgH,cssFn,svgFn,standAloneFl, panZoomFl) != kOkSvgRC )
  3575. rc = cmErrMsg(&p->err,kSvgFailXsRC,"SVG file write to '%s' failed.",cmStringNullGuard(svgFn));
  3576. errLabel:
  3577. cmSvgWriterFree(&svgH);
  3578. //cmFsFreeFn(svgFn);
  3579. cmFsFreeFn(cssFn);
  3580. cmMemFree(t0);
  3581. return rc;
  3582. }
  3583. void _cmXsPushSvgEvent( cmXScore_t* p, cmXsMidiFile_t* mf, unsigned flags, unsigned tick, unsigned durTick, unsigned voice, unsigned d0, unsigned d1, const cmXsComplexity_t* cplx )
  3584. {
  3585. cmXsSvgEvt_t* e = cmLhAllocZ(p->lhH,cmXsSvgEvt_t,1);
  3586. e->flags = flags;
  3587. e->tick = tick;
  3588. e->durTicks = durTick;
  3589. e->voice = voice;
  3590. e->d0 = d0; // note=pitch bar=number pedal=ctl# metronome=BPM
  3591. e->d1 = d1;
  3592. if( cplx != NULL )
  3593. e->cplx = *cplx;
  3594. if( mf->eol != NULL )
  3595. mf->eol->link = e;
  3596. else
  3597. mf->elist = e;
  3598. // track the min/max pitch
  3599. if( cmIsFlag(flags,kOnsetXsFl) )
  3600. {
  3601. mf->pitch_min = mf->eol==NULL ? d0 : cmMin(mf->pitch_min,d0);
  3602. mf->pitch_max = mf->eol==NULL ? d0 : cmMin(mf->pitch_max,d0);
  3603. }
  3604. mf->eol = e;
  3605. }
  3606. cmXsRC_t _cmXScoreGenSvg( cmCtx_t* ctx, cmXsH_t h, int beginMeasNumb, const cmChar_t* svgFn, bool standAloneFl, bool panZoomFl )
  3607. {
  3608. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3609. cmXsPart_t* pp = p->partL;
  3610. cmXsMidiFile_t mf;
  3611. memset(&mf,0,sizeof(mf));
  3612. for(; pp!=NULL; pp=pp->link)
  3613. {
  3614. const cmXsMeas_t* meas = pp->measL;
  3615. for(; meas!=NULL; meas=meas->link)
  3616. {
  3617. if( meas->number < beginMeasNumb )
  3618. continue;
  3619. const cmXsNote_t* note = meas->noteL;
  3620. for(; note!=NULL; note=note->slink)
  3621. {
  3622. // if this is a metronome marker
  3623. if( cmIsFlag(note->flags,kMetronomeXsFl) )
  3624. {
  3625. // set BPM as d0
  3626. _cmXsPushSvgEvent(p,&mf,note->flags,note->tick,0,0,note->duration,0,NULL);
  3627. continue;
  3628. }
  3629. // if this is a sounding note
  3630. if( cmIsFlag(note->flags,kOnsetXsFl) )
  3631. {
  3632. unsigned d0 = cmSciPitchToMidiPitch( note->step, note->alter, note->octave );
  3633. unsigned durTick = note->tied_dur;
  3634. if( note->tied != NULL )
  3635. {
  3636. cmXsNote_t* tn = note->tied;
  3637. for(; tn!=NULL; tn=tn->tied)
  3638. durTick += tn->tied_dur;
  3639. }
  3640. _cmXsPushSvgEvent(p,&mf,note->flags,note->tick,durTick,note->voice->id,d0,note->vel,&note->cplx);
  3641. continue;
  3642. }
  3643. // if this is a bar event
  3644. if( cmIsFlag(note->flags,kBarXsFl) )
  3645. {
  3646. _cmXsPushSvgEvent(p,&mf,note->flags,note->tick,0,0,note->meas->number,0,NULL);
  3647. continue;
  3648. }
  3649. // if this is a pedal event
  3650. if( cmIsFlag(note->flags,kDampDnXsFl|kDampUpDnXsFl|kSostDnXsFl) )
  3651. {
  3652. unsigned d0 = cmIsFlag(note->flags,kSostDnXsFl) ? kSostenutoCtlMdId : kSustainCtlMdId;
  3653. _cmXsPushSvgEvent(p,&mf,note->flags,note->tick,note->duration,0,d0,127,NULL);
  3654. continue;
  3655. }
  3656. }
  3657. }
  3658. }
  3659. return _cmXsWriteMidiSvg( ctx, p, &mf, svgFn, standAloneFl, panZoomFl );
  3660. }
  3661. cmXsRC_t cmXScoreTest(
  3662. cmCtx_t* ctx,
  3663. const cmChar_t* xmlFn,
  3664. const cmChar_t* editFn,
  3665. const cmChar_t* csvOutFn,
  3666. const cmChar_t* midiOutFn,
  3667. const cmChar_t* svgOutFn,
  3668. bool reportFl,
  3669. int beginMeasNumb,
  3670. int beginBPM,
  3671. bool standAloneFl,
  3672. bool panZoomFl,
  3673. bool damperRptFl)
  3674. {
  3675. cmXsRC_t rc;
  3676. cmXsH_t h = cmXsNullHandle;
  3677. if( editFn!=NULL && !cmFsIsFile(editFn) )
  3678. {
  3679. cmRptPrintf(&ctx->rpt,"The edit file %s does not exist. A new edit file will be created.\n",editFn);
  3680. cmXScoreGenEditFile(ctx,xmlFn,editFn,damperRptFl);
  3681. editFn = NULL;
  3682. }
  3683. // Parse the XML file and apply the changes in editFn.
  3684. if((rc = cmXScoreInitialize( ctx, &h, xmlFn,editFn, damperRptFl )) != kOkXsRC )
  3685. return cmErrMsg(&ctx->err,rc,"XScore alloc failed.");
  3686. if( csvOutFn != NULL )
  3687. {
  3688. cmXScoreWriteCsv(h,beginMeasNumb,csvOutFn);
  3689. _cmXsIsCsvValid(ctx,h,csvOutFn);
  3690. }
  3691. // measure the score complexity
  3692. double wndSecs = 1.0;
  3693. _cmXsMeasComplexity(h,wndSecs);
  3694. if( midiOutFn != NULL )
  3695. {
  3696. cmFileSysPathPart_t* pp = cmFsPathParts(midiOutFn);
  3697. _cmXsWriteMidiFile(ctx, h, beginMeasNumb, beginBPM, pp->dirStr, pp->fnStr );
  3698. _cmXsIsMidiFileValid(ctx, h, pp->dirStr, pp->fnStr );
  3699. cmFsFreePathParts(pp);
  3700. }
  3701. if( svgOutFn != NULL )
  3702. _cmXScoreGenSvg( ctx, h, beginMeasNumb, svgOutFn, standAloneFl, panZoomFl );
  3703. if(reportFl)
  3704. cmXScoreReport(h,&ctx->rpt,true);
  3705. return cmXScoreFinalize(&h);
  3706. }
  3707. cmXsRC_t cmXScoreMergeEditFiles( cmCtx_t* ctx, const cmChar_t* xmlFn, const cmChar_t* refEditFn, unsigned refBegMeasNumb, const cmChar_t* editFn, unsigned keyMeasNumb, const cmChar_t* outFn )
  3708. {
  3709. cmXsH_t h = cmXsNullHandle;
  3710. cmXsRC_t rc;
  3711. bool damperRptFl = false;
  3712. if((rc = cmXScoreInitialize(ctx, &h, xmlFn, refEditFn, damperRptFl )) == kOkXsRC )
  3713. {
  3714. cmXScore_t* p = _cmXScoreHandleToPtr(h);
  3715. if((rc = _cmXsMergeEditFiles(p, refBegMeasNumb, editFn, keyMeasNumb, outFn )) == kOkXsRC )
  3716. rc = _cmXScoreEditFileRpt( ctx, p, outFn, damperRptFl );
  3717. cmXScoreFinalize(&h);
  3718. }
  3719. return rc;
  3720. }