mirror of
				https://github.com/Atmosphere-NX/Atmosphere.git
				synced 2025-10-22 16:35:46 +02:00 
			
		
		
		
	
		
			
				
	
	
		
			1280 lines
		
	
	
		
			50 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			1280 lines
		
	
	
		
			50 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /*
 | |
|  * Copyright (c) 2020 Adubbz
 | |
|  *
 | |
|  * This program is free software; you can redistribute it and/or modify it
 | |
|  * under the terms and conditions of the GNU General Public License,
 | |
|  * version 2, as published by the Free Software Foundation.
 | |
|  *
 | |
|  * This program is distributed in the hope 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/>.
 | |
|  */
 | |
| #include <algorithm>
 | |
| #include <cstdarg>
 | |
| #include <cstdio>
 | |
| #include <cstring>
 | |
| #include <dirent.h>
 | |
| #include "ui.hpp"
 | |
| #include "ui_util.hpp"
 | |
| #include "assert.hpp"
 | |
| 
 | |
| namespace dbk {
 | |
| 
 | |
|     namespace {
 | |
| 
 | |
|         static constexpr u32 ExosphereApiVersionConfigItem = 65000;
 | |
|         static constexpr u32 ExosphereHasRcmBugPatch       = 65004;
 | |
|         static constexpr u32 ExosphereEmummcType           = 65007;
 | |
| 
 | |
|         /* Insets of content within windows. */
 | |
|         static constexpr float HorizontalInset       = 20.0f;
 | |
|         static constexpr float BottomInset           = 20.0f;
 | |
| 
 | |
|         /* Insets of content within text areas. */
 | |
|         static constexpr float TextHorizontalInset   = 8.0f;
 | |
|         static constexpr float TextVerticalInset     = 8.0f;
 | |
| 
 | |
|         static constexpr float ButtonHeight          = 60.0f;
 | |
|         static constexpr float ButtonHorizontalGap   = 10.0f;
 | |
| 
 | |
|         static constexpr float VerticalGap           = 10.0f;
 | |
| 
 | |
|         u32 g_screen_width;
 | |
|         u32 g_screen_height;
 | |
| 
 | |
|         std::shared_ptr<Menu> g_current_menu;
 | |
|         bool g_initialized = false;
 | |
|         bool g_exit_requested = false;
 | |
| 
 | |
|         PadState g_pad;
 | |
| 
 | |
|         u32 g_prev_touch_count = -1;
 | |
|         HidTouchScreenState g_start_touch;
 | |
|         bool g_started_touching = false;
 | |
|         bool g_tapping = false;
 | |
|         bool g_touches_moving = false;
 | |
|         bool g_finished_touching = false;
 | |
| 
 | |
|         /* Update install state. */
 | |
|         char g_update_path[FS_MAX_PATH];
 | |
|         bool g_reset_to_factory = false;
 | |
|         bool g_exfat_supported = false;
 | |
|         bool g_use_exfat = false;
 | |
| 
 | |
|         constexpr u32 MaxTapMovement = 20;
 | |
| 
 | |
|         void UpdateInput() {
 | |
|             /* Scan for input and update touch state. */
 | |
|             padUpdate(&g_pad);
 | |
|             HidTouchScreenState current_touch;
 | |
|             hidGetTouchScreenStates(¤t_touch, 1);
 | |
|             const u32 touch_count = current_touch.count;
 | |
| 
 | |
|             if (g_prev_touch_count == 0 && touch_count > 0) {
 | |
|                 hidGetTouchScreenStates(&g_start_touch, 1);
 | |
|                 g_started_touching = true;
 | |
|                 g_tapping = true;
 | |
|             } else {
 | |
|                 g_started_touching = false;
 | |
|             }
 | |
| 
 | |
|             if (g_prev_touch_count > 0 && touch_count == 0) {
 | |
|                 g_finished_touching = true;
 | |
|                 g_tapping = false;
 | |
|             } else {
 | |
|                 g_finished_touching = false;
 | |
|             }
 | |
| 
 | |
|             /* Check if currently moving. */
 | |
|             if (g_prev_touch_count > 0 && touch_count > 0) {
 | |
|                 if ((abs(current_touch.touches[0].x - g_start_touch.touches[0].x) > MaxTapMovement || abs(current_touch.touches[0].y - g_start_touch.touches[0].y) > MaxTapMovement)) {
 | |
|                     g_touches_moving = true;
 | |
|                     g_tapping = false;
 | |
|                 } else {
 | |
|                     g_touches_moving = false;
 | |
|                 }
 | |
|             } else {
 | |
|                 g_touches_moving = false;
 | |
|             }
 | |
| 
 | |
|             /* Update the previous touch count. */
 | |
|             g_prev_touch_count = current_touch.count;
 | |
|         }
 | |
| 
 | |
|         void ChangeMenu(std::shared_ptr<Menu> menu) {
 | |
|             g_current_menu = menu;
 | |
|         }
 | |
| 
 | |
|         void ReturnToPreviousMenu() {
 | |
|             /* Go to the previous menu if there is one. */
 | |
|             if (g_current_menu->GetPrevMenu() != nullptr) {
 | |
|                 g_current_menu = g_current_menu->GetPrevMenu();
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         Result IsPathBottomLevel(const char *path, bool *out) {
 | |
|             Result rc = 0;
 | |
|             FsFileSystem *fs;
 | |
|             char translated_path[FS_MAX_PATH] = {};
 | |
|             DBK_ABORT_UNLESS(fsdevTranslatePath(path, &fs, translated_path) != -1);
 | |
| 
 | |
|             FsDir dir;
 | |
|             if (R_FAILED(rc = fsFsOpenDirectory(fs, translated_path, FsDirOpenMode_ReadDirs, &dir))) {
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             s64 entry_count;
 | |
|             if (R_FAILED(rc = fsDirGetEntryCount(&dir, &entry_count))) {
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             *out = entry_count == 0;
 | |
|             fsDirClose(&dir);
 | |
|             return rc;
 | |
|         }
 | |
| 
 | |
|     }
 | |
| 
 | |
|     void Menu::AddButton(u32 id, const char *text, float x, float y, float w, float h) {
 | |
|         DBK_ABORT_UNLESS(id < MaxButtons);
 | |
|         Button button = {
 | |
|             .id = id,
 | |
|             .selected = false,
 | |
|             .enabled = true,
 | |
|             .x = x,
 | |
|             .y = y,
 | |
|             .w = w,
 | |
|             .h = h,
 | |
|         };
 | |
| 
 | |
|         strncpy(button.text, text, sizeof(button.text)-1);
 | |
|         m_buttons[id] = button;
 | |
|     }
 | |
| 
 | |
|     void Menu::SetButtonSelected(u32 id, bool selected) {
 | |
|         DBK_ABORT_UNLESS(id < MaxButtons);
 | |
|         auto &button = m_buttons[id];
 | |
| 
 | |
|         if (button) {
 | |
|             button->selected = selected;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Menu::DeselectAllButtons() {
 | |
|         for (auto &button : m_buttons) {
 | |
|             /* Ensure button is present. */
 | |
|             if (!button) {
 | |
|                 continue;
 | |
|             }
 | |
|             button->selected = false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Menu::SetButtonEnabled(u32 id, bool enabled) {
 | |
|         DBK_ABORT_UNLESS(id < MaxButtons);
 | |
|         auto &button = m_buttons[id];
 | |
|         button->enabled = enabled;
 | |
|     }
 | |
| 
 | |
|     Button *Menu::GetButton(u32 id) {
 | |
|         DBK_ABORT_UNLESS(id < MaxButtons);
 | |
|         return !m_buttons[id] ? nullptr : &(*m_buttons[id]);
 | |
|     }
 | |
| 
 | |
|     Button *Menu::GetSelectedButton() {
 | |
|         for (auto &button : m_buttons) {
 | |
|             if (button && button->enabled && button->selected) {
 | |
|                 return &(*button);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     Button *Menu::GetClosestButtonToSelection(Direction direction) {
 | |
|         const Button *selected_button = this->GetSelectedButton();
 | |
| 
 | |
|         if (selected_button == nullptr || direction == Direction::Invalid) {
 | |
|             return nullptr;
 | |
|         }
 | |
| 
 | |
|         Button *closest_button = nullptr;
 | |
|         float closest_distance = 0.0f;
 | |
| 
 | |
|         for (auto &button : m_buttons) {
 | |
|             /* Skip absent button. */
 | |
|             if (!button || !button->enabled) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* Skip buttons that are in the wrong direction. */
 | |
|             if ((direction == Direction::Down && button->y <= selected_button->y)  ||
 | |
|                 (direction == Direction::Up && button->y >= selected_button->y)    ||
 | |
|                 (direction == Direction::Right && button->x <= selected_button->x) ||
 | |
|                 (direction == Direction::Left && button->x >= selected_button->x)) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             const float x_dist = button->x - selected_button->x;
 | |
|             const float y_dist = button->y - selected_button->y;
 | |
|             const float sq_dist = x_dist * x_dist + y_dist * y_dist;
 | |
| 
 | |
|             /* If we don't already have a closest button, set it. */
 | |
|             if (closest_button == nullptr) {
 | |
|                 closest_button = &(*button);
 | |
|                 closest_distance = sq_dist;
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* Update the closest button if this one is closer. */
 | |
|             if (sq_dist < closest_distance) {
 | |
|                 closest_button = &(*button);
 | |
|                 closest_distance = sq_dist;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return closest_button;
 | |
|     }
 | |
| 
 | |
|     Button *Menu::GetTouchedButton() {
 | |
|         HidTouchScreenState current_touch;
 | |
|         hidGetTouchScreenStates(¤t_touch, 1);
 | |
|         const u32 touch_count = current_touch.count;
 | |
| 
 | |
|         for (u32 i = 0; i < touch_count && g_started_touching; i++) {
 | |
|             for (auto &button : m_buttons) {
 | |
|                 if (button && button->enabled && button->IsPositionInBounds(current_touch.touches[i].x, current_touch.touches[i].y)) {
 | |
|                     return &(*button);
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     Button *Menu::GetActivatedButton() {
 | |
|         Button *selected_button = this->GetSelectedButton();
 | |
| 
 | |
|         if (selected_button == nullptr) {
 | |
|             return nullptr;
 | |
|         }
 | |
| 
 | |
|         const u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         if (k_down & HidNpadButton_A || this->GetTouchedButton() == selected_button) {
 | |
|             return selected_button;
 | |
|         }
 | |
| 
 | |
|         return nullptr;
 | |
|     }
 | |
| 
 | |
|     void Menu::UpdateButtons() {
 | |
|         const u64 k_down = padGetButtonsDown(&g_pad);
 | |
|         Direction direction = Direction::Invalid;
 | |
| 
 | |
|         if (k_down & HidNpadButton_AnyDown) {
 | |
|             direction = Direction::Down;
 | |
|         } else if (k_down & HidNpadButton_AnyUp) {
 | |
|             direction = Direction::Up;
 | |
|         } else if (k_down & HidNpadButton_AnyLeft) {
 | |
|             direction = Direction::Left;
 | |
|         } else if (k_down & HidNpadButton_AnyRight) {
 | |
|             direction = Direction::Right;
 | |
|         }
 | |
| 
 | |
|         /* Select the closest button. */
 | |
|         if (const Button *closest_button = this->GetClosestButtonToSelection(direction); closest_button != nullptr) {
 | |
|             this->DeselectAllButtons();
 | |
|             this->SetButtonSelected(closest_button->id, true);
 | |
|         }
 | |
| 
 | |
|         /* Select the touched button. */
 | |
|         if (const Button *touched_button = this->GetTouchedButton(); touched_button != nullptr) {
 | |
|             this->DeselectAllButtons();
 | |
|             this->SetButtonSelected(touched_button->id, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Menu::DrawButtons(NVGcontext *vg, u64 ns) {
 | |
|         for (auto &button : m_buttons) {
 | |
|             /* Ensure button is present. */
 | |
|             if (!button) {
 | |
|                 continue;
 | |
|             }
 | |
| 
 | |
|             /* Set the button style. */
 | |
|             auto style = ButtonStyle::StandardDisabled;
 | |
|             if (button->enabled) {
 | |
|                 style = button->selected ? ButtonStyle::StandardSelected : ButtonStyle::Standard;
 | |
|             }
 | |
| 
 | |
|             DrawButton(vg, button->text, button->x, button->y, button->w, button->h, style, ns);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void Menu::LogText(const char *format, ...) {
 | |
|         /* Create a temporary string. */
 | |
|         char tmp[0x100];
 | |
|         va_list args;
 | |
|         va_start(args, format);
 | |
|         vsnprintf(tmp, sizeof(tmp), format, args);
 | |
|         va_end(args);
 | |
| 
 | |
|         /* Append the text to the log buffer. */
 | |
|         strncat(m_log_buffer, tmp, sizeof(m_log_buffer)-1);
 | |
|     }
 | |
| 
 | |
|     std::shared_ptr<Menu> Menu::GetPrevMenu() {
 | |
|         return m_prev_menu;
 | |
|     }
 | |
| 
 | |
|     AlertMenu::AlertMenu(std::shared_ptr<Menu> prev_menu, const char *text, const char *subtext, Result rc) : Menu(prev_menu), m_text{}, m_subtext{}, m_result_text{}, m_rc(rc){
 | |
|         /* Copy the input text. */
 | |
|         strncpy(m_text, text, sizeof(m_text)-1);
 | |
|         strncpy(m_subtext, subtext, sizeof(m_subtext)-1);
 | |
| 
 | |
|         /* Copy result text if there is a result. */
 | |
|         if (R_FAILED(rc)) {
 | |
|             snprintf(m_result_text, sizeof(m_result_text), "Result: 0x%08x", rc);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void AlertMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float window_height = WindowHeight + (R_FAILED(m_rc) ? SubTextHeight : 0.0f);
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - window_height / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, m_text, x, y, WindowWidth, window_height);
 | |
|         DrawText(vg, x + HorizontalInset, y + TitleGap, WindowWidth - HorizontalInset * 2.0f, m_subtext);
 | |
| 
 | |
|         /* Draw the result if there is one. */
 | |
|         if (R_FAILED(m_rc)) {
 | |
|             DrawText(vg, x + HorizontalInset, y + TitleGap + SubTextHeight, WindowWidth - HorizontalInset * 2.0f, m_result_text);
 | |
|         }
 | |
| 
 | |
|         this->DrawButtons(vg, ns);
 | |
|     }
 | |
| 
 | |
|     ErrorMenu::ErrorMenu(const char *text, const char *subtext, Result rc) : AlertMenu(nullptr, text, subtext, rc)  {
 | |
|         const float window_height = WindowHeight + (R_FAILED(m_rc) ? SubTextHeight : 0.0f);
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - window_height / 2.0f;
 | |
|         const float button_y = y + TitleGap + SubTextHeight + VerticalGap * 2.0f + (R_FAILED(m_rc) ? SubTextHeight : 0.0f);
 | |
|         const float button_width = WindowWidth - HorizontalInset * 2.0f;
 | |
| 
 | |
|         /* Add buttons. */
 | |
|         this->AddButton(ExitButtonId, "Exit", x + HorizontalInset, button_y, button_width, ButtonHeight);
 | |
|         this->SetButtonSelected(ExitButtonId, true);
 | |
|     }
 | |
| 
 | |
|     void ErrorMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             g_exit_requested = true;
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case ExitButtonId:
 | |
|                     g_exit_requested = true;
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
| 
 | |
|         /* Fallback on selecting the exfat button. */
 | |
|         if (const Button *selected_button = this->GetSelectedButton(); k_down && selected_button == nullptr) {
 | |
|             this->SetButtonSelected(ExitButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     WarningMenu::WarningMenu(std::shared_ptr<Menu> prev_menu, std::shared_ptr<Menu> next_menu, const char *text, const char *subtext, Result rc) : AlertMenu(prev_menu, text, subtext, rc), m_next_menu(next_menu) {
 | |
|         const float window_height = WindowHeight + (R_FAILED(m_rc) ? SubTextHeight : 0.0f);
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - window_height / 2.0f;
 | |
| 
 | |
|         const float button_y = y + TitleGap + SubTextHeight + VerticalGap * 2.0f + (R_FAILED(m_rc) ? SubTextHeight : 0.0f);
 | |
|         const float button_width = (WindowWidth - HorizontalInset * 2.0f) / 2.0f - ButtonHorizontalGap;
 | |
|         this->AddButton(BackButtonId, "Back", x + HorizontalInset, button_y, button_width, ButtonHeight);
 | |
|         this->AddButton(ContinueButtonId, "Continue", x + HorizontalInset + button_width + ButtonHorizontalGap, button_y, button_width, ButtonHeight);
 | |
|         this->SetButtonSelected(ContinueButtonId, true);
 | |
|     }
 | |
| 
 | |
|     void WarningMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             ReturnToPreviousMenu();
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case BackButtonId:
 | |
|                     ReturnToPreviousMenu();
 | |
|                     return;
 | |
|                 case ContinueButtonId:
 | |
|                     ChangeMenu(m_next_menu);
 | |
|                     return;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
| 
 | |
|         /* Fallback on selecting the exfat button. */
 | |
|         if (const Button *selected_button = this->GetSelectedButton(); k_down && selected_button == nullptr) {
 | |
|             this->SetButtonSelected(ContinueButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     MainMenu::MainMenu() : Menu(nullptr) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         this->AddButton(InstallButtonId, "Install", x + HorizontalInset, y + TitleGap, WindowWidth - HorizontalInset * 2, ButtonHeight);
 | |
|         this->AddButton(ExitButtonId, "Exit", x + HorizontalInset, y + TitleGap + ButtonHeight + VerticalGap, WindowWidth - HorizontalInset * 2, ButtonHeight);
 | |
|         this->SetButtonSelected(InstallButtonId, true);
 | |
|     }
 | |
| 
 | |
|     void MainMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             g_exit_requested = true;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case InstallButtonId:
 | |
|                 {
 | |
|                     const auto file_menu = std::make_shared<FileMenu>(g_current_menu, "/");
 | |
| 
 | |
|                     Result rc = 0;
 | |
|                     u64 hardware_type;
 | |
|                     u64 has_rcm_bug_patch;
 | |
|                     u64 is_emummc;
 | |
| 
 | |
|                     if (R_FAILED(rc = splGetConfig(SplConfigItem_HardwareType, &hardware_type))) {
 | |
|                         ChangeMenu(std::make_shared<ErrorMenu>("An error has occurred", "Failed to get hardware type.", rc));
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (R_FAILED(rc = splGetConfig(static_cast<SplConfigItem>(ExosphereHasRcmBugPatch), &has_rcm_bug_patch))) {
 | |
|                         ChangeMenu(std::make_shared<ErrorMenu>("An error has occurred", "Failed to check RCM bug status.", rc));
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     if (R_FAILED(rc = splGetConfig(static_cast<SplConfigItem>(ExosphereEmummcType), &is_emummc))) {
 | |
|                         ChangeMenu(std::make_shared<ErrorMenu>("An error has occurred", "Failed to check emuMMC status.", rc));
 | |
|                         return;
 | |
|                     }
 | |
| 
 | |
|                     /* Warn if we're working with a patched unit. */
 | |
|                     const bool is_erista = hardware_type == 0 || hardware_type == 1;
 | |
|                     if (is_erista && has_rcm_bug_patch && !is_emummc) {
 | |
|                         ChangeMenu(std::make_shared<WarningMenu>(g_current_menu, file_menu, "Warning: Patched unit detected", "You may burn fuses or render your switch inoperable."));
 | |
|                     } else {
 | |
|                         ChangeMenu(file_menu);
 | |
|                     }
 | |
| 
 | |
|                     return;
 | |
|                 }
 | |
|                 case ExitButtonId:
 | |
|                     g_exit_requested = true;
 | |
|                     return;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
| 
 | |
|         /* Fallback on selecting the install button. */
 | |
|         if (const Button *selected_button = this->GetSelectedButton(); k_down && selected_button == nullptr) {
 | |
|             this->SetButtonSelected(InstallButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void MainMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         DrawWindow(vg, "Daybreak", g_screen_width / 2.0f - WindowWidth / 2.0f, g_screen_height / 2.0f - WindowHeight / 2.0f, WindowWidth, WindowHeight);
 | |
|         this->DrawButtons(vg, ns);
 | |
|     }
 | |
| 
 | |
|     FileMenu::FileMenu(std::shared_ptr<Menu> prev_menu, const char *root) : Menu(prev_menu), m_current_index(0), m_scroll_offset(0), m_touch_start_scroll_offset(0), m_touch_finalize_selection(false) {
 | |
|         Result rc = 0;
 | |
| 
 | |
|         strncpy(m_root, root, sizeof(m_root)-1);
 | |
| 
 | |
|         if (R_FAILED(rc = this->PopulateFileEntries())) {
 | |
|             fatalThrow(rc);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Result FileMenu::PopulateFileEntries() {
 | |
|         /* Open the directory. */
 | |
|         DIR *dir = opendir(m_root);
 | |
|         if (dir == nullptr) {
 | |
|             return fsdevGetLastResult();
 | |
|         }
 | |
| 
 | |
|         /* Add file entries to the list. */
 | |
|         struct dirent *ent;
 | |
|         while ((ent = readdir(dir)) != nullptr) {
 | |
|             if (ent->d_type == DT_DIR) {
 | |
|                 FileEntry file_entry = {};
 | |
|                 strncpy(file_entry.name, ent->d_name, sizeof(file_entry.name));
 | |
|                 m_file_entries.push_back(file_entry);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Close the directory. */
 | |
|         closedir(dir);
 | |
| 
 | |
|         /* Sort the file entries. */
 | |
|         std::sort(m_file_entries.begin(), m_file_entries.end(), [](const FileEntry &a, const FileEntry &b) {
 | |
|             return strncmp(a.name, b.name, sizeof(a.name)) < 0;
 | |
|         });
 | |
| 
 | |
|         return 0;
 | |
|     }
 | |
| 
 | |
|     bool FileMenu::IsSelectionVisible() {
 | |
|         const float visible_start = m_scroll_offset;
 | |
|         const float visible_end = visible_start + FileListHeight;
 | |
|         const float entry_start = static_cast<float>(m_current_index) * (FileRowHeight + FileRowGap);
 | |
|         const float entry_end = entry_start + (FileRowHeight + FileRowGap);
 | |
|         return entry_start >= visible_start && entry_end <= visible_end;
 | |
|     }
 | |
| 
 | |
|     void FileMenu::ScrollToSelection() {
 | |
|         const float visible_start = m_scroll_offset;
 | |
|         const float visible_end = visible_start + FileListHeight;
 | |
|         const float entry_start = static_cast<float>(m_current_index) * (FileRowHeight + FileRowGap);
 | |
|         const float entry_end = entry_start + (FileRowHeight + FileRowGap);
 | |
| 
 | |
|         if (entry_end > visible_end) {
 | |
|             m_scroll_offset += entry_end - visible_end;
 | |
|         } else if (entry_end < visible_end) {
 | |
|             m_scroll_offset = entry_start;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     bool FileMenu::IsEntryTouched(u32 i) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         HidTouchScreenState current_touch;
 | |
|         hidGetTouchScreenStates(¤t_touch, 1);
 | |
| 
 | |
|         /* Check if the tap is within the x bounds. */
 | |
|         if (current_touch.touches[0].x >= x + TextBackgroundOffset + FileRowHorizontalInset && current_touch.touches[0].x <= WindowWidth - (TextBackgroundOffset + FileRowHorizontalInset) * 2.0f) {
 | |
|             const float y_min = y + TitleGap + FileRowGap + i * (FileRowHeight + FileRowGap) - m_scroll_offset;
 | |
|             const float y_max = y_min + FileRowHeight;
 | |
| 
 | |
|             /* Check if the tap is within the y bounds. */
 | |
|             if (current_touch.touches[0].y >= y_min && current_touch.touches[0].y <= y_max) {
 | |
|                 return true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         return false;
 | |
|     }
 | |
| 
 | |
|     void FileMenu::UpdateTouches() {
 | |
|         /* Setup values on initial touch. */
 | |
|         if (g_started_touching) {
 | |
|             m_touch_start_scroll_offset = m_scroll_offset;
 | |
| 
 | |
|             /* We may potentially finalize the selection later if we start off touching it. */
 | |
|             if (this->IsEntryTouched(m_current_index)) {
 | |
|                 m_touch_finalize_selection = true;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Scroll based on touch movement. */
 | |
|         if (g_touches_moving) {
 | |
|             HidTouchScreenState current_touch;
 | |
|             hidGetTouchScreenStates(¤t_touch, 1);
 | |
| 
 | |
|             const int dist_y = current_touch.touches[0].y - g_start_touch.touches[0].y;
 | |
|             float new_scroll_offset = m_touch_start_scroll_offset - static_cast<float>(dist_y);
 | |
|             float max_scroll = (FileRowHeight + FileRowGap) * static_cast<float>(m_file_entries.size()) - FileListHeight;
 | |
| 
 | |
|             /* Don't allow scrolling if there is not enough elements. */
 | |
|             if (max_scroll < 0.0f) {
 | |
|                 max_scroll = 0.0f;
 | |
|             }
 | |
| 
 | |
|             /* Don't allow scrolling before the first element. */
 | |
|             if (new_scroll_offset < 0.0f) {
 | |
|                 new_scroll_offset = 0.0f;
 | |
|             }
 | |
| 
 | |
|             /* Don't allow scrolling past the last element. */
 | |
|             if (new_scroll_offset > max_scroll) {
 | |
|                 new_scroll_offset = max_scroll;
 | |
|             }
 | |
| 
 | |
|             m_scroll_offset = new_scroll_offset;
 | |
|         }
 | |
| 
 | |
|         /* Select any tapped entries. */
 | |
|         if (g_tapping) {
 | |
|             for (u32 i = 0; i < m_file_entries.size(); i++) {
 | |
|                 if (this->IsEntryTouched(i)) {
 | |
|                     /* The current index is checked later. */
 | |
|                     if (i == m_current_index) {
 | |
|                         continue;
 | |
|                     }
 | |
| 
 | |
|                     m_current_index = i;
 | |
| 
 | |
|                     /* Don't finalize selection if we touch something else. */
 | |
|                     m_touch_finalize_selection = false;
 | |
|                     break;
 | |
|                 }
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Don't finalize selection if we aren't finished and we've either stopped tapping or are no longer touching the selection. */
 | |
|         if (!g_finished_touching && (!g_tapping || !this->IsEntryTouched(m_current_index))) {
 | |
|             m_touch_finalize_selection = false;
 | |
|         }
 | |
| 
 | |
|         /* Finalize selection if the currently selected entry is touched for the second time. */
 | |
|         if (g_finished_touching && m_touch_finalize_selection) {
 | |
|             this->FinalizeSelection();
 | |
|             m_touch_finalize_selection = false;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void FileMenu::FinalizeSelection() {
 | |
|         DBK_ABORT_UNLESS(m_current_index < m_file_entries.size());
 | |
|         FileEntry &entry = m_file_entries[m_current_index];
 | |
| 
 | |
|         /* Determine the selected path. */
 | |
|         char current_path[FS_MAX_PATH] = {};
 | |
|         const int path_len = snprintf(current_path, sizeof(current_path), "%s%s/", m_root, entry.name);
 | |
|         DBK_ABORT_UNLESS(path_len >= 0 && path_len < static_cast<int>(sizeof(current_path)));
 | |
| 
 | |
|         /* Determine if the chosen path is the bottom level. */
 | |
|         Result rc = 0;
 | |
|         bool bottom_level;
 | |
|         if (R_FAILED(rc = IsPathBottomLevel(current_path, &bottom_level))) {
 | |
|             fatalThrow(rc);
 | |
|         }
 | |
| 
 | |
|         /* Show exfat settings or the next file menu. */
 | |
|         if (bottom_level) {
 | |
|             /* Set the update path. */
 | |
|             snprintf(g_update_path, sizeof(g_update_path), "%s", current_path);
 | |
| 
 | |
|             /* Change the menu. */
 | |
|             ChangeMenu(std::make_shared<ValidateUpdateMenu>(g_current_menu));
 | |
|         } else {
 | |
|             ChangeMenu(std::make_shared<FileMenu>(g_current_menu, current_path));
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void FileMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             ReturnToPreviousMenu();
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Finalize selection on pressing A. */
 | |
|         if (k_down & HidNpadButton_A) {
 | |
|             this->FinalizeSelection();
 | |
|         }
 | |
| 
 | |
|         /* Update touch input. */
 | |
|         this->UpdateTouches();
 | |
| 
 | |
|         const u32 prev_index = m_current_index;
 | |
| 
 | |
|         if (k_down & HidNpadButton_AnyDown) {
 | |
|             /* Scroll down. */
 | |
|             if (m_current_index >= (m_file_entries.size() - 1)) {
 | |
|                 m_current_index = 0;
 | |
|             } else {
 | |
|                 m_current_index++;
 | |
|             }
 | |
|         } else if (k_down & HidNpadButton_AnyUp) {
 | |
|             /* Scroll up. */
 | |
|             if (m_current_index == 0) {
 | |
|                 m_current_index = m_file_entries.size() - 1;
 | |
|             } else {
 | |
|                 m_current_index--;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         /* Scroll to the selection if it isn't visible. */
 | |
|         if (prev_index != m_current_index && !this->IsSelectionVisible()) {
 | |
|             this->ScrollToSelection();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void FileMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, "Select an update directory", x, y, WindowWidth, WindowHeight);
 | |
|         DrawTextBackground(vg, x + TextBackgroundOffset, y + TitleGap, WindowWidth - TextBackgroundOffset * 2.0f, (FileRowHeight + FileRowGap) * MaxFileRows + FileRowGap);
 | |
| 
 | |
|         nvgSave(vg);
 | |
|         nvgScissor(vg, x + TextBackgroundOffset, y + TitleGap, WindowWidth - TextBackgroundOffset * 2.0f, (FileRowHeight + FileRowGap) * MaxFileRows + FileRowGap);
 | |
| 
 | |
|         for (u32 i = 0; i < m_file_entries.size(); i++) {
 | |
|             FileEntry &entry = m_file_entries[i];
 | |
|             auto style = ButtonStyle::FileSelect;
 | |
| 
 | |
|             if (i == m_current_index) {
 | |
|                 style = ButtonStyle::FileSelectSelected;
 | |
|             }
 | |
| 
 | |
|             DrawButton(vg, entry.name, x + TextBackgroundOffset + FileRowHorizontalInset, y + TitleGap + FileRowGap + i * (FileRowHeight + FileRowGap) - m_scroll_offset, WindowWidth - (TextBackgroundOffset + FileRowHorizontalInset) * 2.0f, FileRowHeight, style, ns);
 | |
|         }
 | |
| 
 | |
|         nvgRestore(vg);
 | |
|     }
 | |
| 
 | |
|     ValidateUpdateMenu::ValidateUpdateMenu(std::shared_ptr<Menu> prev_menu) : Menu(prev_menu), m_has_drawn(false), m_has_info(false), m_has_validated(false) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
|         const float button_width = (WindowWidth - HorizontalInset * 2.0f) / 2.0f - ButtonHorizontalGap;
 | |
| 
 | |
|         /* Add buttons. */
 | |
|         this->AddButton(BackButtonId, "Back", x + HorizontalInset, y + WindowHeight - BottomInset - ButtonHeight, button_width, ButtonHeight);
 | |
|         this->AddButton(ContinueButtonId, "Continue", x + HorizontalInset + button_width + ButtonHorizontalGap, y + WindowHeight - BottomInset - ButtonHeight, button_width, ButtonHeight);
 | |
|         this->SetButtonEnabled(BackButtonId, false);
 | |
|         this->SetButtonEnabled(ContinueButtonId, false);
 | |
| 
 | |
|         /* Obtain update information. */
 | |
|         if (R_FAILED(this->GetUpdateInformation())) {
 | |
|             this->SetButtonEnabled(BackButtonId, true);
 | |
|             this->SetButtonSelected(BackButtonId, true);
 | |
|         } else {
 | |
|             /* Log this early so it is printed out before validation causes stalling. */
 | |
|             this->LogText("Validating update, this may take a moment...\n");
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     Result ValidateUpdateMenu::GetUpdateInformation() {
 | |
|         Result rc = 0;
 | |
|         this->LogText("Directory %s\n", g_update_path);
 | |
| 
 | |
|         /* Attempt to get the update information. */
 | |
|         if (R_FAILED(rc = amssuGetUpdateInformation(&m_update_info, g_update_path))) {
 | |
|             if (rc == 0x1a405) {
 | |
|                 this->LogText("No update found in folder.\nEnsure your ncas are named correctly!\nResult: 0x%08x\n", rc);
 | |
|             } else {
 | |
|                 this->LogText("Failed to get update information.\nResult: 0x%08x\n", rc);
 | |
|             }
 | |
|             return rc;
 | |
|         }
 | |
| 
 | |
|         /* Print update information. */
 | |
|         this->LogText("- Version: %d.%d.%d\n", (m_update_info.version >> 26) & 0x1f, (m_update_info.version >> 20) & 0x1f, (m_update_info.version >> 16) & 0xf);
 | |
|         if (m_update_info.exfat_supported) {
 | |
|             this->LogText("- exFAT: Supported\n");
 | |
|         } else {
 | |
|             this->LogText("- exFAT: Unsupported\n");
 | |
|         }
 | |
|         this->LogText("- Firmware variations: %d\n", m_update_info.num_firmware_variations);
 | |
| 
 | |
|         /* Mark as having obtained update info. */
 | |
|         m_has_info = true;
 | |
|         return rc;
 | |
|     }
 | |
| 
 | |
|     void ValidateUpdateMenu::ValidateUpdate() {
 | |
|         Result rc = 0;
 | |
| 
 | |
|         /* Validate the update. */
 | |
|         if (R_FAILED(rc = amssuValidateUpdate(&m_validation_info, g_update_path))) {
 | |
|             this->LogText("Failed to validate update.\nResult: 0x%08x\n", rc);
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Check the result. */
 | |
|         if (R_SUCCEEDED(m_validation_info.result)) {
 | |
|             this->LogText("Update is valid!\n");
 | |
| 
 | |
|             if (R_FAILED(m_validation_info.exfat_result)) {
 | |
|                 const u32 version = m_validation_info.invalid_key.version;
 | |
|                 this->LogText("exFAT Validation failed with result: 0x%08x\n", m_validation_info.exfat_result);
 | |
|                 this->LogText("Missing content:\n- Program id: %016lx\n- Version: %d.%d.%d\n", m_validation_info.invalid_key.id, (version >> 26) & 0x1f, (version >> 20) & 0x1f, (version >> 16) & 0xf);
 | |
| 
 | |
|                 /* Log the missing content id. */
 | |
|                 this->LogText("- Content id: ");
 | |
|                 for (size_t i = 0; i < sizeof(NcmContentId); i++) {
 | |
|                     this->LogText("%02x", m_validation_info.invalid_content_id.c[i]);
 | |
|                 }
 | |
|                 this->LogText("\n");
 | |
|             }
 | |
| 
 | |
|             /* Enable the back and continue buttons and select the continue button. */
 | |
|             this->SetButtonEnabled(BackButtonId, true);
 | |
|             this->SetButtonEnabled(ContinueButtonId, true);
 | |
|             this->SetButtonSelected(ContinueButtonId, true);
 | |
|         } else {
 | |
|             /* Log the missing content info. */
 | |
|             const u32 version = m_validation_info.invalid_key.version;
 | |
|             this->LogText("Validation failed with result: 0x%08x\n", m_validation_info.result);
 | |
|             this->LogText("Missing content:\n- Program id: %016lx\n- Version: %d.%d.%d\n", m_validation_info.invalid_key.id, (version >> 26) & 0x1f, (version >> 20) & 0x1f, (version >> 16) & 0xf);
 | |
| 
 | |
|             /* Log the missing content id. */
 | |
|             this->LogText("- Content id: ");
 | |
|             for (size_t i = 0; i < sizeof(NcmContentId); i++) {
 | |
|                 this->LogText("%02x", m_validation_info.invalid_content_id.c[i]);
 | |
|             }
 | |
|             this->LogText("\n");
 | |
| 
 | |
|             /* Enable the back button and select it. */
 | |
|             this->SetButtonEnabled(BackButtonId, true);
 | |
|             this->SetButtonSelected(BackButtonId, true);
 | |
|         }
 | |
| 
 | |
|         /* Mark validation as being complete. */
 | |
|         m_has_validated = true;
 | |
|     }
 | |
| 
 | |
|     void ValidateUpdateMenu::Update(u64 ns) {
 | |
|         /* Perform validation if it hasn't been done already. */
 | |
|         if (m_has_info && m_has_drawn && !m_has_validated) {
 | |
|             this->ValidateUpdate();
 | |
|         }
 | |
| 
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             ReturnToPreviousMenu();
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case BackButtonId:
 | |
|                     ReturnToPreviousMenu();
 | |
|                     return;
 | |
|                 case ContinueButtonId:
 | |
|                     /* Don't continue if validation hasn't been done or has failed. */
 | |
|                     if (!m_has_validated || R_FAILED(m_validation_info.result)) {
 | |
|                         break;
 | |
|                     }
 | |
| 
 | |
|                     /* Check if exfat is supported. */
 | |
|                     g_exfat_supported = m_update_info.exfat_supported && R_SUCCEEDED(m_validation_info.exfat_result);
 | |
|                     if (!g_exfat_supported) {
 | |
|                         g_use_exfat = false;
 | |
|                     }
 | |
| 
 | |
|                     /* Warn the user if they're updating with exFAT supposed to be supported but not present/corrupted. */
 | |
|                     if (m_update_info.exfat_supported && R_FAILED(m_validation_info.exfat_result)) {
 | |
|                         ChangeMenu(std::make_shared<WarningMenu>(g_current_menu, std::make_shared<ChooseResetMenu>(g_current_menu), "Warning: exFAT firmware is missing or corrupt", "Are you sure you want to proceed?"));
 | |
|                     } else {
 | |
|                         ChangeMenu(std::make_shared<ChooseResetMenu>(g_current_menu));
 | |
|                     }
 | |
| 
 | |
|                     return;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
|     }
 | |
| 
 | |
|     void ValidateUpdateMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, "Update information", x, y, WindowWidth, WindowHeight);
 | |
|         DrawTextBackground(vg, x + HorizontalInset, y + TitleGap, WindowWidth - HorizontalInset * 2.0f, TextAreaHeight);
 | |
|         DrawTextBlock(vg, m_log_buffer, x + HorizontalInset + TextHorizontalInset, y + TitleGap + TextVerticalInset, WindowWidth - (HorizontalInset + TextHorizontalInset) * 2.0f, TextAreaHeight - TextVerticalInset * 2.0f);
 | |
| 
 | |
|         this->DrawButtons(vg, ns);
 | |
|         m_has_drawn = true;
 | |
|     }
 | |
| 
 | |
|     ChooseResetMenu::ChooseResetMenu(std::shared_ptr<Menu> prev_menu) : Menu(prev_menu) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
|         const float button_width = (WindowWidth - HorizontalInset * 2.0f) / 2.0f - ButtonHorizontalGap;
 | |
| 
 | |
|         /* Add buttons. */
 | |
|         this->AddButton(ResetToFactorySettingsButtonId, "Reset to factory settings", x + HorizontalInset, y + TitleGap, button_width, ButtonHeight);
 | |
|         this->AddButton(PreserveSettingsButtonId, "Preserve settings", x + HorizontalInset + button_width + ButtonHorizontalGap, y + TitleGap, button_width, ButtonHeight);
 | |
|         this->SetButtonSelected(PreserveSettingsButtonId, true);
 | |
|     }
 | |
| 
 | |
|     void ChooseResetMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             ReturnToPreviousMenu();
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case ResetToFactorySettingsButtonId:
 | |
|                     g_reset_to_factory = true;
 | |
|                     break;
 | |
|                 case PreserveSettingsButtonId:
 | |
|                     g_reset_to_factory = false;
 | |
|                     break;
 | |
|             }
 | |
| 
 | |
|             std::shared_ptr<Menu> next_menu;
 | |
| 
 | |
|             if (g_exfat_supported) {
 | |
|                 next_menu = std::make_shared<ChooseExfatMenu>(g_current_menu);
 | |
|             } else {
 | |
|                 next_menu = std::make_shared<WarningMenu>(g_current_menu, std::make_shared<InstallUpdateMenu>(g_current_menu), "Ready to begin update installation", "Are you sure you want to proceed?");
 | |
|             }
 | |
| 
 | |
|             if (g_reset_to_factory) {
 | |
|                 ChangeMenu(std::make_shared<WarningMenu>(g_current_menu, next_menu, "Warning: Factory reset selected", "Saves and installed games will be permanently deleted."));
 | |
|             } else {
 | |
|                 ChangeMenu(next_menu);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
| 
 | |
|         /* Fallback on selecting the exfat button. */
 | |
|         if (const Button *selected_button = this->GetSelectedButton(); k_down && selected_button == nullptr) {
 | |
|             this->SetButtonSelected(PreserveSettingsButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void ChooseResetMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, "Select settings mode", x, y, WindowWidth, WindowHeight);
 | |
|         this->DrawButtons(vg, ns);
 | |
|     }
 | |
| 
 | |
|     ChooseExfatMenu::ChooseExfatMenu(std::shared_ptr<Menu> prev_menu) : Menu(prev_menu) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
|         const float button_width = (WindowWidth - HorizontalInset * 2.0f) / 2.0f - ButtonHorizontalGap;
 | |
| 
 | |
|         /* Add buttons. */
 | |
|         this->AddButton(Fat32ButtonId, "Install (FAT32)", x + HorizontalInset, y + TitleGap, button_width, ButtonHeight);
 | |
|         this->AddButton(ExFatButtonId, "Install (FAT32 + exFAT)", x + HorizontalInset + button_width + ButtonHorizontalGap, y + TitleGap, button_width, ButtonHeight);
 | |
| 
 | |
|         /* Set the default selected button based on the user's current install. We aren't particularly concerned if fsIsExFatSupported fails. */
 | |
|         bool exfat_supported = false;
 | |
|         fsIsExFatSupported(&exfat_supported);
 | |
| 
 | |
|         if (exfat_supported) {
 | |
|             this->SetButtonSelected(ExFatButtonId, true);
 | |
|         } else {
 | |
|             this->SetButtonSelected(Fat32ButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void ChooseExfatMenu::Update(u64 ns) {
 | |
|         u64 k_down = padGetButtonsDown(&g_pad);
 | |
| 
 | |
|         /* Go back if B is pressed. */
 | |
|         if (k_down & HidNpadButton_B) {
 | |
|             ReturnToPreviousMenu();
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case Fat32ButtonId:
 | |
|                     g_use_exfat = false;
 | |
|                     break;
 | |
|                 case ExFatButtonId:
 | |
|                     g_use_exfat = true;
 | |
|                     break;
 | |
|             }
 | |
| 
 | |
|             ChangeMenu(std::make_shared<WarningMenu>(g_current_menu, std::make_shared<InstallUpdateMenu>(g_current_menu), "Ready to begin update installation", "Are you sure you want to proceed?"));
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
| 
 | |
|         /* Fallback on selecting the exfat button. */
 | |
|         if (const Button *selected_button = this->GetSelectedButton(); k_down && selected_button == nullptr) {
 | |
|             this->SetButtonSelected(ExFatButtonId, true);
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void ChooseExfatMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, "Select driver variant", x, y, WindowWidth, WindowHeight);
 | |
|         this->DrawButtons(vg, ns);
 | |
|     }
 | |
| 
 | |
|     InstallUpdateMenu::InstallUpdateMenu(std::shared_ptr<Menu> prev_menu) : Menu(prev_menu), m_install_state(InstallState::NeedsDraw), m_progress_percent(0.0f) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
|         const float button_width = (WindowWidth - HorizontalInset * 2.0f) / 2.0f - ButtonHorizontalGap;
 | |
| 
 | |
|         /* Add buttons. */
 | |
|         this->AddButton(ShutdownButtonId, "Shutdown", x + HorizontalInset, y + WindowHeight - BottomInset - ButtonHeight, button_width, ButtonHeight);
 | |
|         this->AddButton(RebootButtonId, "Reboot", x + HorizontalInset + button_width + ButtonHorizontalGap, y + WindowHeight - BottomInset - ButtonHeight, button_width, ButtonHeight);
 | |
|         this->SetButtonEnabled(ShutdownButtonId, false);
 | |
|         this->SetButtonEnabled(RebootButtonId, false);
 | |
| 
 | |
|         /* Prevent the home button from being pressed during installation. */
 | |
|         hiddbgDeactivateHomeButton();
 | |
|     }
 | |
| 
 | |
|     void InstallUpdateMenu::MarkForReboot() {
 | |
|         this->SetButtonEnabled(ShutdownButtonId, true);
 | |
|         this->SetButtonEnabled(RebootButtonId, true);
 | |
|         this->SetButtonSelected(RebootButtonId, true);
 | |
|         m_install_state = InstallState::AwaitingReboot;
 | |
|     }
 | |
| 
 | |
|     Result InstallUpdateMenu::TransitionUpdateState() {
 | |
|         Result rc = 0;
 | |
|         if (m_install_state == InstallState::NeedsSetup) {
 | |
|             /* Setup the update. */
 | |
|             if (R_FAILED(rc = amssuSetupUpdate(nullptr, UpdateTaskBufferSize, g_update_path, g_use_exfat))) {
 | |
|                 this->LogText("Failed to setup update.\nResult: 0x%08x\n", rc);
 | |
|                 this->MarkForReboot();
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             /* Log setup completion. */
 | |
|             this->LogText("Update setup complete.\n");
 | |
|             m_install_state = InstallState::NeedsPrepare;
 | |
|         } else if (m_install_state == InstallState::NeedsPrepare) {
 | |
|             /* Request update preparation. */
 | |
|             if (R_FAILED(rc = amssuRequestPrepareUpdate(&m_prepare_result))) {
 | |
|                 this->LogText("Failed to request update preparation.\nResult: 0x%08x\n", rc);
 | |
|                 this->MarkForReboot();
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             /* Log awaiting prepare. */
 | |
|             this->LogText("Preparing update...\n");
 | |
|             m_install_state = InstallState::AwaitingPrepare;
 | |
|         } else if (m_install_state == InstallState::AwaitingPrepare) {
 | |
|             /* Check if preparation has a result. */
 | |
|             if (R_FAILED(rc = asyncResultWait(&m_prepare_result, 0)) && rc != 0xea01) {
 | |
|                 this->LogText("Failed to check update preparation result.\nResult: 0x%08x\n", rc);
 | |
|                 this->MarkForReboot();
 | |
|                 return rc;
 | |
|             } else if (R_SUCCEEDED(rc)) {
 | |
|                 if (R_FAILED(rc = asyncResultGet(&m_prepare_result))) {
 | |
|                     this->LogText("Failed to prepare update.\nResult: 0x%08x\n", rc);
 | |
|                     this->MarkForReboot();
 | |
|                     return rc;
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             /* Check if the update has been prepared. */
 | |
|             bool prepared;
 | |
|             if (R_FAILED(rc = amssuHasPreparedUpdate(&prepared))) {
 | |
|                 this->LogText("Failed to check if update has been prepared.\nResult: 0x%08x\n", rc);
 | |
|                 this->MarkForReboot();
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             /* Mark for application if preparation complete. */
 | |
|             if (prepared) {
 | |
|                 this->LogText("Update preparation complete.\nApplying update...\n");
 | |
|                 m_install_state = InstallState::NeedsApply;
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             /* Check update progress. */
 | |
|             NsSystemUpdateProgress update_progress = {};
 | |
|             if (R_FAILED(rc = amssuGetPrepareUpdateProgress(&update_progress))) {
 | |
|                 this->LogText("Failed to check update progress.\nResult: 0x%08x\n", rc);
 | |
|                 this->MarkForReboot();
 | |
|                 return rc;
 | |
|             }
 | |
| 
 | |
|             /* Update progress percent. */
 | |
|             if (update_progress.total_size > 0.0f) {
 | |
|                 m_progress_percent = static_cast<float>(update_progress.current_size) / static_cast<float>(update_progress.total_size);
 | |
|             } else {
 | |
|                 m_progress_percent = 0.0f;
 | |
|             }
 | |
|         } else if (m_install_state == InstallState::NeedsApply) {
 | |
|             /* Apply the prepared update. */
 | |
|             if (R_FAILED(rc = amssuApplyPreparedUpdate())) {
 | |
|                 this->LogText("Failed to apply update.\nResult: 0x%08x\n", rc);
 | |
|             } else {
 | |
|                 /* Log success. */
 | |
|                 this->LogText("Update applied successfully.\n");
 | |
| 
 | |
|                 if (g_reset_to_factory) {
 | |
|                     if (R_FAILED(rc = nsResetToFactorySettingsForRefurbishment())) {
 | |
|                         /* Fallback on ResetToFactorySettings. */
 | |
|                         if (rc == MAKERESULT(Module_Libnx, LibnxError_IncompatSysVer)) {
 | |
|                             if (R_FAILED(rc = nsResetToFactorySettings())) {
 | |
|                                 this->LogText("Failed to reset to factory settings.\nResult: 0x%08x\n", rc);
 | |
|                                 this->MarkForReboot();
 | |
|                                 return rc;
 | |
|                             }
 | |
|                         } else {
 | |
|                             this->LogText("Failed to reset to factory settings for refurbishment.\nResult: 0x%08x\n", rc);
 | |
|                             this->MarkForReboot();
 | |
|                             return rc;
 | |
|                         }
 | |
|                     }
 | |
| 
 | |
|                     this->LogText("Successfully reset to factory settings.\n", rc);
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             this->MarkForReboot();
 | |
|             return rc;
 | |
|         }
 | |
| 
 | |
|         return rc;
 | |
|     }
 | |
| 
 | |
|     void InstallUpdateMenu::Update(u64 ns) {
 | |
|         /* Transition to the next update state. */
 | |
|         if (m_install_state != InstallState::NeedsDraw && m_install_state != InstallState::AwaitingReboot) {
 | |
|             this->TransitionUpdateState();
 | |
|         }
 | |
| 
 | |
|         /* Take action if a button has been activated. */
 | |
|         if (const Button *activated_button = this->GetActivatedButton(); activated_button != nullptr) {
 | |
|             switch (activated_button->id) {
 | |
|                 case ShutdownButtonId:
 | |
|                     if (R_FAILED(appletRequestToShutdown())) {
 | |
|                         spsmShutdown(false);
 | |
|                     }
 | |
|                     break;
 | |
|                 case RebootButtonId:
 | |
|                     if (R_FAILED(appletRequestToReboot())) {
 | |
|                         spsmShutdown(true);
 | |
|                     }
 | |
|                     break;
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         this->UpdateButtons();
 | |
|     }
 | |
| 
 | |
|     void InstallUpdateMenu::Draw(NVGcontext *vg, u64 ns) {
 | |
|         const float x = g_screen_width / 2.0f - WindowWidth / 2.0f;
 | |
|         const float y = g_screen_height / 2.0f - WindowHeight / 2.0f;
 | |
| 
 | |
|         DrawWindow(vg, "Installing update", x, y, WindowWidth, WindowHeight);
 | |
|         DrawProgressText(vg, x + HorizontalInset, y + TitleGap, m_progress_percent);
 | |
|         DrawProgressBar(vg, x + HorizontalInset, y + TitleGap + ProgressTextHeight, WindowWidth - HorizontalInset * 2.0f, ProgressBarHeight, m_progress_percent);
 | |
|         DrawTextBackground(vg, x + HorizontalInset, y + TitleGap + ProgressTextHeight + ProgressBarHeight + VerticalGap, WindowWidth - HorizontalInset * 2.0f, TextAreaHeight);
 | |
|         DrawTextBlock(vg, m_log_buffer, x + HorizontalInset + TextHorizontalInset, y + TitleGap + ProgressTextHeight + ProgressBarHeight + VerticalGap + TextVerticalInset, WindowWidth - (HorizontalInset + TextHorizontalInset) * 2.0f, TextAreaHeight - TextVerticalInset * 2.0f);
 | |
| 
 | |
|         this->DrawButtons(vg, ns);
 | |
| 
 | |
|         /* We have drawn now, allow setup to occur. */
 | |
|         if (m_install_state == InstallState::NeedsDraw) {
 | |
|             this->LogText("Beginning update setup...\n");
 | |
|             m_install_state = InstallState::NeedsSetup;
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     void InitializeMenu(u32 screen_width, u32 screen_height) {
 | |
|         Result rc = 0;
 | |
| 
 | |
|         /* Configure and initialize the gamepad. */
 | |
|         padConfigureInput(1, HidNpadStyleSet_NpadStandard);
 | |
|         padInitializeDefault(&g_pad);
 | |
| 
 | |
|         /* Initialize the touch screen. */
 | |
|         hidInitializeTouchScreen();
 | |
| 
 | |
|         /* Set the screen width and height. */
 | |
|         g_screen_width = screen_width;
 | |
|         g_screen_height = screen_height;
 | |
| 
 | |
|         /* Mark as initialized. */
 | |
|         g_initialized = true;
 | |
| 
 | |
|         /* Attempt to get the exosphere version. */
 | |
|         u64 version;
 | |
|         if (R_FAILED(rc = splGetConfig(static_cast<SplConfigItem>(ExosphereApiVersionConfigItem), &version))) {
 | |
|             ChangeMenu(std::make_shared<ErrorMenu>("Atmosphere not found", "Daybreak requires Atmosphere to be installed.", rc));
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         const u32 version_micro = (version >> 40) & 0xff;
 | |
|         const u32 version_minor = (version >> 48) & 0xff;
 | |
|         const u32 version_major = (version >> 56) & 0xff;
 | |
| 
 | |
|         /* Validate the exosphere version. */
 | |
|         const bool ams_supports_sysupdate_api = version_major >= 0 && version_minor >= 14 && version_micro >= 0;
 | |
|         if (!ams_supports_sysupdate_api) {
 | |
|             ChangeMenu(std::make_shared<ErrorMenu>("Outdated Atmosphere version", "Daybreak requires Atmosphere 0.14.0 or later.", rc));
 | |
|             return;
 | |
|         }
 | |
| 
 | |
|         /* Initialize ams:su. */
 | |
|         if (R_FAILED(rc = amssuInitialize())) {
 | |
|             fatalThrow(rc);
 | |
|         }
 | |
| 
 | |
|         /* Change the current menu to the main menu. */
 | |
|         g_current_menu = std::make_shared<MainMenu>();
 | |
|     }
 | |
| 
 | |
|     void UpdateMenu(u64 ns) {
 | |
|         DBK_ABORT_UNLESS(g_initialized);
 | |
|         DBK_ABORT_UNLESS(g_current_menu != nullptr);
 | |
|         UpdateInput();
 | |
|         g_current_menu->Update(ns);
 | |
|     }
 | |
| 
 | |
|     void RenderMenu(NVGcontext *vg, u64 ns) {
 | |
|         DBK_ABORT_UNLESS(g_initialized);
 | |
|         DBK_ABORT_UNLESS(g_current_menu != nullptr);
 | |
| 
 | |
|         /* Draw background. */
 | |
|         DrawBackground(vg, g_screen_width, g_screen_height);
 | |
| 
 | |
|         /* Draw stars. */
 | |
|         DrawStar(vg, 40.0f, 64.0f, 3.0f);
 | |
|         DrawStar(vg, 110.0f, 300.0f, 3.0f);
 | |
|         DrawStar(vg, 200.0f, 150.0f, 4.0f);
 | |
|         DrawStar(vg, 370.0f, 280.0f, 3.0f);
 | |
|         DrawStar(vg, 450.0f, 40.0f, 3.5f);
 | |
|         DrawStar(vg, 710.0f, 90.0f, 3.0f);
 | |
|         DrawStar(vg, 900.0f, 240.0f, 3.0f);
 | |
|         DrawStar(vg, 970.0f, 64.0f, 4.0f);
 | |
|         DrawStar(vg, 1160.0f, 160.0f, 3.5f);
 | |
|         DrawStar(vg, 1210.0f, 350.0f, 3.0f);
 | |
| 
 | |
|         g_current_menu->Draw(vg, ns);
 | |
|     }
 | |
| 
 | |
|     bool IsExitRequested() {
 | |
|         return g_exit_requested;
 | |
|     }
 | |
| 
 | |
| }
 |