From: "Shannon Sterz" <s.sterz@proxmox.com>
To: "Dominik Csapak" <d.csapak@proxmox.com>
Cc: Proxmox Datacenter Manager development discussion
<pdm-devel@lists.proxmox.com>
Subject: Re: [pdm-devel] [PATCH datacenter-manager v2 01/16] ui: dashboard: refactor guest panel creation to its own module
Date: Thu, 23 Oct 2025 13:19:00 +0200 [thread overview]
Message-ID: <DDPNSO3DHNU1.2TDKSXFLY8NO8@proxmox.com> (raw)
In-Reply-To: <20251023083253.1038119-2-d.csapak@proxmox.com>
On Thu Oct 23, 2025 at 10:28 AM CEST, Dominik Csapak wrote:
> so we can more easily reuse it. For this, also make the 'create_title_with_icon'
> a freestanding function that is public so we can reuse it outside the
> dashboard struct.
>
> Signed-off-by: Dominik Csapak <d.csapak@proxmox.com>
> ---
> ui/src/dashboard/guest_panel.rs | 75 +++++++++++++++++++++++++--------
> ui/src/dashboard/mod.rs | 66 ++++++++++++-----------------
> 2 files changed, 84 insertions(+), 57 deletions(-)
>
> diff --git a/ui/src/dashboard/guest_panel.rs b/ui/src/dashboard/guest_panel.rs
> index 814ecfa5..3197feb4 100644
> --- a/ui/src/dashboard/guest_panel.rs
> +++ b/ui/src/dashboard/guest_panel.rs
> @@ -1,30 +1,32 @@
> use std::rc::Rc;
>
> -use pdm_api_types::resource::{GuestStatusCount, ResourceType};
> +use pdm_api_types::resource::{GuestStatusCount, ResourceType, ResourcesStatus};
> use pdm_search::{Search, SearchTerm};
> use proxmox_yew_comp::GuestState;
> use pwt::{
> css::{self, TextAlign},
> prelude::*,
> - widget::{Container, Fa, List, ListTile},
> + widget::{Container, Fa, List, ListTile, Panel},
> };
> use yew::{
> virtual_dom::{VComp, VNode},
> Properties,
> };
>
> -use crate::{pve::GuestType, search_provider::get_search_provider};
> +use crate::{
> + dashboard::create_title_with_icon, pve::GuestType, search_provider::get_search_provider,
> +};
>
> use super::loading_column;
>
> #[derive(PartialEq, Clone, Properties)]
> pub struct GuestPanel {
> - guest_type: GuestType,
> - status: Option<GuestStatusCount>,
> + guest_type: Option<GuestType>,
> + status: Option<ResourcesStatus>,
> }
>
> impl GuestPanel {
> - pub fn new(guest_type: GuestType, status: Option<GuestStatusCount>) -> Self {
> + pub fn new(guest_type: Option<GuestType>, status: Option<ResourcesStatus>) -> Self {
> yew::props!(Self { guest_type, status })
would be nice to have a comment here that explains that passing
`guest_type` as `None` will render a panel that includes both guest
types.
> }
> }
> @@ -63,7 +65,16 @@ impl yew::Component for PdmGuestPanel {
> let props = ctx.props();
> let guest_type = props.guest_type;
> let status = match &props.status {
> - Some(status) => status,
> + Some(status) => match guest_type {
> + Some(GuestType::Qemu) => status.qemu.clone(),
> + Some(GuestType::Lxc) => status.lxc.clone(),
> + None => GuestStatusCount {
> + running: status.qemu.running + status.lxc.running,
> + stopped: status.qemu.stopped + status.lxc.stopped,
> + template: status.qemu.template + status.lxc.template,
> + unknown: status.qemu.unknown + status.lxc.unknown,
> + },
> + },
> None => return loading_column().into(),
> };
>
> @@ -93,7 +104,7 @@ impl yew::Component for PdmGuestPanel {
>
> fn create_list_tile(
> link: &html::Scope<PdmGuestPanel>,
> - guest_type: GuestType,
> + guest_type: Option<GuestType>,
> status_row: StatusRow,
> ) -> Option<ListTile> {
> let (icon, text, count, status, template) = match status_row {
> @@ -129,7 +140,13 @@ fn create_list_tile(
> None,
> ),
> },
> - StatusRow::All(count) => (Fa::from(guest_type), tr!("All"), count, None, None),
> + StatusRow::All(count) => (
> + Fa::from(guest_type.unwrap_or(GuestType::Qemu)),
> + tr!("All"),
> + count,
> + None,
> + None,
> + ),
> };
>
> Some(
> @@ -158,19 +175,29 @@ fn create_list_tile(
> }
>
> fn create_guest_search_term(
> - guest_type: GuestType,
> + guest_type: Option<GuestType>,
> status: Option<&'static str>,
> template: Option<bool>,
> ) -> Search {
> - let resource_type: ResourceType = guest_type.into();
> - if status.is_none() && template.is_none() {
> - return Search::with_terms(vec![
> - SearchTerm::new(resource_type.as_str()).category(Some("type"))
> - ]);
> + let mut terms = Vec::new();
> + match guest_type {
> + Some(guest_type) => {
> + let resource_type: ResourceType = guest_type.into();
> + terms.push(SearchTerm::new(resource_type.as_str()).category(Some("type")));
> + }
> + None => {
> + terms.push(
> + SearchTerm::new(ResourceType::PveQemu.as_str())
> + .category(Some("type"))
> + .optional(true),
> + );
> + terms.push(
> + SearchTerm::new(ResourceType::PveLxc.as_str())
> + .category(Some("type"))
> + .optional(true),
> + );
> + }
> }
> -
> - let mut terms = vec![SearchTerm::new(resource_type.as_str()).category(Some("type"))];
> -
> if let Some(template) = template {
> terms.push(SearchTerm::new(template.to_string()).category(Some("template")));
> }
> @@ -179,3 +206,15 @@ fn create_guest_search_term(
> }
> Search::with_terms(terms)
> }
> +
> +pub fn create_guest_panel(guest_type: Option<GuestType>, status: Option<ResourcesStatus>) -> Panel {
nit: a doc comment explaining what the parameters do would be nice on a
public function
> + let (icon, title) = match guest_type {
> + Some(GuestType::Qemu) => ("desktop", tr!("Virtual Machines")),
> + Some(GuestType::Lxc) => ("cubes", tr!("Linux Container")),
> + None => ("desktop", tr!("Guests")),
> + };
> + Panel::new()
> + .title(create_title_with_icon(icon, title))
> + .border(true)
> + .with_child(GuestPanel::new(guest_type, status))
> +}
> diff --git a/ui/src/dashboard/mod.rs b/ui/src/dashboard/mod.rs
> index f54c509c..602c8a3b 100644
> --- a/ui/src/dashboard/mod.rs
> +++ b/ui/src/dashboard/mod.rs
> @@ -46,7 +46,7 @@ mod remote_panel;
> use remote_panel::RemotePanel;
>
> mod guest_panel;
> -use guest_panel::GuestPanel;
> +pub use guest_panel::create_guest_panel;
>
> mod sdn_zone_panel;
> use sdn_zone_panel::SdnZonePanel;
> @@ -149,15 +149,6 @@ pub struct PdmDashboard {
> }
>
> impl PdmDashboard {
> - fn create_title_with_icon(&self, icon: &str, title: String) -> Html {
> - Row::new()
> - .class(AlignItems::Center)
> - .gap(2)
> - .with_child(Fa::new(icon))
> - .with_child(title)
> - .into()
> - }
> -
> fn create_node_panel(&self, ctx: &yew::Context<Self>, icon: &str, title: String) -> Panel {
> let mut search_terms = vec![SearchTerm::new("node").category(Some("type"))];
> let (status_icon, text): (Fa, String) = match &self.status {
> @@ -203,7 +194,7 @@ impl PdmDashboard {
> Panel::new()
> .flex(1.0)
> .width(300)
> - .title(self.create_title_with_icon(icon, title))
> + .title(create_title_with_icon(icon, title))
> .border(true)
> .with_child(
> Column::new()
> @@ -233,34 +224,13 @@ impl PdmDashboard {
> )
> }
>
> - fn create_guest_panel(&self, guest_type: GuestType) -> Panel {
> - let (icon, title, status) = match guest_type {
> - GuestType::Qemu => (
> - "desktop",
> - tr!("Virtual Machines"),
> - self.status.as_ref().map(|s| s.qemu.clone()),
> - ),
> - GuestType::Lxc => (
> - "cubes",
> - tr!("Linux Container"),
> - self.status.as_ref().map(|s| s.lxc.clone()),
> - ),
> - };
> - Panel::new()
> - .flex(1.0)
> - .width(300)
> - .title(self.create_title_with_icon(icon, title))
> - .border(true)
> - .with_child(GuestPanel::new(guest_type, status))
> - }
> -
> fn create_sdn_panel(&self) -> Panel {
> let sdn_zones_status = self.status.as_ref().map(|status| status.sdn_zones.clone());
>
> Panel::new()
> .flex(1.0)
> .width(200)
> - .title(self.create_title_with_icon("sdn", tr!("SDN Zones")))
> + .title(create_title_with_icon("sdn", tr!("SDN Zones")))
> .border(true)
> .with_child(SdnZonePanel::new(
> (!self.loading).then_some(sdn_zones_status).flatten(),
> @@ -281,7 +251,7 @@ impl PdmDashboard {
> .flex(1.0)
> .width(500)
> .border(true)
> - .title(self.create_title_with_icon("list", title))
> + .title(create_title_with_icon("list", title))
> .with_child(
> Container::new()
> .class(FlexFit)
> @@ -318,7 +288,7 @@ impl PdmDashboard {
> .width(500)
> .min_width(400)
> .border(true)
> - .title(self.create_title_with_icon(icon, title))
> + .title(create_title_with_icon(icon, title))
> .with_optional_child(
> entities
> .map(|entities| TopEntities::new(entities.clone(), metrics_title, threshold)),
> @@ -537,7 +507,7 @@ impl Component for PdmDashboard {
> .padding_top(0)
> .with_child(
> Panel::new()
> - .title(self.create_title_with_icon("server", tr!("Remotes")))
> + .title(create_title_with_icon("server", tr!("Remotes")))
> .flex(1.0)
> //.border(true)
> .width(300)
> @@ -568,8 +538,16 @@ impl Component for PdmDashboard {
> "building",
> tr!("Virtual Environment Nodes"),
> ))
> - .with_child(self.create_guest_panel(GuestType::Qemu))
> - .with_child(self.create_guest_panel(GuestType::Lxc))
> + .with_child(
> + create_guest_panel(Some(GuestType::Qemu), self.status.clone())
> + .flex(1.0)
> + .width(300),
> + )
> + .with_child(
> + create_guest_panel(Some(GuestType::Lxc), self.status.clone())
> + .flex(1.0)
> + .width(300),
> + )
> // FIXME: add PBS support
> //.with_child(self.create_node_panel(
> // "building-o",
> @@ -580,7 +558,7 @@ impl Component for PdmDashboard {
> // Panel::new()
> // .flex(1.0)
> // .width(300)
> - // .title(self.create_title_with_icon(
> + // .title(create_title_with_icon(
> // "floppy-o",
> // tr!("Backup Server Datastores"),
> // ))
> @@ -777,3 +755,13 @@ fn loading_column() -> Column {
> .class(AlignItems::Center)
> .with_child(html! {<i class={"pwt-loading-icon"} />})
> }
> +
> +/// Create a consistent title component for the given title and icon
> +pub fn create_title_with_icon(icon: &str, title: String) -> Html {
> + Row::new()
> + .class(AlignItems::Center)
> + .gap(2)
> + .with_child(Fa::new(icon))
> + .with_child(title)
> + .into()
> +}
_______________________________________________
pdm-devel mailing list
pdm-devel@lists.proxmox.com
https://lists.proxmox.com/cgi-bin/mailman/listinfo/pdm-devel
next prev parent reply other threads:[~2025-10-23 11:19 UTC|newest]
Thread overview: 28+ messages / expand[flat|nested] mbox.gz Atom feed top
2025-10-23 8:28 [pdm-devel] [PATCH datacenter-manager v2 00/16] prepare ui for customizable views Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 01/16] ui: dashboard: refactor guest panel creation to its own module Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz [this message]
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 02/16] ui: dashboard: refactor creating the node panel into " Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 03/16] ui: dashboard: refactor remote panel creation " Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 04/16] ui: dashboard: remote panel: make wizard menu optional Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 05/16] ui: dashboard: refactor sdn panel creation into its own module Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 06/16] ui: dashboard: refactor task summary panel creation to " Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 07/16] ui: dashboard: task summary: disable virtual scrolling Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 08/16] ui: dashboard: refactor subscription panel creation to its own module Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 09/16] ui: dashboard: refactor top entities " Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 10/16] ui: dashboard: refactor DashboardConfig editing/constants to their module Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 11/16] ui: dashboard: factor out task parameter calculation Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 12/16] ui: dashboard: remove unused remote list Dominik Csapak
2025-10-23 8:28 ` [pdm-devel] [PATCH datacenter-manager v2 13/16] ui: dashboard: status row: make loading less jarring Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 8:28 ` [pdm-devel] [RFC PATCH datacenter-manager v2 14/16] ui: introduce `LoadResult` helper type Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 8:28 ` [pdm-devel] [RFC PATCH datacenter-manager v2 15/16] ui: dashboard: implement 'View' Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 11:44 ` Dominik Csapak
2025-10-23 11:48 ` Dominik Csapak
2025-10-24 10:17 ` Shannon Sterz
2025-10-23 8:28 ` [pdm-devel] [RFC PATCH datacenter-manager v2 16/16] ui: dashboard: use 'View' instead of the Dashboard Dominik Csapak
2025-10-23 11:19 ` Shannon Sterz
2025-10-23 11:20 ` [pdm-devel] [PATCH datacenter-manager v2 00/16] prepare ui for customizable views Shannon Sterz
Reply instructions:
You may reply publicly to this message via plain-text email
using any one of the following methods:
* Save the following mbox file, import it into your mail client,
and reply-to-all from there: mbox
Avoid top-posting and favor interleaved quoting:
https://en.wikipedia.org/wiki/Posting_style#Interleaved_style
* Reply using the --to, --cc, and --in-reply-to
switches of git-send-email(1):
git send-email \
--in-reply-to=DDPNSO3DHNU1.2TDKSXFLY8NO8@proxmox.com \
--to=s.sterz@proxmox.com \
--cc=d.csapak@proxmox.com \
--cc=pdm-devel@lists.proxmox.com \
/path/to/YOUR_REPLY
https://kernel.org/pub/software/scm/git/docs/git-send-email.html
* If your mail client supports setting the In-Reply-To header
via mailto: links, try the mailto: link
Be sure your reply has a Subject: header at the top and a blank line
before the message body.
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox