Compare commits
55 Commits
v0.5.0
...
9ed558562b
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
9ed558562b | ||
|
|
43f5c1a764 | ||
|
|
46149c09db | ||
|
|
a0757efe8b | ||
|
|
10f4b9d8e2 | ||
|
|
42db496ad7 | ||
|
|
d6fd672409 | ||
|
|
60eb1c9f51 | ||
|
|
a09c804595 | ||
|
|
a17f73fd54 | ||
|
|
2373ae4b8c | ||
|
|
16dd460469 | ||
|
|
58f109ca91 | ||
|
|
75da9c0f4b | ||
|
|
833b918c5b | ||
|
|
72c2691a17 | ||
|
|
cf79bc7bd5 | ||
|
|
f5f2f2cdef | ||
|
|
19a9bab8c2 | ||
|
|
6e221ef8c1 | ||
|
|
e142f56706 | ||
|
|
a794f22366 | ||
|
|
cfe4903c79 | ||
|
|
a0a473f96c | ||
|
|
9e4dd3b4c7 | ||
|
|
e5db0334c0 | ||
|
|
d641ad1bbb | ||
|
|
18393ff661 | ||
|
|
b2a82fba30 | ||
|
|
f6c2fd627f | ||
|
|
15d9b31cb6 | ||
|
|
06cc1663b3 | ||
|
|
88a4b2d69c | ||
|
|
e6072d25c5 | ||
|
|
fc2b65601e | ||
|
|
597bdde7e1 | ||
|
|
f56092e86c | ||
|
|
d5cfe59f47 | ||
|
|
f281eaa662 | ||
|
|
cbb3ed7c48 | ||
|
|
41a0b85376 | ||
|
|
b5a31ee81c | ||
|
|
dceb031822 | ||
|
|
78bc9fc432 | ||
|
|
b9072e4d7c | ||
|
|
5d97e63f93 | ||
|
|
957f5bf9f0 | ||
|
|
6833ac5fad | ||
|
|
3dff2ced6c | ||
|
|
ea7ff3796f | ||
|
|
310617d62b | ||
|
|
1d94e82f4b | ||
|
|
00dad5d673 | ||
|
|
414c6957e7 | ||
|
|
f127298e5a |
10
Cargo.lock
generated
10
Cargo.lock
generated
@@ -493,7 +493,7 @@ checksum = "d71b6127be86fdcfddb610f7182ac57211d4b18a3e9c82eb2d17662f2227ad6a"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "canvas"
|
name = "canvas"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@@ -584,7 +584,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "client"
|
name = "client"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"async-trait",
|
"async-trait",
|
||||||
@@ -635,7 +635,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "common"
|
name = "common"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"prost",
|
"prost",
|
||||||
"prost-types",
|
"prost-types",
|
||||||
@@ -3022,7 +3022,7 @@ checksum = "1c107b6f4780854c8b126e228ea8869f4d7b71260f962fefb57b996b8959ba6b"
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "search"
|
name = "search"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"common",
|
"common",
|
||||||
@@ -3121,7 +3121,7 @@ dependencies = [
|
|||||||
|
|
||||||
[[package]]
|
[[package]]
|
||||||
name = "server"
|
name = "server"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
dependencies = [
|
dependencies = [
|
||||||
"anyhow",
|
"anyhow",
|
||||||
"bcrypt",
|
"bcrypt",
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ resolver = "2"
|
|||||||
[workspace.package]
|
[workspace.package]
|
||||||
# TODO: idk how to do the name, fix later
|
# TODO: idk how to do the name, fix later
|
||||||
# name = "komp_ac"
|
# name = "komp_ac"
|
||||||
version = "0.4.2"
|
version = "0.5.0"
|
||||||
edition = "2021"
|
edition = "2021"
|
||||||
license = "GPL-3.0-or-later"
|
license = "GPL-3.0-or-later"
|
||||||
authors = ["Filip Priečinský <filippriec@gmail.com>"]
|
authors = ["Filip Priečinský <filippriec@gmail.com>"]
|
||||||
|
|||||||
@@ -7,16 +7,14 @@ previous_buffer = ["space+b+p"]
|
|||||||
close_buffer = ["space+b+d"]
|
close_buffer = ["space+b+d"]
|
||||||
|
|
||||||
[keybindings.general]
|
[keybindings.general]
|
||||||
move_up = ["k", "Up"]
|
up = ["k", "Up"]
|
||||||
move_down = ["j", "Down"]
|
down = ["j", "Down"]
|
||||||
next_option = ["l", "Right"]
|
left = ["h", "Left"]
|
||||||
previous_option = ["h", "Left"]
|
right = ["l", "Right"]
|
||||||
|
next = ["Tab"]
|
||||||
|
previous = ["Shift+Tab"]
|
||||||
select = ["Enter"]
|
select = ["Enter"]
|
||||||
toggle_sidebar = ["ctrl+t"]
|
esc = ["esc"]
|
||||||
toggle_buffer_list = ["ctrl+b"]
|
|
||||||
next_field = ["Tab"]
|
|
||||||
prev_field = ["Shift+Tab"]
|
|
||||||
exit_table_scroll = ["esc"]
|
|
||||||
open_search = ["ctrl+f"]
|
open_search = ["ctrl+f"]
|
||||||
|
|
||||||
[keybindings.common]
|
[keybindings.common]
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
// src/components/common/find_file_palette.rs
|
// src/bottom_panel/find_file_palette.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::modes::general::command_navigation::NavigationState; // Corrected path
|
use crate::modes::general::command_navigation::NavigationState; // Corrected path
|
||||||
98
client/src/bottom_panel/layout.rs
Normal file
98
client/src/bottom_panel/layout.rs
Normal file
@@ -0,0 +1,98 @@
|
|||||||
|
// src/bottom_panel/layout.rs
|
||||||
|
|
||||||
|
use ratatui::{layout::Constraint, layout::Rect, Frame};
|
||||||
|
use crate::bottom_panel::{status_line::render_status_line, command_line::render_command_line};
|
||||||
|
use crate::bottom_panel::find_file_palette;
|
||||||
|
use crate::config::colors::themes::Theme;
|
||||||
|
use crate::modes::general::command_navigation::NavigationState;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::pages::routing::Router;
|
||||||
|
|
||||||
|
/// Calculate the layout constraints for the bottom panel (status line + command line/palette).
|
||||||
|
pub fn bottom_panel_constraints(
|
||||||
|
app_state: &AppState,
|
||||||
|
navigation_state: &NavigationState,
|
||||||
|
event_handler_command_mode_active: bool,
|
||||||
|
) -> Vec<Constraint> {
|
||||||
|
let mut status_line_height = 1;
|
||||||
|
#[cfg(feature = "ui-debug")]
|
||||||
|
{
|
||||||
|
if let Some(debug_state) = &app_state.debug_state {
|
||||||
|
if debug_state.is_error {
|
||||||
|
status_line_height = 4;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const PALETTE_OPTIONS_HEIGHT_FOR_LAYOUT: u16 = 15;
|
||||||
|
let command_palette_area_height = if navigation_state.active {
|
||||||
|
1 + PALETTE_OPTIONS_HEIGHT_FOR_LAYOUT
|
||||||
|
} else if event_handler_command_mode_active {
|
||||||
|
1
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
};
|
||||||
|
|
||||||
|
let mut constraints = vec![Constraint::Length(status_line_height)];
|
||||||
|
if command_palette_area_height > 0 {
|
||||||
|
constraints.push(Constraint::Length(command_palette_area_height));
|
||||||
|
}
|
||||||
|
constraints
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Render the bottom panel (status line + command line/palette).
|
||||||
|
pub fn render_bottom_panel(
|
||||||
|
f: &mut Frame,
|
||||||
|
root_chunks: &[Rect],
|
||||||
|
chunk_idx: &mut usize,
|
||||||
|
current_dir: &str,
|
||||||
|
theme: &Theme,
|
||||||
|
current_fps: f64,
|
||||||
|
app_state: &AppState,
|
||||||
|
router: &Router,
|
||||||
|
navigation_state: &NavigationState,
|
||||||
|
event_handler_command_input: &str,
|
||||||
|
event_handler_command_mode_active: bool,
|
||||||
|
event_handler_command_message: &str,
|
||||||
|
) {
|
||||||
|
// --- Status line area ---
|
||||||
|
let status_line_area = root_chunks[*chunk_idx];
|
||||||
|
*chunk_idx += 1;
|
||||||
|
|
||||||
|
// --- Command line / palette area ---
|
||||||
|
let command_render_area = if root_chunks.len() > *chunk_idx {
|
||||||
|
Some(root_chunks[*chunk_idx])
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
if command_render_area.is_some() {
|
||||||
|
*chunk_idx += 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- Render status line ---
|
||||||
|
render_status_line(
|
||||||
|
f,
|
||||||
|
status_line_area,
|
||||||
|
current_dir,
|
||||||
|
theme,
|
||||||
|
current_fps,
|
||||||
|
app_state,
|
||||||
|
router,
|
||||||
|
);
|
||||||
|
|
||||||
|
// --- Render command line or palette ---
|
||||||
|
if let Some(area) = command_render_area {
|
||||||
|
if navigation_state.active {
|
||||||
|
find_file_palette::render_find_file_palette(f, area, theme, navigation_state);
|
||||||
|
} else if event_handler_command_mode_active {
|
||||||
|
render_command_line(
|
||||||
|
f,
|
||||||
|
area,
|
||||||
|
event_handler_command_input,
|
||||||
|
true,
|
||||||
|
theme,
|
||||||
|
event_handler_command_message,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
6
client/src/bottom_panel/mod.rs
Normal file
6
client/src/bottom_panel/mod.rs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
// src/bottom_panel/mod.rs
|
||||||
|
|
||||||
|
pub mod status_line;
|
||||||
|
pub mod command_line;
|
||||||
|
pub mod layout;
|
||||||
|
pub mod find_file_palette;
|
||||||
@@ -5,10 +5,11 @@ use ratatui::{
|
|||||||
layout::Rect,
|
layout::Rect,
|
||||||
style::Style,
|
style::Style,
|
||||||
text::{Line, Span, Text},
|
text::{Line, Span, Text},
|
||||||
widgets::Paragraph,
|
widgets::{Paragraph, Wrap},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use ratatui::widgets::Wrap;
|
use crate::pages::routing::Page;
|
||||||
|
use crate::pages::routing::Router;
|
||||||
use std::path::Path;
|
use std::path::Path;
|
||||||
use unicode_width::UnicodeWidthStr;
|
use unicode_width::UnicodeWidthStr;
|
||||||
|
|
||||||
@@ -17,9 +18,9 @@ pub fn render_status_line(
|
|||||||
area: Rect,
|
area: Rect,
|
||||||
current_dir: &str,
|
current_dir: &str,
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
is_edit_mode: bool,
|
|
||||||
current_fps: f64,
|
current_fps: f64,
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
|
router: &Router,
|
||||||
) {
|
) {
|
||||||
#[cfg(feature = "ui-debug")]
|
#[cfg(feature = "ui-debug")]
|
||||||
{
|
{
|
||||||
@@ -49,7 +50,20 @@ pub fn render_status_line(
|
|||||||
|
|
||||||
// --- The normal status line rendering logic (unchanged) ---
|
// --- The normal status line rendering logic (unchanged) ---
|
||||||
let program_info = format!("komp_ac v{}", env!("CARGO_PKG_VERSION"));
|
let program_info = format!("komp_ac v{}", env!("CARGO_PKG_VERSION"));
|
||||||
let mode_text = if is_edit_mode { "[EDIT]" } else { "[READ-ONLY]" };
|
let mode_text = if let Page::Form(path) = &router.current {
|
||||||
|
if let Some(editor) = app_state.editor_for_path_ref(path) {
|
||||||
|
match editor.mode() {
|
||||||
|
canvas::AppMode::Edit => "[EDIT]",
|
||||||
|
canvas::AppMode::ReadOnly => "[READ-ONLY]",
|
||||||
|
canvas::AppMode::Highlight => "[VISUAL]",
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
""
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
"" // No canvas active
|
||||||
|
};
|
||||||
|
|
||||||
let home_dir = dirs::home_dir()
|
let home_dir = dirs::home_dir()
|
||||||
.map(|p| p.to_string_lossy().into_owned())
|
.map(|p| p.to_string_lossy().into_owned())
|
||||||
@@ -1,13 +1,13 @@
|
|||||||
// src/functions/common/buffer.rs
|
// src/buffer/functions/buffer.rs
|
||||||
|
|
||||||
use crate::state::app::buffer::BufferState;
|
use crate::buffer::state::BufferState;
|
||||||
use crate::state::app::buffer::AppView;
|
use crate::buffer::state::AppView;
|
||||||
|
|
||||||
pub fn get_view_layer(view: &AppView) -> u8 {
|
pub fn get_view_layer(view: &AppView) -> u8 {
|
||||||
match view {
|
match view {
|
||||||
AppView::Intro => 1,
|
AppView::Intro => 1,
|
||||||
AppView::Login | AppView::Register | AppView::Admin | AppView::AddTable | AppView::AddLogic => 2,
|
AppView::Login | AppView::Register | AppView::Admin | AppView::AddTable | AppView::AddLogic => 2,
|
||||||
AppView::Form | AppView::Scratch => 3,
|
AppView::Form(_) | AppView::Scratch => 3,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
20
client/src/buffer/logic.rs
Normal file
20
client/src/buffer/logic.rs
Normal file
@@ -0,0 +1,20 @@
|
|||||||
|
// src/buffer/logic.rs
|
||||||
|
use crossterm::event::{KeyCode, KeyModifiers};
|
||||||
|
use crate::config::binds::config::Config;
|
||||||
|
use crate::state::app::state::UiState;
|
||||||
|
|
||||||
|
/// Toggle the buffer list visibility based on keybindings.
|
||||||
|
pub fn toggle_buffer_list(
|
||||||
|
ui_state: &mut UiState,
|
||||||
|
config: &Config,
|
||||||
|
key: KeyCode,
|
||||||
|
modifiers: KeyModifiers,
|
||||||
|
) -> bool {
|
||||||
|
if let Some(action) = config.get_common_action(key, modifiers) {
|
||||||
|
if action == "toggle_buffer_list" {
|
||||||
|
ui_state.show_buffer_list = !ui_state.show_buffer_list;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
11
client/src/buffer/mod.rs
Normal file
11
client/src/buffer/mod.rs
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
// src/buffer/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod functions;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
|
||||||
|
pub use state::{AppView, BufferState};
|
||||||
|
pub use functions::*;
|
||||||
|
pub use ui::render_buffer_list;
|
||||||
|
pub use logic::toggle_buffer_list;
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
// src/state/app/buffer.rs
|
// src/buffer/state/buffer.rs
|
||||||
|
|
||||||
#[derive(Debug, Clone, PartialEq, Eq)]
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
pub enum AppView {
|
pub enum AppView {
|
||||||
@@ -8,7 +8,7 @@ pub enum AppView {
|
|||||||
Admin,
|
Admin,
|
||||||
AddTable,
|
AddTable,
|
||||||
AddLogic,
|
AddLogic,
|
||||||
Form,
|
Form(String),
|
||||||
Scratch,
|
Scratch,
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -23,7 +23,7 @@ impl AppView {
|
|||||||
AppView::Admin => "Admin_Panel",
|
AppView::Admin => "Admin_Panel",
|
||||||
AppView::AddTable => "Add_Table",
|
AppView::AddTable => "Add_Table",
|
||||||
AppView::AddLogic => "Add_Logic",
|
AppView::AddLogic => "Add_Logic",
|
||||||
AppView::Form => "Form",
|
AppView::Form(_) => "Form",
|
||||||
AppView::Scratch => "*scratch*",
|
AppView::Scratch => "*scratch*",
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -31,10 +31,14 @@ impl AppView {
|
|||||||
/// Returns the display name with dynamic context (for Form buffers)
|
/// Returns the display name with dynamic context (for Form buffers)
|
||||||
pub fn display_name_with_context(&self, current_table_name: Option<&str>) -> String {
|
pub fn display_name_with_context(&self, current_table_name: Option<&str>) -> String {
|
||||||
match self {
|
match self {
|
||||||
AppView::Form => {
|
AppView::Form(path) => {
|
||||||
current_table_name
|
// Derive table name from "profile/table" path
|
||||||
.unwrap_or("Data Form")
|
let table = path.split('/').nth(1).unwrap_or("");
|
||||||
.to_string()
|
if !table.is_empty() {
|
||||||
|
table.to_string()
|
||||||
|
} else {
|
||||||
|
current_table_name.unwrap_or("Data Form").to_string()
|
||||||
|
}
|
||||||
}
|
}
|
||||||
_ => self.display_name().to_string(),
|
_ => self.display_name().to_string(),
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
// src/components/handlers/buffer_list.rs
|
// src/buffer/ui.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::app::buffer::BufferState;
|
use crate::buffer::state::BufferState;
|
||||||
use crate::state::app::state::AppState; // Add this import
|
use crate::state::app::state::AppState; // Add this import
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Rect},
|
layout::{Alignment, Rect},
|
||||||
@@ -11,7 +11,7 @@ use ratatui::{
|
|||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use unicode_width::UnicodeWidthStr;
|
use unicode_width::UnicodeWidthStr;
|
||||||
use crate::functions::common::buffer::get_view_layer;
|
use crate::buffer::functions::get_view_layer;
|
||||||
|
|
||||||
pub fn render_buffer_list(
|
pub fn render_buffer_list(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
// src/components/admin.rs
|
|
||||||
pub mod admin_panel;
|
|
||||||
pub mod admin_panel_admin;
|
|
||||||
pub mod add_table;
|
|
||||||
pub mod add_logic;
|
|
||||||
|
|
||||||
pub use admin_panel::*;
|
|
||||||
pub use admin_panel_admin::*;
|
|
||||||
pub use add_table::*;
|
|
||||||
pub use add_logic::*;
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// src/components/form.rs
|
|
||||||
pub mod login;
|
|
||||||
pub mod register;
|
|
||||||
|
|
||||||
pub use login::*;
|
|
||||||
pub use register::*;
|
|
||||||
@@ -1,18 +1,9 @@
|
|||||||
// src/components/common.rs
|
// src/components/common.rs
|
||||||
pub mod command_line;
|
|
||||||
pub mod status_line;
|
|
||||||
pub mod text_editor;
|
pub mod text_editor;
|
||||||
pub mod background;
|
pub mod background;
|
||||||
pub mod dialog;
|
|
||||||
pub mod autocomplete;
|
pub mod autocomplete;
|
||||||
pub mod search_palette;
|
|
||||||
pub mod find_file_palette;
|
|
||||||
|
|
||||||
pub use command_line::*;
|
|
||||||
pub use status_line::*;
|
|
||||||
pub use text_editor::*;
|
pub use text_editor::*;
|
||||||
pub use background::*;
|
pub use background::*;
|
||||||
pub use dialog::*;
|
|
||||||
pub use autocomplete::*;
|
pub use autocomplete::*;
|
||||||
pub use search_palette::*;
|
|
||||||
pub use find_file_palette::*;
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
// src/components/common/autocomplete.rs
|
// src/components/common/autocomplete.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::pages::form::FormState;
|
|
||||||
use common::proto::komp_ac::search::search_response::Hit;
|
use common::proto::komp_ac::search::search_response::Hit;
|
||||||
|
use crate::pages::forms::FormState;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::Rect,
|
layout::Rect,
|
||||||
style::{Color, Modifier, Style},
|
style::{Color, Modifier, Style},
|
||||||
|
|||||||
@@ -1,4 +0,0 @@
|
|||||||
// src/components/form.rs
|
|
||||||
pub mod form;
|
|
||||||
|
|
||||||
pub use form::*;
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// src/components/handlers.rs
|
|
||||||
pub mod sidebar;
|
|
||||||
pub mod buffer_list;
|
|
||||||
|
|
||||||
pub use sidebar::*;
|
|
||||||
pub use buffer_list::*;
|
|
||||||
@@ -1,4 +0,0 @@
|
|||||||
// src/components/intro.rs
|
|
||||||
pub mod intro;
|
|
||||||
|
|
||||||
pub use intro::*;
|
|
||||||
@@ -1,16 +1,7 @@
|
|||||||
// src/components/mod.rs
|
// src/components/mod.rs
|
||||||
pub mod handlers;
|
|
||||||
pub mod intro;
|
|
||||||
pub mod admin;
|
|
||||||
pub mod common;
|
pub mod common;
|
||||||
pub mod form;
|
|
||||||
pub mod auth;
|
|
||||||
pub mod utils;
|
pub mod utils;
|
||||||
|
|
||||||
pub use handlers::*;
|
|
||||||
pub use intro::*;
|
|
||||||
pub use admin::*;
|
|
||||||
pub use common::*;
|
pub use common::*;
|
||||||
pub use form::*;
|
|
||||||
pub use auth::*;
|
|
||||||
pub use utils::*;
|
pub use utils::*;
|
||||||
|
|||||||
@@ -148,19 +148,17 @@ impl Config {
|
|||||||
/// Context-aware keybinding resolution
|
/// Context-aware keybinding resolution
|
||||||
pub fn get_action_for_current_context(
|
pub fn get_action_for_current_context(
|
||||||
&self,
|
&self,
|
||||||
is_edit_mode: bool,
|
|
||||||
command_mode: bool,
|
command_mode: bool,
|
||||||
key: KeyCode,
|
key: KeyCode,
|
||||||
modifiers: KeyModifiers
|
modifiers: KeyModifiers
|
||||||
) -> Option<&str> {
|
) -> Option<&str> {
|
||||||
match (command_mode, is_edit_mode) {
|
if command_mode {
|
||||||
(true, _) => self.get_command_action_for_key(key, modifiers),
|
self.get_command_action_for_key(key, modifiers)
|
||||||
(_, true) => self.get_edit_action_for_key(key, modifiers)
|
} else {
|
||||||
.or_else(|| self.get_common_action(key, modifiers)),
|
// fallback: read-only + common + global
|
||||||
_ => self.get_read_only_action_for_key(key, modifiers)
|
self.get_read_only_action_for_key(key, modifiers)
|
||||||
.or_else(|| self.get_common_action(key, modifiers))
|
.or_else(|| self.get_common_action(key, modifiers))
|
||||||
// Add global bindings check for read-only mode
|
.or_else(|| self.get_action_for_key_in_mode(&self.keybindings.global, key, modifiers))
|
||||||
.or_else(|| self.get_action_for_key_in_mode(&self.keybindings.global, key, modifiers)),
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
85
client/src/dialog/functions.rs
Normal file
85
client/src/dialog/functions.rs
Normal file
@@ -0,0 +1,85 @@
|
|||||||
|
// src/dialog/functions.rs
|
||||||
|
|
||||||
|
use crate::dialog::DialogState;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
|
|
||||||
|
impl AppState {
|
||||||
|
pub fn show_dialog(
|
||||||
|
&mut self,
|
||||||
|
title: &str,
|
||||||
|
message: &str,
|
||||||
|
buttons: Vec<String>,
|
||||||
|
purpose: DialogPurpose,
|
||||||
|
) {
|
||||||
|
self.ui.dialog.dialog_title = title.to_string();
|
||||||
|
self.ui.dialog.dialog_message = message.to_string();
|
||||||
|
self.ui.dialog.dialog_buttons = buttons;
|
||||||
|
self.ui.dialog.dialog_active_button_index = 0;
|
||||||
|
self.ui.dialog.purpose = Some(purpose);
|
||||||
|
self.ui.dialog.is_loading = false;
|
||||||
|
self.ui.dialog.dialog_show = true;
|
||||||
|
self.ui.focus_outside_canvas = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn show_loading_dialog(&mut self, title: &str, message: &str) {
|
||||||
|
self.ui.dialog.dialog_title = title.to_string();
|
||||||
|
self.ui.dialog.dialog_message = message.to_string();
|
||||||
|
self.ui.dialog.dialog_buttons.clear();
|
||||||
|
self.ui.dialog.dialog_active_button_index = 0;
|
||||||
|
self.ui.dialog.purpose = None;
|
||||||
|
self.ui.dialog.is_loading = true;
|
||||||
|
self.ui.dialog.dialog_show = true;
|
||||||
|
self.ui.focus_outside_canvas = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn update_dialog_content(
|
||||||
|
&mut self,
|
||||||
|
message: &str,
|
||||||
|
buttons: Vec<String>,
|
||||||
|
purpose: DialogPurpose,
|
||||||
|
) {
|
||||||
|
if self.ui.dialog.dialog_show {
|
||||||
|
self.ui.dialog.dialog_message = message.to_string();
|
||||||
|
self.ui.dialog.dialog_buttons = buttons;
|
||||||
|
self.ui.dialog.dialog_active_button_index = 0;
|
||||||
|
self.ui.dialog.purpose = Some(purpose);
|
||||||
|
self.ui.dialog.is_loading = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn hide_dialog(&mut self) {
|
||||||
|
self.ui.dialog.dialog_show = false;
|
||||||
|
self.ui.dialog.dialog_title.clear();
|
||||||
|
self.ui.dialog.dialog_message.clear();
|
||||||
|
self.ui.dialog.dialog_buttons.clear();
|
||||||
|
self.ui.dialog.dialog_active_button_index = 0;
|
||||||
|
self.ui.dialog.purpose = None;
|
||||||
|
self.ui.focus_outside_canvas = false;
|
||||||
|
self.ui.dialog.is_loading = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn next_dialog_button(&mut self) {
|
||||||
|
if !self.ui.dialog.dialog_buttons.is_empty() {
|
||||||
|
let next_index = (self.ui.dialog.dialog_active_button_index + 1)
|
||||||
|
% self.ui.dialog.dialog_buttons.len();
|
||||||
|
self.ui.dialog.dialog_active_button_index = next_index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn previous_dialog_button(&mut self) {
|
||||||
|
if !self.ui.dialog.dialog_buttons.is_empty() {
|
||||||
|
let len = self.ui.dialog.dialog_buttons.len();
|
||||||
|
let prev_index =
|
||||||
|
(self.ui.dialog.dialog_active_button_index + len - 1) % len;
|
||||||
|
self.ui.dialog.dialog_active_button_index = prev_index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_active_dialog_button_label(&self) -> Option<&str> {
|
||||||
|
self.ui.dialog
|
||||||
|
.dialog_buttons
|
||||||
|
.get(self.ui.dialog.dialog_active_button_index)
|
||||||
|
.map(|s| s.as_str())
|
||||||
|
}
|
||||||
|
}
|
||||||
207
client/src/dialog/logic.rs
Normal file
207
client/src/dialog/logic.rs
Normal file
@@ -0,0 +1,207 @@
|
|||||||
|
// src/dialog/logic.rs
|
||||||
|
|
||||||
|
// TODO(dialog-refactor):
|
||||||
|
// Currently this module (`handle_dialog_event`) contains page-specific logic
|
||||||
|
// (e.g. Login, Register, Admin, SaveTable). This couples the dialog crate
|
||||||
|
// to application pages and business logic.
|
||||||
|
//
|
||||||
|
// Refactor plan:
|
||||||
|
// 1. Keep dialog generic: only handle navigation (next/prev/select) and return
|
||||||
|
// a `DialogResult` (Dismissed | Selected { purpose, index }).
|
||||||
|
// 2. Move all page-specific actions (e.g. login::back_to_main, register::back_to_login,
|
||||||
|
// handle_delete_selected_columns, buffer_state.update_history) into the
|
||||||
|
// respective page or event handler (e.g. modes/handlers/event.rs).
|
||||||
|
// 3. Dialog crate should only provide state, rendering, and generic navigation.
|
||||||
|
// Pages decide what to do when a dialog button is pressed.
|
||||||
|
|
||||||
|
use crossterm::event::{Event, KeyCode};
|
||||||
|
use crate::config::binds::config::Config;
|
||||||
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::buffer::AppView;
|
||||||
|
use crate::buffer::state::BufferState;
|
||||||
|
use crate::modes::handlers::event::EventOutcome;
|
||||||
|
use crate::pages::register;
|
||||||
|
use crate::pages::login;
|
||||||
|
use crate::pages::admin_panel::add_table::logic::handle_delete_selected_columns;
|
||||||
|
use crate::pages::routing::{Router, Page};
|
||||||
|
use anyhow::Result;
|
||||||
|
|
||||||
|
/// Handles key events specifically when a dialog is active.
|
||||||
|
/// Returns Some(Result<EventOutcome, Error>) if the event was handled (consumed),
|
||||||
|
/// otherwise returns None.
|
||||||
|
pub async fn handle_dialog_event(
|
||||||
|
event: &Event,
|
||||||
|
config: &Config,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
buffer_state: &mut BufferState,
|
||||||
|
router: &mut Router,
|
||||||
|
) -> Option<Result<EventOutcome>> {
|
||||||
|
if let Event::Key(key) = event {
|
||||||
|
// Always allow Esc to dismiss
|
||||||
|
if key.code == KeyCode::Esc {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok("Dialog dismissed".to_string())));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check general bindings for dialog actions
|
||||||
|
if let Some(action) = config.get_general_action(key.code, key.modifiers) {
|
||||||
|
match action {
|
||||||
|
"move_down" | "next_option" => {
|
||||||
|
let current_index = app_state.ui.dialog.dialog_active_button_index;
|
||||||
|
let num_buttons = app_state.ui.dialog.dialog_buttons.len();
|
||||||
|
if num_buttons > 0 && current_index < num_buttons - 1 {
|
||||||
|
app_state.ui.dialog.dialog_active_button_index += 1;
|
||||||
|
}
|
||||||
|
return Some(Ok(EventOutcome::Ok(String::new())));
|
||||||
|
}
|
||||||
|
"move_up" | "previous_option" => {
|
||||||
|
let current_index = app_state.ui.dialog.dialog_active_button_index;
|
||||||
|
if current_index > 0 {
|
||||||
|
app_state.ui.dialog.dialog_active_button_index -= 1;
|
||||||
|
}
|
||||||
|
return Some(Ok(EventOutcome::Ok(String::new())));
|
||||||
|
}
|
||||||
|
"select" => {
|
||||||
|
let selected_index = app_state.ui.dialog.dialog_active_button_index;
|
||||||
|
let purpose = match app_state.ui.dialog.purpose {
|
||||||
|
Some(p) => p,
|
||||||
|
None => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Internal Error: Dialog context lost".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// Handle Dialog Actions Directly Here
|
||||||
|
match purpose {
|
||||||
|
DialogPurpose::LoginSuccess => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "Menu" button selected
|
||||||
|
app_state.hide_dialog();
|
||||||
|
if let Page::Login(state) = &mut router.current {
|
||||||
|
let message =
|
||||||
|
login::back_to_main(state, app_state, buffer_state).await;
|
||||||
|
return Some(Ok(EventOutcome::Ok(message)));
|
||||||
|
}
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Login state not active".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
1 => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok("Exiting dialog".to_string())));
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Unknown dialog button selected".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
DialogPurpose::LoginFailed => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "OK" button selected
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Login failed dialog dismissed".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Unknown dialog button selected".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
DialogPurpose::RegisterSuccess => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "OK" button for RegisterSuccess
|
||||||
|
app_state.hide_dialog();
|
||||||
|
if let Page::Register(state) = &mut router.current {
|
||||||
|
let message =
|
||||||
|
register::back_to_login(state, app_state, buffer_state)
|
||||||
|
.await;
|
||||||
|
return Some(Ok(EventOutcome::Ok(message)));
|
||||||
|
}
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Register state not active".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Unknown dialog button selected".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
DialogPurpose::RegisterFailed => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "OK" button for RegisterFailed
|
||||||
|
app_state.hide_dialog(); // Just dismiss
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Register failed dialog dismissed".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Unknown dialog button selected".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
},
|
||||||
|
DialogPurpose::ConfirmDeleteColumns => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "Confirm" button selected
|
||||||
|
if let Page::Admin(state) = &mut router.current {
|
||||||
|
let outcome_message =
|
||||||
|
handle_delete_selected_columns(&mut state.add_table_state);
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok(outcome_message)));
|
||||||
|
}
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Admin state not active".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
1 => {
|
||||||
|
// "Cancel" button selected
|
||||||
|
app_state.hide_dialog();
|
||||||
|
return Some(Ok(EventOutcome::Ok("Deletion cancelled.".to_string())));
|
||||||
|
}
|
||||||
|
_ => { /* Handle unexpected index */ }
|
||||||
|
},
|
||||||
|
DialogPurpose::SaveTableSuccess => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "OK" button selected
|
||||||
|
app_state.hide_dialog();
|
||||||
|
buffer_state.update_history(AppView::Admin); // Navigate back
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Save success dialog dismissed.".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
_ => { /* Handle unexpected index */ }
|
||||||
|
},
|
||||||
|
DialogPurpose::SaveLogicSuccess => match selected_index {
|
||||||
|
0 => {
|
||||||
|
// "OK" button selected
|
||||||
|
app_state.hide_dialog();
|
||||||
|
buffer_state.update_history(AppView::Admin);
|
||||||
|
return Some(Ok(EventOutcome::Ok(
|
||||||
|
"Save success dialog dismissed.".to_string(),
|
||||||
|
)));
|
||||||
|
}
|
||||||
|
_ => { /* Handle unexpected index */ }
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {} // Ignore other general actions when dialog is shown
|
||||||
|
}
|
||||||
|
}
|
||||||
|
// If it was a key event but not handled above, consume it
|
||||||
|
Some(Ok(EventOutcome::Ok(String::new())))
|
||||||
|
} else {
|
||||||
|
// If it wasn't a key event, consume it too while dialog is active
|
||||||
|
Some(Ok(EventOutcome::Ok(String::new())))
|
||||||
|
}
|
||||||
|
}
|
||||||
10
client/src/dialog/mod.rs
Normal file
10
client/src/dialog/mod.rs
Normal file
@@ -0,0 +1,10 @@
|
|||||||
|
// src/dialog/mod.rs
|
||||||
|
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
pub mod state;
|
||||||
|
pub mod functions;
|
||||||
|
|
||||||
|
pub use ui::render_dialog;
|
||||||
|
pub use logic::handle_dialog_event;
|
||||||
|
pub use state::DialogState;
|
||||||
26
client/src/dialog/state.rs
Normal file
26
client/src/dialog/state.rs
Normal file
@@ -0,0 +1,26 @@
|
|||||||
|
// src/dialog/state.rs
|
||||||
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
|
|
||||||
|
pub struct DialogState {
|
||||||
|
pub dialog_show: bool,
|
||||||
|
pub dialog_title: String,
|
||||||
|
pub dialog_message: String,
|
||||||
|
pub dialog_buttons: Vec<String>,
|
||||||
|
pub dialog_active_button_index: usize,
|
||||||
|
pub purpose: Option<DialogPurpose>,
|
||||||
|
pub is_loading: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for DialogState {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
dialog_show: false,
|
||||||
|
dialog_title: String::new(),
|
||||||
|
dialog_message: String::new(),
|
||||||
|
dialog_buttons: Vec::new(),
|
||||||
|
dialog_active_button_index: 0,
|
||||||
|
purpose: None,
|
||||||
|
is_loading: false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,3 +1,5 @@
|
|||||||
|
// src/dialog/ui.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Constraint, Direction, Layout, Margin, Rect},
|
layout::{Constraint, Direction, Layout, Margin, Rect},
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
// src/functions/common.rs
|
|
||||||
|
|
||||||
pub mod buffer;
|
|
||||||
|
|
||||||
pub use buffer::*;
|
|
||||||
@@ -1,6 +0,0 @@
|
|||||||
// src/functions/mod.rs
|
|
||||||
|
|
||||||
pub mod common;
|
|
||||||
pub mod modes;
|
|
||||||
|
|
||||||
pub use modes::*;
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
// src/functions/modes.rs
|
|
||||||
|
|
||||||
pub mod navigation;
|
|
||||||
|
|
||||||
pub use navigation::*;
|
|
||||||
@@ -1,5 +0,0 @@
|
|||||||
// src/functions/modes/navigation.rs
|
|
||||||
|
|
||||||
pub mod admin_nav;
|
|
||||||
pub mod add_table_nav;
|
|
||||||
pub mod add_logic_nav;
|
|
||||||
@@ -1,440 +0,0 @@
|
|||||||
// src/functions/modes/navigation/add_logic_nav.rs
|
|
||||||
use crate::config::binds::config::{Config, EditorKeybindingMode};
|
|
||||||
use crate::state::{
|
|
||||||
app::state::AppState,
|
|
||||||
pages::add_logic::{AddLogicFocus, AddLogicState},
|
|
||||||
app::buffer::AppView,
|
|
||||||
app::buffer::BufferState,
|
|
||||||
};
|
|
||||||
use crossterm::event::{KeyEvent, KeyCode, KeyModifiers};
|
|
||||||
use crate::services::GrpcClient;
|
|
||||||
use tokio::sync::mpsc;
|
|
||||||
use anyhow::Result;
|
|
||||||
use crate::components::common::text_editor::TextEditor;
|
|
||||||
use crate::services::ui_service::UiService;
|
|
||||||
use tui_textarea::CursorMove; // Ensure this import is present
|
|
||||||
|
|
||||||
pub type SaveLogicResultSender = mpsc::Sender<Result<String>>;
|
|
||||||
|
|
||||||
pub fn handle_add_logic_navigation(
|
|
||||||
key_event: KeyEvent,
|
|
||||||
config: &Config,
|
|
||||||
app_state: &mut AppState,
|
|
||||||
add_logic_state: &mut AddLogicState,
|
|
||||||
is_edit_mode: &mut bool,
|
|
||||||
buffer_state: &mut BufferState,
|
|
||||||
grpc_client: GrpcClient,
|
|
||||||
_save_logic_sender: SaveLogicResultSender, // Marked as unused
|
|
||||||
command_message: &mut String,
|
|
||||||
) -> bool {
|
|
||||||
// === FULLSCREEN SCRIPT EDITING - COMPLETE ISOLATION ===
|
|
||||||
if add_logic_state.current_focus == AddLogicFocus::InsideScriptContent {
|
|
||||||
// === AUTOCOMPLETE HANDLING ===
|
|
||||||
if add_logic_state.script_editor_autocomplete_active {
|
|
||||||
match key_event.code {
|
|
||||||
// ... (Char, Backspace, Tab, Down, Up cases remain the same) ...
|
|
||||||
KeyCode::Char(c) if c.is_alphanumeric() || c == '_' => {
|
|
||||||
add_logic_state.script_editor_filter_text.push(c);
|
|
||||||
add_logic_state.update_script_editor_suggestions();
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
*command_message = format!("Filtering: @{}", add_logic_state.script_editor_filter_text);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
KeyCode::Backspace => {
|
|
||||||
if !add_logic_state.script_editor_filter_text.is_empty() {
|
|
||||||
add_logic_state.script_editor_filter_text.pop();
|
|
||||||
add_logic_state.update_script_editor_suggestions();
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
*command_message = if add_logic_state.script_editor_filter_text.is_empty() {
|
|
||||||
"Autocomplete: @".to_string()
|
|
||||||
} else {
|
|
||||||
format!("Filtering: @{}", add_logic_state.script_editor_filter_text)
|
|
||||||
};
|
|
||||||
} else {
|
|
||||||
let should_deactivate = if let Some((trigger_line, trigger_col)) = add_logic_state.script_editor_trigger_position {
|
|
||||||
let current_cursor = {
|
|
||||||
let editor_borrow = add_logic_state.script_content_editor.borrow();
|
|
||||||
editor_borrow.cursor()
|
|
||||||
};
|
|
||||||
current_cursor.0 == trigger_line && current_cursor.1 == trigger_col + 1
|
|
||||||
} else {
|
|
||||||
false
|
|
||||||
};
|
|
||||||
if should_deactivate {
|
|
||||||
add_logic_state.deactivate_script_editor_autocomplete();
|
|
||||||
*command_message = "Autocomplete cancelled".to_string();
|
|
||||||
}
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
KeyCode::Tab | KeyCode::Down => {
|
|
||||||
if !add_logic_state.script_editor_suggestions.is_empty() {
|
|
||||||
let current = add_logic_state.script_editor_selected_suggestion_index.unwrap_or(0);
|
|
||||||
let next = (current + 1) % add_logic_state.script_editor_suggestions.len();
|
|
||||||
add_logic_state.script_editor_selected_suggestion_index = Some(next);
|
|
||||||
*command_message = format!("Selected: {}", add_logic_state.script_editor_suggestions[next]);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
KeyCode::Up => {
|
|
||||||
if !add_logic_state.script_editor_suggestions.is_empty() {
|
|
||||||
let current = add_logic_state.script_editor_selected_suggestion_index.unwrap_or(0);
|
|
||||||
let prev = if current == 0 {
|
|
||||||
add_logic_state.script_editor_suggestions.len() - 1
|
|
||||||
} else {
|
|
||||||
current - 1
|
|
||||||
};
|
|
||||||
add_logic_state.script_editor_selected_suggestion_index = Some(prev);
|
|
||||||
*command_message = format!("Selected: {}", add_logic_state.script_editor_suggestions[prev]);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
KeyCode::Enter => {
|
|
||||||
if let Some(selected_idx) = add_logic_state.script_editor_selected_suggestion_index {
|
|
||||||
if let Some(suggestion) = add_logic_state.script_editor_suggestions.get(selected_idx).cloned() {
|
|
||||||
let trigger_pos = add_logic_state.script_editor_trigger_position;
|
|
||||||
let filter_len = add_logic_state.script_editor_filter_text.len();
|
|
||||||
|
|
||||||
add_logic_state.deactivate_script_editor_autocomplete();
|
|
||||||
add_logic_state.has_unsaved_changes = true;
|
|
||||||
|
|
||||||
if let Some(pos) = trigger_pos {
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
|
|
||||||
if suggestion == "sql" {
|
|
||||||
replace_autocomplete_text(&mut editor_borrow, pos, filter_len, "sql");
|
|
||||||
editor_borrow.insert_str("('')");
|
|
||||||
// Move cursor back twice to be between the single quotes
|
|
||||||
editor_borrow.move_cursor(CursorMove::Back); // Before ')'
|
|
||||||
editor_borrow.move_cursor(CursorMove::Back); // Before ''' (inside '')
|
|
||||||
*command_message = "Inserted: @sql('')".to_string();
|
|
||||||
} else {
|
|
||||||
let is_table_selection = add_logic_state.is_table_name_suggestion(&suggestion);
|
|
||||||
replace_autocomplete_text(&mut editor_borrow, pos, filter_len, &suggestion);
|
|
||||||
|
|
||||||
if is_table_selection {
|
|
||||||
editor_borrow.insert_str(".");
|
|
||||||
let new_cursor = editor_borrow.cursor();
|
|
||||||
drop(editor_borrow); // Release borrow before calling add_logic_state methods
|
|
||||||
|
|
||||||
add_logic_state.script_editor_trigger_position = Some(new_cursor);
|
|
||||||
add_logic_state.script_editor_autocomplete_active = true;
|
|
||||||
add_logic_state.script_editor_filter_text.clear();
|
|
||||||
add_logic_state.trigger_column_autocomplete_for_table(suggestion.clone());
|
|
||||||
|
|
||||||
let profile_name = add_logic_state.profile_name.clone();
|
|
||||||
let table_name_for_fetch = suggestion.clone();
|
|
||||||
let mut client_clone = grpc_client.clone();
|
|
||||||
tokio::spawn(async move {
|
|
||||||
match UiService::fetch_columns_for_table(&mut client_clone, &profile_name, &table_name_for_fetch).await {
|
|
||||||
Ok(_columns) => {
|
|
||||||
// Result handled by main UI loop
|
|
||||||
}
|
|
||||||
Err(e) => {
|
|
||||||
tracing::error!("Failed to fetch columns for {}.{}: {}", profile_name, table_name_for_fetch, e);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
*command_message = format!("Selected table '{}', fetching columns...", suggestion);
|
|
||||||
} else {
|
|
||||||
*command_message = format!("Inserted: {}", suggestion);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
add_logic_state.deactivate_script_editor_autocomplete();
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
KeyCode::Esc => {
|
|
||||||
add_logic_state.deactivate_script_editor_autocomplete();
|
|
||||||
*command_message = "Autocomplete cancelled".to_string();
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
add_logic_state.deactivate_script_editor_autocomplete();
|
|
||||||
*command_message = "Autocomplete cancelled".to_string();
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if key_event.code == KeyCode::Char('@') && key_event.modifiers == KeyModifiers::NONE {
|
|
||||||
let should_trigger = match add_logic_state.editor_keybinding_mode {
|
|
||||||
EditorKeybindingMode::Vim => *is_edit_mode,
|
|
||||||
_ => true,
|
|
||||||
};
|
|
||||||
if should_trigger {
|
|
||||||
let cursor_before = {
|
|
||||||
let editor_borrow = add_logic_state.script_content_editor.borrow();
|
|
||||||
editor_borrow.cursor()
|
|
||||||
};
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
add_logic_state.script_editor_trigger_position = Some(cursor_before);
|
|
||||||
add_logic_state.script_editor_autocomplete_active = true;
|
|
||||||
add_logic_state.script_editor_filter_text.clear();
|
|
||||||
add_logic_state.update_script_editor_suggestions();
|
|
||||||
add_logic_state.has_unsaved_changes = true;
|
|
||||||
*command_message = "Autocomplete: @ (Tab/↑↓ to navigate, Enter to select, Esc to cancel)".to_string();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if key_event.code == KeyCode::Esc && key_event.modifiers == KeyModifiers::NONE {
|
|
||||||
match add_logic_state.editor_keybinding_mode {
|
|
||||||
EditorKeybindingMode::Vim => {
|
|
||||||
if *is_edit_mode {
|
|
||||||
{
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
if TextEditor::is_vim_normal_mode(&add_logic_state.vim_state) {
|
|
||||||
*is_edit_mode = false;
|
|
||||||
*command_message = "VIM: Normal Mode. Esc again to exit script.".to_string();
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
add_logic_state.current_focus = AddLogicFocus::ScriptContentPreview;
|
|
||||||
app_state.ui.focus_outside_canvas = true;
|
|
||||||
*is_edit_mode = false;
|
|
||||||
*command_message = "Exited script editing.".to_string();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
if *is_edit_mode {
|
|
||||||
*is_edit_mode = false;
|
|
||||||
*command_message = "Exited script edit. Esc again to exit script.".to_string();
|
|
||||||
} else {
|
|
||||||
add_logic_state.current_focus = AddLogicFocus::ScriptContentPreview;
|
|
||||||
app_state.ui.focus_outside_canvas = true;
|
|
||||||
*is_edit_mode = false;
|
|
||||||
*command_message = "Exited script editing.".to_string();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
let changed = {
|
|
||||||
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
|
||||||
TextEditor::handle_input(
|
|
||||||
&mut editor_borrow,
|
|
||||||
key_event,
|
|
||||||
&add_logic_state.editor_keybinding_mode,
|
|
||||||
&mut add_logic_state.vim_state,
|
|
||||||
)
|
|
||||||
};
|
|
||||||
if changed {
|
|
||||||
add_logic_state.has_unsaved_changes = true;
|
|
||||||
}
|
|
||||||
if add_logic_state.editor_keybinding_mode == EditorKeybindingMode::Vim {
|
|
||||||
*is_edit_mode = !TextEditor::is_vim_normal_mode(&add_logic_state.vim_state);
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
let action = config.get_general_action(key_event.code, key_event.modifiers);
|
|
||||||
let current_focus = add_logic_state.current_focus;
|
|
||||||
let mut handled = true;
|
|
||||||
let mut new_focus = current_focus;
|
|
||||||
|
|
||||||
match action.as_deref() {
|
|
||||||
Some("exit_table_scroll") => {
|
|
||||||
handled = false;
|
|
||||||
}
|
|
||||||
Some("move_up") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName => {}
|
|
||||||
AddLogicFocus::InputTargetColumn => new_focus = AddLogicFocus::InputLogicName,
|
|
||||||
AddLogicFocus::InputDescription => new_focus = AddLogicFocus::InputTargetColumn,
|
|
||||||
AddLogicFocus::ScriptContentPreview => new_focus = AddLogicFocus::InputDescription,
|
|
||||||
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::ScriptContentPreview,
|
|
||||||
AddLogicFocus::CancelButton => new_focus = AddLogicFocus::SaveButton,
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some("move_down") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName => new_focus = AddLogicFocus::InputTargetColumn,
|
|
||||||
AddLogicFocus::InputTargetColumn => new_focus = AddLogicFocus::InputDescription,
|
|
||||||
AddLogicFocus::InputDescription => {
|
|
||||||
add_logic_state.last_canvas_field = 2;
|
|
||||||
new_focus = AddLogicFocus::ScriptContentPreview;
|
|
||||||
},
|
|
||||||
AddLogicFocus::ScriptContentPreview => new_focus = AddLogicFocus::SaveButton,
|
|
||||||
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::CancelButton,
|
|
||||||
AddLogicFocus::CancelButton => {}
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some("next_option") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName | AddLogicFocus::InputTargetColumn | AddLogicFocus::InputDescription =>
|
|
||||||
{ new_focus = AddLogicFocus::ScriptContentPreview; }
|
|
||||||
AddLogicFocus::ScriptContentPreview => new_focus = AddLogicFocus::SaveButton,
|
|
||||||
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::CancelButton,
|
|
||||||
AddLogicFocus::CancelButton => { }
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some("previous_option") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName | AddLogicFocus::InputTargetColumn | AddLogicFocus::InputDescription =>
|
|
||||||
{ }
|
|
||||||
AddLogicFocus::ScriptContentPreview => new_focus = AddLogicFocus::InputDescription,
|
|
||||||
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::ScriptContentPreview,
|
|
||||||
AddLogicFocus::CancelButton => new_focus = AddLogicFocus::SaveButton,
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some("next_field") => {
|
|
||||||
new_focus = match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName => AddLogicFocus::InputTargetColumn,
|
|
||||||
AddLogicFocus::InputTargetColumn => AddLogicFocus::InputDescription,
|
|
||||||
AddLogicFocus::InputDescription => AddLogicFocus::ScriptContentPreview,
|
|
||||||
AddLogicFocus::ScriptContentPreview => AddLogicFocus::SaveButton,
|
|
||||||
AddLogicFocus::SaveButton => AddLogicFocus::CancelButton,
|
|
||||||
AddLogicFocus::CancelButton => AddLogicFocus::InputLogicName,
|
|
||||||
_ => current_focus,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
Some("prev_field") => {
|
|
||||||
new_focus = match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName => AddLogicFocus::CancelButton,
|
|
||||||
AddLogicFocus::InputTargetColumn => AddLogicFocus::InputLogicName,
|
|
||||||
AddLogicFocus::InputDescription => AddLogicFocus::InputTargetColumn,
|
|
||||||
AddLogicFocus::ScriptContentPreview => AddLogicFocus::InputDescription,
|
|
||||||
AddLogicFocus::SaveButton => AddLogicFocus::ScriptContentPreview,
|
|
||||||
AddLogicFocus::CancelButton => AddLogicFocus::SaveButton,
|
|
||||||
_ => current_focus,
|
|
||||||
};
|
|
||||||
}
|
|
||||||
Some("select") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::ScriptContentPreview => {
|
|
||||||
new_focus = AddLogicFocus::InsideScriptContent;
|
|
||||||
*is_edit_mode = false;
|
|
||||||
app_state.ui.focus_outside_canvas = false;
|
|
||||||
let mode_hint = match add_logic_state.editor_keybinding_mode {
|
|
||||||
EditorKeybindingMode::Vim => "VIM mode - 'i'/'a'/'o' to edit",
|
|
||||||
_ => "Enter/Ctrl+E to edit",
|
|
||||||
};
|
|
||||||
*command_message = format!("Fullscreen script editing. {} or Esc to exit.", mode_hint);
|
|
||||||
}
|
|
||||||
AddLogicFocus::SaveButton => {
|
|
||||||
*command_message = "Save logic action".to_string();
|
|
||||||
}
|
|
||||||
AddLogicFocus::CancelButton => {
|
|
||||||
buffer_state.update_history(AppView::Admin);
|
|
||||||
app_state.ui.show_add_logic = false;
|
|
||||||
*command_message = "Cancelled Add Logic".to_string();
|
|
||||||
*is_edit_mode = false;
|
|
||||||
}
|
|
||||||
AddLogicFocus::InputLogicName | AddLogicFocus::InputTargetColumn | AddLogicFocus::InputDescription => {
|
|
||||||
*is_edit_mode = !*is_edit_mode;
|
|
||||||
*command_message = format!("Field edit mode: {}", if *is_edit_mode { "ON" } else { "OFF" });
|
|
||||||
}
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some("toggle_edit_mode") => {
|
|
||||||
match current_focus {
|
|
||||||
AddLogicFocus::InputLogicName | AddLogicFocus::InputTargetColumn | AddLogicFocus::InputDescription => {
|
|
||||||
*is_edit_mode = !*is_edit_mode;
|
|
||||||
*command_message = format!("Canvas field edit mode: {}", if *is_edit_mode { "ON" } else { "OFF" });
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
*command_message = "Cannot toggle edit mode here.".to_string();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => handled = false,
|
|
||||||
}
|
|
||||||
|
|
||||||
if handled && current_focus != new_focus {
|
|
||||||
add_logic_state.current_focus = new_focus;
|
|
||||||
let new_is_canvas_input_focus = matches!(new_focus,
|
|
||||||
AddLogicFocus::InputLogicName | AddLogicFocus::InputTargetColumn | AddLogicFocus::InputDescription
|
|
||||||
);
|
|
||||||
if new_is_canvas_input_focus {
|
|
||||||
*is_edit_mode = false;
|
|
||||||
app_state.ui.focus_outside_canvas = false;
|
|
||||||
} else {
|
|
||||||
app_state.ui.focus_outside_canvas = true;
|
|
||||||
if matches!(new_focus, AddLogicFocus::ScriptContentPreview) {
|
|
||||||
*is_edit_mode = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
handled
|
|
||||||
}
|
|
||||||
|
|
||||||
fn replace_autocomplete_text(
|
|
||||||
editor: &mut tui_textarea::TextArea,
|
|
||||||
trigger_pos: (usize, usize),
|
|
||||||
filter_len: usize,
|
|
||||||
replacement: &str,
|
|
||||||
) {
|
|
||||||
// use tui_textarea::CursorMove; // Already imported at the top of the module
|
|
||||||
let filter_start_pos = (trigger_pos.0, trigger_pos.1 + 1);
|
|
||||||
editor.move_cursor(CursorMove::Jump(filter_start_pos.0 as u16, filter_start_pos.1 as u16));
|
|
||||||
for _ in 0..filter_len {
|
|
||||||
editor.delete_next_char();
|
|
||||||
}
|
|
||||||
editor.insert_str(replacement);
|
|
||||||
}
|
|
||||||
@@ -5,9 +5,15 @@ pub mod config;
|
|||||||
pub mod state;
|
pub mod state;
|
||||||
pub mod components;
|
pub mod components;
|
||||||
pub mod modes;
|
pub mod modes;
|
||||||
pub mod functions;
|
|
||||||
pub mod services;
|
pub mod services;
|
||||||
pub mod utils;
|
pub mod utils;
|
||||||
|
pub mod buffer;
|
||||||
|
pub mod sidebar;
|
||||||
|
pub mod dialog;
|
||||||
|
pub mod search;
|
||||||
|
pub mod bottom_panel;
|
||||||
|
pub mod pages;
|
||||||
|
pub mod movement;
|
||||||
|
|
||||||
pub use ui::run_ui;
|
pub use ui::run_ui;
|
||||||
|
|
||||||
|
|||||||
@@ -1,41 +1,42 @@
|
|||||||
// src/modes/canvas/common_mode.rs
|
// src/modes/canvas/common_mode.rs
|
||||||
|
|
||||||
use crate::tui::terminal::core::TerminalCore;
|
use crate::tui::terminal::core::TerminalCore;
|
||||||
use crate::state::pages::{form::FormState, auth::LoginState, auth::RegisterState, auth::AuthState};
|
use crate::state::pages::auth::AuthState;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::services::grpc_client::GrpcClient;
|
use crate::services::grpc_client::GrpcClient;
|
||||||
use crate::services::auth::AuthClient;
|
use crate::services::auth::AuthClient;
|
||||||
use crate::modes::handlers::event::EventOutcome;
|
use crate::modes::handlers::event::EventOutcome;
|
||||||
use crate::tui::functions::common::form::SaveOutcome;
|
crate::pages::forms::logic::SaveOutcome;
|
||||||
use anyhow::{Context, Result};
|
use anyhow::{Context, Result};
|
||||||
use crate::tui::functions::common::{
|
use crate::tui::functions::common::{
|
||||||
form::{save as form_save, revert as form_revert},
|
form::{save as form_save, revert as form_revert},
|
||||||
login::{save as login_save, revert as login_revert},
|
login::{save as login_save, revert as login_revert},
|
||||||
register::{revert as register_revert},
|
register::{revert as register_revert},
|
||||||
};
|
};
|
||||||
|
use crate::pages::routing::{Router, Page};
|
||||||
|
|
||||||
pub async fn handle_core_action(
|
pub async fn handle_core_action(
|
||||||
action: &str,
|
action: &str,
|
||||||
form_state: &mut FormState,
|
|
||||||
auth_state: &mut AuthState,
|
auth_state: &mut AuthState,
|
||||||
login_state: &mut LoginState,
|
|
||||||
register_state: &mut RegisterState,
|
|
||||||
grpc_client: &mut GrpcClient,
|
grpc_client: &mut GrpcClient,
|
||||||
auth_client: &mut AuthClient,
|
auth_client: &mut AuthClient,
|
||||||
terminal: &mut TerminalCore,
|
terminal: &mut TerminalCore,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
|
router: &mut Router,
|
||||||
) -> Result<EventOutcome> {
|
) -> Result<EventOutcome> {
|
||||||
match action {
|
match action {
|
||||||
"save" => {
|
"save" => {
|
||||||
if app_state.ui.show_login {
|
match &mut router.current {
|
||||||
let message = login_save(auth_state, login_state, auth_client, app_state).await.context("Login save action failed")?;
|
Page::Login(state) => {
|
||||||
|
let message = login_save(auth_state, state, auth_client, app_state)
|
||||||
|
.await
|
||||||
|
.context("Login save action failed")?;
|
||||||
Ok(EventOutcome::Ok(message))
|
Ok(EventOutcome::Ok(message))
|
||||||
} else {
|
}
|
||||||
let save_outcome = form_save(
|
Page::Form(form_state) => {
|
||||||
app_state,
|
let save_outcome = form_save(app_state, form_state, grpc_client)
|
||||||
form_state,
|
.await
|
||||||
grpc_client,
|
.context("Form save action failed")?;
|
||||||
).await.context("Register save action failed")?;
|
|
||||||
let message = match save_outcome {
|
let message = match save_outcome {
|
||||||
SaveOutcome::NoChange => "No changes to save.".to_string(),
|
SaveOutcome::NoChange => "No changes to save.".to_string(),
|
||||||
SaveOutcome::UpdatedExisting => "Entry updated.".to_string(),
|
SaveOutcome::UpdatedExisting => "Entry updated.".to_string(),
|
||||||
@@ -43,44 +44,52 @@ pub async fn handle_core_action(
|
|||||||
};
|
};
|
||||||
Ok(EventOutcome::DataSaved(save_outcome, message))
|
Ok(EventOutcome::DataSaved(save_outcome, message))
|
||||||
}
|
}
|
||||||
},
|
_ => Ok(EventOutcome::Ok("Save not applicable".into())),
|
||||||
|
}
|
||||||
|
}
|
||||||
"force_quit" => {
|
"force_quit" => {
|
||||||
terminal.cleanup()?;
|
terminal.cleanup()?;
|
||||||
Ok(EventOutcome::Exit("Force exiting without saving.".to_string()))
|
Ok(EventOutcome::Exit("Force exiting without saving.".to_string()))
|
||||||
},
|
}
|
||||||
"save_and_quit" => {
|
"save_and_quit" => {
|
||||||
let message = if app_state.ui.show_login {
|
let message = match &mut router.current {
|
||||||
login_save(auth_state, login_state, auth_client, app_state).await.context("Login save n quit action failed")?
|
Page::Login(state) => {
|
||||||
} else {
|
login_save(auth_state, state, auth_client, app_state)
|
||||||
let save_outcome = form_save(
|
.await
|
||||||
app_state,
|
.context("Login save and quit action failed")?
|
||||||
form_state,
|
}
|
||||||
grpc_client,
|
Page::Form(form_state) => {
|
||||||
).await?;
|
let save_outcome = form_save(app_state, form_state, grpc_client).await?;
|
||||||
match save_outcome {
|
match save_outcome {
|
||||||
SaveOutcome::NoChange => "No changes to save.".to_string(),
|
SaveOutcome::NoChange => "No changes to save.".to_string(),
|
||||||
SaveOutcome::UpdatedExisting => "Entry updated.".to_string(),
|
SaveOutcome::UpdatedExisting => "Entry updated.".to_string(),
|
||||||
SaveOutcome::CreatedNew(_) => "New entry created.".to_string(),
|
SaveOutcome::CreatedNew(_) => "New entry created.".to_string(),
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
_ => "Save not applicable".to_string(),
|
||||||
};
|
};
|
||||||
terminal.cleanup()?;
|
terminal.cleanup()?;
|
||||||
Ok(EventOutcome::Exit(format!("{}. Exiting application.", message)))
|
Ok(EventOutcome::Exit(format!("{}. Exiting application.", message)))
|
||||||
},
|
}
|
||||||
"revert" => {
|
"revert" => {
|
||||||
if app_state.ui.show_login {
|
match &mut router.current {
|
||||||
let message = login_revert(login_state, app_state).await;
|
Page::Login(state) => {
|
||||||
Ok(EventOutcome::Ok(message))
|
let message = login_revert(state, app_state).await;
|
||||||
} else if app_state.ui.show_register {
|
|
||||||
let message = register_revert(register_state, app_state).await;
|
|
||||||
Ok(EventOutcome::Ok(message))
|
|
||||||
} else {
|
|
||||||
let message = form_revert(
|
|
||||||
form_state,
|
|
||||||
grpc_client,
|
|
||||||
).await.context("Form revert x action failed")?;
|
|
||||||
Ok(EventOutcome::Ok(message))
|
Ok(EventOutcome::Ok(message))
|
||||||
}
|
}
|
||||||
},
|
Page::Register(state) => {
|
||||||
|
let message = register_revert(state, app_state).await;
|
||||||
|
Ok(EventOutcome::Ok(message))
|
||||||
|
}
|
||||||
|
Page::Form(form_state) => {
|
||||||
|
let message = form_revert(form_state, grpc_client)
|
||||||
|
.await
|
||||||
|
.context("Form revert action failed")?;
|
||||||
|
Ok(EventOutcome::Ok(message))
|
||||||
|
}
|
||||||
|
_ => Ok(EventOutcome::Ok("Revert not applicable".into())),
|
||||||
|
}
|
||||||
|
}
|
||||||
_ => Ok(EventOutcome::Ok(format!("Core action not handled: {}", action))),
|
_ => Ok(EventOutcome::Ok(format!("Core action not handled: {}", action))),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,21 +3,19 @@
|
|||||||
use crossterm::event::{KeyEvent, KeyCode, KeyModifiers};
|
use crossterm::event::{KeyEvent, KeyCode, KeyModifiers};
|
||||||
use crate::config::binds::config::Config;
|
use crate::config::binds::config::Config;
|
||||||
use crate::services::grpc_client::GrpcClient;
|
use crate::services::grpc_client::GrpcClient;
|
||||||
use crate::state::pages::form::FormState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::{app::state::AppState, pages::auth::LoginState, pages::auth::RegisterState};
|
|
||||||
use crate::modes::common::commands::CommandHandler;
|
use crate::modes::common::commands::CommandHandler;
|
||||||
use crate::tui::terminal::core::TerminalCore;
|
use crate::tui::terminal::core::TerminalCore;
|
||||||
use crate::tui::functions::common::form::{save, revert};
|
use crate::pages::forms::logic::{save, revert ,SaveOutcome};
|
||||||
use crate::modes::handlers::event::EventOutcome;
|
use crate::modes::handlers::event::EventOutcome;
|
||||||
use crate::tui::functions::common::form::SaveOutcome;
|
use crate::pages::routing::{Router, Page};
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
pub async fn handle_command_event(
|
pub async fn handle_command_event(
|
||||||
key: KeyEvent,
|
key: KeyEvent,
|
||||||
config: &Config,
|
config: &Config,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
login_state: &LoginState,
|
router: &mut Router,
|
||||||
register_state: &RegisterState,
|
|
||||||
command_input: &mut String,
|
command_input: &mut String,
|
||||||
command_message: &mut String,
|
command_message: &mut String,
|
||||||
grpc_client: &mut GrpcClient,
|
grpc_client: &mut GrpcClient,
|
||||||
@@ -26,20 +24,19 @@ pub async fn handle_command_event(
|
|||||||
current_position: &mut u64,
|
current_position: &mut u64,
|
||||||
total_count: u64,
|
total_count: u64,
|
||||||
) -> Result<EventOutcome> {
|
) -> Result<EventOutcome> {
|
||||||
// Exit command mode (via configurable keybinding)
|
// Exit command mode
|
||||||
if config.is_exit_command_mode(key.code, key.modifiers) {
|
if config.is_exit_command_mode(key.code, key.modifiers) {
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
*command_message = "".to_string();
|
*command_message = "".to_string();
|
||||||
return Ok(EventOutcome::Ok("Exited command mode".to_string()));
|
return Ok(EventOutcome::Ok("Exited command mode".to_string()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Execute command (via configurable keybinding, defaults to Enter)
|
// Execute command
|
||||||
if config.is_command_execute(key.code, key.modifiers) {
|
if config.is_command_execute(key.code, key.modifiers) {
|
||||||
return process_command(
|
return process_command(
|
||||||
config,
|
config,
|
||||||
app_state,
|
app_state,
|
||||||
login_state,
|
router,
|
||||||
register_state,
|
|
||||||
command_input,
|
command_input,
|
||||||
command_message,
|
command_message,
|
||||||
grpc_client,
|
grpc_client,
|
||||||
@@ -47,33 +44,31 @@ pub async fn handle_command_event(
|
|||||||
terminal,
|
terminal,
|
||||||
current_position,
|
current_position,
|
||||||
total_count,
|
total_count,
|
||||||
).await;
|
)
|
||||||
|
.await;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Backspace (via configurable keybinding, defaults to Backspace)
|
// Backspace
|
||||||
if config.is_command_backspace(key.code, key.modifiers) {
|
if config.is_command_backspace(key.code, key.modifiers) {
|
||||||
command_input.pop();
|
command_input.pop();
|
||||||
return Ok(EventOutcome::Ok("".to_string()));
|
return Ok(EventOutcome::Ok("".to_string()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Regular character input - accept any character in command mode
|
// Regular character input
|
||||||
if let KeyCode::Char(c) = key.code {
|
if let KeyCode::Char(c) = key.code {
|
||||||
// Accept regular or shifted characters (e.g., 'a' or 'A')
|
|
||||||
if key.modifiers.is_empty() || key.modifiers == KeyModifiers::SHIFT {
|
if key.modifiers.is_empty() || key.modifiers == KeyModifiers::SHIFT {
|
||||||
command_input.push(c);
|
command_input.push(c);
|
||||||
return Ok(EventOutcome::Ok("".to_string()));
|
return Ok(EventOutcome::Ok("".to_string()));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ignore all other keys
|
|
||||||
Ok(EventOutcome::Ok("".to_string()))
|
Ok(EventOutcome::Ok("".to_string()))
|
||||||
}
|
}
|
||||||
|
|
||||||
async fn process_command(
|
async fn process_command(
|
||||||
config: &Config,
|
config: &Config,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
login_state: &LoginState,
|
router: &mut Router,
|
||||||
register_state: &RegisterState,
|
|
||||||
command_input: &mut String,
|
command_input: &mut String,
|
||||||
command_message: &mut String,
|
command_message: &mut String,
|
||||||
grpc_client: &mut GrpcClient,
|
grpc_client: &mut GrpcClient,
|
||||||
@@ -82,27 +77,18 @@ async fn process_command(
|
|||||||
current_position: &mut u64,
|
current_position: &mut u64,
|
||||||
total_count: u64,
|
total_count: u64,
|
||||||
) -> Result<EventOutcome> {
|
) -> Result<EventOutcome> {
|
||||||
// Clone the trimmed command to avoid borrow issues
|
|
||||||
let command = command_input.trim().to_string();
|
let command = command_input.trim().to_string();
|
||||||
if command.is_empty() {
|
if command.is_empty() {
|
||||||
*command_message = "Empty command".to_string();
|
*command_message = "Empty command".to_string();
|
||||||
return Ok(EventOutcome::Ok(command_message.clone()));
|
return Ok(EventOutcome::Ok(command_message.clone()));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Get the action for the command (now checks global and common bindings too)
|
let action = config.get_action_for_command(&command).unwrap_or("unknown");
|
||||||
let action = config.get_action_for_command(&command)
|
|
||||||
.unwrap_or("unknown");
|
|
||||||
|
|
||||||
match action {
|
match action {
|
||||||
"force_quit" | "save_and_quit" | "quit" => {
|
"force_quit" | "save_and_quit" | "quit" => {
|
||||||
let (should_exit, message) = command_handler
|
let (should_exit, message) = command_handler
|
||||||
.handle_command(
|
.handle_command(action, terminal, app_state, router)
|
||||||
action,
|
|
||||||
terminal,
|
|
||||||
app_state,
|
|
||||||
login_state,
|
|
||||||
register_state,
|
|
||||||
)
|
|
||||||
.await?;
|
.await?;
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
if should_exit {
|
if should_exit {
|
||||||
@@ -110,12 +96,10 @@ async fn process_command(
|
|||||||
} else {
|
} else {
|
||||||
Ok(EventOutcome::Ok(message))
|
Ok(EventOutcome::Ok(message))
|
||||||
}
|
}
|
||||||
},
|
}
|
||||||
"save" => {
|
"save" => {
|
||||||
let outcome = save(
|
if let Page::Form(path) = &router.current {
|
||||||
app_state,
|
let outcome = save(app_state, path, grpc_client).await?;
|
||||||
grpc_client,
|
|
||||||
).await?;
|
|
||||||
let message = match outcome {
|
let message = match outcome {
|
||||||
SaveOutcome::CreatedNew(_) => "New entry created".to_string(),
|
SaveOutcome::CreatedNew(_) => "New entry created".to_string(),
|
||||||
SaveOutcome::UpdatedExisting => "Entry updated".to_string(),
|
SaveOutcome::UpdatedExisting => "Entry updated".to_string(),
|
||||||
@@ -123,15 +107,19 @@ async fn process_command(
|
|||||||
};
|
};
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
Ok(EventOutcome::DataSaved(outcome, message))
|
Ok(EventOutcome::DataSaved(outcome, message))
|
||||||
},
|
} else {
|
||||||
|
Ok(EventOutcome::Ok("Not in a form page".to_string()))
|
||||||
|
}
|
||||||
|
}
|
||||||
"revert" => {
|
"revert" => {
|
||||||
let message = revert(
|
if let Page::Form(path) = &router.current {
|
||||||
app_state,
|
let message = revert(app_state, path, grpc_client).await?;
|
||||||
grpc_client,
|
|
||||||
).await?;
|
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
Ok(EventOutcome::Ok(message))
|
Ok(EventOutcome::Ok(message))
|
||||||
},
|
} else {
|
||||||
|
Ok(EventOutcome::Ok("Not in a form page".to_string()))
|
||||||
|
}
|
||||||
|
}
|
||||||
_ => {
|
_ => {
|
||||||
let message = format!("Unhandled action: {}", action);
|
let message = format!("Unhandled action: {}", action);
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
// src/modes/common/commands.rs
|
// src/modes/common/commands.rs
|
||||||
use crate::tui::terminal::core::TerminalCore;
|
use crate::tui::terminal::core::TerminalCore;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::{form::FormState, auth::LoginState, auth::RegisterState};
|
use crate::pages::routing::{Router, Page};
|
||||||
use anyhow::Result;
|
use anyhow::Result;
|
||||||
|
|
||||||
pub struct CommandHandler;
|
pub struct CommandHandler;
|
||||||
@@ -16,11 +16,10 @@ impl CommandHandler {
|
|||||||
action: &str,
|
action: &str,
|
||||||
terminal: &mut TerminalCore,
|
terminal: &mut TerminalCore,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
login_state: &LoginState,
|
router: &Router,
|
||||||
register_state: &RegisterState,
|
|
||||||
) -> Result<(bool, String)> {
|
) -> Result<(bool, String)> {
|
||||||
match action {
|
match action {
|
||||||
"quit" => self.handle_quit(terminal, app_state, login_state, register_state).await,
|
"quit" => self.handle_quit(terminal, app_state, router).await,
|
||||||
"force_quit" => self.handle_force_quit(terminal).await,
|
"force_quit" => self.handle_force_quit(terminal).await,
|
||||||
"save_and_quit" => self.handle_save_quit(terminal).await,
|
"save_and_quit" => self.handle_save_quit(terminal).await,
|
||||||
_ => Ok((false, format!("Unknown command: {}", action))),
|
_ => Ok((false, format!("Unknown command: {}", action))),
|
||||||
@@ -31,18 +30,17 @@ impl CommandHandler {
|
|||||||
&self,
|
&self,
|
||||||
terminal: &mut TerminalCore,
|
terminal: &mut TerminalCore,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
login_state: &LoginState,
|
router: &Router,
|
||||||
register_state: &RegisterState,
|
|
||||||
) -> Result<(bool, String)> {
|
) -> Result<(bool, String)> {
|
||||||
// Use actual unsaved changes state instead of is_saved flag
|
// Use router to check unsaved changes
|
||||||
let has_unsaved = if app_state.ui.show_login {
|
let has_unsaved = match &router.current {
|
||||||
login_state.has_unsaved_changes()
|
Page::Login(page) => page.state.has_unsaved_changes(),
|
||||||
} else if app_state.ui.show_register {
|
Page::Register(state) => state.has_unsaved_changes(),
|
||||||
register_state.has_unsaved_changes()
|
Page::Form(path) => app_state
|
||||||
} else if let Some(fs) = app_state.form_state_mut() {
|
.form_state_for_path_ref(path)
|
||||||
fs.has_unsaved_changes
|
.map(|fs| fs.has_unsaved_changes())
|
||||||
} else {
|
.unwrap_or(false),
|
||||||
false
|
_ => false,
|
||||||
};
|
};
|
||||||
|
|
||||||
if !has_unsaved {
|
if !has_unsaved {
|
||||||
|
|||||||
@@ -1,4 +1,3 @@
|
|||||||
// src/client/modes/general.rs
|
// src/client/modes/general.rs
|
||||||
pub mod navigation;
|
pub mod navigation;
|
||||||
pub mod dialog;
|
|
||||||
pub mod command_navigation;
|
pub mod command_navigation;
|
||||||
|
|||||||
@@ -82,8 +82,6 @@ impl TableDependencyGraph {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// ... (NavigationState struct and its new(), activate_*, deactivate(), add_char(), remove_char(), move_*, autocomplete_selected(), get_display_input() methods are unchanged) ...
|
|
||||||
pub struct NavigationState {
|
pub struct NavigationState {
|
||||||
pub active: bool,
|
pub active: bool,
|
||||||
pub input: String,
|
pub input: String,
|
||||||
|
|||||||
@@ -1,163 +0,0 @@
|
|||||||
// src/modes/general/dialog.rs
|
|
||||||
|
|
||||||
use crossterm::event::{Event, KeyCode};
|
|
||||||
use crate::config::binds::config::Config;
|
|
||||||
use crate::ui::handlers::context::DialogPurpose;
|
|
||||||
use crate::state::app::{state::AppState, buffer::AppView};
|
|
||||||
use crate::state::app::buffer::BufferState;
|
|
||||||
use crate::state::pages::auth::{LoginState, RegisterState};
|
|
||||||
use crate::state::pages::admin::AdminState;
|
|
||||||
use crate::modes::handlers::event::EventOutcome;
|
|
||||||
use crate::tui::functions::common::{login, register};
|
|
||||||
use crate::tui::functions::common::add_table::handle_delete_selected_columns;
|
|
||||||
use anyhow::Result;
|
|
||||||
|
|
||||||
/// Handles key events specifically when a dialog is active.
|
|
||||||
/// Returns Some(Result<EventOutcome, Error>) if the event was handled (consumed),
|
|
||||||
/// otherwise returns None.
|
|
||||||
pub async fn handle_dialog_event(
|
|
||||||
event: &Event,
|
|
||||||
config: &Config,
|
|
||||||
app_state: &mut AppState,
|
|
||||||
login_state: &mut LoginState,
|
|
||||||
register_state: &mut RegisterState,
|
|
||||||
buffer_state: &mut BufferState,
|
|
||||||
admin_state: &mut AdminState,
|
|
||||||
) -> Option<Result<EventOutcome>> {
|
|
||||||
if let Event::Key(key) = event {
|
|
||||||
// Always allow Esc to dismiss
|
|
||||||
if key.code == KeyCode::Esc {
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Dialog dismissed".to_string())));
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check general bindings for dialog actions
|
|
||||||
if let Some(action) = config.get_general_action(key.code, key.modifiers) {
|
|
||||||
match action {
|
|
||||||
"move_down" | "next_option" => {
|
|
||||||
let current_index = app_state.ui.dialog.dialog_active_button_index;
|
|
||||||
let num_buttons = app_state.ui.dialog.dialog_buttons.len();
|
|
||||||
if num_buttons > 0 && current_index < num_buttons - 1 {
|
|
||||||
app_state.ui.dialog.dialog_active_button_index += 1;
|
|
||||||
}
|
|
||||||
return Some(Ok(EventOutcome::Ok(String::new())));
|
|
||||||
}
|
|
||||||
"move_up" | "previous_option" => {
|
|
||||||
let current_index = app_state.ui.dialog.dialog_active_button_index;
|
|
||||||
if current_index > 0 {
|
|
||||||
app_state.ui.dialog.dialog_active_button_index -= 1;
|
|
||||||
}
|
|
||||||
return Some(Ok(EventOutcome::Ok(String::new())));
|
|
||||||
}
|
|
||||||
"select" => {
|
|
||||||
let selected_index = app_state.ui.dialog.dialog_active_button_index;
|
|
||||||
let purpose = match app_state.ui.dialog.purpose {
|
|
||||||
Some(p) => p,
|
|
||||||
None => {
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Internal Error: Dialog context lost".to_string())));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
// Handle Dialog Actions Directly Here
|
|
||||||
match purpose {
|
|
||||||
DialogPurpose::LoginSuccess => {
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "Menu" button selected
|
|
||||||
app_state.hide_dialog();
|
|
||||||
let message = login::back_to_main(login_state, app_state, buffer_state).await;
|
|
||||||
return Some(Ok(EventOutcome::Ok(message)));
|
|
||||||
}
|
|
||||||
1 => {
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Exiting dialog".to_string())));
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Unknown dialog button selected".to_string())));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::LoginFailed => {
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "OK" button selected
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Login failed dialog dismissed".to_string())));
|
|
||||||
}
|
|
||||||
_ => {
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Unknown dialog button selected".to_string())));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::RegisterSuccess => { // Add this arm
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "OK" button for RegisterSuccess
|
|
||||||
app_state.hide_dialog();
|
|
||||||
let message = register::back_to_login(register_state, app_state, buffer_state).await;
|
|
||||||
return Some(Ok(EventOutcome::Ok(message)));
|
|
||||||
}
|
|
||||||
_ => { // Default for RegisterSuccess
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Unknown dialog button selected".to_string())));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::RegisterFailed => { // Add this arm
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "OK" button for RegisterFailed
|
|
||||||
app_state.hide_dialog(); // Just dismiss
|
|
||||||
return Some(Ok(EventOutcome::Ok("Register failed dialog dismissed".to_string())));
|
|
||||||
}
|
|
||||||
_ => { // Default for RegisterFailed
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Unknown dialog button selected".to_string())));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::ConfirmDeleteColumns => {
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "Confirm" button selected
|
|
||||||
let outcome_message = handle_delete_selected_columns(&mut admin_state.add_table_state);
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok(outcome_message)));
|
|
||||||
}
|
|
||||||
1 => { // "Cancel" button selected
|
|
||||||
app_state.hide_dialog();
|
|
||||||
return Some(Ok(EventOutcome::Ok("Deletion cancelled.".to_string())));
|
|
||||||
}
|
|
||||||
_ => { /* Handle unexpected index */ }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::SaveTableSuccess => {
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "OK" button selected
|
|
||||||
app_state.hide_dialog();
|
|
||||||
buffer_state.update_history(AppView::Admin); // Navigate back
|
|
||||||
return Some(Ok(EventOutcome::Ok("Save success dialog dismissed.".to_string())));
|
|
||||||
}
|
|
||||||
_ => { /* Handle unexpected index */ }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
DialogPurpose::SaveLogicSuccess => {
|
|
||||||
match selected_index {
|
|
||||||
0 => { // "OK" button selected
|
|
||||||
app_state.hide_dialog();
|
|
||||||
buffer_state.update_history(AppView::Admin);
|
|
||||||
return Some(Ok(EventOutcome::Ok("Save success dialog dismissed.".to_string())));
|
|
||||||
}
|
|
||||||
_ => { /* Handle unexpected index */ }
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
_ => {} // Ignore other general actions when dialog is shown
|
|
||||||
}
|
|
||||||
}
|
|
||||||
// If it was a key event but not handled above, consume it
|
|
||||||
Some(Ok(EventOutcome::Ok(String::new())))
|
|
||||||
} else {
|
|
||||||
// If it wasn't a key event, consume it too while dialog is active
|
|
||||||
Some(Ok(EventOutcome::Ok(String::new())))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -3,11 +3,8 @@
|
|||||||
use crossterm::event::KeyEvent;
|
use crossterm::event::KeyEvent;
|
||||||
use crate::config::binds::config::Config;
|
use crate::config::binds::config::Config;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::form::FormState;
|
use crate::pages::routing::{Router, Page};
|
||||||
use crate::state::pages::auth::LoginState;
|
use crate::pages::forms::FormState;
|
||||||
use crate::state::pages::auth::RegisterState;
|
|
||||||
use crate::state::pages::intro::IntroState;
|
|
||||||
use crate::state::pages::admin::AdminState;
|
|
||||||
use crate::ui::handlers::context::UiContext;
|
use crate::ui::handlers::context::UiContext;
|
||||||
use crate::modes::handlers::event::EventOutcome;
|
use crate::modes::handlers::event::EventOutcome;
|
||||||
use crate::modes::general::command_navigation::{handle_command_navigation_event, NavigationState};
|
use crate::modes::general::command_navigation::{handle_command_navigation_event, NavigationState};
|
||||||
@@ -18,10 +15,7 @@ pub async fn handle_navigation_event(
|
|||||||
key: KeyEvent,
|
key: KeyEvent,
|
||||||
config: &Config,
|
config: &Config,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
login_state: &mut LoginState,
|
router: &mut Router,
|
||||||
register_state: &mut RegisterState,
|
|
||||||
intro_state: &mut IntroState,
|
|
||||||
admin_state: &mut AdminState,
|
|
||||||
command_mode: &mut bool,
|
command_mode: &mut bool,
|
||||||
command_input: &mut String,
|
command_input: &mut String,
|
||||||
command_message: &mut String,
|
command_message: &mut String,
|
||||||
@@ -34,32 +28,36 @@ pub async fn handle_navigation_event(
|
|||||||
|
|
||||||
if let Some(action) = config.get_general_action(key.code, key.modifiers) {
|
if let Some(action) = config.get_general_action(key.code, key.modifiers) {
|
||||||
match action {
|
match action {
|
||||||
"move_up" => {
|
"up" => {
|
||||||
move_up(app_state, login_state, register_state, intro_state, admin_state);
|
up(app_state, router);
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"move_down" => {
|
"down" => {
|
||||||
move_down(app_state, intro_state, admin_state);
|
down(app_state, router);
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"next_option" => {
|
"next_option" => {
|
||||||
next_option(app_state, intro_state);
|
next_option(app_state, router);
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"previous_option" => {
|
"previous_option" => {
|
||||||
previous_option(app_state, intro_state);
|
previous_option(app_state, router);
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"next_field" => {
|
"next_field" => {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Page::Form(path) = &router.current {
|
||||||
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
next_field(fs);
|
next_field(fs);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"prev_field" => {
|
"prev_field" => {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Page::Form(path) = &router.current {
|
||||||
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
prev_field(fs);
|
prev_field(fs);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"enter_command_mode" => {
|
"enter_command_mode" => {
|
||||||
@@ -67,18 +65,21 @@ pub async fn handle_navigation_event(
|
|||||||
return Ok(EventOutcome::Ok(String::new()));
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
}
|
}
|
||||||
"select" => {
|
"select" => {
|
||||||
let (context, index) = if app_state.ui.show_intro {
|
let (context, index) = match &router.current {
|
||||||
(UiContext::Intro, intro_state.selected_option)
|
Page::Intro(state) => (UiContext::Intro, state.selected_option),
|
||||||
} else if app_state.ui.show_login && app_state.ui.focus_outside_canvas {
|
Page::Login(_) if app_state.ui.focus_outside_canvas => {
|
||||||
(UiContext::Login, app_state.focused_button_index)
|
(UiContext::Login, app_state.focused_button_index)
|
||||||
} else if app_state.ui.show_register && app_state.ui.focus_outside_canvas {
|
}
|
||||||
|
Page::Register(_) if app_state.ui.focus_outside_canvas => {
|
||||||
(UiContext::Register, app_state.focused_button_index)
|
(UiContext::Register, app_state.focused_button_index)
|
||||||
} else if app_state.ui.show_admin {
|
}
|
||||||
(UiContext::Admin, admin_state.get_selected_index().unwrap_or(0))
|
Page::Admin(state) => {
|
||||||
} else if app_state.ui.dialog.dialog_show {
|
(UiContext::Admin, state.get_selected_index().unwrap_or(0))
|
||||||
|
}
|
||||||
|
_ if app_state.ui.dialog.dialog_show => {
|
||||||
(UiContext::Dialog, app_state.ui.dialog.dialog_active_button_index)
|
(UiContext::Dialog, app_state.ui.dialog.dialog_active_button_index)
|
||||||
} else {
|
}
|
||||||
return Ok(EventOutcome::Ok("Select (No Action)".to_string()));
|
_ => return Ok(EventOutcome::Ok("Select (No Action)".to_string())),
|
||||||
};
|
};
|
||||||
return Ok(EventOutcome::ButtonSelected { context, index });
|
return Ok(EventOutcome::ButtonSelected { context, index });
|
||||||
}
|
}
|
||||||
@@ -88,55 +89,68 @@ pub async fn handle_navigation_event(
|
|||||||
Ok(EventOutcome::Ok(String::new()))
|
Ok(EventOutcome::Ok(String::new()))
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn move_up(app_state: &mut AppState, login_state: &mut LoginState, register_state: &mut RegisterState, intro_state: &mut IntroState, admin_state: &mut AdminState) {
|
pub fn up(app_state: &mut AppState, router: &mut Router) {
|
||||||
if app_state.ui.focus_outside_canvas && app_state.ui.show_login || app_state.ui.show_register{
|
match &mut router.current {
|
||||||
|
Page::Login(page) if app_state.ui.focus_outside_canvas => {
|
||||||
if app_state.focused_button_index == 0 {
|
if app_state.focused_button_index == 0 {
|
||||||
app_state.ui.focus_outside_canvas = false;
|
app_state.ui.focus_outside_canvas = false;
|
||||||
if app_state.ui.show_login {
|
let last_field_index = page.state.field_count().saturating_sub(1);
|
||||||
let last_field_index = login_state.field_count().saturating_sub(1);
|
page.state.set_current_field(last_field_index);
|
||||||
login_state.set_current_field(last_field_index);
|
|
||||||
} else {
|
} else {
|
||||||
let last_field_index = register_state.field_count().saturating_sub(1);
|
app_state.focused_button_index =
|
||||||
register_state.set_current_field(last_field_index);
|
app_state.focused_button_index.saturating_sub(1);
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
Page::Register(state) if app_state.ui.focus_outside_canvas => {
|
||||||
|
if app_state.focused_button_index == 0 {
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
let last_field_index = state.state.field_count().saturating_sub(1);
|
||||||
|
state.set_current_field(last_field_index);
|
||||||
} else {
|
} else {
|
||||||
app_state.focused_button_index = app_state.focused_button_index.saturating_sub(1);
|
app_state.focused_button_index =
|
||||||
|
app_state.focused_button_index.saturating_sub(1);
|
||||||
}
|
}
|
||||||
} else if app_state.ui.show_intro {
|
}
|
||||||
intro_state.previous_option();
|
Page::Intro(state) => state.previous_option(),
|
||||||
} else if app_state.ui.show_admin {
|
Page::Admin(state) => state.previous(),
|
||||||
admin_state.previous();
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn move_down(app_state: &mut AppState, intro_state: &mut IntroState, admin_state: &mut AdminState) {
|
pub fn down(app_state: &mut AppState, router: &mut Router) {
|
||||||
if app_state.ui.focus_outside_canvas && app_state.ui.show_login || app_state.ui.show_register {
|
match &mut router.current {
|
||||||
|
Page::Login(_) | Page::Register(_) if app_state.ui.focus_outside_canvas => {
|
||||||
let num_general_elements = 2;
|
let num_general_elements = 2;
|
||||||
if app_state.focused_button_index < num_general_elements - 1 {
|
if app_state.focused_button_index < num_general_elements - 1 {
|
||||||
app_state.focused_button_index += 1;
|
app_state.focused_button_index += 1;
|
||||||
}
|
}
|
||||||
} else if app_state.ui.show_intro {
|
}
|
||||||
intro_state.next_option();
|
Page::Intro(state) => state.next_option(),
|
||||||
} else if app_state.ui.show_admin {
|
Page::Admin(state) => state.next(),
|
||||||
admin_state.next();
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn next_option(app_state: &mut AppState, intro_state: &mut IntroState) {
|
pub fn next_option(app_state: &mut AppState, router: &mut Router) {
|
||||||
if app_state.ui.show_intro {
|
match &mut router.current {
|
||||||
intro_state.next_option();
|
Page::Intro(state) => state.next_option(),
|
||||||
} else {
|
Page::Admin(_) => {
|
||||||
// Get option count from state instead of parameter
|
|
||||||
let option_count = app_state.profile_tree.profiles.len();
|
let option_count = app_state.profile_tree.profiles.len();
|
||||||
app_state.focused_button_index = (app_state.focused_button_index + 1) % option_count;
|
if option_count > 0 {
|
||||||
|
app_state.focused_button_index =
|
||||||
|
(app_state.focused_button_index + 1) % option_count;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
pub fn previous_option(app_state: &mut AppState, intro_state: &mut IntroState) {
|
pub fn previous_option(app_state: &mut AppState, router: &mut Router) {
|
||||||
if app_state.ui.show_intro {
|
match &mut router.current {
|
||||||
intro_state.previous_option();
|
Page::Intro(state) => state.previous_option(),
|
||||||
} else {
|
Page::Admin(_) => {
|
||||||
let option_count = app_state.profile_tree.profiles.len();
|
let option_count = app_state.profile_tree.profiles.len();
|
||||||
|
if option_count > 0 {
|
||||||
app_state.focused_button_index = if app_state.focused_button_index == 0 {
|
app_state.focused_button_index = if app_state.focused_button_index == 0 {
|
||||||
option_count.saturating_sub(1)
|
option_count.saturating_sub(1)
|
||||||
} else {
|
} else {
|
||||||
@@ -144,6 +158,9 @@ pub fn previous_option(app_state: &mut AppState, intro_state: &mut IntroState) {
|
|||||||
};
|
};
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
pub fn next_field(form_state: &mut FormState) {
|
pub fn next_field(form_state: &mut FormState) {
|
||||||
if !form_state.fields.is_empty() {
|
if !form_state.fields.is_empty() {
|
||||||
@@ -164,7 +181,7 @@ pub fn prev_field(form_state: &mut FormState) {
|
|||||||
pub fn handle_enter_command_mode(
|
pub fn handle_enter_command_mode(
|
||||||
command_mode: &mut bool,
|
command_mode: &mut bool,
|
||||||
command_input: &mut String,
|
command_input: &mut String,
|
||||||
command_message: &mut String
|
command_message: &mut String,
|
||||||
) {
|
) {
|
||||||
*command_mode = true;
|
*command_mode = true;
|
||||||
command_input.clear();
|
command_input.clear();
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -1,110 +1,59 @@
|
|||||||
// src/modes/handlers/mode_manager.rs
|
// src/modes/handlers/mode_manager.rs
|
||||||
|
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::modes::handlers::event::EventHandler;
|
use crate::modes::handlers::event::EventHandler;
|
||||||
use crate::state::pages::add_logic::AddLogicFocus;
|
use crate::pages::routing::{Router, Page};
|
||||||
use crate::state::pages::admin::AdminState;
|
|
||||||
use canvas::AppMode as CanvasMode;
|
|
||||||
|
|
||||||
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
pub enum AppMode {
|
pub enum AppMode {
|
||||||
General, // For intro and admin screens
|
/// General mode = when focus is outside any canvas
|
||||||
ReadOnly, // Canvas read-only mode
|
/// (Intro, Admin, Login/Register buttons, AddTable/AddLogic menus, dialogs, etc.)
|
||||||
Edit, // Canvas edit mode
|
General,
|
||||||
Highlight, // Canvas highlight/visual mode
|
|
||||||
Command, // Command mode overlay
|
|
||||||
}
|
|
||||||
|
|
||||||
impl From<canvas::AppMode> for AppMode {
|
/// Command overlay (":" or "ctrl+;"), available globally
|
||||||
fn from(mode: canvas::AppMode) -> Self {
|
Command,
|
||||||
match mode {
|
|
||||||
canvas::AppMode::General => AppMode::General,
|
|
||||||
canvas::AppMode::ReadOnly => AppMode::ReadOnly,
|
|
||||||
canvas::AppMode::Edit => AppMode::Edit,
|
|
||||||
canvas::AppMode::Highlight => AppMode::Highlight,
|
|
||||||
canvas::AppMode::Command => AppMode::Command,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pub struct ModeManager;
|
pub struct ModeManager;
|
||||||
|
|
||||||
impl ModeManager {
|
impl ModeManager {
|
||||||
/// Determine current mode based on app state
|
/// Determine current mode:
|
||||||
|
/// - If navigation palette is active → General
|
||||||
|
/// - If command overlay is active → Command
|
||||||
|
/// - If focus is inside a canvas (Form, Login, Register, AddTable, AddLogic) → let canvas handle its own mode
|
||||||
|
/// - Otherwise → General
|
||||||
pub fn derive_mode(
|
pub fn derive_mode(
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
event_handler: &EventHandler,
|
event_handler: &EventHandler,
|
||||||
admin_state: &AdminState,
|
router: &Router,
|
||||||
) -> AppMode {
|
) -> AppMode {
|
||||||
// Navigation palette always forces General
|
// Navigation palette always forces General
|
||||||
if event_handler.navigation_state.active {
|
if event_handler.navigation_state.active {
|
||||||
return AppMode::General;
|
return AppMode::General;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Explicit command mode flag
|
// Explicit command overlay flag
|
||||||
if event_handler.command_mode {
|
if event_handler.command_mode {
|
||||||
return AppMode::Command;
|
return AppMode::Command;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Always trust the FormEditor when a form is active
|
// If focus is inside a canvas, we don't duplicate canvas modes here.
|
||||||
if app_state.ui.show_form && !app_state.ui.focus_outside_canvas {
|
// Canvas crate owns ReadOnly/Edit/Highlight internally.
|
||||||
if let Some(editor) = &app_state.form_editor {
|
match &router.current {
|
||||||
return AppMode::from(editor.mode());
|
Page::Form(_)
|
||||||
}
|
| Page::Login(_)
|
||||||
}
|
| Page::Register(_)
|
||||||
|
| Page::AddTable(_)
|
||||||
// --- Non-form views (add_logic, add_table, etc.) ---
|
| Page::AddLogic(_) if !app_state.ui.focus_outside_canvas => {
|
||||||
if app_state.ui.show_add_logic {
|
// Canvas active → let canvas handle its own AppMode
|
||||||
match admin_state.add_logic_state.current_focus {
|
AppMode::General
|
||||||
AddLogicFocus::InputLogicName
|
|
||||||
| AddLogicFocus::InputTargetColumn
|
|
||||||
| AddLogicFocus::InputDescription => {
|
|
||||||
if event_handler.is_edit_mode {
|
|
||||||
AppMode::Edit
|
|
||||||
} else {
|
|
||||||
AppMode::ReadOnly
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
_ => AppMode::General,
|
_ => AppMode::General,
|
||||||
}
|
}
|
||||||
} else if app_state.ui.show_add_table {
|
|
||||||
if app_state.ui.focus_outside_canvas {
|
|
||||||
AppMode::General
|
|
||||||
} else if event_handler.is_edit_mode {
|
|
||||||
AppMode::Edit
|
|
||||||
} else {
|
|
||||||
AppMode::ReadOnly
|
|
||||||
}
|
|
||||||
} else if app_state.ui.show_login
|
|
||||||
|| app_state.ui.show_register
|
|
||||||
{
|
|
||||||
// login/register still use the old flag
|
|
||||||
if event_handler.is_edit_mode {
|
|
||||||
AppMode::Edit
|
|
||||||
} else {
|
|
||||||
AppMode::ReadOnly
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
AppMode::General
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Mode transition rules
|
/// Command overlay can be entered from anywhere (General or Canvas).
|
||||||
pub fn can_enter_command_mode(current_mode: AppMode) -> bool {
|
pub fn can_enter_command_mode(_current_mode: AppMode) -> bool {
|
||||||
!matches!(current_mode, AppMode::Edit)
|
true
|
||||||
}
|
|
||||||
|
|
||||||
pub fn can_enter_edit_mode(current_mode: AppMode) -> bool {
|
|
||||||
matches!(current_mode, AppMode::ReadOnly)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn can_enter_read_only_mode(current_mode: AppMode) -> bool {
|
|
||||||
matches!(
|
|
||||||
current_mode,
|
|
||||||
AppMode::Edit | AppMode::Command | AppMode::Highlight
|
|
||||||
)
|
|
||||||
}
|
|
||||||
|
|
||||||
pub fn can_enter_highlight_mode(current_mode: AppMode) -> bool {
|
|
||||||
matches!(current_mode, AppMode::ReadOnly)
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -7,4 +7,3 @@ pub mod canvas;
|
|||||||
pub use handlers::*;
|
pub use handlers::*;
|
||||||
pub use general::*;
|
pub use general::*;
|
||||||
pub use common::*;
|
pub use common::*;
|
||||||
pub use canvas::*;
|
|
||||||
|
|||||||
12
client/src/movement/actions.rs
Normal file
12
client/src/movement/actions.rs
Normal file
@@ -0,0 +1,12 @@
|
|||||||
|
// src/movement/actions.rs
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
|
||||||
|
pub enum MovementAction {
|
||||||
|
Next,
|
||||||
|
Previous,
|
||||||
|
Up,
|
||||||
|
Down,
|
||||||
|
Left,
|
||||||
|
Right,
|
||||||
|
Select,
|
||||||
|
Esc,
|
||||||
|
}
|
||||||
32
client/src/movement/lib.rs
Normal file
32
client/src/movement/lib.rs
Normal file
@@ -0,0 +1,32 @@
|
|||||||
|
// src/movement/lib.rs
|
||||||
|
|
||||||
|
use crate::movement::MovementAction;
|
||||||
|
|
||||||
|
#[inline]
|
||||||
|
pub fn move_focus<T: Copy + Eq>(
|
||||||
|
order: &[T],
|
||||||
|
current: &mut T,
|
||||||
|
action: MovementAction,
|
||||||
|
) -> bool {
|
||||||
|
if order.is_empty() {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if let Some(pos) = order.iter().position(|k| *k == *current) {
|
||||||
|
match action {
|
||||||
|
MovementAction::Previous | MovementAction::Up | MovementAction::Left => {
|
||||||
|
if pos > 0 {
|
||||||
|
*current = order[pos - 1];
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Down | MovementAction::Right => {
|
||||||
|
if pos + 1 < order.len() {
|
||||||
|
*current = order[pos + 1];
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
6
client/src/movement/mod.rs
Normal file
6
client/src/movement/mod.rs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
// src/movement/mod.rs
|
||||||
|
pub mod actions;
|
||||||
|
pub mod lib;
|
||||||
|
|
||||||
|
pub use actions::MovementAction;
|
||||||
|
pub use lib::move_focus;
|
||||||
60
client/src/pages/admin/admin/event.rs
Normal file
60
client/src/pages/admin/admin/event.rs
Normal file
@@ -0,0 +1,60 @@
|
|||||||
|
// src/pages/admin/admin/event.rs
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::event::KeyEvent;
|
||||||
|
|
||||||
|
use crate::buffer::state::BufferState;
|
||||||
|
use crate::config::binds::config::Config;
|
||||||
|
use crate::pages::admin::AdminState;
|
||||||
|
use crate::pages::admin::main::logic::handle_admin_navigation;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
|
||||||
|
/// Handle all Admin page-specific key events (movement + actions).
|
||||||
|
/// Returns true if the key was handled (so the caller should stop propagation).
|
||||||
|
pub fn handle_admin_event(
|
||||||
|
key_event: KeyEvent,
|
||||||
|
config: &Config,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
admin_state: &mut AdminState,
|
||||||
|
buffer_state: &mut BufferState,
|
||||||
|
command_message: &mut String,
|
||||||
|
) -> Result<bool> {
|
||||||
|
// 1) Map general action to MovementAction (same mapping used in event.rs)
|
||||||
|
let movement_action = if let Some(act) =
|
||||||
|
config.get_general_action(key_event.code, key_event.modifiers)
|
||||||
|
{
|
||||||
|
use crate::movement::MovementAction;
|
||||||
|
match act {
|
||||||
|
"up" => Some(MovementAction::Up),
|
||||||
|
"down" => Some(MovementAction::Down),
|
||||||
|
"left" => Some(MovementAction::Left),
|
||||||
|
"right" => Some(MovementAction::Right),
|
||||||
|
"next" => Some(MovementAction::Next),
|
||||||
|
"previous" => Some(MovementAction::Previous),
|
||||||
|
"select" => Some(MovementAction::Select),
|
||||||
|
"esc" => Some(MovementAction::Esc),
|
||||||
|
_ => None,
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(ma) = movement_action {
|
||||||
|
if admin_state.handle_movement(app_state, ma) {
|
||||||
|
return Ok(true);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2) Rich Admin navigation (buttons, selections, etc.)
|
||||||
|
if handle_admin_navigation(
|
||||||
|
key_event,
|
||||||
|
config,
|
||||||
|
app_state,
|
||||||
|
admin_state,
|
||||||
|
buffer_state,
|
||||||
|
command_message,
|
||||||
|
) {
|
||||||
|
return Ok(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(false)
|
||||||
|
}
|
||||||
54
client/src/pages/admin/admin/loader.rs
Normal file
54
client/src/pages/admin/admin/loader.rs
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
// src/pages/admin/admin/loader.rs
|
||||||
|
use anyhow::{Context, Result};
|
||||||
|
|
||||||
|
use crate::pages::admin::{AdminFocus, AdminState};
|
||||||
|
use crate::services::grpc_client::GrpcClient;
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
|
||||||
|
/// Refresh admin data and ensure focus and selections are valid.
|
||||||
|
pub async fn refresh_admin_state(
|
||||||
|
grpc_client: &mut GrpcClient,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
admin_state: &mut AdminState,
|
||||||
|
) -> Result<()> {
|
||||||
|
// Fetch latest profile tree
|
||||||
|
let refreshed_tree = grpc_client
|
||||||
|
.get_profile_tree()
|
||||||
|
.await
|
||||||
|
.context("Failed to refresh profile tree for Admin panel")?;
|
||||||
|
app_state.profile_tree = refreshed_tree;
|
||||||
|
|
||||||
|
// Populate profile names for AdminState's list
|
||||||
|
let profile_names = app_state
|
||||||
|
.profile_tree
|
||||||
|
.profiles
|
||||||
|
.iter()
|
||||||
|
.map(|p| p.name.clone())
|
||||||
|
.collect::<Vec<_>>();
|
||||||
|
admin_state.set_profiles(profile_names);
|
||||||
|
|
||||||
|
// Ensure a sane focus
|
||||||
|
if admin_state.current_focus == AdminFocus::default()
|
||||||
|
|| !matches!(
|
||||||
|
admin_state.current_focus,
|
||||||
|
AdminFocus::InsideProfilesList
|
||||||
|
| AdminFocus::Tables
|
||||||
|
| AdminFocus::InsideTablesList
|
||||||
|
| AdminFocus::Button1
|
||||||
|
| AdminFocus::Button2
|
||||||
|
| AdminFocus::Button3
|
||||||
|
| AdminFocus::ProfilesPane
|
||||||
|
)
|
||||||
|
{
|
||||||
|
admin_state.current_focus = AdminFocus::ProfilesPane;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Ensure a selection exists when profiles are present
|
||||||
|
if admin_state.profile_list_state.selected().is_none()
|
||||||
|
&& !app_state.profile_tree.profiles.is_empty()
|
||||||
|
{
|
||||||
|
admin_state.profile_list_state.select(Some(0));
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
9
client/src/pages/admin/admin/mod.rs
Normal file
9
client/src/pages/admin/admin/mod.rs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
// src/pages/admin/admin/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod tui;
|
||||||
|
pub mod event;
|
||||||
|
pub mod loader;
|
||||||
|
|
||||||
|
pub use state::{AdminState, AdminFocus};
|
||||||
195
client/src/pages/admin/admin/state.rs
Normal file
195
client/src/pages/admin/admin/state.rs
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
// src/pages/admin/admin/state.rs
|
||||||
|
use ratatui::widgets::ListState;
|
||||||
|
use crate::pages::admin_panel::add_table::state::AddTableState;
|
||||||
|
use crate::pages::admin_panel::add_logic::state::AddLogicState;
|
||||||
|
use crate::movement::{move_focus, MovementAction};
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
|
||||||
|
/// Focus states for the admin panel
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
|
||||||
|
pub enum AdminFocus {
|
||||||
|
#[default]
|
||||||
|
ProfilesPane,
|
||||||
|
InsideProfilesList,
|
||||||
|
Tables,
|
||||||
|
InsideTablesList,
|
||||||
|
Button1,
|
||||||
|
Button2,
|
||||||
|
Button3,
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Full admin panel state (for logged-in admins)
|
||||||
|
#[derive(Default, Clone, Debug)]
|
||||||
|
pub struct AdminState {
|
||||||
|
pub profiles: Vec<String>,
|
||||||
|
pub profile_list_state: ListState,
|
||||||
|
pub table_list_state: ListState,
|
||||||
|
pub selected_profile_index: Option<usize>,
|
||||||
|
pub selected_table_index: Option<usize>,
|
||||||
|
pub current_focus: AdminFocus,
|
||||||
|
pub add_table_state: AddTableState,
|
||||||
|
pub add_logic_state: AddLogicState,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AdminState {
|
||||||
|
pub fn get_selected_index(&self) -> Option<usize> {
|
||||||
|
self.profile_list_state.selected()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_selected_profile_name(&self) -> Option<&String> {
|
||||||
|
self.profile_list_state.selected().and_then(|i| self.profiles.get(i))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_profiles(&mut self, new_profiles: Vec<String>) {
|
||||||
|
let current_selection_index = self.profile_list_state.selected();
|
||||||
|
self.profiles = new_profiles;
|
||||||
|
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
} else {
|
||||||
|
let new_selection = match current_selection_index {
|
||||||
|
Some(index) => Some(index.min(self.profiles.len() - 1)),
|
||||||
|
None => Some(0),
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(new_selection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn next(&mut self) {
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let i = match self.profile_list_state.selected() {
|
||||||
|
Some(i) => if i >= self.profiles.len() - 1 { 0 } else { i + 1 },
|
||||||
|
None => 0,
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(Some(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn previous(&mut self) {
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let i = match self.profile_list_state.selected() {
|
||||||
|
Some(i) => if i == 0 { self.profiles.len() - 1 } else { i - 1 },
|
||||||
|
None => self.profiles.len() - 1,
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(Some(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_movement(
|
||||||
|
&mut self,
|
||||||
|
app: &AppState,
|
||||||
|
action: MovementAction,
|
||||||
|
) -> bool {
|
||||||
|
use AdminFocus::*;
|
||||||
|
|
||||||
|
const ORDER: [AdminFocus; 5] = [
|
||||||
|
ProfilesPane,
|
||||||
|
Tables,
|
||||||
|
Button1,
|
||||||
|
Button2,
|
||||||
|
Button3,
|
||||||
|
];
|
||||||
|
|
||||||
|
match (self.current_focus, action) {
|
||||||
|
(ProfilesPane, MovementAction::Select) => {
|
||||||
|
if !app.profile_tree.profiles.is_empty()
|
||||||
|
&& self.profile_list_state.selected().is_none()
|
||||||
|
{
|
||||||
|
self.profile_list_state.select(Some(0));
|
||||||
|
}
|
||||||
|
self.current_focus = InsideProfilesList;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
(Tables, MovementAction::Select) => {
|
||||||
|
let p_idx = self
|
||||||
|
.selected_profile_index
|
||||||
|
.or_else(|| self.profile_list_state.selected());
|
||||||
|
if let Some(pi) = p_idx {
|
||||||
|
let len = app
|
||||||
|
.profile_tree
|
||||||
|
.profiles
|
||||||
|
.get(pi)
|
||||||
|
.map(|p| p.tables.len())
|
||||||
|
.unwrap_or(0);
|
||||||
|
if len > 0 && self.table_list_state.selected().is_none() {
|
||||||
|
self.table_list_state.select(Some(0));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
self.current_focus = InsideTablesList;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
match self.current_focus {
|
||||||
|
InsideProfilesList => match action {
|
||||||
|
MovementAction::Up => {
|
||||||
|
if !app.profile_tree.profiles.is_empty() {
|
||||||
|
let curr = self.profile_list_state.selected().unwrap_or(0);
|
||||||
|
let next = curr.saturating_sub(1);
|
||||||
|
self.profile_list_state.select(Some(next));
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Down => {
|
||||||
|
let len = app.profile_tree.profiles.len();
|
||||||
|
if len > 0 {
|
||||||
|
let curr = self.profile_list_state.selected().unwrap_or(0);
|
||||||
|
let next = if curr + 1 < len { curr + 1 } else { curr };
|
||||||
|
self.profile_list_state.select(Some(next));
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
self.current_focus = ProfilesPane;
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Previous => true,
|
||||||
|
MovementAction::Select => false,
|
||||||
|
_ => false,
|
||||||
|
},
|
||||||
|
InsideTablesList => {
|
||||||
|
let tables_len = {
|
||||||
|
let p_idx = self
|
||||||
|
.selected_profile_index
|
||||||
|
.or_else(|| self.profile_list_state.selected());
|
||||||
|
p_idx.and_then(|pi| app.profile_tree.profiles.get(pi))
|
||||||
|
.map(|p| p.tables.len())
|
||||||
|
.unwrap_or(0)
|
||||||
|
};
|
||||||
|
match action {
|
||||||
|
MovementAction::Up => {
|
||||||
|
if tables_len > 0 {
|
||||||
|
let curr = self.table_list_state.selected().unwrap_or(0);
|
||||||
|
let next = curr.saturating_sub(1);
|
||||||
|
self.table_list_state.select(Some(next));
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Down => {
|
||||||
|
if tables_len > 0 {
|
||||||
|
let curr = self.table_list_state.selected().unwrap_or(0);
|
||||||
|
let next = if curr + 1 < tables_len { curr + 1 } else { curr };
|
||||||
|
self.table_list_state.select(Some(next));
|
||||||
|
}
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
self.current_focus = Tables;
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Previous => true,
|
||||||
|
MovementAction::Select => false,
|
||||||
|
_ => false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
move_focus(&ORDER, &mut self.current_focus, action)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,5 +1,6 @@
|
|||||||
|
// src/pages/admin/admin/tui.rs
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::admin::AdminState;
|
use crate::pages::admin::AdminState;
|
||||||
|
|
||||||
pub fn handle_admin_selection(app_state: &mut AppState, admin_state: &AdminState) {
|
pub fn handle_admin_selection(app_state: &mut AppState, admin_state: &AdminState) {
|
||||||
let profiles = &app_state.profile_tree.profiles;
|
let profiles = &app_state.profile_tree.profiles;
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
// src/components/admin/admin_panel_admin.rs
|
// src/pages/admin/admin/ui.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::pages::admin::{AdminFocus, AdminState};
|
use crate::pages::admin::{AdminFocus, AdminState};
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
||||||
@@ -1,11 +1,11 @@
|
|||||||
// src/functions/modes/navigation/admin_nav.rs
|
// src/pages/admin/main/logic.rs
|
||||||
use crate::state::pages::admin::{AdminFocus, AdminState};
|
use crate::pages::admin::{AdminFocus, AdminState};
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::config::binds::config::Config;
|
use crate::config::binds::config::Config;
|
||||||
use crate::state::app::buffer::{BufferState, AppView};
|
use crate::buffer::state::{BufferState, AppView};
|
||||||
use crate::state::pages::add_table::{AddTableState, LinkDefinition};
|
use crate::pages::admin_panel::add_table::state::{AddTableState, LinkDefinition};
|
||||||
use ratatui::widgets::ListState;
|
use ratatui::widgets::ListState;
|
||||||
use crate::state::pages::add_logic::{AddLogicState, AddLogicFocus}; // Added AddLogicFocus import
|
use crate::pages::admin_panel::add_logic::state::{AddLogicState, AddLogicFocus};
|
||||||
|
|
||||||
// Helper functions list_select_next and list_select_previous remain the same
|
// Helper functions list_select_next and list_select_previous remain the same
|
||||||
fn list_select_next(list_state: &mut ListState, item_count: usize) {
|
fn list_select_next(list_state: &mut ListState, item_count: usize) {
|
||||||
7
client/src/pages/admin/main/mod.rs
Normal file
7
client/src/pages/admin/main/mod.rs
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
// src/pages/admin/main/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
|
||||||
|
pub use state::NonAdminState;
|
||||||
55
client/src/pages/admin/main/state.rs
Normal file
55
client/src/pages/admin/main/state.rs
Normal file
@@ -0,0 +1,55 @@
|
|||||||
|
// src/pages/admin/main/state.rs
|
||||||
|
use ratatui::widgets::ListState;
|
||||||
|
|
||||||
|
/// State for non-admin users (simple profile browser)
|
||||||
|
#[derive(Default, Clone, Debug)]
|
||||||
|
pub struct NonAdminState {
|
||||||
|
pub profiles: Vec<String>, // profile names
|
||||||
|
pub profile_list_state: ListState, // highlight state
|
||||||
|
pub selected_profile_index: Option<usize>, // persistent selection
|
||||||
|
}
|
||||||
|
|
||||||
|
impl NonAdminState {
|
||||||
|
pub fn get_selected_index(&self) -> Option<usize> {
|
||||||
|
self.profile_list_state.selected()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_profiles(&mut self, new_profiles: Vec<String>) {
|
||||||
|
let current_selection_index = self.profile_list_state.selected();
|
||||||
|
self.profiles = new_profiles;
|
||||||
|
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
} else {
|
||||||
|
let new_selection = match current_selection_index {
|
||||||
|
Some(index) => Some(index.min(self.profiles.len() - 1)),
|
||||||
|
None => Some(0),
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(new_selection);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn next(&mut self) {
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let i = match self.profile_list_state.selected() {
|
||||||
|
Some(i) => if i >= self.profiles.len() - 1 { 0 } else { i + 1 },
|
||||||
|
None => 0,
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(Some(i));
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn previous(&mut self) {
|
||||||
|
if self.profiles.is_empty() {
|
||||||
|
self.profile_list_state.select(None);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
let i = match self.profile_list_state.selected() {
|
||||||
|
Some(i) => if i == 0 { self.profiles.len() - 1 } else { i - 1 },
|
||||||
|
None => self.profiles.len() - 1,
|
||||||
|
};
|
||||||
|
self.profile_list_state.select(Some(i));
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,9 @@
|
|||||||
// src/components/admin/admin_panel.rs
|
// src/pages/admin/main/ui.rs
|
||||||
|
|
||||||
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::app::state::AppState;
|
||||||
use crate::state::pages::admin::AdminState;
|
use crate::pages::admin::AdminState;
|
||||||
use common::proto::komp_ac::table_definition::ProfileTreeResponse;
|
use common::proto::komp_ac::table_definition::ProfileTreeResponse;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Constraint, Direction, Layout, Rect},
|
layout::{Constraint, Direction, Layout, Rect},
|
||||||
@@ -12,7 +12,8 @@ use ratatui::{
|
|||||||
widgets::{Block, BorderType, Borders, List, ListItem, Paragraph, Wrap},
|
widgets::{Block, BorderType, Borders, List, ListItem, Paragraph, Wrap},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use super::admin_panel_admin::render_admin_panel_admin;
|
use crate::state::pages::auth::UserRole;
|
||||||
|
use crate::pages::admin::admin::ui::render_admin_panel_admin;
|
||||||
|
|
||||||
pub fn render_admin_panel(
|
pub fn render_admin_panel(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
@@ -44,7 +45,11 @@ pub fn render_admin_panel(
|
|||||||
.constraints([Constraint::Percentage(30), Constraint::Percentage(70)])
|
.constraints([Constraint::Percentage(30), Constraint::Percentage(70)])
|
||||||
.split(chunks[1]);
|
.split(chunks[1]);
|
||||||
|
|
||||||
if auth_state.role.as_deref() != Some("admin") {
|
match auth_state.role {
|
||||||
|
Some(UserRole::Admin) => {
|
||||||
|
render_admin_panel_admin(f, chunks[1], app_state, admin_state, theme);
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
render_admin_panel_non_admin(
|
render_admin_panel_non_admin(
|
||||||
f,
|
f,
|
||||||
admin_state,
|
admin_state,
|
||||||
@@ -53,21 +58,14 @@ pub fn render_admin_panel(
|
|||||||
profile_tree,
|
profile_tree,
|
||||||
selected_profile,
|
selected_profile,
|
||||||
);
|
);
|
||||||
} else {
|
}
|
||||||
render_admin_panel_admin(
|
|
||||||
f,
|
|
||||||
chunks[1],
|
|
||||||
app_state,
|
|
||||||
admin_state,
|
|
||||||
theme,
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Renders the view for non-admin users (profile list and details).
|
/// Renders the view for non-admin users (profile list and details).
|
||||||
fn render_admin_panel_non_admin(
|
fn render_admin_panel_non_admin(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
admin_state: &AdminState,
|
admin_state: &mut AdminState,
|
||||||
content_chunks: &[Rect],
|
content_chunks: &[Rect],
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
profile_tree: &ProfileTreeResponse,
|
profile_tree: &ProfileTreeResponse,
|
||||||
@@ -92,8 +90,7 @@ fn render_admin_panel_non_admin(
|
|||||||
.block(Block::default().title("Profiles"))
|
.block(Block::default().title("Profiles"))
|
||||||
.highlight_style(Style::default().bg(theme.highlight).fg(theme.bg));
|
.highlight_style(Style::default().bg(theme.highlight).fg(theme.bg));
|
||||||
|
|
||||||
let mut profile_list_state_clone = admin_state.profile_list_state.clone();
|
f.render_stateful_widget(list, content_chunks[0], &mut admin_state.profile_list_state);
|
||||||
f.render_stateful_widget(list, content_chunks[0], &mut profile_list_state_clone);
|
|
||||||
|
|
||||||
// Profile details - Use selection info from admin_state
|
// Profile details - Use selection info from admin_state
|
||||||
if let Some(profile) = admin_state
|
if let Some(profile) = admin_state
|
||||||
7
client/src/pages/admin/mod.rs
Normal file
7
client/src/pages/admin/mod.rs
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
// src/pages/admin/mod.rs
|
||||||
|
|
||||||
|
pub mod main; // non-admin
|
||||||
|
pub mod admin; // full admin panel
|
||||||
|
|
||||||
|
pub use main::NonAdminState;
|
||||||
|
pub use admin::{AdminState, AdminFocus};
|
||||||
5
client/src/pages/admin_panel/add_logic/mod.rs
Normal file
5
client/src/pages/admin_panel/add_logic/mod.rs
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
// src/pages/admin_panel/add_logic/mod.rs
|
||||||
|
|
||||||
|
pub mod ui;
|
||||||
|
pub mod nav;
|
||||||
|
pub mod state;
|
||||||
531
client/src/pages/admin_panel/add_logic/nav.rs
Normal file
531
client/src/pages/admin_panel/add_logic/nav.rs
Normal file
@@ -0,0 +1,531 @@
|
|||||||
|
// src/pages/admin_panel/add_logic/nav.rs
|
||||||
|
|
||||||
|
use crate::config::binds::config::{Config, EditorKeybindingMode};
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::pages::admin_panel::add_logic::state::{AddLogicFocus, AddLogicState};
|
||||||
|
use crate::buffer::{AppView, BufferState};
|
||||||
|
use crossterm::event::{KeyEvent, KeyCode, KeyModifiers};
|
||||||
|
use crate::services::GrpcClient;
|
||||||
|
use tokio::sync::mpsc;
|
||||||
|
use anyhow::Result;
|
||||||
|
use crate::components::common::text_editor::TextEditor;
|
||||||
|
use crate::services::ui_service::UiService;
|
||||||
|
use tui_textarea::CursorMove;
|
||||||
|
use crate::pages::admin::AdminState;
|
||||||
|
use crate::pages::routing::{Router, Page};
|
||||||
|
|
||||||
|
pub type SaveLogicResultSender = mpsc::Sender<Result<String>>;
|
||||||
|
|
||||||
|
pub fn handle_add_logic_navigation(
|
||||||
|
key_event: KeyEvent,
|
||||||
|
config: &Config,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
buffer_state: &mut BufferState,
|
||||||
|
grpc_client: GrpcClient,
|
||||||
|
save_logic_sender: SaveLogicResultSender,
|
||||||
|
command_message: &mut String,
|
||||||
|
router: &mut Router,
|
||||||
|
) -> bool {
|
||||||
|
if let Page::AddLogic(add_logic_state) = &mut router.current {
|
||||||
|
// === FULLSCREEN SCRIPT EDITING ===
|
||||||
|
if add_logic_state.current_focus == AddLogicFocus::InsideScriptContent {
|
||||||
|
// === AUTOCOMPLETE HANDLING ===
|
||||||
|
if add_logic_state.script_editor_autocomplete_active {
|
||||||
|
match key_event.code {
|
||||||
|
KeyCode::Char(c) if c.is_alphanumeric() || c == '_' => {
|
||||||
|
add_logic_state.script_editor_filter_text.push(c);
|
||||||
|
add_logic_state.update_script_editor_suggestions();
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
*command_message =
|
||||||
|
format!("Filtering: @{}", add_logic_state.script_editor_filter_text);
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
KeyCode::Backspace => {
|
||||||
|
if !add_logic_state.script_editor_filter_text.is_empty() {
|
||||||
|
add_logic_state.script_editor_filter_text.pop();
|
||||||
|
add_logic_state.update_script_editor_suggestions();
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
*command_message =
|
||||||
|
if add_logic_state.script_editor_filter_text.is_empty() {
|
||||||
|
"Autocomplete: @".to_string()
|
||||||
|
} else {
|
||||||
|
format!(
|
||||||
|
"Filtering: @{}",
|
||||||
|
add_logic_state.script_editor_filter_text
|
||||||
|
)
|
||||||
|
};
|
||||||
|
} else {
|
||||||
|
let should_deactivate =
|
||||||
|
if let Some((trigger_line, trigger_col)) =
|
||||||
|
add_logic_state.script_editor_trigger_position
|
||||||
|
{
|
||||||
|
let current_cursor = {
|
||||||
|
let editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow();
|
||||||
|
editor_borrow.cursor()
|
||||||
|
};
|
||||||
|
current_cursor.0 == trigger_line
|
||||||
|
&& current_cursor.1 == trigger_col + 1
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
};
|
||||||
|
if should_deactivate {
|
||||||
|
add_logic_state.deactivate_script_editor_autocomplete();
|
||||||
|
*command_message = "Autocomplete cancelled".to_string();
|
||||||
|
}
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
KeyCode::Tab | KeyCode::Down => {
|
||||||
|
if !add_logic_state.script_editor_suggestions.is_empty() {
|
||||||
|
let current = add_logic_state
|
||||||
|
.script_editor_selected_suggestion_index
|
||||||
|
.unwrap_or(0);
|
||||||
|
let next =
|
||||||
|
(current + 1) % add_logic_state.script_editor_suggestions.len();
|
||||||
|
add_logic_state.script_editor_selected_suggestion_index = Some(next);
|
||||||
|
*command_message = format!(
|
||||||
|
"Selected: {}",
|
||||||
|
add_logic_state.script_editor_suggestions[next]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
KeyCode::Up => {
|
||||||
|
if !add_logic_state.script_editor_suggestions.is_empty() {
|
||||||
|
let current = add_logic_state
|
||||||
|
.script_editor_selected_suggestion_index
|
||||||
|
.unwrap_or(0);
|
||||||
|
let prev = if current == 0 {
|
||||||
|
add_logic_state.script_editor_suggestions.len() - 1
|
||||||
|
} else {
|
||||||
|
current - 1
|
||||||
|
};
|
||||||
|
add_logic_state.script_editor_selected_suggestion_index = Some(prev);
|
||||||
|
*command_message = format!(
|
||||||
|
"Selected: {}",
|
||||||
|
add_logic_state.script_editor_suggestions[prev]
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
KeyCode::Enter => {
|
||||||
|
if let Some(selected_idx) =
|
||||||
|
add_logic_state.script_editor_selected_suggestion_index
|
||||||
|
{
|
||||||
|
if let Some(suggestion) = add_logic_state
|
||||||
|
.script_editor_suggestions
|
||||||
|
.get(selected_idx)
|
||||||
|
.cloned()
|
||||||
|
{
|
||||||
|
let trigger_pos =
|
||||||
|
add_logic_state.script_editor_trigger_position;
|
||||||
|
let filter_len =
|
||||||
|
add_logic_state.script_editor_filter_text.len();
|
||||||
|
|
||||||
|
add_logic_state.deactivate_script_editor_autocomplete();
|
||||||
|
add_logic_state.has_unsaved_changes = true;
|
||||||
|
|
||||||
|
if let Some(pos) = trigger_pos {
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
|
||||||
|
if suggestion == "sql" {
|
||||||
|
replace_autocomplete_text(
|
||||||
|
&mut editor_borrow,
|
||||||
|
pos,
|
||||||
|
filter_len,
|
||||||
|
"sql",
|
||||||
|
);
|
||||||
|
editor_borrow.insert_str("('')");
|
||||||
|
editor_borrow.move_cursor(CursorMove::Back);
|
||||||
|
editor_borrow.move_cursor(CursorMove::Back);
|
||||||
|
*command_message = "Inserted: @sql('')".to_string();
|
||||||
|
} else {
|
||||||
|
let is_table_selection =
|
||||||
|
add_logic_state.is_table_name_suggestion(&suggestion);
|
||||||
|
replace_autocomplete_text(
|
||||||
|
&mut editor_borrow,
|
||||||
|
pos,
|
||||||
|
filter_len,
|
||||||
|
&suggestion,
|
||||||
|
);
|
||||||
|
|
||||||
|
if is_table_selection {
|
||||||
|
editor_borrow.insert_str(".");
|
||||||
|
let new_cursor = editor_borrow.cursor();
|
||||||
|
drop(editor_borrow);
|
||||||
|
|
||||||
|
add_logic_state.script_editor_trigger_position =
|
||||||
|
Some(new_cursor);
|
||||||
|
add_logic_state.script_editor_autocomplete_active = true;
|
||||||
|
add_logic_state.script_editor_filter_text.clear();
|
||||||
|
add_logic_state
|
||||||
|
.trigger_column_autocomplete_for_table(
|
||||||
|
suggestion.clone(),
|
||||||
|
);
|
||||||
|
|
||||||
|
let profile_name =
|
||||||
|
add_logic_state.profile_name.clone();
|
||||||
|
let table_name_for_fetch = suggestion.clone();
|
||||||
|
let mut client_clone = grpc_client.clone();
|
||||||
|
tokio::spawn(async move {
|
||||||
|
if let Err(e) = UiService::fetch_columns_for_table(
|
||||||
|
&mut client_clone,
|
||||||
|
&profile_name,
|
||||||
|
&table_name_for_fetch,
|
||||||
|
)
|
||||||
|
.await
|
||||||
|
{
|
||||||
|
tracing::error!(
|
||||||
|
"Failed to fetch columns for {}.{}: {}",
|
||||||
|
profile_name,
|
||||||
|
table_name_for_fetch,
|
||||||
|
e
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
*command_message = format!(
|
||||||
|
"Selected table '{}', fetching columns...",
|
||||||
|
suggestion
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
*command_message =
|
||||||
|
format!("Inserted: {}", suggestion);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
add_logic_state.deactivate_script_editor_autocomplete();
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
KeyCode::Esc => {
|
||||||
|
add_logic_state.deactivate_script_editor_autocomplete();
|
||||||
|
*command_message = "Autocomplete cancelled".to_string();
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
add_logic_state.deactivate_script_editor_autocomplete();
|
||||||
|
*command_message = "Autocomplete cancelled".to_string();
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Trigger autocomplete with '@'
|
||||||
|
if key_event.code == KeyCode::Char('@') && key_event.modifiers == KeyModifiers::NONE {
|
||||||
|
let should_trigger = match add_logic_state.editor_keybinding_mode {
|
||||||
|
EditorKeybindingMode::Vim => {
|
||||||
|
TextEditor::is_vim_insert_mode(&add_logic_state.vim_state)
|
||||||
|
}
|
||||||
|
_ => true,
|
||||||
|
};
|
||||||
|
if should_trigger {
|
||||||
|
let cursor_before = {
|
||||||
|
let editor_borrow = add_logic_state.script_content_editor.borrow();
|
||||||
|
editor_borrow.cursor()
|
||||||
|
};
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
add_logic_state.script_editor_trigger_position = Some(cursor_before);
|
||||||
|
add_logic_state.script_editor_autocomplete_active = true;
|
||||||
|
add_logic_state.script_editor_filter_text.clear();
|
||||||
|
add_logic_state.update_script_editor_suggestions();
|
||||||
|
add_logic_state.has_unsaved_changes = true;
|
||||||
|
*command_message = "Autocomplete: @ (Tab/↑↓ to navigate, Enter to select, Esc to cancel)".to_string();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Esc handling
|
||||||
|
if key_event.code == KeyCode::Esc && key_event.modifiers == KeyModifiers::NONE {
|
||||||
|
match add_logic_state.editor_keybinding_mode {
|
||||||
|
EditorKeybindingMode::Vim => {
|
||||||
|
let was_insert =
|
||||||
|
TextEditor::is_vim_insert_mode(&add_logic_state.vim_state);
|
||||||
|
{
|
||||||
|
let mut editor_borrow =
|
||||||
|
add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
if was_insert {
|
||||||
|
*command_message =
|
||||||
|
"VIM: Normal Mode. Esc again to exit script.".to_string();
|
||||||
|
} else {
|
||||||
|
add_logic_state.current_focus =
|
||||||
|
AddLogicFocus::ScriptContentPreview;
|
||||||
|
app_state.ui.focus_outside_canvas = true;
|
||||||
|
*command_message = "Exited script editing.".to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
add_logic_state.current_focus = AddLogicFocus::ScriptContentPreview;
|
||||||
|
app_state.ui.focus_outside_canvas = true;
|
||||||
|
*command_message = "Exited script editing.".to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Normal text input
|
||||||
|
let changed = {
|
||||||
|
let mut editor_borrow = add_logic_state.script_content_editor.borrow_mut();
|
||||||
|
TextEditor::handle_input(
|
||||||
|
&mut editor_borrow,
|
||||||
|
key_event,
|
||||||
|
&add_logic_state.editor_keybinding_mode,
|
||||||
|
&mut add_logic_state.vim_state,
|
||||||
|
)
|
||||||
|
};
|
||||||
|
if changed {
|
||||||
|
add_logic_state.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
|
// === NON-FULLSCREEN NAVIGATION ===
|
||||||
|
let action = config.get_general_action(key_event.code, key_event.modifiers);
|
||||||
|
let current_focus = add_logic_state.current_focus;
|
||||||
|
let mut handled = true;
|
||||||
|
let mut new_focus = current_focus;
|
||||||
|
|
||||||
|
match action.as_deref() {
|
||||||
|
Some("exit_table_scroll") => {
|
||||||
|
handled = false;
|
||||||
|
}
|
||||||
|
Some("move_up") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName => {}
|
||||||
|
AddLogicFocus::InputTargetColumn => new_focus = AddLogicFocus::InputLogicName,
|
||||||
|
AddLogicFocus::InputDescription => {
|
||||||
|
new_focus = AddLogicFocus::InputTargetColumn
|
||||||
|
}
|
||||||
|
AddLogicFocus::ScriptContentPreview => {
|
||||||
|
new_focus = AddLogicFocus::InputDescription
|
||||||
|
}
|
||||||
|
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::ScriptContentPreview,
|
||||||
|
AddLogicFocus::CancelButton => new_focus = AddLogicFocus::SaveButton,
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some("move_down") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName => {
|
||||||
|
new_focus = AddLogicFocus::InputTargetColumn
|
||||||
|
}
|
||||||
|
AddLogicFocus::InputTargetColumn => {
|
||||||
|
new_focus = AddLogicFocus::InputDescription
|
||||||
|
}
|
||||||
|
AddLogicFocus::InputDescription => {
|
||||||
|
add_logic_state.last_canvas_field = 2;
|
||||||
|
new_focus = AddLogicFocus::ScriptContentPreview;
|
||||||
|
}
|
||||||
|
AddLogicFocus::ScriptContentPreview => {
|
||||||
|
new_focus = AddLogicFocus::SaveButton
|
||||||
|
}
|
||||||
|
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::CancelButton,
|
||||||
|
AddLogicFocus::CancelButton => {}
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some("next_option") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName
|
||||||
|
| AddLogicFocus::InputTargetColumn
|
||||||
|
| AddLogicFocus::InputDescription => {
|
||||||
|
new_focus = AddLogicFocus::ScriptContentPreview
|
||||||
|
}
|
||||||
|
AddLogicFocus::ScriptContentPreview => {
|
||||||
|
new_focus = AddLogicFocus::SaveButton
|
||||||
|
}
|
||||||
|
AddLogicFocus::SaveButton => new_focus = AddLogicFocus::CancelButton,
|
||||||
|
AddLogicFocus::CancelButton => {}
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some("previous_option") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName
|
||||||
|
| AddLogicFocus::InputTargetColumn
|
||||||
|
| AddLogicFocus::InputDescription => {}
|
||||||
|
AddLogicFocus::ScriptContentPreview => {
|
||||||
|
new_focus = AddLogicFocus::InputDescription
|
||||||
|
}
|
||||||
|
AddLogicFocus::SaveButton => {
|
||||||
|
new_focus = AddLogicFocus::ScriptContentPreview
|
||||||
|
}
|
||||||
|
AddLogicFocus::CancelButton => new_focus = AddLogicFocus::SaveButton,
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some("next_field") => {
|
||||||
|
new_focus = match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName => AddLogicFocus::InputTargetColumn,
|
||||||
|
AddLogicFocus::InputTargetColumn => AddLogicFocus::InputDescription,
|
||||||
|
AddLogicFocus::InputDescription => AddLogicFocus::ScriptContentPreview,
|
||||||
|
AddLogicFocus::ScriptContentPreview => AddLogicFocus::SaveButton,
|
||||||
|
AddLogicFocus::SaveButton => AddLogicFocus::CancelButton,
|
||||||
|
AddLogicFocus::CancelButton => AddLogicFocus::InputLogicName,
|
||||||
|
_ => current_focus,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Some("prev_field") => {
|
||||||
|
new_focus = match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName => AddLogicFocus::CancelButton,
|
||||||
|
AddLogicFocus::InputTargetColumn => AddLogicFocus::InputLogicName,
|
||||||
|
AddLogicFocus::InputDescription => AddLogicFocus::InputTargetColumn,
|
||||||
|
AddLogicFocus::ScriptContentPreview => AddLogicFocus::InputDescription,
|
||||||
|
AddLogicFocus::SaveButton => AddLogicFocus::ScriptContentPreview,
|
||||||
|
AddLogicFocus::CancelButton => AddLogicFocus::SaveButton,
|
||||||
|
_ => current_focus,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
Some("select") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::ScriptContentPreview => {
|
||||||
|
new_focus = AddLogicFocus::InsideScriptContent;
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
let mode_hint = match add_logic_state.editor_keybinding_mode {
|
||||||
|
EditorKeybindingMode::Vim => {
|
||||||
|
"VIM mode - 'i'/'a'/'o' to edit"
|
||||||
|
}
|
||||||
|
_ => "Enter/Ctrl+E to edit",
|
||||||
|
};
|
||||||
|
*command_message = format!(
|
||||||
|
"Fullscreen script editing. {} or Esc to exit.",
|
||||||
|
mode_hint
|
||||||
|
);
|
||||||
|
handled = true;
|
||||||
|
}
|
||||||
|
AddLogicFocus::SaveButton => {
|
||||||
|
*command_message = "Save logic action".to_string();
|
||||||
|
handled = true;
|
||||||
|
}
|
||||||
|
AddLogicFocus::CancelButton => {
|
||||||
|
buffer_state.update_history(AppView::Admin);
|
||||||
|
*command_message = "Cancelled Add Logic".to_string();
|
||||||
|
handled = true;
|
||||||
|
}
|
||||||
|
AddLogicFocus::InputLogicName
|
||||||
|
| AddLogicFocus::InputTargetColumn
|
||||||
|
| AddLogicFocus::InputDescription => {
|
||||||
|
// Focus canvas inputs; let canvas keymap handle editing
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
handled = false; // forward to canvas
|
||||||
|
}
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Some("toggle_edit_mode") => {
|
||||||
|
match current_focus {
|
||||||
|
AddLogicFocus::InputLogicName
|
||||||
|
| AddLogicFocus::InputTargetColumn
|
||||||
|
| AddLogicFocus::InputDescription => {
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
*command_message =
|
||||||
|
"Focus moved to input. Use i/a (Vim) or type to edit.".to_string();
|
||||||
|
handled = true;
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
*command_message = "Cannot toggle edit mode here.".to_string();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => handled = false,
|
||||||
|
}
|
||||||
|
|
||||||
|
if handled && current_focus != new_focus {
|
||||||
|
add_logic_state.current_focus = new_focus;
|
||||||
|
let new_is_canvas_input_focus = matches!(
|
||||||
|
new_focus,
|
||||||
|
AddLogicFocus::InputLogicName
|
||||||
|
| AddLogicFocus::InputTargetColumn
|
||||||
|
| AddLogicFocus::InputDescription
|
||||||
|
);
|
||||||
|
if new_is_canvas_input_focus {
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
} else {
|
||||||
|
app_state.ui.focus_outside_canvas = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
handled
|
||||||
|
} else {
|
||||||
|
false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn replace_autocomplete_text(
|
||||||
|
editor: &mut tui_textarea::TextArea,
|
||||||
|
trigger_pos: (usize, usize),
|
||||||
|
filter_len: usize,
|
||||||
|
replacement: &str,
|
||||||
|
) {
|
||||||
|
let filter_start_pos = (trigger_pos.0, trigger_pos.1 + 1);
|
||||||
|
editor.move_cursor(CursorMove::Jump(filter_start_pos.0 as u16, filter_start_pos.1 as u16));
|
||||||
|
for _ in 0..filter_len {
|
||||||
|
editor.delete_next_char();
|
||||||
|
}
|
||||||
|
editor.insert_str(replacement);
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
// src/state/pages/add_logic.rs
|
// src/pages/admin_panel/add_logic/state.rs
|
||||||
use crate::config::binds::config::{EditorConfig, EditorKeybindingMode};
|
use crate::config::binds::config::{EditorConfig, EditorKeybindingMode};
|
||||||
use crate::components::common::text_editor::{TextEditor, VimState};
|
use crate::components::common::text_editor::{TextEditor, VimState};
|
||||||
use canvas::{DataProvider, AppMode};
|
use canvas::{DataProvider, AppMode};
|
||||||
@@ -54,7 +54,7 @@ pub struct AddLogicState {
|
|||||||
// New fields for same-profile table names and column autocomplete
|
// New fields for same-profile table names and column autocomplete
|
||||||
pub same_profile_table_names: Vec<String>, // Tables from same profile only
|
pub same_profile_table_names: Vec<String>, // Tables from same profile only
|
||||||
pub script_editor_awaiting_column_autocomplete: Option<String>, // Table name waiting for column fetch
|
pub script_editor_awaiting_column_autocomplete: Option<String>, // Table name waiting for column fetch
|
||||||
pub app_mode: AppMode,
|
pub app_mode: canvas::AppMode,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl AddLogicState {
|
impl AddLogicState {
|
||||||
@@ -92,7 +92,7 @@ impl AddLogicState {
|
|||||||
|
|
||||||
same_profile_table_names: Vec::new(),
|
same_profile_table_names: Vec::new(),
|
||||||
script_editor_awaiting_column_autocomplete: None,
|
script_editor_awaiting_column_autocomplete: None,
|
||||||
app_mode: AppMode::Edit,
|
app_mode: canvas::AppMode::Edit,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -272,7 +272,7 @@ impl AddLogicState {
|
|||||||
impl Default for AddLogicState {
|
impl Default for AddLogicState {
|
||||||
fn default() -> Self {
|
fn default() -> Self {
|
||||||
let mut state = Self::new(&EditorConfig::default());
|
let mut state = Self::new(&EditorConfig::default());
|
||||||
state.app_mode = AppMode::Edit;
|
state.app_mode = canvas::AppMode::Edit;
|
||||||
state
|
state
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
// src/components/admin/add_logic.rs
|
// src/pages/admin_panel/add_logic/ui.rs
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::add_logic::{AddLogicFocus, AddLogicState};
|
use crate::pages::admin_panel::add_logic::state::{AddLogicFocus, AddLogicState};
|
||||||
use canvas::{render_canvas, FormEditor};
|
use canvas::{render_canvas, FormEditor};
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
||||||
@@ -10,7 +10,8 @@ use ratatui::{
|
|||||||
widgets::{Block, BorderType, Borders, Paragraph},
|
widgets::{Block, BorderType, Borders, Paragraph},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use crate::components::common::{dialog, autocomplete}; // Added autocomplete
|
use crate::components::common::autocomplete;
|
||||||
|
use crate::dialog;
|
||||||
use crate::config::binds::config::EditorKeybindingMode;
|
use crate::config::binds::config::EditorKeybindingMode;
|
||||||
|
|
||||||
pub fn render_add_logic(
|
pub fn render_add_logic(
|
||||||
@@ -19,7 +20,6 @@ pub fn render_add_logic(
|
|||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
add_logic_state: &mut AddLogicState,
|
add_logic_state: &mut AddLogicState,
|
||||||
is_edit_mode: bool,
|
|
||||||
) {
|
) {
|
||||||
let main_block = Block::default()
|
let main_block = Block::default()
|
||||||
.title(" Add New Logic Script ")
|
.title(" Add New Logic Script ")
|
||||||
@@ -34,7 +34,11 @@ pub fn render_add_logic(
|
|||||||
// Handle full-screen script editing
|
// Handle full-screen script editing
|
||||||
if add_logic_state.current_focus == AddLogicFocus::InsideScriptContent {
|
if add_logic_state.current_focus == AddLogicFocus::InsideScriptContent {
|
||||||
let mut editor_ref = add_logic_state.script_content_editor.borrow_mut();
|
let mut editor_ref = add_logic_state.script_content_editor.borrow_mut();
|
||||||
let border_style_color = if is_edit_mode { theme.highlight } else { theme.secondary };
|
let border_style_color = if crate::components::common::text_editor::TextEditor::is_vim_insert_mode(&add_logic_state.vim_state) {
|
||||||
|
theme.highlight
|
||||||
|
} else {
|
||||||
|
theme.secondary
|
||||||
|
};
|
||||||
let border_style = Style::default().fg(border_style_color);
|
let border_style = Style::default().fg(border_style_color);
|
||||||
|
|
||||||
editor_ref.set_cursor_line_style(Style::default());
|
editor_ref.set_cursor_line_style(Style::default());
|
||||||
@@ -46,7 +50,7 @@ pub fn render_add_logic(
|
|||||||
format!("Script {}", vim_mode_status)
|
format!("Script {}", vim_mode_status)
|
||||||
}
|
}
|
||||||
EditorKeybindingMode::Emacs | EditorKeybindingMode::Default => {
|
EditorKeybindingMode::Emacs | EditorKeybindingMode::Default => {
|
||||||
if is_edit_mode {
|
if crate::components::common::text_editor::TextEditor::is_vim_insert_mode(&add_logic_state.vim_state) {
|
||||||
"Script (Editing)".to_string()
|
"Script (Editing)".to_string()
|
||||||
} else {
|
} else {
|
||||||
"Script".to_string()
|
"Script".to_string()
|
||||||
@@ -161,8 +165,7 @@ pub fn render_add_logic(
|
|||||||
let active_field_rect = render_canvas(f, canvas_area, &editor, theme);
|
let active_field_rect = render_canvas(f, canvas_area, &editor, theme);
|
||||||
|
|
||||||
// --- Render Autocomplete for Target Column ---
|
// --- Render Autocomplete for Target Column ---
|
||||||
// `is_edit_mode` here refers to the general edit mode of the EventHandler
|
if editor.mode() == canvas::AppMode::Edit && editor.current_field() == 1 { // Target Column field
|
||||||
if is_edit_mode && editor.current_field() == 1 { // Target Column field
|
|
||||||
if add_logic_state.in_target_column_suggestion_mode && add_logic_state.show_target_column_suggestions {
|
if add_logic_state.in_target_column_suggestion_mode && add_logic_state.show_target_column_suggestions {
|
||||||
if !add_logic_state.target_column_suggestions.is_empty() {
|
if !add_logic_state.target_column_suggestions.is_empty() {
|
||||||
if let Some(input_rect) = active_field_rect {
|
if let Some(input_rect) = active_field_rect {
|
||||||
@@ -1,7 +1,6 @@
|
|||||||
// src/tui/functions/common/add_table.rs
|
// src/pages/admin_panel/add_table/logic.rs
|
||||||
use crate::state::pages::add_table::{
|
use crate::pages::admin_panel::add_table::state;
|
||||||
AddTableFocus, AddTableState, ColumnDefinition, IndexDefinition,
|
use crate::pages::admin_panel::add_table::state::{AddTableState, AddTableFocus, IndexDefinition, ColumnDefinition};
|
||||||
};
|
|
||||||
use crate::services::GrpcClient;
|
use crate::services::GrpcClient;
|
||||||
use anyhow::{anyhow, Result};
|
use anyhow::{anyhow, Result};
|
||||||
use common::proto::komp_ac::table_definition::{
|
use common::proto::komp_ac::table_definition::{
|
||||||
6
client/src/pages/admin_panel/add_table/mod.rs
Normal file
6
client/src/pages/admin_panel/add_table/mod.rs
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
// src/pages/admin_panel/add_table/mod.rs
|
||||||
|
|
||||||
|
pub mod ui;
|
||||||
|
pub mod nav;
|
||||||
|
pub mod state;
|
||||||
|
pub mod logic;
|
||||||
@@ -1,12 +1,13 @@
|
|||||||
// src/functions/modes/navigation/add_table_nav.rs
|
// src/pages/admin_panel/add_table/nav.rs
|
||||||
|
|
||||||
use crate::config::binds::config::Config;
|
use crate::config::binds::config::Config;
|
||||||
use crate::state::{
|
use crate::state::{
|
||||||
app::state::AppState,
|
app::state::AppState,
|
||||||
pages::add_table::{AddTableFocus, AddTableState},
|
|
||||||
};
|
};
|
||||||
|
use crate::pages::admin_panel::add_table::state::{AddTableFocus, AddTableState};
|
||||||
use crossterm::event::{KeyEvent};
|
use crossterm::event::{KeyEvent};
|
||||||
use ratatui::widgets::TableState;
|
use ratatui::widgets::TableState;
|
||||||
use crate::tui::functions::common::add_table::{handle_add_column_action, handle_save_table_action};
|
use crate::pages::admin_panel::add_table::logic::{handle_add_column_action, handle_save_table_action};
|
||||||
use crate::ui::handlers::context::DialogPurpose;
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
use crate::services::GrpcClient;
|
use crate::services::GrpcClient;
|
||||||
use tokio::sync::mpsc;
|
use tokio::sync::mpsc;
|
||||||
383
client/src/pages/admin_panel/add_table/state.rs
Normal file
383
client/src/pages/admin_panel/add_table/state.rs
Normal file
@@ -0,0 +1,383 @@
|
|||||||
|
// src/pages/admin_panel/add_table/state.rs
|
||||||
|
|
||||||
|
use canvas::{DataProvider, AppMode};
|
||||||
|
use ratatui::widgets::TableState;
|
||||||
|
use serde::{Deserialize, Serialize};
|
||||||
|
use crate::movement::{move_focus, MovementAction};
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
|
pub struct ColumnDefinition {
|
||||||
|
pub name: String,
|
||||||
|
pub data_type: String,
|
||||||
|
pub selected: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq, Serialize, Deserialize)]
|
||||||
|
pub struct IndexDefinition {
|
||||||
|
pub name: String,
|
||||||
|
pub selected: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, PartialEq, Eq)]
|
||||||
|
pub struct LinkDefinition {
|
||||||
|
pub linked_table_name: String,
|
||||||
|
pub is_required: bool,
|
||||||
|
pub selected: bool,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone, Copy, PartialEq, Eq, Default)]
|
||||||
|
pub enum AddTableFocus {
|
||||||
|
#[default]
|
||||||
|
InputTableName, // Field 0 for CanvasState
|
||||||
|
InputColumnName, // Field 1 for CanvasState
|
||||||
|
InputColumnType, // Field 2 for CanvasState
|
||||||
|
AddColumnButton,
|
||||||
|
// Result Tables
|
||||||
|
ColumnsTable,
|
||||||
|
IndexesTable,
|
||||||
|
LinksTable,
|
||||||
|
// Inside Tables (Scrolling Focus)
|
||||||
|
InsideColumnsTable,
|
||||||
|
InsideIndexesTable,
|
||||||
|
InsideLinksTable,
|
||||||
|
// Buttons
|
||||||
|
SaveButton,
|
||||||
|
DeleteSelectedButton,
|
||||||
|
CancelButton,
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct AddTableState {
|
||||||
|
pub profile_name: String,
|
||||||
|
pub table_name: String,
|
||||||
|
pub table_name_input: String,
|
||||||
|
pub column_name_input: String,
|
||||||
|
pub column_type_input: String,
|
||||||
|
pub columns: Vec<ColumnDefinition>,
|
||||||
|
pub indexes: Vec<IndexDefinition>,
|
||||||
|
pub links: Vec<LinkDefinition>,
|
||||||
|
pub current_focus: AddTableFocus,
|
||||||
|
pub last_canvas_field: usize,
|
||||||
|
pub column_table_state: TableState,
|
||||||
|
pub index_table_state: TableState,
|
||||||
|
pub link_table_state: TableState,
|
||||||
|
pub table_name_cursor_pos: usize,
|
||||||
|
pub column_name_cursor_pos: usize,
|
||||||
|
pub column_type_cursor_pos: usize,
|
||||||
|
pub has_unsaved_changes: bool,
|
||||||
|
pub app_mode: canvas::AppMode,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for AddTableState {
|
||||||
|
fn default() -> Self {
|
||||||
|
AddTableState {
|
||||||
|
profile_name: "default".to_string(),
|
||||||
|
table_name: String::new(),
|
||||||
|
table_name_input: String::new(),
|
||||||
|
column_name_input: String::new(),
|
||||||
|
column_type_input: String::new(),
|
||||||
|
columns: Vec::new(),
|
||||||
|
indexes: Vec::new(),
|
||||||
|
links: Vec::new(),
|
||||||
|
current_focus: AddTableFocus::InputTableName,
|
||||||
|
last_canvas_field: 2,
|
||||||
|
column_table_state: TableState::default(),
|
||||||
|
index_table_state: TableState::default(),
|
||||||
|
link_table_state: TableState::default(),
|
||||||
|
table_name_cursor_pos: 0,
|
||||||
|
column_name_cursor_pos: 0,
|
||||||
|
column_type_cursor_pos: 0,
|
||||||
|
has_unsaved_changes: false,
|
||||||
|
app_mode: canvas::AppMode::Edit,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl AddTableState {
|
||||||
|
pub const INPUT_FIELD_COUNT: usize = 3;
|
||||||
|
|
||||||
|
/// Helper method to add a column from current inputs
|
||||||
|
pub fn add_column_from_inputs(&mut self) -> Option<String> {
|
||||||
|
if self.column_name_input.trim().is_empty() || self.column_type_input.trim().is_empty() {
|
||||||
|
return Some("Both column name and type are required".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Check for duplicate column names
|
||||||
|
if self.columns.iter().any(|col| col.name == self.column_name_input.trim()) {
|
||||||
|
return Some("Column name already exists".to_string());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Add the column
|
||||||
|
self.columns.push(ColumnDefinition {
|
||||||
|
name: self.column_name_input.trim().to_string(),
|
||||||
|
data_type: self.column_type_input.trim().to_string(),
|
||||||
|
selected: false,
|
||||||
|
});
|
||||||
|
|
||||||
|
// Clear inputs and reset focus to column name for next entry
|
||||||
|
self.column_name_input.clear();
|
||||||
|
self.column_type_input.clear();
|
||||||
|
self.column_name_cursor_pos = 0;
|
||||||
|
self.column_type_cursor_pos = 0;
|
||||||
|
self.current_focus = AddTableFocus::InputColumnName;
|
||||||
|
self.last_canvas_field = 1;
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
|
||||||
|
Some(format!("Column '{}' added successfully", self.columns.last().unwrap().name))
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Helper method to delete selected items
|
||||||
|
pub fn delete_selected_items(&mut self) -> Option<String> {
|
||||||
|
let mut deleted_items = Vec::new();
|
||||||
|
|
||||||
|
// Remove selected columns
|
||||||
|
let initial_column_count = self.columns.len();
|
||||||
|
self.columns.retain(|col| {
|
||||||
|
if col.selected {
|
||||||
|
deleted_items.push(format!("column '{}'", col.name));
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Remove selected indexes
|
||||||
|
let initial_index_count = self.indexes.len();
|
||||||
|
self.indexes.retain(|idx| {
|
||||||
|
if idx.selected {
|
||||||
|
deleted_items.push(format!("index '{}'", idx.name));
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Remove selected links
|
||||||
|
let initial_link_count = self.links.len();
|
||||||
|
self.links.retain(|link| {
|
||||||
|
if link.selected {
|
||||||
|
deleted_items.push(format!("link to '{}'", link.linked_table_name));
|
||||||
|
false
|
||||||
|
} else {
|
||||||
|
true
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if deleted_items.is_empty() {
|
||||||
|
Some("No items selected for deletion".to_string())
|
||||||
|
} else {
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
Some(format!("Deleted: {}", deleted_items.join(", ")))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DataProvider for AddTableState {
|
||||||
|
fn field_count(&self) -> usize {
|
||||||
|
3 // Table name, Column name, Column type
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_name(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => "Table name",
|
||||||
|
1 => "Name",
|
||||||
|
2 => "Type",
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_value(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => &self.table_name_input,
|
||||||
|
1 => &self.column_name_input,
|
||||||
|
2 => &self.column_type_input,
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_field_value(&mut self, index: usize, value: String) {
|
||||||
|
match index {
|
||||||
|
0 => self.table_name_input = value,
|
||||||
|
1 => self.column_name_input = value,
|
||||||
|
2 => self.column_type_input = value,
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn supports_suggestions(&self, _field_index: usize) -> bool {
|
||||||
|
false // AddTableState doesn’t use suggestions
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
impl AddTableState {
|
||||||
|
pub fn handle_movement(&mut self, action: MovementAction) -> bool {
|
||||||
|
use AddTableFocus::*;
|
||||||
|
|
||||||
|
// Linear outer focus order
|
||||||
|
const ORDER: [AddTableFocus; 10] = [
|
||||||
|
InputTableName,
|
||||||
|
InputColumnName,
|
||||||
|
InputColumnType,
|
||||||
|
AddColumnButton,
|
||||||
|
ColumnsTable,
|
||||||
|
IndexesTable,
|
||||||
|
LinksTable,
|
||||||
|
SaveButton,
|
||||||
|
DeleteSelectedButton,
|
||||||
|
CancelButton,
|
||||||
|
];
|
||||||
|
|
||||||
|
// Enter "inside" on Select from outer panes
|
||||||
|
match (self.current_focus, action) {
|
||||||
|
(ColumnsTable, MovementAction::Select) => {
|
||||||
|
if !self.columns.is_empty() && self.column_table_state.selected().is_none() {
|
||||||
|
self.column_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
self.current_focus = InsideColumnsTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
(IndexesTable, MovementAction::Select) => {
|
||||||
|
if !self.indexes.is_empty() && self.index_table_state.selected().is_none() {
|
||||||
|
self.index_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
self.current_focus = InsideIndexesTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
(LinksTable, MovementAction::Select) => {
|
||||||
|
if !self.links.is_empty() && self.link_table_state.selected().is_none() {
|
||||||
|
self.link_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
self.current_focus = InsideLinksTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Handle "inside" states: Up/Down/Select/Esc; block outer movement keys
|
||||||
|
match self.current_focus {
|
||||||
|
InsideColumnsTable => {
|
||||||
|
match action {
|
||||||
|
MovementAction::Up => {
|
||||||
|
if let Some(i) = self.column_table_state.selected() {
|
||||||
|
let next = i.saturating_sub(1);
|
||||||
|
self.column_table_state.select(Some(next));
|
||||||
|
} else if !self.columns.is_empty() {
|
||||||
|
self.column_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Down => {
|
||||||
|
if let Some(i) = self.column_table_state.selected() {
|
||||||
|
let last = self.columns.len().saturating_sub(1);
|
||||||
|
let next = if i < last { i + 1 } else { i };
|
||||||
|
self.column_table_state.select(Some(next));
|
||||||
|
} else if !self.columns.is_empty() {
|
||||||
|
self.column_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Select => {
|
||||||
|
if let Some(i) = self.column_table_state.selected() {
|
||||||
|
if let Some(col) = self.columns.get_mut(i) {
|
||||||
|
col.selected = !col.selected;
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
self.column_table_state.select(None);
|
||||||
|
self.current_focus = ColumnsTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Previous => return true, // block outer moves
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
InsideIndexesTable => {
|
||||||
|
match action {
|
||||||
|
MovementAction::Up => {
|
||||||
|
if let Some(i) = self.index_table_state.selected() {
|
||||||
|
let next = i.saturating_sub(1);
|
||||||
|
self.index_table_state.select(Some(next));
|
||||||
|
} else if !self.indexes.is_empty() {
|
||||||
|
self.index_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Down => {
|
||||||
|
if let Some(i) = self.index_table_state.selected() {
|
||||||
|
let last = self.indexes.len().saturating_sub(1);
|
||||||
|
let next = if i < last { i + 1 } else { i };
|
||||||
|
self.index_table_state.select(Some(next));
|
||||||
|
} else if !self.indexes.is_empty() {
|
||||||
|
self.index_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Select => {
|
||||||
|
if let Some(i) = self.index_table_state.selected() {
|
||||||
|
if let Some(ix) = self.indexes.get_mut(i) {
|
||||||
|
ix.selected = !ix.selected;
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
self.index_table_state.select(None);
|
||||||
|
self.current_focus = IndexesTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Previous => return true, // block outer moves
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
InsideLinksTable => {
|
||||||
|
match action {
|
||||||
|
MovementAction::Up => {
|
||||||
|
if let Some(i) = self.link_table_state.selected() {
|
||||||
|
let next = i.saturating_sub(1);
|
||||||
|
self.link_table_state.select(Some(next));
|
||||||
|
} else if !self.links.is_empty() {
|
||||||
|
self.link_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Down => {
|
||||||
|
if let Some(i) = self.link_table_state.selected() {
|
||||||
|
let last = self.links.len().saturating_sub(1);
|
||||||
|
let next = if i < last { i + 1 } else { i };
|
||||||
|
self.link_table_state.select(Some(next));
|
||||||
|
} else if !self.links.is_empty() {
|
||||||
|
self.link_table_state.select(Some(0));
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Select => {
|
||||||
|
if let Some(i) = self.link_table_state.selected() {
|
||||||
|
if let Some(link) = self.links.get_mut(i) {
|
||||||
|
link.selected = !link.selected;
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
self.link_table_state.select(None);
|
||||||
|
self.current_focus = LinksTable;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
MovementAction::Next | MovementAction::Previous => return true, // block outer moves
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default: outer navigation via helper
|
||||||
|
move_focus(&ORDER, &mut self.current_focus, action)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,8 +1,8 @@
|
|||||||
// src/components/admin/add_table.rs
|
// src/pages/admin_panel/add_table/ui.rs
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::add_table::{AddTableFocus, AddTableState};
|
use crate::pages::admin_panel::add_table::state::{AddTableFocus, AddTableState};
|
||||||
use canvas::{render_canvas_default, render_canvas, FormEditor};
|
use canvas::{render_canvas, FormEditor};
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
||||||
style::{Modifier, Style},
|
style::{Modifier, Style},
|
||||||
@@ -10,7 +10,7 @@ use ratatui::{
|
|||||||
widgets::{Block, BorderType, Borders, Cell, Paragraph, Row, Table},
|
widgets::{Block, BorderType, Borders, Cell, Paragraph, Row, Table},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use crate::components::common::dialog;
|
use crate::dialog;
|
||||||
|
|
||||||
/// Renders the Add New Table page layout, structuring the display of table information,
|
/// Renders the Add New Table page layout, structuring the display of table information,
|
||||||
/// input fields, and action buttons. Adapts layout based on terminal width.
|
/// input fields, and action buttons. Adapts layout based on terminal width.
|
||||||
@@ -20,7 +20,6 @@ pub fn render_add_table(
|
|||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
add_table_state: &mut AddTableState,
|
add_table_state: &mut AddTableState,
|
||||||
is_edit_mode: bool, // Determines if canvas inputs are in edit mode
|
|
||||||
) {
|
) {
|
||||||
// --- Configuration ---
|
// --- Configuration ---
|
||||||
// Threshold width to switch between wide and narrow layouts
|
// Threshold width to switch between wide and narrow layouts
|
||||||
4
client/src/pages/admin_panel/mod.rs
Normal file
4
client/src/pages/admin_panel/mod.rs
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
// src/pages/admin_panel/mod.rs
|
||||||
|
|
||||||
|
pub mod add_table;
|
||||||
|
pub mod add_logic;
|
||||||
62
client/src/pages/forms/event.rs
Normal file
62
client/src/pages/forms/event.rs
Normal file
@@ -0,0 +1,62 @@
|
|||||||
|
// src/pages/forms/event.rs
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::event::Event;
|
||||||
|
use canvas::keymap::KeyEventOutcome;
|
||||||
|
use crate::{
|
||||||
|
state::app::state::AppState,
|
||||||
|
pages::forms::{FormState, logic},
|
||||||
|
modes::handlers::event::EventOutcome,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub fn handle_form_event(
|
||||||
|
event: Event,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
path: &str,
|
||||||
|
ideal_cursor_column: &mut usize,
|
||||||
|
) -> Result<EventOutcome> {
|
||||||
|
if let Event::Key(key_event) = event {
|
||||||
|
if let Some(editor) = app_state.editor_for_path(path) {
|
||||||
|
match editor.handle_key_event(key_event) {
|
||||||
|
KeyEventOutcome::Consumed(Some(msg)) => {
|
||||||
|
return Ok(EventOutcome::Ok(msg));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Consumed(None) => {
|
||||||
|
return Ok(EventOutcome::Ok("Form input updated".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Pending => {
|
||||||
|
return Ok(EventOutcome::Ok("Waiting for next key...".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::NotMatched => {
|
||||||
|
// fall through to navigation / save / revert
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(EventOutcome::Ok(String::new()))
|
||||||
|
}
|
||||||
|
|
||||||
|
// Save wrapper
|
||||||
|
pub async fn save_form(
|
||||||
|
app_state: &mut AppState,
|
||||||
|
path: &str,
|
||||||
|
grpc_client: &mut crate::services::grpc_client::GrpcClient,
|
||||||
|
) -> Result<EventOutcome> {
|
||||||
|
let outcome = logic::save(app_state, path, grpc_client).await?;
|
||||||
|
let message = match outcome {
|
||||||
|
logic::SaveOutcome::NoChange => "No changes to save.".to_string(),
|
||||||
|
logic::SaveOutcome::UpdatedExisting => "Entry updated.".to_string(),
|
||||||
|
logic::SaveOutcome::CreatedNew(_) => "New entry created.".to_string(),
|
||||||
|
};
|
||||||
|
Ok(EventOutcome::DataSaved(outcome, message))
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn revert_form(
|
||||||
|
app_state: &mut AppState,
|
||||||
|
path: &str,
|
||||||
|
grpc_client: &mut crate::services::grpc_client::GrpcClient,
|
||||||
|
) -> Result<EventOutcome> {
|
||||||
|
let message = logic::revert(app_state, path, grpc_client).await?;
|
||||||
|
Ok(EventOutcome::Ok(message))
|
||||||
|
}
|
||||||
39
client/src/pages/forms/loader.rs
Normal file
39
client/src/pages/forms/loader.rs
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
// src/pages/forms/loader.rs
|
||||||
|
use anyhow::{Context, Result};
|
||||||
|
use crate::{
|
||||||
|
state::app::state::AppState,
|
||||||
|
services::grpc_client::GrpcClient,
|
||||||
|
services::ui_service::UiService, // ✅ import UiService
|
||||||
|
config::binds::Config,
|
||||||
|
pages::forms::FormState,
|
||||||
|
};
|
||||||
|
|
||||||
|
pub async fn ensure_form_loaded_and_count(
|
||||||
|
grpc_client: &mut GrpcClient,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
config: &Config,
|
||||||
|
profile: &str,
|
||||||
|
table: &str,
|
||||||
|
) -> Result<()> {
|
||||||
|
let path = format!("{}/{}", profile, table);
|
||||||
|
|
||||||
|
app_state.ensure_form_editor(&path, config, || {
|
||||||
|
FormState::new(profile.to_string(), table.to_string(), vec![])
|
||||||
|
});
|
||||||
|
|
||||||
|
if let Some(form_state) = app_state.form_state_for_path(&path) {
|
||||||
|
UiService::fetch_and_set_table_count(grpc_client, form_state)
|
||||||
|
.await
|
||||||
|
.context("Failed to fetch table count")?;
|
||||||
|
|
||||||
|
if form_state.total_count > 0 {
|
||||||
|
UiService::load_table_data_by_position(grpc_client, form_state)
|
||||||
|
.await
|
||||||
|
.context("Failed to load table data")?;
|
||||||
|
} else {
|
||||||
|
form_state.reset_to_empty();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
@@ -1,6 +1,7 @@
|
|||||||
// src/tui/functions/common/form.rs
|
// src/pages/forms/logic.rs
|
||||||
use crate::services::grpc_client::GrpcClient;
|
use crate::services::grpc_client::GrpcClient;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::pages::forms::FormState;
|
||||||
use crate::utils::data_converter;
|
use crate::utils::data_converter;
|
||||||
use anyhow::{anyhow, Context, Result};
|
use anyhow::{anyhow, Context, Result};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
@@ -14,14 +15,14 @@ pub enum SaveOutcome {
|
|||||||
|
|
||||||
pub async fn save(
|
pub async fn save(
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
|
path: &str,
|
||||||
grpc_client: &mut GrpcClient,
|
grpc_client: &mut GrpcClient,
|
||||||
) -> Result<SaveOutcome> {
|
) -> Result<SaveOutcome> {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
if !fs.has_unsaved_changes {
|
if !fs.has_unsaved_changes {
|
||||||
return Ok(SaveOutcome::NoChange);
|
return Ok(SaveOutcome::NoChange);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Copy out what we need before dropping the mutable borrow
|
|
||||||
let profile_name = fs.profile_name.clone();
|
let profile_name = fs.profile_name.clone();
|
||||||
let table_name = fs.table_name.clone();
|
let table_name = fs.table_name.clone();
|
||||||
let fields = fs.fields.clone();
|
let fields = fs.fields.clone();
|
||||||
@@ -62,7 +63,7 @@ pub async fn save(
|
|||||||
.context("Failed to post new table data")?;
|
.context("Failed to post new table data")?;
|
||||||
|
|
||||||
if response.success {
|
if response.success {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
fs.id = response.inserted_id;
|
fs.id = response.inserted_id;
|
||||||
fs.total_count += 1;
|
fs.total_count += 1;
|
||||||
fs.current_position = fs.total_count;
|
fs.current_position = fs.total_count;
|
||||||
@@ -84,7 +85,7 @@ pub async fn save(
|
|||||||
.context("Failed to put (update) table data")?;
|
.context("Failed to put (update) table data")?;
|
||||||
|
|
||||||
if response.success {
|
if response.success {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
fs.has_unsaved_changes = false;
|
fs.has_unsaved_changes = false;
|
||||||
}
|
}
|
||||||
SaveOutcome::UpdatedExisting
|
SaveOutcome::UpdatedExisting
|
||||||
@@ -101,9 +102,10 @@ pub async fn save(
|
|||||||
|
|
||||||
pub async fn revert(
|
pub async fn revert(
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
|
path: &str,
|
||||||
grpc_client: &mut GrpcClient,
|
grpc_client: &mut GrpcClient,
|
||||||
) -> Result<String> {
|
) -> Result<String> {
|
||||||
if let Some(fs) = app_state.form_state_mut() {
|
if let Some(fs) = app_state.form_state_for_path(path) {
|
||||||
if fs.id == 0
|
if fs.id == 0
|
||||||
|| (fs.total_count > 0 && fs.current_position > fs.total_count)
|
|| (fs.total_count > 0 && fs.current_position > fs.total_count)
|
||||||
|| (fs.total_count == 0 && fs.current_position == 1)
|
|| (fs.total_count == 0 && fs.current_position == 1)
|
||||||
@@ -146,3 +148,37 @@ pub async fn revert(
|
|||||||
Ok("Nothing to revert".to_string())
|
Ok("Nothing to revert".to_string())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
pub async fn handle_action(
|
||||||
|
action: &str,
|
||||||
|
form_state: &mut FormState,
|
||||||
|
_grpc_client: &mut GrpcClient,
|
||||||
|
ideal_cursor_column: &mut usize,
|
||||||
|
) -> Result<String> {
|
||||||
|
if form_state.has_unsaved_changes() {
|
||||||
|
return Ok(
|
||||||
|
"Unsaved changes. Save (Ctrl+S) or Revert (Ctrl+R) before navigating."
|
||||||
|
.to_string(),
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
let total_count = form_state.total_count;
|
||||||
|
|
||||||
|
match action {
|
||||||
|
"previous_entry" => {
|
||||||
|
if form_state.current_position > 1 {
|
||||||
|
form_state.current_position -= 1;
|
||||||
|
*ideal_cursor_column = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
"next_entry" => {
|
||||||
|
if form_state.current_position <= total_count {
|
||||||
|
form_state.current_position += 1;
|
||||||
|
*ideal_cursor_column = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => return Err(anyhow!("Unknown form action: {}", action)),
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(String::new())
|
||||||
|
}
|
||||||
13
client/src/pages/forms/mod.rs
Normal file
13
client/src/pages/forms/mod.rs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
// src/pages/forms/mod.rs
|
||||||
|
|
||||||
|
pub mod ui;
|
||||||
|
pub mod state;
|
||||||
|
pub mod logic;
|
||||||
|
pub mod event;
|
||||||
|
pub mod loader;
|
||||||
|
|
||||||
|
pub use ui::*;
|
||||||
|
pub use state::*;
|
||||||
|
pub use logic::*;
|
||||||
|
pub use event::*;
|
||||||
|
pub use loader::*;
|
||||||
@@ -1,10 +1,7 @@
|
|||||||
// src/state/pages/form.rs
|
// src/pages/forms/state.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use canvas::{DataProvider, AppMode};
|
||||||
use canvas::{DataProvider, AppMode, EditorState, FormEditor};
|
|
||||||
use common::proto::komp_ac::search::search_response::Hit;
|
use common::proto::komp_ac::search::search_response::Hit;
|
||||||
use ratatui::layout::Rect;
|
|
||||||
use ratatui::Frame;
|
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
|
|
||||||
fn json_value_to_string(value: &serde_json::Value) -> String {
|
fn json_value_to_string(value: &serde_json::Value) -> String {
|
||||||
@@ -24,7 +21,7 @@ pub struct FieldDefinition {
|
|||||||
pub link_target_table: Option<String>,
|
pub link_target_table: Option<String>,
|
||||||
}
|
}
|
||||||
|
|
||||||
#[derive(Clone)]
|
#[derive(Debug, Clone)]
|
||||||
pub struct FormState {
|
pub struct FormState {
|
||||||
pub id: i64,
|
pub id: i64,
|
||||||
pub profile_name: String,
|
pub profile_name: String,
|
||||||
@@ -75,7 +72,7 @@ impl FormState {
|
|||||||
selected_suggestion_index: None,
|
selected_suggestion_index: None,
|
||||||
autocomplete_loading: false,
|
autocomplete_loading: false,
|
||||||
link_display_map: HashMap::new(),
|
link_display_map: HashMap::new(),
|
||||||
app_mode: AppMode::Edit,
|
app_mode: canvas::AppMode::Edit,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1,23 +1,20 @@
|
|||||||
// src/components/form/form.rs
|
// src/pages/forms/ui.rs
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::app::state::AppState;
|
|
||||||
use crate::state::pages::form::FormState;
|
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Margin, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Margin, Rect},
|
||||||
style::Style,
|
style::Style,
|
||||||
widgets::{Block, Borders, Paragraph},
|
widgets::{Block, Borders, Paragraph},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use canvas::canvas::HighlightState;
|
|
||||||
use canvas::{
|
use canvas::{
|
||||||
render_canvas, render_suggestions_dropdown, DefaultCanvasTheme, FormEditor,
|
render_canvas, render_suggestions_dropdown, DefaultCanvasTheme, FormEditor,
|
||||||
};
|
};
|
||||||
|
use crate::pages::forms::FormState;
|
||||||
|
|
||||||
pub fn render_form(
|
pub fn render_form_page(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
area: Rect,
|
area: Rect,
|
||||||
app_state: &AppState,
|
editor: &FormEditor<FormState>,
|
||||||
form_state: &FormState, // not needed directly anymore, editor holds it
|
|
||||||
table_name: &str,
|
table_name: &str,
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
total_count: u64,
|
total_count: u64,
|
||||||
@@ -62,10 +59,7 @@ pub fn render_form(
|
|||||||
f.render_widget(count_para, main_layout[0]);
|
f.render_widget(count_para, main_layout[0]);
|
||||||
|
|
||||||
// --- FORM RENDERING (Using persistent FormEditor) ---
|
// --- FORM RENDERING (Using persistent FormEditor) ---
|
||||||
if let Some(editor) = &app_state.form_editor {
|
|
||||||
let active_field_rect = render_canvas(f, main_layout[1], editor, theme);
|
let active_field_rect = render_canvas(f, main_layout[1], editor, theme);
|
||||||
|
|
||||||
// --- SUGGESTIONS DROPDOWN ---
|
|
||||||
if let Some(active_rect) = active_field_rect {
|
if let Some(active_rect) = active_field_rect {
|
||||||
render_suggestions_dropdown(
|
render_suggestions_dropdown(
|
||||||
f,
|
f,
|
||||||
@@ -76,4 +70,3 @@ pub fn render_form(
|
|||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
76
client/src/pages/intro/logic.rs
Normal file
76
client/src/pages/intro/logic.rs
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
// src/pages/intro/logic.rs
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::buffer::state::{AppView, BufferState};
|
||||||
|
|
||||||
|
/// Handles intro screen selection by updating view history and managing focus state.
|
||||||
|
/// 0: Continue (restores last form or default)
|
||||||
|
/// 1: Admin view
|
||||||
|
/// 2: Login view
|
||||||
|
/// 3: Register view (with focus reset)
|
||||||
|
pub fn handle_intro_selection(
|
||||||
|
app_state: &mut AppState,
|
||||||
|
buffer_state: &mut BufferState,
|
||||||
|
index: usize,
|
||||||
|
) {
|
||||||
|
match index {
|
||||||
|
// Continue: go to the most recent existing Form tab, or open a sensible default
|
||||||
|
0 => {
|
||||||
|
// 1) Try to switch to an already open Form buffer (most recent)
|
||||||
|
if let Some(existing_path) = buffer_state
|
||||||
|
.history
|
||||||
|
.iter()
|
||||||
|
.rev()
|
||||||
|
.find_map(|view| {
|
||||||
|
if let AppView::Form(p) = view {
|
||||||
|
Some(p.clone())
|
||||||
|
} else {
|
||||||
|
None
|
||||||
|
}
|
||||||
|
})
|
||||||
|
{
|
||||||
|
buffer_state.update_history(AppView::Form(existing_path));
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2) Otherwise pick a fallback path
|
||||||
|
let fallback_path = if let (Some(profile), Some(table)) = (
|
||||||
|
app_state.current_view_profile_name.clone(),
|
||||||
|
app_state.current_view_table_name.clone(),
|
||||||
|
) {
|
||||||
|
Some(format!("{}/{}", profile, table))
|
||||||
|
} else if let Some(any_key) = app_state.form_editor.keys().next().cloned() {
|
||||||
|
// Use any existing editor key if available
|
||||||
|
Some(any_key)
|
||||||
|
} else {
|
||||||
|
// Otherwise pick the first available table from the profile tree
|
||||||
|
let mut found: Option<String> = None;
|
||||||
|
for prof in &app_state.profile_tree.profiles {
|
||||||
|
if let Some(tbl) = prof.tables.first() {
|
||||||
|
found = Some(format!("{}/{}", prof.name, tbl.name));
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
found
|
||||||
|
};
|
||||||
|
|
||||||
|
if let Some(path) = fallback_path {
|
||||||
|
buffer_state.update_history(AppView::Form(path));
|
||||||
|
} else {
|
||||||
|
// No sensible default; stay on Intro
|
||||||
|
}
|
||||||
|
}
|
||||||
|
1 => {
|
||||||
|
buffer_state.update_history(AppView::Admin);
|
||||||
|
}
|
||||||
|
2 => {
|
||||||
|
buffer_state.update_history(AppView::Login);
|
||||||
|
}
|
||||||
|
3 => {
|
||||||
|
buffer_state.update_history(AppView::Register);
|
||||||
|
// Register view requires focus reset
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
app_state.focused_button_index = 0;
|
||||||
|
}
|
||||||
|
_ => return,
|
||||||
|
}
|
||||||
|
}
|
||||||
9
client/src/pages/intro/mod.rs
Normal file
9
client/src/pages/intro/mod.rs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
// src/pages/intro/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
|
||||||
|
pub use state::*;
|
||||||
|
pub use ui::render_intro;
|
||||||
|
pub use logic::*;
|
||||||
48
client/src/pages/intro/state.rs
Normal file
48
client/src/pages/intro/state.rs
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
// src/state/pages/intro.rs
|
||||||
|
use crate::movement::MovementAction;
|
||||||
|
|
||||||
|
#[derive(Default, Clone, Debug)]
|
||||||
|
pub struct IntroState {
|
||||||
|
pub selected_option: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl IntroState {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self::default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn next_option(&mut self) {
|
||||||
|
if self.selected_option < 3 {
|
||||||
|
self.selected_option += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn previous_option(&mut self) {
|
||||||
|
if self.selected_option > 0 {
|
||||||
|
self.selected_option -= 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl IntroState {
|
||||||
|
pub fn handle_movement(&mut self, action: MovementAction) -> bool {
|
||||||
|
match action {
|
||||||
|
MovementAction::Next | MovementAction::Right | MovementAction::Down => {
|
||||||
|
self.next_option();
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Previous | MovementAction::Left | MovementAction::Up => {
|
||||||
|
self.previous_option();
|
||||||
|
true
|
||||||
|
}
|
||||||
|
MovementAction::Select => {
|
||||||
|
// Actual selection handled in event loop (UiContext::Intro)
|
||||||
|
false
|
||||||
|
}
|
||||||
|
MovementAction::Esc => {
|
||||||
|
// Nothing special for Intro, but could be used to quit
|
||||||
|
true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
// src/components/intro/intro.rs
|
// src/pages/intro/ui.rs
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
layout::{Alignment, Constraint, Direction, Layout, Rect},
|
||||||
style::Style,
|
style::Style,
|
||||||
@@ -8,7 +8,7 @@ use ratatui::{
|
|||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::pages::intro::IntroState;
|
use crate::pages::intro::IntroState;
|
||||||
|
|
||||||
pub fn render_intro(f: &mut Frame, intro_state: &IntroState, area: Rect, theme: &Theme) {
|
pub fn render_intro(f: &mut Frame, intro_state: &IntroState, area: Rect, theme: &Theme) {
|
||||||
let block = Block::default()
|
let block = Block::default()
|
||||||
73
client/src/pages/login/event.rs
Normal file
73
client/src/pages/login/event.rs
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
// src/pages/login/event.rs
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::event::{Event, KeyCode, KeyModifiers};
|
||||||
|
use canvas::{keymap::KeyEventOutcome, AppMode as CanvasMode};
|
||||||
|
use crate::{
|
||||||
|
state::app::state::AppState,
|
||||||
|
pages::login::LoginFormState,
|
||||||
|
modes::handlers::event::EventOutcome,
|
||||||
|
};
|
||||||
|
use canvas::DataProvider;
|
||||||
|
|
||||||
|
/// Handles all Login page-specific events
|
||||||
|
pub fn handle_login_event(
|
||||||
|
event: Event,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
login_page: &mut LoginFormState,
|
||||||
|
) -> Result<EventOutcome> {
|
||||||
|
if let Event::Key(key_event) = event {
|
||||||
|
let key_code = key_event.code;
|
||||||
|
let modifiers = key_event.modifiers;
|
||||||
|
|
||||||
|
// From buttons (outside) back into the canvas (ReadOnly) with Up/k from the left-most button
|
||||||
|
if login_page.focus_outside_canvas
|
||||||
|
&& login_page.focused_button_index == 0
|
||||||
|
&& matches!(key_code, KeyCode::Up | KeyCode::Char('k'))
|
||||||
|
&& modifiers.is_empty()
|
||||||
|
{
|
||||||
|
login_page.focus_outside_canvas = false;
|
||||||
|
app_state.ui.focus_outside_canvas = false; // 🔑 keep global in sync
|
||||||
|
login_page.editor.set_mode(CanvasMode::ReadOnly);
|
||||||
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Focus handoff: inside canvas → buttons
|
||||||
|
if !login_page.focus_outside_canvas {
|
||||||
|
let last_idx = login_page.editor.data_provider().field_count().saturating_sub(1);
|
||||||
|
let at_last = login_page.editor.current_field() >= last_idx;
|
||||||
|
if at_last
|
||||||
|
&& matches!(
|
||||||
|
(key_code, modifiers),
|
||||||
|
(KeyCode::Char('j'), KeyModifiers::NONE) | (KeyCode::Down, _)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
login_page.focus_outside_canvas = true;
|
||||||
|
login_page.focused_button_index = 0; // focus "Login" button
|
||||||
|
app_state.ui.focus_outside_canvas = true;
|
||||||
|
app_state.focused_button_index = 0;
|
||||||
|
login_page.editor.set_mode(CanvasMode::ReadOnly);
|
||||||
|
return Ok(EventOutcome::Ok("Focus moved to buttons".into()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forward to canvas if focus is inside
|
||||||
|
if !login_page.focus_outside_canvas {
|
||||||
|
match login_page.handle_key_event(key_event) {
|
||||||
|
KeyEventOutcome::Consumed(Some(msg)) => {
|
||||||
|
return Ok(EventOutcome::Ok(msg));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Consumed(None) => {
|
||||||
|
return Ok(EventOutcome::Ok("Login input updated".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Pending => {
|
||||||
|
return Ok(EventOutcome::Ok("Waiting for next key...".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::NotMatched => {
|
||||||
|
// fall through to button handling
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(EventOutcome::Ok(String::new()))
|
||||||
|
}
|
||||||
@@ -1,14 +1,15 @@
|
|||||||
// src/tui/functions/common/login.rs
|
// src/pages/login/logic.rs
|
||||||
|
|
||||||
use crate::services::auth::AuthClient;
|
use crate::services::auth::AuthClient;
|
||||||
use crate::state::pages::auth::AuthState;
|
use crate::state::pages::auth::AuthState;
|
||||||
use crate::state::pages::auth::LoginState;
|
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::app::buffer::{AppView, BufferState};
|
use crate::buffer::state::{AppView, BufferState};
|
||||||
use crate::config::storage::storage::{StoredAuthData, save_auth_data};
|
use crate::config::storage::storage::{StoredAuthData, save_auth_data};
|
||||||
use crate::ui::handlers::context::DialogPurpose;
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
use common::proto::komp_ac::auth::LoginResponse;
|
use common::proto::komp_ac::auth::LoginResponse;
|
||||||
use anyhow::{Context, Result};
|
use crate::pages::login::LoginFormState;
|
||||||
|
use crate::state::pages::auth::UserRole;
|
||||||
|
use anyhow::{Context, Result, anyhow};
|
||||||
use tokio::spawn;
|
use tokio::spawn;
|
||||||
use tokio::sync::mpsc;
|
use tokio::sync::mpsc;
|
||||||
use tracing::{info, error};
|
use tracing::{info, error};
|
||||||
@@ -24,15 +25,14 @@ pub enum LoginResult {
|
|||||||
/// Updates AuthState and AppState on success or failure.
|
/// Updates AuthState and AppState on success or failure.
|
||||||
pub async fn save(
|
pub async fn save(
|
||||||
auth_state: &mut AuthState,
|
auth_state: &mut AuthState,
|
||||||
login_state: &mut LoginState,
|
login_state: &mut LoginFormState,
|
||||||
auth_client: &mut AuthClient,
|
auth_client: &mut AuthClient,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
) -> Result<String> {
|
) -> Result<String> {
|
||||||
let identifier = login_state.username.clone();
|
let identifier = login_state.username().to_string();
|
||||||
let password = login_state.password.clone();
|
let password = login_state.password().to_string();
|
||||||
|
|
||||||
// --- Client-side validation ---
|
// --- Client-side validation ---
|
||||||
// Prevent login attempt if the identifier field is empty or whitespace.
|
|
||||||
if identifier.trim().is_empty() {
|
if identifier.trim().is_empty() {
|
||||||
let error_message = "Username/Email cannot be empty.".to_string();
|
let error_message = "Username/Email cannot be empty.".to_string();
|
||||||
app_state.show_dialog(
|
app_state.show_dialog(
|
||||||
@@ -41,28 +41,28 @@ pub async fn save(
|
|||||||
vec!["OK".to_string()],
|
vec!["OK".to_string()],
|
||||||
DialogPurpose::LoginFailed,
|
DialogPurpose::LoginFailed,
|
||||||
);
|
);
|
||||||
login_state.error_message = Some(error_message.clone());
|
login_state.set_error_message(Some(error_message.clone()));
|
||||||
return Err(anyhow::anyhow!(error_message));
|
return Err(anyhow!(error_message));
|
||||||
}
|
}
|
||||||
|
|
||||||
// Clear previous error/dialog state before attempting
|
// Clear previous error/dialog state before attempting
|
||||||
login_state.error_message = None;
|
login_state.set_error_message(None);
|
||||||
app_state.hide_dialog(); // Hide any previous dialog
|
app_state.hide_dialog();
|
||||||
|
|
||||||
// Call the gRPC login method
|
// Call the gRPC login method
|
||||||
match auth_client.login(identifier.clone(), password).await
|
match auth_client.login(identifier.clone(), password).await
|
||||||
.with_context(|| format!("gRPC login attempt failed for identifier: {}", identifier))
|
.with_context(|| format!("gRPC login attempt failed for identifier: {}", identifier))
|
||||||
{
|
{
|
||||||
Ok(response) => {
|
Ok(response) => {
|
||||||
// Store authentication details using correct field names
|
// Store authentication details
|
||||||
auth_state.auth_token = Some(response.access_token.clone());
|
auth_state.auth_token = Some(response.access_token.clone());
|
||||||
auth_state.user_id = Some(response.user_id.clone());
|
auth_state.user_id = Some(response.user_id.clone());
|
||||||
auth_state.role = Some(response.role.clone());
|
auth_state.role = Some(UserRole::from_str(&response.role));
|
||||||
auth_state.decoded_username = Some(response.username.clone());
|
auth_state.decoded_username = Some(response.username.clone());
|
||||||
login_state.set_has_unsaved_changes(false);
|
|
||||||
login_state.error_message = None;
|
|
||||||
|
|
||||||
// Format the success message using response data
|
login_state.set_has_unsaved_changes(false);
|
||||||
|
login_state.set_error_message(None);
|
||||||
|
|
||||||
let success_message = format!(
|
let success_message = format!(
|
||||||
"Login Successful!\n\n\
|
"Login Successful!\n\n\
|
||||||
Username: {}\n\
|
Username: {}\n\
|
||||||
@@ -79,9 +79,11 @@ pub async fn save(
|
|||||||
vec!["Menu".to_string(), "Exit".to_string()],
|
vec!["Menu".to_string(), "Exit".to_string()],
|
||||||
DialogPurpose::LoginSuccess,
|
DialogPurpose::LoginSuccess,
|
||||||
);
|
);
|
||||||
login_state.password.clear();
|
|
||||||
login_state.username.clear();
|
login_state.username_mut().clear();
|
||||||
login_state.current_cursor_pos = 0;
|
login_state.password_mut().clear();
|
||||||
|
login_state.set_current_cursor_pos(0);
|
||||||
|
|
||||||
Ok("Login successful, details shown in dialog.".to_string())
|
Ok("Login successful, details shown in dialog.".to_string())
|
||||||
}
|
}
|
||||||
Err(e) => {
|
Err(e) => {
|
||||||
@@ -92,10 +94,10 @@ pub async fn save(
|
|||||||
vec!["OK".to_string()],
|
vec!["OK".to_string()],
|
||||||
DialogPurpose::LoginFailed,
|
DialogPurpose::LoginFailed,
|
||||||
);
|
);
|
||||||
login_state.error_message = Some(error_message.clone());
|
login_state.set_error_message(Some(error_message.clone()));
|
||||||
login_state.set_has_unsaved_changes(true);
|
login_state.set_has_unsaved_changes(true);
|
||||||
login_state.username.clear();
|
login_state.username_mut().clear();
|
||||||
login_state.password.clear();
|
login_state.password_mut().clear();
|
||||||
Err(e)
|
Err(e)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -103,39 +105,26 @@ pub async fn save(
|
|||||||
|
|
||||||
/// Reverts the login form fields to empty and returns to the previous screen (Intro).
|
/// Reverts the login form fields to empty and returns to the previous screen (Intro).
|
||||||
pub async fn revert(
|
pub async fn revert(
|
||||||
login_state: &mut LoginState,
|
login_state: &mut LoginFormState,
|
||||||
_app_state: &mut AppState, // Keep signature consistent if needed elsewhere
|
app_state: &mut AppState,
|
||||||
) -> String {
|
) -> String {
|
||||||
// Clear the input fields
|
login_state.clear();
|
||||||
login_state.username.clear();
|
app_state.hide_dialog();
|
||||||
login_state.password.clear();
|
|
||||||
login_state.error_message = None;
|
|
||||||
login_state.set_has_unsaved_changes(false);
|
|
||||||
login_state.login_request_pending = false; // Ensure flag is reset on revert
|
|
||||||
|
|
||||||
"Login reverted".to_string()
|
"Login reverted".to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Clears login form and navigates back to main menu.
|
||||||
pub async fn back_to_main(
|
pub async fn back_to_main(
|
||||||
login_state: &mut LoginState,
|
login_state: &mut LoginFormState,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
buffer_state: &mut BufferState,
|
buffer_state: &mut BufferState,
|
||||||
) -> String {
|
) -> String {
|
||||||
// Clear the input fields
|
login_state.clear();
|
||||||
login_state.username.clear();
|
|
||||||
login_state.password.clear();
|
|
||||||
login_state.error_message = None;
|
|
||||||
login_state.set_has_unsaved_changes(false);
|
|
||||||
login_state.login_request_pending = false; // Ensure flag is reset
|
|
||||||
|
|
||||||
// Ensure dialog is hidden if revert is called
|
|
||||||
app_state.hide_dialog();
|
app_state.hide_dialog();
|
||||||
|
|
||||||
// Navigation logic
|
|
||||||
buffer_state.close_active_buffer();
|
buffer_state.close_active_buffer();
|
||||||
buffer_state.update_history(AppView::Intro);
|
buffer_state.update_history(AppView::Intro);
|
||||||
|
|
||||||
// Reset focus state
|
|
||||||
app_state.ui.focus_outside_canvas = false;
|
app_state.ui.focus_outside_canvas = false;
|
||||||
app_state.focused_button_index = 0;
|
app_state.focused_button_index = 0;
|
||||||
|
|
||||||
@@ -144,15 +133,15 @@ pub async fn back_to_main(
|
|||||||
|
|
||||||
/// Validates input, shows loading, and spawns the login task.
|
/// Validates input, shows loading, and spawns the login task.
|
||||||
pub fn initiate_login(
|
pub fn initiate_login(
|
||||||
login_state: &LoginState,
|
login_state: &mut LoginFormState,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
mut auth_client: AuthClient,
|
mut auth_client: AuthClient,
|
||||||
sender: mpsc::Sender<LoginResult>,
|
sender: mpsc::Sender<LoginResult>,
|
||||||
) -> String {
|
) -> String {
|
||||||
let username = login_state.username.clone();
|
login_state.sync_from_editor();
|
||||||
let password = login_state.password.clone();
|
let username = login_state.username().to_string();
|
||||||
|
let password = login_state.password().to_string();
|
||||||
|
|
||||||
// 1. Client-side validation
|
|
||||||
if username.trim().is_empty() {
|
if username.trim().is_empty() {
|
||||||
app_state.show_dialog(
|
app_state.show_dialog(
|
||||||
"Login Failed",
|
"Login Failed",
|
||||||
@@ -162,25 +151,20 @@ pub fn initiate_login(
|
|||||||
);
|
);
|
||||||
"Username cannot be empty.".to_string()
|
"Username cannot be empty.".to_string()
|
||||||
} else {
|
} else {
|
||||||
// 2. Show Loading Dialog
|
|
||||||
app_state.show_loading_dialog("Logging In", "Please wait...");
|
app_state.show_loading_dialog("Logging In", "Please wait...");
|
||||||
|
|
||||||
// 3. Spawn the login task
|
|
||||||
spawn(async move {
|
spawn(async move {
|
||||||
// Use the passed-in (and moved) auth_client directly
|
|
||||||
let login_outcome = match auth_client.login(username.clone(), password).await
|
let login_outcome = match auth_client.login(username.clone(), password).await
|
||||||
.with_context(|| format!("Spawned login task failed for identifier: {}", username))
|
.with_context(|| format!("Spawned login task failed for identifier: {}", username))
|
||||||
{
|
{
|
||||||
Ok(response) => LoginResult::Success(response),
|
Ok(response) => LoginResult::Success(response),
|
||||||
Err(e) => LoginResult::Failure(format!("{}", e)),
|
Err(e) => LoginResult::Failure(format!("{}", e)),
|
||||||
};
|
};
|
||||||
// Send result back to the main UI thread
|
|
||||||
if let Err(e) = sender.send(login_outcome).await {
|
if let Err(e) = sender.send(login_outcome).await {
|
||||||
error!("Failed to send login result: {}", e);
|
error!("Failed to send login result: {}", e);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
// 4. Return immediately
|
|
||||||
"Login initiated.".to_string()
|
"Login initiated.".to_string()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -191,28 +175,24 @@ pub fn handle_login_result(
|
|||||||
result: LoginResult,
|
result: LoginResult,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
auth_state: &mut AuthState,
|
auth_state: &mut AuthState,
|
||||||
login_state: &mut LoginState,
|
login_state: &mut LoginFormState,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
match result {
|
match result {
|
||||||
LoginResult::Success(response) => {
|
LoginResult::Success(response) => {
|
||||||
auth_state.auth_token = Some(response.access_token.clone());
|
auth_state.auth_token = Some(response.access_token.clone());
|
||||||
auth_state.user_id = Some(response.user_id.clone());
|
auth_state.user_id = Some(response.user_id.clone());
|
||||||
auth_state.role = Some(response.role.clone());
|
auth_state.role = Some(UserRole::from_str(&response.role));
|
||||||
auth_state.decoded_username = Some(response.username.clone());
|
auth_state.decoded_username = Some(response.username.clone());
|
||||||
|
|
||||||
// --- NEW: Save auth data to file ---
|
|
||||||
let data_to_store = StoredAuthData {
|
let data_to_store = StoredAuthData {
|
||||||
access_token: response.access_token.clone(),
|
access_token: response.access_token.clone(),
|
||||||
user_id: response.user_id.clone(),
|
user_id: response.user_id.clone(),
|
||||||
role: response.role.clone(),
|
role: response.role.clone(),
|
||||||
username: response.username.clone(),
|
username: response.username.clone(),
|
||||||
};
|
};
|
||||||
|
|
||||||
if let Err(e) = save_auth_data(&data_to_store) {
|
if let Err(e) = save_auth_data(&data_to_store) {
|
||||||
error!("Failed to save auth data to file: {}", e);
|
error!("Failed to save auth data to file: {}", e);
|
||||||
// Continue anyway - user is still logged in for this session
|
|
||||||
}
|
}
|
||||||
// --- END NEW ---
|
|
||||||
|
|
||||||
let success_message = format!(
|
let success_message = format!(
|
||||||
"Login Successful!\n\nUsername: {}\nUser ID: {}\nRole: {}",
|
"Login Successful!\n\nUsername: {}\nUser ID: {}\nRole: {}",
|
||||||
@@ -226,14 +206,28 @@ pub fn handle_login_result(
|
|||||||
info!(message = %success_message, "Login successful");
|
info!(message = %success_message, "Login successful");
|
||||||
}
|
}
|
||||||
LoginResult::Failure(err_msg) | LoginResult::ConnectionError(err_msg) => {
|
LoginResult::Failure(err_msg) | LoginResult::ConnectionError(err_msg) => {
|
||||||
app_state.update_dialog_content(&err_msg, vec!["OK".to_string()], DialogPurpose::LoginFailed);
|
app_state.update_dialog_content(
|
||||||
login_state.error_message = Some(err_msg.clone());
|
&err_msg,
|
||||||
|
vec!["OK".to_string()],
|
||||||
|
DialogPurpose::LoginFailed,
|
||||||
|
);
|
||||||
|
login_state.set_error_message(Some(err_msg.clone()));
|
||||||
error!(error = %err_msg, "Login failed/connection error");
|
error!(error = %err_msg, "Login failed/connection error");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
login_state.username.clear();
|
|
||||||
login_state.password.clear();
|
login_state.username_mut().clear();
|
||||||
|
login_state.password_mut().clear();
|
||||||
login_state.set_has_unsaved_changes(false);
|
login_state.set_has_unsaved_changes(false);
|
||||||
login_state.current_cursor_pos = 0;
|
login_state.set_current_cursor_pos(0);
|
||||||
true // Request redraw as dialog content changed
|
|
||||||
|
true
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn handle_action(action: &str) -> Result<String> {
|
||||||
|
match action {
|
||||||
|
"previous_entry" => Ok("Previous entry not implemented".into()),
|
||||||
|
"next_entry" => Ok("Next entry not implemented".into()),
|
||||||
|
_ => Err(anyhow!("Unknown login action: {}", action)),
|
||||||
|
}
|
||||||
}
|
}
|
||||||
11
client/src/pages/login/mod.rs
Normal file
11
client/src/pages/login/mod.rs
Normal file
@@ -0,0 +1,11 @@
|
|||||||
|
// src/pages/login/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
pub mod event;
|
||||||
|
|
||||||
|
pub use state::*;
|
||||||
|
pub use ui::render_login;
|
||||||
|
pub use logic::*;
|
||||||
|
pub use event::*;
|
||||||
248
client/src/pages/login/state.rs
Normal file
248
client/src/pages/login/state.rs
Normal file
@@ -0,0 +1,248 @@
|
|||||||
|
// src/pages/login/state.rs
|
||||||
|
|
||||||
|
use canvas::{AppMode, DataProvider};
|
||||||
|
use canvas::FormEditor;
|
||||||
|
use std::fmt;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct LoginState {
|
||||||
|
pub username: String,
|
||||||
|
pub password: String,
|
||||||
|
pub error_message: Option<String>,
|
||||||
|
pub current_field: usize,
|
||||||
|
pub current_cursor_pos: usize,
|
||||||
|
pub has_unsaved_changes: bool,
|
||||||
|
pub login_request_pending: bool,
|
||||||
|
pub app_mode: AppMode,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for LoginState {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
username: String::new(),
|
||||||
|
password: String::new(),
|
||||||
|
error_message: None,
|
||||||
|
current_field: 0,
|
||||||
|
current_cursor_pos: 0,
|
||||||
|
has_unsaved_changes: false,
|
||||||
|
login_request_pending: false,
|
||||||
|
app_mode: canvas::AppMode::Edit,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LoginState {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
app_mode: canvas::AppMode::Edit,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_field(&self) -> usize {
|
||||||
|
self.current_field
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_cursor_pos(&self) -> usize {
|
||||||
|
self.current_cursor_pos
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_field(&mut self, index: usize) {
|
||||||
|
if index < 2 {
|
||||||
|
self.current_field = index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_cursor_pos(&mut self, pos: usize) {
|
||||||
|
self.current_cursor_pos = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input(&self) -> &str {
|
||||||
|
match self.current_field {
|
||||||
|
0 => &self.username,
|
||||||
|
1 => &self.password,
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input_mut(&mut self) -> &mut String {
|
||||||
|
match self.current_field {
|
||||||
|
0 => &mut self.username,
|
||||||
|
1 => &mut self.password,
|
||||||
|
_ => panic!("Invalid current_field index in LoginState"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_mode(&self) -> AppMode {
|
||||||
|
self.app_mode
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn has_unsaved_changes(&self) -> bool {
|
||||||
|
self.has_unsaved_changes
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_has_unsaved_changes(&mut self, changed: bool) {
|
||||||
|
self.has_unsaved_changes = changed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Implement DataProvider for LoginState
|
||||||
|
impl DataProvider for LoginState {
|
||||||
|
fn field_count(&self) -> usize {
|
||||||
|
2
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_name(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => "Username/Email",
|
||||||
|
1 => "Password",
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_value(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => &self.username,
|
||||||
|
1 => &self.password,
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_field_value(&mut self, index: usize, value: String) {
|
||||||
|
match index {
|
||||||
|
0 => self.username = value,
|
||||||
|
1 => self.password = value,
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn supports_suggestions(&self, _field_index: usize) -> bool {
|
||||||
|
false // Login form doesn't support suggestions
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wrapper that owns both the raw login state and its editor
|
||||||
|
|
||||||
|
pub struct LoginFormState {
|
||||||
|
pub state: LoginState,
|
||||||
|
pub editor: FormEditor<LoginState>,
|
||||||
|
pub focus_outside_canvas: bool,
|
||||||
|
pub focused_button_index: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
// manual debug because FormEditor doesnt implement debug
|
||||||
|
impl fmt::Debug for LoginFormState {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
f.debug_struct("LoginFormState")
|
||||||
|
.field("state", &self.state) // ✅ only print the data
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl LoginFormState {
|
||||||
|
/// Sync the editor's data provider back into our state
|
||||||
|
pub fn sync_from_editor(&mut self) {
|
||||||
|
// FormEditor holds the authoritative data
|
||||||
|
let dp = self.editor.data_provider();
|
||||||
|
self.state = dp.clone(); // LoginState implements Clone
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Create a new LoginFormState with default LoginState and FormEditor
|
||||||
|
pub fn new() -> Self {
|
||||||
|
let state = LoginState::default();
|
||||||
|
let editor = FormEditor::new(state.clone());
|
||||||
|
Self {
|
||||||
|
state,
|
||||||
|
editor,
|
||||||
|
focus_outside_canvas: false,
|
||||||
|
focused_button_index: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to LoginState fields ===
|
||||||
|
|
||||||
|
pub fn username(&self) -> &str {
|
||||||
|
&self.state.username
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn username_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.username
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn password(&self) -> &str {
|
||||||
|
&self.state.password
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn password_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.password
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn error_message(&self) -> Option<&String> {
|
||||||
|
self.state.error_message.as_ref()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_error_message(&mut self, msg: Option<String>) {
|
||||||
|
self.state.error_message = msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn has_unsaved_changes(&self) -> bool {
|
||||||
|
self.state.has_unsaved_changes
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_has_unsaved_changes(&mut self, changed: bool) {
|
||||||
|
self.state.has_unsaved_changes = changed;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn clear(&mut self) {
|
||||||
|
self.state.username.clear();
|
||||||
|
self.state.password.clear();
|
||||||
|
self.state.error_message = None;
|
||||||
|
self.state.has_unsaved_changes = false;
|
||||||
|
self.state.login_request_pending = false;
|
||||||
|
self.state.current_cursor_pos = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to LoginState cursor/input ===
|
||||||
|
|
||||||
|
pub fn current_field(&self) -> usize {
|
||||||
|
self.state.current_field()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_field(&mut self, index: usize) {
|
||||||
|
self.state.set_current_field(index);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_cursor_pos(&self) -> usize {
|
||||||
|
self.state.current_cursor_pos()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_cursor_pos(&mut self, pos: usize) {
|
||||||
|
self.state.set_current_cursor_pos(pos);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input(&self) -> &str {
|
||||||
|
self.state.get_current_input()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input_mut(&mut self) -> &mut String {
|
||||||
|
self.state.get_current_input_mut()
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to FormEditor ===
|
||||||
|
|
||||||
|
pub fn mode(&self) -> AppMode {
|
||||||
|
self.editor.mode()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn cursor_position(&self) -> usize {
|
||||||
|
self.editor.cursor_position()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_key_event(
|
||||||
|
&mut self,
|
||||||
|
key_event: crossterm::event::KeyEvent,
|
||||||
|
) -> canvas::keymap::KeyEventOutcome {
|
||||||
|
self.editor.handle_key_event(key_event)
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,9 +1,7 @@
|
|||||||
// src/components/auth/login.rs
|
// src/pages/login/ui.rs
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
config::colors::themes::Theme,
|
config::colors::themes::Theme,
|
||||||
state::pages::auth::LoginState,
|
|
||||||
components::common::dialog,
|
|
||||||
state::app::state::AppState,
|
state::app::state::AppState,
|
||||||
};
|
};
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
@@ -19,15 +17,19 @@ use canvas::{
|
|||||||
DefaultCanvasTheme,
|
DefaultCanvasTheme,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
use crate::pages::login::LoginFormState;
|
||||||
|
use crate::dialog;
|
||||||
|
|
||||||
pub fn render_login(
|
pub fn render_login(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
area: Rect,
|
area: Rect,
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
// FIX: take &LoginState (reference), not owned
|
login_page: &LoginFormState,
|
||||||
login_state: &LoginState,
|
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
is_edit_mode: bool,
|
|
||||||
) {
|
) {
|
||||||
|
let login_state = &login_page.state;
|
||||||
|
let editor = &login_page.editor;
|
||||||
|
|
||||||
// Main container
|
// Main container
|
||||||
let block = Block::default()
|
let block = Block::default()
|
||||||
.borders(Borders::ALL)
|
.borders(Borders::ALL)
|
||||||
@@ -53,14 +55,10 @@ pub fn render_login(
|
|||||||
])
|
])
|
||||||
.split(inner_area);
|
.split(inner_area);
|
||||||
|
|
||||||
// Wrap LoginState in FormEditor (no clone needed)
|
|
||||||
let editor = FormEditor::new(login_state.clone());
|
|
||||||
|
|
||||||
// Use DefaultCanvasTheme instead of app Theme
|
|
||||||
let input_rect = render_canvas(
|
let input_rect = render_canvas(
|
||||||
f,
|
f,
|
||||||
chunks[0],
|
chunks[0],
|
||||||
&editor,
|
editor,
|
||||||
&DefaultCanvasTheme,
|
&DefaultCanvasTheme,
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -82,7 +80,7 @@ pub fn render_login(
|
|||||||
|
|
||||||
// Login Button
|
// Login Button
|
||||||
let login_button_index = 0;
|
let login_button_index = 0;
|
||||||
let login_active = if app_state.ui.focus_outside_canvas {
|
let login_active = if login_page.focus_outside_canvas {
|
||||||
app_state.focused_button_index == login_button_index
|
app_state.focused_button_index == login_button_index
|
||||||
} else {
|
} else {
|
||||||
false
|
false
|
||||||
@@ -135,14 +133,14 @@ pub fn render_login(
|
|||||||
);
|
);
|
||||||
|
|
||||||
// --- SUGGESTIONS DROPDOWN (if active) ---
|
// --- SUGGESTIONS DROPDOWN (if active) ---
|
||||||
if app_state.current_mode == crate::modes::handlers::mode_manager::AppMode::Edit {
|
if editor.mode() == canvas::AppMode::Edit {
|
||||||
if let Some(input_rect) = input_rect {
|
if let Some(input_rect) = input_rect {
|
||||||
render_suggestions_dropdown(
|
render_suggestions_dropdown(
|
||||||
f,
|
f,
|
||||||
f.area(),
|
chunks[0],
|
||||||
input_rect,
|
input_rect,
|
||||||
&DefaultCanvasTheme,
|
&DefaultCanvasTheme,
|
||||||
&editor, // FIX: pass &editor
|
editor,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
9
client/src/pages/mod.rs
Normal file
9
client/src/pages/mod.rs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
// src/pages/mod.rs
|
||||||
|
|
||||||
|
pub mod routing;
|
||||||
|
pub mod intro;
|
||||||
|
pub mod login;
|
||||||
|
pub mod register;
|
||||||
|
pub mod forms;
|
||||||
|
pub mod admin;
|
||||||
|
pub mod admin_panel;
|
||||||
76
client/src/pages/register/event.rs
Normal file
76
client/src/pages/register/event.rs
Normal file
@@ -0,0 +1,76 @@
|
|||||||
|
// src/pages/register/event.rs
|
||||||
|
use anyhow::Result;
|
||||||
|
use crossterm::event::{Event, KeyCode, KeyModifiers};
|
||||||
|
use canvas::{keymap::KeyEventOutcome, AppMode as CanvasMode};
|
||||||
|
use canvas::DataProvider;
|
||||||
|
use crate::{
|
||||||
|
state::app::state::AppState,
|
||||||
|
pages::register::RegisterFormState,
|
||||||
|
modes::handlers::event::EventOutcome,
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Handles all Register page-specific events.
|
||||||
|
/// Return a non-empty Ok(message) only when the page actually consumed the key,
|
||||||
|
/// otherwise return Ok("") to let global handling proceed.
|
||||||
|
pub fn handle_register_event(
|
||||||
|
event: Event,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
register_page: &mut RegisterFormState,
|
||||||
|
)-> Result<EventOutcome> {
|
||||||
|
if let Event::Key(key_event) = event {
|
||||||
|
let key_code = key_event.code;
|
||||||
|
let modifiers = key_event.modifiers;
|
||||||
|
|
||||||
|
// From buttons (outside) back into the canvas (ReadOnly) with Up/k from the left-most button
|
||||||
|
if register_page.focus_outside_canvas
|
||||||
|
&& register_page.focused_button_index == 0
|
||||||
|
&& matches!(key_code, KeyCode::Up | KeyCode::Char('k'))
|
||||||
|
&& modifiers.is_empty()
|
||||||
|
{
|
||||||
|
register_page.focus_outside_canvas = false;
|
||||||
|
// Keep global in sync for now (cursor styling elsewhere still reads it)
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
register_page.editor.set_mode(CanvasMode::ReadOnly);
|
||||||
|
return Ok(EventOutcome::Ok(String::new()));
|
||||||
|
}
|
||||||
|
|
||||||
|
// Focus handoff: inside canvas → buttons
|
||||||
|
if !register_page.focus_outside_canvas {
|
||||||
|
let last_idx = register_page.editor.data_provider().field_count().saturating_sub(1);
|
||||||
|
let at_last = register_page.editor.current_field() >= last_idx;
|
||||||
|
if at_last
|
||||||
|
&& matches!(
|
||||||
|
(key_code, modifiers),
|
||||||
|
(KeyCode::Char('j'), KeyModifiers::NONE) | (KeyCode::Down, _)
|
||||||
|
)
|
||||||
|
{
|
||||||
|
register_page.focus_outside_canvas = true;
|
||||||
|
register_page.focused_button_index = 0; // focus "Register" button
|
||||||
|
// Keep global in sync for now
|
||||||
|
app_state.ui.focus_outside_canvas = true;
|
||||||
|
app_state.focused_button_index = 0;
|
||||||
|
register_page.editor.set_mode(CanvasMode::ReadOnly);
|
||||||
|
return Ok(EventOutcome::Ok("Focus moved to buttons".into()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Forward to canvas if focus is inside
|
||||||
|
if !register_page.focus_outside_canvas {
|
||||||
|
match register_page.handle_key_event(key_event) {
|
||||||
|
KeyEventOutcome::Consumed(Some(msg)) => {
|
||||||
|
return Ok(EventOutcome::Ok(msg));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Consumed(None) => {
|
||||||
|
return Ok(EventOutcome::Ok("Register input updated".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::Pending => {
|
||||||
|
return Ok(EventOutcome::Ok("Waiting for next key...".into()));
|
||||||
|
}
|
||||||
|
KeyEventOutcome::NotMatched => {
|
||||||
|
// fall through
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Ok(EventOutcome::Ok(String::new()))
|
||||||
|
}
|
||||||
@@ -1,13 +1,11 @@
|
|||||||
// src/tui/functions/common/register.rs
|
// src/pages/register/logic.rs
|
||||||
|
|
||||||
use crate::services::auth::AuthClient;
|
use crate::services::auth::AuthClient;
|
||||||
use crate::state::{
|
use crate::state::app::state::AppState;
|
||||||
pages::auth::RegisterState,
|
|
||||||
app::state::AppState,
|
|
||||||
};
|
|
||||||
use crate::ui::handlers::context::DialogPurpose;
|
use crate::ui::handlers::context::DialogPurpose;
|
||||||
use crate::state::app::buffer::{AppView, BufferState};
|
use crate::buffer::state::{AppView, BufferState};
|
||||||
use common::proto::komp_ac::auth::AuthResponse;
|
use common::proto::komp_ac::auth::AuthResponse;
|
||||||
|
use crate::pages::register::RegisterFormState;
|
||||||
use anyhow::Context;
|
use anyhow::Context;
|
||||||
use tokio::spawn;
|
use tokio::spawn;
|
||||||
use tokio::sync::mpsc;
|
use tokio::sync::mpsc;
|
||||||
@@ -22,24 +20,26 @@ pub enum RegisterResult {
|
|||||||
|
|
||||||
/// Clears the registration form fields.
|
/// Clears the registration form fields.
|
||||||
pub async fn revert(
|
pub async fn revert(
|
||||||
register_state: &mut RegisterState,
|
register_state: &mut RegisterFormState,
|
||||||
_app_state: &mut AppState, // Keep signature consistent if needed elsewhere
|
app_state: &mut AppState,
|
||||||
) -> String {
|
) -> String {
|
||||||
register_state.username.clear();
|
register_state.username_mut().clear();
|
||||||
register_state.email.clear();
|
register_state.email_mut().clear();
|
||||||
register_state.password.clear();
|
register_state.password_mut().clear();
|
||||||
register_state.password_confirmation.clear();
|
register_state.password_confirmation_mut().clear();
|
||||||
register_state.role.clear();
|
register_state.role_mut().clear();
|
||||||
register_state.error_message = None;
|
register_state.set_error_message(None);
|
||||||
register_state.set_has_unsaved_changes(false);
|
register_state.set_has_unsaved_changes(false);
|
||||||
register_state.current_field = 0; // Reset focus to first field
|
register_state.set_current_field(0); // Reset focus to first field
|
||||||
register_state.current_cursor_pos = 0;
|
register_state.set_current_cursor_pos(0);
|
||||||
|
|
||||||
|
app_state.hide_dialog();
|
||||||
"Registration form cleared".to_string()
|
"Registration form cleared".to_string()
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Clears the form and returns to the intro screen.
|
/// Clears the form and returns to the intro screen.
|
||||||
pub async fn back_to_login(
|
pub async fn back_to_login(
|
||||||
register_state: &mut RegisterState,
|
register_state: &mut RegisterFormState,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
buffer_state: &mut BufferState,
|
buffer_state: &mut BufferState,
|
||||||
) -> String {
|
) -> String {
|
||||||
@@ -54,6 +54,8 @@ pub async fn back_to_login(
|
|||||||
buffer_state.update_history(AppView::Login);
|
buffer_state.update_history(AppView::Login);
|
||||||
|
|
||||||
// Reset focus state
|
// Reset focus state
|
||||||
|
register_state.focus_outside_canvas = false;
|
||||||
|
register_state.focused_button_index = 0;
|
||||||
app_state.ui.focus_outside_canvas = false;
|
app_state.ui.focus_outside_canvas = false;
|
||||||
app_state.focused_button_index = 0;
|
app_state.focused_button_index = 0;
|
||||||
|
|
||||||
@@ -62,24 +64,34 @@ pub async fn back_to_login(
|
|||||||
|
|
||||||
/// Validates input, shows loading, and spawns the registration task.
|
/// Validates input, shows loading, and spawns the registration task.
|
||||||
pub fn initiate_registration(
|
pub fn initiate_registration(
|
||||||
register_state: &RegisterState,
|
register_state: &mut RegisterFormState,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
mut auth_client: AuthClient,
|
mut auth_client: AuthClient,
|
||||||
sender: mpsc::Sender<RegisterResult>,
|
sender: mpsc::Sender<RegisterResult>,
|
||||||
) -> String {
|
) -> String {
|
||||||
// Clone necessary data
|
register_state.sync_from_editor();
|
||||||
let username = register_state.username.clone();
|
let username = register_state.username().to_string();
|
||||||
let email = register_state.email.clone();
|
let email = register_state.email().to_string();
|
||||||
let password = register_state.password.clone();
|
let password = register_state.password().to_string();
|
||||||
let password_confirmation = register_state.password_confirmation.clone();
|
let password_confirmation = register_state.password_confirmation().to_string();
|
||||||
let role = register_state.role.clone();
|
let role = register_state.role().to_string();
|
||||||
|
|
||||||
// 1. Client-side validation
|
// 1. Client-side validation
|
||||||
if username.trim().is_empty() {
|
if username.trim().is_empty() {
|
||||||
app_state.show_dialog("Registration Failed", "Username cannot be empty.", vec!["OK".to_string()], DialogPurpose::RegisterFailed);
|
app_state.show_dialog(
|
||||||
|
"Registration Failed",
|
||||||
|
"Username cannot be empty.",
|
||||||
|
vec!["OK".to_string()],
|
||||||
|
DialogPurpose::RegisterFailed,
|
||||||
|
);
|
||||||
"Username cannot be empty.".to_string()
|
"Username cannot be empty.".to_string()
|
||||||
} else if !password.is_empty() && password != password_confirmation {
|
} else if !password.is_empty() && password != password_confirmation {
|
||||||
app_state.show_dialog("Registration Failed", "Passwords do not match.", vec!["OK".to_string()], DialogPurpose::RegisterFailed);
|
app_state.show_dialog(
|
||||||
|
"Registration Failed",
|
||||||
|
"Passwords do not match.",
|
||||||
|
vec!["OK".to_string()],
|
||||||
|
DialogPurpose::RegisterFailed,
|
||||||
|
);
|
||||||
"Passwords do not match.".to_string()
|
"Passwords do not match.".to_string()
|
||||||
} else {
|
} else {
|
||||||
// 2. Show Loading Dialog
|
// 2. Show Loading Dialog
|
||||||
@@ -88,14 +100,19 @@ pub fn initiate_registration(
|
|||||||
// 3. Spawn the registration task
|
// 3. Spawn the registration task
|
||||||
spawn(async move {
|
spawn(async move {
|
||||||
let password_opt = if password.is_empty() { None } else { Some(password) };
|
let password_opt = if password.is_empty() { None } else { Some(password) };
|
||||||
let password_conf_opt = if password_confirmation.is_empty() { None } else { Some(password_confirmation) };
|
let password_conf_opt =
|
||||||
|
if password_confirmation.is_empty() { None } else { Some(password_confirmation) };
|
||||||
let role_opt = if role.is_empty() { None } else { Some(role) };
|
let role_opt = if role.is_empty() { None } else { Some(role) };
|
||||||
let register_outcome = match auth_client.register(username.clone(), email, password_opt, password_conf_opt, role_opt).await
|
|
||||||
|
let register_outcome = match auth_client
|
||||||
|
.register(username.clone(), email, password_opt, password_conf_opt, role_opt)
|
||||||
|
.await
|
||||||
.with_context(|| format!("Spawned register task failed for username: {}", username))
|
.with_context(|| format!("Spawned register task failed for username: {}", username))
|
||||||
{
|
{
|
||||||
Ok(response) => RegisterResult::Success(response),
|
Ok(response) => RegisterResult::Success(response),
|
||||||
Err(e) => RegisterResult::Failure(format!("{}", e)),
|
Err(e) => RegisterResult::Failure(format!("{}", e)),
|
||||||
};
|
};
|
||||||
|
|
||||||
// Send result back to the main UI thread
|
// Send result back to the main UI thread
|
||||||
if let Err(e) = sender.send(register_outcome).await {
|
if let Err(e) = sender.send(register_outcome).await {
|
||||||
error!("Failed to send registration result: {}", e);
|
error!("Failed to send registration result: {}", e);
|
||||||
@@ -112,7 +129,7 @@ pub fn initiate_registration(
|
|||||||
pub fn handle_registration_result(
|
pub fn handle_registration_result(
|
||||||
result: RegisterResult,
|
result: RegisterResult,
|
||||||
app_state: &mut AppState,
|
app_state: &mut AppState,
|
||||||
register_state: &mut RegisterState,
|
register_state: &mut RegisterFormState,
|
||||||
) -> bool {
|
) -> bool {
|
||||||
match result {
|
match result {
|
||||||
RegisterResult::Success(response) => {
|
RegisterResult::Success(response) => {
|
||||||
@@ -133,7 +150,7 @@ pub fn handle_registration_result(
|
|||||||
vec!["OK".to_string()],
|
vec!["OK".to_string()],
|
||||||
DialogPurpose::RegisterFailed,
|
DialogPurpose::RegisterFailed,
|
||||||
);
|
);
|
||||||
register_state.error_message = Some(err_msg.clone());
|
register_state.set_error_message(Some(err_msg.clone()));
|
||||||
error!(error = %err_msg, "Registration failed/connection error");
|
error!(error = %err_msg, "Registration failed/connection error");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
14
client/src/pages/register/mod.rs
Normal file
14
client/src/pages/register/mod.rs
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
// src/pages/register/mod.rs
|
||||||
|
|
||||||
|
// pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod state;
|
||||||
|
pub mod logic;
|
||||||
|
pub mod suggestions;
|
||||||
|
pub mod event;
|
||||||
|
|
||||||
|
// pub use state::*;
|
||||||
|
pub use ui::render_register;
|
||||||
|
pub use logic::*;
|
||||||
|
pub use state::*;
|
||||||
|
pub use event::*;
|
||||||
370
client/src/pages/register/state.rs
Normal file
370
client/src/pages/register/state.rs
Normal file
@@ -0,0 +1,370 @@
|
|||||||
|
// src/pages/register/state.rs
|
||||||
|
|
||||||
|
use canvas::{DataProvider, AppMode, FormEditor};
|
||||||
|
use std::fmt;
|
||||||
|
|
||||||
|
use crossterm::event::{KeyCode, KeyEvent, KeyModifiers};
|
||||||
|
use canvas::keymap::KeyEventOutcome;
|
||||||
|
use crate::pages::register::suggestions::role_suggestions_sync;
|
||||||
|
|
||||||
|
/// Represents the state of the Registration form UI
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub struct RegisterState {
|
||||||
|
pub username: String,
|
||||||
|
pub email: String,
|
||||||
|
pub password: String,
|
||||||
|
pub password_confirmation: String,
|
||||||
|
pub role: String,
|
||||||
|
pub error_message: Option<String>,
|
||||||
|
pub current_field: usize,
|
||||||
|
pub current_cursor_pos: usize,
|
||||||
|
pub has_unsaved_changes: bool,
|
||||||
|
pub app_mode: canvas::AppMode,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for RegisterState {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self {
|
||||||
|
username: String::new(),
|
||||||
|
email: String::new(),
|
||||||
|
password: String::new(),
|
||||||
|
password_confirmation: String::new(),
|
||||||
|
role: String::new(),
|
||||||
|
error_message: None,
|
||||||
|
current_field: 0,
|
||||||
|
current_cursor_pos: 0,
|
||||||
|
has_unsaved_changes: false,
|
||||||
|
app_mode: canvas::AppMode::Edit,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RegisterState {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
app_mode: canvas::AppMode::Edit,
|
||||||
|
..Default::default()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_field(&self) -> usize {
|
||||||
|
self.current_field
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_cursor_pos(&self) -> usize {
|
||||||
|
self.current_cursor_pos
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_field(&mut self, index: usize) {
|
||||||
|
if index < 5 {
|
||||||
|
self.current_field = index;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_current_cursor_pos(&mut self, pos: usize) {
|
||||||
|
self.current_cursor_pos = pos;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input(&self) -> &str {
|
||||||
|
match self.current_field {
|
||||||
|
0 => &self.username,
|
||||||
|
1 => &self.email,
|
||||||
|
2 => &self.password,
|
||||||
|
3 => &self.password_confirmation,
|
||||||
|
4 => &self.role,
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_current_input_mut(&mut self, index: usize) -> &mut String {
|
||||||
|
match index {
|
||||||
|
0 => &mut self.username,
|
||||||
|
1 => &mut self.email,
|
||||||
|
2 => &mut self.password,
|
||||||
|
3 => &mut self.password_confirmation,
|
||||||
|
4 => &mut self.role,
|
||||||
|
_ => panic!("Invalid current_field index in RegisterState"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn current_mode(&self) -> AppMode {
|
||||||
|
self.app_mode
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn has_unsaved_changes(&self) -> bool {
|
||||||
|
self.has_unsaved_changes
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_has_unsaved_changes(&mut self, changed: bool) {
|
||||||
|
self.has_unsaved_changes = changed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl DataProvider for RegisterState {
|
||||||
|
fn field_count(&self) -> usize { 5 }
|
||||||
|
|
||||||
|
fn field_name(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => "Username",
|
||||||
|
1 => "Email (Optional)",
|
||||||
|
2 => "Password (Optional)",
|
||||||
|
3 => "Confirm Password",
|
||||||
|
4 => "Role (Optional)",
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn field_value(&self, index: usize) -> &str {
|
||||||
|
match index {
|
||||||
|
0 => &self.username,
|
||||||
|
1 => &self.email,
|
||||||
|
2 => &self.password,
|
||||||
|
3 => &self.password_confirmation,
|
||||||
|
4 => &self.role,
|
||||||
|
_ => "",
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
fn set_field_value(&mut self, index: usize, value: String) {
|
||||||
|
match index {
|
||||||
|
0 => self.username = value,
|
||||||
|
1 => self.email = value,
|
||||||
|
2 => self.password = value,
|
||||||
|
3 => self.password_confirmation = value,
|
||||||
|
4 => self.role = value,
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
self.has_unsaved_changes = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
fn supports_suggestions(&self, field_index: usize) -> bool {
|
||||||
|
field_index == 4 // only Role field supports suggestions
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/// Wrapper that owns both the raw register state and its editor
|
||||||
|
pub struct RegisterFormState {
|
||||||
|
pub state: RegisterState,
|
||||||
|
pub editor: FormEditor<RegisterState>,
|
||||||
|
pub focus_outside_canvas: bool,
|
||||||
|
pub focused_button_index: usize,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Default for RegisterFormState {
|
||||||
|
fn default() -> Self {
|
||||||
|
Self::new()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// manual Debug because FormEditor doesn’t implement Debug
|
||||||
|
impl fmt::Debug for RegisterFormState {
|
||||||
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
||||||
|
f.debug_struct("RegisterFormState")
|
||||||
|
.field("state", &self.state)
|
||||||
|
.finish()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl RegisterFormState {
|
||||||
|
/// Sync the editor's data provider back into our state
|
||||||
|
pub fn sync_from_editor(&mut self) {
|
||||||
|
// The FormEditor holds the authoritative data
|
||||||
|
let dp = self.editor.data_provider();
|
||||||
|
self.state = dp.clone(); // because RegisterState: Clone
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new() -> Self {
|
||||||
|
let state = RegisterState::default();
|
||||||
|
let editor = FormEditor::new(state.clone());
|
||||||
|
Self {
|
||||||
|
state,
|
||||||
|
editor,
|
||||||
|
focus_outside_canvas: false,
|
||||||
|
focused_button_index: 0,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to RegisterState ===
|
||||||
|
pub fn username(&self) -> &str {
|
||||||
|
&self.state.username
|
||||||
|
}
|
||||||
|
pub fn username_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.username
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn email(&self) -> &str {
|
||||||
|
&self.state.email
|
||||||
|
}
|
||||||
|
pub fn email_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.email
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn password(&self) -> &str {
|
||||||
|
&self.state.password
|
||||||
|
}
|
||||||
|
pub fn password_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.password
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn password_confirmation(&self) -> &str {
|
||||||
|
&self.state.password_confirmation
|
||||||
|
}
|
||||||
|
pub fn password_confirmation_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.password_confirmation
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn role(&self) -> &str {
|
||||||
|
&self.state.role
|
||||||
|
}
|
||||||
|
pub fn role_mut(&mut self) -> &mut String {
|
||||||
|
&mut self.state.role
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn error_message(&self) -> Option<&String> {
|
||||||
|
self.state.error_message.as_ref()
|
||||||
|
}
|
||||||
|
pub fn set_error_message(&mut self, msg: Option<String>) {
|
||||||
|
self.state.error_message = msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn has_unsaved_changes(&self) -> bool {
|
||||||
|
self.state.has_unsaved_changes
|
||||||
|
}
|
||||||
|
pub fn set_has_unsaved_changes(&mut self, changed: bool) {
|
||||||
|
self.state.has_unsaved_changes = changed;
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn clear(&mut self) {
|
||||||
|
self.state.username.clear();
|
||||||
|
self.state.email.clear();
|
||||||
|
self.state.password.clear();
|
||||||
|
self.state.password_confirmation.clear();
|
||||||
|
self.state.role.clear();
|
||||||
|
self.state.error_message = None;
|
||||||
|
self.state.has_unsaved_changes = false;
|
||||||
|
self.state.current_field = 0;
|
||||||
|
self.state.current_cursor_pos = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to cursor/input ===
|
||||||
|
pub fn current_field(&self) -> usize {
|
||||||
|
self.state.current_field()
|
||||||
|
}
|
||||||
|
pub fn set_current_field(&mut self, index: usize) {
|
||||||
|
self.state.set_current_field(index);
|
||||||
|
}
|
||||||
|
pub fn current_cursor_pos(&self) -> usize {
|
||||||
|
self.state.current_cursor_pos()
|
||||||
|
}
|
||||||
|
pub fn set_current_cursor_pos(&mut self, pos: usize) {
|
||||||
|
self.state.set_current_cursor_pos(pos);
|
||||||
|
}
|
||||||
|
pub fn get_current_input(&self) -> &str {
|
||||||
|
self.state.get_current_input()
|
||||||
|
}
|
||||||
|
pub fn get_current_input_mut(&mut self) -> &mut String {
|
||||||
|
self.state.get_current_input_mut(self.state.current_field)
|
||||||
|
}
|
||||||
|
|
||||||
|
// === Delegates to FormEditor ===
|
||||||
|
pub fn mode(&self) -> canvas::AppMode {
|
||||||
|
self.editor.mode()
|
||||||
|
}
|
||||||
|
pub fn cursor_position(&self) -> usize {
|
||||||
|
self.editor.cursor_position()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn handle_key_event(
|
||||||
|
&mut self,
|
||||||
|
key_event: crossterm::event::KeyEvent,
|
||||||
|
) -> canvas::keymap::KeyEventOutcome {
|
||||||
|
// Only customize behavior for the Role field (index 4) in Edit mode
|
||||||
|
let in_role_field = self.editor.current_field() == 4;
|
||||||
|
let in_edit_mode = self.editor.mode() == canvas::AppMode::Edit;
|
||||||
|
|
||||||
|
if in_role_field && in_edit_mode {
|
||||||
|
match key_event.code {
|
||||||
|
// Tab: open suggestions if inactive; otherwise cycle next
|
||||||
|
KeyCode::Tab => {
|
||||||
|
if !self.editor.is_suggestions_active() {
|
||||||
|
if let Some(query) = self.editor.start_suggestions(4) {
|
||||||
|
let items = role_suggestions_sync(&query);
|
||||||
|
let applied =
|
||||||
|
self.editor.apply_suggestions_result(4, &query, items);
|
||||||
|
if applied {
|
||||||
|
self.editor.update_inline_completion();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Cycle to next suggestion
|
||||||
|
self.editor.suggestions_next();
|
||||||
|
}
|
||||||
|
return KeyEventOutcome::Consumed(None);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Shift+Tab (BackTab): cycle suggestions too (fallback to next)
|
||||||
|
KeyCode::BackTab => {
|
||||||
|
if self.editor.is_suggestions_active() {
|
||||||
|
// If your canvas exposes suggestions_prev(), use it here.
|
||||||
|
// Fallback: cycle next.
|
||||||
|
self.editor.suggestions_next();
|
||||||
|
return KeyEventOutcome::Consumed(None);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enter: if suggestions active — apply selected suggestion
|
||||||
|
KeyCode::Enter => {
|
||||||
|
if self.editor.is_suggestions_active() {
|
||||||
|
let _ = self.editor.apply_suggestion();
|
||||||
|
return KeyEventOutcome::Consumed(None);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Esc: close suggestions if active
|
||||||
|
KeyCode::Esc => {
|
||||||
|
if self.editor.is_suggestions_active() {
|
||||||
|
self.editor.close_suggestions();
|
||||||
|
return KeyEventOutcome::Consumed(None);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Character input: first let editor mutate text, then refilter if active
|
||||||
|
KeyCode::Char(_) => {
|
||||||
|
let outcome = self.editor.handle_key_event(key_event);
|
||||||
|
if self.editor.is_suggestions_active() {
|
||||||
|
if let Some(query) = self.editor.start_suggestions(4) {
|
||||||
|
let items = role_suggestions_sync(&query);
|
||||||
|
let applied =
|
||||||
|
self.editor.apply_suggestions_result(4, &query, items);
|
||||||
|
if applied {
|
||||||
|
self.editor.update_inline_completion();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return outcome;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Backspace/Delete: mutate then refilter if active
|
||||||
|
KeyCode::Backspace | KeyCode::Delete => {
|
||||||
|
let outcome = self.editor.handle_key_event(key_event);
|
||||||
|
if self.editor.is_suggestions_active() {
|
||||||
|
if let Some(query) = self.editor.start_suggestions(4) {
|
||||||
|
let items = role_suggestions_sync(&query);
|
||||||
|
let applied =
|
||||||
|
self.editor.apply_suggestions_result(4, &query, items);
|
||||||
|
if applied {
|
||||||
|
self.editor.update_inline_completion();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return outcome;
|
||||||
|
}
|
||||||
|
|
||||||
|
_ => { /* fall through to default */ }
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default: let canvas handle it
|
||||||
|
self.editor.handle_key_event(key_event)
|
||||||
|
}
|
||||||
|
}
|
||||||
36
client/src/pages/register/suggestions.rs
Normal file
36
client/src/pages/register/suggestions.rs
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
// src/pages/register/suggestions.rs
|
||||||
|
|
||||||
|
use anyhow::Result;
|
||||||
|
use async_trait::async_trait;
|
||||||
|
use canvas::{SuggestionItem, SuggestionsProvider};
|
||||||
|
|
||||||
|
// Keep the async provider if you want, but add this sync helper and shared data.
|
||||||
|
const ROLES: &[&str] = &["admin", "moderator", "accountant", "viewer"];
|
||||||
|
|
||||||
|
pub fn role_suggestions_sync(query: &str) -> Vec<SuggestionItem> {
|
||||||
|
let q = query.to_lowercase();
|
||||||
|
ROLES
|
||||||
|
.iter()
|
||||||
|
.filter(|r| q.is_empty() || r.to_lowercase().contains(&q))
|
||||||
|
.map(|r| SuggestionItem {
|
||||||
|
display_text: (*r).to_string(),
|
||||||
|
value_to_store: (*r).to_string(),
|
||||||
|
})
|
||||||
|
.collect()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct RoleSuggestionsProvider;
|
||||||
|
|
||||||
|
#[async_trait]
|
||||||
|
impl SuggestionsProvider for RoleSuggestionsProvider {
|
||||||
|
async fn fetch_suggestions(
|
||||||
|
&mut self,
|
||||||
|
field_index: usize,
|
||||||
|
query: &str,
|
||||||
|
) -> Result<Vec<SuggestionItem>> {
|
||||||
|
if field_index != 4 {
|
||||||
|
return Ok(Vec::new());
|
||||||
|
}
|
||||||
|
Ok(role_suggestions_sync(query))
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,11 +1,8 @@
|
|||||||
// src/components/auth/register.rs
|
// src/pages/register/ui.rs
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
config::colors::themes::Theme,
|
config::colors::themes::Theme,
|
||||||
state::pages::auth::RegisterState,
|
|
||||||
components::common::dialog,
|
|
||||||
state::app::state::AppState,
|
state::app::state::AppState,
|
||||||
modes::handlers::mode_manager::AppMode,
|
|
||||||
};
|
};
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Alignment, Constraint, Direction, Layout, Rect, Margin},
|
layout::{Alignment, Constraint, Direction, Layout, Rect, Margin},
|
||||||
@@ -13,16 +10,24 @@ use ratatui::{
|
|||||||
widgets::{Block, BorderType, Borders, Paragraph},
|
widgets::{Block, BorderType, Borders, Paragraph},
|
||||||
Frame,
|
Frame,
|
||||||
};
|
};
|
||||||
use canvas::{FormEditor, render_canvas_default, render_canvas, render_suggestions_dropdown, DefaultCanvasTheme};
|
use crate::dialog;
|
||||||
|
use crate::pages::register::RegisterFormState;
|
||||||
|
use crate::pages::register::suggestions::RoleSuggestionsProvider;
|
||||||
|
use tokio::runtime::Handle;
|
||||||
|
use canvas::{render_canvas, render_suggestions_dropdown, DefaultCanvasTheme};
|
||||||
|
use canvas::SuggestionsProvider;
|
||||||
|
|
||||||
pub fn render_register(
|
pub fn render_register(
|
||||||
f: &mut Frame,
|
f: &mut Frame,
|
||||||
area: Rect,
|
area: Rect,
|
||||||
theme: &Theme,
|
theme: &Theme,
|
||||||
state: &RegisterState,
|
register_page: &RegisterFormState,
|
||||||
app_state: &AppState,
|
app_state: &AppState,
|
||||||
is_edit_mode: bool,
|
|
||||||
) {
|
) {
|
||||||
|
let state = ®ister_page.state;
|
||||||
|
let editor = ®ister_page.editor;
|
||||||
|
|
||||||
|
// Outer block
|
||||||
let block = Block::default()
|
let block = Block::default()
|
||||||
.borders(Borders::ALL)
|
.borders(Borders::ALL)
|
||||||
.border_type(BorderType::Plain)
|
.border_type(BorderType::Plain)
|
||||||
@@ -47,15 +52,9 @@ pub fn render_register(
|
|||||||
])
|
])
|
||||||
.split(inner_area);
|
.split(inner_area);
|
||||||
|
|
||||||
// Wrap RegisterState in FormEditor
|
// Render the form canvas
|
||||||
let editor = FormEditor::new(state.clone());
|
let input_rect = render_canvas(f, chunks[0], editor, theme);
|
||||||
|
|
||||||
let input_rect = render_canvas(
|
|
||||||
f,
|
|
||||||
chunks[0],
|
|
||||||
&editor,
|
|
||||||
theme,
|
|
||||||
);
|
|
||||||
|
|
||||||
// --- HELP TEXT ---
|
// --- HELP TEXT ---
|
||||||
let help_text = Paragraph::new("* are optional fields")
|
let help_text = Paragraph::new("* are optional fields")
|
||||||
@@ -81,11 +80,9 @@ pub fn render_register(
|
|||||||
|
|
||||||
// Register Button
|
// Register Button
|
||||||
let register_button_index = 0;
|
let register_button_index = 0;
|
||||||
let register_active = if app_state.ui.focus_outside_canvas {
|
let register_active =
|
||||||
app_state.focused_button_index == register_button_index
|
register_page.focus_outside_canvas
|
||||||
} else {
|
&& register_page.focused_button_index == register_button_index;
|
||||||
false
|
|
||||||
};
|
|
||||||
let mut register_style = Style::default().fg(theme.fg);
|
let mut register_style = Style::default().fg(theme.fg);
|
||||||
let mut register_border = Style::default().fg(theme.border);
|
let mut register_border = Style::default().fg(theme.border);
|
||||||
if register_active {
|
if register_active {
|
||||||
@@ -108,11 +105,9 @@ pub fn render_register(
|
|||||||
|
|
||||||
// Return Button
|
// Return Button
|
||||||
let return_button_index = 1;
|
let return_button_index = 1;
|
||||||
let return_active = if app_state.ui.focus_outside_canvas {
|
let return_active =
|
||||||
app_state.focused_button_index == return_button_index
|
register_page.focus_outside_canvas
|
||||||
} else {
|
&& register_page.focused_button_index == return_button_index;
|
||||||
false
|
|
||||||
};
|
|
||||||
let mut return_style = Style::default().fg(theme.fg);
|
let mut return_style = Style::default().fg(theme.fg);
|
||||||
let mut return_border = Style::default().fg(theme.border);
|
let mut return_border = Style::default().fg(theme.border);
|
||||||
if return_active {
|
if return_active {
|
||||||
@@ -133,19 +128,6 @@ pub fn render_register(
|
|||||||
button_chunks[1],
|
button_chunks[1],
|
||||||
);
|
);
|
||||||
|
|
||||||
// --- AUTOCOMPLETE DROPDOWN (Using new canvas suggestions) ---
|
|
||||||
if app_state.current_mode == AppMode::Edit {
|
|
||||||
if let Some(input_rect) = input_rect {
|
|
||||||
render_suggestions_dropdown(
|
|
||||||
f,
|
|
||||||
f.area(), // Frame area
|
|
||||||
input_rect, // Current input field rect
|
|
||||||
&DefaultCanvasTheme,
|
|
||||||
&editor,
|
|
||||||
);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// --- DIALOG ---
|
// --- DIALOG ---
|
||||||
if app_state.ui.dialog.dialog_show {
|
if app_state.ui.dialog.dialog_show {
|
||||||
dialog::render_dialog(
|
dialog::render_dialog(
|
||||||
@@ -159,4 +141,17 @@ pub fn render_register(
|
|||||||
app_state.ui.dialog.is_loading,
|
app_state.ui.dialog.is_loading,
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Render suggestions dropdown if active (library GUI)
|
||||||
|
if editor.mode() == canvas::AppMode::Edit {
|
||||||
|
if let Some(input_rect) = input_rect {
|
||||||
|
render_suggestions_dropdown(
|
||||||
|
f,
|
||||||
|
f.area(),
|
||||||
|
input_rect,
|
||||||
|
&DefaultCanvasTheme,
|
||||||
|
editor,
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
5
client/src/pages/routing/mod.rs
Normal file
5
client/src/pages/routing/mod.rs
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
// src/pages/routing/mod.rs
|
||||||
|
|
||||||
|
pub mod router;
|
||||||
|
|
||||||
|
pub use router::{Page, Router};
|
||||||
36
client/src/pages/routing/router.rs
Normal file
36
client/src/pages/routing/router.rs
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
// src/pages/routing/router.rs
|
||||||
|
use crate::state::pages::auth::AuthState;
|
||||||
|
use crate::pages::admin_panel::add_logic::state::AddLogicState;
|
||||||
|
use crate::pages::admin_panel::add_table::state::AddTableState;
|
||||||
|
use crate::pages::admin::AdminState;
|
||||||
|
use crate::pages::forms::FormState;
|
||||||
|
use crate::pages::login::LoginFormState;
|
||||||
|
use crate::pages::register::RegisterFormState;
|
||||||
|
use crate::pages::intro::IntroState;
|
||||||
|
|
||||||
|
#[derive(Debug)]
|
||||||
|
pub enum Page {
|
||||||
|
Intro(IntroState),
|
||||||
|
Login(LoginFormState),
|
||||||
|
Register(RegisterFormState),
|
||||||
|
Admin(AdminState),
|
||||||
|
AddLogic(AddLogicState),
|
||||||
|
AddTable(AddTableState),
|
||||||
|
Form(String),
|
||||||
|
}
|
||||||
|
|
||||||
|
pub struct Router {
|
||||||
|
pub current: Page,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Router {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self {
|
||||||
|
current: Page::Intro(IntroState::default()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn navigate(&mut self, page: Page) {
|
||||||
|
self.current = page;
|
||||||
|
}
|
||||||
|
}
|
||||||
113
client/src/search/event.rs
Normal file
113
client/src/search/event.rs
Normal file
@@ -0,0 +1,113 @@
|
|||||||
|
// src/search/event.rs
|
||||||
|
use crate::state::app::state::AppState;
|
||||||
|
use crate::services::grpc_client::GrpcClient;
|
||||||
|
use common::proto::komp_ac::search::search_response::Hit;
|
||||||
|
use crossterm::event::KeyCode;
|
||||||
|
use tokio::sync::mpsc;
|
||||||
|
use tracing::{error, info};
|
||||||
|
use std::collections::HashMap;
|
||||||
|
use anyhow::Result;
|
||||||
|
|
||||||
|
pub async fn handle_search_palette_event(
|
||||||
|
key_event: crossterm::event::KeyEvent,
|
||||||
|
app_state: &mut AppState,
|
||||||
|
grpc_client: &mut GrpcClient,
|
||||||
|
search_result_sender: mpsc::UnboundedSender<Vec<Hit>>,
|
||||||
|
) -> Result<Option<String>> {
|
||||||
|
let mut should_close = false;
|
||||||
|
let mut outcome_message = None;
|
||||||
|
let mut trigger_search = false;
|
||||||
|
|
||||||
|
if let Some(search_state) = app_state.search_state.as_mut() {
|
||||||
|
match key_event.code {
|
||||||
|
KeyCode::Esc => {
|
||||||
|
should_close = true;
|
||||||
|
outcome_message = Some("Search cancelled".to_string());
|
||||||
|
}
|
||||||
|
KeyCode::Enter => {
|
||||||
|
// Step 1: Extract the data we need while holding the borrow
|
||||||
|
let maybe_data = search_state
|
||||||
|
.results
|
||||||
|
.get(search_state.selected_index)
|
||||||
|
.map(|hit| (hit.id, hit.content_json.clone()));
|
||||||
|
|
||||||
|
// Step 2: Process outside the borrow
|
||||||
|
if let Some((id, content_json)) = maybe_data {
|
||||||
|
if let Ok(data) = serde_json::from_str::<HashMap<String, String>>(&content_json) {
|
||||||
|
// Use current view path to access the active form
|
||||||
|
if let (Some(profile), Some(table)) = (
|
||||||
|
app_state.current_view_profile_name.clone(),
|
||||||
|
app_state.current_view_table_name.clone(),
|
||||||
|
) {
|
||||||
|
let path = format!("{}/{}", profile, table);
|
||||||
|
if let Some(fs) = app_state.form_state_for_path(&path) {
|
||||||
|
let detached_pos = fs.total_count + 2;
|
||||||
|
fs.update_from_response(&data, detached_pos);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
should_close = true;
|
||||||
|
outcome_message = Some(format!("Loaded record ID {}", id));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
KeyCode::Up => search_state.previous_result(),
|
||||||
|
KeyCode::Down => search_state.next_result(),
|
||||||
|
KeyCode::Char(c) => {
|
||||||
|
search_state.input.insert(search_state.cursor_position, c);
|
||||||
|
search_state.cursor_position += 1;
|
||||||
|
trigger_search = true;
|
||||||
|
}
|
||||||
|
KeyCode::Backspace => {
|
||||||
|
if search_state.cursor_position > 0 {
|
||||||
|
search_state.cursor_position -= 1;
|
||||||
|
search_state.input.remove(search_state.cursor_position);
|
||||||
|
trigger_search = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
KeyCode::Left => {
|
||||||
|
search_state.cursor_position =
|
||||||
|
search_state.cursor_position.saturating_sub(1);
|
||||||
|
}
|
||||||
|
KeyCode::Right => {
|
||||||
|
if search_state.cursor_position < search_state.input.len() {
|
||||||
|
search_state.cursor_position += 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if trigger_search {
|
||||||
|
if let Some(search_state) = app_state.search_state.as_mut() {
|
||||||
|
search_state.is_loading = true;
|
||||||
|
search_state.results.clear();
|
||||||
|
search_state.selected_index = 0;
|
||||||
|
|
||||||
|
let query = search_state.input.clone();
|
||||||
|
let table_name = search_state.table_name.clone();
|
||||||
|
let sender = search_result_sender.clone();
|
||||||
|
let mut grpc_client = grpc_client.clone();
|
||||||
|
|
||||||
|
info!("Spawning search task for query: '{}'", query);
|
||||||
|
tokio::spawn(async move {
|
||||||
|
match grpc_client.search_table(table_name, query).await {
|
||||||
|
Ok(response) => {
|
||||||
|
let _ = sender.send(response.hits);
|
||||||
|
}
|
||||||
|
Err(e) => {
|
||||||
|
error!("Search failed: {:?}", e);
|
||||||
|
let _ = sender.send(vec![]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if should_close {
|
||||||
|
app_state.search_state = None;
|
||||||
|
app_state.ui.show_search_palette = false;
|
||||||
|
app_state.ui.focus_outside_canvas = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(outcome_message)
|
||||||
|
}
|
||||||
31
client/src/search/grpc.rs
Normal file
31
client/src/search/grpc.rs
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
// src/search/grpc.rs
|
||||||
|
|
||||||
|
use common::proto::komp_ac::search::{
|
||||||
|
searcher_client::SearcherClient, SearchRequest, SearchResponse,
|
||||||
|
};
|
||||||
|
use tonic::transport::Channel;
|
||||||
|
use anyhow::Result;
|
||||||
|
|
||||||
|
/// Internal search gRPC wrapper
|
||||||
|
#[derive(Clone)]
|
||||||
|
pub struct SearchGrpc {
|
||||||
|
client: SearcherClient<Channel>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl SearchGrpc {
|
||||||
|
pub fn new(channel: Channel) -> Self {
|
||||||
|
Self {
|
||||||
|
client: SearcherClient::new(channel),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub async fn search_table(
|
||||||
|
&mut self,
|
||||||
|
table_name: String,
|
||||||
|
query: String,
|
||||||
|
) -> Result<SearchResponse> {
|
||||||
|
let request = tonic::Request::new(SearchRequest { table_name, query });
|
||||||
|
let response = self.client.search_table(request).await?;
|
||||||
|
Ok(response.into_inner())
|
||||||
|
}
|
||||||
|
}
|
||||||
9
client/src/search/mod.rs
Normal file
9
client/src/search/mod.rs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
// src/search/mod.rs
|
||||||
|
|
||||||
|
pub mod state;
|
||||||
|
pub mod ui;
|
||||||
|
pub mod event;
|
||||||
|
pub mod grpc;
|
||||||
|
|
||||||
|
pub use ui::*;
|
||||||
|
pub use grpc::SearchGrpc;
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
// src/state/app/search.rs
|
// src/search/state.rs
|
||||||
|
|
||||||
use common::proto::komp_ac::search::search_response::Hit;
|
use common::proto::komp_ac::search::search_response::Hit;
|
||||||
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
// src/components/common/search_palette.rs
|
// src/search/ui.rs
|
||||||
|
|
||||||
use crate::config::colors::themes::Theme;
|
use crate::config::colors::themes::Theme;
|
||||||
use crate::state::app::search::SearchState;
|
use crate::search::state::SearchState;
|
||||||
use ratatui::{
|
use ratatui::{
|
||||||
layout::{Constraint, Direction, Layout, Rect},
|
layout::{Constraint, Direction, Layout, Rect},
|
||||||
style::{Modifier, Style},
|
style::{Modifier, Style},
|
||||||
@@ -22,9 +22,8 @@ use common::proto::komp_ac::tables_data::{
|
|||||||
PostTableDataRequest, PostTableDataResponse, PutTableDataRequest,
|
PostTableDataRequest, PostTableDataResponse, PutTableDataRequest,
|
||||||
PutTableDataResponse,
|
PutTableDataResponse,
|
||||||
};
|
};
|
||||||
use common::proto::komp_ac::search::{
|
use crate::search::SearchGrpc;
|
||||||
searcher_client::SearcherClient, SearchRequest, SearchResponse,
|
use common::proto::komp_ac::search::SearchResponse;
|
||||||
};
|
|
||||||
use anyhow::{Context, Result};
|
use anyhow::{Context, Result};
|
||||||
use std::collections::HashMap;
|
use std::collections::HashMap;
|
||||||
use tonic::transport::Channel;
|
use tonic::transport::Channel;
|
||||||
@@ -36,7 +35,7 @@ pub struct GrpcClient {
|
|||||||
table_definition_client: TableDefinitionClient<Channel>,
|
table_definition_client: TableDefinitionClient<Channel>,
|
||||||
table_script_client: TableScriptClient<Channel>,
|
table_script_client: TableScriptClient<Channel>,
|
||||||
tables_data_client: TablesDataClient<Channel>,
|
tables_data_client: TablesDataClient<Channel>,
|
||||||
search_client: SearcherClient<Channel>,
|
search_client: SearchGrpc,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl GrpcClient {
|
impl GrpcClient {
|
||||||
@@ -52,7 +51,7 @@ impl GrpcClient {
|
|||||||
TableDefinitionClient::new(channel.clone());
|
TableDefinitionClient::new(channel.clone());
|
||||||
let table_script_client = TableScriptClient::new(channel.clone());
|
let table_script_client = TableScriptClient::new(channel.clone());
|
||||||
let tables_data_client = TablesDataClient::new(channel.clone());
|
let tables_data_client = TablesDataClient::new(channel.clone());
|
||||||
let search_client = SearcherClient::new(channel.clone());
|
let search_client = SearchGrpc::new(channel.clone());
|
||||||
|
|
||||||
Ok(Self {
|
Ok(Self {
|
||||||
table_structure_client,
|
table_structure_client,
|
||||||
@@ -247,11 +246,6 @@ impl GrpcClient {
|
|||||||
table_name: String,
|
table_name: String,
|
||||||
query: String,
|
query: String,
|
||||||
) -> Result<SearchResponse> {
|
) -> Result<SearchResponse> {
|
||||||
let request = tonic::Request::new(SearchRequest { table_name, query });
|
self.search_client.search_table(table_name, query).await
|
||||||
let response = self
|
|
||||||
.search_client
|
|
||||||
.search_table(request)
|
|
||||||
.await?;
|
|
||||||
Ok(response.into_inner())
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -2,10 +2,10 @@
|
|||||||
|
|
||||||
use crate::services::grpc_client::GrpcClient;
|
use crate::services::grpc_client::GrpcClient;
|
||||||
use crate::state::app::state::AppState;
|
use crate::state::app::state::AppState;
|
||||||
use crate::state::pages::add_logic::AddLogicState;
|
use crate::pages::admin_panel::add_logic::state::AddLogicState;
|
||||||
use crate::state::pages::form::{FieldDefinition, FormState};
|
use crate::pages::forms::logic::SaveOutcome;
|
||||||
use crate::tui::functions::common::form::SaveOutcome;
|
|
||||||
use crate::utils::columns::filter_user_columns;
|
use crate::utils::columns::filter_user_columns;
|
||||||
|
use crate::pages::forms::{FieldDefinition, FormState};
|
||||||
use anyhow::{anyhow, Context, Result};
|
use anyhow::{anyhow, Context, Result};
|
||||||
use std::sync::Arc;
|
use std::sync::Arc;
|
||||||
|
|
||||||
|
|||||||
21
client/src/sidebar/logic.rs
Normal file
21
client/src/sidebar/logic.rs
Normal file
@@ -0,0 +1,21 @@
|
|||||||
|
// src/sidebar/state.rs
|
||||||
|
use crossterm::event::{KeyCode, KeyModifiers};
|
||||||
|
use crate::config::binds::config::Config;
|
||||||
|
use crate::state::app::state::UiState;
|
||||||
|
|
||||||
|
pub fn toggle_sidebar(
|
||||||
|
ui_state: &mut UiState,
|
||||||
|
config: &Config,
|
||||||
|
key: KeyCode,
|
||||||
|
modifiers: KeyModifiers,
|
||||||
|
) -> bool {
|
||||||
|
if let Some(action) =
|
||||||
|
config.get_action_for_key_in_mode(&config.keybindings.common, key, modifiers)
|
||||||
|
{
|
||||||
|
if action == "toggle_sidebar" {
|
||||||
|
ui_state.show_sidebar = !ui_state.show_sidebar;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
false
|
||||||
|
}
|
||||||
7
client/src/sidebar/mod.rs
Normal file
7
client/src/sidebar/mod.rs
Normal file
@@ -0,0 +1,7 @@
|
|||||||
|
// src/sidebar/mod.rs
|
||||||
|
|
||||||
|
pub mod ui;
|
||||||
|
pub mod logic;
|
||||||
|
|
||||||
|
pub use ui::{calculate_sidebar_layout, render_sidebar};
|
||||||
|
pub use logic::toggle_sidebar;
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user