2024-01-03 15:27:32 -07:00
|
|
|
use cosmic::{
|
|
|
|
|
app::Core,
|
|
|
|
|
cosmic_theme,
|
2024-01-05 08:55:18 -07:00
|
|
|
iced::{
|
|
|
|
|
alignment::{Horizontal, Vertical},
|
2024-02-22 16:17:39 -07:00
|
|
|
futures::SinkExt,
|
2024-01-10 10:47:12 -07:00
|
|
|
keyboard::Modifiers,
|
2024-02-22 16:17:39 -07:00
|
|
|
subscription::{self, Subscription},
|
2024-01-29 11:09:05 -07:00
|
|
|
//TODO: export in cosmic::widget
|
2024-02-26 12:05:29 -07:00
|
|
|
widget::{horizontal_rule, scrollable::Viewport},
|
2024-01-29 11:09:05 -07:00
|
|
|
Alignment,
|
2024-02-22 21:34:21 -07:00
|
|
|
ContentFit,
|
2024-01-29 11:09:05 -07:00
|
|
|
Length,
|
|
|
|
|
Point,
|
2024-02-26 12:05:29 -07:00
|
|
|
Rectangle,
|
|
|
|
|
Size,
|
2024-01-05 08:55:18 -07:00
|
|
|
},
|
2024-01-03 15:27:32 -07:00
|
|
|
theme, widget, Element,
|
|
|
|
|
};
|
2024-02-22 15:04:37 -07:00
|
|
|
use mime_guess::MimeGuess;
|
2024-01-24 09:30:06 -05:00
|
|
|
use once_cell::sync::Lazy;
|
2024-01-03 15:27:32 -07:00
|
|
|
use std::{
|
|
|
|
|
cmp::Ordering,
|
2024-01-04 15:56:01 -07:00
|
|
|
collections::HashMap,
|
2024-01-05 14:44:20 -07:00
|
|
|
fmt,
|
|
|
|
|
fs::{self, Metadata},
|
2024-01-03 15:27:32 -07:00
|
|
|
path::PathBuf,
|
|
|
|
|
process,
|
|
|
|
|
time::{Duration, Instant},
|
|
|
|
|
};
|
|
|
|
|
|
2024-02-18 02:44:54 -05:00
|
|
|
use crate::{
|
2024-02-26 12:05:29 -07:00
|
|
|
app::Action,
|
2024-02-18 02:44:54 -05:00
|
|
|
config::{IconSizes, TabConfig},
|
|
|
|
|
dialog::DialogKind,
|
2024-02-26 12:05:29 -07:00
|
|
|
fl, menu,
|
2024-02-18 02:44:54 -05:00
|
|
|
mime_icon::mime_icon,
|
2024-02-26 12:05:29 -07:00
|
|
|
mouse_area,
|
2024-02-18 02:44:54 -05:00
|
|
|
};
|
2024-01-03 15:33:28 -07:00
|
|
|
|
2024-01-03 15:27:32 -07:00
|
|
|
const DOUBLE_CLICK_DURATION: Duration = Duration::from_millis(500);
|
2024-02-22 15:04:37 -07:00
|
|
|
//TODO: adjust for locales?
|
|
|
|
|
const TIME_FORMAT: &'static str = "%a %-d %b %-Y %r";
|
2024-01-24 09:30:06 -05:00
|
|
|
static SPECIAL_DIRS: Lazy<HashMap<PathBuf, &'static str>> = Lazy::new(|| {
|
|
|
|
|
let mut special_dirs = HashMap::new();
|
|
|
|
|
if let Some(dir) = dirs::document_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-documents");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::download_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-download");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::audio_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-music");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::picture_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-pictures");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::public_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-publicshare");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::template_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-templates");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::video_dir() {
|
|
|
|
|
special_dirs.insert(dir, "folder-videos");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::desktop_dir() {
|
|
|
|
|
special_dirs.insert(dir, "user-desktop");
|
|
|
|
|
}
|
|
|
|
|
if let Some(dir) = dirs::home_dir() {
|
|
|
|
|
special_dirs.insert(dir, "user-home");
|
|
|
|
|
}
|
|
|
|
|
special_dirs
|
|
|
|
|
});
|
2024-01-05 09:36:16 -07:00
|
|
|
fn button_style(selected: bool) -> theme::Button {
|
|
|
|
|
//TODO: move to libcosmic
|
|
|
|
|
theme::Button::Custom {
|
|
|
|
|
active: Box::new(move |focused, theme| {
|
|
|
|
|
let mut appearance =
|
|
|
|
|
widget::button::StyleSheet::active(theme, focused, &theme::Button::MenuItem);
|
|
|
|
|
if !selected {
|
|
|
|
|
appearance.background = None;
|
|
|
|
|
}
|
|
|
|
|
appearance
|
|
|
|
|
}),
|
|
|
|
|
disabled: Box::new(move |theme| {
|
|
|
|
|
let mut appearance =
|
|
|
|
|
widget::button::StyleSheet::disabled(theme, &theme::Button::MenuItem);
|
|
|
|
|
if !selected {
|
|
|
|
|
appearance.background = None;
|
|
|
|
|
}
|
|
|
|
|
appearance
|
|
|
|
|
}),
|
|
|
|
|
hovered: Box::new(move |focused, theme| {
|
|
|
|
|
widget::button::StyleSheet::hovered(theme, focused, &theme::Button::MenuItem)
|
|
|
|
|
}),
|
|
|
|
|
pressed: Box::new(move |focused, theme| {
|
|
|
|
|
widget::button::StyleSheet::pressed(theme, focused, &theme::Button::MenuItem)
|
|
|
|
|
}),
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 15:34:48 -07:00
|
|
|
pub fn folder_icon(path: &PathBuf, icon_size: u16) -> widget::icon::Handle {
|
2024-01-04 15:56:01 -07:00
|
|
|
widget::icon::from_name(SPECIAL_DIRS.get(path).map_or("folder", |x| *x))
|
|
|
|
|
.size(icon_size)
|
2024-01-05 08:55:18 -07:00
|
|
|
.handle()
|
2024-01-04 15:56:01 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-09 15:34:48 -07:00
|
|
|
pub fn folder_icon_symbolic(path: &PathBuf, icon_size: u16) -> widget::icon::Handle {
|
|
|
|
|
widget::icon::from_name(format!(
|
|
|
|
|
"{}-symbolic",
|
|
|
|
|
SPECIAL_DIRS.get(path).map_or("folder", |x| *x)
|
|
|
|
|
))
|
|
|
|
|
.size(icon_size)
|
|
|
|
|
.handle()
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-31 14:15:21 -07:00
|
|
|
pub fn trash_icon_symbolic(icon_size: u16) -> widget::icon::Handle {
|
2024-01-10 12:57:30 -07:00
|
|
|
let full = match trash::os_limited::list() {
|
|
|
|
|
Ok(entries) => !entries.is_empty(),
|
|
|
|
|
Err(_err) => false,
|
|
|
|
|
};
|
|
|
|
|
widget::icon::from_name(if full {
|
|
|
|
|
"user-trash-full-symbolic"
|
|
|
|
|
} else {
|
|
|
|
|
"user-trash-symbolic"
|
|
|
|
|
})
|
|
|
|
|
.size(icon_size)
|
|
|
|
|
.handle()
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 15:32:42 -07:00
|
|
|
//TODO: translate, add more levels?
|
|
|
|
|
fn format_size(size: u64) -> String {
|
2024-01-29 10:34:09 -07:00
|
|
|
const KB: u64 = 1000;
|
|
|
|
|
const MB: u64 = 1000 * KB;
|
|
|
|
|
const GB: u64 = 1000 * MB;
|
|
|
|
|
const TB: u64 = 1000 * GB;
|
2024-01-05 15:32:42 -07:00
|
|
|
|
2024-01-29 10:34:09 -07:00
|
|
|
if size >= TB {
|
|
|
|
|
format!("{:.1} TB", size as f64 / TB as f64)
|
|
|
|
|
} else if size >= GB {
|
|
|
|
|
format!("{:.1} GB", size as f64 / GB as f64)
|
|
|
|
|
} else if size >= MB {
|
|
|
|
|
format!("{:.1} MB", size as f64 / MB as f64)
|
|
|
|
|
} else if size >= KB {
|
|
|
|
|
format!("{:.1} KB", size as f64 / KB as f64)
|
2024-01-05 15:32:42 -07:00
|
|
|
} else {
|
|
|
|
|
format!("{} B", size)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-04 16:08:22 -07:00
|
|
|
#[cfg(not(target_os = "windows"))]
|
2024-01-05 14:44:20 -07:00
|
|
|
fn hidden_attribute(_metadata: &Metadata) -> bool {
|
2024-01-04 16:08:22 -07:00
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(target_os = "windows")]
|
2024-01-05 14:44:20 -07:00
|
|
|
fn hidden_attribute(metadata: &Metadata) -> bool {
|
2024-01-04 16:08:22 -07:00
|
|
|
use std::os::windows::fs::MetadataExt;
|
2024-01-05 14:44:20 -07:00
|
|
|
// https://learn.microsoft.com/en-us/windows/win32/fileio/file-attribute-constants
|
|
|
|
|
const FILE_ATTRIBUTE_HIDDEN: u32 = 2;
|
|
|
|
|
metadata.file_attributes() & FILE_ATTRIBUTE_HIDDEN == FILE_ATTRIBUTE_HIDDEN
|
2024-01-04 16:08:22 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-03 15:41:01 -07:00
|
|
|
#[cfg(target_os = "linux")]
|
2024-01-04 16:08:22 -07:00
|
|
|
fn open_command(path: &PathBuf) -> process::Command {
|
2024-01-03 15:41:01 -07:00
|
|
|
let mut command = process::Command::new("xdg-open");
|
|
|
|
|
command.arg(path);
|
|
|
|
|
command
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(target_os = "macos")]
|
2024-01-04 16:08:22 -07:00
|
|
|
fn open_command(path: &PathBuf) -> process::Command {
|
2024-01-03 15:41:01 -07:00
|
|
|
let mut command = process::Command::new("open");
|
|
|
|
|
command.arg(path);
|
|
|
|
|
command
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(target_os = "redox")]
|
2024-01-04 16:08:22 -07:00
|
|
|
fn open_command(path: &PathBuf) -> process::Command {
|
2024-01-03 15:41:01 -07:00
|
|
|
let mut command = process::Command::new("launcher");
|
|
|
|
|
command.arg(path);
|
|
|
|
|
command
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[cfg(target_os = "windows")]
|
2024-01-04 16:08:22 -07:00
|
|
|
fn open_command(path: &PathBuf) -> process::Command {
|
2024-01-28 06:23:40 -05:00
|
|
|
use std::os::windows::process::CommandExt;
|
|
|
|
|
|
2024-01-03 15:41:01 -07:00
|
|
|
let mut command = process::Command::new("cmd");
|
2024-01-28 06:23:40 -05:00
|
|
|
|
|
|
|
|
command
|
|
|
|
|
.arg("/c")
|
|
|
|
|
.arg("start")
|
|
|
|
|
.raw_arg("\"\"")
|
|
|
|
|
.arg(path)
|
|
|
|
|
.creation_flags(0x08000000);
|
2024-01-03 15:41:01 -07:00
|
|
|
command
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-18 02:44:54 -05:00
|
|
|
pub fn scan_path(tab_path: &PathBuf, sizes: IconSizes) -> Vec<Item> {
|
2024-01-05 08:55:18 -07:00
|
|
|
let mut items = Vec::new();
|
2024-01-05 16:17:23 -07:00
|
|
|
match fs::read_dir(tab_path) {
|
2024-01-05 08:55:18 -07:00
|
|
|
Ok(entries) => {
|
|
|
|
|
for entry_res in entries {
|
|
|
|
|
let entry = match entry_res {
|
|
|
|
|
Ok(ok) => ok,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to read entry in {:?}: {}", tab_path, err);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let name = match entry.file_name().into_string() {
|
2024-01-05 14:44:20 -07:00
|
|
|
Ok(ok) => ok,
|
2024-01-05 08:55:18 -07:00
|
|
|
Err(name_os) => {
|
|
|
|
|
log::warn!(
|
|
|
|
|
"failed to parse entry in {:?}: {:?} is not valid UTF-8",
|
|
|
|
|
tab_path,
|
|
|
|
|
name_os,
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2024-01-05 14:44:20 -07:00
|
|
|
let metadata = match entry.metadata() {
|
|
|
|
|
Ok(ok) => ok,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!(
|
|
|
|
|
"failed to read metadata for entry in {:?}: {}",
|
|
|
|
|
tab_path,
|
|
|
|
|
err
|
|
|
|
|
);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let hidden = name.starts_with(".") || hidden_attribute(&metadata);
|
|
|
|
|
|
2024-01-05 08:55:18 -07:00
|
|
|
let path = entry.path();
|
2024-01-05 14:44:20 -07:00
|
|
|
|
2024-02-22 15:04:37 -07:00
|
|
|
let mime_guess = MimeGuess::from_path(&path);
|
|
|
|
|
|
2024-02-22 21:46:25 -07:00
|
|
|
let (icon_handle_grid, icon_handle_list) = if metadata.is_dir() {
|
2024-01-05 09:36:16 -07:00
|
|
|
(
|
2024-02-18 02:44:54 -05:00
|
|
|
folder_icon(&path, sizes.grid()),
|
|
|
|
|
folder_icon(&path, sizes.list()),
|
2024-01-05 09:36:16 -07:00
|
|
|
)
|
2024-01-05 08:55:18 -07:00
|
|
|
} else {
|
2024-01-05 09:36:16 -07:00
|
|
|
(
|
2024-02-18 02:44:54 -05:00
|
|
|
mime_icon(&path, sizes.grid()),
|
|
|
|
|
mime_icon(&path, sizes.list()),
|
2024-01-05 09:36:16 -07:00
|
|
|
)
|
2024-01-05 08:55:18 -07:00
|
|
|
};
|
|
|
|
|
|
2024-01-06 20:31:00 -07:00
|
|
|
let children = if metadata.is_dir() {
|
|
|
|
|
//TODO: calculate children in the background (and make it cancellable?)
|
|
|
|
|
match fs::read_dir(&path) {
|
|
|
|
|
Ok(entries) => entries.count(),
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to read directory {:?}: {}", path, err);
|
|
|
|
|
0
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
} else {
|
|
|
|
|
0
|
|
|
|
|
};
|
|
|
|
|
|
2024-01-05 08:55:18 -07:00
|
|
|
items.push(Item {
|
|
|
|
|
name,
|
2024-01-30 10:47:41 -07:00
|
|
|
metadata: ItemMetadata::Path { metadata, children },
|
2024-01-05 08:55:18 -07:00
|
|
|
hidden,
|
2024-01-05 14:44:20 -07:00
|
|
|
path,
|
2024-02-22 15:04:37 -07:00
|
|
|
mime_guess,
|
2024-01-05 09:36:16 -07:00
|
|
|
icon_handle_grid,
|
|
|
|
|
icon_handle_list,
|
2024-02-22 16:17:39 -07:00
|
|
|
thumbnail_res_opt: match mime_guess.first() {
|
|
|
|
|
Some(mime) if mime.type_() == "image" => None,
|
|
|
|
|
_ => Some(Err(())),
|
|
|
|
|
},
|
2024-01-10 09:47:47 -07:00
|
|
|
selected: false,
|
|
|
|
|
click_time: None,
|
2024-01-05 08:55:18 -07:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to read directory {:?}: {}", tab_path, err);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-06 12:59:36 -07:00
|
|
|
items.sort_by(|a, b| match (a.metadata.is_dir(), b.metadata.is_dir()) {
|
|
|
|
|
(true, false) => Ordering::Less,
|
|
|
|
|
(false, true) => Ordering::Greater,
|
|
|
|
|
_ => lexical_sort::natural_lexical_cmp(&a.name, &b.name),
|
|
|
|
|
});
|
2024-01-05 16:17:23 -07:00
|
|
|
items
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This config statement is from trash::os_limited, inverted
|
|
|
|
|
#[cfg(not(any(
|
|
|
|
|
target_os = "windows",
|
|
|
|
|
all(
|
|
|
|
|
unix,
|
|
|
|
|
not(target_os = "macos"),
|
|
|
|
|
not(target_os = "ios"),
|
|
|
|
|
not(target_os = "android")
|
|
|
|
|
)
|
|
|
|
|
)))]
|
|
|
|
|
pub fn scan_trash() -> Vec<Item> {
|
|
|
|
|
log::warn!("viewing trash not supported on this platform");
|
|
|
|
|
Vec::new()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// This config statement is from trash::os_limited
|
|
|
|
|
#[cfg(any(
|
|
|
|
|
target_os = "windows",
|
|
|
|
|
all(
|
|
|
|
|
unix,
|
|
|
|
|
not(target_os = "macos"),
|
|
|
|
|
not(target_os = "ios"),
|
|
|
|
|
not(target_os = "android")
|
|
|
|
|
)
|
|
|
|
|
))]
|
2024-02-18 02:44:54 -05:00
|
|
|
pub fn scan_trash(sizes: IconSizes) -> Vec<Item> {
|
2024-01-05 16:17:23 -07:00
|
|
|
let mut items: Vec<Item> = Vec::new();
|
|
|
|
|
match trash::os_limited::list() {
|
|
|
|
|
Ok(entries) => {
|
|
|
|
|
for entry in entries {
|
2024-01-06 12:59:36 -07:00
|
|
|
let metadata = match trash::os_limited::metadata(&entry) {
|
|
|
|
|
Ok(ok) => ok,
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to get metadata for trash item {:?}: {}", entry, err);
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
|
2024-01-05 16:17:23 -07:00
|
|
|
let path = entry.original_path();
|
2024-01-30 10:47:41 -07:00
|
|
|
let name = entry.name.clone();
|
2024-01-05 16:17:23 -07:00
|
|
|
|
2024-02-22 15:04:37 -07:00
|
|
|
let mime_guess = MimeGuess::from_path(&path);
|
|
|
|
|
|
2024-02-22 21:46:25 -07:00
|
|
|
let (icon_handle_grid, icon_handle_list) = match metadata.size {
|
2024-01-10 08:53:22 -07:00
|
|
|
trash::TrashItemSize::Entries(_) => (
|
2024-02-18 02:44:54 -05:00
|
|
|
folder_icon(&path, sizes.grid()),
|
|
|
|
|
folder_icon(&path, sizes.list()),
|
2024-01-10 08:53:22 -07:00
|
|
|
),
|
|
|
|
|
trash::TrashItemSize::Bytes(_) => (
|
2024-02-18 02:44:54 -05:00
|
|
|
mime_icon(&path, sizes.grid()),
|
|
|
|
|
mime_icon(&path, sizes.list()),
|
2024-01-10 08:53:22 -07:00
|
|
|
),
|
2024-01-06 12:59:36 -07:00
|
|
|
};
|
2024-01-05 16:17:23 -07:00
|
|
|
|
|
|
|
|
items.push(Item {
|
|
|
|
|
name,
|
2024-01-30 10:47:41 -07:00
|
|
|
metadata: ItemMetadata::Trash { metadata, entry },
|
2024-01-05 16:17:23 -07:00
|
|
|
hidden: false,
|
|
|
|
|
path,
|
2024-02-22 15:04:37 -07:00
|
|
|
mime_guess,
|
2024-01-05 16:17:23 -07:00
|
|
|
icon_handle_grid,
|
|
|
|
|
icon_handle_list,
|
2024-02-22 16:17:39 -07:00
|
|
|
thumbnail_res_opt: Some(Err(())),
|
2024-01-10 09:47:47 -07:00
|
|
|
selected: false,
|
|
|
|
|
click_time: None,
|
2024-01-05 16:17:23 -07:00
|
|
|
});
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to read trash items: {}", err);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-06 12:59:36 -07:00
|
|
|
items.sort_by(|a, b| match (a.metadata.is_dir(), b.metadata.is_dir()) {
|
2024-01-05 08:55:18 -07:00
|
|
|
(true, false) => Ordering::Less,
|
|
|
|
|
(false, true) => Ordering::Greater,
|
2024-01-05 09:44:47 -07:00
|
|
|
_ => lexical_sort::natural_lexical_cmp(&a.name, &b.name),
|
2024-01-05 08:55:18 -07:00
|
|
|
});
|
|
|
|
|
items
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-09 15:42:42 -07:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq)]
|
2024-01-05 16:17:23 -07:00
|
|
|
pub enum Location {
|
|
|
|
|
Path(PathBuf),
|
|
|
|
|
Trash,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl Location {
|
2024-02-18 02:44:54 -05:00
|
|
|
pub fn scan(&self, sizes: IconSizes) -> Vec<Item> {
|
2024-01-05 16:17:23 -07:00
|
|
|
match self {
|
2024-02-18 02:44:54 -05:00
|
|
|
Self::Path(path) => scan_path(path, sizes),
|
|
|
|
|
Self::Trash => scan_trash(sizes),
|
2024-01-05 16:17:23 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-09 15:34:48 -07:00
|
|
|
#[derive(Clone, Debug)]
|
2024-01-05 09:36:16 -07:00
|
|
|
pub enum Message {
|
2024-01-10 10:26:30 -07:00
|
|
|
Click(Option<usize>),
|
2024-02-11 00:24:35 -05:00
|
|
|
Config(TabConfig),
|
2024-02-26 12:05:29 -07:00
|
|
|
ContextAction(Action),
|
|
|
|
|
ContextMenu(Option<Point>),
|
|
|
|
|
Drag(Option<Point>),
|
2024-01-29 11:58:36 -07:00
|
|
|
EditLocation(Option<Location>),
|
2024-01-30 10:47:41 -07:00
|
|
|
GoNext,
|
|
|
|
|
GoPrevious,
|
2024-01-09 15:34:48 -07:00
|
|
|
Location(Location),
|
2024-02-06 22:58:29 -05:00
|
|
|
LocationUp,
|
2024-01-10 10:26:30 -07:00
|
|
|
RightClick(usize),
|
2024-02-26 12:05:29 -07:00
|
|
|
Scroll(Viewport),
|
2024-02-22 16:17:39 -07:00
|
|
|
Thumbnail(PathBuf, Result<image::RgbaImage, ()>),
|
2024-02-11 00:24:35 -05:00
|
|
|
ToggleShowHidden,
|
2024-01-05 15:17:38 -07:00
|
|
|
View(View),
|
2024-01-05 09:36:16 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-06 12:59:36 -07:00
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
|
pub enum ItemMetadata {
|
2024-01-30 10:47:41 -07:00
|
|
|
Path {
|
|
|
|
|
metadata: Metadata,
|
|
|
|
|
children: usize,
|
|
|
|
|
},
|
|
|
|
|
Trash {
|
|
|
|
|
metadata: trash::TrashItemMetadata,
|
|
|
|
|
entry: trash::TrashItem,
|
|
|
|
|
},
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
impl ItemMetadata {
|
|
|
|
|
pub fn is_dir(&self) -> bool {
|
|
|
|
|
match self {
|
2024-01-30 10:47:41 -07:00
|
|
|
Self::Path { metadata, .. } => metadata.is_dir(),
|
|
|
|
|
Self::Trash { metadata, .. } => match metadata.size {
|
2024-01-10 08:53:22 -07:00
|
|
|
trash::TrashItemSize::Entries(_) => true,
|
|
|
|
|
trash::TrashItemSize::Bytes(_) => false,
|
|
|
|
|
},
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
#[derive(Clone)]
|
|
|
|
|
pub struct Item {
|
|
|
|
|
pub name: String,
|
2024-01-06 12:59:36 -07:00
|
|
|
pub metadata: ItemMetadata,
|
2024-01-05 09:36:16 -07:00
|
|
|
pub hidden: bool,
|
2024-01-05 14:44:20 -07:00
|
|
|
pub path: PathBuf,
|
2024-02-22 15:04:37 -07:00
|
|
|
pub mime_guess: MimeGuess,
|
2024-01-05 09:36:16 -07:00
|
|
|
pub icon_handle_grid: widget::icon::Handle,
|
|
|
|
|
pub icon_handle_list: widget::icon::Handle,
|
2024-02-22 16:17:39 -07:00
|
|
|
pub thumbnail_res_opt: Option<Result<image::RgbaImage, ()>>,
|
2024-01-10 09:47:47 -07:00
|
|
|
pub selected: bool,
|
|
|
|
|
pub click_time: Option<Instant>,
|
2024-01-05 09:36:16 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-05 14:44:20 -07:00
|
|
|
impl Item {
|
2024-02-18 02:44:54 -05:00
|
|
|
pub fn property_view(&self, core: &Core, sizes: IconSizes) -> Element<crate::app::Message> {
|
2024-02-22 16:17:39 -07:00
|
|
|
let cosmic_theme::Spacing { space_xxxs, .. } = core.system_theme().cosmic().spacing;
|
2024-02-22 15:04:37 -07:00
|
|
|
|
2024-02-22 16:17:39 -07:00
|
|
|
let mut column = widget::column().spacing(space_xxxs);
|
|
|
|
|
|
|
|
|
|
let is_image = if let Some(mime) = self.mime_guess.first() {
|
|
|
|
|
mime.type_() == "image" && self.path.is_file()
|
|
|
|
|
} else {
|
|
|
|
|
false
|
|
|
|
|
};
|
2024-02-22 15:04:37 -07:00
|
|
|
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::row::with_children(vec![
|
|
|
|
|
widget::horizontal_space(Length::Fill).into(),
|
|
|
|
|
if is_image {
|
2024-02-22 21:24:16 -07:00
|
|
|
widget::image::viewer(widget::image::Handle::from_path(&self.path))
|
|
|
|
|
.min_scale(1.0)
|
|
|
|
|
.into()
|
2024-02-22 16:17:39 -07:00
|
|
|
} else {
|
|
|
|
|
widget::icon::icon(self.icon_handle_grid.clone())
|
2024-02-22 21:34:21 -07:00
|
|
|
.content_fit(ContentFit::Contain)
|
2024-02-22 16:17:39 -07:00
|
|
|
.size(sizes.grid())
|
|
|
|
|
.into()
|
|
|
|
|
},
|
|
|
|
|
widget::horizontal_space(Length::Fill).into(),
|
2024-01-05 14:58:50 -07:00
|
|
|
]));
|
2024-01-05 14:44:20 -07:00
|
|
|
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text::heading(self.name.clone()));
|
|
|
|
|
|
2024-02-22 15:04:37 -07:00
|
|
|
if let Some(mime) = self.mime_guess.first() {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!("Type: {}", mime)));
|
2024-02-22 15:04:37 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-05 14:44:20 -07:00
|
|
|
//TODO: translate!
|
2024-01-05 14:45:45 -07:00
|
|
|
//TODO: correct display of folder size?
|
2024-01-06 12:59:36 -07:00
|
|
|
match &self.metadata {
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Path { metadata, children } => {
|
2024-01-06 20:31:00 -07:00
|
|
|
if metadata.is_dir() {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!("Items: {}", children)));
|
2024-01-06 20:31:00 -07:00
|
|
|
} else {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!(
|
|
|
|
|
"Size: {}",
|
|
|
|
|
format_size(metadata.len())
|
|
|
|
|
)));
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
2024-01-05 14:44:20 -07:00
|
|
|
|
2024-02-22 15:04:37 -07:00
|
|
|
if let Ok(time) = metadata.created() {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!(
|
|
|
|
|
"Created: {}",
|
|
|
|
|
chrono::DateTime::<chrono::Local>::from(time).format(TIME_FORMAT)
|
|
|
|
|
)));
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
2024-01-05 14:44:20 -07:00
|
|
|
|
2024-01-06 12:59:36 -07:00
|
|
|
if let Ok(time) = metadata.modified() {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!(
|
|
|
|
|
"Modified: {}",
|
|
|
|
|
chrono::DateTime::<chrono::Local>::from(time).format(TIME_FORMAT)
|
|
|
|
|
)));
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
2024-01-05 14:44:20 -07:00
|
|
|
|
2024-02-22 15:04:37 -07:00
|
|
|
if let Ok(time) = metadata.accessed() {
|
2024-02-22 16:17:39 -07:00
|
|
|
column = column.push(widget::text(format!(
|
|
|
|
|
"Accessed: {}",
|
|
|
|
|
chrono::DateTime::<chrono::Local>::from(time).format(TIME_FORMAT)
|
|
|
|
|
)));
|
2024-01-06 12:59:36 -07:00
|
|
|
}
|
|
|
|
|
}
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Trash { .. } => {
|
2024-01-06 12:59:36 -07:00
|
|
|
//TODO: trash metadata
|
2024-01-05 16:17:23 -07:00
|
|
|
}
|
2024-01-05 14:44:20 -07:00
|
|
|
}
|
|
|
|
|
|
2024-02-22 16:17:39 -07:00
|
|
|
column.into()
|
2024-01-05 14:44:20 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
impl fmt::Debug for Item {
|
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
|
|
|
|
|
f.debug_struct("Item")
|
|
|
|
|
.field("name", &self.name)
|
2024-01-06 12:59:36 -07:00
|
|
|
.field("metadata", &self.metadata)
|
2024-01-05 09:36:16 -07:00
|
|
|
.field("hidden", &self.hidden)
|
2024-01-05 14:44:20 -07:00
|
|
|
.field("path", &self.path)
|
|
|
|
|
// icon_handles
|
2024-01-10 09:47:47 -07:00
|
|
|
.field("selected", &self.selected)
|
|
|
|
|
.field("click_time", &self.click_time)
|
2024-01-05 09:36:16 -07:00
|
|
|
.finish()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Copy, Debug)]
|
|
|
|
|
pub enum View {
|
|
|
|
|
Grid,
|
|
|
|
|
List,
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[derive(Clone, Debug)]
|
|
|
|
|
pub struct Tab {
|
2024-01-05 16:17:23 -07:00
|
|
|
pub location: Location,
|
2024-01-05 09:36:16 -07:00
|
|
|
pub context_menu: Option<Point>,
|
|
|
|
|
pub items_opt: Option<Vec<Item>>,
|
|
|
|
|
pub view: View,
|
2024-02-15 15:03:01 -07:00
|
|
|
pub dialog: Option<DialogKind>,
|
2024-02-26 12:05:29 -07:00
|
|
|
pub drag_opt: Option<Point>,
|
|
|
|
|
pub scroll_opt: Option<Viewport>,
|
2024-01-29 11:58:36 -07:00
|
|
|
pub edit_location: Option<Location>,
|
2024-01-30 10:47:41 -07:00
|
|
|
pub history_i: usize,
|
|
|
|
|
pub history: Vec<Location>,
|
2024-02-10 02:13:13 -05:00
|
|
|
pub config: TabConfig,
|
2024-01-05 09:36:16 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-03 15:27:32 -07:00
|
|
|
impl Tab {
|
2024-02-10 02:13:13 -05:00
|
|
|
pub fn new(location: Location, config: TabConfig) -> Self {
|
2024-01-30 10:47:41 -07:00
|
|
|
let history = vec![location.clone()];
|
2024-01-05 08:55:18 -07:00
|
|
|
Self {
|
2024-01-05 16:17:23 -07:00
|
|
|
location,
|
2024-01-03 15:27:32 -07:00
|
|
|
context_menu: None,
|
2024-01-05 08:55:18 -07:00
|
|
|
items_opt: None,
|
2024-02-26 12:05:29 -07:00
|
|
|
view: View::Grid,
|
2024-02-15 15:03:01 -07:00
|
|
|
dialog: None,
|
2024-02-26 12:05:29 -07:00
|
|
|
drag_opt: None,
|
|
|
|
|
scroll_opt: None,
|
2024-01-29 11:58:36 -07:00
|
|
|
edit_location: None,
|
2024-01-30 10:47:41 -07:00
|
|
|
history_i: 0,
|
|
|
|
|
history,
|
2024-02-10 02:13:13 -05:00
|
|
|
config,
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn title(&self) -> String {
|
|
|
|
|
//TODO: better title
|
2024-01-05 16:17:23 -07:00
|
|
|
match &self.location {
|
|
|
|
|
Location::Path(path) => {
|
|
|
|
|
format!("{}", path.display())
|
|
|
|
|
}
|
|
|
|
|
Location::Trash => {
|
|
|
|
|
fl!("trash")
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
|
2024-02-26 12:05:29 -07:00
|
|
|
fn select_by_drag(&mut self, rect: Rectangle) {
|
|
|
|
|
let items = match &mut self.items_opt {
|
|
|
|
|
Some(some) => some,
|
|
|
|
|
None => return,
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
println!("{:?}", rect);
|
|
|
|
|
let (row, col) = match self.view {
|
|
|
|
|
View::Grid => (0, 0),
|
|
|
|
|
View::List => (0, 0),
|
|
|
|
|
};
|
|
|
|
|
for (i, item) in items.iter_mut().enumerate() {
|
|
|
|
|
item.selected = false;
|
|
|
|
|
//TODO
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-10 10:47:12 -07:00
|
|
|
pub fn update(&mut self, message: Message, modifiers: Modifiers) -> bool {
|
2024-01-03 15:27:32 -07:00
|
|
|
let mut cd = None;
|
2024-01-30 10:47:41 -07:00
|
|
|
let mut history_i_opt = None;
|
2024-01-03 15:27:32 -07:00
|
|
|
match message {
|
2024-01-10 10:26:30 -07:00
|
|
|
Message::Click(click_i_opt) => {
|
2024-01-05 08:55:18 -07:00
|
|
|
if let Some(ref mut items) = self.items_opt {
|
|
|
|
|
for (i, item) in items.iter_mut().enumerate() {
|
2024-01-05 11:18:38 -07:00
|
|
|
if Some(i) == click_i_opt {
|
2024-01-10 09:47:47 -07:00
|
|
|
item.selected = true;
|
2024-01-10 10:26:30 -07:00
|
|
|
if let Some(click_time) = item.click_time {
|
|
|
|
|
if click_time.elapsed() < DOUBLE_CLICK_DURATION {
|
|
|
|
|
match self.location {
|
|
|
|
|
Location::Path(_) => {
|
|
|
|
|
if item.path.is_dir() {
|
|
|
|
|
cd = Some(Location::Path(item.path.clone()));
|
2024-02-15 15:03:01 -07:00
|
|
|
} else if !self.dialog.is_some() {
|
2024-01-10 10:26:30 -07:00
|
|
|
let mut command = open_command(&item.path);
|
|
|
|
|
match command.spawn() {
|
|
|
|
|
Ok(_) => (),
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!(
|
|
|
|
|
"failed to open {:?}: {}",
|
|
|
|
|
item.path,
|
|
|
|
|
err
|
|
|
|
|
);
|
2024-01-10 09:47:47 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-10 10:26:30 -07:00
|
|
|
}
|
|
|
|
|
Location::Trash => {
|
|
|
|
|
//TODO: open properties?
|
2024-01-03 15:41:01 -07:00
|
|
|
}
|
|
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
}
|
2024-01-10 10:29:49 -07:00
|
|
|
//TODO: prevent triple-click and beyond from opening file?
|
|
|
|
|
item.click_time = Some(Instant::now());
|
2024-02-15 15:03:01 -07:00
|
|
|
} else if modifiers.contains(Modifiers::CTRL)
|
2024-02-20 11:58:39 -07:00
|
|
|
&& self.dialog.as_ref().map_or(true, |x| x.multiple())
|
2024-02-15 15:03:01 -07:00
|
|
|
{
|
2024-01-10 10:47:12 -07:00
|
|
|
// Holding control allows multiple selection
|
|
|
|
|
item.click_time = None;
|
2024-01-05 08:55:18 -07:00
|
|
|
} else {
|
2024-01-10 09:47:47 -07:00
|
|
|
item.selected = false;
|
|
|
|
|
item.click_time = None;
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-05 11:18:38 -07:00
|
|
|
self.context_menu = None;
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
2024-02-11 00:24:35 -05:00
|
|
|
Message::Config(config) => {
|
|
|
|
|
self.config = config;
|
|
|
|
|
}
|
2024-02-26 12:05:29 -07:00
|
|
|
Message::ContextAction(action) => {
|
|
|
|
|
// Close context menu
|
|
|
|
|
self.context_menu = None;
|
|
|
|
|
|
|
|
|
|
// TODO: run actions message
|
|
|
|
|
println!("TODO {:?}", action);
|
|
|
|
|
}
|
|
|
|
|
Message::ContextMenu(point_opt) => {
|
|
|
|
|
self.context_menu = point_opt;
|
|
|
|
|
}
|
|
|
|
|
Message::Drag(point_opt) => match point_opt {
|
|
|
|
|
Some(point) => {
|
|
|
|
|
let drag = match self.drag_opt {
|
|
|
|
|
Some(some) => some,
|
|
|
|
|
None => {
|
|
|
|
|
self.drag_opt = Some(point);
|
|
|
|
|
point
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
let min_x = drag.x.min(point.x);
|
|
|
|
|
let max_x = drag.x.max(point.x);
|
|
|
|
|
let min_y = drag.y.min(point.y);
|
|
|
|
|
let max_y = drag.y.max(point.y);
|
|
|
|
|
let offset_y = self
|
|
|
|
|
.scroll_opt
|
|
|
|
|
.map(|x| x.absolute_offset().y)
|
|
|
|
|
.unwrap_or_default();
|
|
|
|
|
let rect = Rectangle::new(
|
|
|
|
|
Point::new(min_x, min_y + offset_y),
|
|
|
|
|
Size::new(max_x - min_x, max_y - min_y),
|
|
|
|
|
);
|
|
|
|
|
self.select_by_drag(rect);
|
|
|
|
|
}
|
|
|
|
|
None => {
|
|
|
|
|
self.drag_opt = None;
|
|
|
|
|
}
|
|
|
|
|
},
|
2024-01-29 11:58:36 -07:00
|
|
|
Message::EditLocation(edit_location) => {
|
|
|
|
|
self.edit_location = edit_location;
|
|
|
|
|
}
|
2024-01-30 10:47:41 -07:00
|
|
|
Message::GoNext => {
|
|
|
|
|
if let Some(history_i) = self.history_i.checked_add(1) {
|
|
|
|
|
if let Some(location) = self.history.get(history_i) {
|
|
|
|
|
cd = Some(location.clone());
|
|
|
|
|
history_i_opt = Some(history_i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Message::GoPrevious => {
|
|
|
|
|
if let Some(history_i) = self.history_i.checked_sub(1) {
|
|
|
|
|
if let Some(location) = self.history.get(history_i) {
|
|
|
|
|
cd = Some(location.clone());
|
|
|
|
|
history_i_opt = Some(history_i);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-09 15:34:48 -07:00
|
|
|
Message::Location(location) => {
|
|
|
|
|
cd = Some(location);
|
2024-01-03 17:04:08 -07:00
|
|
|
}
|
2024-02-06 22:58:29 -05:00
|
|
|
Message::LocationUp => {
|
|
|
|
|
// Sets location to the path's parent
|
|
|
|
|
// Does nothing if path is root or location is Trash
|
|
|
|
|
if let Location::Path(ref path) = self.location {
|
2024-02-07 21:11:57 -05:00
|
|
|
if let Some(parent) = path.parent() {
|
|
|
|
|
cd = Some(Location::Path(parent.to_owned()));
|
2024-02-06 22:58:29 -05:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-10 10:26:30 -07:00
|
|
|
Message::RightClick(click_i) => {
|
|
|
|
|
if let Some(ref mut items) = self.items_opt {
|
|
|
|
|
if !items.get(click_i).map_or(false, |x| x.selected) {
|
|
|
|
|
// If item not selected, clear selection on other items
|
|
|
|
|
for (i, item) in items.iter_mut().enumerate() {
|
2024-01-10 10:47:12 -07:00
|
|
|
if i == click_i {
|
|
|
|
|
item.selected = true;
|
2024-02-15 15:03:01 -07:00
|
|
|
} else if modifiers.contains(Modifiers::CTRL)
|
2024-02-20 11:58:39 -07:00
|
|
|
&& self.dialog.as_ref().map_or(true, |x| x.multiple())
|
2024-02-15 15:03:01 -07:00
|
|
|
{
|
2024-01-10 10:47:12 -07:00
|
|
|
// Holding control allows multiple selection
|
|
|
|
|
} else {
|
|
|
|
|
item.selected = false;
|
|
|
|
|
}
|
|
|
|
|
item.click_time = None;
|
2024-01-10 10:26:30 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-26 12:05:29 -07:00
|
|
|
Message::Scroll(viewport) => {
|
|
|
|
|
self.scroll_opt = Some(viewport);
|
|
|
|
|
}
|
2024-02-22 16:17:39 -07:00
|
|
|
Message::Thumbnail(path, thumbnail_res) => {
|
|
|
|
|
if let Some(ref mut items) = self.items_opt {
|
|
|
|
|
for item in items.iter_mut() {
|
|
|
|
|
if item.path == path {
|
|
|
|
|
if let Ok(thumbnail) = &thumbnail_res {
|
|
|
|
|
//TODO: pass handles already generated to avoid blocking main thread
|
|
|
|
|
let handle = widget::icon::from_raster_pixels(
|
|
|
|
|
thumbnail.width(),
|
|
|
|
|
thumbnail.height(),
|
|
|
|
|
thumbnail.as_raw().clone(),
|
|
|
|
|
);
|
|
|
|
|
item.icon_handle_grid = handle.clone();
|
|
|
|
|
item.icon_handle_list = handle;
|
|
|
|
|
}
|
|
|
|
|
item.thumbnail_res_opt = Some(thumbnail_res);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-11 00:24:35 -05:00
|
|
|
Message::ToggleShowHidden => self.config.show_hidden = !self.config.show_hidden,
|
2024-01-05 15:17:38 -07:00
|
|
|
Message::View(view) => {
|
|
|
|
|
self.view = view;
|
|
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
2024-02-20 11:58:39 -07:00
|
|
|
if let Some(location) = cd {
|
2024-01-09 15:42:42 -07:00
|
|
|
if location != self.location {
|
2024-01-30 10:47:41 -07:00
|
|
|
self.location = location.clone();
|
2024-01-09 15:42:42 -07:00
|
|
|
self.items_opt = None;
|
2024-01-29 11:58:36 -07:00
|
|
|
self.edit_location = None;
|
2024-01-30 10:47:41 -07:00
|
|
|
if let Some(history_i) = history_i_opt {
|
|
|
|
|
// Navigating in history
|
|
|
|
|
self.history_i = history_i;
|
|
|
|
|
} else {
|
|
|
|
|
// Truncate history to remove next entries
|
|
|
|
|
self.history.truncate(self.history_i + 1);
|
|
|
|
|
|
|
|
|
|
// Push to the front of history
|
|
|
|
|
self.history_i = self.history.len();
|
|
|
|
|
self.history.push(location);
|
|
|
|
|
}
|
2024-01-09 15:42:42 -07:00
|
|
|
true
|
|
|
|
|
} else {
|
|
|
|
|
false
|
|
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
} else {
|
|
|
|
|
false
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-30 10:47:41 -07:00
|
|
|
pub fn location_view(&self, core: &Core) -> Element<Message> {
|
2024-01-29 11:58:36 -07:00
|
|
|
let cosmic_theme::Spacing {
|
|
|
|
|
space_xxxs,
|
|
|
|
|
space_xxs,
|
2024-01-30 10:47:41 -07:00
|
|
|
space_s,
|
2024-01-29 11:58:36 -07:00
|
|
|
..
|
|
|
|
|
} = core.system_theme().cosmic().spacing;
|
|
|
|
|
|
2024-01-30 10:47:41 -07:00
|
|
|
let mut row = widget::row::with_capacity(5).align_items(Alignment::Center);
|
|
|
|
|
|
|
|
|
|
let mut prev_button =
|
|
|
|
|
widget::button(widget::icon::from_name("go-previous-symbolic").size(16))
|
|
|
|
|
.padding(space_xxs)
|
|
|
|
|
.style(theme::Button::Icon);
|
|
|
|
|
if self.history_i > 0 && !self.history.is_empty() {
|
|
|
|
|
prev_button = prev_button.on_press(Message::GoPrevious);
|
|
|
|
|
}
|
|
|
|
|
row = row.push(prev_button);
|
|
|
|
|
|
|
|
|
|
let mut next_button = widget::button(widget::icon::from_name("go-next-symbolic").size(16))
|
|
|
|
|
.padding(space_xxs)
|
|
|
|
|
.style(theme::Button::Icon);
|
|
|
|
|
if self.history_i + 1 < self.history.len() {
|
|
|
|
|
next_button = next_button.on_press(Message::GoNext);
|
|
|
|
|
}
|
|
|
|
|
row = row.push(next_button);
|
|
|
|
|
|
|
|
|
|
row = row.push(widget::horizontal_space(Length::Fixed(space_s.into())));
|
|
|
|
|
|
2024-01-29 11:58:36 -07:00
|
|
|
if let Some(location) = &self.edit_location {
|
|
|
|
|
match location {
|
|
|
|
|
Location::Path(path) => {
|
2024-01-30 10:47:41 -07:00
|
|
|
row = row.push(
|
2024-01-29 11:58:36 -07:00
|
|
|
widget::button(widget::icon::from_name("window-close-symbolic").size(16))
|
|
|
|
|
.on_press(Message::EditLocation(None))
|
|
|
|
|
.padding(space_xxs)
|
2024-01-30 10:47:41 -07:00
|
|
|
.style(theme::Button::Icon),
|
|
|
|
|
);
|
|
|
|
|
row = row.push(
|
2024-01-29 11:58:36 -07:00
|
|
|
widget::text_input("", path.to_string_lossy())
|
|
|
|
|
.on_input(|input| {
|
|
|
|
|
Message::EditLocation(Some(Location::Path(PathBuf::from(input))))
|
|
|
|
|
})
|
2024-01-30 10:47:41 -07:00
|
|
|
.on_submit(Message::Location(location.clone())),
|
|
|
|
|
);
|
|
|
|
|
return row.into();
|
2024-01-29 11:58:36 -07:00
|
|
|
}
|
|
|
|
|
_ => {
|
|
|
|
|
//TODO: allow editing other locations
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-30 11:26:23 -07:00
|
|
|
} else if let Location::Path(_) = &self.location {
|
2024-01-30 10:47:41 -07:00
|
|
|
row = row.push(
|
|
|
|
|
widget::button(widget::icon::from_name("edit-symbolic").size(16))
|
|
|
|
|
.on_press(Message::EditLocation(Some(self.location.clone())))
|
|
|
|
|
.padding(space_xxs)
|
|
|
|
|
.style(theme::Button::Icon),
|
|
|
|
|
);
|
2024-01-29 11:58:36 -07:00
|
|
|
}
|
2024-01-10 12:57:30 -07:00
|
|
|
|
|
|
|
|
let mut children: Vec<Element<_>> = Vec::new();
|
|
|
|
|
match &self.location {
|
|
|
|
|
Location::Path(path) => {
|
|
|
|
|
let home_dir = crate::home_dir();
|
|
|
|
|
for ancestor in path.ancestors() {
|
|
|
|
|
let ancestor = ancestor.to_path_buf();
|
|
|
|
|
let mut found_home = false;
|
|
|
|
|
let mut row = widget::row::with_capacity(2)
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.spacing(space_xxxs);
|
2024-01-29 11:16:19 -07:00
|
|
|
|
|
|
|
|
let name = match ancestor.file_name() {
|
2024-01-10 12:57:30 -07:00
|
|
|
Some(name) => {
|
|
|
|
|
if ancestor == home_dir {
|
|
|
|
|
row = row.push(
|
|
|
|
|
widget::icon::icon(folder_icon_symbolic(&ancestor, 16))
|
|
|
|
|
.size(16),
|
|
|
|
|
);
|
|
|
|
|
found_home = true;
|
2024-01-29 11:16:19 -07:00
|
|
|
fl!("home")
|
2024-01-29 11:09:05 -07:00
|
|
|
} else {
|
2024-01-29 11:16:19 -07:00
|
|
|
name.to_string_lossy().to_string()
|
2024-01-10 12:57:30 -07:00
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
None => {
|
|
|
|
|
row = row.push(
|
|
|
|
|
widget::icon::from_name("drive-harddisk-system-symbolic")
|
|
|
|
|
.size(16)
|
|
|
|
|
.icon(),
|
|
|
|
|
);
|
2024-01-29 11:16:19 -07:00
|
|
|
fl!("filesystem")
|
2024-01-10 12:57:30 -07:00
|
|
|
}
|
2024-01-29 11:16:19 -07:00
|
|
|
};
|
2024-01-10 12:57:30 -07:00
|
|
|
|
2024-01-29 11:16:19 -07:00
|
|
|
if children.is_empty() {
|
|
|
|
|
row = row.push(widget::text::heading(name));
|
|
|
|
|
} else {
|
2024-01-29 11:09:05 -07:00
|
|
|
children.push(
|
|
|
|
|
widget::icon::from_name("go-next-symbolic")
|
|
|
|
|
.size(16)
|
|
|
|
|
.icon()
|
|
|
|
|
.into(),
|
|
|
|
|
);
|
2024-01-29 11:16:19 -07:00
|
|
|
row = row.push(widget::text(name));
|
2024-01-10 12:57:30 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
children.push(
|
|
|
|
|
widget::button(row)
|
|
|
|
|
.padding(space_xxxs)
|
|
|
|
|
.on_press(Message::Location(Location::Path(ancestor)))
|
2024-01-29 11:09:05 -07:00
|
|
|
.style(theme::Button::Link)
|
2024-01-10 12:57:30 -07:00
|
|
|
.into(),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
if found_home {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
children.reverse();
|
|
|
|
|
}
|
|
|
|
|
Location::Trash => {
|
|
|
|
|
let mut row = widget::row::with_capacity(2)
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.spacing(space_xxxs);
|
|
|
|
|
row = row.push(widget::icon::icon(trash_icon_symbolic(16)).size(16));
|
2024-01-29 11:16:19 -07:00
|
|
|
row = row.push(widget::text::heading(fl!("trash")));
|
2024-01-10 12:57:30 -07:00
|
|
|
|
|
|
|
|
children.push(
|
|
|
|
|
widget::button(row)
|
|
|
|
|
.padding(space_xxxs)
|
|
|
|
|
.on_press(Message::Location(Location::Trash))
|
|
|
|
|
.style(theme::Button::Text)
|
|
|
|
|
.into(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-29 11:58:36 -07:00
|
|
|
|
2024-01-30 10:47:41 -07:00
|
|
|
for child in children {
|
|
|
|
|
row = row.push(child);
|
|
|
|
|
}
|
|
|
|
|
row.into()
|
2024-01-10 12:57:30 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
pub fn empty_view(&self, has_hidden: bool, core: &Core) -> Element<Message> {
|
2024-01-03 15:27:32 -07:00
|
|
|
let cosmic_theme::Spacing { space_xxs, .. } = core.system_theme().cosmic().spacing;
|
|
|
|
|
|
2024-02-26 12:05:29 -07:00
|
|
|
widget::column::with_children(vec![widget::container(
|
|
|
|
|
widget::column::with_children(vec![
|
|
|
|
|
widget::icon::from_name("folder-symbolic")
|
|
|
|
|
.size(64)
|
|
|
|
|
.icon()
|
2024-01-05 09:36:16 -07:00
|
|
|
.into(),
|
2024-02-26 12:05:29 -07:00
|
|
|
widget::text(if has_hidden {
|
|
|
|
|
fl!("empty-folder-hidden")
|
|
|
|
|
} else {
|
|
|
|
|
fl!("empty-folder")
|
|
|
|
|
})
|
|
|
|
|
.into(),
|
|
|
|
|
])
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.spacing(space_xxs),
|
|
|
|
|
)
|
|
|
|
|
.align_x(Horizontal::Center)
|
|
|
|
|
.align_y(Vertical::Center)
|
|
|
|
|
.width(Length::Fill)
|
|
|
|
|
.height(Length::Fill)
|
|
|
|
|
.into()])
|
2024-01-05 09:36:16 -07:00
|
|
|
.into()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn grid_view(&self, core: &Core) -> Element<Message> {
|
|
|
|
|
let cosmic_theme::Spacing { space_xxs, .. } = core.system_theme().cosmic().spacing;
|
|
|
|
|
|
2024-01-29 11:25:43 -07:00
|
|
|
//TODO: get from config
|
|
|
|
|
let item_width = Length::Fixed(96.0);
|
|
|
|
|
let item_height = Length::Fixed(116.0);
|
2024-02-18 02:44:54 -05:00
|
|
|
let TabConfig {
|
|
|
|
|
show_hidden,
|
|
|
|
|
icon_sizes,
|
|
|
|
|
} = self.config;
|
2024-01-29 11:25:43 -07:00
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
let mut children: Vec<Element<_>> = Vec::new();
|
2024-01-05 08:55:18 -07:00
|
|
|
if let Some(ref items) = self.items_opt {
|
|
|
|
|
let mut count = 0;
|
|
|
|
|
let mut hidden = 0;
|
|
|
|
|
for (i, item) in items.iter().enumerate() {
|
2024-02-11 00:24:35 -05:00
|
|
|
if !show_hidden && item.hidden {
|
2024-01-05 08:55:18 -07:00
|
|
|
hidden += 1;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-05 15:10:46 -07:00
|
|
|
let button = widget::button(
|
|
|
|
|
widget::column::with_children(vec![
|
|
|
|
|
widget::icon::icon(item.icon_handle_grid.clone())
|
2024-02-22 21:34:21 -07:00
|
|
|
.content_fit(ContentFit::Contain)
|
2024-02-18 02:44:54 -05:00
|
|
|
.size(icon_sizes.grid())
|
2024-01-05 15:10:46 -07:00
|
|
|
.into(),
|
|
|
|
|
widget::text(item.name.clone()).into(),
|
|
|
|
|
])
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.spacing(space_xxs)
|
2024-01-29 11:25:43 -07:00
|
|
|
.height(item_height)
|
|
|
|
|
.width(item_width),
|
2024-01-05 15:10:46 -07:00
|
|
|
)
|
2024-02-26 12:05:29 -07:00
|
|
|
.padding(0)
|
2024-01-10 09:47:47 -07:00
|
|
|
.style(button_style(item.selected))
|
2024-01-10 10:26:30 -07:00
|
|
|
.on_press(Message::Click(Some(i)));
|
2024-01-05 15:10:46 -07:00
|
|
|
if self.context_menu.is_some() {
|
|
|
|
|
children.push(button.into());
|
|
|
|
|
} else {
|
|
|
|
|
children.push(
|
2024-02-26 12:05:29 -07:00
|
|
|
mouse_area::MouseArea::new(button)
|
2024-01-10 10:26:30 -07:00
|
|
|
.on_right_press_no_capture(move |_point_opt| Message::RightClick(i))
|
2024-01-05 15:10:46 -07:00
|
|
|
.into(),
|
|
|
|
|
);
|
|
|
|
|
}
|
2024-01-05 09:36:16 -07:00
|
|
|
count += 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if count == 0 {
|
|
|
|
|
return self.empty_view(hidden > 0, core);
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-02-26 12:05:29 -07:00
|
|
|
widget::scrollable(widget::flex_row(children))
|
|
|
|
|
.on_scroll(Message::Scroll)
|
|
|
|
|
.width(Length::Fill)
|
|
|
|
|
.into()
|
2024-01-05 09:36:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn list_view(&self, core: &Core) -> Element<Message> {
|
|
|
|
|
let cosmic_theme::Spacing { space_xxs, .. } = core.system_theme().cosmic().spacing;
|
|
|
|
|
|
2024-01-29 10:39:12 -07:00
|
|
|
//TODO: make adaptive?
|
2024-02-13 12:49:37 -07:00
|
|
|
let modified_width = Length::Fixed(200.0);
|
|
|
|
|
let size_width = Length::Fixed(100.0);
|
2024-01-29 10:39:12 -07:00
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
let mut children: Vec<Element<_>> = Vec::new();
|
2024-01-05 15:32:42 -07:00
|
|
|
|
|
|
|
|
children.push(
|
|
|
|
|
widget::row::with_children(vec![
|
2024-01-29 10:39:12 -07:00
|
|
|
widget::text::heading(fl!("name"))
|
|
|
|
|
.width(Length::Fill)
|
|
|
|
|
.into(),
|
|
|
|
|
//TODO: do not show modified column when in the trash
|
|
|
|
|
widget::text::heading(fl!("modified"))
|
2024-02-13 12:49:37 -07:00
|
|
|
.width(modified_width)
|
2024-01-29 10:39:12 -07:00
|
|
|
.into(),
|
2024-02-15 15:03:01 -07:00
|
|
|
widget::text::heading(fl!("size")).width(size_width).into(),
|
2024-01-05 15:32:42 -07:00
|
|
|
])
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.padding(space_xxs)
|
|
|
|
|
.spacing(space_xxs)
|
|
|
|
|
.into(),
|
|
|
|
|
);
|
|
|
|
|
|
2024-01-29 11:09:05 -07:00
|
|
|
children.push(horizontal_rule(1).into());
|
2024-01-05 15:32:42 -07:00
|
|
|
|
2024-01-05 09:36:16 -07:00
|
|
|
if let Some(ref items) = self.items_opt {
|
|
|
|
|
let mut count = 0;
|
|
|
|
|
let mut hidden = 0;
|
2024-02-18 02:44:54 -05:00
|
|
|
let TabConfig {
|
|
|
|
|
show_hidden,
|
|
|
|
|
icon_sizes,
|
|
|
|
|
} = self.config;
|
2024-01-05 09:36:16 -07:00
|
|
|
for (i, item) in items.iter().enumerate() {
|
2024-02-11 00:24:35 -05:00
|
|
|
if !show_hidden && item.hidden {
|
2024-01-05 09:36:16 -07:00
|
|
|
hidden += 1;
|
|
|
|
|
continue;
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-29 11:09:05 -07:00
|
|
|
if count > 0 {
|
|
|
|
|
children.push(horizontal_rule(1).into());
|
|
|
|
|
}
|
|
|
|
|
|
2024-01-29 10:39:12 -07:00
|
|
|
let modified_text = match &item.metadata {
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Path { metadata, .. } => match metadata.modified() {
|
2024-01-29 10:39:12 -07:00
|
|
|
Ok(time) => chrono::DateTime::<chrono::Local>::from(time)
|
2024-02-22 15:04:37 -07:00
|
|
|
.format(TIME_FORMAT)
|
2024-01-29 10:39:12 -07:00
|
|
|
.to_string(),
|
|
|
|
|
Err(_) => String::new(),
|
|
|
|
|
},
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Trash { .. } => String::new(),
|
2024-01-29 10:39:12 -07:00
|
|
|
};
|
|
|
|
|
|
|
|
|
|
let size_text = match &item.metadata {
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Path { metadata, children } => {
|
2024-01-29 10:39:12 -07:00
|
|
|
if metadata.is_dir() {
|
|
|
|
|
format!("{} items", children)
|
|
|
|
|
} else {
|
|
|
|
|
format_size(metadata.len())
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-30 10:47:41 -07:00
|
|
|
ItemMetadata::Trash { metadata, .. } => match metadata.size {
|
2024-01-29 10:39:12 -07:00
|
|
|
trash::TrashItemSize::Entries(entries) => {
|
|
|
|
|
//TODO: translate
|
|
|
|
|
if entries == 1 {
|
|
|
|
|
format!("{} item", entries)
|
|
|
|
|
} else {
|
|
|
|
|
format!("{} items", entries)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
trash::TrashItemSize::Bytes(bytes) => format_size(bytes),
|
|
|
|
|
},
|
|
|
|
|
};
|
|
|
|
|
|
2024-01-05 15:32:42 -07:00
|
|
|
//TODO: align columns
|
2024-01-05 15:10:46 -07:00
|
|
|
let button = widget::button(
|
|
|
|
|
widget::row::with_children(vec![
|
2024-02-22 21:46:25 -07:00
|
|
|
widget::icon::icon(item.icon_handle_list.clone())
|
|
|
|
|
.content_fit(ContentFit::Contain)
|
|
|
|
|
.size(icon_sizes.list())
|
|
|
|
|
.into(),
|
2024-01-29 10:39:12 -07:00
|
|
|
widget::text(item.name.clone()).width(Length::Fill).into(),
|
2024-02-13 12:49:37 -07:00
|
|
|
widget::text(modified_text).width(modified_width).into(),
|
|
|
|
|
widget::text(size_text).width(size_width).into(),
|
2024-01-05 15:10:46 -07:00
|
|
|
])
|
|
|
|
|
.align_items(Alignment::Center)
|
|
|
|
|
.spacing(space_xxs),
|
|
|
|
|
)
|
2024-02-22 21:46:25 -07:00
|
|
|
.padding(space_xxs)
|
2024-01-10 09:47:47 -07:00
|
|
|
.style(button_style(item.selected))
|
2024-01-10 10:26:30 -07:00
|
|
|
.on_press(Message::Click(Some(i)));
|
2024-01-05 15:10:46 -07:00
|
|
|
if self.context_menu.is_some() {
|
|
|
|
|
children.push(button.into());
|
|
|
|
|
} else {
|
|
|
|
|
children.push(
|
2024-02-26 12:05:29 -07:00
|
|
|
mouse_area::MouseArea::new(button)
|
2024-01-10 10:26:30 -07:00
|
|
|
.on_right_press_no_capture(move |_point_opt| Message::RightClick(i))
|
2024-01-05 15:10:46 -07:00
|
|
|
.into(),
|
|
|
|
|
);
|
|
|
|
|
}
|
2024-01-05 08:55:18 -07:00
|
|
|
count += 1;
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
|
|
|
|
|
2024-01-05 08:55:18 -07:00
|
|
|
if count == 0 {
|
2024-01-05 09:36:16 -07:00
|
|
|
return self.empty_view(hidden > 0, core);
|
2024-01-05 08:55:18 -07:00
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
2024-02-01 15:55:52 -07:00
|
|
|
|
2024-02-26 12:05:29 -07:00
|
|
|
widget::scrollable(
|
|
|
|
|
widget::column::with_children(children)
|
|
|
|
|
// Hack to make room for scroll bar
|
|
|
|
|
.padding([0, space_xxs, 0, 0]),
|
|
|
|
|
)
|
|
|
|
|
.on_scroll(Message::Scroll)
|
|
|
|
|
.width(Length::Fill)
|
2024-01-29 10:39:12 -07:00
|
|
|
.into()
|
2024-01-05 09:36:16 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
pub fn view(&self, core: &Core) -> Element<Message> {
|
2024-02-26 12:05:29 -07:00
|
|
|
let location_view = self.location_view(core);
|
|
|
|
|
let item_view = match self.view {
|
2024-01-05 16:17:23 -07:00
|
|
|
View::Grid => self.grid_view(core),
|
|
|
|
|
View::List => self.list_view(core),
|
2024-02-26 12:05:29 -07:00
|
|
|
};
|
|
|
|
|
let mut mouse_area =
|
|
|
|
|
mouse_area::MouseArea::new(widget::container(item_view).height(Length::Fill))
|
|
|
|
|
.on_drag(move |point_opt| Message::Drag(point_opt))
|
|
|
|
|
.on_press(move |_point_opt| Message::Click(None))
|
|
|
|
|
.on_release(move |point_opt| Message::Drag(None))
|
|
|
|
|
.on_back_press(move |_point_opt| Message::GoPrevious)
|
|
|
|
|
.on_forward_press(move |_point_opt| Message::GoNext)
|
|
|
|
|
.show_drag_box(true);
|
|
|
|
|
if self.context_menu.is_some() {
|
|
|
|
|
mouse_area = mouse_area.on_right_press(move |_point_opt| Message::ContextMenu(None));
|
|
|
|
|
} else {
|
|
|
|
|
mouse_area =
|
|
|
|
|
mouse_area.on_right_press(move |point_opt| Message::ContextMenu(point_opt));
|
|
|
|
|
}
|
|
|
|
|
let mut popover = widget::popover(mouse_area, menu::context_menu(&self));
|
|
|
|
|
match self.context_menu {
|
|
|
|
|
Some(point) => {
|
|
|
|
|
let rounded = Point::new(point.x.round(), point.y.round());
|
|
|
|
|
popover = popover.position(rounded);
|
|
|
|
|
}
|
|
|
|
|
None => {
|
|
|
|
|
popover = popover.show_popup(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
widget::container(widget::column::with_children(vec![
|
|
|
|
|
location_view,
|
|
|
|
|
popover.into(),
|
|
|
|
|
]))
|
2024-01-05 11:18:38 -07:00
|
|
|
.height(Length::Fill)
|
2024-01-05 09:36:16 -07:00
|
|
|
.width(Length::Fill)
|
|
|
|
|
.into()
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
2024-02-22 16:17:39 -07:00
|
|
|
|
|
|
|
|
pub fn subscription(&self) -> Subscription<Message> {
|
|
|
|
|
if let Some(items) = &self.items_opt {
|
|
|
|
|
//TODO: how many thumbnail loads should be in flight at once?
|
|
|
|
|
let jobs = 8;
|
|
|
|
|
let mut subscriptions = Vec::with_capacity(jobs);
|
|
|
|
|
for item in items.iter() {
|
|
|
|
|
match item.thumbnail_res_opt {
|
|
|
|
|
Some(_) => continue,
|
|
|
|
|
None => {
|
|
|
|
|
let path = item.path.clone();
|
|
|
|
|
subscriptions.push(subscription::channel(
|
|
|
|
|
path.clone(),
|
|
|
|
|
1,
|
|
|
|
|
|mut output| async move {
|
|
|
|
|
let (path, thumbnail_res) =
|
|
|
|
|
tokio::task::spawn_blocking(move || {
|
|
|
|
|
let start = std::time::Instant::now();
|
|
|
|
|
let thumbnail_res = match image::io::Reader::open(&path) {
|
|
|
|
|
Ok(reader) => match reader.decode() {
|
|
|
|
|
Ok(image) => {
|
|
|
|
|
//TODO: configurable thumbnail size
|
2024-02-22 21:24:16 -07:00
|
|
|
let thumbnail = image.thumbnail(64, 64);
|
2024-02-22 16:17:39 -07:00
|
|
|
Ok(thumbnail.to_rgba8())
|
|
|
|
|
}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!(
|
|
|
|
|
"failed to decode {:?}: {}",
|
|
|
|
|
path,
|
|
|
|
|
err
|
|
|
|
|
);
|
|
|
|
|
Err(())
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!("failed to read {:?}: {}", path, err);
|
|
|
|
|
Err(())
|
|
|
|
|
}
|
|
|
|
|
};
|
|
|
|
|
log::info!(
|
|
|
|
|
"thumbnailed {:?} in {:?}",
|
|
|
|
|
path,
|
|
|
|
|
start.elapsed()
|
|
|
|
|
);
|
|
|
|
|
(path, thumbnail_res)
|
|
|
|
|
})
|
|
|
|
|
.await
|
|
|
|
|
.unwrap();
|
|
|
|
|
|
|
|
|
|
match output
|
|
|
|
|
.send(Message::Thumbnail(path.clone(), thumbnail_res))
|
|
|
|
|
.await
|
|
|
|
|
{
|
|
|
|
|
Ok(()) => {}
|
|
|
|
|
Err(err) => {
|
|
|
|
|
log::warn!(
|
|
|
|
|
"failed to send thumbnail for {:?}: {}",
|
|
|
|
|
path,
|
|
|
|
|
err
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
//TODO: how to properly kill this task?
|
|
|
|
|
loop {
|
|
|
|
|
tokio::time::sleep(std::time::Duration::new(1, 0)).await;
|
|
|
|
|
}
|
|
|
|
|
},
|
|
|
|
|
));
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if subscriptions.len() >= jobs {
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
Subscription::batch(subscriptions)
|
|
|
|
|
} else {
|
|
|
|
|
Subscription::none()
|
|
|
|
|
}
|
|
|
|
|
}
|
2024-01-03 15:27:32 -07:00
|
|
|
}
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
|
mod tests {
|
2024-02-05 20:58:17 -05:00
|
|
|
use std::{io, path::PathBuf};
|
2024-02-01 16:29:10 +00:00
|
|
|
|
2024-02-03 02:31:08 -05:00
|
|
|
use cosmic::iced_runtime::keyboard::Modifiers;
|
2024-02-05 20:58:17 -05:00
|
|
|
use log::{debug, trace};
|
|
|
|
|
use tempfile::TempDir;
|
2024-02-01 16:29:10 +00:00
|
|
|
use test_log::test;
|
|
|
|
|
|
2024-02-03 02:31:08 -05:00
|
|
|
use super::{scan_path, Item, Location, Message, Tab};
|
2024-02-10 02:13:13 -05:00
|
|
|
use crate::{
|
|
|
|
|
app::test_utils::{
|
|
|
|
|
assert_eq_tab_path, assert_eq_tab_path_contents, empty_fs, eq_path_item, filter_dirs,
|
|
|
|
|
read_dir_sorted, simple_fs, sort_files, tab_click_new, NAME_LEN, NUM_DIRS, NUM_FILES,
|
|
|
|
|
NUM_HIDDEN, NUM_NESTED,
|
|
|
|
|
},
|
2024-02-18 02:44:54 -05:00
|
|
|
config::{IconSizes, TabConfig},
|
2024-02-01 16:29:10 +00:00
|
|
|
};
|
|
|
|
|
|
2024-02-04 07:28:51 -05:00
|
|
|
// Boilerplate for tab tests. Checks if simulated clicks selected items.
|
|
|
|
|
fn tab_selects_item(
|
|
|
|
|
clicks: &[usize],
|
|
|
|
|
modifiers: Modifiers,
|
|
|
|
|
expected_selected: &[bool],
|
|
|
|
|
) -> io::Result<()> {
|
|
|
|
|
let (_fs, mut tab) = tab_click_new(NUM_FILES, NUM_NESTED, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
|
|
|
|
|
|
|
|
|
// Simulate clicks by triggering Message::Click
|
|
|
|
|
for &click in clicks {
|
|
|
|
|
debug!("Emitting Message::Click(Some({click})) with modifiers: {modifiers:?}");
|
|
|
|
|
tab.update(Message::Click(Some(click)), modifiers);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
let items = tab
|
|
|
|
|
.items_opt
|
|
|
|
|
.as_deref()
|
|
|
|
|
.expect("tab should be populated with items");
|
|
|
|
|
|
|
|
|
|
for (i, (&expected, actual)) in expected_selected.into_iter().zip(items).enumerate() {
|
|
|
|
|
assert_eq!(
|
|
|
|
|
expected,
|
|
|
|
|
actual.selected,
|
|
|
|
|
"expected index {i} to be {}",
|
|
|
|
|
if expected {
|
|
|
|
|
"selected but it was deselected"
|
|
|
|
|
} else {
|
|
|
|
|
"deselected but it was selected"
|
|
|
|
|
}
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-05 20:58:17 -05:00
|
|
|
fn tab_history() -> io::Result<(TempDir, Tab, Vec<PathBuf>)> {
|
|
|
|
|
let fs = simple_fs(NUM_FILES, NUM_NESTED, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
|
|
|
|
let path = fs.path();
|
2024-02-10 02:13:13 -05:00
|
|
|
let mut tab = Tab::new(Location::Path(path.into()), TabConfig::default());
|
2024-02-05 20:58:17 -05:00
|
|
|
|
|
|
|
|
// All directories (simple_fs only produces one nested layer)
|
|
|
|
|
let dirs: Vec<PathBuf> = filter_dirs(path)?
|
|
|
|
|
.flat_map(|dir| {
|
|
|
|
|
filter_dirs(&dir).map(|nested_dirs| std::iter::once(dir).chain(nested_dirs))
|
|
|
|
|
})
|
|
|
|
|
.flatten()
|
|
|
|
|
.collect();
|
|
|
|
|
assert!(
|
|
|
|
|
dirs.len() == NUM_DIRS + NUM_DIRS * NUM_NESTED,
|
|
|
|
|
"Sanity check: Have {} dirs instead of {}",
|
|
|
|
|
dirs.len(),
|
|
|
|
|
NUM_DIRS + NUM_DIRS * NUM_NESTED
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
debug!("Building history by emitting Message::Location");
|
|
|
|
|
for dir in &dirs {
|
|
|
|
|
debug!(
|
|
|
|
|
"Emitting Message::Location(Location::Path(\"{}\"))",
|
|
|
|
|
dir.display()
|
|
|
|
|
);
|
|
|
|
|
tab.update(
|
|
|
|
|
Message::Location(Location::Path(dir.clone())),
|
|
|
|
|
Modifiers::empty(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
trace!("Tab history: {:?}", tab.history);
|
|
|
|
|
|
|
|
|
|
Ok((fs, tab, dirs))
|
|
|
|
|
}
|
|
|
|
|
|
2024-02-01 16:29:10 +00:00
|
|
|
#[test]
|
|
|
|
|
fn scan_path_succeeds_on_valid_path() -> io::Result<()> {
|
2024-02-01 23:31:50 -05:00
|
|
|
let fs = simple_fs(NUM_FILES, NUM_HIDDEN, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
2024-02-01 16:29:10 +00:00
|
|
|
let path = fs.path();
|
|
|
|
|
|
2024-02-04 07:28:51 -05:00
|
|
|
// Read directory entries and sort as cosmic-files does
|
|
|
|
|
let entries = read_dir_sorted(path)?;
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
debug!("Calling scan_path(\"{}\")", path.display());
|
2024-02-18 02:44:54 -05:00
|
|
|
let actual = scan_path(&path.to_owned(), IconSizes::default());
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
// scan_path shouldn't skip any entries
|
|
|
|
|
assert_eq!(entries.len(), actual.len());
|
|
|
|
|
|
|
|
|
|
// Correct files should be scanned
|
|
|
|
|
assert!(entries
|
|
|
|
|
.into_iter()
|
|
|
|
|
.zip(actual.into_iter())
|
|
|
|
|
.all(|(path, item)| eq_path_item(&path, &item)));
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn scan_path_returns_empty_vec_for_invalid_path() -> io::Result<()> {
|
2024-02-01 23:31:50 -05:00
|
|
|
let fs = simple_fs(NUM_FILES, NUM_NESTED, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
2024-02-01 16:29:10 +00:00
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
// A nonexisting path within the temp dir
|
|
|
|
|
let invalid_path = path.join("ferris");
|
|
|
|
|
assert!(!invalid_path.exists());
|
|
|
|
|
|
|
|
|
|
debug!("Calling scan_path(\"{}\")", invalid_path.display());
|
2024-02-18 02:44:54 -05:00
|
|
|
let actual = scan_path(&invalid_path, IconSizes::default());
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
assert!(actual.is_empty());
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn scan_path_empty_dir_returns_empty_vec() -> io::Result<()> {
|
|
|
|
|
let fs = empty_fs()?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
debug!("Calling scan_path(\"{}\")", path.display());
|
2024-02-18 02:44:54 -05:00
|
|
|
let actual = scan_path(&path.to_owned(), IconSizes::default());
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
assert_eq!(0, path.read_dir()?.count());
|
2024-02-03 02:31:08 -05:00
|
|
|
assert!(actual.is_empty());
|
2024-02-01 16:29:10 +00:00
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2024-02-03 02:31:08 -05:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_location_changes_location() -> io::Result<()> {
|
|
|
|
|
let fs = simple_fs(NUM_FILES, NUM_NESTED, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
// Next directory in temp directory
|
2024-02-05 20:58:17 -05:00
|
|
|
// This does not have to be sorted
|
2024-02-04 07:28:51 -05:00
|
|
|
let next_dir = filter_dirs(path)?
|
2024-02-03 02:31:08 -05:00
|
|
|
.next()
|
|
|
|
|
.expect("temp directory should have at least one directory");
|
|
|
|
|
|
2024-02-10 02:13:13 -05:00
|
|
|
let mut tab = Tab::new(Location::Path(path.to_owned()), TabConfig::default());
|
2024-02-03 02:31:08 -05:00
|
|
|
debug!(
|
|
|
|
|
"Emitting Message::Location(Location::Path(\"{}\"))",
|
|
|
|
|
next_dir.display()
|
|
|
|
|
);
|
|
|
|
|
tab.update(
|
|
|
|
|
Message::Location(Location::Path(next_dir.clone())),
|
|
|
|
|
Modifiers::empty(),
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
// Validate that the tab's path updated
|
|
|
|
|
// NOTE: `items_opt` is set to None with Message::Location so this ONLY checks for equal paths
|
|
|
|
|
// If item contents are NOT None then this needs to be reevaluated for correctness
|
|
|
|
|
assert_eq_tab_path(&tab, &next_dir);
|
|
|
|
|
assert!(
|
|
|
|
|
tab.items_opt.is_none(),
|
|
|
|
|
"Tab's `items` is not None which means this test needs to be updated"
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_click_single_selects_item() -> io::Result<()> {
|
2024-02-04 07:28:51 -05:00
|
|
|
// Select the second directory with no keys held down
|
|
|
|
|
tab_selects_item(&[1], Modifiers::empty(), &[false, true])
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_click_double_opens_folder() -> io::Result<()> {
|
2024-02-04 07:28:51 -05:00
|
|
|
let (fs, mut tab) = tab_click_new(NUM_FILES, NUM_NESTED, NUM_DIRS, NUM_NESTED, NAME_LEN)?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
// Simulate double clicking second directory
|
|
|
|
|
debug!("Emitting first Message::Click(Some(1))");
|
|
|
|
|
tab.update(Message::Click(Some(1)), Modifiers::empty());
|
|
|
|
|
debug!("Emitting second Message::Click(Some(1))");
|
|
|
|
|
tab.update(Message::Click(Some(1)), Modifiers::empty());
|
|
|
|
|
|
|
|
|
|
// Path to second directory
|
|
|
|
|
let second_dir = read_dir_sorted(path)?
|
2024-02-05 20:58:17 -05:00
|
|
|
.into_iter()
|
2024-02-04 07:28:51 -05:00
|
|
|
.filter(|p| p.is_dir())
|
|
|
|
|
.nth(1)
|
|
|
|
|
.expect("should be at least two directories");
|
|
|
|
|
|
|
|
|
|
// Location should have changed to second_dir
|
|
|
|
|
assert_eq_tab_path(&tab, &second_dir);
|
|
|
|
|
|
|
|
|
|
Ok(())
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_click_ctrl_selects_multiple() -> io::Result<()> {
|
2024-02-04 07:28:51 -05:00
|
|
|
// Select the first and second directory by holding down ctrl
|
|
|
|
|
tab_selects_item(&[0, 1], Modifiers::CTRL, &[true, true])
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_gonext_moves_forward_in_history() -> io::Result<()> {
|
2024-02-05 20:58:17 -05:00
|
|
|
let (fs, mut tab, dirs) = tab_history()?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
// Rewind to the start
|
|
|
|
|
for _ in 0..dirs.len() {
|
|
|
|
|
debug!("Emitting Message::GoPrevious to rewind to the start",);
|
|
|
|
|
tab.update(Message::GoPrevious, Modifiers::empty());
|
|
|
|
|
}
|
|
|
|
|
assert_eq_tab_path(&tab, path);
|
|
|
|
|
|
|
|
|
|
// Back to the future. Directories should be in the order they were opened.
|
|
|
|
|
for dir in dirs {
|
|
|
|
|
debug!("Emitting Message::GoNext",);
|
|
|
|
|
tab.update(Message::GoNext, Modifiers::empty());
|
|
|
|
|
assert_eq_tab_path(&tab, &dir);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_goprev_moves_backward_in_history() -> io::Result<()> {
|
2024-02-05 20:58:17 -05:00
|
|
|
let (fs, mut tab, dirs) = tab_history()?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
|
|
|
|
|
for dir in dirs.into_iter().rev() {
|
|
|
|
|
assert_eq_tab_path(&tab, &dir);
|
|
|
|
|
debug!("Emitting Message::GoPrevious",);
|
|
|
|
|
tab.update(Message::GoPrevious, Modifiers::empty());
|
|
|
|
|
}
|
|
|
|
|
assert_eq_tab_path(&tab, path);
|
|
|
|
|
|
|
|
|
|
Ok(())
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_empty_history_does_nothing_on_prev_next() -> io::Result<()> {
|
2024-02-05 20:58:17 -05:00
|
|
|
let fs = simple_fs(0, NUM_NESTED, NUM_DIRS, 0, NAME_LEN)?;
|
|
|
|
|
let path = fs.path();
|
2024-02-10 02:13:13 -05:00
|
|
|
let mut tab = Tab::new(Location::Path(path.into()), TabConfig::default());
|
2024-02-05 20:58:17 -05:00
|
|
|
|
|
|
|
|
// Tab's location shouldn't change if GoPrev or GoNext is triggered
|
|
|
|
|
debug!("Emitting Message::GoPrevious",);
|
|
|
|
|
tab.update(Message::GoPrevious, Modifiers::empty());
|
|
|
|
|
assert_eq_tab_path(&tab, path);
|
|
|
|
|
|
|
|
|
|
debug!("Emitting Message::GoNext",);
|
|
|
|
|
tab.update(Message::GoNext, Modifiers::empty());
|
|
|
|
|
assert_eq_tab_path(&tab, path);
|
|
|
|
|
|
|
|
|
|
Ok(())
|
2024-02-03 02:31:08 -05:00
|
|
|
}
|
2024-02-06 22:58:29 -05:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn tab_locationup_moves_up_hierarchy() -> io::Result<()> {
|
|
|
|
|
let fs = simple_fs(0, NUM_NESTED, NUM_DIRS, 0, NAME_LEN)?;
|
|
|
|
|
let path = fs.path();
|
|
|
|
|
let mut next_dir = filter_dirs(path)?
|
|
|
|
|
.next()
|
|
|
|
|
.expect("should be at least one directory");
|
|
|
|
|
|
2024-02-10 02:13:13 -05:00
|
|
|
let mut tab = Tab::new(Location::Path(next_dir.clone()), TabConfig::default());
|
2024-02-06 22:58:29 -05:00
|
|
|
// This will eventually yield false once root is hit
|
|
|
|
|
while next_dir.pop() {
|
|
|
|
|
debug!("Emitting Message::LocationUp",);
|
|
|
|
|
tab.update(Message::LocationUp, Modifiers::empty());
|
|
|
|
|
assert_eq_tab_path(&tab, &next_dir);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Ok(())
|
|
|
|
|
}
|
2024-02-01 16:29:10 +00:00
|
|
|
}
|