1085 lines
43 KiB
Rust
1085 lines
43 KiB
Rust
//! ## FileTransferActivity
|
|
//!
|
|
//! `filetransfer_activiy` is the module which implements the Filetransfer activity, which is the main activity afterall
|
|
|
|
/**
|
|
* MIT License
|
|
*
|
|
* termscp - Copyright (c) 2021 Christian Visintin
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
* of this software and associated documentation files (the "Software"), to deal
|
|
* in the Software without restriction, including without limitation the rights
|
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
* copies of the Software, and to permit persons to whom the Software is
|
|
* furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
* copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
* SOFTWARE.
|
|
*/
|
|
// locals
|
|
use super::{browser::FileExplorerTab, Context, FileTransferActivity};
|
|
use crate::fs::explorer::FileSorting;
|
|
use crate::fs::FsEntry;
|
|
use crate::ui::components::{
|
|
file_list::{FileList, FileListPropsBuilder},
|
|
logbox::{LogBox, LogboxPropsBuilder},
|
|
};
|
|
use crate::ui::store::Store;
|
|
use crate::utils::fmt::fmt_time;
|
|
use crate::utils::ui::draw_area_in;
|
|
// Ext
|
|
use bytesize::ByteSize;
|
|
use std::path::PathBuf;
|
|
use tui_realm_stdlib::{
|
|
Input, InputPropsBuilder, List, ListPropsBuilder, Paragraph, ParagraphPropsBuilder,
|
|
ProgressBar, ProgressBarPropsBuilder, Radio, RadioPropsBuilder, Span, SpanPropsBuilder, Table,
|
|
TablePropsBuilder,
|
|
};
|
|
use tuirealm::props::{Alignment, PropsBuilder, TableBuilder, TextSpan};
|
|
use tuirealm::tui::{
|
|
layout::{Constraint, Direction, Layout},
|
|
style::Color,
|
|
widgets::{BorderType, Borders, Clear},
|
|
};
|
|
#[cfg(target_family = "unix")]
|
|
use users::{get_group_by_gid, get_user_by_uid};
|
|
|
|
impl FileTransferActivity {
|
|
// -- init
|
|
|
|
/// ### init
|
|
///
|
|
/// Initialize file transfer activity's view
|
|
pub(super) fn init(&mut self) {
|
|
// Mount local file explorer
|
|
let local_explorer_background = self.theme().transfer_local_explorer_background;
|
|
let local_explorer_foreground = self.theme().transfer_local_explorer_foreground;
|
|
let local_explorer_highlighted = self.theme().transfer_local_explorer_highlighted;
|
|
let remote_explorer_background = self.theme().transfer_remote_explorer_background;
|
|
let remote_explorer_foreground = self.theme().transfer_remote_explorer_foreground;
|
|
let remote_explorer_highlighted = self.theme().transfer_remote_explorer_highlighted;
|
|
let log_panel = self.theme().transfer_log_window;
|
|
let log_background = self.theme().transfer_log_background;
|
|
self.view.mount(
|
|
super::COMPONENT_EXPLORER_LOCAL,
|
|
Box::new(FileList::new(
|
|
FileListPropsBuilder::default()
|
|
.with_highlight_color(local_explorer_highlighted)
|
|
.with_background(local_explorer_background)
|
|
.with_foreground(local_explorer_foreground)
|
|
.with_borders(Borders::ALL, BorderType::Plain, local_explorer_highlighted)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Mount remote file explorer
|
|
self.view.mount(
|
|
super::COMPONENT_EXPLORER_REMOTE,
|
|
Box::new(FileList::new(
|
|
FileListPropsBuilder::default()
|
|
.with_highlight_color(remote_explorer_highlighted)
|
|
.with_background(remote_explorer_background)
|
|
.with_foreground(remote_explorer_foreground)
|
|
.with_borders(Borders::ALL, BorderType::Plain, remote_explorer_highlighted)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Mount log box
|
|
self.view.mount(
|
|
super::COMPONENT_LOG_BOX,
|
|
Box::new(LogBox::new(
|
|
LogboxPropsBuilder::default()
|
|
.with_title("Log", Alignment::Left)
|
|
.with_background(log_background)
|
|
.with_borders(Borders::ALL, BorderType::Plain, log_panel)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Mount status bars
|
|
self.view.mount(
|
|
super::COMPONENT_SPAN_STATUS_BAR_LOCAL,
|
|
Box::new(Span::new(SpanPropsBuilder::default().build())),
|
|
);
|
|
self.view.mount(
|
|
super::COMPONENT_SPAN_STATUS_BAR_REMOTE,
|
|
Box::new(Span::new(SpanPropsBuilder::default().build())),
|
|
);
|
|
// Load process bar
|
|
self.refresh_local_status_bar();
|
|
self.refresh_remote_status_bar();
|
|
// Update components
|
|
let _ = self.update_local_filelist();
|
|
let _ = self.update_remote_filelist();
|
|
// Give focus to local explorer
|
|
self.view.active(super::COMPONENT_EXPLORER_LOCAL);
|
|
}
|
|
|
|
// -- view
|
|
|
|
/// ### view
|
|
///
|
|
/// View gui
|
|
pub(super) fn view(&mut self) {
|
|
let mut context: Context = self.context.take().unwrap();
|
|
let store: &mut Store = &mut context.store;
|
|
let _ = context.terminal.draw(|f| {
|
|
// Prepare chunks
|
|
let chunks = Layout::default()
|
|
.direction(Direction::Vertical)
|
|
.margin(1)
|
|
.constraints(
|
|
[
|
|
Constraint::Percentage(70), // Explorer
|
|
Constraint::Percentage(30), // Log
|
|
]
|
|
.as_ref(),
|
|
)
|
|
.split(f.size());
|
|
// Create explorer chunks
|
|
let tabs_chunks = Layout::default()
|
|
.constraints([Constraint::Percentage(50), Constraint::Percentage(50)].as_ref())
|
|
.direction(Direction::Horizontal)
|
|
.split(chunks[0]);
|
|
// Create log box chunks
|
|
let bottom_chunks = Layout::default()
|
|
.constraints([Constraint::Length(1), Constraint::Length(10)].as_ref())
|
|
.direction(Direction::Vertical)
|
|
.split(chunks[1]);
|
|
// Create status bar chunks
|
|
let status_bar_chunks = Layout::default()
|
|
.constraints([Constraint::Percentage(50), Constraint::Percentage(50)].as_ref())
|
|
.direction(Direction::Horizontal)
|
|
.horizontal_margin(1)
|
|
.split(bottom_chunks[0]);
|
|
// If width is unset in the storage, set width
|
|
if !store.isset(super::STORAGE_EXPLORER_WIDTH) {
|
|
store.set_unsigned(super::STORAGE_EXPLORER_WIDTH, tabs_chunks[0].width as usize);
|
|
}
|
|
// Draw explorers
|
|
// @! Local explorer (Find or default)
|
|
match self.browser.tab() {
|
|
FileExplorerTab::FindLocal => {
|
|
self.view
|
|
.render(super::COMPONENT_EXPLORER_FIND, f, tabs_chunks[0])
|
|
}
|
|
_ => self
|
|
.view
|
|
.render(super::COMPONENT_EXPLORER_LOCAL, f, tabs_chunks[0]),
|
|
}
|
|
// @! Remote explorer (Find or default)
|
|
match self.browser.tab() {
|
|
FileExplorerTab::FindRemote => {
|
|
self.view
|
|
.render(super::COMPONENT_EXPLORER_FIND, f, tabs_chunks[1])
|
|
}
|
|
_ => self
|
|
.view
|
|
.render(super::COMPONENT_EXPLORER_REMOTE, f, tabs_chunks[1]),
|
|
}
|
|
// Draw log box
|
|
self.view
|
|
.render(super::COMPONENT_LOG_BOX, f, bottom_chunks[1]);
|
|
// Draw status bar
|
|
self.view.render(
|
|
super::COMPONENT_SPAN_STATUS_BAR_LOCAL,
|
|
f,
|
|
status_bar_chunks[0],
|
|
);
|
|
self.view.render(
|
|
super::COMPONENT_SPAN_STATUS_BAR_REMOTE,
|
|
f,
|
|
status_bar_chunks[1],
|
|
);
|
|
// @! Draw popups
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_COPY) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_COPY, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_FIND) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_FIND, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_GOTO) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_GOTO, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_MKDIR) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_MKDIR, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_NEWFILE) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_NEWFILE, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_OPEN_WITH) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_OPEN_WITH, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_RENAME) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_RENAME, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_SAVEAS) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_SAVEAS, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_INPUT_EXEC) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_INPUT_EXEC, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_LIST_FILEINFO) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 50);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_LIST_FILEINFO, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_PROGRESS_BAR_PARTIAL) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 20);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
let popup_chunks = Layout::default()
|
|
.direction(Direction::Vertical)
|
|
.constraints(
|
|
[
|
|
Constraint::Percentage(50), // Full
|
|
Constraint::Percentage(50), // Partial
|
|
]
|
|
.as_ref(),
|
|
)
|
|
.split(popup);
|
|
self.view
|
|
.render(super::COMPONENT_PROGRESS_BAR_FULL, f, popup_chunks[0]);
|
|
self.view
|
|
.render(super::COMPONENT_PROGRESS_BAR_PARTIAL, f, popup_chunks[1]);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_RADIO_DELETE) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 30, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_RADIO_DELETE, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_RADIO_REPLACE) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_RADIO_REPLACE, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_RADIO_DISCONNECT) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 30, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view
|
|
.render(super::COMPONENT_RADIO_DISCONNECT, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_RADIO_QUIT) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 30, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_RADIO_QUIT, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_RADIO_SORTING) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_RADIO_SORTING, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_TEXT_ERROR) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_TEXT_ERROR, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_TEXT_FATAL) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_TEXT_FATAL, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_TEXT_WAIT) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_TEXT_WAIT, f, popup);
|
|
}
|
|
}
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_TEXT_HELP) {
|
|
if props.visible {
|
|
let popup = draw_area_in(f.size(), 50, 80);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_TEXT_HELP, f, popup);
|
|
}
|
|
}
|
|
});
|
|
// Re-give context
|
|
self.context = Some(context);
|
|
}
|
|
|
|
// -- partials
|
|
|
|
/// ### mount_error
|
|
///
|
|
/// Mount error box
|
|
pub(super) fn mount_error(&mut self, text: &str) {
|
|
// Mount
|
|
let error_color = self.theme().misc_error_dialog;
|
|
self.mount_text_dialog(super::COMPONENT_TEXT_ERROR, text, error_color);
|
|
}
|
|
|
|
/// ### umount_error
|
|
///
|
|
/// Umount error message
|
|
pub(super) fn umount_error(&mut self) {
|
|
self.view.umount(super::COMPONENT_TEXT_ERROR);
|
|
}
|
|
|
|
pub(super) fn mount_fatal(&mut self, text: &str) {
|
|
// Mount
|
|
let error_color = self.theme().misc_error_dialog;
|
|
self.mount_text_dialog(super::COMPONENT_TEXT_FATAL, text, error_color);
|
|
}
|
|
|
|
pub(super) fn mount_wait(&mut self, text: &str) {
|
|
self.mount_wait_ex(text);
|
|
}
|
|
|
|
pub(super) fn mount_blocking_wait(&mut self, text: &str) {
|
|
self.mount_wait_ex(text);
|
|
self.view();
|
|
}
|
|
|
|
fn mount_wait_ex(&mut self, text: &str) {
|
|
let color = self.theme().misc_info_dialog;
|
|
self.mount_text_dialog(super::COMPONENT_TEXT_WAIT, text, color);
|
|
}
|
|
|
|
pub(super) fn umount_wait(&mut self) {
|
|
self.view.umount(super::COMPONENT_TEXT_WAIT);
|
|
}
|
|
|
|
/// ### mount_quit
|
|
///
|
|
/// Mount quit popup
|
|
pub(super) fn mount_quit(&mut self) {
|
|
// Protocol
|
|
let quit_color = self.theme().misc_quit_dialog;
|
|
self.mount_radio_dialog(
|
|
super::COMPONENT_RADIO_QUIT,
|
|
"Are you sure you want to quit?",
|
|
&["Yes", "No"],
|
|
0,
|
|
quit_color,
|
|
);
|
|
}
|
|
|
|
/// ### umount_quit
|
|
///
|
|
/// Umount quit popup
|
|
pub(super) fn umount_quit(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_QUIT);
|
|
}
|
|
|
|
/// ### mount_disconnect
|
|
///
|
|
/// Mount disconnect popup
|
|
pub(super) fn mount_disconnect(&mut self) {
|
|
// Protocol
|
|
let quit_color = self.theme().misc_quit_dialog;
|
|
self.mount_radio_dialog(
|
|
super::COMPONENT_RADIO_DISCONNECT,
|
|
"Are you sure you want to disconnect?",
|
|
&["Yes", "No"],
|
|
0,
|
|
quit_color,
|
|
);
|
|
}
|
|
|
|
/// ### umount_disconnect
|
|
///
|
|
/// Umount disconnect popup
|
|
pub(super) fn umount_disconnect(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_DISCONNECT);
|
|
}
|
|
|
|
pub(super) fn mount_copy(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_COPY,
|
|
"Copy file(s) to…",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_copy(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_COPY);
|
|
}
|
|
|
|
pub(super) fn mount_exec(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_EXEC,
|
|
"Execute command",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_exec(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_EXEC);
|
|
}
|
|
|
|
pub(super) fn mount_find(&mut self, search: &str) {
|
|
// Get color
|
|
let (bg, fg, hg): (Color, Color, Color) = match self.browser.tab() {
|
|
FileExplorerTab::Local | FileExplorerTab::FindLocal => (
|
|
self.theme().transfer_local_explorer_background,
|
|
self.theme().transfer_local_explorer_foreground,
|
|
self.theme().transfer_local_explorer_highlighted,
|
|
),
|
|
FileExplorerTab::Remote | FileExplorerTab::FindRemote => (
|
|
self.theme().transfer_remote_explorer_background,
|
|
self.theme().transfer_remote_explorer_foreground,
|
|
self.theme().transfer_remote_explorer_highlighted,
|
|
),
|
|
};
|
|
// Mount component
|
|
self.view.mount(
|
|
super::COMPONENT_EXPLORER_FIND,
|
|
Box::new(FileList::new(
|
|
FileListPropsBuilder::default()
|
|
.with_title(
|
|
format!("Search results for \"{}\"", search),
|
|
Alignment::Left,
|
|
)
|
|
.with_borders(Borders::ALL, BorderType::Plain, hg)
|
|
.with_highlight_color(hg)
|
|
.with_background(bg)
|
|
.with_foreground(fg)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Give focus to explorer findd
|
|
self.view.active(super::COMPONENT_EXPLORER_FIND);
|
|
}
|
|
|
|
pub(super) fn umount_find(&mut self) {
|
|
self.view.umount(super::COMPONENT_EXPLORER_FIND);
|
|
}
|
|
|
|
pub(super) fn mount_find_input(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_FIND,
|
|
"Search files by name",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_find_input(&mut self) {
|
|
// Umount input find
|
|
self.view.umount(super::COMPONENT_INPUT_FIND);
|
|
}
|
|
|
|
pub(super) fn mount_goto(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_GOTO,
|
|
"Change working directory",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_goto(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_GOTO);
|
|
}
|
|
|
|
pub(super) fn mount_mkdir(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_MKDIR,
|
|
"Insert directory name",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_mkdir(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_MKDIR);
|
|
}
|
|
|
|
pub(super) fn mount_newfile(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_NEWFILE,
|
|
"New file name",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_newfile(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_NEWFILE);
|
|
}
|
|
|
|
pub(super) fn mount_openwith(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_OPEN_WITH,
|
|
"Open file with…",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_openwith(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_OPEN_WITH);
|
|
}
|
|
|
|
pub(super) fn mount_rename(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(
|
|
super::COMPONENT_INPUT_RENAME,
|
|
"Move file(s) to…",
|
|
"",
|
|
input_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_rename(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_RENAME);
|
|
}
|
|
|
|
pub(super) fn mount_saveas(&mut self) {
|
|
let input_color = self.theme().misc_input_dialog;
|
|
self.mount_input_dialog(super::COMPONENT_INPUT_SAVEAS, "Save as…", "", input_color);
|
|
}
|
|
|
|
pub(super) fn umount_saveas(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_SAVEAS);
|
|
}
|
|
|
|
pub(super) fn mount_progress_bar(&mut self, root_name: String) {
|
|
let prog_color_full = self.theme().transfer_progress_bar_full;
|
|
let prog_color_partial = self.theme().transfer_progress_bar_partial;
|
|
self.view.mount(
|
|
super::COMPONENT_PROGRESS_BAR_FULL,
|
|
Box::new(ProgressBar::new(
|
|
ProgressBarPropsBuilder::default()
|
|
.with_progbar_color(prog_color_full)
|
|
.with_background(Color::Black)
|
|
.with_borders(
|
|
Borders::TOP | Borders::RIGHT | Borders::LEFT,
|
|
BorderType::Rounded,
|
|
Color::Reset,
|
|
)
|
|
.with_title(root_name, Alignment::Center)
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.mount(
|
|
super::COMPONENT_PROGRESS_BAR_PARTIAL,
|
|
Box::new(ProgressBar::new(
|
|
ProgressBarPropsBuilder::default()
|
|
.with_progbar_color(prog_color_partial)
|
|
.with_background(Color::Black)
|
|
.with_borders(
|
|
Borders::BOTTOM | Borders::RIGHT | Borders::LEFT,
|
|
BorderType::Rounded,
|
|
Color::Reset,
|
|
)
|
|
.with_title("Please wait", Alignment::Center)
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_PROGRESS_BAR_PARTIAL);
|
|
}
|
|
|
|
pub(super) fn umount_progress_bar(&mut self) {
|
|
self.view.umount(super::COMPONENT_PROGRESS_BAR_PARTIAL);
|
|
self.view.umount(super::COMPONENT_PROGRESS_BAR_FULL);
|
|
}
|
|
|
|
pub(super) fn mount_file_sorting(&mut self) {
|
|
let sorting_color = self.theme().transfer_status_sorting;
|
|
let sorting: FileSorting = match self.browser.tab() {
|
|
FileExplorerTab::Local => self.local().get_file_sorting(),
|
|
FileExplorerTab::Remote => self.remote().get_file_sorting(),
|
|
_ => panic!("You can't mount file sorting when in found result"),
|
|
};
|
|
let index: usize = match sorting {
|
|
FileSorting::CreationTime => 2,
|
|
FileSorting::ModifyTime => 1,
|
|
FileSorting::Name => 0,
|
|
FileSorting::Size => 3,
|
|
};
|
|
self.mount_radio_dialog(
|
|
super::COMPONENT_RADIO_SORTING,
|
|
"Sort files by",
|
|
&["Name", "Modify time", "Creation time", "Size"],
|
|
index,
|
|
sorting_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_file_sorting(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_SORTING);
|
|
}
|
|
|
|
pub(super) fn mount_radio_delete(&mut self) {
|
|
let warn_color = self.theme().misc_warn_dialog;
|
|
self.mount_radio_dialog(
|
|
super::COMPONENT_RADIO_DELETE,
|
|
"Delete file",
|
|
&["Yes", "No"],
|
|
1,
|
|
warn_color,
|
|
);
|
|
}
|
|
|
|
pub(super) fn umount_radio_delete(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_DELETE);
|
|
}
|
|
|
|
pub(super) fn mount_radio_replace(&mut self, file_name: &str) {
|
|
let warn_color = self.theme().misc_warn_dialog;
|
|
self.view.mount(
|
|
super::COMPONENT_RADIO_REPLACE,
|
|
Box::new(Radio::new(
|
|
RadioPropsBuilder::default()
|
|
.with_color(warn_color)
|
|
.with_inverted_color(Color::Black)
|
|
.with_borders(Borders::ALL, BorderType::Plain, warn_color)
|
|
.with_title(
|
|
format!("File '{}' already exists. Overwrite file?", file_name),
|
|
Alignment::Center,
|
|
)
|
|
.with_options(&[String::from("Yes"), String::from("No")])
|
|
.with_value(0)
|
|
.rewind(true)
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_RADIO_REPLACE);
|
|
}
|
|
|
|
pub(super) fn umount_radio_replace(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_REPLACE);
|
|
}
|
|
|
|
pub(super) fn mount_file_info(&mut self, file: &FsEntry) {
|
|
let mut texts: TableBuilder = TableBuilder::default();
|
|
// Abs path
|
|
let real_path: Option<PathBuf> = {
|
|
let real_file: FsEntry = file.get_realfile();
|
|
match real_file.get_abs_path() != file.get_abs_path() {
|
|
true => Some(real_file.get_abs_path()),
|
|
false => None,
|
|
}
|
|
};
|
|
let path: String = match real_path {
|
|
Some(symlink) => format!("{} -> {}", file.get_abs_path().display(), symlink.display()),
|
|
None => format!("{}", file.get_abs_path().display()),
|
|
};
|
|
// Make texts
|
|
texts
|
|
.add_col(TextSpan::from("Path: "))
|
|
.add_col(TextSpan::new(path.as_str()).fg(Color::Yellow));
|
|
if let Some(filetype) = file.get_ftype() {
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("File type: "))
|
|
.add_col(TextSpan::new(filetype.as_str()).fg(Color::LightGreen));
|
|
}
|
|
let (bsize, size): (ByteSize, usize) = (ByteSize(file.get_size() as u64), file.get_size());
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Size: "))
|
|
.add_col(TextSpan::new(format!("{} ({})", bsize, size).as_str()).fg(Color::Cyan));
|
|
let ctime: String = fmt_time(file.get_creation_time(), "%b %d %Y %H:%M:%S");
|
|
let atime: String = fmt_time(file.get_last_access_time(), "%b %d %Y %H:%M:%S");
|
|
let mtime: String = fmt_time(file.get_creation_time(), "%b %d %Y %H:%M:%S");
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Creation time: "))
|
|
.add_col(TextSpan::new(ctime.as_str()).fg(Color::LightGreen));
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Last modified time: "))
|
|
.add_col(TextSpan::new(mtime.as_str()).fg(Color::LightBlue));
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Last access time: "))
|
|
.add_col(TextSpan::new(atime.as_str()).fg(Color::LightRed));
|
|
// User
|
|
#[cfg(target_family = "unix")]
|
|
let username: String = match file.get_user() {
|
|
Some(uid) => match get_user_by_uid(uid) {
|
|
Some(user) => user.name().to_string_lossy().to_string(),
|
|
None => uid.to_string(),
|
|
},
|
|
None => String::from("0"),
|
|
};
|
|
#[cfg(target_os = "windows")]
|
|
let username: String = format!("{}", file.get_user().unwrap_or(0));
|
|
// Group
|
|
#[cfg(target_family = "unix")]
|
|
let group: String = match file.get_group() {
|
|
Some(gid) => match get_group_by_gid(gid) {
|
|
Some(group) => group.name().to_string_lossy().to_string(),
|
|
None => gid.to_string(),
|
|
},
|
|
None => String::from("0"),
|
|
};
|
|
#[cfg(target_os = "windows")]
|
|
let group: String = format!("{}", file.get_group().unwrap_or(0));
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("User: "))
|
|
.add_col(TextSpan::new(username.as_str()).fg(Color::LightYellow));
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Group: "))
|
|
.add_col(TextSpan::new(group.as_str()).fg(Color::Blue));
|
|
self.view.mount(
|
|
super::COMPONENT_LIST_FILEINFO,
|
|
Box::new(Table::new(
|
|
TablePropsBuilder::default()
|
|
.with_borders(Borders::ALL, BorderType::Rounded, Color::White)
|
|
.with_title(file.get_name(), Alignment::Left)
|
|
.with_table(texts.build())
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_LIST_FILEINFO);
|
|
}
|
|
|
|
pub(super) fn umount_file_info(&mut self) {
|
|
self.view.umount(super::COMPONENT_LIST_FILEINFO);
|
|
}
|
|
|
|
pub(super) fn refresh_local_status_bar(&mut self) {
|
|
let sorting_color = self.theme().transfer_status_sorting;
|
|
let hidden_color = self.theme().transfer_status_hidden;
|
|
let local_bar_spans: Vec<TextSpan> = vec![
|
|
TextSpan::new("File sorting: ").fg(sorting_color),
|
|
TextSpan::new(Self::get_file_sorting_str(self.local().get_file_sorting()))
|
|
.fg(sorting_color)
|
|
.reversed(),
|
|
TextSpan::new(" Hidden files: ").fg(hidden_color),
|
|
TextSpan::new(Self::get_hidden_files_str(
|
|
self.local().hidden_files_visible(),
|
|
))
|
|
.fg(hidden_color)
|
|
.reversed(),
|
|
];
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_SPAN_STATUS_BAR_LOCAL) {
|
|
self.view.update(
|
|
super::COMPONENT_SPAN_STATUS_BAR_LOCAL,
|
|
SpanPropsBuilder::from(props)
|
|
.with_spans(local_bar_spans)
|
|
.build(),
|
|
);
|
|
}
|
|
}
|
|
|
|
pub(super) fn refresh_remote_status_bar(&mut self) {
|
|
let sorting_color = self.theme().transfer_status_sorting;
|
|
let hidden_color = self.theme().transfer_status_hidden;
|
|
let sync_color = self.theme().transfer_status_sync_browsing;
|
|
let remote_bar_spans: Vec<TextSpan> = vec![
|
|
TextSpan::new("File sorting: ").fg(sorting_color),
|
|
TextSpan::new(Self::get_file_sorting_str(self.remote().get_file_sorting()))
|
|
.fg(sorting_color)
|
|
.reversed(),
|
|
TextSpan::new(" Hidden files: ").fg(hidden_color),
|
|
TextSpan::new(Self::get_hidden_files_str(
|
|
self.remote().hidden_files_visible(),
|
|
))
|
|
.fg(hidden_color)
|
|
.reversed(),
|
|
TextSpan::new(" Sync Browsing: ").fg(sync_color),
|
|
TextSpan::new(match self.browser.sync_browsing {
|
|
true => "ON ",
|
|
false => "OFF",
|
|
})
|
|
.fg(sync_color)
|
|
.reversed(),
|
|
];
|
|
if let Some(props) = self.view.get_props(super::COMPONENT_SPAN_STATUS_BAR_REMOTE) {
|
|
self.view.update(
|
|
super::COMPONENT_SPAN_STATUS_BAR_REMOTE,
|
|
SpanPropsBuilder::from(props)
|
|
.with_spans(remote_bar_spans)
|
|
.build(),
|
|
);
|
|
}
|
|
}
|
|
|
|
/// ### mount_help
|
|
///
|
|
/// Mount help
|
|
pub(super) fn mount_help(&mut self) {
|
|
let key_color = self.theme().misc_keys;
|
|
self.view.mount(
|
|
super::COMPONENT_TEXT_HELP,
|
|
Box::new(List::new(
|
|
ListPropsBuilder::default()
|
|
.with_borders(Borders::ALL, BorderType::Rounded, Color::White)
|
|
.with_highlighted_str(Some("?"))
|
|
.with_max_scroll_step(8)
|
|
.bold()
|
|
.scrollable(true)
|
|
.with_title("Help", Alignment::Center)
|
|
.with_rows(
|
|
TableBuilder::default()
|
|
.add_col(TextSpan::new("<ESC>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Disconnect"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<TAB>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(
|
|
" Switch between explorer and logs",
|
|
))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<BACKSPACE>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Go to previous directory"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<RIGHT/LEFT>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Change explorer tab"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<UP/DOWN>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Move up/down in list"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<ENTER>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Enter directory"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<SPACE>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Upload/Download file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<A>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Toggle hidden files"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<B>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Change file sorting mode"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<C>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Copy"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<D>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Make directory"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<G>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Go to path"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<H>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Show help"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<I>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Show info about selected file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<L>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Reload directory content"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<M>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Select file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<N>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Create new file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<O>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(
|
|
" Open text file with preferred editor",
|
|
))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<Q>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Quit termscp"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<R>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Rename file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<S>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Save file as"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<U>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Go to parent directory"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<V>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(
|
|
" Open file with default application for file type",
|
|
))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<W>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(
|
|
" Open file with specified application",
|
|
))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<X>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Execute shell command"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<Y>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Toggle synchronized browsing"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<DEL|E>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Delete selected file"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<CTRL+A>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Select all files"))
|
|
.add_row()
|
|
.add_col(TextSpan::new("<CTRL+C>").bold().fg(key_color))
|
|
.add_col(TextSpan::from(" Interrupt file transfer"))
|
|
.build(),
|
|
)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Active help
|
|
self.view.active(super::COMPONENT_TEXT_HELP);
|
|
}
|
|
|
|
pub(super) fn umount_help(&mut self) {
|
|
self.view.umount(super::COMPONENT_TEXT_HELP);
|
|
}
|
|
|
|
fn get_file_sorting_str(mode: FileSorting) -> &'static str {
|
|
match mode {
|
|
FileSorting::Name => "By name",
|
|
FileSorting::CreationTime => "By creation time",
|
|
FileSorting::ModifyTime => "By modify time",
|
|
FileSorting::Size => "By size",
|
|
}
|
|
}
|
|
|
|
fn get_hidden_files_str(show: bool) -> &'static str {
|
|
match show {
|
|
true => "Show",
|
|
false => "Hide",
|
|
}
|
|
}
|
|
|
|
// -- Mount helpers
|
|
|
|
fn mount_text_dialog(&mut self, id: &str, text: &str, color: Color) {
|
|
// Mount
|
|
self.view.mount(
|
|
id,
|
|
Box::new(Paragraph::new(
|
|
ParagraphPropsBuilder::default()
|
|
.with_borders(Borders::ALL, BorderType::Thick, color)
|
|
.with_foreground(color)
|
|
.bold()
|
|
.with_text_alignment(Alignment::Center)
|
|
.with_texts(vec![TextSpan::from(text)])
|
|
.build(),
|
|
)),
|
|
);
|
|
// Give focus to error
|
|
self.view.active(id);
|
|
}
|
|
|
|
fn mount_input_dialog(&mut self, id: &str, text: &str, val: &str, color: Color) {
|
|
self.view.mount(
|
|
id,
|
|
Box::new(Input::new(
|
|
InputPropsBuilder::default()
|
|
.with_foreground(color)
|
|
.with_label(text, Alignment::Center)
|
|
.with_borders(Borders::ALL, BorderType::Rounded, color)
|
|
.with_value(val.to_string())
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(id);
|
|
}
|
|
|
|
fn mount_radio_dialog(
|
|
&mut self,
|
|
id: &str,
|
|
text: &str,
|
|
opts: &[&str],
|
|
default: usize,
|
|
color: Color,
|
|
) {
|
|
self.view.mount(
|
|
id,
|
|
Box::new(Radio::new(
|
|
RadioPropsBuilder::default()
|
|
.with_color(color)
|
|
.with_inverted_color(Color::Black)
|
|
.with_borders(Borders::ALL, BorderType::Rounded, color)
|
|
.with_title(text, Alignment::Center)
|
|
.with_options(opts)
|
|
.with_value(default)
|
|
.rewind(true)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Active
|
|
self.view.active(id);
|
|
}
|
|
}
|