mirror of
https://github.com/veeso/termscp.git
synced 2025-12-07 09:36:00 -08:00
923 lines
36 KiB
Rust
923 lines
36 KiB
Rust
//! ## FileTransferActivity
|
|
//!
|
|
//! `filetransfer_activiy` is the module which implements the Filetransfer activity, which is the main activity afterall
|
|
|
|
/*
|
|
*
|
|
* Copyright (C) 2020-2021 Christian Visintin - christian.visintin1997@gmail.com
|
|
*
|
|
* This file is part of "TermSCP"
|
|
*
|
|
* TermSCP is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU General Public License as published by
|
|
* the Free Software Foundation, either version 3 of the License, or
|
|
* (at your option) any later version.
|
|
*
|
|
* TermSCP is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with TermSCP. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
*/
|
|
|
|
// Deps
|
|
extern crate bytesize;
|
|
extern crate hostname;
|
|
#[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))]
|
|
extern crate users;
|
|
// locals
|
|
use super::{Context, FileExplorerTab, FileTransferActivity};
|
|
use crate::fs::explorer::FileSorting;
|
|
use crate::fs::FsEntry;
|
|
use crate::ui::layout::components::{
|
|
msgbox::MsgBox, file_list::FileList, input::Input, logbox::LogBox, progress_bar::ProgressBar,
|
|
radio_group::RadioGroup, table::Table,
|
|
};
|
|
use crate::ui::layout::props::{
|
|
PropValue, PropsBuilder, TableBuilder, TextParts, TextSpan, TextSpanBuilder,
|
|
};
|
|
use crate::ui::layout::utils::draw_area_in;
|
|
use crate::ui::store::Store;
|
|
use crate::utils::fmt::fmt_time;
|
|
// Ext
|
|
use bytesize::ByteSize;
|
|
use std::path::PathBuf;
|
|
use tui::{
|
|
layout::{Constraint, Direction, Layout},
|
|
style::Color,
|
|
widgets::Clear,
|
|
};
|
|
#[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))]
|
|
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
|
|
self.view.mount(
|
|
super::COMPONENT_EXPLORER_LOCAL,
|
|
Box::new(FileList::new(
|
|
PropsBuilder::default()
|
|
.with_background(Color::Yellow)
|
|
.with_foreground(Color::Yellow)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Mount remote file explorer
|
|
self.view.mount(
|
|
super::COMPONENT_EXPLORER_REMOTE,
|
|
Box::new(FileList::new(
|
|
PropsBuilder::default()
|
|
.with_background(Color::LightBlue)
|
|
.with_foreground(Color::LightBlue)
|
|
.build(),
|
|
)),
|
|
);
|
|
// Mount log box
|
|
self.view.mount(
|
|
super::COMPONENT_LOG_BOX,
|
|
Box::new(LogBox::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::LightGreen)
|
|
.bold()
|
|
.build(),
|
|
)),
|
|
);
|
|
// 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]);
|
|
// 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);
|
|
}
|
|
if !store.isset(super::STORAGE_LOGBOX_WIDTH) {
|
|
store.set_unsigned(super::STORAGE_LOGBOX_WIDTH, chunks[1].width as usize);
|
|
}
|
|
// Draw explorers
|
|
self.view
|
|
.render(super::COMPONENT_EXPLORER_LOCAL, f, tabs_chunks[0]);
|
|
self.view
|
|
.render(super::COMPONENT_EXPLORER_REMOTE, f, tabs_chunks[1]);
|
|
// Draw log box
|
|
self.view.render(super::COMPONENT_LOG_BOX, f, chunks[1]);
|
|
// Draw popups
|
|
if let Some(mut props) = self.view.get_props(super::COMPONENT_INPUT_COPY) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_GOTO) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_MKDIR) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_NEWFILE) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_RENAME) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_SAVEAS) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_INPUT_EXEC) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_LIST_FILEINFO) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_PROGRESS_BAR) {
|
|
if props.build().visible {
|
|
let popup = draw_area_in(f.size(), 40, 10);
|
|
f.render_widget(Clear, popup);
|
|
// make popup
|
|
self.view.render(super::COMPONENT_PROGRESS_BAR, f, popup);
|
|
}
|
|
}
|
|
if let Some(mut props) = self.view.get_props(super::COMPONENT_RADIO_DELETE) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_RADIO_DISCONNECT) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_RADIO_QUIT) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_RADIO_SORTING) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_TEXT_ERROR) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_TEXT_FATAL) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_TEXT_WAIT) {
|
|
if props.build().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(mut props) = self.view.get_props(super::COMPONENT_TEXT_HELP) {
|
|
if props.build().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
|
|
self.view.mount(
|
|
super::COMPONENT_TEXT_ERROR,
|
|
Box::new(MsgBox::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::Red)
|
|
.bold()
|
|
.with_texts(TextParts::new(None, Some(vec![TextSpan::from(text)])))
|
|
.build(),
|
|
)),
|
|
);
|
|
// Give focus to error
|
|
self.view.active(super::COMPONENT_TEXT_ERROR);
|
|
}
|
|
|
|
/// ### 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
|
|
self.view.mount(
|
|
super::COMPONENT_TEXT_FATAL,
|
|
Box::new(MsgBox::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::Red)
|
|
.bold()
|
|
.with_texts(TextParts::new(None, Some(vec![TextSpan::from(text)])))
|
|
.build(),
|
|
)),
|
|
);
|
|
// Give focus to error
|
|
self.view.active(super::COMPONENT_TEXT_FATAL);
|
|
}
|
|
|
|
pub(super) fn mount_wait(&mut self, text: &str) {
|
|
// Mount
|
|
self.view.mount(
|
|
super::COMPONENT_TEXT_WAIT,
|
|
Box::new(MsgBox::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::White)
|
|
.bold()
|
|
.with_texts(TextParts::new(None, Some(vec![TextSpan::from(text)])))
|
|
.build(),
|
|
)),
|
|
);
|
|
// Give focus to info
|
|
self.view.active(super::COMPONENT_TEXT_WAIT);
|
|
}
|
|
|
|
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
|
|
self.view.mount(
|
|
super::COMPONENT_RADIO_QUIT,
|
|
Box::new(RadioGroup::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::Yellow)
|
|
.with_background(Color::Black)
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Are you sure you want to quit?")),
|
|
Some(vec![TextSpan::from("Yes"), TextSpan::from("No")]),
|
|
))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_RADIO_QUIT);
|
|
}
|
|
|
|
/// ### 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
|
|
self.view.mount(
|
|
super::COMPONENT_RADIO_DISCONNECT,
|
|
Box::new(RadioGroup::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::Yellow)
|
|
.with_background(Color::Black)
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Are you sure you want to disconnect?")),
|
|
Some(vec![TextSpan::from("Yes"), TextSpan::from("No")]),
|
|
))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_RADIO_DISCONNECT);
|
|
}
|
|
|
|
/// ### 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) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_COPY,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Insert destination name")),
|
|
None,
|
|
))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_COPY);
|
|
}
|
|
|
|
pub(super) fn umount_copy(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_COPY);
|
|
}
|
|
|
|
pub(super) fn mount_exec(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_EXEC,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(Some(String::from("Execute command")), None))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_EXEC);
|
|
}
|
|
|
|
pub(super) fn umount_exec(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_EXEC);
|
|
}
|
|
|
|
pub(super) fn mount_goto(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_GOTO,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Change working directory")),
|
|
None,
|
|
))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_GOTO);
|
|
}
|
|
|
|
pub(super) fn umount_goto(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_GOTO);
|
|
}
|
|
|
|
pub(super) fn mount_mkdir(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_MKDIR,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Insert directory name")),
|
|
None,
|
|
))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_MKDIR);
|
|
}
|
|
|
|
pub(super) fn umount_mkdir(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_MKDIR);
|
|
}
|
|
|
|
pub(super) fn mount_newfile(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_NEWFILE,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(Some(String::from("New file name")), None))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_NEWFILE);
|
|
}
|
|
|
|
pub(super) fn umount_newfile(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_NEWFILE);
|
|
}
|
|
|
|
pub(super) fn mount_rename(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_RENAME,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(Some(String::from("Insert new name")), None))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_RENAME);
|
|
}
|
|
|
|
pub(super) fn umount_rename(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_RENAME);
|
|
}
|
|
|
|
pub(super) fn mount_saveas(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_INPUT_SAVEAS,
|
|
Box::new(Input::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::new(Some(String::from("Save as...")), None))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_INPUT_SAVEAS);
|
|
}
|
|
|
|
pub(super) fn umount_saveas(&mut self) {
|
|
self.view.umount(super::COMPONENT_INPUT_SAVEAS);
|
|
}
|
|
|
|
pub(super) fn mount_progress_bar(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_PROGRESS_BAR,
|
|
Box::new(ProgressBar::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::LightGreen)
|
|
.with_background(Color::Black)
|
|
.with_texts(TextParts::new(Some(String::from("Please wait")), None))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_PROGRESS_BAR);
|
|
}
|
|
|
|
pub(super) fn umount_progress_bar(&mut self) {
|
|
self.view.umount(super::COMPONENT_PROGRESS_BAR);
|
|
}
|
|
|
|
pub(super) fn mount_file_sorting(&mut self) {
|
|
let sorting: FileSorting = match self.tab {
|
|
FileExplorerTab::Local => self.local.get_file_sorting(),
|
|
FileExplorerTab::Remote => self.remote.get_file_sorting(),
|
|
};
|
|
let index: usize = match sorting {
|
|
FileSorting::ByCreationTime => 2,
|
|
FileSorting::ByModifyTime => 1,
|
|
FileSorting::ByName => 0,
|
|
FileSorting::BySize => 3,
|
|
};
|
|
self.view.mount(
|
|
super::COMPONENT_RADIO_SORTING,
|
|
Box::new(RadioGroup::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::LightMagenta)
|
|
.with_background(Color::Black)
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Sort files by")),
|
|
Some(vec![
|
|
TextSpan::from("Name"),
|
|
TextSpan::from("Modify time"),
|
|
TextSpan::from("Creation time"),
|
|
TextSpan::from("Size"),
|
|
]),
|
|
))
|
|
.with_value(PropValue::Unsigned(index))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_RADIO_SORTING);
|
|
}
|
|
|
|
pub(super) fn umount_file_sorting(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_SORTING);
|
|
}
|
|
|
|
pub(super) fn mount_radio_delete(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_RADIO_DELETE,
|
|
Box::new(RadioGroup::new(
|
|
PropsBuilder::default()
|
|
.with_foreground(Color::Red)
|
|
.with_background(Color::Black)
|
|
.with_texts(TextParts::new(
|
|
Some(String::from("Delete file")),
|
|
Some(vec![TextSpan::from("Yes"), TextSpan::from("No")]),
|
|
))
|
|
.with_value(PropValue::Unsigned(1))
|
|
.build(),
|
|
)),
|
|
);
|
|
self.view.active(super::COMPONENT_RADIO_DELETE);
|
|
}
|
|
|
|
pub(super) fn umount_radio_delete(&mut self) {
|
|
self.view.umount(super::COMPONENT_RADIO_DELETE);
|
|
}
|
|
|
|
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(
|
|
TextSpanBuilder::new(path.as_str())
|
|
.with_foreground(Color::Yellow)
|
|
.build(),
|
|
);
|
|
if let Some(filetype) = file.get_ftype() {
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("File type: "))
|
|
.add_col(
|
|
TextSpanBuilder::new(filetype.as_str())
|
|
.with_foreground(Color::LightGreen)
|
|
.build(),
|
|
);
|
|
}
|
|
let (bsize, size): (ByteSize, usize) = (ByteSize(file.get_size() as u64), file.get_size());
|
|
texts.add_row().add_col(TextSpan::from("Size: ")).add_col(
|
|
TextSpanBuilder::new(format!("{} ({})", bsize, size).as_str())
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
);
|
|
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(
|
|
TextSpanBuilder::new(ctime.as_str())
|
|
.with_foreground(Color::LightGreen)
|
|
.build(),
|
|
);
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Last modified time: "))
|
|
.add_col(
|
|
TextSpanBuilder::new(mtime.as_str())
|
|
.with_foreground(Color::LightBlue)
|
|
.build(),
|
|
);
|
|
texts
|
|
.add_row()
|
|
.add_col(TextSpan::from("Last access time: "))
|
|
.add_col(
|
|
TextSpanBuilder::new(atime.as_str())
|
|
.with_foreground(Color::LightRed)
|
|
.build(),
|
|
);
|
|
// User
|
|
#[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))]
|
|
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(any(target_os = "unix", target_os = "macos", target_os = "linux"))]
|
|
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(
|
|
TextSpanBuilder::new(username.as_str())
|
|
.with_foreground(Color::LightYellow)
|
|
.build(),
|
|
);
|
|
texts.add_row().add_col(TextSpan::from("Group: ")).add_col(
|
|
TextSpanBuilder::new(group.as_str())
|
|
.with_foreground(Color::Blue)
|
|
.build(),
|
|
);
|
|
self.view.mount(
|
|
super::COMPONENT_LIST_FILEINFO,
|
|
Box::new(Table::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::table(
|
|
Some(file.get_name().to_string()),
|
|
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);
|
|
}
|
|
|
|
/// ### mount_help
|
|
///
|
|
/// Mount help
|
|
pub(super) fn mount_help(&mut self) {
|
|
self.view.mount(
|
|
super::COMPONENT_TEXT_HELP,
|
|
Box::new(Table::new(
|
|
PropsBuilder::default()
|
|
.with_texts(TextParts::table(
|
|
Some(String::from("Help")),
|
|
TableBuilder::default()
|
|
.add_col(
|
|
TextSpanBuilder::new("<ESC>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Disconnect"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<TAB>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(
|
|
" Switch between explorer and logs",
|
|
))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<BACKSPACE>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Go to previous directory"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<RIGHT/LEFT>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Change explorer tab"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<UP/DOWN>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Move up/down in list"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<ENTER>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Enter directory"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<SPACE>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Upload/Download file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<A>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Toggle hidden files"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<B>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Change file sorting mode"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<C>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Copy"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<D>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Make directory"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<G>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Go to path"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<H>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Show help"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<I>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Show info about selected file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<L>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Reload directory content"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<N>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Create new file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<O>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Open text file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<Q>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Quit termscp"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<R>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Rename file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<S>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Save file as"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<U>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Go to parent directory"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<DEL|E>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.add_col(TextSpan::from(" Delete selected file"))
|
|
.add_row()
|
|
.add_col(
|
|
TextSpanBuilder::new("<CTRL+C>")
|
|
.bold()
|
|
.with_foreground(Color::Cyan)
|
|
.build(),
|
|
)
|
|
.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);
|
|
}
|
|
}
|