graph: Move link data into new GObject subclass

This commit is contained in:
Tom A. Wagner
2023-07-19 15:06:51 +02:00
parent 475a83fab7
commit f986902929
6 changed files with 147 additions and 26 deletions

View File

@@ -25,10 +25,7 @@ use gtk::{
use log::info;
use pipewire::{channel::Sender, spa::Direction};
use crate::{
ui,
GtkMessage, MediaType, NodeType, PipewireLink, PipewireMessage,
};
use crate::{ui, GtkMessage, MediaType, NodeType, PipewireLink, PipewireMessage};
static STYLE: &str = include_str!("style.css");

View File

@@ -14,8 +14,6 @@
//
// SPDX-License-Identifier: GPL-3.0-only
use super::{Node, Port};
use gtk::{
glib::{self, clone},
graphene,
@@ -28,6 +26,7 @@ use log::{error, warn};
use std::{cmp::Ordering, collections::HashMap};
use super::{Link, Node, Port};
use crate::NodeType;
const CANVAS_SIZE: f64 = 5000.0;
@@ -59,7 +58,7 @@ mod imp {
/// Stores nodes and their positions.
pub(super) nodes: RefCell<HashMap<u32, (Node, Point)>>,
/// Stores the link and whether it is currently active.
pub(super) links: RefCell<HashMap<u32, (crate::PipewireLink, bool)>>,
pub(super) links: RefCell<HashMap<u32, Link>>,
pub hadjustment: RefCell<Option<gtk::Adjustment>>,
pub vadjustment: RefCell<Option<gtk::Adjustment>>,
pub zoom_factor: Cell<f64>,
@@ -431,13 +430,13 @@ mod imp {
rgba.alpha().into(),
);
for (link, active) in self.links.borrow().values() {
for link in self.links.borrow().values() {
// TODO: Do not draw links when they are outside the view
if let Some((from_x, from_y, to_x, to_y)) = self.get_link_coordinates(link) {
link_cr.move_to(from_x, from_y);
// Use dashed line for inactive links, full line otherwise.
if *active {
if link.active() {
link_cr.set_dash(&[], 0.0);
} else {
link_cr.set_dash(&[10.0, 5.0], 0.0);
@@ -478,11 +477,10 @@ mod imp {
///
/// # Returns
/// `Some((from_x, from_y, to_x, to_y))` if all objects the links refers to exist as widgets.
fn get_link_coordinates(&self, link: &crate::PipewireLink) -> Option<(f64, f64, f64, f64)> {
fn get_link_coordinates(&self, link: &Link) -> Option<(f64, f64, f64, f64)> {
let widget = &*self.obj();
let nodes = self.nodes.borrow();
let output_port = &nodes.get(&link.node_from)?.0.get_port(link.port_from)?;
let output_port = link.output_port()?;
let output_port_padding =
(output_port.allocated_width() - output_port.width()) as f64 / 2.0;
@@ -493,7 +491,7 @@ mod imp {
(output_port.height() / 2) as f64,
)?;
let input_port = &nodes.get(&link.node_to)?.0.get_port(link.port_to)?;
let input_port = link.input_port()?;
let input_port_padding =
(input_port.allocated_width() - input_port.width()) as f64 / 2.0;
@@ -671,16 +669,28 @@ impl GraphView {
}
pub fn add_link(&self, link_id: u32, link: crate::PipewireLink, active: bool) {
self.imp()
.links
.borrow_mut()
.insert(link_id, (link, active));
let nodes = self.imp().nodes.borrow();
let output_port = nodes
.get(&link.node_from)
.and_then(|(node, _)| node.get_port(link.port_from));
let input_port = nodes
.get(&link.node_to)
.and_then(|(node, _)| node.get_port(link.port_to));
let link = Link::new();
link.set_input_port(input_port.as_ref());
link.set_output_port(output_port.as_ref());
link.set_active(active);
self.imp().links.borrow_mut().insert(link_id, link);
self.queue_draw();
}
pub fn set_link_state(&self, link_id: u32, active: bool) {
if let Some((_, state)) = self.imp().links.borrow_mut().get_mut(&link_id) {
*state = active;
if let Some(link) = self.imp().links.borrow_mut().get_mut(&link_id) {
link.set_active(active);
self.queue_draw();
} else {
warn!("Link state changed on unknown link (id={})", link_id);

114
src/ui/graph/link.rs Normal file
View File

@@ -0,0 +1,114 @@
// Copyright 2021 Tom A. Wagner <tom.a.wagner@protonmail.com>
//
// This program is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License version 3 as published by
// the Free Software Foundation.
//
// This program is distributed in the hope that it will be useful,
// but WITHOUT ANY WARRANTY; without even the implied warranty of
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
// GNU General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with this program. If not, see <http://www.gnu.org/licenses/>.
//
// SPDX-License-Identifier: GPL-3.0-only
use gtk::{glib, prelude::*, subclass::prelude::*};
use super::Port;
mod imp {
use super::*;
use std::cell::Cell;
use once_cell::sync::Lazy;
#[derive(Default)]
pub struct Link {
pub output_port: glib::WeakRef<Port>,
pub input_port: glib::WeakRef<Port>,
pub active: Cell<bool>,
}
#[glib::object_subclass]
impl ObjectSubclass for Link {
const NAME: &'static str = "HelvumLink";
type Type = super::Link;
type ParentType = glib::Object;
}
impl ObjectImpl for Link {
fn properties() -> &'static [glib::ParamSpec] {
static PROPERTIES: Lazy<Vec<glib::ParamSpec>> = Lazy::new(|| {
vec![
glib::ParamSpecObject::builder::<Port>("output-port")
.flags(glib::ParamFlags::READWRITE)
.build(),
glib::ParamSpecObject::builder::<Port>("input-port")
.flags(glib::ParamFlags::READWRITE)
.build(),
glib::ParamSpecBoolean::builder("active")
.default_value(false)
.flags(glib::ParamFlags::READWRITE)
.build(),
]
});
PROPERTIES.as_ref()
}
fn property(&self, _id: usize, pspec: &glib::ParamSpec) -> glib::Value {
match pspec.name() {
"output-port" => self.output_port.upgrade().to_value(),
"input-port" => self.input_port.upgrade().to_value(),
"active" => self.active.get().to_value(),
_ => unimplemented!(),
}
}
fn set_property(&self, _id: usize, value: &glib::Value, pspec: &glib::ParamSpec) {
match pspec.name() {
"output-port" => self.output_port.set(value.get().unwrap()),
"input-port" => self.input_port.set(value.get().unwrap()),
"active" => self.active.set(value.get().unwrap()),
_ => unimplemented!(),
}
}
}
}
glib::wrapper! {
pub struct Link(ObjectSubclass<imp::Link>);
}
impl Link {
pub fn new() -> Self {
glib::Object::new()
}
pub fn output_port(&self) -> Option<Port> {
self.property("output-port")
}
pub fn set_output_port(&self, port: Option<&Port>) {
self.set_property("output-port", port);
}
pub fn input_port(&self) -> Option<Port> {
self.property("input-port")
}
pub fn set_input_port(&self, port: Option<&Port>) {
self.set_property("input-port", port);
}
pub fn active(&self) -> bool {
self.property("active")
}
pub fn set_active(&self, active: bool) {
self.set_property("active", active);
}
}

View File

@@ -20,5 +20,7 @@ mod node;
pub use node::*;
mod port;
pub use port::*;
mod link;
pub use link::*;
mod zoomentry;
pub use zoomentry::*;
pub use zoomentry::*;

View File

@@ -109,11 +109,9 @@ mod imp {
fn properties() -> &'static [glib::ParamSpec] {
static PROPERTIES: Lazy<Vec<glib::ParamSpec>> = Lazy::new(|| {
vec![
glib::ParamSpecObject::builder::<GraphView>("zoomed-widget")
.flags(glib::ParamFlags::READWRITE | glib::ParamFlags::CONSTRUCT)
.build(),
]
vec![glib::ParamSpecObject::builder::<GraphView>("zoomed-widget")
.flags(glib::ParamFlags::READWRITE | glib::ParamFlags::CONSTRUCT)
.build()]
});
PROPERTIES.as_ref()

View File

@@ -18,4 +18,4 @@
//!
//! This module contains gtk widgets needed to present the graphical user interface.
pub mod graph;
pub mod graph;