Closes: #157 XDG portals expose a D-Bus interface allowing apps to prevent screen idling, user switching, and other actions. That interface, org.freedesktop.portal.Inhibit, does all of the heavy lifting here. Idling = the screen dimming or shutting off. Idling is inhibited when a video is actively playing. Idling is NOT inhibited when videos aren't playing - this includes paused or stopped videos.
79 lines
2.9 KiB
Rust
79 lines
2.9 KiB
Rust
use iced_video_player::{
|
|
Video,
|
|
gst::{self, prelude::*},
|
|
gst_app, gst_pbutils,
|
|
};
|
|
|
|
use cosmic::app::{Command, message};
|
|
|
|
pub fn new_video(
|
|
url: &url::Url,
|
|
) -> Result<Video, cosmic::Command<cosmic::app::Message<super::Message>>> {
|
|
//TODO: this code came from iced_video_player::Video::new and has been modified to stop the pipeline on error
|
|
//TODO: remove unwraps and enable playback of files with only audio.
|
|
gst::init().unwrap();
|
|
|
|
let pipeline = format!(
|
|
"playbin uri=\"{}\" video-sink=\"videoscale ! videoconvert ! videoflip method=automatic ! appsink name=iced_video drop=true caps=video/x-raw,format=NV12,pixel-aspect-ratio=1/1\"",
|
|
url.as_str()
|
|
);
|
|
let pipeline = gst::parse::launch(pipeline.as_ref())
|
|
.unwrap()
|
|
.downcast::<gst::Pipeline>()
|
|
.map_err(|_| iced_video_player::Error::Cast)
|
|
.unwrap();
|
|
pipeline.connect("element-setup", false, |vals| {
|
|
let Ok(elem) = vals[1].get::<gst::Element>() else {
|
|
return None;
|
|
};
|
|
if let Some(factory) = elem.factory() {
|
|
if factory.name() == "souphttpsrc" {
|
|
elem.set_property(
|
|
"user-agent",
|
|
"Mozilla/5.0 (X11; Linux x86_64; rv:142.0) Gecko/20100101 Firefox/142.0",
|
|
);
|
|
}
|
|
}
|
|
None
|
|
});
|
|
let video_sink: gst::Element = pipeline.property("video-sink");
|
|
let pad = video_sink.pads().first().cloned().unwrap();
|
|
let pad = pad.dynamic_cast::<gst::GhostPad>().unwrap();
|
|
let bin = pad
|
|
.parent_element()
|
|
.unwrap()
|
|
.downcast::<gst::Bin>()
|
|
.unwrap();
|
|
let video_sink = bin.by_name("iced_video").unwrap();
|
|
let video_sink = video_sink.downcast::<gst_app::AppSink>().unwrap();
|
|
|
|
match Video::from_gst_pipeline(pipeline.clone(), video_sink, None) {
|
|
Ok(ok) => Ok(ok),
|
|
Err(err) => {
|
|
log::warn!("failed to open {}: {err}", url);
|
|
// Handle codecs required before the file can play
|
|
let mut commands = Vec::new();
|
|
while let Some(msg) = pipeline
|
|
.bus()
|
|
.unwrap()
|
|
.pop_filtered(&[gst::MessageType::Element])
|
|
{
|
|
match msg.view() {
|
|
gst::MessageView::Element(element) => {
|
|
if gst_pbutils::MissingPluginMessage::is(&element) {
|
|
commands.push(Command::perform(
|
|
async { message::app(super::Message::MissingPlugin(msg)) },
|
|
|x| x,
|
|
));
|
|
// Do one codec install at a time
|
|
break;
|
|
}
|
|
}
|
|
_ => {}
|
|
}
|
|
}
|
|
pipeline.set_state(gst::State::Null).unwrap();
|
|
Err(Command::batch(commands))
|
|
}
|
|
}
|
|
}
|