forked from mirrors/gstreamer-rs
Add basic-tutorial-4 from the GStreamer tutorials
https://cgit.freedesktop.org/gstreamer/gst-docs/tree/examples/tutorials/basic-tutorial-4.c Fixes https://github.com/sdroege/gstreamer-rs/pull/39
This commit is contained in:
parent
d85be4fe03
commit
f253caac19
1 changed files with 149 additions and 0 deletions
149
tutorials/src/bin/basic-tutorial-4.rs
Normal file
149
tutorials/src/bin/basic-tutorial-4.rs
Normal file
|
@ -0,0 +1,149 @@
|
|||
extern crate gstreamer as gst;
|
||||
use std::io;
|
||||
use std::io::Write;
|
||||
use gst::prelude::*;
|
||||
use gst::MessageView;
|
||||
|
||||
struct CustomData {
|
||||
playbin: gst::Element, // Our one and only element
|
||||
playing: bool, // Are we in the PLAYING state?
|
||||
terminate: bool, // Should we terminate execution?
|
||||
seek_enabled: bool, // Is seeking enabled for this media?
|
||||
seek_done: bool, // Have we performed the seek already?
|
||||
duration: gst::ClockTime, // How long does this media last, in nanoseconds
|
||||
}
|
||||
|
||||
fn main() {
|
||||
// Initialize GStreamer
|
||||
gst::init().unwrap();
|
||||
|
||||
// Creat the playbin element
|
||||
let playbin =
|
||||
gst::ElementFactory::make("playbin", "playbin").expect("Failed to create playbin element");
|
||||
|
||||
// Set the URI to play
|
||||
let uri = "https://www.freedesktop.org/software/gstreamer-sdk/data/media/sintel_trailer-480p.webm";
|
||||
playbin
|
||||
.set_property("uri", &uri)
|
||||
.expect("Can't set uri property on playbin");
|
||||
|
||||
// Start playing
|
||||
let ret = playbin.set_state(gst::State::Playing);
|
||||
assert_ne!(ret, gst::StateChangeReturn::Failure);
|
||||
|
||||
// Listen to the bus
|
||||
let bus = playbin.get_bus().unwrap();
|
||||
let mut custom_data = CustomData {
|
||||
playbin: playbin,
|
||||
playing: false,
|
||||
terminate: false,
|
||||
seek_enabled: false,
|
||||
seek_done: false,
|
||||
duration: gst::CLOCK_TIME_NONE,
|
||||
};
|
||||
|
||||
while !custom_data.terminate {
|
||||
let msg = bus.timed_pop(100 * gst::MSECOND);
|
||||
|
||||
match msg {
|
||||
Some(msg) => {
|
||||
handle_message(&mut custom_data, &msg);
|
||||
}
|
||||
None => {
|
||||
if custom_data.playing {
|
||||
let position = custom_data
|
||||
.playbin
|
||||
.query_position(gst::Format::Time)
|
||||
.expect("Could not query current position.");
|
||||
let position = position as gst::ClockTime;
|
||||
|
||||
// If we didn't know it yet, query the stream duration
|
||||
if custom_data.duration == gst::CLOCK_TIME_NONE {
|
||||
custom_data.duration = custom_data
|
||||
.playbin
|
||||
.query_duration(gst::Format::Time)
|
||||
.expect("Could not query current duration.") as
|
||||
gst::ClockTime;
|
||||
}
|
||||
|
||||
// Print current position and total duration
|
||||
print!("\rPosition {} / {}", position, custom_data.duration);
|
||||
io::stdout().flush().unwrap();
|
||||
|
||||
if custom_data.seek_enabled && !custom_data.seek_done &&
|
||||
position > 10 * gst::SECOND
|
||||
{
|
||||
println!("\nReached 10s, performing seek...");
|
||||
custom_data
|
||||
.playbin
|
||||
.seek_simple(
|
||||
gst::Format::Time,
|
||||
gst::SeekFlags::FLUSH | gst::SeekFlags::KEY_UNIT,
|
||||
(30 * gst::SECOND) as i64,
|
||||
)
|
||||
.expect("Failed to seek.");
|
||||
custom_data.seek_done = true;
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Shutdown pipeline
|
||||
let ret = custom_data.playbin.set_state(gst::State::Null);
|
||||
assert_ne!(ret, gst::StateChangeReturn::Failure);
|
||||
}
|
||||
|
||||
fn handle_message(custom_data: &mut CustomData, msg: &gst::GstRc<gst::MessageRef>) {
|
||||
match msg.view() {
|
||||
MessageView::Error(err) => {
|
||||
println!(
|
||||
"Error received from element {}: {} ({:?})",
|
||||
msg.get_src().get_path_string(),
|
||||
err.get_error(),
|
||||
err.get_debug()
|
||||
);
|
||||
custom_data.terminate = true;
|
||||
}
|
||||
MessageView::Eos(..) => {
|
||||
println!("End-Of-Stream reached.");
|
||||
custom_data.terminate = true;
|
||||
}
|
||||
MessageView::DurationChanged(_) => {
|
||||
// The duration has changed, mark the current one as invalid
|
||||
custom_data.duration = gst::CLOCK_TIME_NONE;
|
||||
}
|
||||
MessageView::StateChanged(state) => if msg.get_src() == custom_data.playbin {
|
||||
let new_state = state.get_current();
|
||||
let old_state = state.get_old();
|
||||
|
||||
println!(
|
||||
"Pipeline state changed from {:?} to {:?}",
|
||||
old_state,
|
||||
new_state
|
||||
);
|
||||
|
||||
custom_data.playing = new_state == gst::State::Playing;
|
||||
if custom_data.playing {
|
||||
let query = gst::Query::new_seeking(gst::Format::Time);
|
||||
if custom_data.playbin.query(query.get_mut().unwrap()) {
|
||||
match query.view() {
|
||||
gst::QueryView::Seeking(seek) => {
|
||||
let (_fmt, seekable, start, end) = seek.get();
|
||||
custom_data.seek_enabled = seekable;
|
||||
if seekable {
|
||||
println!("Seeking is ENABLED from {} to {}", start, end)
|
||||
} else {
|
||||
println!("Seeking is DISABLED for this stream.")
|
||||
}
|
||||
}
|
||||
_ => unreachable!(),
|
||||
}
|
||||
} else {
|
||||
eprintln!("Seeking query failed.")
|
||||
}
|
||||
}
|
||||
},
|
||||
_ => (),
|
||||
}
|
||||
}
|
Loading…
Reference in a new issue