2020-01-18 09:38:21 +01:00
|
|
|
/*
|
|
|
|
* Copyright (c) 2018-2020, Andreas Kling <kling@serenityos.org>
|
2022-02-26 10:50:04 -07:00
|
|
|
* Copyright (c) 2022, the SerenityOS developers.
|
2020-01-18 09:38:21 +01:00
|
|
|
*
|
2021-04-22 01:24:48 -07:00
|
|
|
* SPDX-License-Identifier: BSD-2-Clause
|
2020-01-18 09:38:21 +01:00
|
|
|
*/
|
|
|
|
|
2019-03-29 17:03:30 +01:00
|
|
|
#pragma once
|
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
#include <AK/HashMap.h>
|
|
|
|
#include <AK/NonnullOwnPtrVector.h>
|
2020-02-11 20:05:44 +01:00
|
|
|
#include <LibCore/DateTime.h>
|
2021-02-08 20:42:34 +01:00
|
|
|
#include <LibCore/FileWatcher.h>
|
2020-02-06 20:33:02 +01:00
|
|
|
#include <LibGUI/Model.h>
|
2020-04-21 21:26:17 +02:00
|
|
|
#include <string.h>
|
2020-01-10 18:58:00 +03:00
|
|
|
#include <sys/stat.h>
|
|
|
|
#include <time.h>
|
2019-03-29 17:03:30 +01:00
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
namespace GUI {
|
|
|
|
|
|
|
|
class FileSystemModel
|
|
|
|
: public Model
|
|
|
|
, public Weakable<FileSystemModel> {
|
2020-01-10 18:58:00 +03:00
|
|
|
friend struct Node;
|
2019-05-28 11:53:16 +02:00
|
|
|
|
2019-03-29 17:03:30 +01:00
|
|
|
public:
|
2019-06-07 17:13:23 +02:00
|
|
|
enum Mode {
|
2019-05-28 11:53:16 +02:00
|
|
|
Invalid,
|
|
|
|
DirectoriesOnly,
|
|
|
|
FilesAndDirectories
|
|
|
|
};
|
2019-03-29 17:14:03 +01:00
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
enum Column {
|
|
|
|
Icon = 0,
|
|
|
|
Name,
|
|
|
|
Size,
|
2021-08-31 01:11:05 +02:00
|
|
|
User,
|
2020-01-10 18:58:00 +03:00
|
|
|
Group,
|
|
|
|
Permissions,
|
|
|
|
ModificationTime,
|
|
|
|
Inode,
|
2020-01-27 22:19:05 +01:00
|
|
|
SymlinkTarget,
|
2020-01-10 18:58:00 +03:00
|
|
|
__Count,
|
|
|
|
};
|
|
|
|
|
|
|
|
struct Node {
|
2022-02-26 10:50:04 -07:00
|
|
|
~Node() = default;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
|
|
|
String name;
|
2020-01-27 22:19:05 +01:00
|
|
|
String symlink_target;
|
2020-01-10 18:58:00 +03:00
|
|
|
size_t size { 0 };
|
|
|
|
mode_t mode { 0 };
|
|
|
|
uid_t uid { 0 };
|
|
|
|
gid_t gid { 0 };
|
|
|
|
ino_t inode { 0 };
|
|
|
|
time_t mtime { 0 };
|
2020-09-18 18:21:28 +02:00
|
|
|
bool is_accessible_directory { false };
|
2020-01-10 18:58:00 +03:00
|
|
|
|
|
|
|
size_t total_size { 0 };
|
|
|
|
|
2020-02-06 11:56:38 +01:00
|
|
|
mutable RefPtr<Gfx::Bitmap> thumbnail;
|
2020-01-10 18:58:00 +03:00
|
|
|
bool is_directory() const { return S_ISDIR(mode); }
|
2021-04-09 23:02:15 +02:00
|
|
|
bool is_symlink_to_directory() const;
|
2020-01-28 16:23:59 +03:00
|
|
|
bool is_executable() const { return mode & (S_IXUSR | S_IXGRP | S_IXOTH); }
|
2020-01-10 18:58:00 +03:00
|
|
|
|
2020-07-10 09:51:15 -04:00
|
|
|
bool is_selected() const { return m_selected; }
|
|
|
|
void set_selected(bool selected);
|
|
|
|
|
2020-04-21 21:26:17 +02:00
|
|
|
bool has_error() const { return m_error != 0; }
|
|
|
|
int error() const { return m_error; }
|
2021-09-07 18:14:34 +02:00
|
|
|
char const* error_string() const { return strerror(m_error); }
|
2020-04-21 21:26:17 +02:00
|
|
|
|
2020-08-17 22:02:21 +02:00
|
|
|
String full_path() const;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
|
|
|
private:
|
2020-02-02 15:07:41 +01:00
|
|
|
friend class FileSystemModel;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
2020-08-17 22:02:21 +02:00
|
|
|
explicit Node(FileSystemModel& model)
|
|
|
|
: m_model(model)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
|
|
|
|
FileSystemModel& m_model;
|
|
|
|
|
2021-08-08 10:00:13 +00:00
|
|
|
Node* m_parent { nullptr };
|
|
|
|
NonnullOwnPtrVector<Node> m_children;
|
|
|
|
bool m_has_traversed { false };
|
2020-01-10 18:58:00 +03:00
|
|
|
|
2020-07-10 09:51:15 -04:00
|
|
|
bool m_selected { false };
|
|
|
|
|
2020-04-21 21:26:17 +02:00
|
|
|
int m_error { 0 };
|
2020-09-16 17:47:34 +02:00
|
|
|
bool m_parent_of_root { false };
|
2020-04-21 21:26:17 +02:00
|
|
|
|
2020-08-17 22:02:21 +02:00
|
|
|
ModelIndex index(int column) const;
|
|
|
|
void traverse_if_needed();
|
|
|
|
void reify_if_needed();
|
2021-08-08 10:01:03 +00:00
|
|
|
bool fetch_data(String const& full_path, bool is_root);
|
|
|
|
|
|
|
|
OwnPtr<Node> create_child(String const& child_name);
|
2020-01-10 18:58:00 +03:00
|
|
|
};
|
|
|
|
|
2021-04-17 00:47:36 +02:00
|
|
|
static NonnullRefPtr<FileSystemModel> create(String root_path = "/", Mode mode = Mode::FilesAndDirectories)
|
2019-03-29 17:03:30 +01:00
|
|
|
{
|
2021-04-23 16:46:57 +02:00
|
|
|
return adopt_ref(*new FileSystemModel(root_path, mode));
|
2019-03-29 17:03:30 +01:00
|
|
|
}
|
2022-02-26 10:50:04 -07:00
|
|
|
virtual ~FileSystemModel() override = default;
|
2019-03-29 17:03:30 +01:00
|
|
|
|
|
|
|
String root_path() const { return m_root_path; }
|
2021-04-17 00:47:36 +02:00
|
|
|
void set_root_path(String);
|
2021-09-07 18:14:34 +02:00
|
|
|
String full_path(ModelIndex const&) const;
|
2021-04-17 00:47:36 +02:00
|
|
|
ModelIndex index(String path, int column) const;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
2022-04-01 20:58:27 +03:00
|
|
|
void update_node_on_selection(ModelIndex const&, bool const);
|
2020-07-10 09:51:15 -04:00
|
|
|
ModelIndex m_previously_selected_index {};
|
|
|
|
|
2021-09-07 18:14:34 +02:00
|
|
|
Node const& node(ModelIndex const& index) const;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
|
|
|
Function<void(int done, int total)> on_thumbnail_progress;
|
2020-04-21 21:26:17 +02:00
|
|
|
Function<void()> on_complete;
|
2021-09-07 18:14:34 +02:00
|
|
|
Function<void(int error, char const* error_string)> on_directory_change_error;
|
|
|
|
Function<void(int error, char const* error_string)> on_rename_error;
|
2021-09-24 18:30:12 +02:00
|
|
|
Function<void(String const& old_name, String const& new_name)> on_rename_successful;
|
2019-03-29 17:03:30 +01:00
|
|
|
|
2020-02-02 15:07:41 +01:00
|
|
|
virtual int tree_column() const override { return Column::Name; }
|
2021-09-07 18:14:34 +02:00
|
|
|
virtual int row_count(ModelIndex const& = ModelIndex()) const override;
|
|
|
|
virtual int column_count(ModelIndex const& = ModelIndex()) const override;
|
2020-01-10 18:58:00 +03:00
|
|
|
virtual String column_name(int column) const override;
|
2021-09-07 18:14:34 +02:00
|
|
|
virtual Variant data(ModelIndex const&, ModelRole = ModelRole::Display) const override;
|
|
|
|
virtual ModelIndex parent_index(ModelIndex const&) const override;
|
|
|
|
virtual ModelIndex index(int row, int column = 0, ModelIndex const& parent = ModelIndex()) const override;
|
2020-02-14 13:18:34 +01:00
|
|
|
virtual StringView drag_data_type() const override { return "text/uri-list"; }
|
2021-09-07 18:14:34 +02:00
|
|
|
virtual bool accepts_drag(ModelIndex const&, Vector<String> const& mime_types) const override;
|
2020-05-21 19:45:15 +02:00
|
|
|
virtual bool is_column_sortable(int column_index) const override { return column_index != Column::Icon; }
|
2021-09-07 18:14:34 +02:00
|
|
|
virtual bool is_editable(ModelIndex const&) const override;
|
2020-10-20 15:13:28 -06:00
|
|
|
virtual bool is_searchable() const override { return true; }
|
2021-09-07 18:14:34 +02:00
|
|
|
virtual void set_data(ModelIndex const&, Variant const&) override;
|
2021-11-11 00:55:02 +01:00
|
|
|
virtual Vector<ModelIndex> matches(StringView, unsigned = MatchesFlag::AllMatching, ModelIndex const& = ModelIndex()) override;
|
2021-05-25 14:13:19 +00:00
|
|
|
virtual void invalidate() override;
|
2019-03-29 17:03:30 +01:00
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
static String timestamp_string(time_t timestamp)
|
|
|
|
{
|
2020-02-11 20:05:44 +01:00
|
|
|
return Core::DateTime::from_timestamp(timestamp).to_string();
|
2020-01-10 18:58:00 +03:00
|
|
|
}
|
|
|
|
|
2020-07-18 18:46:37 +02:00
|
|
|
bool should_show_dotfiles() const { return m_should_show_dotfiles; }
|
|
|
|
void set_should_show_dotfiles(bool);
|
|
|
|
|
2019-03-29 17:03:30 +01:00
|
|
|
private:
|
2021-04-17 00:47:36 +02:00
|
|
|
FileSystemModel(String root_path, Mode);
|
2019-03-29 17:03:30 +01:00
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
String name_for_uid(uid_t) const;
|
|
|
|
String name_for_gid(gid_t) const;
|
|
|
|
|
2022-04-03 18:45:23 +04:30
|
|
|
Optional<Node const&> node_for_path(String const&) const;
|
2021-05-09 17:09:30 +00:00
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
HashMap<uid_t, String> m_user_names;
|
|
|
|
HashMap<gid_t, String> m_group_names;
|
|
|
|
|
2021-09-07 18:14:34 +02:00
|
|
|
bool fetch_thumbnail_for(Node const& node);
|
|
|
|
GUI::Icon icon_for(Node const& node) const;
|
2020-01-10 18:58:00 +03:00
|
|
|
|
2021-08-08 10:01:03 +00:00
|
|
|
void handle_file_event(Core::FileWatcherEvent const& event);
|
|
|
|
|
2019-03-29 17:03:30 +01:00
|
|
|
String m_root_path;
|
2019-03-29 17:14:03 +01:00
|
|
|
Mode m_mode { Invalid };
|
2020-01-10 18:58:00 +03:00
|
|
|
OwnPtr<Node> m_root { nullptr };
|
2019-03-29 17:03:30 +01:00
|
|
|
|
2020-01-10 18:58:00 +03:00
|
|
|
unsigned m_thumbnail_progress { 0 };
|
|
|
|
unsigned m_thumbnail_progress_total { 0 };
|
2020-07-18 18:46:37 +02:00
|
|
|
|
|
|
|
bool m_should_show_dotfiles { false };
|
2021-05-09 17:09:30 +00:00
|
|
|
|
|
|
|
RefPtr<Core::FileWatcher> m_file_watcher;
|
2019-03-29 17:03:30 +01:00
|
|
|
};
|
2020-02-02 15:07:41 +01:00
|
|
|
|
|
|
|
}
|