public inbox for pdm-devel@lists.proxmox.com
 help / color / mirror / Atom feed
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


  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
Service provided by Proxmox Server Solutions GmbH | Privacy | Legal