VoiceEntry.ts 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373
  1. import { Fraction } from "../../Common/DataObjects/Fraction";
  2. import {Voice} from "./Voice";
  3. import {SourceStaffEntry} from "./SourceStaffEntry";
  4. import {Note} from "./Note";
  5. import {LyricsEntry} from "./Lyrics/LyricsEntry";
  6. import {TechnicalInstruction} from "./Instructions/TechnicalInstruction";
  7. import {OrnamentContainer} from "./OrnamentContainer";
  8. import { Dictionary } from "typescript-collections";
  9. import {Arpeggio} from "./Arpeggio";
  10. import { PlaybackEntry } from "../Playback/PlaybackEntry";
  11. import { Articulation } from "./Articulation";
  12. import { PlaybackNote } from "../Playback/PlaybackNote";
  13. /**
  14. * A [[VoiceEntry]] contains the notes in a voice at a timestamp.
  15. */
  16. export class VoiceEntry {
  17. /**
  18. *
  19. * @param timestamp The relative timestamp within the source measure.
  20. * @param parentVoice
  21. * @param parentSourceStaffEntry
  22. * @param isGrace States whether the VoiceEntry has (only) grace notes.
  23. * @param graceNoteSlash States whether the grace note(s) have a slash (Acciaccatura, played before the beat)
  24. */
  25. constructor(timestamp: Fraction, parentVoice: Voice, parentSourceStaffEntry: SourceStaffEntry, addToStaffEntry: boolean = true,
  26. isGrace: boolean = false, graceNoteSlash: boolean = false, graceSlur: boolean = false) {
  27. this.timestamp = timestamp;
  28. this.parentVoice = parentVoice;
  29. this.parentSourceStaffEntry = parentSourceStaffEntry;
  30. this.isGrace = isGrace;
  31. this.graceAfterMainNote = false;
  32. this.graceNoteSlash = graceNoteSlash;
  33. this.graceSlur = graceSlur;
  34. if (!isGrace) {
  35. parentVoice.VoiceEntries.push(this);
  36. }
  37. // add currentVoiceEntry to staff entry:
  38. if (addToStaffEntry && parentSourceStaffEntry !== undefined) {
  39. const list: VoiceEntry[] = parentSourceStaffEntry.VoiceEntries;
  40. if (list.indexOf(this) === -1) {
  41. list.push(this);
  42. }
  43. }
  44. // ToDo: at this moment there are no notes added to the voice entry
  45. this.mainPlaybackEntry = new PlaybackEntry(this);
  46. this.PlaybackEntries.push(this.mainPlaybackEntry);
  47. }
  48. private parentVoice: Voice;
  49. private parentSourceStaffEntry: SourceStaffEntry;
  50. private timestamp: Fraction;
  51. private notes: Note[] = [];
  52. private graceVoiceEntriesBefore: VoiceEntry[] = [];
  53. private graceVoiceEntriesAfter: VoiceEntry[] = [];
  54. private isGrace: boolean;
  55. /** States whether the grace notes come after a main note (at end of measure). */
  56. private graceAfterMainNote: boolean;
  57. private graceNoteSlash: boolean;
  58. private graceSlur: boolean; // TODO grace slur system could be refined to be non-binary
  59. private articulations: Articulation[] = [];
  60. private playbackEntries: PlaybackEntry[] = [];
  61. private fermata: Articulation;
  62. private technicalInstructions: TechnicalInstruction[] = [];
  63. private lyricsEntries: Dictionary<string, LyricsEntry> = new Dictionary<string, LyricsEntry>();
  64. /** The Arpeggio consisting of this VoiceEntry's notes. Undefined if no arpeggio exists. */
  65. private arpeggio: Arpeggio;
  66. private ornamentContainer: OrnamentContainer;
  67. private wantedStemDirection: StemDirectionType = StemDirectionType.Undefined;
  68. /** Stem direction specified in the xml stem element. */
  69. private stemDirectionXml: StemDirectionType = StemDirectionType.Undefined;
  70. private stemDirection: StemDirectionType = StemDirectionType.Undefined;
  71. /** Color of the stem given in XML. RGB Hexadecimal, like #00FF00. */
  72. private stemColorXml: string;
  73. /** Color of the stem currently set. RGB Hexadecimal, like #00FF00. */
  74. private stemColor: string;
  75. private mainPlaybackEntry: PlaybackEntry;
  76. private volumeModifier: Articulation;
  77. private durationModifier: Articulation;
  78. public get ParentSourceStaffEntry(): SourceStaffEntry {
  79. return this.parentSourceStaffEntry;
  80. }
  81. public get ParentVoice(): Voice {
  82. return this.parentVoice;
  83. }
  84. public get Timestamp(): Fraction {
  85. return this.timestamp;
  86. }
  87. public set Timestamp(value: Fraction) {
  88. this.timestamp = value;
  89. }
  90. public get Notes(): Note[] {
  91. return this.notes;
  92. }
  93. public addNote(note: Note): void {
  94. this.notes.push(note);
  95. // only add playback notes when these are no rests and are not tied notes (besides the first note of a tie)
  96. if (!note.isRest() &&
  97. (note.NoteTie === undefined || note.NoteTie.StartNote === note)) {
  98. this.MainPlaybackEntry.Notes.push(new PlaybackNote(this.MainPlaybackEntry, note));
  99. }
  100. }
  101. public get GraceVoiceEntriesBefore(): VoiceEntry[] {
  102. return this.graceVoiceEntriesBefore;
  103. }
  104. public set GraceVoiceEntriesBefore(value: VoiceEntry[] ) {
  105. this.graceVoiceEntriesBefore = value;
  106. for (const ve of this.graceVoiceEntriesBefore) {
  107. ve.parentSourceStaffEntry = this.ParentSourceStaffEntry;
  108. }
  109. }
  110. public get GraceVoiceEntriesAfter(): VoiceEntry[] {
  111. return this.graceVoiceEntriesAfter;
  112. }
  113. public set GraceVoiceEntriesAfter(value: VoiceEntry[] ) {
  114. this.graceVoiceEntriesAfter = value;
  115. for (const ve of this.graceVoiceEntriesAfter) {
  116. ve.parentSourceStaffEntry = this.ParentSourceStaffEntry;
  117. }
  118. }
  119. public get IsGrace(): boolean {
  120. return this.isGrace;
  121. }
  122. public set IsGrace(value: boolean) {
  123. this.isGrace = value;
  124. }
  125. public get GraceAfterMainNote(): boolean {
  126. return this.graceAfterMainNote;
  127. }
  128. public set GraceAfterMainNote(value: boolean) {
  129. this.graceAfterMainNote = value;
  130. }
  131. public get GraceNoteSlash(): boolean {
  132. return this.graceNoteSlash;
  133. }
  134. public set GraceNoteSlash(value: boolean) {
  135. this.graceNoteSlash = value;
  136. }
  137. public get GraceSlur(): boolean {
  138. return this.graceSlur;
  139. }
  140. public set GraceSlur(value: boolean) {
  141. this.graceSlur = value;
  142. }
  143. public get Articulations(): Articulation[] {
  144. return this.articulations;
  145. }
  146. /** Stores all playback entries (e.g. extra grace and ornament entries).
  147. * Also holds the main playback entry.
  148. * The entries are sorted in ascending timestamp.
  149. */
  150. public get PlaybackEntries(): PlaybackEntry[] {
  151. return this.playbackEntries;
  152. }
  153. public get Fermata(): Articulation {
  154. return this.fermata;
  155. }
  156. public get MainPlaybackEntry(): PlaybackEntry {
  157. return this.mainPlaybackEntry;
  158. }
  159. public set MainPlaybackEntry(value: PlaybackEntry) {
  160. this.mainPlaybackEntry = value;
  161. }
  162. public removeMainPlaybackEntry(): void {
  163. if (this.mainPlaybackEntry !== undefined) {
  164. this.removePlaybackEntry(this.mainPlaybackEntry);
  165. }
  166. }
  167. public removePlaybackEntry(value: PlaybackEntry): void {
  168. if (this.mainPlaybackEntry === value) {
  169. this.mainPlaybackEntry = undefined;
  170. }
  171. const index: number = this.playbackEntries.indexOf(value);
  172. if (index > -1) {
  173. this.playbackEntries.splice(index, 1);
  174. }
  175. }
  176. public set Articulations(value: Articulation[]) {
  177. this.articulations = value;
  178. }
  179. public get TechnicalInstructions(): TechnicalInstruction[] {
  180. return this.technicalInstructions;
  181. }
  182. public get LyricsEntries(): Dictionary<string, LyricsEntry> {
  183. return this.lyricsEntries;
  184. }
  185. public get Arpeggio(): Arpeggio {
  186. return this.arpeggio;
  187. }
  188. public set Arpeggio(value: Arpeggio) {
  189. this.arpeggio = value;
  190. }
  191. public get OrnamentContainer(): OrnamentContainer {
  192. return this.ornamentContainer;
  193. }
  194. public set OrnamentContainer(value: OrnamentContainer) {
  195. this.ornamentContainer = value;
  196. }
  197. // WantedStemDirection provides the stem direction to VexFlow in case of more than 1 voice
  198. // for optimal graphical appearance
  199. public set WantedStemDirection(value: StemDirectionType) {
  200. this.wantedStemDirection = value;
  201. }
  202. public get WantedStemDirection(): StemDirectionType {
  203. return this.wantedStemDirection;
  204. }
  205. public set StemDirectionXml(value: StemDirectionType) {
  206. this.stemDirectionXml = value;
  207. }
  208. public get StemDirectionXml(): StemDirectionType {
  209. return this.stemDirectionXml;
  210. }
  211. // StemDirection holds the actual value of the stem
  212. public set StemDirection(value: StemDirectionType) {
  213. this.stemDirection = value;
  214. }
  215. public get StemDirection(): StemDirectionType {
  216. return this.stemDirection;
  217. }
  218. public get StemColorXml(): string {
  219. return this.stemColorXml;
  220. }
  221. public set StemColorXml(value: string) {
  222. this.stemColorXml = value;
  223. }
  224. public get StemColor(): string {
  225. return this.stemColor;
  226. }
  227. public set StemColor(value: string) {
  228. this.stemColor = value;
  229. }
  230. public get VolumeModifier(): Articulation {
  231. return this.volumeModifier;
  232. }
  233. public set VolumeModifier(value: Articulation) {
  234. this.volumeModifier = value;
  235. }
  236. public get DurationModifier(): Articulation {
  237. return this.durationModifier;
  238. }
  239. public set DurationModifier(value: Articulation) {
  240. this.durationModifier = value;
  241. }
  242. public hasArticulation(articulation: Articulation): boolean {
  243. for (const existingArticulation of this.articulations) {
  244. if (existingArticulation.Equals(articulation)) {
  245. return true;
  246. }
  247. }
  248. return false;
  249. }
  250. public static isSupportedArticulation(articulation: ArticulationEnum): boolean {
  251. switch (articulation) {
  252. case ArticulationEnum.accent:
  253. case ArticulationEnum.strongaccent:
  254. case ArticulationEnum.softaccent:
  255. case ArticulationEnum.invertedstrongaccent:
  256. case ArticulationEnum.staccato:
  257. case ArticulationEnum.staccatissimo:
  258. case ArticulationEnum.spiccato:
  259. case ArticulationEnum.tenuto:
  260. case ArticulationEnum.fermata:
  261. case ArticulationEnum.invertedfermata:
  262. case ArticulationEnum.breathmark:
  263. case ArticulationEnum.caesura:
  264. case ArticulationEnum.lefthandpizzicato:
  265. case ArticulationEnum.naturalharmonic:
  266. case ArticulationEnum.snappizzicato:
  267. case ArticulationEnum.upbow:
  268. case ArticulationEnum.downbow:
  269. case ArticulationEnum.bend:
  270. return true;
  271. default:
  272. return false;
  273. }
  274. }
  275. public hasTie(): boolean {
  276. for (let idx: number = 0, len: number = this.Notes.length; idx < len; ++idx) {
  277. const note: Note = this.Notes[idx];
  278. if (note.NoteTie) { return true; }
  279. }
  280. return false;
  281. }
  282. public hasSlur(): boolean {
  283. for (let idx: number = 0, len: number = this.Notes.length; idx < len; ++idx) {
  284. const note: Note = this.Notes[idx];
  285. if (note.NoteSlurs.length > 0) { return true; }
  286. }
  287. return false;
  288. }
  289. public isStaccato(): boolean {
  290. for (const articulation of this.Articulations) {
  291. if (articulation.articulationEnum === ArticulationEnum.staccato) {
  292. return true;
  293. }
  294. }
  295. return false;
  296. }
  297. public isAccent(): boolean {
  298. for (const articulation of this.Articulations) {
  299. if (articulation.articulationEnum === ArticulationEnum.accent || articulation.articulationEnum === ArticulationEnum.strongaccent) {
  300. return true;
  301. }
  302. }
  303. return false;
  304. }
  305. public getVerseNumberForLyricEntry(lyricsEntry: LyricsEntry): string {
  306. let verseNumber: string = "1";
  307. this.lyricsEntries.forEach((key: string, value: LyricsEntry): void => {
  308. if (lyricsEntry === value) {
  309. verseNumber = key;
  310. }
  311. });
  312. return verseNumber;
  313. }
  314. }
  315. export enum ArticulationEnum {
  316. accent,
  317. strongaccent,
  318. softaccent,
  319. marcatoup,
  320. marcatodown,
  321. invertedstrongaccent,
  322. staccato,
  323. staccatissimo,
  324. spiccato,
  325. tenuto,
  326. fermata,
  327. invertedfermata,
  328. breathmark,
  329. caesura,
  330. lefthandpizzicato,
  331. naturalharmonic,
  332. snappizzicato,
  333. upbow,
  334. downbow,
  335. scoop,
  336. plop,
  337. doit,
  338. falloff,
  339. stress,
  340. unstress,
  341. detachedlegato,
  342. otherarticulation,
  343. bend
  344. }
  345. export enum StemDirectionType {
  346. Undefined = -1,
  347. Up = 0,
  348. Down = 1,
  349. None = 2,
  350. Double = 3
  351. }