/* * Copyright (c) 2018-2020, Andreas Kling * * SPDX-License-Identifier: BSD-2-Clause */ #pragma once #include #include #include #include #include #include #include #include namespace GUI { class FileSystemModel : public Model , public Weakable { friend struct Node; public: enum Mode { Invalid, DirectoriesOnly, FilesAndDirectories }; enum Column { Icon = 0, Name, Size, Owner, Group, Permissions, ModificationTime, Inode, SymlinkTarget, __Count, }; struct Node { ~Node() { } String name; String symlink_target; size_t size { 0 }; mode_t mode { 0 }; uid_t uid { 0 }; gid_t gid { 0 }; ino_t inode { 0 }; time_t mtime { 0 }; bool is_accessible_directory { false }; size_t total_size { 0 }; mutable RefPtr thumbnail; bool is_directory() const { return S_ISDIR(mode); } bool is_symlink_to_directory() const; bool is_executable() const { return mode & (S_IXUSR | S_IXGRP | S_IXOTH); } bool is_selected() const { return m_selected; } void set_selected(bool selected); bool has_error() const { return m_error != 0; } int error() const { return m_error; } const char* error_string() const { return strerror(m_error); } String full_path() const; private: friend class FileSystemModel; explicit Node(FileSystemModel& model) : m_model(model) { } FileSystemModel& m_model; Node* parent { nullptr }; NonnullOwnPtrVector children; bool has_traversed { false }; bool m_selected { false }; RefPtr m_file_watcher; int m_error { 0 }; bool m_parent_of_root { false }; ModelIndex index(int column) const; void traverse_if_needed(); void reify_if_needed(); bool fetch_data(const String& full_path, bool is_root); }; static NonnullRefPtr create(String root_path = "/", Mode mode = Mode::FilesAndDirectories) { return adopt_ref(*new FileSystemModel(root_path, mode)); } virtual ~FileSystemModel() override; String root_path() const { return m_root_path; } void set_root_path(String); String full_path(const ModelIndex&) const; ModelIndex index(String path, int column) const; void update_node_on_selection(const ModelIndex&, const bool); ModelIndex m_previously_selected_index {}; const Node& node(const ModelIndex& index) const; Function on_thumbnail_progress; Function on_complete; Function on_error; virtual int tree_column() const override { return Column::Name; } virtual int row_count(const ModelIndex& = ModelIndex()) const override; virtual int column_count(const ModelIndex& = ModelIndex()) const override; virtual String column_name(int column) const override; virtual Variant data(const ModelIndex&, ModelRole = ModelRole::Display) const override; virtual void update() override; virtual ModelIndex parent_index(const ModelIndex&) const override; virtual ModelIndex index(int row, int column = 0, const ModelIndex& parent = ModelIndex()) const override; virtual StringView drag_data_type() const override { return "text/uri-list"; } virtual bool accepts_drag(const ModelIndex&, const Vector& mime_types) const override; virtual bool is_column_sortable(int column_index) const override { return column_index != Column::Icon; } virtual bool is_editable(const ModelIndex&) const override; virtual bool is_searchable() const override { return true; } virtual void set_data(const ModelIndex&, const Variant&) override; virtual Vector matches(const StringView&, unsigned = MatchesFlag::AllMatching, const ModelIndex& = ModelIndex()) override; static String timestamp_string(time_t timestamp) { return Core::DateTime::from_timestamp(timestamp).to_string(); } bool should_show_dotfiles() const { return m_should_show_dotfiles; } void set_should_show_dotfiles(bool); private: FileSystemModel(String root_path, Mode); String name_for_uid(uid_t) const; String name_for_gid(gid_t) const; HashMap m_user_names; HashMap m_group_names; bool fetch_thumbnail_for(const Node& node); GUI::Icon icon_for(const Node& node) const; String m_root_path; Mode m_mode { Invalid }; OwnPtr m_root { nullptr }; unsigned m_thumbnail_progress { 0 }; unsigned m_thumbnail_progress_total { 0 }; bool m_should_show_dotfiles { false }; }; }