mirror of
https://github.com/BoredDevNL/BoredOS.git
synced 2026-05-15 10:48:38 +00:00
976 lines
No EOL
34 KiB
C
976 lines
No EOL
34 KiB
C
#include "explorer.h"
|
|
#include "graphics.h"
|
|
#include "fat32.h"
|
|
#include "wm.h"
|
|
#include "editor.h"
|
|
#include "markdown.h"
|
|
#include <stdbool.h>
|
|
#include <stddef.h>
|
|
|
|
// === File Explorer State ===
|
|
Window win_explorer;
|
|
|
|
#define EXPLORER_MAX_FILES 64
|
|
#define EXPLORER_ITEM_HEIGHT 80
|
|
#define EXPLORER_ITEM_WIDTH 120
|
|
#define EXPLORER_COLS 3
|
|
#define EXPLORER_ROWS 4
|
|
#define EXPLORER_PADDING 15
|
|
|
|
// Dialog states
|
|
#define DIALOG_NONE 0
|
|
#define DIALOG_CREATE_FILE 1
|
|
#define DIALOG_CREATE_FOLDER 2
|
|
#define DIALOG_DELETE_CONFIRM 3
|
|
#define DIALOG_INPUT_MAX 256
|
|
|
|
typedef struct {
|
|
char name[256];
|
|
bool is_directory;
|
|
uint32_t size;
|
|
uint32_t color;
|
|
} ExplorerItem;
|
|
|
|
typedef enum {
|
|
ACTION_NONE,
|
|
ACTION_CREATE_FILE,
|
|
ACTION_CREATE_FOLDER,
|
|
ACTION_DELETE_FILE,
|
|
ACTION_DELETE_FOLDER
|
|
} ContextMenuAction;
|
|
|
|
static ExplorerItem items[EXPLORER_MAX_FILES];
|
|
static int item_count = 0;
|
|
static int selected_item = -1;
|
|
static char current_path[256] = "/";
|
|
static int last_clicked_item = -1;
|
|
static uint32_t last_click_time = 0;
|
|
|
|
// Dialog state
|
|
static int dialog_state = DIALOG_NONE;
|
|
static char dialog_input[DIALOG_INPUT_MAX] = "";
|
|
static int dialog_input_cursor = 0;
|
|
static char dialog_target_path[256] = ""; // For delete confirmations
|
|
static bool dialog_target_is_dir = false; // For delete confirmations
|
|
|
|
// Dropdown menu state
|
|
static bool dropdown_menu_visible = false;
|
|
static int dropdown_menu_item_height = 25;
|
|
#define DROPDOWN_MENU_WIDTH 120
|
|
#define DROPDOWN_MENU_ITEMS 3
|
|
|
|
// File context menu state
|
|
static bool file_context_menu_visible = false;
|
|
static int file_context_menu_x = 0;
|
|
static int file_context_menu_y = 0;
|
|
static int file_context_menu_item = -1; // Which item is being right-clicked
|
|
#define FILE_CONTEXT_MENU_WIDTH 140
|
|
#define FILE_CONTEXT_MENU_HEIGHT 50
|
|
#define FILE_CONTEXT_ITEMS 2 // "Open with Text Editor" and "Open with Markdown Viewer"
|
|
|
|
// === Helper Functions ===
|
|
|
|
static size_t explorer_strlen(const char *str);
|
|
static void explorer_strcpy(char *dest, const char *src);
|
|
static int explorer_strcmp(const char *s1, const char *s2);
|
|
static void explorer_strcat(char *dest, const char *src);
|
|
static void explorer_load_directory(const char *path);
|
|
static void explorer_handle_right_click(Window *win, int x, int y);
|
|
static void explorer_handle_file_context_menu_click(Window *win, int x, int y);
|
|
|
|
static size_t explorer_strlen(const char *str) {
|
|
size_t len = 0;
|
|
while (str[len]) len++;
|
|
return len;
|
|
}
|
|
|
|
static void explorer_strcpy(char *dest, const char *src) {
|
|
while (*src) *dest++ = *src++;
|
|
*dest = 0;
|
|
}
|
|
|
|
static int explorer_strcmp(const char *s1, const char *s2) {
|
|
while (*s1 && (*s1 == *s2)) {
|
|
s1++;
|
|
s2++;
|
|
}
|
|
return *(const unsigned char*)s1 - *(const unsigned char*)s2;
|
|
}
|
|
|
|
static void explorer_strcat(char *dest, const char *src) {
|
|
while (*dest) dest++;
|
|
explorer_strcpy(dest, src);
|
|
}
|
|
|
|
// Get file extension (e.g., "md" from "file.md")
|
|
static const char* explorer_get_extension(const char *filename) {
|
|
const char *dot = filename;
|
|
const char *ext = "";
|
|
|
|
// Find the last dot
|
|
while (*dot) {
|
|
if (*dot == '.') {
|
|
ext = dot + 1;
|
|
}
|
|
dot++;
|
|
}
|
|
|
|
return ext;
|
|
}
|
|
|
|
// Check if file is markdown
|
|
static bool explorer_is_markdown_file(const char *filename) {
|
|
const char *ext = explorer_get_extension(filename);
|
|
return explorer_strcmp(ext, "md") == 0;
|
|
}
|
|
|
|
// === Dialog and File Operations ===
|
|
|
|
static void dialog_open_create_file(void) {
|
|
dialog_state = DIALOG_CREATE_FILE;
|
|
dialog_input[0] = 0;
|
|
dialog_input_cursor = 0;
|
|
}
|
|
|
|
static void dialog_open_create_folder(void) {
|
|
dialog_state = DIALOG_CREATE_FOLDER;
|
|
dialog_input[0] = 0;
|
|
dialog_input_cursor = 0;
|
|
}
|
|
|
|
static void dialog_open_delete_confirm(int item_idx) {
|
|
if (item_idx < 0 || item_idx >= item_count) return;
|
|
|
|
dialog_state = DIALOG_DELETE_CONFIRM;
|
|
dialog_target_is_dir = items[item_idx].is_directory;
|
|
|
|
// Build full path to target
|
|
explorer_strcpy(dialog_target_path, current_path);
|
|
if (dialog_target_path[explorer_strlen(dialog_target_path) - 1] != '/') {
|
|
explorer_strcat(dialog_target_path, "/");
|
|
}
|
|
explorer_strcat(dialog_target_path, items[item_idx].name);
|
|
}
|
|
|
|
static void dialog_close(void) {
|
|
dialog_state = DIALOG_NONE;
|
|
dialog_input[0] = 0;
|
|
dialog_input_cursor = 0;
|
|
dialog_target_path[0] = 0;
|
|
}
|
|
|
|
static void dialog_confirm_create_file(void) {
|
|
if (dialog_input[0] == 0) return;
|
|
|
|
char full_path[256];
|
|
explorer_strcpy(full_path, current_path);
|
|
if (full_path[explorer_strlen(full_path) - 1] != '/') {
|
|
explorer_strcat(full_path, "/");
|
|
}
|
|
explorer_strcat(full_path, dialog_input);
|
|
|
|
// Create empty file
|
|
FAT32_FileHandle *file = fat32_open(full_path, "w");
|
|
if (file) {
|
|
fat32_close(file);
|
|
explorer_load_directory(current_path);
|
|
}
|
|
|
|
dialog_close();
|
|
}
|
|
|
|
static void dialog_confirm_create_folder(void) {
|
|
if (dialog_input[0] == 0) return;
|
|
|
|
char full_path[256];
|
|
explorer_strcpy(full_path, current_path);
|
|
if (full_path[explorer_strlen(full_path) - 1] != '/') {
|
|
explorer_strcat(full_path, "/");
|
|
}
|
|
explorer_strcat(full_path, dialog_input);
|
|
|
|
// Create directory
|
|
if (fat32_mkdir(full_path)) {
|
|
explorer_load_directory(current_path);
|
|
}
|
|
|
|
dialog_close();
|
|
}
|
|
|
|
// Recursive delete for directories
|
|
static bool explorer_delete_recursive(const char *path) {
|
|
if (fat32_is_directory(path)) {
|
|
// List contents and delete recursively
|
|
FAT32_FileInfo entries[64];
|
|
int count = fat32_list_directory(path, entries, 64);
|
|
|
|
for (int i = 0; i < count; i++) {
|
|
char child_path[256];
|
|
explorer_strcpy(child_path, path);
|
|
if (child_path[explorer_strlen(child_path) - 1] != '/') {
|
|
explorer_strcat(child_path, "/");
|
|
}
|
|
explorer_strcat(child_path, entries[i].name);
|
|
|
|
if (entries[i].is_directory) {
|
|
explorer_delete_recursive(child_path);
|
|
} else {
|
|
fat32_delete(child_path);
|
|
}
|
|
}
|
|
// Delete the directory itself
|
|
return fat32_rmdir(path);
|
|
} else {
|
|
// Regular file
|
|
return fat32_delete(path);
|
|
}
|
|
}
|
|
|
|
static void dialog_confirm_delete(void) {
|
|
explorer_delete_recursive(dialog_target_path);
|
|
explorer_load_directory(current_path);
|
|
dialog_close();
|
|
}
|
|
|
|
static void dropdown_menu_toggle(void) {
|
|
dropdown_menu_visible = !dropdown_menu_visible;
|
|
}
|
|
|
|
// === Helper Functions (continued)
|
|
|
|
// === Explorer Logic ===
|
|
|
|
static uint32_t explorer_get_folder_color(const char *folder_path) {
|
|
char color_file_path[256];
|
|
explorer_strcpy(color_file_path, folder_path);
|
|
if (color_file_path[explorer_strlen(color_file_path) - 1] != '/') {
|
|
explorer_strcat(color_file_path, "/");
|
|
}
|
|
explorer_strcat(color_file_path, ".color");
|
|
|
|
FAT32_FileHandle *file = fat32_open(color_file_path, "r");
|
|
if (file) {
|
|
uint32_t color = 0;
|
|
int bytes_read = fat32_read(file, &color, sizeof(uint32_t));
|
|
fat32_close(file);
|
|
if (bytes_read == sizeof(uint32_t)) {
|
|
return color;
|
|
}
|
|
}
|
|
return COLOR_APPLE_YELLOW;
|
|
}
|
|
|
|
static void explorer_set_folder_color(const char *folder_path, uint32_t color) {
|
|
char color_file_path[256];
|
|
explorer_strcpy(color_file_path, folder_path);
|
|
if (color_file_path[explorer_strlen(color_file_path) - 1] != '/') {
|
|
explorer_strcat(color_file_path, "/");
|
|
}
|
|
explorer_strcat(color_file_path, ".color");
|
|
|
|
FAT32_FileHandle *file = fat32_open(color_file_path, "w");
|
|
if (file) {
|
|
fat32_write(file, &color, sizeof(uint32_t));
|
|
fat32_close(file);
|
|
}
|
|
}
|
|
|
|
static void explorer_load_directory(const char *path) {
|
|
explorer_strcpy(current_path, path);
|
|
|
|
FAT32_FileInfo entries[EXPLORER_MAX_FILES];
|
|
int count = fat32_list_directory(path, entries, EXPLORER_MAX_FILES);
|
|
|
|
item_count = 0;
|
|
for (int i = 0; i < count && i < EXPLORER_MAX_FILES; i++) {
|
|
// Skip .color files
|
|
if (explorer_strcmp(entries[i].name, ".color") == 0) {
|
|
continue;
|
|
}
|
|
|
|
explorer_strcpy(items[item_count].name, entries[i].name);
|
|
items[item_count].is_directory = entries[i].is_directory;
|
|
items[item_count].size = entries[i].size;
|
|
|
|
if (items[item_count].is_directory) {
|
|
char subfolder_path[256];
|
|
explorer_strcpy(subfolder_path, current_path);
|
|
if (subfolder_path[explorer_strlen(subfolder_path) - 1] != '/') {
|
|
explorer_strcat(subfolder_path, "/");
|
|
}
|
|
explorer_strcat(subfolder_path, items[item_count].name);
|
|
items[item_count].color = explorer_get_folder_color(subfolder_path);
|
|
} else {
|
|
items[item_count].color = COLOR_APPLE_YELLOW;
|
|
}
|
|
item_count++;
|
|
}
|
|
|
|
selected_item = -1;
|
|
}
|
|
|
|
static void explorer_navigate_to(const char *dirname) {
|
|
char new_path[256];
|
|
|
|
if (explorer_strcmp(dirname, "..") == 0) {
|
|
// Go to parent directory
|
|
int len = explorer_strlen(current_path);
|
|
int i = len - 1;
|
|
|
|
// Skip trailing slashes
|
|
while (i > 0 && current_path[i] == '/') i--;
|
|
|
|
// Find last slash
|
|
while (i > 0 && current_path[i] != '/') i--;
|
|
|
|
if (i == 0) {
|
|
explorer_strcpy(new_path, "/");
|
|
} else {
|
|
for (int j = 0; j < i; j++) {
|
|
new_path[j] = current_path[j];
|
|
}
|
|
new_path[i] = 0;
|
|
}
|
|
} else {
|
|
// Go to subdirectory
|
|
explorer_strcpy(new_path, current_path);
|
|
if (new_path[explorer_strlen(new_path) - 1] != '/') {
|
|
explorer_strcat(new_path, "/");
|
|
}
|
|
explorer_strcat(new_path, dirname);
|
|
}
|
|
|
|
explorer_load_directory(new_path);
|
|
}
|
|
|
|
// Draw a simple file icon
|
|
static void explorer_draw_file_icon(int x, int y, bool is_dir, uint32_t color) {
|
|
if (is_dir) {
|
|
// Folder icon (colored folder) - Desktop style
|
|
// Folder tab
|
|
draw_rect(x + 10, y + 10, 15, 6, COLOR_LTGRAY);
|
|
draw_rect(x + 10, y + 10, 15, 1, COLOR_BLACK);
|
|
draw_rect(x + 10, y + 10, 1, 6, COLOR_BLACK);
|
|
draw_rect(x + 24, y + 10, 1, 6, COLOR_BLACK);
|
|
|
|
// Folder body
|
|
draw_rect(x + 10, y + 16, 25, 15, color);
|
|
draw_rect(x + 10, y + 16, 25, 1, COLOR_BLACK);
|
|
draw_rect(x + 10, y + 16, 1, 15, COLOR_BLACK);
|
|
draw_rect(x + 34, y + 16, 1, 15, COLOR_BLACK);
|
|
draw_rect(x + 10, y + 30, 25, 1, COLOR_BLACK);
|
|
} else {
|
|
// Document icon - larger
|
|
draw_rect(x + 12, y + 10, 20, 25, COLOR_WHITE);
|
|
draw_rect(x + 12, y + 10, 20, 2, COLOR_BLACK);
|
|
draw_rect(x + 12, y + 10, 2, 25, COLOR_BLACK);
|
|
draw_rect(x + 30, y + 10, 2, 25, COLOR_BLACK);
|
|
draw_rect(x + 12, y + 33, 20, 2, COLOR_BLACK);
|
|
// Lines on document
|
|
draw_rect(x + 15, y + 18, 14, 1, COLOR_DKGRAY);
|
|
draw_rect(x + 15, y + 23, 14, 1, COLOR_DKGRAY);
|
|
draw_rect(x + 15, y + 28, 14, 1, COLOR_DKGRAY);
|
|
}
|
|
}
|
|
|
|
// === Paint Function ===
|
|
|
|
static void explorer_paint(Window *win) {
|
|
int offset_x = win->x + 4;
|
|
int offset_y = win->y + 24;
|
|
|
|
// Fill background
|
|
draw_rect(offset_x, offset_y, win->w - 8, win->h - 28, COLOR_LTGRAY);
|
|
|
|
// Draw path bar
|
|
int path_height = 30;
|
|
draw_bevel_rect(offset_x + 4, offset_y + 4, win->w - 16, path_height, true);
|
|
draw_string(offset_x + 10, offset_y + 10, "Path: ", COLOR_BLACK);
|
|
draw_string(offset_x + 50, offset_y + 10, current_path, COLOR_BLACK);
|
|
|
|
// Draw dropdown menu button (right-aligned, before back button)
|
|
int dropdown_btn_x = win->x + win->w - 90;
|
|
draw_button(dropdown_btn_x, offset_y + 4, 35, 30, "...", false);
|
|
|
|
// Draw back button (right-aligned)
|
|
draw_button(win->x + win->w - 40, offset_y + 4, 30, 30, "<", false);
|
|
|
|
// Draw dropdown menu if visible
|
|
if (dropdown_menu_visible) {
|
|
int menu_x = dropdown_btn_x;
|
|
int menu_y = offset_y + 34;
|
|
|
|
// Draw menu background
|
|
draw_rect(menu_x, menu_y, DROPDOWN_MENU_WIDTH, dropdown_menu_item_height * DROPDOWN_MENU_ITEMS, COLOR_LTGRAY);
|
|
draw_bevel_rect(menu_x, menu_y, DROPDOWN_MENU_WIDTH, dropdown_menu_item_height * DROPDOWN_MENU_ITEMS, true);
|
|
|
|
// Draw menu items
|
|
draw_string(menu_x + 8, menu_y + 5, "New File", COLOR_BLACK);
|
|
draw_string(menu_x + 8, menu_y + dropdown_menu_item_height + 5, "New Folder", COLOR_BLACK);
|
|
draw_string(menu_x + 8, menu_y + dropdown_menu_item_height * 2 + 5, "Delete", COLOR_RED);
|
|
}
|
|
|
|
// Draw file list
|
|
int content_start_y = offset_y + 40;
|
|
|
|
for (int i = 0; i < item_count; i++) {
|
|
int row = i / EXPLORER_COLS;
|
|
int col = i % EXPLORER_COLS;
|
|
|
|
int item_x = offset_x + 10 + (col * (EXPLORER_ITEM_WIDTH + EXPLORER_PADDING));
|
|
int item_y = content_start_y + (row * (EXPLORER_ITEM_HEIGHT + EXPLORER_PADDING));
|
|
|
|
// Draw item background
|
|
uint32_t bg_color = (i == selected_item) ? COLOR_BLUE : COLOR_WHITE;
|
|
draw_bevel_rect(item_x, item_y, EXPLORER_ITEM_WIDTH, EXPLORER_ITEM_HEIGHT, false);
|
|
draw_rect(item_x + 2, item_y + 2, EXPLORER_ITEM_WIDTH - 4, EXPLORER_ITEM_HEIGHT - 4, bg_color);
|
|
|
|
// Draw icon (larger area)
|
|
explorer_draw_file_icon(item_x + 5, item_y + 5, items[i].is_directory, items[i].color);
|
|
|
|
// Draw name below icon with text wrapping
|
|
uint32_t text_color = (i == selected_item) ? COLOR_WHITE : COLOR_BLACK;
|
|
int name_len = explorer_strlen(items[i].name);
|
|
int text_x = item_x + 5;
|
|
int text_y = item_y + 50;
|
|
int max_name_width = EXPLORER_ITEM_WIDTH - 10; // 110 pixels available for text
|
|
int chars_per_line = max_name_width / 8; // 8 pixels per character
|
|
|
|
// Draw wrapped filename
|
|
int line_offset = 0;
|
|
char line_buffer[25];
|
|
for (int j = 0; j < name_len; j++) {
|
|
int pos_in_line = j % chars_per_line;
|
|
line_buffer[pos_in_line] = items[i].name[j];
|
|
line_buffer[pos_in_line + 1] = 0;
|
|
|
|
// Draw line when we reach end of line or end of name
|
|
if (pos_in_line == chars_per_line - 1 || j == name_len - 1) {
|
|
draw_string(text_x, text_y + (line_offset * 10), line_buffer, text_color);
|
|
line_offset++;
|
|
}
|
|
}
|
|
}
|
|
|
|
// Draw dialogs
|
|
if (dialog_state == DIALOG_CREATE_FILE) {
|
|
int dlg_x = win->x + win->w / 2 - 150;
|
|
int dlg_y = win->y + win->h / 2 - 60;
|
|
|
|
// Dialog background
|
|
draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY);
|
|
draw_bevel_rect(dlg_x, dlg_y, 300, 110, true);
|
|
|
|
// Title
|
|
draw_string(dlg_x + 10, dlg_y + 10, "Create New File", COLOR_BLACK);
|
|
|
|
// Input field
|
|
draw_bevel_rect(dlg_x + 10, dlg_y + 35, 280, 20, false);
|
|
draw_string(dlg_x + 15, dlg_y + 40, dialog_input, COLOR_BLACK);
|
|
draw_string(dlg_x + 15 + dialog_input_cursor * 8, dlg_y + 40, "|", COLOR_BLACK);
|
|
|
|
// Buttons
|
|
draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Create", false);
|
|
draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false);
|
|
} else if (dialog_state == DIALOG_CREATE_FOLDER) {
|
|
int dlg_x = win->x + win->w / 2 - 150;
|
|
int dlg_y = win->y + win->h / 2 - 60;
|
|
|
|
// Dialog background
|
|
draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY);
|
|
draw_bevel_rect(dlg_x, dlg_y, 300, 110, true);
|
|
|
|
// Title
|
|
draw_string(dlg_x + 10, dlg_y + 10, "Create New Folder", COLOR_BLACK);
|
|
|
|
// Input field
|
|
draw_bevel_rect(dlg_x + 10, dlg_y + 35, 280, 20, false);
|
|
draw_string(dlg_x + 15, dlg_y + 40, dialog_input, COLOR_BLACK);
|
|
draw_string(dlg_x + 15 + dialog_input_cursor * 8, dlg_y + 40, "|", COLOR_BLACK);
|
|
|
|
// Buttons
|
|
draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Create", false);
|
|
draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false);
|
|
} else if (dialog_state == DIALOG_DELETE_CONFIRM) {
|
|
int dlg_x = win->x + win->w / 2 - 150;
|
|
int dlg_y = win->y + win->h / 2 - 60;
|
|
|
|
// Dialog background
|
|
draw_rect(dlg_x - 5, dlg_y - 5, 310, 120, COLOR_LTGRAY);
|
|
draw_bevel_rect(dlg_x, dlg_y, 300, 110, true);
|
|
|
|
// Title
|
|
const char *title = dialog_target_is_dir ? "Delete Folder?" : "Delete File?";
|
|
draw_string(dlg_x + 10, dlg_y + 10, title, COLOR_BLACK);
|
|
|
|
// Message
|
|
draw_string(dlg_x + 10, dlg_y + 35, "This action cannot be undone.", COLOR_BLACK);
|
|
|
|
// Buttons
|
|
draw_button(dlg_x + 50, dlg_y + 65, 80, 25, "Delete", false);
|
|
draw_button(dlg_x + 170, dlg_y + 65, 80, 25, "Cancel", false);
|
|
}
|
|
|
|
// Draw file context menu if visible
|
|
if (file_context_menu_visible && file_context_menu_item >= 0) {
|
|
// Convert window-relative coordinates to screen coordinates for drawing
|
|
int menu_screen_x = win->x + file_context_menu_x;
|
|
int menu_screen_y = win->y + file_context_menu_y;
|
|
|
|
if (items[file_context_menu_item].is_directory) {
|
|
// Folder context menu (Color selection)
|
|
int menu_height = 25 * 5; // 5 items, 25px each
|
|
|
|
// Draw menu background
|
|
draw_rect(menu_screen_x, menu_screen_y, FILE_CONTEXT_MENU_WIDTH, menu_height, COLOR_LTGRAY);
|
|
draw_bevel_rect(menu_screen_x, menu_screen_y, FILE_CONTEXT_MENU_WIDTH, menu_height, true);
|
|
|
|
// Draw menu items
|
|
int item_h = 25;
|
|
draw_string(menu_screen_x + 5, menu_screen_y + 5, "Blue", COLOR_APPLE_BLUE);
|
|
draw_string(menu_screen_x + 5, menu_screen_y + item_h + 5, "Red", COLOR_RED);
|
|
draw_string(menu_screen_x + 5, menu_screen_y + item_h * 2 + 5, "Yellow", COLOR_APPLE_YELLOW); // Text might be hard to read, but requested
|
|
draw_string(menu_screen_x + 5, menu_screen_y + item_h * 3 + 5, "Green", COLOR_APPLE_GREEN);
|
|
draw_string(menu_screen_x + 5, menu_screen_y + item_h * 4 + 5, "Black", COLOR_BLACK);
|
|
} else {
|
|
// File context menu
|
|
// Draw menu background
|
|
draw_rect(menu_screen_x, menu_screen_y, FILE_CONTEXT_MENU_WIDTH, FILE_CONTEXT_MENU_HEIGHT, COLOR_LTGRAY);
|
|
draw_bevel_rect(menu_screen_x, menu_screen_y, FILE_CONTEXT_MENU_WIDTH, FILE_CONTEXT_MENU_HEIGHT, true);
|
|
|
|
// Draw menu items
|
|
int item_height = FILE_CONTEXT_MENU_HEIGHT / FILE_CONTEXT_ITEMS;
|
|
|
|
// Item 1: "Open with Text Editor"
|
|
draw_string(menu_screen_x + 5, menu_screen_y + 5, "Open w/ Editor", COLOR_BLACK);
|
|
|
|
// Item 2: "Open with Markdown Viewer" (only show if file is .md)
|
|
if (explorer_is_markdown_file(items[file_context_menu_item].name)) {
|
|
draw_string(menu_screen_x + 5, menu_screen_y + item_height + 5, "Open w/ Markdown", COLOR_BLACK);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// === Mouse Handler ===
|
|
|
|
static void explorer_handle_click(Window *win, int x, int y) {
|
|
// Handle file context menu clicks first
|
|
if (file_context_menu_visible) {
|
|
explorer_handle_file_context_menu_click(win, x, y);
|
|
return;
|
|
}
|
|
|
|
// Handle dialog clicks
|
|
if (dialog_state == DIALOG_CREATE_FILE || dialog_state == DIALOG_CREATE_FOLDER) {
|
|
int dlg_x = win->w / 2 - 150;
|
|
int dlg_y = win->h / 2 - 60;
|
|
|
|
// Create button
|
|
if (x >= dlg_x + 50 && x < dlg_x + 130 &&
|
|
y >= dlg_y + 65 && y < dlg_y + 90) {
|
|
if (dialog_state == DIALOG_CREATE_FILE) {
|
|
dialog_confirm_create_file();
|
|
} else {
|
|
dialog_confirm_create_folder();
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Cancel button
|
|
if (x >= dlg_x + 170 && x < dlg_x + 250 &&
|
|
y >= dlg_y + 65 && y < dlg_y + 90) {
|
|
dialog_close();
|
|
return;
|
|
}
|
|
|
|
// Input field click
|
|
if (x >= dlg_x + 10 && x < dlg_x + 290 &&
|
|
y >= dlg_y + 35 && y < dlg_y + 55) {
|
|
dialog_input_cursor = (x - dlg_x - 15) / 8;
|
|
if (dialog_input_cursor > (int)explorer_strlen(dialog_input)) {
|
|
dialog_input_cursor = explorer_strlen(dialog_input);
|
|
}
|
|
return;
|
|
}
|
|
} else if (dialog_state == DIALOG_DELETE_CONFIRM) {
|
|
int dlg_x = win->w / 2 - 150;
|
|
int dlg_y = win->h / 2 - 60;
|
|
|
|
// Delete button
|
|
if (x >= dlg_x + 50 && x < dlg_x + 130 &&
|
|
y >= dlg_y + 65 && y < dlg_y + 90) {
|
|
dialog_confirm_delete();
|
|
return;
|
|
}
|
|
|
|
// Cancel button
|
|
if (x >= dlg_x + 170 && x < dlg_x + 250 &&
|
|
y >= dlg_y + 65 && y < dlg_y + 90) {
|
|
dialog_close();
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Handle dropdown menu clicks
|
|
if (dropdown_menu_visible) {
|
|
int dropdown_btn_x = win->w - 90; // Window-relative
|
|
int menu_y = 58; // Window-relative (offset_y + 34, where offset_y = 24)
|
|
|
|
// New File
|
|
if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH &&
|
|
y >= menu_y && y < menu_y + dropdown_menu_item_height) {
|
|
dropdown_menu_toggle();
|
|
dialog_open_create_file();
|
|
return;
|
|
}
|
|
|
|
// New Folder
|
|
if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH &&
|
|
y >= menu_y + dropdown_menu_item_height &&
|
|
y < menu_y + dropdown_menu_item_height * 2) {
|
|
dropdown_menu_toggle();
|
|
dialog_open_create_folder();
|
|
return;
|
|
}
|
|
|
|
// Delete
|
|
if (x >= dropdown_btn_x && x < dropdown_btn_x + DROPDOWN_MENU_WIDTH &&
|
|
y >= menu_y + dropdown_menu_item_height * 2 &&
|
|
y < menu_y + dropdown_menu_item_height * 3) {
|
|
dropdown_menu_toggle();
|
|
if (selected_item >= 0) {
|
|
dialog_open_delete_confirm(selected_item);
|
|
}
|
|
return;
|
|
}
|
|
|
|
// Click outside menu closes it
|
|
dropdown_menu_toggle();
|
|
return;
|
|
}
|
|
|
|
// x, y are already relative to window (0,0 is top-left of window content area)
|
|
// Check dropdown menu button
|
|
int button_y = 28; // Position from top of window title bar
|
|
if (x >= win->w - 90 && x < win->w - 55 &&
|
|
y >= button_y && y < button_y + 30) {
|
|
// Dropdown menu button clicked
|
|
dropdown_menu_toggle();
|
|
return;
|
|
}
|
|
|
|
// Check back button (right-aligned)
|
|
if (x >= win->w - 40 && x < win->w - 10 &&
|
|
y >= button_y && y < button_y + 30) {
|
|
// Back button clicked
|
|
explorer_navigate_to("..");
|
|
return;
|
|
}
|
|
|
|
// File items start at y=64 relative to window
|
|
int content_start_y = 64;
|
|
int offset_x = 4;
|
|
|
|
for (int i = 0; i < item_count; i++) {
|
|
int row = i / EXPLORER_COLS;
|
|
int col = i % EXPLORER_COLS;
|
|
|
|
int item_x = offset_x + 10 + (col * (EXPLORER_ITEM_WIDTH + EXPLORER_PADDING));
|
|
int item_y = content_start_y + (row * (EXPLORER_ITEM_HEIGHT + EXPLORER_PADDING));
|
|
|
|
if (x >= item_x && x < item_x + EXPLORER_ITEM_WIDTH &&
|
|
y >= item_y && y < item_y + EXPLORER_ITEM_HEIGHT) {
|
|
|
|
// Check for double-click
|
|
if (last_clicked_item == i) {
|
|
// Double-click detected
|
|
if (items[i].is_directory) {
|
|
explorer_navigate_to(items[i].name);
|
|
} else {
|
|
// Open file - check type
|
|
char full_path[256];
|
|
explorer_strcpy(full_path, current_path);
|
|
if (full_path[explorer_strlen(full_path) - 1] != '/') {
|
|
explorer_strcat(full_path, "/");
|
|
}
|
|
explorer_strcat(full_path, items[i].name);
|
|
|
|
// Check if markdown file
|
|
if (explorer_is_markdown_file(items[i].name)) {
|
|
// Open with markdown viewer
|
|
win_markdown.visible = true;
|
|
win_markdown.focused = true;
|
|
int max_z = 0;
|
|
if (win_explorer.z_index > max_z) max_z = win_explorer.z_index;
|
|
if (win_cmd.z_index > max_z) max_z = win_cmd.z_index;
|
|
if (win_notepad.z_index > max_z) max_z = win_notepad.z_index;
|
|
if (win_calculator.z_index > max_z) max_z = win_calculator.z_index;
|
|
if (win_editor.z_index > max_z) max_z = win_editor.z_index;
|
|
win_markdown.z_index = max_z + 1;
|
|
markdown_open_file(full_path);
|
|
} else {
|
|
// Open with text editor
|
|
win_editor.visible = true;
|
|
win_editor.focused = true;
|
|
int max_z = 0;
|
|
if (win_explorer.z_index > max_z) max_z = win_explorer.z_index;
|
|
if (win_cmd.z_index > max_z) max_z = win_cmd.z_index;
|
|
if (win_notepad.z_index > max_z) max_z = win_notepad.z_index;
|
|
if (win_calculator.z_index > max_z) max_z = win_calculator.z_index;
|
|
if (win_markdown.z_index > max_z) max_z = win_markdown.z_index;
|
|
win_editor.z_index = max_z + 1;
|
|
editor_open_file(full_path);
|
|
}
|
|
}
|
|
last_clicked_item = -1;
|
|
} else {
|
|
// Single-click - select
|
|
selected_item = i;
|
|
last_clicked_item = i;
|
|
last_click_time = 0; // Reset for next click
|
|
}
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
// === Key Handler ===
|
|
|
|
static void explorer_handle_key(Window *win, char c) {
|
|
(void)win;
|
|
|
|
// Handle dialog input
|
|
if (dialog_state == DIALOG_CREATE_FILE || dialog_state == DIALOG_CREATE_FOLDER) {
|
|
if (c == 27) { // ESC - close dialog
|
|
dialog_close();
|
|
return;
|
|
} else if (c == '\n') { // ENTER - confirm
|
|
if (dialog_state == DIALOG_CREATE_FILE) {
|
|
dialog_confirm_create_file();
|
|
} else {
|
|
dialog_confirm_create_folder();
|
|
}
|
|
return;
|
|
} else if (c == 8 || c == 127) { // BACKSPACE
|
|
if (dialog_input_cursor > 0) {
|
|
dialog_input_cursor--;
|
|
// Shift characters
|
|
for (int i = dialog_input_cursor; i < (int)explorer_strlen(dialog_input); i++) {
|
|
dialog_input[i] = dialog_input[i + 1];
|
|
}
|
|
}
|
|
return;
|
|
} else if (c >= 32 && c < 127) { // Printable character
|
|
int len = explorer_strlen(dialog_input);
|
|
if (len < DIALOG_INPUT_MAX - 1) {
|
|
// Shift characters to make room
|
|
for (int i = len; i >= dialog_input_cursor; i--) {
|
|
dialog_input[i + 1] = dialog_input[i];
|
|
}
|
|
dialog_input[dialog_input_cursor] = c;
|
|
dialog_input_cursor++;
|
|
}
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (dialog_state == DIALOG_DELETE_CONFIRM) {
|
|
if (c == 27) { // ESC
|
|
dialog_close();
|
|
return;
|
|
}
|
|
return;
|
|
}
|
|
|
|
if (c == 'q' || c == 'Q') {
|
|
win->visible = false;
|
|
return;
|
|
}
|
|
|
|
// Close dropdown menu if open with ESC
|
|
if (dropdown_menu_visible && c == 27) {
|
|
dropdown_menu_toggle();
|
|
return;
|
|
}
|
|
|
|
if (c == 17) { // UP
|
|
if (selected_item > 0) {
|
|
selected_item -= EXPLORER_COLS;
|
|
if (selected_item < 0) selected_item = 0;
|
|
}
|
|
} else if (c == 18) { // DOWN
|
|
if (selected_item < item_count - 1) {
|
|
selected_item += EXPLORER_COLS;
|
|
if (selected_item >= item_count) selected_item = item_count - 1;
|
|
}
|
|
} else if (c == 19) { // LEFT
|
|
if (selected_item > 0) {
|
|
selected_item--;
|
|
}
|
|
} else if (c == 20) { // RIGHT
|
|
if (selected_item < item_count - 1) {
|
|
selected_item++;
|
|
}
|
|
} else if (c == '\n') { // ENTER
|
|
if (selected_item >= 0 && selected_item < item_count) {
|
|
if (items[selected_item].is_directory) {
|
|
explorer_navigate_to(items[selected_item].name);
|
|
}
|
|
}
|
|
} else if (c == 'd' || c == 'D') { // Delete key
|
|
if (selected_item >= 0) {
|
|
dialog_open_delete_confirm(selected_item);
|
|
}
|
|
} else if (c == 'n' || c == 'N') { // New file
|
|
dialog_open_create_file();
|
|
} else if (c == 'f' || c == 'F') { // New folder
|
|
dialog_open_create_folder();
|
|
}
|
|
}
|
|
|
|
// === Right-Click Handler ===
|
|
|
|
static void explorer_handle_right_click(Window *win, int x, int y) {
|
|
// File items start at y=64 relative to window
|
|
int content_start_y = 64;
|
|
int offset_x = 4;
|
|
|
|
for (int i = 0; i < item_count; i++) {
|
|
int row = i / EXPLORER_COLS;
|
|
int col = i % EXPLORER_COLS;
|
|
|
|
int item_x = offset_x + 10 + (col * (EXPLORER_ITEM_WIDTH + EXPLORER_PADDING));
|
|
int item_y = content_start_y + (row * (EXPLORER_ITEM_HEIGHT + EXPLORER_PADDING));
|
|
|
|
if (x >= item_x && x < item_x + EXPLORER_ITEM_WIDTH &&
|
|
y >= item_y && y < item_y + EXPLORER_ITEM_HEIGHT) {
|
|
|
|
// Right-click on a file or folder item
|
|
// Show context menu
|
|
file_context_menu_visible = true;
|
|
file_context_menu_item = i;
|
|
file_context_menu_x = x;
|
|
file_context_menu_y = y;
|
|
return;
|
|
}
|
|
}
|
|
|
|
// Close menu if clicking elsewhere
|
|
file_context_menu_visible = false;
|
|
file_context_menu_item = -1;
|
|
}
|
|
|
|
static void explorer_handle_file_context_menu_click(Window *win, int x, int y) {
|
|
(void)win; // Suppress unused warning - we use absolute coordinates instead
|
|
|
|
if (!file_context_menu_visible || file_context_menu_item < 0) {
|
|
return;
|
|
}
|
|
|
|
// Adjust coordinates to be relative to context menu
|
|
int relative_x = x - file_context_menu_x;
|
|
int relative_y = y - file_context_menu_y;
|
|
|
|
int menu_height;
|
|
if (items[file_context_menu_item].is_directory) {
|
|
menu_height = 25 * 5;
|
|
} else {
|
|
menu_height = FILE_CONTEXT_MENU_HEIGHT;
|
|
}
|
|
|
|
if (relative_x < 0 || relative_x > FILE_CONTEXT_MENU_WIDTH ||
|
|
relative_y < 0 || relative_y > menu_height) {
|
|
// Clicked outside menu - close it
|
|
file_context_menu_visible = false;
|
|
file_context_menu_item = -1;
|
|
return;
|
|
}
|
|
|
|
if (items[file_context_menu_item].is_directory) {
|
|
int clicked_item = relative_y / 25;
|
|
uint32_t new_color = items[file_context_menu_item].color;
|
|
|
|
if (clicked_item == 0) new_color = COLOR_APPLE_BLUE;
|
|
else if (clicked_item == 1) new_color = COLOR_RED;
|
|
else if (clicked_item == 2) new_color = COLOR_APPLE_YELLOW;
|
|
else if (clicked_item == 3) new_color = COLOR_APPLE_GREEN;
|
|
else if (clicked_item == 4) new_color = COLOR_BLACK;
|
|
|
|
items[file_context_menu_item].color = new_color;
|
|
|
|
// Save to file
|
|
char full_path[256];
|
|
explorer_strcpy(full_path, current_path);
|
|
if (full_path[explorer_strlen(full_path) - 1] != '/') {
|
|
explorer_strcat(full_path, "/");
|
|
}
|
|
explorer_strcat(full_path, items[file_context_menu_item].name);
|
|
explorer_set_folder_color(full_path, new_color);
|
|
} else {
|
|
int item_height = FILE_CONTEXT_MENU_HEIGHT / FILE_CONTEXT_ITEMS;
|
|
int clicked_item = relative_y / item_height;
|
|
|
|
// Build full path
|
|
char full_path[256];
|
|
explorer_strcpy(full_path, current_path);
|
|
if (full_path[explorer_strlen(full_path) - 1] != '/') {
|
|
explorer_strcat(full_path, "/");
|
|
}
|
|
explorer_strcat(full_path, items[file_context_menu_item].name);
|
|
|
|
if (clicked_item == 0) {
|
|
// "Open with Text Editor"
|
|
win_editor.visible = true;
|
|
win_editor.focused = true;
|
|
int max_z = 0;
|
|
if (win_explorer.z_index > max_z) max_z = win_explorer.z_index;
|
|
if (win_cmd.z_index > max_z) max_z = win_cmd.z_index;
|
|
if (win_notepad.z_index > max_z) max_z = win_notepad.z_index;
|
|
if (win_calculator.z_index > max_z) max_z = win_calculator.z_index;
|
|
if (win_markdown.z_index > max_z) max_z = win_markdown.z_index;
|
|
win_editor.z_index = max_z + 1;
|
|
editor_open_file(full_path);
|
|
} else if (clicked_item == 1 && explorer_is_markdown_file(items[file_context_menu_item].name)) {
|
|
// "Open with Markdown Viewer"
|
|
win_markdown.visible = true;
|
|
win_markdown.focused = true;
|
|
int max_z = 0;
|
|
if (win_explorer.z_index > max_z) max_z = win_explorer.z_index;
|
|
if (win_cmd.z_index > max_z) max_z = win_cmd.z_index;
|
|
if (win_notepad.z_index > max_z) max_z = win_notepad.z_index;
|
|
if (win_calculator.z_index > max_z) max_z = win_calculator.z_index;
|
|
if (win_editor.z_index > max_z) max_z = win_editor.z_index;
|
|
win_markdown.z_index = max_z + 1;
|
|
markdown_open_file(full_path);
|
|
}
|
|
}
|
|
|
|
file_context_menu_visible = false;
|
|
file_context_menu_item = -1;
|
|
}
|
|
|
|
// === Initialization ===
|
|
|
|
void explorer_init(void) {
|
|
win_explorer.title = "File Explorer";
|
|
win_explorer.x = 300;
|
|
win_explorer.y = 100;
|
|
win_explorer.w = 600;
|
|
win_explorer.h = 400;
|
|
win_explorer.visible = false;
|
|
win_explorer.focused = false;
|
|
win_explorer.z_index = 0;
|
|
win_explorer.paint = explorer_paint;
|
|
win_explorer.handle_key = explorer_handle_key;
|
|
win_explorer.handle_click = explorer_handle_click;
|
|
win_explorer.handle_right_click = explorer_handle_right_click;
|
|
|
|
explorer_load_directory("/");
|
|
}
|
|
void explorer_reset(void) {
|
|
// Reset explorer to root directory on close/reopen
|
|
explorer_load_directory("/");
|
|
win_explorer.focused = false;
|
|
} |