mirror of
https://github.com/ByteWelder/Tactility.git
synced 2026-02-18 10:53:17 +00:00
- GitHub actions changed to build simulator on macOS (it's broken, but at least we get a good code portability check for now!) - `Buildscripts/` shell scripts updated to use `/bin/sh` so it works on macOS too - Various includes fixed in various subprojects so the code is more portable
330 lines
11 KiB
C++
330 lines
11 KiB
C++
#include "app/alertdialog/AlertDialog.h"
|
|
#include "app/imageviewer/ImageViewer.h"
|
|
#include "app/inputdialog/InputDialog.h"
|
|
#include "app/textviewer/TextViewer.h"
|
|
#include "app/ElfApp.h"
|
|
#include "lvgl/Toolbar.h"
|
|
#include "lvgl/LvglSync.h"
|
|
#include "service/loader/Loader.h"
|
|
#include "FileUtils.h"
|
|
#include "Tactility.h"
|
|
#include "View.h"
|
|
#include "StringUtils.h"
|
|
#include <cstring>
|
|
#include <unistd.h>
|
|
|
|
#define TAG "files_app"
|
|
|
|
namespace tt::app::files {
|
|
|
|
// region Callbacks
|
|
|
|
static void dirEntryListScrollBeginCallback(lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
view->onDirEntryListScrollBegin();
|
|
}
|
|
|
|
static void onDirEntryPressedCallback(lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
auto* button = lv_event_get_target_obj(event);
|
|
auto index = lv_obj_get_index(button);
|
|
view->onDirEntryPressed(index);
|
|
}
|
|
|
|
static void onDirEntryLongPressedCallback(lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
auto* button = lv_event_get_target_obj(event);
|
|
auto index = lv_obj_get_index(button);
|
|
view->onDirEntryLongPressed(index);
|
|
}
|
|
|
|
static void onRenamePressedCallback(lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
view->onRenamePressed();
|
|
}
|
|
|
|
static void onDeletePressedCallback(lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
view->onDeletePressed();
|
|
}
|
|
|
|
static void onNavigateUpPressedCallback(TT_UNUSED lv_event_t* event) {
|
|
auto* view = (View*)lv_event_get_user_data(event);
|
|
view->onNavigateUpPressed();
|
|
}
|
|
|
|
// endregion
|
|
|
|
void View::viewFile(const std::string& path, const std::string& filename) {
|
|
std::string file_path = path + "/" + filename;
|
|
|
|
// For PC we need to make the path relative to the current work directory,
|
|
// because that's how LVGL maps its 'drive letter' to the file system.
|
|
std::string processed_filepath;
|
|
if (kernel::getPlatform() == kernel::PlatformSimulator) {
|
|
char cwd[PATH_MAX];
|
|
if (getcwd(cwd, sizeof(cwd)) == nullptr) {
|
|
TT_LOG_E(TAG, "Failed to get current working directory");
|
|
return;
|
|
}
|
|
if (!file_path.starts_with(cwd)) {
|
|
TT_LOG_E(TAG, "Can only work with files in working directory %s", cwd);
|
|
return;
|
|
}
|
|
processed_filepath = file_path.substr(strlen(cwd));
|
|
} else {
|
|
processed_filepath = file_path;
|
|
}
|
|
|
|
TT_LOG_I(TAG, "Clicked %s", file_path.c_str());
|
|
|
|
if (isSupportedExecutableFile(filename)) {
|
|
#ifdef ESP_PLATFORM
|
|
app::startElfApp(processed_filepath);
|
|
#endif
|
|
} else if (isSupportedImageFile(filename)) {
|
|
auto bundle = std::make_shared<Bundle>();
|
|
bundle->putString(IMAGE_VIEWER_FILE_ARGUMENT, processed_filepath);
|
|
service::loader::startApp("ImageViewer", false, bundle);
|
|
} else if (isSupportedTextFile(filename)) {
|
|
auto bundle = std::make_shared<Bundle>();
|
|
if (kernel::getPlatform() == kernel::PlatformEsp) {
|
|
bundle->putString(TEXT_VIEWER_FILE_ARGUMENT, processed_filepath);
|
|
} else {
|
|
// Remove forward slash, because we need a relative path
|
|
bundle->putString(TEXT_VIEWER_FILE_ARGUMENT, processed_filepath.substr(1));
|
|
}
|
|
service::loader::startApp("TextViewer", false, bundle);
|
|
} else {
|
|
TT_LOG_W(TAG, "opening files of this type is not supported");
|
|
}
|
|
|
|
onNavigate();
|
|
}
|
|
|
|
void View::onDirEntryPressed(uint32_t index) {
|
|
dirent dir_entry;
|
|
if (state->getDirent(index, dir_entry)) {
|
|
TT_LOG_I(TAG, "Pressed %s %d", dir_entry.d_name, dir_entry.d_type);
|
|
state->setSelectedChildEntry(dir_entry.d_name);
|
|
switch (dir_entry.d_type) {
|
|
case TT_DT_DIR:
|
|
case TT_DT_CHR:
|
|
state->setEntriesForChildPath(dir_entry.d_name);
|
|
onNavigate();
|
|
update();
|
|
break;
|
|
case TT_DT_LNK:
|
|
TT_LOG_W(TAG, "opening links is not supported");
|
|
break;
|
|
case TT_DT_REG:
|
|
viewFile(state->getCurrentPath(), dir_entry.d_name);
|
|
onNavigate();
|
|
break;
|
|
default:
|
|
// Assume it's a file
|
|
// TODO: Find a better way to identify a file
|
|
viewFile(state->getCurrentPath(), dir_entry.d_name);
|
|
onNavigate();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
void View::onDirEntryLongPressed(int32_t index) {
|
|
dirent dir_entry;
|
|
if (state->getDirent(index, dir_entry)) {
|
|
TT_LOG_I(TAG, "Pressed %s %d", dir_entry.d_name, dir_entry.d_type);
|
|
state->setSelectedChildEntry(dir_entry.d_name);
|
|
switch (dir_entry.d_type) {
|
|
case TT_DT_DIR:
|
|
case TT_DT_CHR:
|
|
showActionsForDirectory();
|
|
break;
|
|
case TT_DT_LNK:
|
|
TT_LOG_W(TAG, "opening links is not supported");
|
|
break;
|
|
case TT_DT_REG:
|
|
showActionsForFile();
|
|
break;
|
|
default:
|
|
// Assume it's a file
|
|
// TODO: Find a better way to identify a file
|
|
showActionsForFile();
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
void View::createDirEntryWidget(lv_obj_t* parent, struct dirent& dir_entry) {
|
|
tt_check(parent);
|
|
auto* list = (lv_obj_t*)parent;
|
|
const char* symbol;
|
|
if (dir_entry.d_type == TT_DT_DIR || dir_entry.d_type == TT_DT_CHR) {
|
|
symbol = LV_SYMBOL_DIRECTORY;
|
|
} else if (isSupportedImageFile(dir_entry.d_name)) {
|
|
symbol = LV_SYMBOL_IMAGE;
|
|
} else if (dir_entry.d_type == TT_DT_LNK) {
|
|
symbol = LV_SYMBOL_LOOP;
|
|
} else {
|
|
symbol = LV_SYMBOL_FILE;
|
|
}
|
|
lv_obj_t* button = lv_list_add_button(list, symbol, dir_entry.d_name);
|
|
lv_obj_add_event_cb(button, &onDirEntryPressedCallback, LV_EVENT_SHORT_CLICKED, this);
|
|
lv_obj_add_event_cb(button, &onDirEntryLongPressedCallback, LV_EVENT_LONG_PRESSED, this);
|
|
}
|
|
|
|
void View::onNavigateUpPressed() {
|
|
if (state->getCurrentPath() != "/") {
|
|
TT_LOG_I(TAG, "Navigating upwards");
|
|
std::string new_absolute_path;
|
|
if (string::getPathParent(state->getCurrentPath(), new_absolute_path)) {
|
|
state->setEntriesForPath(new_absolute_path);
|
|
}
|
|
onNavigate();
|
|
update();
|
|
}
|
|
}
|
|
|
|
void View::onRenamePressed() {
|
|
std::string entry_name = state->getSelectedChildEntry();
|
|
TT_LOG_I(TAG, "Pending rename %s", entry_name.c_str());
|
|
state->setPendingAction(State::ActionRename);
|
|
app::inputdialog::start("Rename", "", entry_name);
|
|
}
|
|
|
|
void View::onDeletePressed() {
|
|
std::string file_path = state->getSelectedChildPath();
|
|
TT_LOG_I(TAG, "Pending delete %s", file_path.c_str());
|
|
state->setPendingAction(State::ActionDelete);
|
|
std::string message = "Do you want to delete this?\n" + file_path;
|
|
const std::vector<std::string> choices = { "Yes", "No" };
|
|
app::alertdialog::start("Are you sure?", message, choices);
|
|
}
|
|
|
|
void View::showActionsForDirectory() {
|
|
lv_obj_clean(action_list);
|
|
|
|
auto* rename_button = lv_list_add_button(action_list, LV_SYMBOL_EDIT, "Rename");
|
|
lv_obj_add_event_cb(rename_button, onRenamePressedCallback, LV_EVENT_SHORT_CLICKED, this);
|
|
|
|
lv_obj_remove_flag(action_list, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
|
|
void View::showActionsForFile() {
|
|
lv_obj_clean(action_list);
|
|
|
|
auto* rename_button = lv_list_add_button(action_list, LV_SYMBOL_EDIT, "Rename");
|
|
lv_obj_add_event_cb(rename_button, onRenamePressedCallback, LV_EVENT_SHORT_CLICKED, this);
|
|
auto* delete_button = lv_list_add_button(action_list, LV_SYMBOL_TRASH, "Delete");
|
|
lv_obj_add_event_cb(delete_button, onDeletePressedCallback, LV_EVENT_SHORT_CLICKED, this);
|
|
|
|
lv_obj_remove_flag(action_list, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
|
|
void View::update() {
|
|
auto scoped_lockable = lvgl::getLvglSyncLockable()->scoped();
|
|
if (scoped_lockable->lock(100 / portTICK_PERIOD_MS)) {
|
|
lv_obj_clean(dir_entry_list);
|
|
auto entries = state->lockEntries();
|
|
for (auto entry : entries) {
|
|
TT_LOG_D(TAG, "Entry: %s %d", entry.d_name, entry.d_type);
|
|
createDirEntryWidget(dir_entry_list, entry);
|
|
}
|
|
state->unlockEntries();
|
|
|
|
if (state->getCurrentPath() == "/") {
|
|
lv_obj_add_flag(navigate_up_button, LV_OBJ_FLAG_HIDDEN);
|
|
} else {
|
|
lv_obj_remove_flag(navigate_up_button, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
} else {
|
|
TT_LOG_E(TAG, LOG_MESSAGE_MUTEX_LOCK_FAILED_FMT, "lvgl");
|
|
}
|
|
}
|
|
|
|
void View::init(lv_obj_t* parent) {
|
|
lv_obj_set_flex_flow(parent, LV_FLEX_FLOW_COLUMN);
|
|
|
|
lv_obj_t* toolbar = lvgl::toolbar_create(parent, "Files");
|
|
navigate_up_button = lvgl::toolbar_add_button_action(toolbar, LV_SYMBOL_UP, &onNavigateUpPressedCallback, this);
|
|
|
|
lv_obj_t* wrapper = lv_obj_create(parent);
|
|
lv_obj_set_width(wrapper, LV_PCT(100));
|
|
lv_obj_set_style_border_width(wrapper, 0, 0);
|
|
lv_obj_set_style_pad_all(wrapper, 0, 0);
|
|
lv_obj_set_flex_grow(wrapper, 1);
|
|
lv_obj_set_flex_flow(wrapper, LV_FLEX_FLOW_ROW);
|
|
|
|
dir_entry_list = lv_list_create(wrapper);
|
|
lv_obj_set_height(dir_entry_list, LV_PCT(100));
|
|
lv_obj_set_flex_grow(dir_entry_list, 1);
|
|
|
|
lv_obj_add_event_cb(dir_entry_list, dirEntryListScrollBeginCallback, LV_EVENT_SCROLL_BEGIN, this);
|
|
|
|
action_list = lv_list_create(wrapper);
|
|
lv_obj_set_height(action_list, LV_PCT(100));
|
|
lv_obj_set_flex_grow(action_list, 1);
|
|
lv_obj_add_flag(action_list, LV_OBJ_FLAG_HIDDEN);
|
|
|
|
update();
|
|
}
|
|
|
|
void View::onDirEntryListScrollBegin() {
|
|
auto scoped_lockable = lvgl::getLvglSyncLockable()->scoped();
|
|
if (scoped_lockable->lock(100 / portTICK_PERIOD_MS)) {
|
|
lv_obj_add_flag(action_list, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
}
|
|
|
|
void View::onNavigate() {
|
|
auto scoped_lockable = lvgl::getLvglSyncLockable()->scoped();
|
|
if (scoped_lockable->lock(100 / portTICK_PERIOD_MS)) {
|
|
lv_obj_add_flag(action_list, LV_OBJ_FLAG_HIDDEN);
|
|
}
|
|
}
|
|
|
|
void View::onResult(Result result, const Bundle& bundle) {
|
|
if (result != ResultOk) {
|
|
return;
|
|
}
|
|
|
|
std::string filepath = state->getSelectedChildPath();
|
|
TT_LOG_I(TAG, "Result for %s", filepath.c_str());
|
|
|
|
switch (state->getPendingAction()) {
|
|
case State::ActionDelete: {
|
|
if (alertdialog::getResultIndex(bundle) == 0) {
|
|
int delete_count = (int)remove(filepath.c_str());
|
|
if (delete_count > 0) {
|
|
TT_LOG_I(TAG, "Deleted %d items", delete_count);
|
|
} else {
|
|
TT_LOG_W(TAG, "Failed to delete %s", filepath.c_str());
|
|
}
|
|
state->setEntriesForPath(state->getCurrentPath());
|
|
update();
|
|
}
|
|
break;
|
|
}
|
|
case State::ActionRename: {
|
|
auto new_name = app::inputdialog::getResult(bundle);
|
|
if (!new_name.empty() && new_name != state->getSelectedChildEntry()) {
|
|
std::string rename_to = getChildPath(state->getCurrentPath(), new_name);
|
|
if (rename(filepath.c_str(), rename_to.c_str())) {
|
|
TT_LOG_I(TAG, "Renamed \"%s\" to \"%s\"", filepath.c_str(), rename_to.c_str());
|
|
} else {
|
|
TT_LOG_E(TAG, "Failed to rename \"%s\" to \"%s\"", filepath.c_str(), rename_to.c_str());
|
|
}
|
|
state->setEntriesForPath(state->getCurrentPath());
|
|
update();
|
|
}
|
|
break;
|
|
}
|
|
default:
|
|
break;
|
|
}
|
|
}
|
|
|
|
}
|