use crate::core::FileType;
use crate::core::{FileEntry, browse_dir};
use chrono::{DateTime, Local};
use humansize::{DECIMAL, format_size};
use unicode_width::UnicodeWidthChar;
use std::collections::HashSet;
use std::ffi::OsString;
use std::fs::{File, Metadata};
use std::io::{BufRead, BufReader, ErrorKind, Read, Seek};
use std::path::{Path, PathBuf};
use std::sync::Arc;
use std::time::SystemTime;
const MIN_PREVIEW_LINES: usize = 3;
const MAX_PREVIEW_SIZE: u64 = 10 * 1024 * 1024;
const HEADER_PEEK_BYTES: usize = 8;
const BINARY_PEEK_BYTES: usize = 1024;
pub struct Formatter {
dirs_first: bool,
show_hidden: bool,
show_system: bool,
case_insensitive: bool,
always_show: Arc<HashSet<OsString>>,
always_show_lowercase: Arc<HashSet<String>>,
}
impl Formatter {
pub fn new(
dirs_first: bool,
show_hidden: bool,
show_system: bool,
case_insensitive: bool,
always_show: Arc<HashSet<OsString>>,
) -> Self {
let always_show_lowercase = Arc::new(
always_show
.iter()
.map(|s| s.to_string_lossy().to_lowercase())
.collect::<HashSet<String>>(),
);
Self {
dirs_first,
show_hidden,
show_system,
case_insensitive,
always_show,
always_show_lowercase,
}
}
pub fn sort_entries(&self, entries: &mut [FileEntry]) {
entries.sort_by(|a, b| {
if self.dirs_first {
match (a.is_dir(), b.is_dir()) {
(true, false) => return std::cmp::Ordering::Less,
(false, true) => return std::cmp::Ordering::Greater,
_ => {}
}
}
if self.case_insensitive {
a.lowercase_name().cmp(b.lowercase_name())
} else {
a.name_str().cmp(&b.name_str())
}
});
}
pub fn filter_entries(&self, entries: &mut Vec<FileEntry>) {
entries.retain(|e| {
let is_exception = if self.case_insensitive {
self.always_show_lowercase.contains(e.lowercase_name())
} else {
self.always_show.contains(e.name())
};
if is_exception {
return true;
}
let hidden_ok = self.show_hidden || !e.is_hidden();
let system_ok = self.show_system || !e.is_system();
hidden_ok && system_ok
});
self.sort_entries(entries);
}
}
pub fn format_attributes(meta: &Metadata) -> String {
#[cfg(unix)]
{
use std::os::unix::fs::PermissionsExt;
let file_type = meta.file_type();
let first = if file_type.is_dir() {
'd'
} else if file_type.is_symlink() {
'l'
} else {
'-'
};
let mode = meta.permissions().mode();
let mut chars = [first, '-', '-', '-', '-', '-', '-', '-', '-', '-'];
let shifts = [6, 3, 0];
for (i, &shift) in shifts.iter().enumerate() {
let base = 1 + i * 3;
if (mode >> (shift + 2)) & 1u32 != 0 {
chars[base] = 'r';
}
if (mode >> (shift + 1)) & 1u32 != 0 {
chars[base + 1] = 'w';
}
if (mode >> shift) & 1u32 != 0 {
chars[base + 2] = 'x';
}
}
chars.iter().collect()
}
#[cfg(windows)]
{
use std::os::windows::fs::MetadataExt;
let attr = meta.file_attributes();
let mut out = String::with_capacity(7);
out.push(if attr & 0x10 != 0 {
'd'
} else if attr & 0x400 != 0 {
'l'
} else {
'-'
});
out.push(if attr & 0x02 != 0 { 'h' } else { '-' });
out.push(if attr & 0x04 != 0 { 's' } else { '-' });
out.push(if attr & 0x20 != 0 { 'a' } else { '-' });
out.push(if attr & 0x01 != 0 { 'r' } else { '-' });
out
}
}
pub fn format_file_type(file_type: &FileType) -> &'static str {
match file_type {
FileType::File => "File",
FileType::Directory => "Directory",
FileType::Symlink => "Symlink",
FileType::Other => "Other",
}
}
pub fn format_file_size(size: Option<u64>, is_dir: bool) -> String {
if is_dir {
"-".into()
} else if let Some(sz) = size {
format_size(sz, DECIMAL)
} else {
"-".to_string()
}
}
pub fn format_file_time(modified: Option<SystemTime>) -> String {
modified
.map(|mtime| {
let dt: DateTime<Local> = DateTime::from(mtime);
dt.format("%Y-%m-%d %H:%M:%S").to_string()
})
.unwrap_or_else(|| "-".to_string())
}
pub fn symlink_target_resolved(
entry: &crate::core::FileEntry,
parent_dir: &Path,
) -> Option<PathBuf> {
if !entry.is_symlink() {
return None;
}
let entry_path = parent_dir.join(entry.name());
if let Ok(target) = std::fs::read_link(&entry_path) {
let resolved = if target.is_absolute() {
target
} else {
entry_path
.parent()
.unwrap_or_else(|| Path::new(""))
.join(target)
};
Some(resolved)
} else {
None
}
}
pub fn sanitize_to_exact_width(line: &str, pane_width: usize) -> String {
let mut out = String::with_capacity(pane_width);
let mut current_w = 0;
for char in line.chars() {
if char == '\t' {
let space_count = 4 - (current_w % 4);
if current_w + space_count > pane_width {
break;
}
out.push_str(&" ".repeat(space_count));
current_w += space_count;
continue;
}
if char.is_control() {
continue;
}
let w = char.width().unwrap_or(0);
if current_w + w > pane_width {
break;
}
out.push(char);
current_w += w;
}
if current_w < pane_width {
out.push_str(&" ".repeat(pane_width - current_w));
}
out
}
pub fn preview_directory(path: &Path, max_lines: usize, pane_width: usize) -> Vec<String> {
match browse_dir(path) {
Ok(entries) => {
let mut lines = Vec::with_capacity(max_lines);
let total_entries = entries.len();
for e in entries.iter().take(max_lines) {
let display_name = if e.is_dir() {
e.name().to_string_lossy().clone() + "/"
} else {
e.name().to_string_lossy().clone()
};
lines.push(sanitize_to_exact_width(&display_name, pane_width));
}
if lines.is_empty() {
lines.push(sanitize_to_exact_width("[empty directory]", pane_width));
} else if total_entries > max_lines
&& let Some(last) = lines.last_mut()
{
*last = sanitize_to_exact_width("...", pane_width);
}
while lines.len() < max_lines {
lines.push(" ".repeat(pane_width));
}
lines
}
Err(e) => {
let err_msg = "[Error: ".to_owned() + &e.to_string() + "]";
let mut err_lines = vec![sanitize_to_exact_width(&err_msg, pane_width)];
while err_lines.len() < max_lines {
err_lines.push(" ".repeat(pane_width));
}
err_lines
}
}
}
pub fn safe_read_preview(path: &Path, max_lines: usize, pane_width: usize) -> Vec<String> {
let max_lines = std::cmp::max(max_lines, MIN_PREVIEW_LINES);
let Ok(meta) = std::fs::metadata(path) else {
return vec![sanitize_to_exact_width(
"[Error: Access Denied]",
pane_width,
)];
};
if meta.is_dir() {
return preview_directory(path, max_lines, pane_width);
}
if meta.len() > MAX_PREVIEW_SIZE {
return vec![sanitize_to_exact_width(
"[File too large for preview]",
pane_width,
)];
}
if !meta.is_file() {
return vec![sanitize_to_exact_width("[Not a regular file]", pane_width)];
}
match File::open(path) {
Ok(mut file) => {
let mut header = [0u8; HEADER_PEEK_BYTES];
let read_bytes = file.read(&mut header).unwrap_or(0);
if read_bytes >= 5 && &header[..5] == b"%PDF-" {
return vec![sanitize_to_exact_width(
"[Binary file - preview hidden]",
pane_width,
)];
}
let mut buffer = [0u8; BINARY_PEEK_BYTES];
let n = file.read(&mut buffer).unwrap_or(0);
if buffer[..n].contains(&0) {
return vec![sanitize_to_exact_width(
"[Binary file - preview hidden]",
pane_width,
)];
}
let _ = file.rewind();
let reader = BufReader::new(file);
let mut preview_lines = Vec::with_capacity(max_lines);
for line_result in reader.lines().take(max_lines) {
match line_result {
Ok(line) => {
preview_lines.push(sanitize_to_exact_width(&line, pane_width));
}
Err(_) => break,
}
}
if preview_lines.is_empty() {
preview_lines.push(sanitize_to_exact_width("[Empty file]", pane_width));
}
preview_lines
}
Err(e) => {
let msg = match e.kind() {
ErrorKind::PermissionDenied => "[Error: Permission Denied]",
ErrorKind::NotFound => "[Error: File Not Found]",
_ => {
return vec![sanitize_to_exact_width(
&format!("[Error reading file: {}]", e),
pane_width,
)];
}
};
vec![sanitize_to_exact_width(msg, pane_width)]
}
}
}