| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | #include "DirectoryModel.h"
 | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | #include <dirent.h>
 | 
					
						
							|  |  |  | #include <stdio.h>
 | 
					
						
							|  |  |  | #include <unistd.h>
 | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  | #include <grp.h>
 | 
					
						
							|  |  |  | #include <pwd.h>
 | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | #include <AK/FileSystemPath.h>
 | 
					
						
							|  |  |  | #include <AK/StringBuilder.h>
 | 
					
						
							| 
									
										
										
										
											2019-03-23 12:37:33 +01:00
										 |  |  | #include <SharedGraphics/GraphicsBitmap.h>
 | 
					
						
							|  |  |  | #include <SharedGraphics/Painter.h>
 | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | DirectoryModel::DirectoryModel() | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-22 00:19:53 +01:00
										 |  |  |     m_directory_icon = GraphicsBitmap::load_from_file("/res/icons/folder16.png"); | 
					
						
							|  |  |  |     m_file_icon = GraphicsBitmap::load_from_file("/res/icons/file16.png"); | 
					
						
							|  |  |  |     m_symlink_icon = GraphicsBitmap::load_from_file("/res/icons/link16.png"); | 
					
						
							|  |  |  |     m_socket_icon = GraphicsBitmap::load_from_file("/res/icons/socket16.png"); | 
					
						
							|  |  |  |     m_executable_icon = GraphicsBitmap::load_from_file("/res/icons/executable16.png"); | 
					
						
							| 
									
										
										
										
											2019-03-22 12:54:27 +01:00
										 |  |  |     m_filetype_image_icon = GraphicsBitmap::load_from_file("/res/icons/16x16/filetype-image.png"); | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     setpwent(); | 
					
						
							|  |  |  |     while (auto* passwd = getpwent()) | 
					
						
							|  |  |  |         m_user_names.set(passwd->pw_uid, passwd->pw_name); | 
					
						
							|  |  |  |     endpwent(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     setgrent(); | 
					
						
							|  |  |  |     while (auto* group = getgrent()) | 
					
						
							|  |  |  |         m_group_names.set(group->gr_gid, group->gr_name); | 
					
						
							|  |  |  |     endgrent(); | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | DirectoryModel::~DirectoryModel() | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | int DirectoryModel::row_count() const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     return m_directories.size() + m_files.size(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | int DirectoryModel::column_count() const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     return Column::__Count; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | String DirectoryModel::column_name(int column) const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (column) { | 
					
						
							|  |  |  |     case Column::Icon: return ""; | 
					
						
							|  |  |  |     case Column::Name: return "Name"; | 
					
						
							|  |  |  |     case Column::Size: return "Size"; | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  |     case Column::Owner: return "Owner"; | 
					
						
							|  |  |  |     case Column::Group: return "Group"; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     case Column::Permissions: return "Mode"; | 
					
						
							|  |  |  |     case Column::Inode: return "Inode"; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | GModel::ColumnMetadata DirectoryModel::column_metadata(int column) const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     switch (column) { | 
					
						
							|  |  |  |     case Column::Icon: return { 16, TextAlignment::Center }; | 
					
						
							|  |  |  |     case Column::Name: return { 120, TextAlignment::CenterLeft }; | 
					
						
							|  |  |  |     case Column::Size: return { 80, TextAlignment::CenterRight }; | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  |     case Column::Owner: return { 50, TextAlignment::CenterLeft }; | 
					
						
							|  |  |  |     case Column::Group: return { 50, TextAlignment::CenterLeft }; | 
					
						
							|  |  |  |     case Column::Permissions: return { 80, TextAlignment::CenterLeft }; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     case Column::Inode: return { 80, TextAlignment::CenterRight }; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | const GraphicsBitmap& DirectoryModel::icon_for(const Entry& entry) const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     if (S_ISDIR(entry.mode)) | 
					
						
							|  |  |  |         return *m_directory_icon; | 
					
						
							|  |  |  |     if (S_ISLNK(entry.mode)) | 
					
						
							|  |  |  |         return *m_symlink_icon; | 
					
						
							|  |  |  |     if (S_ISSOCK(entry.mode)) | 
					
						
							|  |  |  |         return *m_socket_icon; | 
					
						
							| 
									
										
										
										
											2019-03-01 14:57:42 +01:00
										 |  |  |     if (entry.mode & S_IXUSR) | 
					
						
							|  |  |  |         return *m_executable_icon; | 
					
						
							| 
									
										
										
										
											2019-03-23 12:37:33 +01:00
										 |  |  |     if (entry.name.ends_with(".png")) { | 
					
						
							|  |  |  |         if (!entry.thumbnail) { | 
					
						
							|  |  |  |             if (auto png_bitmap = GraphicsBitmap::load_from_file(entry.full_path(*this))) { | 
					
						
							|  |  |  |                 entry.thumbnail = GraphicsBitmap::create(png_bitmap->format(), { 32, 32 }); | 
					
						
							|  |  |  |                 Painter painter(*entry.thumbnail); | 
					
						
							|  |  |  |                 painter.draw_scaled_bitmap(entry.thumbnail->rect(), *png_bitmap, png_bitmap->rect()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         if (!entry.thumbnail) | 
					
						
							|  |  |  |             return *m_filetype_image_icon; | 
					
						
							|  |  |  |         return *entry.thumbnail; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     return *m_file_icon; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | static String permission_string(mode_t mode) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     StringBuilder builder; | 
					
						
							|  |  |  |     if (S_ISDIR(mode)) | 
					
						
							|  |  |  |         builder.append("d"); | 
					
						
							|  |  |  |     else if (S_ISLNK(mode)) | 
					
						
							|  |  |  |         builder.append("l"); | 
					
						
							|  |  |  |     else if (S_ISBLK(mode)) | 
					
						
							|  |  |  |         builder.append("b"); | 
					
						
							|  |  |  |     else if (S_ISCHR(mode)) | 
					
						
							|  |  |  |         builder.append("c"); | 
					
						
							|  |  |  |     else if (S_ISFIFO(mode)) | 
					
						
							|  |  |  |         builder.append("f"); | 
					
						
							|  |  |  |     else if (S_ISSOCK(mode)) | 
					
						
							|  |  |  |         builder.append("s"); | 
					
						
							|  |  |  |     else if (S_ISREG(mode)) | 
					
						
							|  |  |  |         builder.append("-"); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         builder.append("?"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     builder.appendf("%c%c%c%c%c%c%c%c", | 
					
						
							|  |  |  |         mode & S_IRUSR ? 'r' : '-', | 
					
						
							|  |  |  |         mode & S_IWUSR ? 'w' : '-', | 
					
						
							|  |  |  |         mode & S_ISUID ? 's' : (mode & S_IXUSR ? 'x' : '-'), | 
					
						
							|  |  |  |         mode & S_IRGRP ? 'r' : '-', | 
					
						
							|  |  |  |         mode & S_IWGRP ? 'w' : '-', | 
					
						
							|  |  |  |         mode & S_ISGID ? 's' : (mode & S_IXGRP ? 'x' : '-'), | 
					
						
							|  |  |  |         mode & S_IROTH ? 'r' : '-', | 
					
						
							|  |  |  |         mode & S_IWOTH ? 'w' : '-' | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (mode & S_ISVTX) | 
					
						
							|  |  |  |         builder.append("t"); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         builder.appendf("%c", mode & S_IXOTH ? 'x' : '-'); | 
					
						
							|  |  |  |     return builder.to_string(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | String DirectoryModel::name_for_uid(uid_t uid) const | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     auto it = m_user_names.find(uid); | 
					
						
							|  |  |  |     if (it == m_user_names.end()) | 
					
						
							|  |  |  |         return String::format("%u", uid); | 
					
						
							|  |  |  |     return (*it).value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | String DirectoryModel::name_for_gid(uid_t gid) const | 
					
						
							| 
									
										
										
										
											2019-03-02 02:05:49 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     auto it = m_user_names.find(gid); | 
					
						
							|  |  |  |     if (it == m_user_names.end()) | 
					
						
							|  |  |  |         return String::format("%u", gid); | 
					
						
							|  |  |  |     return (*it).value; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | GVariant DirectoryModel::data(const GModelIndex& index, Role role) const | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-09 14:52:25 +01:00
										 |  |  |     ASSERT(is_valid(index)); | 
					
						
							| 
									
										
										
										
											2019-03-06 19:56:47 +01:00
										 |  |  |     auto& entry = this->entry(index.row()); | 
					
						
							| 
									
										
										
										
											2019-03-09 14:52:25 +01:00
										 |  |  |     if (role == Role::Sort) { | 
					
						
							|  |  |  |         switch (index.column()) { | 
					
						
							|  |  |  |         case Column::Icon: return entry.is_directory() ? 0 : 1; | 
					
						
							|  |  |  |         case Column::Name: return entry.name; | 
					
						
							|  |  |  |         case Column::Size: return (int)entry.size; | 
					
						
							|  |  |  |         case Column::Owner: return name_for_uid(entry.uid); | 
					
						
							|  |  |  |         case Column::Group: return name_for_gid(entry.gid); | 
					
						
							|  |  |  |         case Column::Permissions: return permission_string(entry.mode); | 
					
						
							|  |  |  |         case Column::Inode: return (int)entry.inode; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  |     if (role == Role::Display) { | 
					
						
							|  |  |  |         switch (index.column()) { | 
					
						
							|  |  |  |         case Column::Icon: return icon_for(entry); | 
					
						
							|  |  |  |         case Column::Name: return entry.name; | 
					
						
							|  |  |  |         case Column::Size: return (int)entry.size; | 
					
						
							|  |  |  |         case Column::Owner: return name_for_uid(entry.uid); | 
					
						
							|  |  |  |         case Column::Group: return name_for_gid(entry.gid); | 
					
						
							|  |  |  |         case Column::Permissions: return permission_string(entry.mode); | 
					
						
							|  |  |  |         case Column::Inode: return (int)entry.inode; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-23 03:53:51 +01:00
										 |  |  |     if (role == Role::Icon) { | 
					
						
							|  |  |  |         return icon_for(entry); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-18 04:54:07 +01:00
										 |  |  |     return { }; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | void DirectoryModel::update() | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     DIR* dirp = opendir(m_path.characters()); | 
					
						
							|  |  |  |     if (!dirp) { | 
					
						
							|  |  |  |         perror("opendir"); | 
					
						
							|  |  |  |         exit(1); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     m_directories.clear(); | 
					
						
							|  |  |  |     m_files.clear(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     m_bytes_in_files = 0; | 
					
						
							|  |  |  |     while (auto* de = readdir(dirp)) { | 
					
						
							|  |  |  |         Entry entry; | 
					
						
							|  |  |  |         entry.name = de->d_name; | 
					
						
							|  |  |  |         struct stat st; | 
					
						
							|  |  |  |         int rc = lstat(String::format("%s/%s", m_path.characters(), de->d_name).characters(), &st); | 
					
						
							|  |  |  |         if (rc < 0) { | 
					
						
							|  |  |  |             perror("lstat"); | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         entry.size = st.st_size; | 
					
						
							|  |  |  |         entry.mode = st.st_mode; | 
					
						
							|  |  |  |         entry.uid = st.st_uid; | 
					
						
							|  |  |  |         entry.gid = st.st_gid; | 
					
						
							|  |  |  |         entry.inode = st.st_ino; | 
					
						
							|  |  |  |         auto& entries = S_ISDIR(st.st_mode) ? m_directories : m_files; | 
					
						
							|  |  |  |         entries.append(move(entry)); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (S_ISREG(entry.mode)) | 
					
						
							|  |  |  |             m_bytes_in_files += st.st_size; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     closedir(dirp); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     did_update(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | void DirectoryModel::open(const String& a_path) | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-03 00:40:50 +01:00
										 |  |  |     FileSystemPath canonical_path(a_path); | 
					
						
							|  |  |  |     auto path = canonical_path.string(); | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     if (m_path == path) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     DIR* dirp = opendir(path.characters()); | 
					
						
							|  |  |  |     if (!dirp) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     closedir(dirp); | 
					
						
							|  |  |  |     m_path = path; | 
					
						
							|  |  |  |     update(); | 
					
						
							|  |  |  |     set_selected_index({ 0, 0 }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-23 01:42:49 +01:00
										 |  |  | void DirectoryModel::activate(const GModelIndex& index) | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2019-03-08 16:02:00 +01:00
										 |  |  |     if (!index.is_valid()) | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     auto& entry = this->entry(index.row()); | 
					
						
							| 
									
										
										
										
											2019-03-01 14:57:42 +01:00
										 |  |  |     FileSystemPath path(String::format("%s/%s", m_path.characters(), entry.name.characters())); | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     if (entry.is_directory()) { | 
					
						
							| 
									
										
										
										
											2019-03-01 14:57:42 +01:00
										 |  |  |         open(path.string()); | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (entry.is_executable()) { | 
					
						
							|  |  |  |         if (fork() == 0) { | 
					
						
							|  |  |  |             int rc = execl(path.string().characters(), path.string().characters(), nullptr); | 
					
						
							|  |  |  |             if (rc < 0) | 
					
						
							|  |  |  |                 perror("exec"); | 
					
						
							|  |  |  |             ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2019-03-07 14:43:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-22 12:43:29 +01:00
										 |  |  |     if (path.string().ends_with(".png")) { | 
					
						
							|  |  |  |         if (fork() == 0) { | 
					
						
							|  |  |  |             int rc = execl("/bin/qs", "/bin/qs", path.string().characters(), nullptr); | 
					
						
							|  |  |  |             if (rc < 0) | 
					
						
							|  |  |  |                 perror("exec"); | 
					
						
							|  |  |  |             ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-03-07 14:43:12 +01:00
										 |  |  |     if (fork() == 0) { | 
					
						
							|  |  |  |         int rc = execl("/bin/TextEditor", "/bin/TextEditor", path.string().characters(), nullptr); | 
					
						
							|  |  |  |         if (rc < 0) | 
					
						
							|  |  |  |             perror("exec"); | 
					
						
							|  |  |  |         ASSERT_NOT_REACHED(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return; | 
					
						
							| 
									
										
										
										
											2019-03-01 13:54:28 +01:00
										 |  |  | } |