gstreamer-rs/tutorials/src/bin/basic-tutorial-3.rs
2019-02-21 19:56:23 +02:00

133 lines
4.3 KiB
Rust

extern crate gstreamer as gst;
use gst::prelude::*;
#[path = "../tutorials-common.rs"]
mod tutorials_common;
fn tutorial_main() {
// Initialize GStreamer
gst::init().unwrap();
// Create the elements
let source = gst::ElementFactory::make("uridecodebin", "source")
.expect("Could not create uridecodebin element.");
let convert = gst::ElementFactory::make("audioconvert", "convert")
.expect("Could not create convert element.");
let sink =
gst::ElementFactory::make("autoaudiosink", "sink").expect("Could not create sink element.");
// Create the empty pipeline
let pipeline = gst::Pipeline::new("test-pipeline");
// Build the pipeline Note that we are NOT linking the source at this
// point. We will do it later.
pipeline.add_many(&[&source, &convert, &sink]).unwrap();
convert.link(&sink).expect("Elements could not be linked.");
// Set the URI to play
let uri =
"https://www.freedesktop.org/software/gstreamer-sdk/data/media/sintel_trailer-480p.webm";
source
.set_property("uri", &uri)
.expect("Can't set uri property on uridecodebin");
// Connect the pad-added signal
let pipeline_weak = pipeline.downgrade();
let convert_weak = convert.downgrade();
source.connect_pad_added(move |_, src_pad| {
let pipeline = match pipeline_weak.upgrade() {
Some(pipeline) => pipeline,
None => return,
};
let convert = match convert_weak.upgrade() {
Some(convert) => convert,
None => return,
};
println!(
"Received new pad {} from {}",
src_pad.get_name(),
pipeline.get_name()
);
let sink_pad = convert
.get_static_pad("sink")
.expect("Failed to get static sink pad from convert");
if sink_pad.is_linked() {
println!("We are already linked. Ignoring.");
return;
}
let new_pad_caps = src_pad
.get_current_caps()
.expect("Failed to get caps of new pad.");
let new_pad_struct = new_pad_caps
.get_structure(0)
.expect("Failed to get first structure of caps.");
let new_pad_type = new_pad_struct.get_name();
let is_audio = new_pad_type.starts_with("audio/x-raw");
if !is_audio {
println!(
"It has type {} which is not raw audio. Ignoring.",
new_pad_type
);
return;
}
let res = src_pad.link(&sink_pad);
if res.is_err() {
println!("Type is {} but link failed.", new_pad_type);
} else {
println!("Link succeeded (type {}).", new_pad_type);
}
});
// Start playing
pipeline
.set_state(gst::State::Playing)
.expect("Unable to set the pipeline to the `Playing` state");
// Wait until error or EOS
let bus = pipeline.get_bus().unwrap();
for msg in bus.iter_timed(gst::CLOCK_TIME_NONE) {
use gst::MessageView;
match msg.view() {
MessageView::Error(err) => {
eprintln!(
"Error received from element {:?} {}",
err.get_src().map(|s| s.get_path_string()),
err.get_error()
);
eprintln!("Debugging information: {:?}", err.get_debug());
break;
}
MessageView::StateChanged(state_changed) => {
if state_changed
.get_src()
.map(|s| s == pipeline)
.unwrap_or(false)
{
println!(
"Pipeline state changed from {:?} to {:?}",
state_changed.get_old(),
state_changed.get_current()
);
}
}
MessageView::Eos(..) => break,
_ => (),
}
}
pipeline
.set_state(gst::State::Null)
.expect("Unable to set the pipeline to the `Null` state");
}
fn main() {
// tutorials_common::run is only required to set up the application environment on macOS
// (but not necessary in normal Cocoa applications where this is set up automatically)
tutorials_common::run(tutorial_main);
}