diff --git a/front/components/PartitionCoord.tsx b/front/components/PartitionCoord.tsx index 76216b1..1ad6a28 100644 --- a/front/components/PartitionCoord.tsx +++ b/front/components/PartitionCoord.tsx @@ -31,7 +31,9 @@ const PartitionCoord = ({ setPartitionData([base64data, a]); onPartitionReady(); }} - onEndReached={onEndReached} + onEndReached={() => { + console.log('osmd end reached'); + }} timestamp={timestamp} /> )} @@ -39,6 +41,9 @@ const PartitionCoord = ({ { + onEndReached(); + }} /> )} diff --git a/front/components/PartitionView.tsx b/front/components/PartitionView.tsx index c985daf..20e3781 100644 --- a/front/components/PartitionView.tsx +++ b/front/components/PartitionView.tsx @@ -28,7 +28,7 @@ const PartitionView = (props: PartitionViewProps) => { const [osmd, setOsmd] = useState(); const [soundPlayer, setSoundPlayer] = useState(); const audioContext = new SAC.AudioContext(); - const [wholeNoteLength, setWholeNoteLength] = useState(0); // Length of Whole note, in ms (?) + // const [wholeNoteLength, setWholeNoteLength] = useState(0); // Length of Whole note, in ms (?) const colorScheme = useColorScheme(); const dimensions = useWindowDimensions(); const OSMD_DIV_ID = 'osmd-div'; @@ -45,15 +45,15 @@ const PartitionView = (props: PartitionViewProps) => { autoResize: false, }; // Turns note.Length or timestamp in ms - const timestampToMs = (timestamp: Fraction) => { + const timestampToMs = (timestamp: Fraction, wholeNoteLength: number) => { return timestamp.RealValue * wholeNoteLength; }; - const getActualNoteLength = (note: Note) => { - let duration = timestampToMs(note.Length); + const getActualNoteLength = (note: Note, wholeNoteLength: number) => { + let duration = timestampToMs(note.Length, wholeNoteLength); if (note.NoteTie) { const firstNote = note.NoteTie.Notes.at(1); if (Object.is(note.NoteTie.StartNote, note) && firstNote) { - duration += timestampToMs(firstNote.Length); + duration += timestampToMs(firstNote.Length, wholeNoteLength); } else { duration = 0; } @@ -97,9 +97,38 @@ const PartitionView = (props: PartitionViewProps) => { _osmd.render(); _osmd.cursor.show(); // get the current cursor position + const bpm = _osmd.Sheet.HasBPMInfo ? _osmd.Sheet.getExpressionsStartTempoInBPM() : 60; + // setWholeNoteLength(Math.round((60 / bpm) * 4000)); + const wholeNoteLength = Math.round((60 / bpm) * 4000); const curPos = []; while (!_osmd.cursor.iterator.EndReached) { - curPos.push(_osmd.cursor.cursorElement.offsetLeft); + const notesToPlay = _osmd.cursor + .NotesUnderCursor() + .filter((note) => { + return note.isRest() == false && note.Pitch; + }) + .map((note) => { + return { + note: note, + duration: getActualNoteLength(note, wholeNoteLength), + }; + }); + const shortestNotes = _osmd!.cursor + .NotesUnderCursor() + .sort((n1, n2) => n1.Length.CompareTo(n2.Length)) + .at(0); + const ts = timestampToMs(shortestNotes?.getAbsoluteTimestamp() ?? new Fraction(-1), wholeNoteLength); + const sNL = timestampToMs(shortestNotes?.Length ?? new Fraction(-1), wholeNoteLength); + curPos.push({ + offset: _osmd.cursor.cursorElement.offsetLeft, + notes: notesToPlay, + shortedNotes: shortestNotes, + sNinfos: { + ts, + sNL, + isRest: shortestNotes?.isRest(), + } + }); _osmd.cursor.next(); } console.log('curPos', curPos); @@ -110,14 +139,14 @@ const PartitionView = (props: PartitionViewProps) => { console.log('current measure index', _osmd.cursor.iterator.CurrentMeasureIndex); const osmdCanvas = document.querySelector('#' + OSMD_DIV_ID + ' canvas'); // Ty https://github.com/jimutt/osmd-audio-player/blob/ec205a6e46ee50002c1fa8f5999389447bba7bbf/src/PlaybackEngine.ts#LL77C12-L77C63 - const bpm = _osmd.Sheet.HasBPMInfo ? _osmd.Sheet.getExpressionsStartTempoInBPM() : 60; - setWholeNoteLength(Math.round((60 / bpm) * 4000)); props.onPartitionReady( osmdCanvas.toDataURL(), curPos.map((pos) => { return { - x: pos, - timing: Math.floor(Math.random() * 600) + 100, + x: pos.offset, + timing: pos.sNinfos.sNL, + timestamp: pos.sNinfos.ts, + notes: pos.notes, }; }) ); diff --git a/front/components/PartitionVisualizer/PhaserCanvas.tsx b/front/components/PartitionVisualizer/PhaserCanvas.tsx index 2cf2b74..dccc9b2 100644 --- a/front/components/PartitionVisualizer/PhaserCanvas.tsx +++ b/front/components/PartitionVisualizer/PhaserCanvas.tsx @@ -4,16 +4,16 @@ import * as React from 'react'; import { useEffect, useRef } from 'react'; import Phaser from 'phaser'; import useColorScheme from '../../hooks/colorScheme'; +import { PartitionContext } from '../../views/PlayView'; +import { on } from 'events'; -const b64data = - ''; - -const cursorB64 = - ''; +let globalTimestamp = 0; +let globalStatus: 'playing' | 'paused' | 'stopped' = 'playing'; const getPianoScene = ( partitionB64: string, cursorPositions: PianoCursorPosition[], + onEndReached: () => void, colorScheme: 'light' | 'dark' ) => { class PianoScene extends Phaser.Scene { @@ -30,63 +30,39 @@ const getPianoScene = ( this.cursor = this.add.rectangle(0, 0, 30, 350, 0x31ef8c, 0.5).setOrigin(0, 0); this.cameras.main.startFollow(this.cursor, true, 0.05, 0.05); - - this.cursorMoveNext(); - - // // const raster = this.add.image(0, 0, 'raster'); - // const group = this.add.group(); - - // group.createMultiple({ key: 'expo', repeat: 8 }); - - // let ci = 0; - // const colors = [ - // 0xef658c, 0xff9a52, 0xffdf00, 0x31ef8c, 0x21dfff, 0x31aade, 0x5275de, 0x9c55ad, - // 0xbd208c, - // ]; - - // const _this = this; - - // group.children.iterate((child) => { - // child.x = 100; - // child.y = 300; - // child.depth = 9 - ci; - - // child.tint = colors[ci]; - - // ci++; - - // _this.tweens.add({ - // targets: child, - // x: 900, - // yoyo: true, - // repeat: -1, - // ease: 'Sine.easeInOut', - // duration: 1500, - // delay: 100 * ci, - // }); - // }); }); } - cursorMoveNext(gameobject) { - if (this.cursorPositionsIdx >= cursorPositions.length) { - return; - } - this.cursorPositionsIdx++; - const cursorPosition = cursorPositions[this.cursorPositionsIdx]; - if (cursorPosition) { - this.tweens.add({ - targets: this.cursor, - x: cursorPosition.x, - duration: 200, - ease: 'Sine.easeInOut', - onComplete: () => { - // wait cursorPosition.timing - setTimeout(() => { - this.cursorMoveNext(); - }, cursorPosition.timing); - }, + override update() { + const currentTimestamp = globalTimestamp; + const status = globalStatus; + + if (status === 'playing') { + const transitionTime = 75; + const cP = cursorPositions.findLast((cP, idx) => { + if ( + cP.timestamp < currentTimestamp + transitionTime && + idx > this.cursorPositionsIdx + ) { + this.cursorPositionsIdx = idx; + return true; + } + return false; }); + if (cP) { + const tw = { + targets: this!.cursor, + x: cP!.x, + duration: transitionTime, + ease: 'Sine.easeInOut', + }; + if (this.cursorPositionsIdx === cursorPositions.length - 1) { + tw.onComplete = () => { + onEndReached(); + }; + } + this.tweens.add(tw); + } } } } @@ -98,25 +74,39 @@ export type PianoCursorPosition = { x: number; // timestamp in ms timing: number; + timestamp: number; + notes: any[]; }; -type PhaserCanvasProps = { +export type UpdateInfo = { + currentTimestamp: number; + status: 'playing' | 'paused' | 'stopped'; +}; + +export type PhaserCanvasProps = { partitionB64: string; cursorPositions: PianoCursorPosition[]; + onEndReached: () => void; }; -const PhaserCanvas = ({ partitionB64, cursorPositions }: PhaserCanvasProps) => { +const PhaserCanvas = ({ + partitionB64, + cursorPositions, + onEndReached, +}: PhaserCanvasProps) => { const colorScheme = useColorScheme(); + const { timestamp } = React.useContext(PartitionContext); const [game, setGame] = React.useState(null); + globalTimestamp = timestamp; useEffect(() => { - const pianoScene = getPianoScene(partitionB64, cursorPositions, colorScheme); + const pianoScene = getPianoScene(partitionB64, cursorPositions, onEndReached, colorScheme); const config = { type: Phaser.AUTO, parent: 'phaser-canvas', width: 1000, - height: 900, + height: 400, scene: pianoScene, scale: { mode: Phaser.Scale.FIT, diff --git a/front/views/PlayView.tsx b/front/views/PlayView.tsx index 1920cf4..f85e044 100644 --- a/front/views/PlayView.tsx +++ b/front/views/PlayView.tsx @@ -21,7 +21,6 @@ import { transformQuery, useQuery } from '../Queries'; import API from '../API'; import LoadingComponent, { LoadingView } from '../components/Loading'; import Constants from 'expo-constants'; -import VirtualPiano from '../components/VirtualPiano/VirtualPiano'; import { strToKey, keyToStr, Note } from '../models/Piano'; import { useSelector } from 'react-redux'; import { RootState } from '../state/Store'; @@ -70,6 +69,14 @@ function parseMidiMessage(message: MIDIMessageEvent) { }; } +export const PartitionContext = React.createContext<{ + // Timestamp of the play session, in milisecond + timestamp: number; +}>({ + timestamp: 0, +}); + + const PlayView = ({ songId, type, route }: RouteProps) => { const accessToken = useSelector((state: RootState) => state.user.accessToken); const navigation = useNavigation(); @@ -79,7 +86,6 @@ const PlayView = ({ songId, type, route }: RouteProps) => { const webSocket = useRef(); const [paused, setPause] = useState(true); const stopwatch = useStopwatch(); - const [isVirtualPianoVisible, setVirtualPianoVisible] = useState(false); const [time, setTime] = useState(0); const [partitionRendered, setPartitionRendered] = useState(false); // Used to know when partitionview can render const [score, setScore] = useState(0); // Between 0 and 100 @@ -218,7 +224,7 @@ const PlayView = ({ songId, type, route }: RouteProps) => { useEffect(() => { ScreenOrientation.lockAsync(ScreenOrientation.OrientationLock.LANDSCAPE).catch(() => {}); const interval = setInterval(() => { - setTime(() => getElapsedTime()); // Countdown + setTime(() =>getElapsedTime()); // Countdown }, 1); return () => { @@ -253,6 +259,11 @@ const PlayView = ({ songId, type, route }: RouteProps) => { } return ( + ) => { /> */} { onEnd(); }} @@ -288,39 +300,6 @@ const PlayView = ({ songId, type, route }: RouteProps) => { {!partitionRendered && } - {isVirtualPianoVisible && ( - - { - console.log('On note down', keyToStr(note)); - }} - onNoteUp={(note) => { - console.log('On note up', keyToStr(note)); - }} - showOctaveNumbers={true} - startNote={Note.C} - endNote={Note.B} - startOctave={2} - endOctave={5} - style={{ - width: '80%', - height: '100%', - }} - highlightedNotes={[ - { key: strToKey('D3') }, - { key: strToKey('A#'), bgColor: '#00FF00' }, - ]} - /> - - )} ) => { } }} /> - - } - onPress={() => { - setVirtualPianoVisible(!isVirtualPianoVisible); - }} - /> {time < 0 ? paused @@ -406,6 +371,7 @@ const PlayView = ({ songId, type, route }: RouteProps) => { + ); };