409 lines
14 KiB
C
409 lines
14 KiB
C
/*
|
|
* This file is part of Anemone3DS
|
|
* Copyright (C) 2016-2017 Alex Taber ("astronautlevel"), Dawid Eckert ("daedreth")
|
|
*
|
|
* 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 3 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, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* Additional Terms 7.b and 7.c of GPLv3 apply to this file:
|
|
* * Requiring preservation of specified reasonable legal notices or
|
|
* author attributions in that material or in the Appropriate Legal
|
|
* Notices displayed by works containing it.
|
|
* * Prohibiting misrepresentation of the origin of that material,
|
|
* or requiring that modified versions of such material be marked in
|
|
* reasonable ways as different from the original version.
|
|
*/
|
|
|
|
#include "draw.h"
|
|
#include "common.h"
|
|
|
|
#include "pp2d/pp2d/pp2d.h"
|
|
#include "quirc/quirc.h"
|
|
|
|
#include <time.h>
|
|
|
|
|
|
enum Colors {
|
|
COLOR_BACKGROUND = ABGR8(255, 32, 28, 35), //silver-y black
|
|
COLOR_ACCENT = RGBA8(55, 122, 168, 255),
|
|
COLOR_WHITE = RGBA8(255, 255, 255, 255),
|
|
COLOR_CURSOR = RGBA8(200, 200, 200, 255),
|
|
COLOR_BLACK = RGBA8(0, 0, 0, 255),
|
|
COLOR_RED = RGBA8(200, 0, 0, 255),
|
|
COLOR_YELLOW = RGBA8(239, 220, 11, 255),
|
|
};
|
|
|
|
void init_screens(void)
|
|
{
|
|
pp2d_init();
|
|
|
|
pp2d_set_screen_color(GFX_TOP, COLOR_BACKGROUND);
|
|
pp2d_set_screen_color(GFX_BOTTOM, COLOR_BACKGROUND);
|
|
|
|
pp2d_load_texture_png(TEXTURE_ARROW, "romfs:/arrow.png");
|
|
pp2d_load_texture_png(TEXTURE_SHUFFLE, "romfs:/shuffle.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_1, "romfs:/battery1.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_2, "romfs:/battery2.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_3, "romfs:/battery3.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_4, "romfs:/battery4.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_5, "romfs:/battery5.png");
|
|
pp2d_load_texture_png(TEXTURE_BATTERY_CHARGE, "romfs:/charging.png");
|
|
}
|
|
|
|
void exit_screens(void)
|
|
{
|
|
pp2d_exit();
|
|
}
|
|
|
|
static int theme_vertical_scroll = 0;
|
|
static int splash_vertical_scroll = 0;
|
|
|
|
void draw_themext_error(void)
|
|
{
|
|
throw_error("Theme extdata does not exist\nSet a default theme from the home menu", ERROR);
|
|
}
|
|
|
|
void draw_qr(void)
|
|
{
|
|
take_picture();
|
|
pp2d_begin_draw(GFX_TOP);
|
|
pp2d_free_texture(TEXTURE_QR);
|
|
u32 *rgba8_buf = malloc(240 * 400 * sizeof(u32));
|
|
for (int i = 0; i < 240 * 400; i++)
|
|
{
|
|
rgba8_buf[i] = RGB565_TO_RGBA8(buf[i]);
|
|
u8 *byte_pointer = (u8*)&rgba8_buf[i];
|
|
u8 r = *(byte_pointer+3);
|
|
u8 g = *(byte_pointer+2);
|
|
u8 b = *(byte_pointer+1);
|
|
u8 a = *(byte_pointer);
|
|
rgba8_buf[i] = RGBA8(r, g, b, a);
|
|
}
|
|
pp2d_load_texture_memory(TEXTURE_QR, rgba8_buf, 400, 240);
|
|
pp2d_draw_texture(TEXTURE_QR, 0, 0);
|
|
free(rgba8_buf);
|
|
pp2d_end_draw();
|
|
|
|
int w;
|
|
int h;
|
|
|
|
u8 *image = (u8*) quirc_begin(context, &w, &h);
|
|
|
|
for (ssize_t x = 0; x < w; x++)
|
|
{
|
|
for (ssize_t y = 0; y < h; y++)
|
|
{
|
|
u16 px = buf[y * 400 + x];
|
|
image[y * w + x] = (u8)(((((px >> 11) & 0x1F) << 3) + (((px >> 5) & 0x3F) << 2) + ((px & 0x1F) << 3)) / 3);
|
|
}
|
|
}
|
|
|
|
quirc_end(context);
|
|
|
|
if (quirc_count(context) > 0)
|
|
{
|
|
struct quirc_code code;
|
|
struct quirc_data data;
|
|
quirc_extract(context, 0, &code);
|
|
if (!quirc_decode(&code, &data))
|
|
{
|
|
qr_mode = false;
|
|
|
|
http_get((char*)data.payload, "/Themes/");
|
|
}
|
|
}
|
|
}
|
|
|
|
void draw_base_interface(void)
|
|
{
|
|
pp2d_begin_draw(GFX_TOP);
|
|
pp2d_draw_rectangle(0, 0, 400, 23, COLOR_ACCENT);
|
|
|
|
time_t t = time(NULL);
|
|
struct tm tm = *localtime(&t);
|
|
|
|
pp2d_draw_textf(7, 2, 0.6, 0.6, COLOR_WHITE, "%.2i", tm.tm_hour);
|
|
pp2d_draw_text(28, 2, 0.6, 0.6, COLOR_WHITE, (tm.tm_sec % 2 == 1) ? ":" : " ");
|
|
pp2d_draw_textf(34, 2, 0.6, 0.6, COLOR_WHITE, "%.2i", tm.tm_min);
|
|
|
|
u8 battery_charging;
|
|
PTMU_GetBatteryChargeState(&battery_charging);
|
|
u8 battery_status;
|
|
PTMU_GetBatteryLevel(&battery_status);
|
|
pp2d_draw_texture(2 + battery_status, 357, 2);
|
|
|
|
if (battery_charging)
|
|
pp2d_draw_texture(TEXTURE_BATTERY_CHARGE, 357, 2);
|
|
|
|
pp2d_draw_on(GFX_BOTTOM);
|
|
pp2d_draw_rectangle(0, 0, 320, 24, COLOR_ACCENT);
|
|
pp2d_draw_rectangle(0, 216, 320, 24, COLOR_ACCENT);
|
|
pp2d_draw_text(7, 219, 0.6, 0.6, COLOR_WHITE, VERSION);
|
|
pp2d_draw_on(GFX_TOP);
|
|
}
|
|
void throw_error(char* error, int error_type) {
|
|
|
|
switch (error_type) {
|
|
case ERROR:
|
|
while (1)
|
|
{
|
|
hidScanInput();
|
|
u32 kDown = hidKeysDown();
|
|
draw_base_interface();
|
|
pp2d_draw_text(70, 120, 0.8, 0.8, COLOR_RED, error);
|
|
pp2d_draw_wtext(70, 150, 0.8, 0.8, COLOR_WHITE, L"Press \uE000 to shut down.");
|
|
pp2d_end_draw();
|
|
if (kDown & KEY_A) {
|
|
if (homebrew)
|
|
APT_HardwareResetAsync();
|
|
else {
|
|
srvPublishToSubscriber(0x202, 0);
|
|
}
|
|
}
|
|
}
|
|
case WARNING:
|
|
while (1)
|
|
{
|
|
hidScanInput();
|
|
u32 kDown = hidKeysDown();
|
|
draw_base_interface();
|
|
pp2d_draw_text(70, 120, 0.8, 0.8, COLOR_YELLOW, error);
|
|
pp2d_draw_wtext(70, 150, 0.8, 0.8, COLOR_WHITE, L"Press \uE000 to continue.");
|
|
pp2d_end_draw();
|
|
if (kDown & KEY_A) break;
|
|
}
|
|
break;
|
|
}
|
|
pp2d_end_draw();
|
|
}
|
|
void draw_theme_install(int install_type)
|
|
{
|
|
draw_base_interface();
|
|
switch(install_type)
|
|
{
|
|
case 0:
|
|
pp2d_draw_text_center(GFX_TOP, 120, 0.8, 0.8, COLOR_WHITE, "Installing a single theme...");
|
|
break;
|
|
case 1:
|
|
pp2d_draw_text_center(GFX_TOP, 120, 0.8, 0.8, COLOR_WHITE, "Installing a shuffle theme...");
|
|
break;
|
|
case 2:
|
|
pp2d_draw_text_center(GFX_TOP, 120, 0.8, 0.8, COLOR_WHITE, "Installing BGM...");
|
|
break;
|
|
case 3:
|
|
pp2d_draw_text_center(GFX_TOP, 120, 0.8, 0.8, COLOR_WHITE, "Downloading...");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
pp2d_end_draw();
|
|
}
|
|
|
|
void draw_theme_interface(Theme_s * themes_list, int theme_count, int selected_theme, bool preview_mode, int shuffle_theme_count)
|
|
{
|
|
|
|
if (themes_list == NULL)
|
|
{
|
|
pp2d_begin_draw(GFX_TOP);
|
|
pp2d_draw_text_center(GFX_TOP, 100, 1, 1, COLOR_WHITE, "NO THEMES FOUND");
|
|
pp2d_end_draw();
|
|
return;
|
|
}
|
|
|
|
Theme_s current_theme = themes_list[selected_theme];
|
|
|
|
if (preview_mode)
|
|
{
|
|
if (current_theme.has_preview)
|
|
{
|
|
pp2d_begin_draw(GFX_TOP);
|
|
pp2d_draw_texture_part(TEXTURE_PREVIEW, 0, 0, current_theme.preview_offset, 0, 400, 240);
|
|
pp2d_draw_on(GFX_BOTTOM);
|
|
pp2d_draw_texture_part(TEXTURE_PREVIEW, 0, 0, 40+current_theme.preview_offset, 240, 320, 240);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
draw_base_interface();
|
|
pp2d_draw_text_center(GFX_TOP, 4, 0.5, 0.5, COLOR_WHITE, "Theme mode");
|
|
wchar_t title[0x40] = {0};
|
|
utf16_to_utf32((u32*)title, current_theme.name, 0x40);
|
|
pp2d_draw_wtext_wrap(20, 30, 0.7, 0.7, COLOR_WHITE, 380, title);
|
|
wchar_t author[0x40] = {0};
|
|
utf16_to_utf32((u32*)author, current_theme.author, 0x40);
|
|
pp2d_draw_text(20, 50, 0.5, 0.5, COLOR_WHITE, "By: ");
|
|
pp2d_draw_wtext_wrap(44, 50, 0.5, 0.5, COLOR_WHITE, 380, author);
|
|
wchar_t description[0xa6] = {0};
|
|
utf16_to_utf32((u32*)description, current_theme.desc, 0xb0);
|
|
pp2d_draw_wtext_wrap(20, 65, 0.5, 0.5, COLOR_WHITE, 363, description);
|
|
|
|
pp2d_draw_wtext(20, 150, 0.6, 0.6, COLOR_WHITE, L"\uE046 Install Shuffle Theme");
|
|
pp2d_draw_wtext(200, 150, 0.6, 0.6, COLOR_WHITE, L"\uE004 Switch to Splashes");
|
|
pp2d_draw_wtext(20, 180, 0.6, 0.6, COLOR_WHITE, L"\uE000 Install Theme");
|
|
pp2d_draw_wtext(200, 180, 0.6, 0.6, COLOR_WHITE, L"\uE001 Queue Shuffle");
|
|
pp2d_draw_wtext(20, 210, 0.6, 0.6, COLOR_WHITE, L"\uE002 Install BGM");
|
|
pp2d_draw_wtext(200, 210, 0.6, 0.6, COLOR_WHITE, L"\uE003 Preview Theme");
|
|
pp2d_draw_wtext(130, 120, 0.6, 0.6, COLOR_WHITE, L"\uE005 Scan QRCode");
|
|
pp2d_draw_on(GFX_BOTTOM);
|
|
|
|
pp2d_draw_textf(7, 3, 0.6, 0.6, COLOR_WHITE, "Selected: %i/10", shuffle_theme_count);
|
|
|
|
// Scroll the menu up or down if the selected theme is out of its bounds
|
|
//----------------------------------------------------------------
|
|
for (int i = 0; i < theme_count; i++) {
|
|
if (theme_count <= THEMES_PER_SCREEN)
|
|
break;
|
|
|
|
if (theme_vertical_scroll > selected_theme)
|
|
theme_vertical_scroll--;
|
|
|
|
if ((i < selected_theme) && \
|
|
((selected_theme - theme_vertical_scroll) >= THEMES_PER_SCREEN) && \
|
|
(theme_vertical_scroll != ( - THEMES_PER_SCREEN)))
|
|
theme_vertical_scroll++;
|
|
}
|
|
//----------------------------------------------------------------
|
|
|
|
// Show arrows if there are themes out of bounds
|
|
//----------------------------------------------------------------
|
|
if (theme_vertical_scroll > 0)
|
|
pp2d_draw_texture(TEXTURE_ARROW, 155, 6);
|
|
if (theme_vertical_scroll + THEMES_PER_SCREEN < theme_count)
|
|
pp2d_draw_texture_flip(TEXTURE_ARROW, 155, 224, VERTICAL);
|
|
|
|
for (int i = theme_vertical_scroll; i < (THEMES_PER_SCREEN + theme_vertical_scroll); i++)
|
|
{
|
|
if (i >= theme_count)
|
|
break;
|
|
|
|
current_theme = themes_list[i];
|
|
wchar_t name[0x80] = {0};
|
|
utf16_to_utf32((u32*)name, current_theme.name, 0x80);
|
|
|
|
int vertical_offset = 48 * (i-theme_vertical_scroll);
|
|
u32 font_color = COLOR_WHITE;
|
|
|
|
if (i == selected_theme)
|
|
{
|
|
font_color = COLOR_BLACK;
|
|
pp2d_draw_rectangle(0, 24 + vertical_offset, 320, 48, COLOR_CURSOR);
|
|
}
|
|
pp2d_draw_wtext(54, 40 + vertical_offset, 0.55, 0.55, font_color, name);
|
|
if (!current_theme.placeholder_color)
|
|
pp2d_draw_texture(current_theme.icon_id, 0, 24 + vertical_offset);
|
|
else
|
|
pp2d_draw_rectangle(0, 24 + vertical_offset, 48, 48, current_theme.placeholder_color);
|
|
|
|
if (current_theme.in_shuffle)
|
|
pp2d_draw_texture_blend(TEXTURE_SHUFFLE, 280, 32 + vertical_offset, font_color);
|
|
}
|
|
}
|
|
|
|
pp2d_end_draw();
|
|
}
|
|
|
|
void draw_splash_install(int install_type)
|
|
{
|
|
draw_base_interface();
|
|
switch (install_type)
|
|
{
|
|
case SINGLE_INSTALL:
|
|
pp2d_draw_text_center(GFX_TOP, 110, 0.7, 0.7, COLOR_WHITE, "Installing a splash...");
|
|
break;
|
|
case UNINSTALL:
|
|
pp2d_draw_text_center(GFX_TOP, 110, 0.7, 0.7, COLOR_WHITE, "Uninstalling a splash...");
|
|
break;
|
|
default:
|
|
break;
|
|
}
|
|
pp2d_end_draw();
|
|
}
|
|
|
|
void draw_splash_interface(Splash_s *splashes_list, int splash_count, int selected_splash, bool preview_mode)
|
|
{
|
|
if (splashes_list == NULL)
|
|
{
|
|
pp2d_begin_draw(GFX_TOP);
|
|
pp2d_draw_text_center(GFX_TOP, 100, 1, 1, COLOR_WHITE, "NO SPLASHES FOUND");
|
|
pp2d_end_draw();
|
|
return;
|
|
}
|
|
|
|
Splash_s current_splash = splashes_list[selected_splash];
|
|
|
|
if (preview_mode)
|
|
{
|
|
// TODO: Splash Previews
|
|
} else {
|
|
draw_base_interface();
|
|
pp2d_draw_text_center(GFX_TOP, 4, 0.5, 0.5, COLOR_WHITE, "Splash mode");
|
|
wchar_t title[0x40] = {0};
|
|
utf16_to_utf32((u32*)title, current_splash.name, 0x40);
|
|
pp2d_draw_wtext_wrap(20, 30, 0.7, 0.7, COLOR_WHITE, 380, title);
|
|
wchar_t author[0x40] = {0};
|
|
utf16_to_utf32((u32*)author, current_splash.author, 0x40);
|
|
pp2d_draw_text(20, 50, 0.5, 0.5, COLOR_WHITE, "By: ");
|
|
pp2d_draw_wtext_wrap(44, 50, 0.5, 0.5, COLOR_WHITE, 380, author);
|
|
wchar_t description[0xa6] = {0};
|
|
utf16_to_utf32((u32*)description, current_splash.desc, 0xb0);
|
|
pp2d_draw_wtext_wrap(20, 65, 0.5, 0.5, COLOR_WHITE, 363, description);
|
|
|
|
pp2d_draw_wtext_center(GFX_TOP, 180, 0.7, 0.7, COLOR_WHITE, L"\uE000 Install Splash \uE004 Switch to Themes");
|
|
pp2d_draw_wtext_center(GFX_TOP, 210, 0.7, 0.7, COLOR_WHITE, L"\uE002 Delete current Splash");
|
|
pp2d_draw_on(GFX_BOTTOM);
|
|
for (int i = 0; i < splash_count; i++) {
|
|
if (splash_count <= THEMES_PER_SCREEN)
|
|
break;
|
|
|
|
if (splash_vertical_scroll > selected_splash)
|
|
splash_vertical_scroll--;
|
|
|
|
if ((i < selected_splash) && \
|
|
((selected_splash - splash_vertical_scroll) >= THEMES_PER_SCREEN) && \
|
|
(splash_vertical_scroll != ( - THEMES_PER_SCREEN)))
|
|
splash_vertical_scroll++;
|
|
}
|
|
|
|
if (splash_vertical_scroll > 0)
|
|
pp2d_draw_texture(TEXTURE_ARROW, 155, 6);
|
|
if (splash_vertical_scroll + THEMES_PER_SCREEN < splash_count)
|
|
pp2d_draw_texture_flip(TEXTURE_ARROW, 155, 224, VERTICAL);
|
|
|
|
for (int i = splash_vertical_scroll; i < (THEMES_PER_SCREEN + splash_vertical_scroll); i++)
|
|
{
|
|
if (i >= splash_count)
|
|
break;
|
|
|
|
current_splash = splashes_list[i];
|
|
wchar_t name[0x106] = {0};
|
|
utf16_to_utf32((u32*)name, current_splash.name, 0x106);
|
|
|
|
int vertical_offset = 48 * (i-splash_vertical_scroll);
|
|
u32 font_color = COLOR_WHITE;
|
|
|
|
if (i == selected_splash)
|
|
{
|
|
font_color = COLOR_BLACK;
|
|
pp2d_draw_rectangle(0, 24 + vertical_offset, 320, 48, COLOR_CURSOR);
|
|
}
|
|
pp2d_draw_wtext(54, 40 + vertical_offset, 0.55, 0.55, font_color, name);
|
|
|
|
if (!current_splash.placeholder_color)
|
|
pp2d_draw_texture(current_splash.icon_id, 0, 24 + vertical_offset);
|
|
else
|
|
pp2d_draw_rectangle(0, 24 + vertical_offset, 48, 48, current_splash.placeholder_color);
|
|
}
|
|
}
|
|
pp2d_end_draw();
|
|
}
|