1
0
Fork 0
mirror of synced 2024-05-09 23:12:29 +12:00
czkawka/czkawka_gui/src/connect_things/connect_selection_of_directories.rs
Rafał Mikrut 378fa1fd6e
Excluded extensions and krokiet new features (#1184)
* AVC

* Import split

* Default thread size

* Hen

* Allowed extensions

* Perf

* Connect

* Excluded

* Zmiany

* Optimization

* 4.10

* At once

* Included

* Chang

* VD

* VD

* Hashes

* Wersja

* SD

* Up

* Up

* 2024

* Dup

* Slint files

* Added  select

* Selections

* Fix

* LTO

* Actions

* Added popup delete

* AB

* V4

* Release

* LTO

* Basic moving

* Commonsy

* Moving probably works

* Popup move
2024-02-14 17:45:25 +01:00

309 lines
13 KiB
Rust

use std::collections::HashSet;
use std::path::PathBuf;
use gdk4::{DragAction, FileList};
use gtk4::prelude::*;
use gtk4::{DropTarget, FileChooserNative, Notebook, Orientation, ResponseType, TreeView, Window};
#[cfg(target_family = "windows")]
use czkawka_core::common::normalize_windows_path;
use crate::flg;
use crate::gui_structs::gui_data::GuiData;
use crate::help_functions::{check_if_value_is_in_list_store, get_list_store, ColumnsExcludedDirectory, ColumnsIncludedDirectory};
use crate::notebook_enums::{to_notebook_upper_enum, NotebookUpperEnum};
pub fn connect_selection_of_directories(gui_data: &GuiData) {
// Add manually directory
{
let tree_view_included_directories = gui_data.upper_notebook.tree_view_included_directories.clone();
let window_main = gui_data.window_main.clone();
let buttons_manual_add_included_directory = gui_data.upper_notebook.buttons_manual_add_included_directory.clone();
buttons_manual_add_included_directory.connect_clicked(move |_| {
add_manually_directories(&window_main, &tree_view_included_directories, false);
});
}
// Add manually excluded directory
{
let tree_view_excluded_directories = gui_data.upper_notebook.tree_view_excluded_directories.clone();
let window_main = gui_data.window_main.clone();
let buttons_manual_add_excluded_directory = gui_data.upper_notebook.buttons_manual_add_excluded_directory.clone();
buttons_manual_add_excluded_directory.connect_clicked(move |_| {
add_manually_directories(&window_main, &tree_view_excluded_directories, true);
});
}
// Add included directory
{
let buttons_add_included_directory = gui_data.upper_notebook.buttons_add_included_directory.clone();
let file_dialog_include_exclude_folder_selection = gui_data.file_dialog_include_exclude_folder_selection.clone();
buttons_add_included_directory.connect_clicked(move |_| {
file_dialog_include_exclude_folder_selection.show();
file_dialog_include_exclude_folder_selection.set_title(&flg!("include_folders_dialog_title"));
});
}
// Add excluded directory
{
let buttons_add_excluded_directory = gui_data.upper_notebook.buttons_add_excluded_directory.clone();
let file_dialog_include_exclude_folder_selection = gui_data.file_dialog_include_exclude_folder_selection.clone();
buttons_add_excluded_directory.connect_clicked(move |_| {
file_dialog_include_exclude_folder_selection.show();
file_dialog_include_exclude_folder_selection.set_title(&flg!("exclude_folders_dialog_title"));
});
}
// Connect
{
let notebook_upper = gui_data.upper_notebook.notebook_upper.clone();
let tree_view_included_directories = gui_data.upper_notebook.tree_view_included_directories.clone();
let tree_view_excluded_directories = gui_data.upper_notebook.tree_view_excluded_directories.clone();
let file_dialog_include_exclude_folder_selection = gui_data.file_dialog_include_exclude_folder_selection.clone();
connect_file_dialog(
&file_dialog_include_exclude_folder_selection,
tree_view_included_directories,
tree_view_excluded_directories,
notebook_upper,
);
}
// Drag and drop
{
configure_directory_drop(&gui_data.upper_notebook.tree_view_included_directories, false);
configure_directory_drop(&gui_data.upper_notebook.tree_view_excluded_directories, true);
}
// Remove Excluded Folder
{
let buttons_remove_excluded_directory = gui_data.upper_notebook.buttons_remove_excluded_directory.clone();
let tree_view_excluded_directories = gui_data.upper_notebook.tree_view_excluded_directories.clone();
buttons_remove_excluded_directory.connect_clicked(move |_| {
let list_store = get_list_store(&tree_view_excluded_directories);
let selection = tree_view_excluded_directories.selection();
let (vec_tree_path, _tree_model) = selection.selected_rows();
for tree_path in vec_tree_path.iter().rev() {
list_store.remove(&list_store.iter(tree_path).unwrap());
}
});
}
// Remove Included Folder
{
let buttons_remove_included_directory = gui_data.upper_notebook.buttons_remove_included_directory.clone();
let tree_view_included_directories = gui_data.upper_notebook.tree_view_included_directories.clone();
buttons_remove_included_directory.connect_clicked(move |_| {
let list_store = get_list_store(&tree_view_included_directories);
let selection = tree_view_included_directories.selection();
let (vec_tree_path, _tree_model) = selection.selected_rows();
for tree_path in vec_tree_path.iter().rev() {
list_store.remove(&list_store.iter(tree_path).unwrap());
}
});
}
}
fn configure_directory_drop(tree_view: &TreeView, excluded_items: bool) {
let tv = tree_view.clone();
let drop_target = DropTarget::builder().name("file-drop-target").actions(DragAction::COPY).build();
drop_target.set_types(&[FileList::static_type()]);
drop_target.connect_drop(move |_, value, _, _| {
if let Ok(file_list) = value.get::<FileList>() {
let mut folders: HashSet<PathBuf> = HashSet::new();
for f in file_list.files() {
if let Some(path) = f.path() {
if path.is_dir() {
folders.insert(path);
} else if let Some(parent) = path.parent() {
if parent.is_dir() {
folders.insert(parent.to_path_buf());
}
}
}
}
add_directories(&tv, &folders.into_iter().collect(), excluded_items);
}
true
});
tree_view.add_controller(drop_target);
}
fn connect_file_dialog(file_dialog_include_exclude_folder_selection: &FileChooserNative, include_tree_view: TreeView, exclude_tree_view: TreeView, notebook_upper: Notebook) {
file_dialog_include_exclude_folder_selection.connect_response(move |file_chooser, response_type| {
if response_type == ResponseType::Accept {
let excluded_items;
let tree_view = match to_notebook_upper_enum(notebook_upper.current_page().unwrap()) {
NotebookUpperEnum::IncludedDirectories => {
excluded_items = false;
&include_tree_view
}
NotebookUpperEnum::ExcludedDirectories => {
excluded_items = true;
&exclude_tree_view
}
_ => panic!(),
};
let mut folders: Vec<PathBuf> = Vec::new();
let g_files = file_chooser.files();
for index in 0..g_files.n_items() {
let file = &g_files.item(index);
if let Some(file) = file {
let ss = file.clone().downcast::<gtk4::gio::File>().unwrap();
if let Some(path_buf) = ss.path() {
folders.push(path_buf);
}
}
}
add_directories(tree_view, &folders, excluded_items);
}
});
}
fn add_directories(tree_view: &TreeView, folders: &Vec<PathBuf>, excluded_items: bool) {
let list_store = get_list_store(tree_view);
if excluded_items {
for file_entry in folders {
let values: [(u32, &dyn ToValue); 1] = [(ColumnsExcludedDirectory::Path as u32, &file_entry.to_string_lossy().to_string())];
list_store.set(&list_store.append(), &values);
}
} else {
for file_entry in folders {
let values: [(u32, &dyn ToValue); 2] = [
(ColumnsIncludedDirectory::Path as u32, &file_entry.to_string_lossy().to_string()),
(ColumnsIncludedDirectory::ReferenceButton as u32, &false),
];
list_store.set(&list_store.append(), &values);
}
}
}
fn add_manually_directories(window_main: &Window, tree_view: &TreeView, excluded_items: bool) {
let dialog = gtk4::Dialog::builder()
.title(flg!("include_manually_directories_dialog_title"))
.transient_for(window_main)
.modal(true)
.build();
dialog.set_default_size(300, 0);
let entry: gtk4::Entry = gtk4::Entry::new();
let added_button = dialog.add_button(&flg!("general_ok_button"), ResponseType::Ok);
dialog.add_button(&flg!("general_close_button"), ResponseType::Cancel);
let parent = added_button.parent().unwrap().parent().unwrap().downcast::<gtk4::Box>().unwrap(); // TODO Hack, but not so ugly as before
parent.set_orientation(Orientation::Vertical);
parent.insert_child_after(&entry, None::<&gtk4::Widget>);
dialog.show();
let tree_view = tree_view.clone();
dialog.connect_response(move |dialog, response_type| {
if response_type == ResponseType::Ok {
for text in entry.text().split(';') {
let mut text = text.trim().to_string();
#[cfg(target_family = "windows")]
let mut text = normalize_windows_path(text).to_string_lossy().to_string();
remove_ending_slashes(&mut text);
if !text.is_empty() {
let list_store = get_list_store(&tree_view);
if excluded_items {
if !(check_if_value_is_in_list_store(&list_store, ColumnsExcludedDirectory::Path as i32, &text)) {
let values: [(u32, &dyn ToValue); 1] = [(ColumnsExcludedDirectory::Path as u32, &text)];
list_store.set(&list_store.append(), &values);
}
} else {
if !check_if_value_is_in_list_store(&list_store, ColumnsIncludedDirectory::Path as i32, &text) {
let values: [(u32, &dyn ToValue); 2] = [(ColumnsIncludedDirectory::Path as u32, &text), (ColumnsIncludedDirectory::ReferenceButton as u32, &false)];
list_store.set(&list_store.append(), &values);
}
}
}
}
}
dialog.close();
});
}
fn remove_ending_slashes(original_string: &mut String) {
let mut windows_disk_path: bool = false;
let mut chars = original_string.chars();
if let Some(first_character) = chars.next() {
if first_character.is_alphabetic() {
if let Some(second_character) = chars.next() {
if second_character == ':' {
windows_disk_path = true;
original_string.push('/'); // In case of adding window path without ending slash e.g. C: instead C:/ or C:\
}
}
}
}
while (original_string != "/" && (original_string.ends_with('/') || original_string.ends_with('\\'))) && (!windows_disk_path || original_string.len() > 3) {
original_string.pop();
}
}
#[test]
pub fn test_remove_ending_slashes() {
let mut original = "/home/rafal".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "/home/rafal/".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "/home/rafal\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "/home/rafal/////////".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "/home/rafal/\\//////\\\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "/home/rafal\\\\\\\\\\\\\\\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/home/rafal");
let mut original = "\\\\\\\\\\\\\\\\\\\\\\\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "");
let mut original = "//////////".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "/");
let mut original = "C:/".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:/");
let mut original = "C:\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:\\");
let mut original = "C://////////".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:/");
let mut original = "C:/roman/function/".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:/roman/function");
let mut original = "C:/staszek/without".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:/staszek/without");
let mut original = "C:\\\\\\\\\\".to_string();
remove_ending_slashes(&mut original);
assert_eq!(&original, "C:\\");
}