Cursor.ts 19 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417
  1. import {MusicPartManagerIterator} from "../MusicalScore/MusicParts/MusicPartManagerIterator";
  2. import {MusicPartManager} from "../MusicalScore/MusicParts/MusicPartManager";
  3. import {VoiceEntry} from "../MusicalScore/VoiceData/VoiceEntry";
  4. import {VexFlowStaffEntry} from "../MusicalScore/Graphical/VexFlow/VexFlowStaffEntry";
  5. import {MusicSystem} from "../MusicalScore/Graphical/MusicSystem";
  6. import {OpenSheetMusicDisplay} from "./OpenSheetMusicDisplay";
  7. import {GraphicalMusicSheet} from "../MusicalScore/Graphical/GraphicalMusicSheet";
  8. import {Instrument} from "../MusicalScore/Instrument";
  9. import {Note} from "../MusicalScore/VoiceData/Note";
  10. import {Fraction} from "../Common/DataObjects/Fraction";
  11. import { EngravingRules } from "../MusicalScore/Graphical/EngravingRules";
  12. import { SourceMeasure } from "../MusicalScore/VoiceData/SourceMeasure";
  13. import { StaffLine } from "../MusicalScore/Graphical/StaffLine";
  14. import { GraphicalMeasure } from "../MusicalScore/Graphical/GraphicalMeasure";
  15. import { VexFlowMeasure } from "../MusicalScore/Graphical/VexFlow/VexFlowMeasure";
  16. import { CursorOptions } from "./OSMDOptions";
  17. import { BoundingBox, GraphicalStaffEntry } from "../MusicalScore";
  18. import { IPlaybackListener } from "../Common/Interfaces/IPlaybackListener";
  19. import { CursorPosChangedData } from "../Common/DataObjects/CursorPosChangedData";
  20. import { PointF2D } from "../Common/DataObjects";
  21. /**
  22. * A cursor which can iterate through the music sheet.
  23. */
  24. export class Cursor implements IPlaybackListener {
  25. constructor(container: HTMLElement, openSheetMusicDisplay: OpenSheetMusicDisplay, cursorOptions: CursorOptions) {
  26. this.container = container;
  27. this.openSheetMusicDisplay = openSheetMusicDisplay;
  28. this.rules = this.openSheetMusicDisplay.EngravingRules;
  29. this.cursorOptions = cursorOptions;
  30. // set cursor id
  31. // TODO add this for the OSMD object as well and refactor this into a util method?
  32. let id: number = 0;
  33. this.cursorElementId = "cursorImg-0";
  34. // find unique cursor id in document
  35. while (document.getElementById(this.cursorElementId)) {
  36. id++;
  37. this.cursorElementId = `cursorImg-${id}`;
  38. }
  39. const curs: HTMLElement = document.createElement("img");
  40. curs.id = this.cursorElementId;
  41. curs.style.position = "absolute";
  42. if (this.cursorOptions.follow === true) {
  43. curs.style.zIndex = "-1";
  44. } else {
  45. curs.style.zIndex = "-2";
  46. }
  47. this.cursorElement = <HTMLImageElement>curs;
  48. this.container.appendChild(curs);
  49. }
  50. public cursorPositionChanged(timestamp: Fraction, data: CursorPosChangedData): void {
  51. // if (this.iterator.CurrentEnrolledTimestamp.lt(timestamp)) {
  52. // this.iterator.moveToNext();
  53. // while (this.iterator.CurrentEnrolledTimestamp.lt(timestamp)) {
  54. // this.iterator.moveToNext();
  55. // }
  56. // } else if (this.iterator.CurrentEnrolledTimestamp.gt(timestamp)) {
  57. // this.iterator = new MusicPartManagerIterator(this.manager.MusicSheet, timestamp);
  58. // }
  59. this.updateWithTimestamp(data.PredictedPosition);
  60. }
  61. public pauseOccurred(o: object): void {
  62. // throw new Error("Method not implemented.");
  63. }
  64. public selectionEndReached(o: object): void {
  65. // throw new Error("Method not implemented.");
  66. }
  67. public resetOccurred(o: object): void {
  68. this.reset();
  69. }
  70. public notesPlaybackEventOccurred(o: object): void {
  71. // throw new Error("Method not implemented.");
  72. }
  73. private container: HTMLElement;
  74. public cursorElement: HTMLImageElement;
  75. /** a unique id of the cursor's HTMLElement in the document.
  76. * Should be constant between re-renders and backend changes,
  77. * but different between different OSMD objects on the same page.
  78. */
  79. public cursorElementId: string;
  80. private openSheetMusicDisplay: OpenSheetMusicDisplay;
  81. private rules: EngravingRules;
  82. private manager: MusicPartManager;
  83. public iterator: MusicPartManagerIterator;
  84. private graphic: GraphicalMusicSheet;
  85. public hidden: boolean = false;
  86. public currentPageNumber: number = 1;
  87. private cursorOptions: CursorOptions;
  88. /** Initialize the cursor. Necessary before using functions like show() and next(). */
  89. public init(manager: MusicPartManager, graphic: GraphicalMusicSheet): void {
  90. this.manager = manager;
  91. this.graphic = graphic;
  92. this.reset();
  93. this.hidden = false;
  94. }
  95. /**
  96. * Make the cursor visible
  97. */
  98. public show(): void {
  99. this.hidden = false;
  100. //this.resetIterator(); // TODO maybe not here? though setting measure range to draw, rerendering, then handling cursor show is difficult
  101. this.update();
  102. }
  103. public resetIterator(): void {
  104. if (!this.openSheetMusicDisplay.Sheet || !this.openSheetMusicDisplay.Sheet.SourceMeasures) { // just a safety measure
  105. console.log("OSMD.Cursor.resetIterator(): sheet or measures were null/undefined.");
  106. return;
  107. }
  108. // set selection start, so that when there's MinMeasureToDraw set, the cursor starts there right away instead of at measure 1
  109. const lastSheetMeasureIndex: number = this.openSheetMusicDisplay.Sheet.SourceMeasures.length - 1; // last measure in data model
  110. let startMeasureIndex: number = this.rules.MinMeasureToDrawIndex;
  111. startMeasureIndex = Math.min(startMeasureIndex, lastSheetMeasureIndex);
  112. let endMeasureIndex: number = this.rules.MaxMeasureToDrawIndex;
  113. endMeasureIndex = Math.min(endMeasureIndex, lastSheetMeasureIndex);
  114. const updateSelectionStart: boolean = this.openSheetMusicDisplay.Sheet && (
  115. !this.openSheetMusicDisplay.Sheet.SelectionStart ||
  116. this.openSheetMusicDisplay.Sheet.SelectionStart.WholeValue < startMeasureIndex) &&
  117. this.openSheetMusicDisplay.Sheet.SourceMeasures.length > startMeasureIndex;
  118. if (updateSelectionStart) {
  119. this.openSheetMusicDisplay.Sheet.SelectionStart = this.openSheetMusicDisplay.Sheet.SourceMeasures[startMeasureIndex].AbsoluteTimestamp;
  120. }
  121. if (this.openSheetMusicDisplay.Sheet && this.openSheetMusicDisplay.Sheet.SourceMeasures.length > endMeasureIndex) {
  122. const lastMeasure: SourceMeasure = this.openSheetMusicDisplay.Sheet.SourceMeasures[endMeasureIndex];
  123. this.openSheetMusicDisplay.Sheet.SelectionEnd = Fraction.plus(lastMeasure.AbsoluteTimestamp, lastMeasure.Duration);
  124. }
  125. this.iterator = this.manager.getIterator();
  126. }
  127. private getStaffEntryFromVoiceEntry(voiceEntry: VoiceEntry): VexFlowStaffEntry {
  128. const measureIndex: number = voiceEntry.ParentSourceStaffEntry.VerticalContainerParent.ParentMeasure.measureListIndex;
  129. const staffIndex: number = voiceEntry.ParentSourceStaffEntry.ParentStaff.idInMusicSheet;
  130. return <VexFlowStaffEntry>this.graphic.findGraphicalStaffEntryFromMeasureList(staffIndex, measureIndex, voiceEntry.ParentSourceStaffEntry);
  131. }
  132. public updateWithTimestamp(timestamp: Fraction): void {
  133. const sheetTimestamp: Fraction = this.manager.absoluteEnrolledToSheetTimestamp(timestamp);
  134. const values: [number, MusicSystem, GraphicalStaffEntry] = this.graphic.calculateXPositionFromTimestamp(sheetTimestamp);
  135. const x: number = values[0];
  136. const currentSystem: MusicSystem = values[1];
  137. this.updateCurrentPageFromSystem(currentSystem);
  138. const previousStaffEntry: GraphicalStaffEntry = values[2];
  139. const topMeasureNumber: number = Math.max(previousStaffEntry.parentMeasure.MeasureNumber, 1);
  140. // we have to find the top measure, otherwise the cursor with type 3 "jumps around" between vertical measures
  141. let topMeasure: GraphicalMeasure;
  142. for (const measure of this.graphic.MeasureList[topMeasureNumber - 1]) {
  143. if (measure) {
  144. topMeasure = measure;
  145. break;
  146. }
  147. }
  148. const points: [PointF2D, PointF2D] = this.graphic.calculateCursorPoints(x, currentSystem);
  149. const y: number = points[0].y;
  150. const height: number = points[1].y - y;
  151. this.updateWidthAndStyle(topMeasure.PositionAndShape, x, y, height);
  152. if (this.openSheetMusicDisplay.FollowCursor) {
  153. const diff: number = this.cursorElement.getBoundingClientRect().top;
  154. this.cursorElement.scrollIntoView({behavior: diff < 1000 ? "smooth" : "auto", block: "center"});
  155. }
  156. // Show cursor
  157. // // Old cursor: this.graphic.Cursors.push(cursor);
  158. this.cursorElement.style.display = "";
  159. }
  160. public update(): void {
  161. if (this.hidden || this.hidden === undefined || this.hidden === null) {
  162. return;
  163. }
  164. this.updateCurrentPage(); // attach cursor to new page DOM if necessary
  165. // this.graphic?.Cursors?.length = 0;
  166. const iterator: MusicPartManagerIterator = this.iterator;
  167. // TODO when measure draw range (drawUpToMeasureNumber) was changed, next/update can fail to move cursor. but of course it can be reset before.
  168. const voiceEntries: VoiceEntry[] = iterator.CurrentVisibleVoiceEntries();
  169. if (iterator.EndReached || !iterator.CurrentVoiceEntries || voiceEntries.length === 0) {
  170. return;
  171. }
  172. let x: number = 0, y: number = 0, height: number = 0;
  173. let musicSystem: MusicSystem;
  174. if (iterator.CurrentMeasure.isReducedToMultiRest) {
  175. const multiRestGMeasure: GraphicalMeasure = this.graphic.findGraphicalMeasure(iterator.CurrentMeasureIndex, 0);
  176. const totalRestMeasures: number = multiRestGMeasure.parentSourceMeasure.multipleRestMeasures;
  177. const currentRestMeasureNumber: number = iterator.CurrentMeasure.multipleRestMeasureNumber;
  178. const progressRatio: number = currentRestMeasureNumber / (totalRestMeasures + 1);
  179. const effectiveWidth: number = multiRestGMeasure.PositionAndShape.Size.width - (multiRestGMeasure as VexFlowMeasure).beginInstructionsWidth;
  180. x = multiRestGMeasure.PositionAndShape.AbsolutePosition.x + (multiRestGMeasure as VexFlowMeasure).beginInstructionsWidth + progressRatio * effectiveWidth;
  181. musicSystem = multiRestGMeasure.ParentMusicSystem;
  182. } else {
  183. // get all staff entries inside the current voice entry
  184. const gseArr: VexFlowStaffEntry[] = voiceEntries.map(ve => this.getStaffEntryFromVoiceEntry(ve));
  185. // sort them by x position and take the leftmost entry
  186. const gse: VexFlowStaffEntry =
  187. gseArr.sort((a, b) => a?.PositionAndShape?.AbsolutePosition?.x <= b?.PositionAndShape?.AbsolutePosition?.x ? -1 : 1 )[0];
  188. x = gse.PositionAndShape.AbsolutePosition.x;
  189. musicSystem = gse.parentMeasure.ParentMusicSystem;
  190. // debug: change color of notes under cursor (needs re-render)
  191. // for (const gve of gse.graphicalVoiceEntries) {
  192. // for (const note of gve.notes) {
  193. // note.sourceNote.NoteheadColor = "#0000FF";
  194. // }
  195. // }
  196. }
  197. if (!musicSystem) {
  198. return;
  199. }
  200. // y is common for both multirest and non-multirest, given the MusicSystem
  201. y = musicSystem.PositionAndShape.AbsolutePosition.y + musicSystem.StaffLines[0].PositionAndShape.RelativePosition.y;
  202. const bottomStaffline: StaffLine = musicSystem.StaffLines[musicSystem.StaffLines.length - 1];
  203. const endY: number = musicSystem.PositionAndShape.AbsolutePosition.y +
  204. bottomStaffline.PositionAndShape.RelativePosition.y + bottomStaffline.StaffHeight;
  205. height = endY - y;
  206. // Update the graphical cursor
  207. const measurePositionAndShape: BoundingBox = this.graphic.findGraphicalMeasure(iterator.CurrentMeasureIndex, 0).PositionAndShape;
  208. this.updateWidthAndStyle(measurePositionAndShape, x, y, height);
  209. if (this.openSheetMusicDisplay.FollowCursor) {
  210. if (!this.openSheetMusicDisplay.EngravingRules.RenderSingleHorizontalStaffline) {
  211. const diff: number = this.cursorElement.getBoundingClientRect().top;
  212. this.cursorElement.scrollIntoView({behavior: diff < 1000 ? "smooth" : "auto", block: "center"});
  213. } else {
  214. this.cursorElement.scrollIntoView({behavior: "smooth", inline: "center"});
  215. }
  216. }
  217. // Show cursor
  218. // // Old cursor: this.graphic.Cursors.push(cursor);
  219. this.cursorElement.style.display = "";
  220. }
  221. public updateWidthAndStyle(measurePositionAndShape: BoundingBox, x: number, y: number, height: number): void {
  222. const cursorElement: HTMLImageElement = this.cursorElement;
  223. let newWidth: number = 0;
  224. switch (this.cursorOptions.type) {
  225. case 1:
  226. cursorElement.style.top = (y * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  227. cursorElement.style.left = ((x - 1.5) * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  228. cursorElement.height = (height * 10.0 * this.openSheetMusicDisplay.zoom);
  229. newWidth = 5 * this.openSheetMusicDisplay.zoom;
  230. break;
  231. case 2:
  232. cursorElement.style.top = ((y-2.5) * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  233. cursorElement.style.left = (x * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  234. cursorElement.height = (1.5 * 10.0 * this.openSheetMusicDisplay.zoom);
  235. newWidth = 5 * this.openSheetMusicDisplay.zoom;
  236. break;
  237. case 3:
  238. cursorElement.style.top = measurePositionAndShape.AbsolutePosition.y * 10.0 * this.openSheetMusicDisplay.zoom +"px";
  239. cursorElement.style.left = measurePositionAndShape.AbsolutePosition.x * 10.0 * this.openSheetMusicDisplay.zoom +"px";
  240. cursorElement.height = (height * 10.0 * this.openSheetMusicDisplay.zoom);
  241. newWidth = measurePositionAndShape.Size.width * 10 * this.openSheetMusicDisplay.zoom;
  242. break;
  243. case 4:
  244. cursorElement.style.top = measurePositionAndShape.AbsolutePosition.y * 10.0 * this.openSheetMusicDisplay.zoom +"px";
  245. cursorElement.style.left = measurePositionAndShape.AbsolutePosition.x * 10.0 * this.openSheetMusicDisplay.zoom +"px";
  246. cursorElement.height = (height * 10.0 * this.openSheetMusicDisplay.zoom);
  247. newWidth = (x-measurePositionAndShape.AbsolutePosition.x) * 10 * this.openSheetMusicDisplay.zoom;
  248. break;
  249. default:
  250. cursorElement.style.top = (y * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  251. cursorElement.style.left = ((x - 1.5) * 10.0 * this.openSheetMusicDisplay.zoom) + "px";
  252. cursorElement.height = (height * 10.0 * this.openSheetMusicDisplay.zoom);
  253. newWidth = 3 * 10.0 * this.openSheetMusicDisplay.zoom;
  254. break;
  255. }
  256. if (newWidth !== cursorElement.width) {
  257. cursorElement.width = newWidth;
  258. this.updateStyle(newWidth, this.cursorOptions);
  259. }
  260. }
  261. /**
  262. * Hide the cursor
  263. */
  264. public hide(): void {
  265. // Hide the actual cursor element
  266. this.cursorElement.style.display = "none";
  267. //this.graphic.Cursors.length = 0;
  268. // Forcing the sheet to re-render is not necessary anymore
  269. //if (!this.hidden) {
  270. // this.openSheetMusicDisplay.render();
  271. //}
  272. this.hidden = true;
  273. }
  274. /**
  275. * Go to next entry
  276. */
  277. public next(): void {
  278. this.iterator.moveToNextVisibleVoiceEntry(false); // moveToNext() would not skip notes in hidden (visible = false) parts
  279. this.update();
  280. }
  281. /**
  282. * reset cursor to start
  283. */
  284. public reset(): void {
  285. this.resetIterator();
  286. //this.iterator.moveToNext();
  287. const iterTmp: MusicPartManagerIterator = this.manager.getIterator(this.graphic.ParentMusicSheet.SelectionStart);
  288. this.updateWithTimestamp(iterTmp.CurrentEnrolledTimestamp);
  289. }
  290. private updateStyle(width: number, cursorOptions: CursorOptions = undefined): void {
  291. if (cursorOptions !== undefined) {
  292. this.cursorOptions = cursorOptions;
  293. }
  294. // Create a dummy canvas to generate the gradient for the cursor
  295. // FIXME This approach needs to be improved
  296. const c: HTMLCanvasElement = document.createElement("canvas");
  297. c.width = this.cursorElement.width;
  298. c.height = 1;
  299. const ctx: CanvasRenderingContext2D = c.getContext("2d");
  300. ctx.globalAlpha = this.cursorOptions.alpha;
  301. // Generate the gradient
  302. const gradient: CanvasGradient = ctx.createLinearGradient(0, 0, this.cursorElement.width, 0);
  303. switch (this.cursorOptions.type) {
  304. case 1:
  305. case 2:
  306. case 3:
  307. case 4:
  308. gradient.addColorStop(1, this.cursorOptions.color);
  309. break;
  310. default:
  311. gradient.addColorStop(0, "white"); // it was: "transparent"
  312. gradient.addColorStop(0.2, this.cursorOptions.color);
  313. gradient.addColorStop(0.8, this.cursorOptions.color);
  314. gradient.addColorStop(1, "white"); // it was: "transparent"
  315. break;
  316. }
  317. ctx.fillStyle = gradient;
  318. ctx.fillRect(0, 0, width, 1);
  319. // Set the actual image
  320. this.cursorElement.src = c.toDataURL("image/png");
  321. }
  322. public get Iterator(): MusicPartManagerIterator {
  323. return this.iterator;
  324. }
  325. public get Hidden(): boolean {
  326. return this.hidden;
  327. }
  328. /** returns voices under the current Cursor position. Without instrument argument, all voices are returned. */
  329. public VoicesUnderCursor(instrument?: Instrument): VoiceEntry[] {
  330. return this.iterator.CurrentVisibleVoiceEntries(instrument);
  331. }
  332. public NotesUnderCursor(instrument?: Instrument): Note[] {
  333. const voiceEntries: VoiceEntry[] = this.VoicesUnderCursor(instrument);
  334. const notes: Note[] = [];
  335. voiceEntries.forEach(voiceEntry => {
  336. notes.push.apply(notes, voiceEntry.Notes);
  337. });
  338. return notes;
  339. }
  340. /** Check if there was a change in current page, and attach cursor element to the corresponding HTMLElement (div).
  341. * This is only necessary if using PageFormat (multiple pages).
  342. */
  343. public updateCurrentPage(): number {
  344. const timestamp: Fraction = this.iterator.currentTimeStamp;
  345. for (const page of this.graphic.MusicPages) {
  346. const lastSystemTimestamp: Fraction = page.MusicSystems.last().GetSystemsLastTimeStamp();
  347. if (lastSystemTimestamp.gt(timestamp)) {
  348. // gt: the last timestamp of the last system is equal to the first of the next page,
  349. // so we do need to use gt, not gte here.
  350. const newPageNumber: number = page.PageNumber;
  351. if (newPageNumber !== this.currentPageNumber) {
  352. this.container.removeChild(this.cursorElement);
  353. this.container = document.getElementById("osmdCanvasPage" + newPageNumber);
  354. this.container.appendChild(this.cursorElement);
  355. // TODO maybe store this.pageCurrentlyAttachedTo, though right now it isn't necessary
  356. // alternative to remove/append:
  357. // this.openSheetMusicDisplay.enableOrDisableCursor(true);
  358. }
  359. return this.currentPageNumber = newPageNumber;
  360. }
  361. }
  362. return 1;
  363. }
  364. public updateCurrentPageFromSystem(system: MusicSystem): number {
  365. if (system !== undefined) {
  366. const newPageNumber: number = system.Parent.PageNumber;
  367. if (newPageNumber !== this.currentPageNumber) {
  368. this.container.removeChild(this.cursorElement);
  369. this.container = document.getElementById("osmdCanvasPage" + newPageNumber);
  370. this.container.appendChild(this.cursorElement);
  371. // TODO maybe store this.pageCurrentlyAttachedTo, though right now it isn't necessary
  372. // alternative to remove/append:
  373. // this.openSheetMusicDisplay.enableOrDisableCursor(true);
  374. }
  375. return this.currentPageNumber = newPageNumber;
  376. }
  377. return 1;
  378. }
  379. }