#include "Tactility/app/fileselection/State.h" #include #include "Tactility/hal/sdcard/SdCardDevice.h" #include #include #include #include #include #include #include #define TAG "fileselection_app" namespace tt::app::fileselection { State::State() { if (kernel::getPlatform() == kernel::PlatformSimulator) { char cwd[PATH_MAX]; if (getcwd(cwd, sizeof(cwd)) != nullptr) { setEntriesForPath(cwd); } else { TT_LOG_E(TAG, "Failed to get current work directory files"); setEntriesForPath("/"); } } else { setEntriesForPath("/"); } } std::string State::getSelectedChildPath() const { return file::getChildPath(current_path, selected_child_entry); } bool State::setEntriesForPath(const std::string& path) { auto lock = mutex.asScopedLock(); if (!lock.lock(100)) { TT_LOG_E(TAG, LOG_MESSAGE_MUTEX_LOCK_FAILED_FMT, "setEntriesForPath"); return false; } TT_LOG_I(TAG, "Changing path: %s -> %s", current_path.c_str(), path.c_str()); /** * ESP32 does not have a root directory, so we have to create it manually. * We'll add the NVS Flash partitions and the binding for the sdcard. */ bool show_custom_root = (kernel::getPlatform() == kernel::PlatformEsp) && (path == "/"); if (show_custom_root) { TT_LOG_I(TAG, "Setting custom root"); dir_entries.clear(); dir_entries.push_back(dirent{ .d_ino = 0, .d_type = file::TT_DT_DIR, .d_name = SYSTEM_PARTITION_NAME }); dir_entries.push_back(dirent{ .d_ino = 1, .d_type = file::TT_DT_DIR, .d_name = DATA_PARTITION_NAME }); auto sdcards = tt::hal::findDevices(hal::Device::Type::SdCard); for (auto& sdcard : sdcards) { auto state = sdcard->getState(); if (state == hal::sdcard::SdCardDevice::State::Mounted) { auto mount_name = sdcard->getMountPath().substr(1); auto dir_entry = dirent { .d_ino = 2, .d_type = file::TT_DT_DIR, .d_name = { 0 } }; assert(mount_name.length() < sizeof(dirent::d_name)); strcpy(dir_entry.d_name, mount_name.c_str()); dir_entries.push_back(dir_entry); } } current_path = path; selected_child_entry = ""; return true; } else { dir_entries.clear(); int count = file::scandir(path, dir_entries, &file::direntFilterDotEntries, file::direntSortAlphaAndType); if (count >= 0) { TT_LOG_I(TAG, "%s has %u entries", path.c_str(), count); current_path = path; selected_child_entry = ""; return true; } else { TT_LOG_E(TAG, "Failed to fetch entries for %s", path.c_str()); return false; } } } bool State::setEntriesForChildPath(const std::string& childPath) { auto path = file::getChildPath(current_path, childPath); TT_LOG_I(TAG, "Navigating from %s to %s", current_path.c_str(), path.c_str()); return setEntriesForPath(path); } bool State::getDirent(uint32_t index, dirent& dirent) { auto lock = mutex.asScopedLock(); if (!lock.lock(50 / portTICK_PERIOD_MS)) { return false; } if (index < dir_entries.size()) { dirent = dir_entries[index]; return true; } else { return false; } } }