2017-10-11 09:15:34 +00:00
|
|
|
#[cfg(feature = "tutorial5")]
|
|
|
|
mod tutorial5 {
|
|
|
|
use std::os::raw::c_void;
|
2018-04-01 08:30:03 +00:00
|
|
|
use std::process;
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
use gdk::prelude::*;
|
|
|
|
use gtk::prelude::*;
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
use gst::prelude::*;
|
|
|
|
use gst_video::prelude::*;
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2018-07-27 10:07:24 +00:00
|
|
|
use std::ops;
|
|
|
|
|
|
|
|
// Custom struct to keep our window reference alive
|
|
|
|
// and to store the timeout id so that we can remove
|
|
|
|
// it from the main context again later and drop the
|
|
|
|
// references it keeps inside its closures
|
|
|
|
struct AppWindow {
|
2020-10-10 08:00:48 +00:00
|
|
|
main_window: gtk::Window,
|
2018-07-27 10:07:24 +00:00
|
|
|
timeout_id: Option<glib::SourceId>,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ops::Deref for AppWindow {
|
2020-10-10 08:00:48 +00:00
|
|
|
type Target = gtk::Window;
|
2018-07-27 10:07:24 +00:00
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
fn deref(&self) -> >k::Window {
|
2018-07-27 10:07:24 +00:00
|
|
|
&self.main_window
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Drop for AppWindow {
|
|
|
|
fn drop(&mut self) {
|
|
|
|
if let Some(source_id) = self.timeout_id.take() {
|
|
|
|
glib::source_remove(source_id);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-10-11 09:15:34 +00:00
|
|
|
// Extract tags from streams of @stype and add the info in the UI.
|
2018-07-27 10:25:47 +00:00
|
|
|
fn add_streams_info(playbin: &gst::Element, textbuf: >k::TextBuffer, stype: &str) {
|
2017-10-11 09:15:34 +00:00
|
|
|
let propname: &str = &format!("n-{}", stype);
|
|
|
|
let signame: &str = &format!("get-{}-tags", stype);
|
|
|
|
|
|
|
|
match playbin.get_property(propname).unwrap().get() {
|
2019-08-11 07:33:34 +00:00
|
|
|
Ok(Some(x)) => {
|
2018-10-08 12:02:23 +00:00
|
|
|
for i in 0..x {
|
|
|
|
let tags = playbin.emit(signame, &[&i]).unwrap().unwrap();
|
|
|
|
|
2019-08-11 07:33:34 +00:00
|
|
|
if let Ok(Some(tags)) = tags.get::<gst::TagList>() {
|
2018-10-08 12:02:23 +00:00
|
|
|
textbuf.insert_at_cursor(&format!("{} stream {}:\n ", stype, i));
|
|
|
|
|
|
|
|
if let Some(codec) = tags.get::<gst::tags::VideoCodec>() {
|
|
|
|
textbuf.insert_at_cursor(&format!(
|
|
|
|
" codec: {} \n",
|
|
|
|
codec.get().unwrap()
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(codec) = tags.get::<gst::tags::AudioCodec>() {
|
|
|
|
textbuf.insert_at_cursor(&format!(
|
|
|
|
" codec: {} \n",
|
|
|
|
codec.get().unwrap()
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(lang) = tags.get::<gst::tags::LanguageCode>() {
|
|
|
|
textbuf.insert_at_cursor(&format!(
|
|
|
|
" language: {} \n",
|
|
|
|
lang.get().unwrap()
|
|
|
|
));
|
|
|
|
}
|
|
|
|
|
|
|
|
if let Some(bitrate) = tags.get::<gst::tags::Bitrate>() {
|
|
|
|
textbuf.insert_at_cursor(&format!(
|
|
|
|
" bitrate: {} \n",
|
|
|
|
bitrate.get().unwrap()
|
|
|
|
));
|
|
|
|
}
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
}
|
2018-10-08 12:02:23 +00:00
|
|
|
}
|
2019-08-11 07:33:34 +00:00
|
|
|
_ => {
|
2017-10-11 09:15:34 +00:00
|
|
|
eprintln!("Could not get {}!", propname);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Extract metadata from all the streams and write it to the text widget in the GUI
|
2018-07-27 10:25:47 +00:00
|
|
|
fn analyze_streams(playbin: &gst::Element, textbuf: >k::TextBuffer) {
|
2017-10-11 09:15:34 +00:00
|
|
|
{
|
2017-10-26 10:56:52 +00:00
|
|
|
textbuf.set_text("");
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
2018-07-27 10:25:47 +00:00
|
|
|
add_streams_info(playbin, textbuf, "video");
|
|
|
|
add_streams_info(playbin, textbuf, "audio");
|
|
|
|
add_streams_info(playbin, textbuf, "text");
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This creates all the GTK+ widgets that compose our application, and registers the callbacks
|
2018-07-27 10:07:24 +00:00
|
|
|
fn create_ui(playbin: &gst::Element) -> AppWindow {
|
2020-10-10 08:00:48 +00:00
|
|
|
let main_window = gtk::Window::new(gtk::WindowType::Toplevel);
|
2017-10-11 09:15:34 +00:00
|
|
|
main_window.connect_delete_event(|_, _| {
|
|
|
|
gtk::main_quit();
|
|
|
|
Inhibit(false)
|
|
|
|
});
|
|
|
|
|
2020-06-16 08:45:12 +00:00
|
|
|
let play_button =
|
|
|
|
gtk::Button::from_icon_name(Some("media-playback-start"), gtk::IconSize::SmallToolbar);
|
2017-10-11 09:15:34 +00:00
|
|
|
let pipeline = playbin.clone();
|
|
|
|
play_button.connect_clicked(move |_| {
|
|
|
|
let pipeline = &pipeline;
|
2017-11-05 17:58:44 +00:00
|
|
|
pipeline
|
|
|
|
.set_state(gst::State::Playing)
|
2019-01-08 16:13:37 +00:00
|
|
|
.expect("Unable to set the pipeline to the `Playing` state");
|
2017-10-11 09:15:34 +00:00
|
|
|
});
|
|
|
|
|
2020-06-16 08:45:12 +00:00
|
|
|
let pause_button =
|
|
|
|
gtk::Button::from_icon_name(Some("media-playback-pause"), gtk::IconSize::SmallToolbar);
|
2017-10-11 09:15:34 +00:00
|
|
|
let pipeline = playbin.clone();
|
|
|
|
pause_button.connect_clicked(move |_| {
|
|
|
|
let pipeline = &pipeline;
|
2017-11-05 17:58:44 +00:00
|
|
|
pipeline
|
|
|
|
.set_state(gst::State::Paused)
|
2019-01-08 16:13:37 +00:00
|
|
|
.expect("Unable to set the pipeline to the `Paused` state");
|
2017-10-11 09:15:34 +00:00
|
|
|
});
|
|
|
|
|
2020-06-16 08:45:12 +00:00
|
|
|
let stop_button =
|
|
|
|
gtk::Button::from_icon_name(Some("media-playback-stop"), gtk::IconSize::SmallToolbar);
|
2017-10-11 09:15:34 +00:00
|
|
|
let pipeline = playbin.clone();
|
|
|
|
stop_button.connect_clicked(move |_| {
|
|
|
|
let pipeline = &pipeline;
|
2019-01-08 16:13:37 +00:00
|
|
|
pipeline
|
|
|
|
.set_state(gst::State::Ready)
|
|
|
|
.expect("Unable to set the pipeline to the `Ready` state");
|
2017-10-11 09:15:34 +00:00
|
|
|
});
|
|
|
|
|
2020-06-16 08:45:12 +00:00
|
|
|
let slider = gtk::Scale::with_range(
|
2017-10-11 09:15:34 +00:00
|
|
|
gtk::Orientation::Horizontal,
|
|
|
|
0.0 as f64,
|
|
|
|
100.0 as f64,
|
|
|
|
1.0 as f64,
|
|
|
|
);
|
|
|
|
let pipeline = playbin.clone();
|
2017-10-17 09:06:51 +00:00
|
|
|
let slider_update_signal_id = slider.connect_value_changed(move |slider| {
|
|
|
|
let pipeline = &pipeline;
|
|
|
|
let value = slider.get_value() as u64;
|
2019-02-28 08:54:32 +00:00
|
|
|
if pipeline
|
|
|
|
.seek_simple(
|
|
|
|
gst::SeekFlags::FLUSH | gst::SeekFlags::KEY_UNIT,
|
|
|
|
value * gst::SECOND,
|
|
|
|
)
|
|
|
|
.is_err()
|
|
|
|
{
|
2017-10-17 09:06:51 +00:00
|
|
|
eprintln!("Seeking to {} failed", value);
|
|
|
|
}
|
|
|
|
});
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
slider.set_draw_value(false);
|
|
|
|
let pipeline = playbin.clone();
|
|
|
|
let lslider = slider.clone();
|
|
|
|
// Update the UI (seekbar) every second
|
2020-08-27 06:49:03 +00:00
|
|
|
let timeout_id = glib::timeout_add_seconds_local(1, move || {
|
2017-10-11 09:15:34 +00:00
|
|
|
let pipeline = &pipeline;
|
|
|
|
let lslider = &lslider;
|
|
|
|
|
2017-12-09 16:20:21 +00:00
|
|
|
if let Some(dur) = pipeline.query_duration::<gst::ClockTime>() {
|
2017-11-11 10:21:55 +00:00
|
|
|
let seconds = dur / gst::SECOND;
|
|
|
|
lslider.set_range(0.0, seconds.map(|v| v as f64).unwrap_or(0.0));
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
2017-12-09 16:20:21 +00:00
|
|
|
if let Some(pos) = pipeline.query_position::<gst::ClockTime>() {
|
2017-11-11 10:21:55 +00:00
|
|
|
let seconds = pos / gst::SECOND;
|
2017-10-11 09:15:34 +00:00
|
|
|
lslider.block_signal(&slider_update_signal_id);
|
2017-11-11 10:21:55 +00:00
|
|
|
lslider.set_value(seconds.map(|v| v as f64).unwrap_or(0.0));
|
2017-10-11 09:15:34 +00:00
|
|
|
lslider.unblock_signal(&slider_update_signal_id);
|
|
|
|
}
|
|
|
|
|
|
|
|
Continue(true)
|
|
|
|
});
|
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
let controls = gtk::Box::new(gtk::Orientation::Horizontal, 0);
|
2017-10-11 09:15:34 +00:00
|
|
|
controls.pack_start(&play_button, false, false, 0);
|
|
|
|
controls.pack_start(&pause_button, false, false, 0);
|
|
|
|
controls.pack_start(&stop_button, false, false, 0);
|
|
|
|
controls.pack_start(&slider, true, true, 2);
|
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
let video_window = gtk::DrawingArea::new();
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
let video_overlay = playbin
|
|
|
|
.clone()
|
|
|
|
.dynamic_cast::<gst_video::VideoOverlay>()
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
video_window.connect_realize(move |video_window| {
|
|
|
|
let video_overlay = &video_overlay;
|
|
|
|
let gdk_window = video_window.get_window().unwrap();
|
|
|
|
|
|
|
|
if !gdk_window.ensure_native() {
|
|
|
|
println!("Can't create native window for widget");
|
|
|
|
process::exit(-1);
|
|
|
|
}
|
|
|
|
|
|
|
|
let display_type_name = gdk_window.get_display().get_type().name();
|
2019-07-05 23:21:50 +00:00
|
|
|
#[cfg(all(target_os = "linux", feature = "tutorial5-x11"))]
|
|
|
|
{
|
2017-10-11 09:15:34 +00:00
|
|
|
// Check if we're using X11 or ...
|
|
|
|
if display_type_name == "GdkX11Display" {
|
|
|
|
extern "C" {
|
|
|
|
pub fn gdk_x11_window_get_xid(
|
|
|
|
window: *mut glib::object::GObject,
|
|
|
|
) -> *mut c_void;
|
|
|
|
}
|
|
|
|
|
2019-02-28 08:32:13 +00:00
|
|
|
#[allow(clippy::cast_ptr_alignment)]
|
2017-10-11 09:15:34 +00:00
|
|
|
unsafe {
|
2019-01-16 11:32:58 +00:00
|
|
|
let xid = gdk_x11_window_get_xid(gdk_window.as_ptr() as *mut _);
|
2017-10-11 09:15:34 +00:00
|
|
|
video_overlay.set_window_handle(xid as usize);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
println!("Add support for display type '{}'", display_type_name);
|
|
|
|
process::exit(-1);
|
|
|
|
}
|
2019-07-05 23:21:50 +00:00
|
|
|
}
|
|
|
|
#[cfg(all(target_os = "macos", feature = "tutorial5-quartz"))]
|
|
|
|
{
|
2017-10-11 09:15:34 +00:00
|
|
|
if display_type_name == "GdkQuartzDisplay" {
|
|
|
|
extern "C" {
|
|
|
|
pub fn gdk_quartz_window_get_nsview(
|
|
|
|
window: *mut glib::object::GObject,
|
|
|
|
) -> *mut c_void;
|
|
|
|
}
|
|
|
|
|
2019-02-28 08:32:13 +00:00
|
|
|
#[allow(clippy::cast_ptr_alignment)]
|
2017-10-11 09:15:34 +00:00
|
|
|
unsafe {
|
2019-01-16 11:32:58 +00:00
|
|
|
let window = gdk_quartz_window_get_nsview(gdk_window.as_ptr() as *mut _);
|
2017-10-11 09:15:34 +00:00
|
|
|
video_overlay.set_window_handle(window as usize);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
println!(
|
|
|
|
"Unsupported display type '{}', compile with `--feature `",
|
|
|
|
display_type_name
|
|
|
|
);
|
|
|
|
process::exit(-1);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
let streams_list = gtk::TextView::new();
|
|
|
|
streams_list.set_editable(false);
|
2018-07-27 10:07:24 +00:00
|
|
|
let pipeline_weak = playbin.downgrade();
|
2018-07-27 10:25:47 +00:00
|
|
|
let streams_list_weak = glib::SendWeakRef::from(streams_list.downgrade());
|
2019-02-28 08:54:32 +00:00
|
|
|
let bus = playbin.get_bus().unwrap();
|
|
|
|
|
|
|
|
#[allow(clippy::single_match)]
|
|
|
|
bus.connect_message(move |_, msg| match msg.view() {
|
|
|
|
gst::MessageView::Application(application) => {
|
|
|
|
let pipeline = match pipeline_weak.upgrade() {
|
|
|
|
Some(pipeline) => pipeline,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
let streams_list = match streams_list_weak.upgrade() {
|
|
|
|
Some(streams_list) => streams_list,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
|
|
|
if application.get_structure().map(|s| s.get_name()) == Some("tags-changed") {
|
|
|
|
let textbuf = streams_list
|
|
|
|
.get_buffer()
|
|
|
|
.expect("Couldn't get buffer from text_view");
|
|
|
|
analyze_streams(&pipeline, &textbuf);
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
2019-02-28 08:54:32 +00:00
|
|
|
}
|
|
|
|
_ => (),
|
|
|
|
});
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
let vbox = gtk::Box::new(gtk::Orientation::Horizontal, 0);
|
2017-10-11 09:15:34 +00:00
|
|
|
vbox.pack_start(&video_window, true, true, 0);
|
|
|
|
vbox.pack_start(&streams_list, false, false, 2);
|
|
|
|
|
2020-10-10 08:00:48 +00:00
|
|
|
let main_box = gtk::Box::new(gtk::Orientation::Vertical, 0);
|
2017-10-11 09:15:34 +00:00
|
|
|
main_box.pack_start(&vbox, true, true, 0);
|
|
|
|
main_box.pack_start(&controls, false, false, 0);
|
|
|
|
main_window.add(&main_box);
|
|
|
|
main_window.set_default_size(640, 480);
|
|
|
|
|
|
|
|
main_window.show_all();
|
2018-07-27 10:07:24 +00:00
|
|
|
|
|
|
|
AppWindow {
|
|
|
|
main_window,
|
|
|
|
timeout_id: Some(timeout_id),
|
|
|
|
}
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// We are possibly in a GStreamer working thread, so we notify the main
|
|
|
|
// thread of this event through a message in the bus
|
|
|
|
fn post_app_message(playbin: &gst::Element) {
|
2020-06-30 20:51:15 +00:00
|
|
|
let _ = playbin.post_message(gst::message::Application::new(gst::Structure::new_empty(
|
2020-06-23 14:56:25 +00:00
|
|
|
"tags-changed",
|
|
|
|
)));
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn run() {
|
|
|
|
// Make sure the right features were activated
|
2019-02-28 08:54:32 +00:00
|
|
|
#[allow(clippy::eq_op)]
|
|
|
|
{
|
|
|
|
if !cfg!(feature = "tutorial5-x11") && !cfg!(feature = "tutorial5-quartz") {
|
|
|
|
eprintln!(
|
|
|
|
"No Gdk backend selected, compile with --features tutorial5[-x11][-quartz]."
|
|
|
|
);
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2019-02-28 08:54:32 +00:00
|
|
|
return;
|
|
|
|
}
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize GTK
|
|
|
|
if let Err(err) = gtk::init() {
|
|
|
|
eprintln!("Failed to initialize GTK: {}", err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Initialize GStreamer
|
|
|
|
if let Err(err) = gst::init() {
|
|
|
|
eprintln!("Failed to initialize Gst: {}", err);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let uri = "https://www.freedesktop.org/software/gstreamer-sdk/\
|
|
|
|
data/media/sintel_trailer-480p.webm";
|
|
|
|
let playbin = gst::ElementFactory::make("playbin", None).unwrap();
|
2018-07-27 10:25:47 +00:00
|
|
|
playbin.set_property("uri", &uri).unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
playbin
|
2018-07-27 10:07:24 +00:00
|
|
|
.connect("video-tags-changed", false, |args| {
|
2019-08-11 07:33:34 +00:00
|
|
|
let pipeline = args[0]
|
|
|
|
.get::<gst::Element>()
|
|
|
|
.expect("playbin \"video-tags-changed\" args[0]")
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
post_app_message(&pipeline);
|
|
|
|
None
|
2018-10-08 12:02:23 +00:00
|
|
|
})
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
playbin
|
2018-07-27 10:07:24 +00:00
|
|
|
.connect("audio-tags-changed", false, |args| {
|
2019-08-11 07:33:34 +00:00
|
|
|
let pipeline = args[0]
|
|
|
|
.get::<gst::Element>()
|
|
|
|
.expect("playbin \"audio-tags-changed\" args[0]")
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
post_app_message(&pipeline);
|
|
|
|
None
|
2018-10-08 12:02:23 +00:00
|
|
|
})
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
playbin
|
2018-07-27 10:07:24 +00:00
|
|
|
.connect("text-tags-changed", false, move |args| {
|
2019-08-11 07:33:34 +00:00
|
|
|
let pipeline = args[0]
|
|
|
|
.get::<gst::Element>()
|
|
|
|
.expect("playbin \"text-tags-changed\" args[0]")
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
post_app_message(&pipeline);
|
|
|
|
None
|
2018-10-08 12:02:23 +00:00
|
|
|
})
|
|
|
|
.unwrap();
|
2017-10-11 09:15:34 +00:00
|
|
|
|
2018-07-27 10:07:24 +00:00
|
|
|
let window = create_ui(&playbin);
|
2017-10-11 09:15:34 +00:00
|
|
|
|
|
|
|
let bus = playbin.get_bus().unwrap();
|
|
|
|
bus.add_signal_watch();
|
2018-07-27 10:07:24 +00:00
|
|
|
|
|
|
|
let pipeline_weak = playbin.downgrade();
|
|
|
|
bus.connect_message(move |_, msg| {
|
|
|
|
let pipeline = match pipeline_weak.upgrade() {
|
|
|
|
Some(pipeline) => pipeline,
|
|
|
|
None => return,
|
|
|
|
};
|
|
|
|
|
2017-10-11 09:15:34 +00:00
|
|
|
match msg.view() {
|
|
|
|
// This is called when an End-Of-Stream message is posted on the bus.
|
|
|
|
// We just set the pipeline to READY (which stops playback).
|
|
|
|
gst::MessageView::Eos(..) => {
|
|
|
|
println!("End-Of-Stream reached.");
|
2019-01-08 16:13:37 +00:00
|
|
|
pipeline
|
|
|
|
.set_state(gst::State::Ready)
|
|
|
|
.expect("Unable to set the pipeline to the `Ready` state");
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// This is called when an error message is posted on the bus
|
|
|
|
gst::MessageView::Error(err) => {
|
|
|
|
println!(
|
2017-11-16 11:58:56 +00:00
|
|
|
"Error from {:?}: {} ({:?})",
|
2018-01-29 14:26:01 +00:00
|
|
|
err.get_src().map(|s| s.get_path_string()),
|
2017-10-11 09:15:34 +00:00
|
|
|
err.get_error(),
|
|
|
|
err.get_debug()
|
|
|
|
);
|
|
|
|
}
|
|
|
|
// This is called when the pipeline changes states. We use it to
|
|
|
|
// keep track of the current state.
|
2018-01-29 14:26:01 +00:00
|
|
|
gst::MessageView::StateChanged(state_changed) => {
|
2018-02-22 10:18:37 +00:00
|
|
|
if state_changed
|
|
|
|
.get_src()
|
|
|
|
.map(|s| s == pipeline)
|
|
|
|
.unwrap_or(false)
|
|
|
|
{
|
2018-01-29 14:26:01 +00:00
|
|
|
println!("State set to {:?}", state_changed.get_current());
|
2017-11-16 11:58:56 +00:00
|
|
|
}
|
|
|
|
}
|
2017-10-11 09:15:34 +00:00
|
|
|
_ => (),
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2018-07-27 10:07:24 +00:00
|
|
|
playbin
|
|
|
|
.set_state(gst::State::Playing)
|
2019-01-08 16:13:37 +00:00
|
|
|
.expect("Unable to set the playbin to the `Playing` state");
|
2018-07-27 10:07:24 +00:00
|
|
|
|
2017-10-11 09:15:34 +00:00
|
|
|
gtk::main();
|
2018-07-27 10:07:24 +00:00
|
|
|
window.hide();
|
2019-01-08 16:13:37 +00:00
|
|
|
playbin
|
|
|
|
.set_state(gst::State::Null)
|
|
|
|
.expect("Unable to set the playbin to the `Null` state");
|
2018-07-27 10:07:24 +00:00
|
|
|
|
|
|
|
bus.remove_signal_watch();
|
2017-10-11 09:15:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(feature = "tutorial5")]
|
|
|
|
fn main() {
|
|
|
|
tutorial5::run();
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(not(feature = "tutorial5"))]
|
|
|
|
fn main() {
|
|
|
|
println!("Please compile with --features tutorial5[-x11][-quartz]");
|
|
|
|
}
|