1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
use std::fmt::Display;
use regex::Regex;
use crate::fileinfo::{FileInfo, FileKind};
#[derive(Clone)]
pub enum FilterKind {
Extension(String),
Name(String),
Directory,
All,
}
impl FilterKind {
pub fn from_input(input: &str) -> Self {
let words: Vec<&str> = input.split_whitespace().collect();
if words.is_empty() {
return Self::All;
}
match words[0] {
"d" => Self::Directory,
"e" if words.len() > 1 => Self::Extension(words[1].to_owned()),
"n" if words.len() > 1 => Self::Name(words[1].to_owned()),
_ => Self::All,
}
}
pub fn filter_by(&self, fileinfo: &FileInfo) -> bool {
match self {
Self::Extension(ext) => Self::filter_by_extension(fileinfo, ext),
Self::Name(filename) => Self::filter_by_name(fileinfo, filename),
Self::Directory => Self::filter_directory(fileinfo),
Self::All => true,
}
}
fn filter_by_extension(fileinfo: &FileInfo, ext: &str) -> bool {
fileinfo.extension == ext
}
fn filter_by_name(fileinfo: &FileInfo, filename: &str) -> bool {
if let Ok(re) = Regex::new(filename) {
re.is_match(&fileinfo.filename)
} else {
false
}
}
fn filter_directory(fileinfo: &FileInfo) -> bool {
matches!(fileinfo.file_kind, FileKind::Directory)
}
}
impl Display for FilterKind {
fn fmt(&self, f: &mut std::fmt::Formatter) -> std::fmt::Result {
match self {
Self::Directory => write!(f, "Filter: Directory only"),
Self::Extension(s) => write!(f, "Filter: by extension \"{}\"", s),
Self::Name(s) => write!(f, "Filter: by name \"{}\"", s),
Self::All => write!(f, ""),
}
}
}