GUI: Menu abstraction

This commit is contained in:
Jeffrey Pfau 2015-08-30 13:14:29 -07:00
parent 1c351d92c7
commit 31f40b05ba
4 changed files with 158 additions and 94 deletions

View File

@ -11,7 +11,6 @@
#include "util/vector.h" #include "util/vector.h"
struct GUIFont; struct GUIFont;
DECLARE_VECTOR(FileList, char*);
enum GUIInput { enum GUIInput {
GUI_INPUT_NONE = -1, GUI_INPUT_NONE = -1,

View File

@ -6,22 +6,21 @@
#include "file-select.h" #include "file-select.h"
#include "util/gui/font.h" #include "util/gui/font.h"
#include "util/gui/menu.h"
#include "util/vfs.h" #include "util/vfs.h"
#include <stdlib.h> #include <stdlib.h>
DEFINE_VECTOR(FileList, char*);
#define ITERATION_SIZE 5 #define ITERATION_SIZE 5
#define SCANNING_THRESHOLD 15 #define SCANNING_THRESHOLD 15
static void _cleanFiles(struct FileList* currentFiles) { static void _cleanFiles(struct GUIMenuItemList* currentFiles) {
size_t size = FileListSize(currentFiles); size_t size = GUIMenuItemListSize(currentFiles);
size_t i; size_t i;
for (i = 1; i < size; ++i) { for (i = 1; i < size; ++i) {
free(*FileListGetPointer(currentFiles, i)); free(GUIMenuItemListGetPointer(currentFiles, i)->title);
} }
FileListClear(currentFiles); GUIMenuItemListClear(currentFiles);
} }
static void _upDirectory(char* currentPath) { static void _upDirectory(char* currentPath) {
@ -41,17 +40,17 @@ static void _upDirectory(char* currentPath) {
} }
static int _strpcmp(const void* a, const void* b) { static int _strpcmp(const void* a, const void* b) {
return strcmp(*(const char**) a, *(const char**) b); return strcmp(((const struct GUIMenuItem*) a)->title, ((const struct GUIMenuItem*) b)->title);
} }
static bool _refreshDirectory(struct GUIParams* params, const char* currentPath, struct FileList* currentFiles, bool (*filter)(struct VFile*)) { static bool _refreshDirectory(struct GUIParams* params, const char* currentPath, struct GUIMenuItemList* currentFiles, bool (*filter)(struct VFile*)) {
_cleanFiles(currentFiles); _cleanFiles(currentFiles);
struct VDir* dir = VDirOpen(currentPath); struct VDir* dir = VDirOpen(currentPath);
if (!dir) { if (!dir) {
return false; return false;
} }
*FileListAppend(currentFiles) = "(Up)"; *GUIMenuItemListAppend(currentFiles) = (struct GUIMenuItem) { .title = "(Up)" };
size_t i = 0; size_t i = 0;
struct VDirEntry* de; struct VDirEntry* de;
while ((de = dir->listNext(dir))) { while ((de = dir->listNext(dir))) {
@ -64,7 +63,7 @@ static bool _refreshDirectory(struct GUIParams* params, const char* currentPath,
} }
params->drawStart(); params->drawStart();
GUIFontPrintf(params->font, 0, GUIFontHeight(params->font), GUI_TEXT_LEFT, 0xFFFFFFFF, "%s", currentPath); GUIFontPrintf(params->font, 0, GUIFontHeight(params->font), GUI_TEXT_LEFT, 0xFFFFFFFF, "%s", currentPath);
GUIFontPrintf(params->font, 0, GUIFontHeight(params->font) * 2, GUI_TEXT_LEFT, 0xFFFFFFFF, "(scanning)"); GUIFontPrintf(params->font, 0, GUIFontHeight(params->font) * 2, GUI_TEXT_LEFT, 0xFFFFFFFF, "(scanning item %z)", i);
params->drawEnd(); params->drawEnd();
} }
const char* name = de->name(de); const char* name = de->name(de);
@ -77,70 +76,38 @@ static bool _refreshDirectory(struct GUIParams* params, const char* currentPath,
continue; continue;
} }
if (!filter || filter(vf)) { if (!filter || filter(vf)) {
*FileListAppend(currentFiles) = strdup(name); *GUIMenuItemListAppend(currentFiles) = (struct GUIMenuItem) { .title = strdup(name) };
} }
vf->close(vf); vf->close(vf);
} else { } else {
*FileListAppend(currentFiles) = strdup(name); *GUIMenuItemListAppend(currentFiles) = (struct GUIMenuItem) { .title = strdup(name) };
} }
} }
dir->close(dir); dir->close(dir);
qsort(FileListGetPointer(currentFiles, 1), FileListSize(currentFiles) - 1, sizeof(char*), _strpcmp); qsort(GUIMenuItemListGetPointer(currentFiles, 1), GUIMenuItemListSize(currentFiles) - 1, sizeof(struct GUIMenuItem), _strpcmp);
return true; return true;
} }
bool GUISelectFile(struct GUIParams* params, char* outPath, size_t outLen, bool (*filter)(struct VFile*)) { bool GUISelectFile(struct GUIParams* params, char* outPath, size_t outLen, bool (*filter)(struct VFile*)) {
size_t start = 0; struct GUIMenu menu = {
size_t pageSize = params->height / GUIFontHeight(params->font); .title = params->currentPath,
if (pageSize > 4) { .index = params->fileIndex,
pageSize -= 4; };
} else { GUIMenuItemListInit(&menu.items, 0);
pageSize = 1; _refreshDirectory(params, params->currentPath, &menu.items, filter);
}
GUIInvalidateKeys(params);
struct FileList currentFiles;
FileListInit(&currentFiles, 0);
_refreshDirectory(params, params->currentPath, &currentFiles, filter);
while (true) { while (true) {
int newInput = 0; struct GUIMenuItem item;
GUIPollInput(params, &newInput, 0); enum GUIMenuExitReason reason = GUIShowMenu(params, &menu, &item);
params->fileIndex = menu.index;
if (newInput & (1 << GUI_INPUT_UP) && params->fileIndex > 0) { if (reason == GUI_MENU_EXIT_CANCEL) {
--params->fileIndex;
}
if (newInput & (1 << GUI_INPUT_DOWN) && params->fileIndex < FileListSize(&currentFiles) - 1) {
++params->fileIndex;
}
if (newInput & (1 << GUI_INPUT_LEFT)) {
if (params->fileIndex >= pageSize) {
params->fileIndex -= pageSize;
} else {
params->fileIndex = 0;
}
}
if (newInput & (1 << GUI_INPUT_RIGHT)) {
if (params->fileIndex + pageSize < FileListSize(&currentFiles)) {
params->fileIndex += pageSize;
} else {
params->fileIndex = FileListSize(&currentFiles) - 1;
}
}
if (params->fileIndex < start) {
start = params->fileIndex;
}
while ((params->fileIndex - start + 4) * GUIFontHeight(params->font) > params->height) {
++start;
}
if (newInput & (1 << GUI_INPUT_CANCEL)) {
break; break;
} }
if (newInput & (1 << GUI_INPUT_SELECT)) { if (reason == GUI_MENU_EXIT_ACCEPT) {
if (params->fileIndex == 0) { if (params->fileIndex == 0) {
_upDirectory(params->currentPath); _upDirectory(params->currentPath);
if (!_refreshDirectory(params, params->currentPath, &currentFiles, filter)) { if (!_refreshDirectory(params, params->currentPath, &menu.items, filter)) {
break; break;
} }
} else { } else {
@ -149,69 +116,47 @@ bool GUISelectFile(struct GUIParams* params, char* outPath, size_t outLen, bool
if (params->currentPath[len - 1] == *sep) { if (params->currentPath[len - 1] == *sep) {
sep = ""; sep = "";
} }
snprintf(outPath, outLen, "%s%s%s", params->currentPath, sep, *FileListGetPointer(&currentFiles, params->fileIndex)); snprintf(outPath, outLen, "%s%s%s", params->currentPath, sep, item.title);
struct FileList newFiles; struct GUIMenuItemList newFiles;
FileListInit(&newFiles, 0); GUIMenuItemListInit(&newFiles, 0);
if (!_refreshDirectory(params, outPath, &newFiles, filter)) { if (!_refreshDirectory(params, outPath, &newFiles, filter)) {
_cleanFiles(&newFiles); _cleanFiles(&newFiles);
FileListDeinit(&newFiles); GUIMenuItemListDeinit(&newFiles);
struct VFile* vf = VFileOpen(outPath, O_RDONLY); struct VFile* vf = VFileOpen(outPath, O_RDONLY);
if (!vf) { if (!vf) {
continue; continue;
} }
if (!filter || filter(vf)) { if (!filter || filter(vf)) {
vf->close(vf); vf->close(vf);
_cleanFiles(&currentFiles); _cleanFiles(&menu.items);
FileListDeinit(&currentFiles); GUIMenuItemListDeinit(&menu.items);
return true; return true;
} }
vf->close(vf); vf->close(vf);
break; break;
} else { } else {
_cleanFiles(&currentFiles); _cleanFiles(&menu.items);
FileListDeinit(&currentFiles); GUIMenuItemListDeinit(&menu.items);
currentFiles = newFiles; menu.items = newFiles;
strncpy(params->currentPath, outPath, PATH_MAX); strncpy(params->currentPath, outPath, PATH_MAX);
} }
} }
params->fileIndex = 0; params->fileIndex = 0;
} }
if (newInput & (1 << GUI_INPUT_BACK)) { if (reason == GUI_MENU_EXIT_BACK) {
if (strncmp(params->currentPath, params->basePath, PATH_MAX) == 0) { if (strncmp(params->currentPath, params->basePath, PATH_MAX) == 0) {
break; break;
} }
_upDirectory(params->currentPath); _upDirectory(params->currentPath);
if (!_refreshDirectory(params, params->currentPath, &currentFiles, filter)) { if (!_refreshDirectory(params, params->currentPath, &menu.items, filter)) {
break; break;
} }
params->fileIndex = 0; params->fileIndex = 0;
} }
params->drawStart();
unsigned y = GUIFontHeight(params->font);
GUIFontPrintf(params->font, 0, y, GUI_TEXT_LEFT, 0xFFFFFFFF, "%s", params->currentPath);
y += 2 * GUIFontHeight(params->font);
size_t i;
for (i = start; i < FileListSize(&currentFiles); ++i) {
int color = 0xE0A0A0A0;
char bullet = ' ';
if (i == params->fileIndex) {
color = 0xFFFFFFFF;
bullet = '>';
}
GUIFontPrintf(params->font, 0, y, GUI_TEXT_LEFT, color, "%c %s", bullet, *FileListGetPointer(&currentFiles, i));
y += GUIFontHeight(params->font);
if (y + GUIFontHeight(params->font) > params->height) {
break;
}
}
y += GUIFontHeight(params->font) * 2;
params->drawEnd();
} }
_cleanFiles(&currentFiles); _cleanFiles(&menu.items);
FileListDeinit(&currentFiles); GUIMenuItemListDeinit(&menu.items);
return false; return false;
} }

88
src/util/gui/menu.c Normal file
View File

@ -0,0 +1,88 @@
/* Copyright (c) 2013-2015 Jeffrey Pfau
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#include "menu.h"
#include "util/gui.h"
#include "util/gui/font.h"
DEFINE_VECTOR(GUIMenuItemList, struct GUIMenuItem);
enum GUIMenuExitReason GUIShowMenu(struct GUIParams* params, struct GUIMenu* menu, struct GUIMenuItem* item) {
size_t start = 0;
size_t pageSize = params->height / GUIFontHeight(params->font);
if (pageSize > 4) {
pageSize -= 4;
} else {
pageSize = 1;
}
GUIInvalidateKeys(params);
while (true) {
int newInput = 0;
GUIPollInput(params, &newInput, 0);
if (newInput & (1 << GUI_INPUT_UP) && menu->index > 0) {
--menu->index;
}
if (newInput & (1 << GUI_INPUT_DOWN) && menu->index < GUIMenuItemListSize(&menu->items) - 1) {
++menu->index;
}
if (newInput & (1 << GUI_INPUT_LEFT)) {
if (menu->index >= pageSize) {
menu->index -= pageSize;
} else {
menu->index = 0;
}
}
if (newInput & (1 << GUI_INPUT_RIGHT)) {
if (menu->index + pageSize < GUIMenuItemListSize(&menu->items)) {
menu->index += pageSize;
} else {
menu->index = GUIMenuItemListSize(&menu->items) - 1;
}
}
if (menu->index < start) {
start = menu->index;
}
while ((menu->index - start + 4) * GUIFontHeight(params->font) > params->height) {
++start;
}
if (newInput & (1 << GUI_INPUT_CANCEL)) {
break;
}
if (newInput & (1 << GUI_INPUT_SELECT)) {
*item = *GUIMenuItemListGetPointer(&menu->items, menu->index);
return GUI_MENU_EXIT_ACCEPT;
}
if (newInput & (1 << GUI_INPUT_BACK)) {
return GUI_MENU_EXIT_BACK;
}
params->drawStart();
unsigned y = GUIFontHeight(params->font);
GUIFontPrint(params->font, 0, y, GUI_TEXT_LEFT, 0xFFFFFFFF, menu->title);
y += 2 * GUIFontHeight(params->font);
size_t i;
for (i = start; i < GUIMenuItemListSize(&menu->items); ++i) {
int color = 0xE0A0A0A0;
char bullet = ' ';
if (i == menu->index) {
color = 0xFFFFFFFF;
bullet = '>';
}
GUIFontPrintf(params->font, 0, y, GUI_TEXT_LEFT, color, "%c %s", bullet, GUIMenuItemListGetPointer(&menu->items, i)->title);
y += GUIFontHeight(params->font);
if (y + GUIFontHeight(params->font) > params->height) {
break;
}
}
y += GUIFontHeight(params->font) * 2;
params->drawEnd();
}
return GUI_MENU_EXIT_CANCEL;
}

32
src/util/gui/menu.h Normal file
View File

@ -0,0 +1,32 @@
/* Copyright (c) 2013-2015 Jeffrey Pfau
*
* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
#ifndef GUI_MENU_H
#define GUI_MENU_H
#include "util/vector.h"
struct GUIMenuItem {
const char* title;
void* data;
};
DECLARE_VECTOR(GUIMenuItemList, struct GUIMenuItem);
struct GUIMenu {
const char* title;
struct GUIMenuItemList items;
size_t index;
};
enum GUIMenuExitReason {
GUI_MENU_EXIT_ACCEPT,
GUI_MENU_EXIT_BACK,
GUI_MENU_EXIT_CANCEL,
};
struct GUIParams;
enum GUIMenuExitReason GUIShowMenu(struct GUIParams* params, struct GUIMenu* menu, struct GUIMenuItem* item);
#endif