//! ## 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 . * */ // Deps extern crate bytesize; extern crate hostname; #[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))] extern crate users; // Local use super::{ Context, DialogYesNoOption, FileExplorerTab, FileTransferActivity, FsEntry, InputField, LogLevel, LogRecord, Popup, }; use crate::fs::explorer::{FileExplorer, FileSorting}; use crate::utils::fmt::{align_text_center, fmt_time}; // Ext use bytesize::ByteSize; use std::path::{Path, PathBuf}; use tui::{ layout::{Constraint, Corner, Direction, Layout, Rect}, style::{Color, Modifier, Style}, text::{Span, Spans}, widgets::{ Block, BorderType, Borders, Clear, Gauge, List, ListItem, ListState, Paragraph, Tabs, }, }; use unicode_width::UnicodeWidthStr; #[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))] use users::{get_group_by_gid, get_user_by_uid}; impl FileTransferActivity { /// ### draw /// /// Draw UI pub(super) fn draw(&mut self) { let mut ctx: Context = self.context.take().unwrap(); let _ = ctx.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]); // Set localhost state let mut localhost_state: ListState = ListState::default(); localhost_state.select(Some(self.local.get_relative_index())); // Set remote state let mut remote_state: ListState = ListState::default(); remote_state.select(Some(self.remote.get_relative_index())); // Draw tabs f.render_stateful_widget( self.draw_local_explorer(tabs_chunks[0].width), tabs_chunks[0], &mut localhost_state, ); f.render_stateful_widget( self.draw_remote_explorer(tabs_chunks[1].width), tabs_chunks[1], &mut remote_state, ); // Set log state let mut log_state: ListState = ListState::default(); log_state.select(Some(self.log_index)); // Draw log f.render_stateful_widget( self.draw_log_list(chunks[1].width), chunks[1], &mut log_state, ); // Draw popup if let Some(popup) = &self.popup { // Calculate popup size let (width, height): (u16, u16) = match popup { Popup::Alert(_, _) => (50, 10), Popup::Fatal(_) => (50, 10), Popup::FileInfo => (50, 50), Popup::FileSortingDialog => (50, 10), Popup::Help => (50, 80), Popup::Input(_, _) => (40, 10), Popup::Progress(_) => (40, 10), Popup::Wait(_) => (50, 10), Popup::YesNo(_, _, _) => (30, 10), }; let popup_area: Rect = self.draw_popup_area(f.size(), width, height); f.render_widget(Clear, popup_area); //this clears out the background match popup { Popup::Alert(color, txt) => f.render_widget( self.draw_popup_alert(*color, txt.clone(), popup_area.width), popup_area, ), Popup::Fatal(txt) => f.render_widget( self.draw_popup_fatal(txt.clone(), popup_area.width), popup_area, ), Popup::FileInfo => f.render_widget(self.draw_popup_fileinfo(), popup_area), Popup::FileSortingDialog => { f.render_widget(self.draw_popup_file_sorting_dialog(), popup_area) } Popup::Help => f.render_widget(self.draw_popup_help(), popup_area), Popup::Input(txt, _) => { f.render_widget(self.draw_popup_input(txt.clone()), popup_area); // Set cursor f.set_cursor( popup_area.x + self.input_txt.width() as u16 + 1, popup_area.y + 1, ) } Popup::Progress(txt) => { f.render_widget(self.draw_popup_progress(txt.clone()), popup_area) } Popup::Wait(txt) => f.render_widget( self.draw_popup_wait(txt.clone(), popup_area.width), popup_area, ), Popup::YesNo(txt, _, _) => { f.render_widget(self.draw_popup_yesno(txt.clone()), popup_area) } } } }); self.context = Some(ctx); } /// ### draw_local_explorer /// /// Draw local explorer list pub(super) fn draw_local_explorer(&self, width: u16) -> List { let hostname: String = match hostname::get() { Ok(h) => { let hostname: String = h.as_os_str().to_string_lossy().to_string(); let tokens: Vec<&str> = hostname.split('.').collect(); String::from(*tokens.get(0).unwrap_or(&"localhost")) } Err(_) => String::from("localhost"), }; let files: Vec = self .local .iter_files() .map(|entry: &FsEntry| ListItem::new(Span::from(self.local.fmt_file(entry)))) .collect(); // Get colors to use; highlight element inverting fg/bg only when tab is active let (fg, bg): (Color, Color) = match self.tab { FileExplorerTab::Local => (Color::Black, Color::LightYellow), _ => (Color::LightYellow, Color::Reset), }; List::new(files) .block( Block::default() .borders(Borders::ALL) .border_style(match self.input_field { InputField::Explorer => match self.tab { FileExplorerTab::Local => Style::default().fg(Color::LightYellow), _ => Style::default(), }, _ => Style::default(), }) .title(format!( "{}:{} ", hostname, FileTransferActivity::elide_wrkdir_path( self.local.wrkdir.as_path(), hostname.as_str(), width ) .display() )), ) .start_corner(Corner::TopLeft) .highlight_style(Style::default().fg(fg).bg(bg).add_modifier(Modifier::BOLD)) } /// ### draw_remote_explorer /// /// Draw remote explorer list pub(super) fn draw_remote_explorer(&self, width: u16) -> List { let files: Vec = self .remote .iter_files() .map(|entry: &FsEntry| ListItem::new(Span::from(self.remote.fmt_file(entry)))) .collect(); // Get colors to use; highlight element inverting fg/bg only when tab is active let (fg, bg): (Color, Color) = match self.tab { FileExplorerTab::Remote => (Color::Black, Color::LightBlue), _ => (Color::LightBlue, Color::Reset), }; List::new(files) .block( Block::default() .borders(Borders::ALL) .border_style(match self.input_field { InputField::Explorer => match self.tab { FileExplorerTab::Remote => Style::default().fg(Color::LightBlue), _ => Style::default(), }, _ => Style::default(), }) .title(format!( "{}:{} ", self.params.address, FileTransferActivity::elide_wrkdir_path( self.remote.wrkdir.as_path(), self.params.address.as_str(), width ) .display() )), ) .start_corner(Corner::TopLeft) .highlight_style(Style::default().bg(bg).fg(fg).add_modifier(Modifier::BOLD)) } /// ### draw_log_list /// /// Draw log list /// Chunk width must be provided to wrap text pub(super) fn draw_log_list(&self, width: u16) -> List { let events: Vec = self .log_records .iter() .map(|record: &LogRecord| { let record_rows = textwrap::wrap(record.msg.as_str(), (width as usize) - 35); // -35 'cause log prefix let s = match record.level { LogLevel::Error => Style::default().fg(Color::Red), LogLevel::Warn => Style::default().fg(Color::Yellow), LogLevel::Info => Style::default().fg(Color::Green), }; let mut rows: Vec = Vec::with_capacity(record_rows.len()); // Iterate over remaining rows for (idx, row) in record_rows.iter().enumerate() { let row: Spans = match idx { 0 => Spans::from(vec![ Span::from(format!("{}", record.time.format("%Y-%m-%dT%H:%M:%S%Z"))), Span::raw(" ["), Span::styled( format!( "{:5}", match record.level { LogLevel::Error => "ERROR", LogLevel::Warn => "WARN", LogLevel::Info => "INFO", } ), s, ), Span::raw("]: "), Span::from(String::from(row.as_ref())), ]), _ => Spans::from(vec![Span::from(textwrap::indent( row.as_ref(), " ", ))]), }; rows.push(row); } ListItem::new(rows) }) .collect(); List::new(events) .block( Block::default() .borders(Borders::ALL) .border_style(match self.input_field { InputField::Logs => Style::default().fg(Color::LightGreen), _ => Style::default(), }) .title("Log"), ) .start_corner(Corner::BottomLeft) .highlight_style(Style::default().add_modifier(Modifier::BOLD)) } /// ### draw_popup_area /// /// Draw popup area pub(super) fn draw_popup_area(&self, area: Rect, width: u16, height: u16) -> Rect { let popup_layout = Layout::default() .direction(Direction::Vertical) .constraints( [ Constraint::Percentage((100 - height) / 2), Constraint::Percentage(height), Constraint::Percentage((100 - height) / 2), ] .as_ref(), ) .split(area); Layout::default() .direction(Direction::Horizontal) .constraints( [ Constraint::Percentage((100 - width) / 2), Constraint::Percentage(width), Constraint::Percentage((100 - width) / 2), ] .as_ref(), ) .split(popup_layout[1])[1] } /// ### draw_popup_alert /// /// Draw alert popup pub(super) fn draw_popup_alert(&self, color: Color, text: String, width: u16) -> List { // Wraps texts let message_rows = textwrap::wrap(text.as_str(), width as usize); let mut lines: Vec = Vec::new(); for msg in message_rows.iter() { lines.push(ListItem::new(Spans::from(align_text_center(msg, width)))); } List::new(lines) .block( Block::default() .borders(Borders::ALL) .border_style(Style::default().fg(color)) .border_type(BorderType::Rounded) .title("Alert"), ) .start_corner(Corner::TopLeft) .style(Style::default().fg(color)) } /// ### draw_popup_fatal /// /// Draw fatal error popup pub(super) fn draw_popup_fatal(&self, text: String, width: u16) -> List { // Wraps texts let message_rows = textwrap::wrap(text.as_str(), width as usize); let mut lines: Vec = Vec::new(); for msg in message_rows.iter() { lines.push(ListItem::new(Spans::from(align_text_center(msg, width)))); } List::new(lines) .block( Block::default() .borders(Borders::ALL) .border_style(Style::default().fg(Color::Red)) .border_type(BorderType::Rounded) .title("Fatal error"), ) .start_corner(Corner::TopLeft) .style(Style::default().fg(Color::Red)) } /// ### draw_popup_file_sorting_dialog /// /// Draw FileSorting mode select popup pub(super) fn draw_popup_file_sorting_dialog(&self) -> Tabs { let choices: Vec = vec![ Spans::from("Name"), Spans::from("Modify time"), Spans::from("Creation time"), Spans::from("Size"), ]; let explorer: &FileExplorer = match self.tab { FileExplorerTab::Local => &self.local, FileExplorerTab::Remote => &self.remote, }; let index: usize = match explorer.get_file_sorting() { FileSorting::ByCreationTime => 2, FileSorting::ByModifyTime => 1, FileSorting::ByName => 0, FileSorting::BySize => 3, }; Tabs::new(choices) .block( Block::default() .borders(Borders::ALL) .border_type(BorderType::Rounded) .title("Sort files by"), ) .select(index) .style(Style::default()) .highlight_style( Style::default() .add_modifier(Modifier::BOLD) .bg(Color::LightMagenta) .fg(Color::DarkGray), ) } /// ### draw_popup_input /// /// Draw input popup pub(super) fn draw_popup_input(&self, text: String) -> Paragraph { Paragraph::new(self.input_txt.as_ref()) .style(Style::default().fg(Color::White)) .block( Block::default() .borders(Borders::ALL) .border_type(BorderType::Rounded) .title(text), ) } /// ### draw_popup_progress /// /// Draw progress popup pub(super) fn draw_popup_progress(&self, text: String) -> Gauge { // Calculate ETA let elapsed_secs: u64 = self.transfer.started.elapsed().as_secs(); let eta: String = match self.transfer.progress as u64 { 0 => String::from("--:--"), // NOTE: would divide by 0 :D _ => { let eta: u64 = ((elapsed_secs * 100) / (self.transfer.progress as u64)) - elapsed_secs; format!("{:0width$}:{:0width$}", (eta / 60), (eta % 60), width = 2) } }; // Calculate bytes/s let label = format!( "{:.2}% - ETA {} ({}/s)", self.transfer.progress, eta, ByteSize(self.transfer.bytes_per_second()) ); Gauge::default() .block(Block::default().borders(Borders::ALL).title(text)) .gauge_style( Style::default() .fg(Color::Green) .bg(Color::Black) .add_modifier(Modifier::BOLD), ) .label(label) .ratio(self.transfer.progress / 100.0) } /// ### draw_popup_wait /// /// Draw wait popup pub(super) fn draw_popup_wait(&self, text: String, width: u16) -> List { // Wraps texts let message_rows = textwrap::wrap(text.as_str(), width as usize); let mut lines: Vec = Vec::new(); for msg in message_rows.iter() { lines.push(ListItem::new(Spans::from(align_text_center(msg, width)))); } List::new(lines) .block( Block::default() .borders(Borders::ALL) .border_style(Style::default().fg(Color::White)) .border_type(BorderType::Rounded) .title("Please wait"), ) .start_corner(Corner::TopLeft) .style(Style::default().add_modifier(Modifier::BOLD)) } /// ### draw_popup_yesno /// /// Draw yes/no select popup pub(super) fn draw_popup_yesno(&self, text: String) -> Tabs { let choices: Vec = vec![Spans::from("Yes"), Spans::from("No")]; let index: usize = match self.choice_opt { DialogYesNoOption::Yes => 0, DialogYesNoOption::No => 1, }; Tabs::new(choices) .block( Block::default() .borders(Borders::ALL) .border_type(BorderType::Rounded) .title(text), ) .select(index) .style(Style::default()) .highlight_style( Style::default() .add_modifier(Modifier::BOLD) .fg(Color::Yellow), ) } /// ### draw_popup_fileinfo /// /// Draw popup containing info about selected fsentry pub(super) fn draw_popup_fileinfo(&self) -> List { let mut info: Vec = Vec::new(); // Get current fsentry let fsentry: Option<&FsEntry> = match self.tab { FileExplorerTab::Local => { // Get selected file match self.local.get_current_file() { Some(entry) => Some(entry), None => None, } } FileExplorerTab::Remote => match self.remote.get_current_file() { Some(entry) => Some(entry), None => None, }, }; // Get file_name and fill info list let file_name: String = match fsentry { Some(fsentry) => { // Get name and path let abs_path: PathBuf = fsentry.get_abs_path(); let name: String = fsentry.get_name().to_string(); let ctime: String = fmt_time(fsentry.get_creation_time(), "%b %d %Y %H:%M:%S"); let atime: String = fmt_time(fsentry.get_last_access_time(), "%b %d %Y %H:%M:%S"); let mtime: String = fmt_time(fsentry.get_creation_time(), "%b %d %Y %H:%M:%S"); let (bsize, size): (ByteSize, usize) = (ByteSize(fsentry.get_size() as u64), fsentry.get_size()); let user: Option = fsentry.get_user(); let group: Option = fsentry.get_group(); let real_path: Option = { let real_file: FsEntry = fsentry.get_realfile(); match real_file.get_abs_path() != abs_path { true => Some(real_file.get_abs_path()), false => None, } }; // Push path info.push(ListItem::new(Spans::from(vec![ Span::styled("Path: ", Style::default()), Span::styled( match real_path { Some(symlink) => { format!("{} => {}", abs_path.display(), symlink.display()) } None => abs_path.to_string_lossy().to_string(), }, Style::default() .fg(Color::LightYellow) .add_modifier(Modifier::BOLD), ), ]))); // Push file type if let Some(ftype) = fsentry.get_ftype() { info.push(ListItem::new(Spans::from(vec![ Span::styled("File type: ", Style::default()), Span::styled( ftype, Style::default() .fg(Color::Green) .add_modifier(Modifier::BOLD), ), ]))); } // Push size info.push(ListItem::new(Spans::from(vec![ Span::styled("Size: ", Style::default()), Span::styled( format!("{} ({})", bsize, size), Style::default() .fg(Color::LightBlue) .add_modifier(Modifier::BOLD), ), ]))); // Push creation time info.push(ListItem::new(Spans::from(vec![ Span::styled("Creation time: ", Style::default()), Span::styled( ctime, Style::default() .fg(Color::LightGreen) .add_modifier(Modifier::BOLD), ), ]))); // Push Last change info.push(ListItem::new(Spans::from(vec![ Span::styled("Last change time: ", Style::default()), Span::styled( mtime, Style::default().fg(Color::Red).add_modifier(Modifier::BOLD), ), ]))); // Push Last access info.push(ListItem::new(Spans::from(vec![ Span::styled("Last access time: ", Style::default()), Span::styled( atime, Style::default() .fg(Color::LightMagenta) .add_modifier(Modifier::BOLD), ), ]))); // User #[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))] let username: String = match 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!("{}", user.unwrap_or(0)); info.push(ListItem::new(Spans::from(vec![ Span::styled("User: ", Style::default()), Span::styled( username, Style::default() .fg(Color::LightRed) .add_modifier(Modifier::BOLD), ), ]))); // Group #[cfg(any(target_os = "unix", target_os = "macos", target_os = "linux"))] let group: String = match 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!("{}", group.unwrap_or(0)); info.push(ListItem::new(Spans::from(vec![ Span::styled("Group: ", Style::default()), Span::styled( group, Style::default() .fg(Color::LightBlue) .add_modifier(Modifier::BOLD), ), ]))); // Finally return file name name } None => String::from(""), }; List::new(info) .block( Block::default() .borders(Borders::ALL) .border_style(Style::default()) .border_type(BorderType::Rounded) .title(file_name), ) .start_corner(Corner::TopLeft) } /// ### draw_footer /// /// Draw authentication page footer pub(super) fn draw_popup_help(&self) -> List { // Write header let cmds: Vec = vec![ ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Disconnect"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Switch between log tab and explorer"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Go to previous directory in stack"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Change explorer tab"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Move up/down in list"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Scroll up/down in list quickly"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Enter directory"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Upload/download file"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Delete file"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Toggle hidden files"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Change file sorting mode"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Copy"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Make directory"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Same as "), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Goto path"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Show help"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Show info about the selected file or directory"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Reload directory content"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("New file"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Open text file"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Quit TermSCP"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Rename file"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Go to parent directory"), ])), ListItem::new(Spans::from(vec![ Span::styled( "", Style::default() .fg(Color::Cyan) .add_modifier(Modifier::BOLD), ), Span::raw(" "), Span::raw("Abort current file transfer"), ])), ]; List::new(cmds) .block( Block::default() .borders(Borders::ALL) .border_style(Style::default()) .border_type(BorderType::Rounded) .title("Help"), ) .start_corner(Corner::TopLeft) } /// ### elide_wrkdir_path /// /// Elide working directory path if longer than width + host.len /// In this case, the path is formatted to {ANCESTOR[0]}/.../{PARENT[0]}/{BASENAME} fn elide_wrkdir_path(wrkdir: &Path, host: &str, width: u16) -> PathBuf { let fmt_path: String = format!("{}", wrkdir.display()); // NOTE: +5 is const match fmt_path.len() + host.len() + 5 > width as usize { false => PathBuf::from(wrkdir), true => { // Elide let ancestors_len: usize = wrkdir.ancestors().count(); let mut ancestors = wrkdir.ancestors(); let mut elided_path: PathBuf = PathBuf::new(); // If ancestors_len's size is bigger than 2, push count - 2 if ancestors_len > 2 { elided_path.push(ancestors.nth(ancestors_len - 2).unwrap()); } // If ancestors_len is bigger than 3, push '...' and parent too if ancestors_len > 3 { elided_path.push("..."); if let Some(parent) = wrkdir.ancestors().nth(1) { elided_path.push(parent.file_name().unwrap()); } } // Push file_name if let Some(name) = wrkdir.file_name() { elided_path.push(name); } elided_path } } } }