Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 777bf5fd53 | |||
| 06cd411089 |
6
Cargo.lock
generated
6
Cargo.lock
generated
@@ -279,7 +279,7 @@ checksum = "a1d728cc89cf3aee9ff92b05e62b19ee65a02b5702cff7d5a377e32c6ae29d8d"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cm-dashboard"
|
name = "cm-dashboard"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"chrono",
|
"chrono",
|
||||||
@@ -301,7 +301,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cm-dashboard-agent"
|
name = "cm-dashboard-agent"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@@ -325,7 +325,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "cm-dashboard-shared"
|
name = "cm-dashboard-shared"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"chrono",
|
"chrono",
|
||||||
"serde",
|
"serde",
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cm-dashboard-agent"
|
name = "cm-dashboard-agent"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cm-dashboard"
|
name = "cm-dashboard"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|||||||
@@ -24,6 +24,7 @@ pub struct Dashboard {
|
|||||||
config: DashboardConfig,
|
config: DashboardConfig,
|
||||||
system_area: Rect, // Store system area for mouse event handling
|
system_area: Rect, // Store system area for mouse event handling
|
||||||
services_area: Rect, // Store services area for mouse event handling
|
services_area: Rect, // Store services area for mouse event handling
|
||||||
|
hosts_area: Rect, // Store hosts area for mouse event handling
|
||||||
}
|
}
|
||||||
|
|
||||||
impl Dashboard {
|
impl Dashboard {
|
||||||
@@ -125,6 +126,7 @@ impl Dashboard {
|
|||||||
config,
|
config,
|
||||||
system_area: Rect::default(),
|
system_area: Rect::default(),
|
||||||
services_area: Rect::default(),
|
services_area: Rect::default(),
|
||||||
|
hosts_area: Rect::default(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -272,9 +274,10 @@ impl Dashboard {
|
|||||||
|
|
||||||
// Render TUI regardless of terminal size
|
// Render TUI regardless of terminal size
|
||||||
if let Err(e) = terminal.draw(|frame| {
|
if let Err(e) = terminal.draw(|frame| {
|
||||||
let (_title_area, system_area, services_area) = tui_app.render(frame, &self.metric_store);
|
let (_title_area, system_area, services_area, hosts_area) = tui_app.render(frame, &self.metric_store);
|
||||||
self.system_area = system_area;
|
self.system_area = system_area;
|
||||||
self.services_area = services_area;
|
self.services_area = services_area;
|
||||||
|
self.hosts_area = hosts_area;
|
||||||
}) {
|
}) {
|
||||||
error!("Error rendering TUI: {}", e);
|
error!("Error rendering TUI: {}", e);
|
||||||
break;
|
break;
|
||||||
@@ -381,8 +384,9 @@ impl Dashboard {
|
|||||||
// Determine which panel the mouse is over
|
// Determine which panel the mouse is over
|
||||||
let in_system_area = is_in_area(x, y, &self.system_area);
|
let in_system_area = is_in_area(x, y, &self.system_area);
|
||||||
let in_services_area = is_in_area(x, y, &self.services_area);
|
let in_services_area = is_in_area(x, y, &self.services_area);
|
||||||
|
let in_hosts_area = is_in_area(x, y, &self.hosts_area);
|
||||||
|
|
||||||
if !in_system_area && !in_services_area {
|
if !in_system_area && !in_services_area && !in_hosts_area {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -431,6 +435,40 @@ impl Dashboard {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
MouseEventKind::Down(button) => {
|
MouseEventKind::Down(button) => {
|
||||||
|
// Handle clicks in hosts area
|
||||||
|
if in_hosts_area {
|
||||||
|
if let MouseButton::Left = button {
|
||||||
|
if let Some(ref mut tui_app) = self.tui_app {
|
||||||
|
// Calculate which host was clicked based on position
|
||||||
|
let relative_x = x.saturating_sub(self.hosts_area.x + 1) as usize; // +1 for border
|
||||||
|
let relative_y = y.saturating_sub(self.hosts_area.y + 1) as usize; // +1 for border
|
||||||
|
|
||||||
|
// Calculate column width based on hostnames
|
||||||
|
let max_name_len = tui_app.available_hosts.iter().map(|h| h.len()).max().unwrap_or(8);
|
||||||
|
let col_width = 2 + 2 + max_name_len + 1 + 1; // icon + arrow + name + asterisk + padding
|
||||||
|
|
||||||
|
// Calculate number of columns and rows
|
||||||
|
let inner_width = self.hosts_area.width.saturating_sub(2) as usize;
|
||||||
|
let num_columns = (inner_width / col_width).max(1);
|
||||||
|
let rows_per_column = (tui_app.available_hosts.len() + num_columns - 1) / num_columns;
|
||||||
|
|
||||||
|
// Determine which column was clicked
|
||||||
|
let clicked_column = relative_x / col_width;
|
||||||
|
let clicked_row = relative_y;
|
||||||
|
|
||||||
|
// Calculate host index
|
||||||
|
let host_index = clicked_column * rows_per_column + clicked_row;
|
||||||
|
|
||||||
|
if host_index < tui_app.available_hosts.len() {
|
||||||
|
let hostname = tui_app.available_hosts[host_index].clone();
|
||||||
|
debug!("Clicked host: {} (index: {})", hostname, host_index);
|
||||||
|
tui_app.set_current_host(Some(hostname));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return Ok(());
|
||||||
|
}
|
||||||
|
|
||||||
// Only handle clicks in services area (not system area)
|
// Only handle clicks in services area (not system area)
|
||||||
if !in_services_area {
|
if !in_services_area {
|
||||||
return Ok(());
|
return Ok(());
|
||||||
|
|||||||
@@ -63,7 +63,7 @@ pub struct TuiApp {
|
|||||||
/// Current active host
|
/// Current active host
|
||||||
pub current_host: Option<String>,
|
pub current_host: Option<String>,
|
||||||
/// Available hosts
|
/// Available hosts
|
||||||
available_hosts: Vec<String>,
|
pub available_hosts: Vec<String>,
|
||||||
/// Should quit application
|
/// Should quit application
|
||||||
should_quit: bool,
|
should_quit: bool,
|
||||||
/// Track if user manually navigated away from localhost
|
/// Track if user manually navigated away from localhost
|
||||||
@@ -440,6 +440,13 @@ impl TuiApp {
|
|||||||
self.switch_to_host(&prev_host);
|
self.switch_to_host(&prev_host);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Set current host (for mouse click selection)
|
||||||
|
pub fn set_current_host(&mut self, hostname: Option<String>) {
|
||||||
|
if let Some(host) = hostname {
|
||||||
|
self.switch_to_host(&host);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@@ -465,7 +472,7 @@ impl TuiApp {
|
|||||||
|
|
||||||
|
|
||||||
/// Render the dashboard (real btop-style multi-panel layout)
|
/// Render the dashboard (real btop-style multi-panel layout)
|
||||||
pub fn render(&mut self, frame: &mut Frame, metric_store: &MetricStore) -> (Rect, Rect, Rect) {
|
pub fn render(&mut self, frame: &mut Frame, metric_store: &MetricStore) -> (Rect, Rect, Rect, Rect) {
|
||||||
let size = frame.size();
|
let size = frame.size();
|
||||||
|
|
||||||
// Clear background to true black like btop
|
// Clear background to true black like btop
|
||||||
@@ -503,7 +510,7 @@ impl TuiApp {
|
|||||||
|
|
||||||
// Calculate hosts panel height dynamically based on available width
|
// Calculate hosts panel height dynamically based on available width
|
||||||
let hosts_inner_width = content_chunks[0].width.saturating_sub(2);
|
let hosts_inner_width = content_chunks[0].width.saturating_sub(2);
|
||||||
let hosts_content_height = HostsWidget::required_height(self.available_hosts.len(), hosts_inner_width);
|
let hosts_content_height = HostsWidget::required_height(self.available_hosts.len(), hosts_inner_width, &self.available_hosts);
|
||||||
let hosts_height = hosts_content_height + 2; // Add borders
|
let hosts_height = hosts_content_height + 2; // Add borders
|
||||||
|
|
||||||
// Left side: hosts panel on top, system panel below
|
// Left side: hosts panel on top, system panel below
|
||||||
@@ -519,7 +526,8 @@ impl TuiApp {
|
|||||||
self.render_btop_title(frame, main_chunks[0], metric_store);
|
self.render_btop_title(frame, main_chunks[0], metric_store);
|
||||||
|
|
||||||
// Render hosts panel on left
|
// Render hosts panel on left
|
||||||
self.render_hosts_panel(frame, left_chunks[0], metric_store);
|
let hosts_area = left_chunks[0];
|
||||||
|
self.render_hosts_panel(frame, hosts_area, metric_store);
|
||||||
|
|
||||||
// Render system panel below hosts
|
// Render system panel below hosts
|
||||||
let system_area = left_chunks[1];
|
let system_area = left_chunks[1];
|
||||||
@@ -543,7 +551,7 @@ impl TuiApp {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Return all areas for mouse event handling
|
// Return all areas for mouse event handling
|
||||||
(main_chunks[0], system_area, services_area)
|
(main_chunks[0], system_area, services_area, hosts_area)
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Render btop-style minimal title with host status colors
|
/// Render btop-style minimal title with host status colors
|
||||||
|
|||||||
@@ -55,17 +55,23 @@ impl HostsWidget {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/// Calculate the required height for hosts panel based on host count and available width
|
/// Calculate the required height for hosts panel based on host count and available width
|
||||||
pub fn required_height(num_hosts: usize, available_width: u16) -> u16 {
|
pub fn required_height(num_hosts: usize, available_width: u16, hostnames: &[String]) -> u16 {
|
||||||
if num_hosts == 0 {
|
if num_hosts == 0 {
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
// Estimate column width: icon(2) + arrow(2) + max_hostname(~12) + padding(2) = ~18
|
let col_width = Self::calculate_column_width(hostnames);
|
||||||
let col_width = 18u16;
|
|
||||||
let num_columns = (available_width / col_width).max(1) as usize;
|
let num_columns = (available_width / col_width).max(1) as usize;
|
||||||
let rows_needed = (num_hosts + num_columns - 1) / num_columns;
|
let rows_needed = (num_hosts + num_columns - 1) / num_columns;
|
||||||
rows_needed.max(1) as u16
|
rows_needed.max(1) as u16
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Calculate column width based on longest hostname
|
||||||
|
fn calculate_column_width(hostnames: &[String]) -> u16 {
|
||||||
|
// icon(2) + arrow(2) + hostname + asterisk(1) + padding(1)
|
||||||
|
let max_name_len = hostnames.iter().map(|h| h.len()).max().unwrap_or(8);
|
||||||
|
(2 + 2 + max_name_len + 1 + 1) as u16
|
||||||
|
}
|
||||||
|
|
||||||
/// Render hosts list in dynamic columns based on available width
|
/// Render hosts list in dynamic columns based on available width
|
||||||
pub fn render<F>(
|
pub fn render<F>(
|
||||||
&mut self,
|
&mut self,
|
||||||
@@ -88,8 +94,8 @@ impl HostsWidget {
|
|||||||
// Store viewport height for scroll calculations
|
// Store viewport height for scroll calculations
|
||||||
self.last_viewport_height = area.height as usize;
|
self.last_viewport_height = area.height as usize;
|
||||||
|
|
||||||
// Calculate column width and number of columns that fit
|
// Calculate column width based on actual hostname lengths
|
||||||
let col_width = 18u16;
|
let col_width = Self::calculate_column_width(available_hosts);
|
||||||
let num_columns = (area.width / col_width).max(1) as usize;
|
let num_columns = (area.width / col_width).max(1) as usize;
|
||||||
let rows_per_column = (available_hosts.len() + num_columns - 1) / num_columns;
|
let rows_per_column = (available_hosts.len() + num_columns - 1) / num_columns;
|
||||||
|
|
||||||
|
|||||||
@@ -19,61 +19,64 @@ struct ColumnVisibility {
|
|||||||
show_ram: bool,
|
show_ram: bool,
|
||||||
show_uptime: bool,
|
show_uptime: bool,
|
||||||
show_restarts: bool,
|
show_restarts: bool,
|
||||||
|
name_width: u16,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl ColumnVisibility {
|
impl ColumnVisibility {
|
||||||
/// Calculate actual width needed for all columns
|
/// Fixed column widths for non-name columns
|
||||||
const NAME_WIDTH: u16 = 23;
|
|
||||||
const STATUS_WIDTH: u16 = 10;
|
const STATUS_WIDTH: u16 = 10;
|
||||||
const RAM_WIDTH: u16 = 8;
|
const RAM_WIDTH: u16 = 8;
|
||||||
const UPTIME_WIDTH: u16 = 8;
|
const UPTIME_WIDTH: u16 = 8;
|
||||||
const RESTARTS_WIDTH: u16 = 5;
|
const RESTARTS_WIDTH: u16 = 5;
|
||||||
const COLUMN_SPACING: u16 = 1; // Space between columns
|
const COLUMN_SPACING: u16 = 1;
|
||||||
|
const MIN_NAME_WIDTH: u16 = 10;
|
||||||
|
|
||||||
|
/// Determine which columns to show based on available width and service names
|
||||||
|
fn from_width_and_names(width: u16, max_name_len: usize) -> Self {
|
||||||
|
// Calculate name width: icon(2) + name + padding(1), min 10
|
||||||
|
let name_width = ((2 + max_name_len + 1) as u16).max(Self::MIN_NAME_WIDTH);
|
||||||
|
|
||||||
/// Determine which columns to show based on available width
|
|
||||||
/// Priority order: Name > Status > RAM > Uptime > Restarts
|
|
||||||
fn from_width(width: u16) -> Self {
|
|
||||||
// Calculate cumulative widths for each configuration
|
// Calculate cumulative widths for each configuration
|
||||||
let minimal = Self::NAME_WIDTH + Self::COLUMN_SPACING + Self::STATUS_WIDTH; // 34
|
let minimal = name_width + Self::COLUMN_SPACING + Self::STATUS_WIDTH;
|
||||||
let with_ram = minimal + Self::COLUMN_SPACING + Self::RAM_WIDTH; // 43
|
let with_ram = minimal + Self::COLUMN_SPACING + Self::RAM_WIDTH;
|
||||||
let with_uptime = with_ram + Self::COLUMN_SPACING + Self::UPTIME_WIDTH; // 52
|
let with_uptime = with_ram + Self::COLUMN_SPACING + Self::UPTIME_WIDTH;
|
||||||
let full = with_uptime + Self::COLUMN_SPACING + Self::RESTARTS_WIDTH; // 58
|
let full = with_uptime + Self::COLUMN_SPACING + Self::RESTARTS_WIDTH;
|
||||||
|
|
||||||
if width >= full {
|
if width >= full {
|
||||||
// Show all columns
|
|
||||||
Self {
|
Self {
|
||||||
show_name: true,
|
show_name: true,
|
||||||
show_status: true,
|
show_status: true,
|
||||||
show_ram: true,
|
show_ram: true,
|
||||||
show_uptime: true,
|
show_uptime: true,
|
||||||
show_restarts: true,
|
show_restarts: true,
|
||||||
|
name_width,
|
||||||
}
|
}
|
||||||
} else if width >= with_uptime {
|
} else if width >= with_uptime {
|
||||||
// Hide restarts
|
|
||||||
Self {
|
Self {
|
||||||
show_name: true,
|
show_name: true,
|
||||||
show_status: true,
|
show_status: true,
|
||||||
show_ram: true,
|
show_ram: true,
|
||||||
show_uptime: true,
|
show_uptime: true,
|
||||||
show_restarts: false,
|
show_restarts: false,
|
||||||
|
name_width,
|
||||||
}
|
}
|
||||||
} else if width >= with_ram {
|
} else if width >= with_ram {
|
||||||
// Hide uptime and restarts
|
|
||||||
Self {
|
Self {
|
||||||
show_name: true,
|
show_name: true,
|
||||||
show_status: true,
|
show_status: true,
|
||||||
show_ram: true,
|
show_ram: true,
|
||||||
show_uptime: false,
|
show_uptime: false,
|
||||||
show_restarts: false,
|
show_restarts: false,
|
||||||
|
name_width,
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Minimal: Name + Status only
|
|
||||||
Self {
|
Self {
|
||||||
show_name: true,
|
show_name: true,
|
||||||
show_status: true,
|
show_status: true,
|
||||||
show_ram: false,
|
show_ram: false,
|
||||||
show_uptime: false,
|
show_uptime: false,
|
||||||
show_restarts: false,
|
show_restarts: false,
|
||||||
|
name_width,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -125,6 +128,15 @@ impl ServicesWidget {
|
|||||||
self.status
|
self.status
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Get max service name length for column width calculation
|
||||||
|
fn get_max_service_name_len(&self) -> usize {
|
||||||
|
self.parent_services
|
||||||
|
.keys()
|
||||||
|
.map(|name| name.len())
|
||||||
|
.max()
|
||||||
|
.unwrap_or(8)
|
||||||
|
}
|
||||||
|
|
||||||
/// Extract service name and determine if it's a parent or sub-service
|
/// Extract service name and determine if it's a parent or sub-service
|
||||||
#[allow(dead_code)]
|
#[allow(dead_code)]
|
||||||
fn extract_service_info(metric_name: &str) -> Option<(String, Option<String>)> {
|
fn extract_service_info(metric_name: &str) -> Option<(String, Option<String>)> {
|
||||||
@@ -156,7 +168,7 @@ impl ServicesWidget {
|
|||||||
/// Format parent service line - returns text without icon for span formatting
|
/// Format parent service line - returns text without icon for span formatting
|
||||||
fn format_parent_service_line(&self, name: &str, info: &ServiceInfo, columns: ColumnVisibility) -> String {
|
fn format_parent_service_line(&self, name: &str, info: &ServiceInfo, columns: ColumnVisibility) -> String {
|
||||||
// Account for icon prefix "● " (2 chars) in name column width
|
// Account for icon prefix "● " (2 chars) in name column width
|
||||||
let name_width = ColumnVisibility::NAME_WIDTH.saturating_sub(2) as usize;
|
let name_width = columns.name_width.saturating_sub(2) as usize;
|
||||||
// Truncate long service names to fit layout
|
// Truncate long service names to fit layout
|
||||||
let max_name_len = name_width.saturating_sub(3); // -3 for "..."
|
let max_name_len = name_width.saturating_sub(3); // -3 for "..."
|
||||||
let short_name = if name.len() > max_name_len {
|
let short_name = if name.len() > max_name_len {
|
||||||
@@ -724,13 +736,16 @@ impl ServicesWidget {
|
|||||||
.constraints([Constraint::Length(1), Constraint::Min(0)])
|
.constraints([Constraint::Length(1), Constraint::Min(0)])
|
||||||
.split(area);
|
.split(area);
|
||||||
|
|
||||||
// Determine which columns to show based on available width
|
// Calculate max service name length for dynamic column width
|
||||||
let columns = ColumnVisibility::from_width(area.width);
|
let max_name_len = self.get_max_service_name_len();
|
||||||
|
|
||||||
|
// Determine which columns to show based on available width and service names
|
||||||
|
let columns = ColumnVisibility::from_width_and_names(area.width, max_name_len);
|
||||||
|
|
||||||
// Build header - columns must align with service row format
|
// Build header - columns must align with service row format
|
||||||
let mut header_parts = Vec::new();
|
let mut header_parts = Vec::new();
|
||||||
if columns.show_name {
|
if columns.show_name {
|
||||||
header_parts.push(format!("{:<width$}", "Service:", width = ColumnVisibility::NAME_WIDTH as usize));
|
header_parts.push(format!("{:<width$}", "Service:", width = columns.name_width as usize));
|
||||||
}
|
}
|
||||||
if columns.show_status {
|
if columns.show_status {
|
||||||
header_parts.push(format!("{:<width$}", "Status:", width = ColumnVisibility::STATUS_WIDTH as usize));
|
header_parts.push(format!("{:<width$}", "Status:", width = ColumnVisibility::STATUS_WIDTH as usize));
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
[package]
|
[package]
|
||||||
name = "cm-dashboard-shared"
|
name = "cm-dashboard-shared"
|
||||||
version = "0.1.280"
|
version = "0.1.282"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
|
|
||||||
[dependencies]
|
[dependencies]
|
||||||
|
|||||||
Reference in New Issue
Block a user