-
Notifications
You must be signed in to change notification settings - Fork 59
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Any chance of a seek API? #7
Comments
I took the day to play with this and I got a basic video seek API working, actually, just implemented a |
Just in case it can be useful for someone, here my seek code: if (IsMouseButtonPressed(MOUSE_LEFT_BUTTON) && CheckCollisionPointRec(GetMousePosition(), timeBar))
{
int timeBarPositionX = GetMouseX();
// Get equivalent audio frame
currentAudioFrame = (timeBarPositionX*totalAudioFrames)/timeBar.width;
// Get equivalent video frame (use audio frame to sync)
currentVideoFrame = (int)((float)currentAudioFrame*(float)totalVideoFrames/(float)totalAudioFrames);
// Reset video/audio
if (plm != NULL)
{
plm_rewind(plm);
frame = NULL;
samples = NULL;
baseTime = 0.0;
timeExcess = 0.0;
}
// Move to required video/audio frame
for (int i = 0; i < currentAudioFrame; i++) samples = plm_decode_audio(plm); // TODO: plm_skip_audio_frame(plm)
for (int i = 0; i < currentVideoFrame; i++) frame = plm_skip_video_frame(plm);
} About sync, I'm requesting video/audio frames manually and measuring time to display one new frame every if ((plm != NULL) && !pause)
{
// Video should run at 'framerate' fps => One new frame every 1/framerate
double time = (GetTime() - baseTime);
double frameTime = (1.0/framerate);
if (time >= frameTime)
{
timeExcess += (time - frameTime);
baseTime = GetTime();
// Decode video frame
frame = plm_decode_video(plm); // Get frame as 3 planes: Y, Cr, Cb
currentVideoFrame++;
if (timeExcess >= frameTime)
{
// Discard previous frame a load new one
frame = plm_decode_video(plm);
currentVideoFrame++;
timeExcess = 0;
}
if (frame != NULL) // We got a video frame!
{
plm_frame_to_rgb(frame, imFrame.data); // Convert (Y, Cr, Cb) to RGB on the CPU (slow)
UpdateTexture(texture, imFrame.data); // Update texture with new data for drawing
}
}
// Refill audio stream if required
while (IsAudioStreamProcessed(stream))
{
// Decode audio sample
samples = plm_decode_audio(plm);
currentAudioFrame++;
if (samples != NULL) // We got an audio frame (1152 samples)!
{
// Copy sample to audio stream
UpdateAudioStream(stream, samples->interleaved, PLM_AUDIO_SAMPLES_PER_FRAME*2);
}
}
} It works really good and fast, just an acceptable seek delay (~0.5s) due to the |
Hmm. I'm inspired to try this out myself! Thank Ray, looks very useful :) |
Hi @RandyGaul! Glad you find it useful! Just in case you or someone else wants to try it, here there are the sources: raylib_video_player.zip I'm not submitting the |
Ping @phoboslab as he is not watching the repo :D |
@phoboslab why aren’t you following your own repo? XD |
Well, the simple honest truth is that this is just a hobby project. If you want to get my attention, pay me. Otherwise I'll work on it whenever I feel like it. On topic: thanks for the seeking code! Looks like a good starting point to put this into pl_mpeg proper. I believe calculating the total video length could be done by just looking at the PS packet headers, as most of them include a presentation timestamp. Then there'd be no need for demuxing or even decoding to find the length. Ideally seeking would be done by a binary search - estimating the byte offset, finding packet headers and jumping forward/backward until we are at the right spot. Or, probably much cleaner: build an index of packet offsets and timestamps at load time. |
Pretty sure everyone assumes these kinds of things are merely hobbies :) |
int plm_seek(plm_t *self, double time, int seek_exact); A seek API is now implemented. Seeking should be instant provided your data source can keep up. The seeking is done on the packet layer and no unnecessary video or audio data is decoded in the process. Note that seeking to inter frames will still decode all frames starting at the previous intra frame. This does not build an index of timestamps up front, but instead guesses the byte offset of the seek time. Writing the function that seeks to the correct PS packet took me longer than I'd like to admit. It's a bit hairy, but heavily documented for anyone interested: https://github.com/phoboslab/pl_mpeg/blob/master/pl_mpeg.h#L1876 From the documentation:
Edit: there's also a new function to get the file's duration: double plm_get_duration(plm_t *self); |
I'm using this library with raylib to play MPEG files, it works flawlessly! In under 150 lines of C I got a nice video player!
Just wondering if there is any plan to add a frame/sample seek API. It would be really great.
(I can keep decoding frames/samples to implement a seek mechanism but it seems a bit inefficient)
EDIT: Just created a function to get video length but it's extremely inefficient, it takes up to 10 seconds on my system. I can probably execute it in a second thread while starting playing the video in main thread... but... any better idea to get video length?
The text was updated successfully, but these errors were encountered: