admin panel for admin
This commit is contained in:
@@ -2,6 +2,7 @@
|
|||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::pages::auth::AuthState;
|
use crate::state::pages::auth::AuthState;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::admin::AdminState;
|
use crate::state::pages::admin::AdminState;
|
||||||
use common::proto::multieko2::table_definition::ProfileTreeResponse;
|
use common::proto::multieko2::table_definition::ProfileTreeResponse;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
@@ -15,8 +16,9 @@ use super::admin_panel_admin::render_admin_panel_admin;
|
|||||||
|
|
||||||
pub fn render_admin_panel(
|
pub fn render_admin_panel(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
|
app_state: &AppState,
|
||||||
auth_state: &AuthState,
|
auth_state: &AuthState,
|
||||||
admin_state: &AdminState,
|
admin_state: &mut AdminState,
|
||||||
area: Rect,
|
area: Rect,
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
profile_tree: &ProfileTreeResponse,
|
profile_tree: &ProfileTreeResponse,
|
||||||
@@ -57,7 +59,13 @@ pub fn render_admin_panel(
|
|||||||
selected_profile,
|
selected_profile,
|
||||||
);
|
);
|
||||||
} else {
|
} else {
|
||||||
render_admin_panel_admin(f, &content_chunks, theme);
|
render_admin_panel_admin(
|
||||||
|
f,
|
||||||
|
content_chunks[0],
|
||||||
|
app_state,
|
||||||
|
admin_state,
|
||||||
|
theme,
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,19 +4,168 @@ use crate::config::colors::themes::Theme;
|
|||||||
use crate::state::pages::admin::{AdminFocus, AdminState};
|
use crate::state::pages::admin::{AdminFocus, AdminState};
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
||||||
style::{Style, Stylize},
|
style::Style,
|
||||||
text::{Line, Span},
|
text::{Line, Span},
|
||||||
widgets::{Block, BorderType, Borders, List, ListItem, Paragraph},
|
widgets::{Block, BorderType, Borders, List, ListItem, ListState, Paragraph},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
|
|
||||||
/// Renders the view specific to admin users.
|
/// Renders the view specific to admin users.
|
||||||
pub fn render_admin_panel_admin(f: &mut Frame, content_chunks: &[Rect], theme: &Theme) {
|
pub fn render_admin_panel_admin(
|
||||||
// Admin-specific view placeholder
|
f: &mut Frame,
|
||||||
let admin_message = Paragraph::new("Admin-specific view. Profile selection not applicable.")
|
area: Rect,
|
||||||
.style(Style::default().fg(theme.fg))
|
app_state: &AppState,
|
||||||
.alignment(Alignment::Center);
|
admin_state: &mut AdminState,
|
||||||
// Render only in the right pane for now, leaving left empty
|
theme: &Theme,
|
||||||
f.render_widget(admin_message, content_chunks[1]);
|
) {
|
||||||
|
let main_block = Block::default()
|
||||||
|
.title(" Admin Panel ")
|
||||||
|
.borders(Borders::ALL)
|
||||||
|
.border_type(BorderType::Rounded)
|
||||||
|
.border_style(Style::default().fg(theme.border));
|
||||||
|
f.render_widget(main_block, area);
|
||||||
|
|
||||||
|
let inner_area = area.inner(ratatui::layout::Margin { vertical: 1, horizontal: 1 });
|
||||||
|
|
||||||
|
// Split the inner area into two panes
|
||||||
|
let chunks = Layout::default()
|
||||||
|
.direction(Direction::Horizontal)
|
||||||
|
.constraints([Constraint::Percentage(30), Constraint::Percentage(70)].as_ref())
|
||||||
|
.split(inner_area);
|
||||||
|
|
||||||
|
let left_pane = chunks[0];
|
||||||
|
let right_pane = chunks[1];
|
||||||
|
|
||||||
|
// --- Profiles Pane (Left) ---
|
||||||
|
let profile_focus = admin_state.current_focus == AdminFocus::Profiles;
|
||||||
|
let profile_border_style = if profile_focus {
|
||||||
|
Style::default().fg(theme.highlight)
|
||||||
|
} else {
|
||||||
|
Style::default().fg(theme.border)
|
||||||
|
};
|
||||||
|
|
||||||
|
let profile_list_items: Vec<ListItem> = app_state
|
||||||
|
.profile_tree
|
||||||
|
.profiles
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(i, profile)| {
|
||||||
|
let is_selected = admin_state.profile_list_state.selected() == Some(i);
|
||||||
|
let prefix = if is_selected { "[*] " } else { "[ ] " };
|
||||||
|
let style = if is_selected {
|
||||||
|
Style::default().fg(theme.highlight).add_modifier(ratatui::style::Modifier::BOLD)
|
||||||
|
} else {
|
||||||
|
Style::default().fg(theme.fg)
|
||||||
|
};
|
||||||
|
ListItem::new(Line::from(vec![Span::styled(prefix, style), Span::styled(&profile.name, style)]))
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let profile_list = List::new(profile_list_items)
|
||||||
|
.block(
|
||||||
|
Block::default()
|
||||||
|
.title(" Profiles ")
|
||||||
|
.borders(Borders::ALL)
|
||||||
|
.border_type(BorderType::Rounded)
|
||||||
|
.border_style(profile_border_style),
|
||||||
|
)
|
||||||
|
.highlight_style(Style::default().add_modifier(ratatui::style::Modifier::REVERSED))
|
||||||
|
.highlight_symbol(if profile_focus { "> " } else { " " });
|
||||||
|
|
||||||
|
f.render_stateful_widget(profile_list, left_pane, &mut admin_state.profile_list_state);
|
||||||
|
|
||||||
|
// --- Tables & Dependencies Pane (Right) ---
|
||||||
|
let table_focus = admin_state.current_focus == AdminFocus::Tables;
|
||||||
|
let table_border_style = if table_focus {
|
||||||
|
Style::default().fg(theme.highlight)
|
||||||
|
} else {
|
||||||
|
Style::default().fg(theme.border)
|
||||||
|
};
|
||||||
|
|
||||||
|
let right_chunks = Layout::default()
|
||||||
|
.direction(Direction::Vertical)
|
||||||
|
.constraints([Constraint::Min(0), Constraint::Length(3)])
|
||||||
|
.split(right_pane);
|
||||||
|
|
||||||
|
let tables_area = right_chunks[0];
|
||||||
|
let deps_area = right_chunks[1];
|
||||||
|
|
||||||
|
let selected_profile_idx = admin_state.profile_list_state.selected();
|
||||||
|
let selected_profile_name = app_state
|
||||||
|
.profile_tree
|
||||||
|
.profiles
|
||||||
|
.get(selected_profile_idx.unwrap_or(usize::MAX))
|
||||||
|
.map_or("None", |p| &p.name);
|
||||||
|
|
||||||
|
let (table_list_items, selected_table_deps): (Vec<ListItem>, Vec<String>) = if let Some(
|
||||||
|
profile,
|
||||||
|
) = selected_profile_idx.and_then(|idx| app_state.profile_tree.profiles.get(idx)) {
|
||||||
|
let items: Vec<ListItem> = profile
|
||||||
|
.tables
|
||||||
|
.iter()
|
||||||
|
.enumerate()
|
||||||
|
.map(|(i, table)| {
|
||||||
|
let is_selected = admin_state.table_list_state.selected() == Some(i);
|
||||||
|
let prefix = if is_selected { "[*] " } else { "[ ] " };
|
||||||
|
let deps_str = if !table.depends_on.is_empty() {
|
||||||
|
format!(" -> [{}]", table.depends_on.join(", "))
|
||||||
|
} else {
|
||||||
|
String::new()
|
||||||
|
};
|
||||||
|
let style = if is_selected {
|
||||||
|
Style::default().fg(theme.highlight).add_modifier(ratatui::style::Modifier::BOLD)
|
||||||
|
} else {
|
||||||
|
Style::default().fg(theme.fg)
|
||||||
|
};
|
||||||
|
ListItem::new(Line::from(vec![
|
||||||
|
Span::styled(prefix, style),
|
||||||
|
Span::styled(&table.name, style),
|
||||||
|
Span::styled(deps_str, style.fg(theme.secondary)),
|
||||||
|
]))
|
||||||
|
})
|
||||||
|
.collect();
|
||||||
|
|
||||||
|
let deps = admin_state.table_list_state.selected()
|
||||||
|
.and_then(|idx| profile.tables.get(idx))
|
||||||
|
.map_or(vec![], |t| t.depends_on.clone());
|
||||||
|
|
||||||
|
(items, deps)
|
||||||
|
} else {
|
||||||
|
(vec![ListItem::new("No profile selected or profile has no tables")], vec![])
|
||||||
|
};
|
||||||
|
|
||||||
|
let table_list = List::new(table_list_items)
|
||||||
|
.block(
|
||||||
|
Block::default()
|
||||||
|
.title(format!(" Tables & Dependencies (Profile: {}) ", selected_profile_name))
|
||||||
|
.borders(Borders::ALL)
|
||||||
|
.border_type(BorderType::Rounded)
|
||||||
|
.border_style(table_border_style),
|
||||||
|
)
|
||||||
|
.highlight_style(Style::default().add_modifier(ratatui::style::Modifier::REVERSED))
|
||||||
|
.highlight_symbol(if table_focus { "> " } else { " " });
|
||||||
|
|
||||||
|
f.render_stateful_widget(table_list, tables_area, &mut admin_state.table_list_state);
|
||||||
|
|
||||||
|
// --- Dependencies Display ---
|
||||||
|
let selected_table_name = selected_profile_idx
|
||||||
|
.and_then(|p_idx| app_state.profile_tree.profiles.get(p_idx))
|
||||||
|
.and_then(|p| admin_state.table_list_state.selected().and_then(|t_idx| p.tables.get(t_idx)))
|
||||||
|
.map_or("N/A", |t| &t.name);
|
||||||
|
|
||||||
|
let deps_text = if !selected_table_deps.is_empty() {
|
||||||
|
format!("Dependencies for {}: {}", selected_table_name, selected_table_deps.join(", "))
|
||||||
|
} else {
|
||||||
|
format!("Dependencies for {}: None", selected_table_name)
|
||||||
|
};
|
||||||
|
|
||||||
|
let deps_paragraph = Paragraph::new(deps_text)
|
||||||
|
.style(Style::default().fg(theme.secondary))
|
||||||
|
.block(
|
||||||
|
Block::default()
|
||||||
|
.borders(Borders::TOP)
|
||||||
|
.border_style(Style::default().fg(theme.border)),
|
||||||
|
);
|
||||||
|
f.render_widget(deps_paragraph, deps_area);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -109,6 +109,7 @@ pub fn render_ui(
|
|||||||
} else if app_state.ui.show_admin {
|
} else if app_state.ui.show_admin {
|
||||||
crate::components::admin::admin_panel::render_admin_panel(
|
crate::components::admin::admin_panel::render_admin_panel(
|
||||||
f,
|
f,
|
||||||
|
app_state,
|
||||||
auth_state,
|
auth_state,
|
||||||
admin_state,
|
admin_state,
|
||||||
main_content_area,
|
main_content_area,
|
||||||
|
|||||||
Reference in New Issue
Block a user