blobwarsAttrition/src/system/widgets.c

322 lines
6.0 KiB
C

/*
Copyright (C) 2018 Parallel Realities
This program is free software; you can redistribute it and/or
modify it under the terms of the GNU General Public License
as published by the Free Software Foundation; either version 2
of the License, or (at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
See the GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program; if not, write to the Free Software
Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA.
*/
#include "widgets.h"
static void loadWidgetGroup(char *filename);
static void loadWidgets(void);
static void createWidgetOptions(Widget *w, char *options);
static void selectWidget(int dir);
static void updateWidgetValue(int dir);
static Widget widgets[MAX_WIDGETS];
static Widget *selectedWidget;
static int widgetIndex;
static int numWidgets;
void initWidgets(void)
{
memset(widgets, 0, sizeof(Widget) * MAX_WIDGETS);
numWidgets = 0;
selectedWidget = NULL;
loadWidgets();
}
void doWidgets(void)
{
if (app.keyboard[SDL_SCANCODE_UP])
{
selectWidget(-1);
app.keyboard[SDL_SCANCODE_UP] = 0;
}
if (app.keyboard[SDL_SCANCODE_DOWN])
{
selectWidget(1);
app.keyboard[SDL_SCANCODE_DOWN] = 0;
}
if (app.keyboard[SDL_SCANCODE_LEFT])
{
updateWidgetValue(-1);
}
if (app.keyboard[SDL_SCANCODE_RIGHT])
{
updateWidgetValue(1);
}
if (app.keyboard[SDL_SCANCODE_RETURN])
{
selectedWidget->action();
}
}
static void updateWidgetValue(int dir)
{
if (selectedWidget->type == WT_SLIDER)
{
selectedWidget->value = limit(selectedWidget->value + dir, selectedWidget->minValue, selectedWidget->maxValue);
selectedWidget->action();
}
else if (selectedWidget->type == WT_SPINNER)
{
selectedWidget->value = limit(selectedWidget->value + dir, 0, selectedWidget->numOptions - 1);
selectedWidget->action();
}
}
void drawWidgets(void)
{
int i;
Widget *w;
for (i = 0 ; i < numWidgets ; i++)
{
w = &widgets[i];
if (w->visible)
{
switch (w->type)
{
case WT_BUTTON:
if (w != selectedWidget)
{
drawRect(w->x, w->y, w->w, w->h, 0, 0, 0, 255);
drawOutlineRect(w->x, w->y, w->w, w->h, 0, 128, 0, 255);
}
else
{
drawRect(w->x, w->y, w->w, w->h, 0, 128, 0, 255);
drawOutlineRect(w->x, w->y, w->w, w->h, 0, 255, 0, 255);
}
drawText(w->x + w->w / 2, w->y + 2, 24, TA_CENTER, colors.white, w->label);
break;
case WT_SLIDER:
break;
case WT_SPINNER:
break;
case WT_INPUT:
break;
}
}
}
}
static void selectWidget(int dir)
{
int oldWidgetIndex = widgetIndex;
do
{
widgetIndex += dir;
if (widgetIndex < 0)
{
widgetIndex = numWidgets - 1;
}
if (widgetIndex >= numWidgets)
{
widgetIndex = 0;
}
selectedWidget = &widgets[widgetIndex];
} while (!selectedWidget->enabled && !selectedWidget->visible);
if (oldWidgetIndex != widgetIndex)
{
playSound(SND_MENU_NAV, 0);
}
}
Widget *getWidget(char *name, char *group)
{
int i;
Widget *w;
for (i = 0 ; i < numWidgets ; i++)
{
w = &widgets[i];
if (strcmp(w->name, name) == 0 && strcmp(w->group, group) == 0)
{
return w;
}
}
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_ERROR, "No such widget '%s', '%s'", name, group);
exit(1);
return NULL;
}
void hideAllWidgets(void)
{
int i;
for (i = 0 ; i < numWidgets ; i++)
{
widgets[i].visible = 0;
}
selectedWidget = NULL;
}
void showWidgetGroup(char *group)
{
int i;
Widget *w;
hideAllWidgets();
for (i = 0 ; i < numWidgets ; i++)
{
w = &widgets[i];
if (strcmp(w->group, group) == 0)
{
if (selectedWidget == NULL)
{
selectedWidget = w;
widgetIndex = i;
}
w->visible = 1;
}
}
}
static void loadWidgets(void)
{
char **filenames;
char path[MAX_FILENAME_LENGTH];
int count, i;
filenames = getFileList("data/widgets", &count);
for (i = 0 ; i < count ; i++)
{
sprintf(path, "data/widgets/%s", filenames[i]);
loadWidgetGroup(path);
free(filenames[i]);
}
free(filenames);
}
static void loadWidgetGroup(char *filename)
{
cJSON *root, *node;
char *text;
Widget *w;
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_INFO, "Loading %s", filename);
text = readFile(filename);
root = cJSON_Parse(text);
for (node = root->child ; node != NULL ; node = node->next)
{
if (++numWidgets >= MAX_WIDGETS)
{
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_ERROR, "Out of widget space.");
exit(1);
}
w = &widgets[numWidgets];
STRNCPY(w->name, cJSON_GetObjectItem(node, "name")->valuestring, MAX_NAME_LENGTH);
STRNCPY(w->group, cJSON_GetObjectItem(node, "group")->valuestring, MAX_NAME_LENGTH);
STRNCPY(w->label, cJSON_GetObjectItem(node, "label")->valuestring, MAX_NAME_LENGTH);
w->x = cJSON_GetObjectItem(node, "x")->valueint;
w->y = cJSON_GetObjectItem(node, "y")->valueint;
w->w = cJSON_GetObjectItem(node, "w")->valueint;
w->h = cJSON_GetObjectItem(node, "h")->valueint;
w->type = lookup(cJSON_GetObjectItem(node, "type")->valuestring);
if (w->x == -1)
{
w->x = (SCREEN_WIDTH - w->w) / 2;
}
switch (w->type)
{
case WT_SPINNER:
createWidgetOptions(w, cJSON_GetObjectItem(node, "options")->valuestring);
break;
default:
break;
}
}
cJSON_Delete(root);
free(text);
}
static void createWidgetOptions(Widget *w, char *options)
{
int i;
char *option;
w->numOptions = 1;
for (i = 0 ; i < strlen(options) ; i++)
{
if (options[i] == '|')
{
w->numOptions++;
}
}
w->options = malloc(w->numOptions * sizeof(char*));
i = 0;
option = strtok(options, "|");
while (option)
{
w->options[i] = malloc(strlen(option) + 1);
strcpy(w->options[i], option);
SDL_LogMessage(SDL_LOG_CATEGORY_APPLICATION, SDL_LOG_PRIORITY_DEBUG, "widget.option[%d] = %s", i, option);
option = strtok(NULL, "|");
i++;
}
}
void destroy(void)
{
}