Merge pull request #5042 from Morph1984/project-aether
Project Aether: Reimplementation of the Web Browser Applet
This commit is contained in:
commit
29ccc7673f
|
@ -135,6 +135,8 @@ add_library(core STATIC
|
||||||
frontend/emu_window.h
|
frontend/emu_window.h
|
||||||
frontend/framebuffer_layout.cpp
|
frontend/framebuffer_layout.cpp
|
||||||
frontend/framebuffer_layout.h
|
frontend/framebuffer_layout.h
|
||||||
|
frontend/input_interpreter.cpp
|
||||||
|
frontend/input_interpreter.h
|
||||||
frontend/input.h
|
frontend/input.h
|
||||||
hardware_interrupt_manager.cpp
|
hardware_interrupt_manager.cpp
|
||||||
hardware_interrupt_manager.h
|
hardware_interrupt_manager.h
|
||||||
|
|
|
@ -53,72 +53,4 @@ void DefaultPhotoViewerApplet::ShowAllPhotos(std::function<void()> finished) con
|
||||||
finished();
|
finished();
|
||||||
}
|
}
|
||||||
|
|
||||||
ECommerceApplet::~ECommerceApplet() = default;
|
|
||||||
|
|
||||||
DefaultECommerceApplet::~DefaultECommerceApplet() = default;
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowApplicationInformation(
|
|
||||||
std::function<void()> finished, u64 title_id, std::optional<u128> user_id,
|
|
||||||
std::optional<bool> full_display, std::optional<std::string> extra_parameter) {
|
|
||||||
const auto value = user_id.value_or(u128{});
|
|
||||||
LOG_INFO(Service_AM,
|
|
||||||
"Application requested frontend show application information for EShop, "
|
|
||||||
"title_id={:016X}, user_id={:016X}{:016X}, full_display={}, extra_parameter={}",
|
|
||||||
title_id, value[1], value[0],
|
|
||||||
full_display.has_value() ? fmt::format("{}", *full_display) : "null",
|
|
||||||
extra_parameter.value_or("null"));
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowAddOnContentList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id,
|
|
||||||
std::optional<bool> full_display) {
|
|
||||||
const auto value = user_id.value_or(u128{});
|
|
||||||
LOG_INFO(Service_AM,
|
|
||||||
"Application requested frontend show add on content list for EShop, "
|
|
||||||
"title_id={:016X}, user_id={:016X}{:016X}, full_display={}",
|
|
||||||
title_id, value[1], value[0],
|
|
||||||
full_display.has_value() ? fmt::format("{}", *full_display) : "null");
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowSubscriptionList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id) {
|
|
||||||
const auto value = user_id.value_or(u128{});
|
|
||||||
LOG_INFO(Service_AM,
|
|
||||||
"Application requested frontend show subscription list for EShop, title_id={:016X}, "
|
|
||||||
"user_id={:016X}{:016X}",
|
|
||||||
title_id, value[1], value[0]);
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowConsumableItemList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id) {
|
|
||||||
const auto value = user_id.value_or(u128{});
|
|
||||||
LOG_INFO(
|
|
||||||
Service_AM,
|
|
||||||
"Application requested frontend show consumable item list for EShop, title_id={:016X}, "
|
|
||||||
"user_id={:016X}{:016X}",
|
|
||||||
title_id, value[1], value[0]);
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowShopHome(std::function<void()> finished, u128 user_id,
|
|
||||||
bool full_display) {
|
|
||||||
LOG_INFO(Service_AM,
|
|
||||||
"Application requested frontend show home menu for EShop, user_id={:016X}{:016X}, "
|
|
||||||
"full_display={}",
|
|
||||||
user_id[1], user_id[0], full_display);
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
void DefaultECommerceApplet::ShowSettings(std::function<void()> finished, u128 user_id,
|
|
||||||
bool full_display) {
|
|
||||||
LOG_INFO(Service_AM,
|
|
||||||
"Application requested frontend show settings menu for EShop, user_id={:016X}{:016X}, "
|
|
||||||
"full_display={}",
|
|
||||||
user_id[1], user_id[0], full_display);
|
|
||||||
finished();
|
|
||||||
}
|
|
||||||
|
|
||||||
} // namespace Core::Frontend
|
} // namespace Core::Frontend
|
||||||
|
|
|
@ -58,55 +58,4 @@ public:
|
||||||
void ShowAllPhotos(std::function<void()> finished) const override;
|
void ShowAllPhotos(std::function<void()> finished) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
class ECommerceApplet {
|
|
||||||
public:
|
|
||||||
virtual ~ECommerceApplet();
|
|
||||||
|
|
||||||
// Shows a page with application icons, description, name, and price.
|
|
||||||
virtual void ShowApplicationInformation(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id = {},
|
|
||||||
std::optional<bool> full_display = {},
|
|
||||||
std::optional<std::string> extra_parameter = {}) = 0;
|
|
||||||
|
|
||||||
// Shows a page with all of the add on content available for a game, with name, description, and
|
|
||||||
// price.
|
|
||||||
virtual void ShowAddOnContentList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id = {},
|
|
||||||
std::optional<bool> full_display = {}) = 0;
|
|
||||||
|
|
||||||
// Shows a page with all of the subscriptions (recurring payments) for a game, with name,
|
|
||||||
// description, price, and renewal period.
|
|
||||||
virtual void ShowSubscriptionList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id = {}) = 0;
|
|
||||||
|
|
||||||
// Shows a page with a list of any additional game related purchasable items (DLC,
|
|
||||||
// subscriptions, etc) for a particular game, with name, description, type, and price.
|
|
||||||
virtual void ShowConsumableItemList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id = {}) = 0;
|
|
||||||
|
|
||||||
// Shows the home page of the shop.
|
|
||||||
virtual void ShowShopHome(std::function<void()> finished, u128 user_id, bool full_display) = 0;
|
|
||||||
|
|
||||||
// Shows the user settings page of the shop.
|
|
||||||
virtual void ShowSettings(std::function<void()> finished, u128 user_id, bool full_display) = 0;
|
|
||||||
};
|
|
||||||
|
|
||||||
class DefaultECommerceApplet : public ECommerceApplet {
|
|
||||||
public:
|
|
||||||
~DefaultECommerceApplet() override;
|
|
||||||
|
|
||||||
void ShowApplicationInformation(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id, std::optional<bool> full_display,
|
|
||||||
std::optional<std::string> extra_parameter) override;
|
|
||||||
void ShowAddOnContentList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id,
|
|
||||||
std::optional<bool> full_display) override;
|
|
||||||
void ShowSubscriptionList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id) override;
|
|
||||||
void ShowConsumableItemList(std::function<void()> finished, u64 title_id,
|
|
||||||
std::optional<u128> user_id) override;
|
|
||||||
void ShowShopHome(std::function<void()> finished, u128 user_id, bool full_display) override;
|
|
||||||
void ShowSettings(std::function<void()> finished, u128 user_id, bool full_display) override;
|
|
||||||
};
|
|
||||||
|
|
||||||
} // namespace Core::Frontend
|
} // namespace Core::Frontend
|
||||||
|
|
|
@ -11,14 +11,22 @@ WebBrowserApplet::~WebBrowserApplet() = default;
|
||||||
|
|
||||||
DefaultWebBrowserApplet::~DefaultWebBrowserApplet() = default;
|
DefaultWebBrowserApplet::~DefaultWebBrowserApplet() = default;
|
||||||
|
|
||||||
void DefaultWebBrowserApplet::OpenPageLocal(std::string_view filename,
|
void DefaultWebBrowserApplet::OpenLocalWebPage(
|
||||||
std::function<void()> unpack_romfs_callback,
|
std::string_view local_url, std::function<void()> extract_romfs_callback,
|
||||||
std::function<void()> finished_callback) {
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback) const {
|
||||||
LOG_INFO(Service_AM,
|
LOG_WARNING(Service_AM, "(STUBBED) called, backend requested to open local web page at {}",
|
||||||
"(STUBBED) called - No suitable web browser implementation found to open website page "
|
local_url);
|
||||||
"at '{}'!",
|
|
||||||
filename);
|
callback(Service::AM::Applets::WebExitReason::WindowClosed, "http://localhost/");
|
||||||
finished_callback();
|
}
|
||||||
|
|
||||||
|
void DefaultWebBrowserApplet::OpenExternalWebPage(
|
||||||
|
std::string_view external_url,
|
||||||
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback) const {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, backend requested to open external web page at {}",
|
||||||
|
external_url);
|
||||||
|
|
||||||
|
callback(Service::AM::Applets::WebExitReason::WindowClosed, "http://localhost/");
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace Core::Frontend
|
} // namespace Core::Frontend
|
||||||
|
|
|
@ -7,22 +7,34 @@
|
||||||
#include <functional>
|
#include <functional>
|
||||||
#include <string_view>
|
#include <string_view>
|
||||||
|
|
||||||
|
#include "core/hle/service/am/applets/web_types.h"
|
||||||
|
|
||||||
namespace Core::Frontend {
|
namespace Core::Frontend {
|
||||||
|
|
||||||
class WebBrowserApplet {
|
class WebBrowserApplet {
|
||||||
public:
|
public:
|
||||||
virtual ~WebBrowserApplet();
|
virtual ~WebBrowserApplet();
|
||||||
|
|
||||||
virtual void OpenPageLocal(std::string_view url, std::function<void()> unpack_romfs_callback,
|
virtual void OpenLocalWebPage(
|
||||||
std::function<void()> finished_callback) = 0;
|
std::string_view local_url, std::function<void()> extract_romfs_callback,
|
||||||
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback) const = 0;
|
||||||
|
|
||||||
|
virtual void OpenExternalWebPage(
|
||||||
|
std::string_view external_url,
|
||||||
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback) const = 0;
|
||||||
};
|
};
|
||||||
|
|
||||||
class DefaultWebBrowserApplet final : public WebBrowserApplet {
|
class DefaultWebBrowserApplet final : public WebBrowserApplet {
|
||||||
public:
|
public:
|
||||||
~DefaultWebBrowserApplet() override;
|
~DefaultWebBrowserApplet() override;
|
||||||
|
|
||||||
void OpenPageLocal(std::string_view url, std::function<void()> unpack_romfs_callback,
|
void OpenLocalWebPage(std::string_view local_url, std::function<void()> extract_romfs_callback,
|
||||||
std::function<void()> finished_callback) override;
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)>
|
||||||
|
callback) const override;
|
||||||
|
|
||||||
|
void OpenExternalWebPage(std::string_view external_url,
|
||||||
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)>
|
||||||
|
callback) const override;
|
||||||
};
|
};
|
||||||
|
|
||||||
} // namespace Core::Frontend
|
} // namespace Core::Frontend
|
||||||
|
|
|
@ -0,0 +1,45 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "core/core.h"
|
||||||
|
#include "core/frontend/input_interpreter.h"
|
||||||
|
#include "core/hle/service/hid/controllers/npad.h"
|
||||||
|
#include "core/hle/service/hid/hid.h"
|
||||||
|
#include "core/hle/service/sm/sm.h"
|
||||||
|
|
||||||
|
InputInterpreter::InputInterpreter(Core::System& system)
|
||||||
|
: npad{system.ServiceManager()
|
||||||
|
.GetService<Service::HID::Hid>("hid")
|
||||||
|
->GetAppletResource()
|
||||||
|
->GetController<Service::HID::Controller_NPad>(Service::HID::HidController::NPad)} {}
|
||||||
|
|
||||||
|
InputInterpreter::~InputInterpreter() = default;
|
||||||
|
|
||||||
|
void InputInterpreter::PollInput() {
|
||||||
|
const u32 button_state = npad.GetAndResetPressState();
|
||||||
|
|
||||||
|
previous_index = current_index;
|
||||||
|
current_index = (current_index + 1) % button_states.size();
|
||||||
|
|
||||||
|
button_states[current_index] = button_state;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InputInterpreter::IsButtonPressedOnce(HIDButton button) const {
|
||||||
|
const bool current_press =
|
||||||
|
(button_states[current_index] & (1U << static_cast<u8>(button))) != 0;
|
||||||
|
const bool previous_press =
|
||||||
|
(button_states[previous_index] & (1U << static_cast<u8>(button))) != 0;
|
||||||
|
|
||||||
|
return current_press && !previous_press;
|
||||||
|
}
|
||||||
|
|
||||||
|
bool InputInterpreter::IsButtonHeld(HIDButton button) const {
|
||||||
|
u32 held_buttons{button_states[0]};
|
||||||
|
|
||||||
|
for (std::size_t i = 1; i < button_states.size(); ++i) {
|
||||||
|
held_buttons &= button_states[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
return (held_buttons & (1U << static_cast<u8>(button))) != 0;
|
||||||
|
}
|
|
@ -0,0 +1,120 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
|
||||||
|
#include "common/common_types.h"
|
||||||
|
|
||||||
|
namespace Core {
|
||||||
|
class System;
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace Service::HID {
|
||||||
|
class Controller_NPad;
|
||||||
|
}
|
||||||
|
|
||||||
|
enum class HIDButton : u8 {
|
||||||
|
A,
|
||||||
|
B,
|
||||||
|
X,
|
||||||
|
Y,
|
||||||
|
LStick,
|
||||||
|
RStick,
|
||||||
|
L,
|
||||||
|
R,
|
||||||
|
ZL,
|
||||||
|
ZR,
|
||||||
|
Plus,
|
||||||
|
Minus,
|
||||||
|
|
||||||
|
DLeft,
|
||||||
|
DUp,
|
||||||
|
DRight,
|
||||||
|
DDown,
|
||||||
|
|
||||||
|
LStickLeft,
|
||||||
|
LStickUp,
|
||||||
|
LStickRight,
|
||||||
|
LStickDown,
|
||||||
|
|
||||||
|
RStickLeft,
|
||||||
|
RStickUp,
|
||||||
|
RStickRight,
|
||||||
|
RStickDown,
|
||||||
|
|
||||||
|
LeftSL,
|
||||||
|
LeftSR,
|
||||||
|
|
||||||
|
RightSL,
|
||||||
|
RightSR,
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The InputInterpreter class interfaces with HID to retrieve button press states.
|
||||||
|
* Input is intended to be polled every 50ms so that a button is considered to be
|
||||||
|
* held down after 400ms has elapsed since the initial button press and subsequent
|
||||||
|
* repeated presses occur every 50ms.
|
||||||
|
*/
|
||||||
|
class InputInterpreter {
|
||||||
|
public:
|
||||||
|
explicit InputInterpreter(Core::System& system);
|
||||||
|
virtual ~InputInterpreter();
|
||||||
|
|
||||||
|
/// Gets a button state from HID and inserts it into the array of button states.
|
||||||
|
void PollInput();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The specified button is considered to be pressed once
|
||||||
|
* if it is currently pressed and not pressed previously.
|
||||||
|
*
|
||||||
|
* @param button The button to check.
|
||||||
|
*
|
||||||
|
* @returns True when the button is pressed once.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] bool IsButtonPressedOnce(HIDButton button) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks whether any of the buttons in the parameter list is pressed once.
|
||||||
|
*
|
||||||
|
* @tparam HIDButton The buttons to check.
|
||||||
|
*
|
||||||
|
* @returns True when at least one of the buttons is pressed once.
|
||||||
|
*/
|
||||||
|
template <HIDButton... T>
|
||||||
|
[[nodiscard]] bool IsAnyButtonPressedOnce() {
|
||||||
|
return (IsButtonPressedOnce(T) || ...);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The specified button is considered to be held down if it is pressed in all 9 button states.
|
||||||
|
*
|
||||||
|
* @param button The button to check.
|
||||||
|
*
|
||||||
|
* @returns True when the button is held down.
|
||||||
|
*/
|
||||||
|
[[nodiscard]] bool IsButtonHeld(HIDButton button) const;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checks whether any of the buttons in the parameter list is held down.
|
||||||
|
*
|
||||||
|
* @tparam HIDButton The buttons to check.
|
||||||
|
*
|
||||||
|
* @returns True when at least one of the buttons is held down.
|
||||||
|
*/
|
||||||
|
template <HIDButton... T>
|
||||||
|
[[nodiscard]] bool IsAnyButtonHeld() {
|
||||||
|
return (IsButtonHeld(T) || ...);
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
Service::HID::Controller_NPad& npad;
|
||||||
|
|
||||||
|
/// Stores 9 consecutive button states polled from HID.
|
||||||
|
std::array<u32, 9> button_states{};
|
||||||
|
|
||||||
|
std::size_t previous_index{};
|
||||||
|
std::size_t current_index{};
|
||||||
|
};
|
|
@ -142,14 +142,14 @@ void Applet::Initialize() {
|
||||||
|
|
||||||
AppletFrontendSet::AppletFrontendSet() = default;
|
AppletFrontendSet::AppletFrontendSet() = default;
|
||||||
|
|
||||||
AppletFrontendSet::AppletFrontendSet(ControllerApplet controller, ECommerceApplet e_commerce,
|
AppletFrontendSet::AppletFrontendSet(ControllerApplet controller_applet, ErrorApplet error_applet,
|
||||||
ErrorApplet error, ParentalControlsApplet parental_controls,
|
ParentalControlsApplet parental_controls_applet,
|
||||||
PhotoViewer photo_viewer, ProfileSelect profile_select,
|
PhotoViewer photo_viewer_, ProfileSelect profile_select_,
|
||||||
SoftwareKeyboard software_keyboard, WebBrowser web_browser)
|
SoftwareKeyboard software_keyboard_, WebBrowser web_browser_)
|
||||||
: controller{std::move(controller)}, e_commerce{std::move(e_commerce)}, error{std::move(error)},
|
: controller{std::move(controller_applet)}, error{std::move(error_applet)},
|
||||||
parental_controls{std::move(parental_controls)}, photo_viewer{std::move(photo_viewer)},
|
parental_controls{std::move(parental_controls_applet)},
|
||||||
profile_select{std::move(profile_select)}, software_keyboard{std::move(software_keyboard)},
|
photo_viewer{std::move(photo_viewer_)}, profile_select{std::move(profile_select_)},
|
||||||
web_browser{std::move(web_browser)} {}
|
software_keyboard{std::move(software_keyboard_)}, web_browser{std::move(web_browser_)} {}
|
||||||
|
|
||||||
AppletFrontendSet::~AppletFrontendSet() = default;
|
AppletFrontendSet::~AppletFrontendSet() = default;
|
||||||
|
|
||||||
|
@ -170,10 +170,6 @@ void AppletManager::SetAppletFrontendSet(AppletFrontendSet set) {
|
||||||
frontend.controller = std::move(set.controller);
|
frontend.controller = std::move(set.controller);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (set.e_commerce != nullptr) {
|
|
||||||
frontend.e_commerce = std::move(set.e_commerce);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (set.error != nullptr) {
|
if (set.error != nullptr) {
|
||||||
frontend.error = std::move(set.error);
|
frontend.error = std::move(set.error);
|
||||||
}
|
}
|
||||||
|
@ -210,10 +206,6 @@ void AppletManager::SetDefaultAppletsIfMissing() {
|
||||||
std::make_unique<Core::Frontend::DefaultControllerApplet>(system.ServiceManager());
|
std::make_unique<Core::Frontend::DefaultControllerApplet>(system.ServiceManager());
|
||||||
}
|
}
|
||||||
|
|
||||||
if (frontend.e_commerce == nullptr) {
|
|
||||||
frontend.e_commerce = std::make_unique<Core::Frontend::DefaultECommerceApplet>();
|
|
||||||
}
|
|
||||||
|
|
||||||
if (frontend.error == nullptr) {
|
if (frontend.error == nullptr) {
|
||||||
frontend.error = std::make_unique<Core::Frontend::DefaultErrorApplet>();
|
frontend.error = std::make_unique<Core::Frontend::DefaultErrorApplet>();
|
||||||
}
|
}
|
||||||
|
@ -257,13 +249,14 @@ std::shared_ptr<Applet> AppletManager::GetApplet(AppletId id) const {
|
||||||
return std::make_shared<ProfileSelect>(system, *frontend.profile_select);
|
return std::make_shared<ProfileSelect>(system, *frontend.profile_select);
|
||||||
case AppletId::SoftwareKeyboard:
|
case AppletId::SoftwareKeyboard:
|
||||||
return std::make_shared<SoftwareKeyboard>(system, *frontend.software_keyboard);
|
return std::make_shared<SoftwareKeyboard>(system, *frontend.software_keyboard);
|
||||||
|
case AppletId::Web:
|
||||||
|
case AppletId::Shop:
|
||||||
|
case AppletId::OfflineWeb:
|
||||||
|
case AppletId::LoginShare:
|
||||||
|
case AppletId::WebAuth:
|
||||||
|
return std::make_shared<WebBrowser>(system, *frontend.web_browser);
|
||||||
case AppletId::PhotoViewer:
|
case AppletId::PhotoViewer:
|
||||||
return std::make_shared<PhotoViewer>(system, *frontend.photo_viewer);
|
return std::make_shared<PhotoViewer>(system, *frontend.photo_viewer);
|
||||||
case AppletId::LibAppletShop:
|
|
||||||
return std::make_shared<WebBrowser>(system, *frontend.web_browser,
|
|
||||||
frontend.e_commerce.get());
|
|
||||||
case AppletId::LibAppletOff:
|
|
||||||
return std::make_shared<WebBrowser>(system, *frontend.web_browser);
|
|
||||||
default:
|
default:
|
||||||
UNIMPLEMENTED_MSG(
|
UNIMPLEMENTED_MSG(
|
||||||
"No backend implementation exists for applet_id={:02X}! Falling back to stub applet.",
|
"No backend implementation exists for applet_id={:02X}! Falling back to stub applet.",
|
||||||
|
|
|
@ -50,13 +50,13 @@ enum class AppletId : u32 {
|
||||||
ProfileSelect = 0x10,
|
ProfileSelect = 0x10,
|
||||||
SoftwareKeyboard = 0x11,
|
SoftwareKeyboard = 0x11,
|
||||||
MiiEdit = 0x12,
|
MiiEdit = 0x12,
|
||||||
LibAppletWeb = 0x13,
|
Web = 0x13,
|
||||||
LibAppletShop = 0x14,
|
Shop = 0x14,
|
||||||
PhotoViewer = 0x15,
|
PhotoViewer = 0x15,
|
||||||
Settings = 0x16,
|
Settings = 0x16,
|
||||||
LibAppletOff = 0x17,
|
OfflineWeb = 0x17,
|
||||||
LibAppletWhitelisted = 0x18,
|
LoginShare = 0x18,
|
||||||
LibAppletAuth = 0x19,
|
WebAuth = 0x19,
|
||||||
MyPage = 0x1A,
|
MyPage = 0x1A,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -157,7 +157,6 @@ protected:
|
||||||
|
|
||||||
struct AppletFrontendSet {
|
struct AppletFrontendSet {
|
||||||
using ControllerApplet = std::unique_ptr<Core::Frontend::ControllerApplet>;
|
using ControllerApplet = std::unique_ptr<Core::Frontend::ControllerApplet>;
|
||||||
using ECommerceApplet = std::unique_ptr<Core::Frontend::ECommerceApplet>;
|
|
||||||
using ErrorApplet = std::unique_ptr<Core::Frontend::ErrorApplet>;
|
using ErrorApplet = std::unique_ptr<Core::Frontend::ErrorApplet>;
|
||||||
using ParentalControlsApplet = std::unique_ptr<Core::Frontend::ParentalControlsApplet>;
|
using ParentalControlsApplet = std::unique_ptr<Core::Frontend::ParentalControlsApplet>;
|
||||||
using PhotoViewer = std::unique_ptr<Core::Frontend::PhotoViewerApplet>;
|
using PhotoViewer = std::unique_ptr<Core::Frontend::PhotoViewerApplet>;
|
||||||
|
@ -166,10 +165,10 @@ struct AppletFrontendSet {
|
||||||
using WebBrowser = std::unique_ptr<Core::Frontend::WebBrowserApplet>;
|
using WebBrowser = std::unique_ptr<Core::Frontend::WebBrowserApplet>;
|
||||||
|
|
||||||
AppletFrontendSet();
|
AppletFrontendSet();
|
||||||
AppletFrontendSet(ControllerApplet controller, ECommerceApplet e_commerce, ErrorApplet error,
|
AppletFrontendSet(ControllerApplet controller_applet, ErrorApplet error_applet,
|
||||||
ParentalControlsApplet parental_controls, PhotoViewer photo_viewer,
|
ParentalControlsApplet parental_controls_applet, PhotoViewer photo_viewer_,
|
||||||
ProfileSelect profile_select, SoftwareKeyboard software_keyboard,
|
ProfileSelect profile_select_, SoftwareKeyboard software_keyboard_,
|
||||||
WebBrowser web_browser);
|
WebBrowser web_browser_);
|
||||||
~AppletFrontendSet();
|
~AppletFrontendSet();
|
||||||
|
|
||||||
AppletFrontendSet(const AppletFrontendSet&) = delete;
|
AppletFrontendSet(const AppletFrontendSet&) = delete;
|
||||||
|
@ -179,7 +178,6 @@ struct AppletFrontendSet {
|
||||||
AppletFrontendSet& operator=(AppletFrontendSet&&) noexcept;
|
AppletFrontendSet& operator=(AppletFrontendSet&&) noexcept;
|
||||||
|
|
||||||
ControllerApplet controller;
|
ControllerApplet controller;
|
||||||
ECommerceApplet e_commerce;
|
|
||||||
ErrorApplet error;
|
ErrorApplet error;
|
||||||
ParentalControlsApplet parental_controls;
|
ParentalControlsApplet parental_controls;
|
||||||
PhotoViewer photo_viewer;
|
PhotoViewer photo_viewer;
|
||||||
|
|
|
@ -1,238 +1,271 @@
|
||||||
// Copyright 2018 yuzu emulator team
|
// Copyright 2020 yuzu Emulator Project
|
||||||
// Licensed under GPLv2 or any later version
|
// Licensed under GPLv2 or any later version
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#include <array>
|
|
||||||
#include <cstring>
|
|
||||||
#include <vector>
|
|
||||||
|
|
||||||
#include "common/assert.h"
|
#include "common/assert.h"
|
||||||
#include "common/common_funcs.h"
|
|
||||||
#include "common/common_paths.h"
|
#include "common/common_paths.h"
|
||||||
#include "common/file_util.h"
|
#include "common/file_util.h"
|
||||||
#include "common/hex_util.h"
|
|
||||||
#include "common/logging/log.h"
|
#include "common/logging/log.h"
|
||||||
#include "common/string_util.h"
|
#include "common/string_util.h"
|
||||||
#include "core/core.h"
|
#include "core/core.h"
|
||||||
#include "core/file_sys/content_archive.h"
|
#include "core/file_sys/content_archive.h"
|
||||||
#include "core/file_sys/mode.h"
|
#include "core/file_sys/mode.h"
|
||||||
#include "core/file_sys/nca_metadata.h"
|
#include "core/file_sys/nca_metadata.h"
|
||||||
|
#include "core/file_sys/patch_manager.h"
|
||||||
#include "core/file_sys/registered_cache.h"
|
#include "core/file_sys/registered_cache.h"
|
||||||
#include "core/file_sys/romfs.h"
|
#include "core/file_sys/romfs.h"
|
||||||
#include "core/file_sys/system_archive/system_archive.h"
|
#include "core/file_sys/system_archive/system_archive.h"
|
||||||
#include "core/file_sys/vfs_types.h"
|
#include "core/file_sys/vfs_vector.h"
|
||||||
#include "core/frontend/applets/general_frontend.h"
|
|
||||||
#include "core/frontend/applets/web_browser.h"
|
#include "core/frontend/applets/web_browser.h"
|
||||||
#include "core/hle/kernel/process.h"
|
#include "core/hle/kernel/process.h"
|
||||||
|
#include "core/hle/result.h"
|
||||||
|
#include "core/hle/service/am/am.h"
|
||||||
#include "core/hle/service/am/applets/web_browser.h"
|
#include "core/hle/service/am/applets/web_browser.h"
|
||||||
#include "core/hle/service/filesystem/filesystem.h"
|
#include "core/hle/service/filesystem/filesystem.h"
|
||||||
#include "core/loader/loader.h"
|
#include "core/hle/service/ns/pl_u.h"
|
||||||
|
|
||||||
namespace Service::AM::Applets {
|
namespace Service::AM::Applets {
|
||||||
|
|
||||||
enum class WebArgTLVType : u16 {
|
|
||||||
InitialURL = 0x1,
|
|
||||||
ShopArgumentsURL = 0x2, ///< TODO(DarkLordZach): This is not the official name.
|
|
||||||
CallbackURL = 0x3,
|
|
||||||
CallbackableURL = 0x4,
|
|
||||||
ApplicationID = 0x5,
|
|
||||||
DocumentPath = 0x6,
|
|
||||||
DocumentKind = 0x7,
|
|
||||||
SystemDataID = 0x8,
|
|
||||||
ShareStartPage = 0x9,
|
|
||||||
Whitelist = 0xA,
|
|
||||||
News = 0xB,
|
|
||||||
UserID = 0xE,
|
|
||||||
AlbumEntry0 = 0xF,
|
|
||||||
ScreenShotEnabled = 0x10,
|
|
||||||
EcClientCertEnabled = 0x11,
|
|
||||||
Unk12 = 0x12,
|
|
||||||
PlayReportEnabled = 0x13,
|
|
||||||
Unk14 = 0x14,
|
|
||||||
Unk15 = 0x15,
|
|
||||||
BootDisplayKind = 0x17,
|
|
||||||
BackgroundKind = 0x18,
|
|
||||||
FooterEnabled = 0x19,
|
|
||||||
PointerEnabled = 0x1A,
|
|
||||||
LeftStickMode = 0x1B,
|
|
||||||
KeyRepeatFrame1 = 0x1C,
|
|
||||||
KeyRepeatFrame2 = 0x1D,
|
|
||||||
BootAsMediaPlayerInv = 0x1E,
|
|
||||||
DisplayUrlKind = 0x1F,
|
|
||||||
BootAsMediaPlayer = 0x21,
|
|
||||||
ShopJumpEnabled = 0x22,
|
|
||||||
MediaAutoPlayEnabled = 0x23,
|
|
||||||
LobbyParameter = 0x24,
|
|
||||||
ApplicationAlbumEntry = 0x26,
|
|
||||||
JsExtensionEnabled = 0x27,
|
|
||||||
AdditionalCommentText = 0x28,
|
|
||||||
TouchEnabledOnContents = 0x29,
|
|
||||||
UserAgentAdditionalString = 0x2A,
|
|
||||||
AdditionalMediaData0 = 0x2B,
|
|
||||||
MediaPlayerAutoCloseEnabled = 0x2C,
|
|
||||||
PageCacheEnabled = 0x2D,
|
|
||||||
WebAudioEnabled = 0x2E,
|
|
||||||
Unk2F = 0x2F,
|
|
||||||
YouTubeVideoWhitelist = 0x31,
|
|
||||||
FooterFixedKind = 0x32,
|
|
||||||
PageFadeEnabled = 0x33,
|
|
||||||
MediaCreatorApplicationRatingAge = 0x34,
|
|
||||||
BootLoadingIconEnabled = 0x35,
|
|
||||||
PageScrollIndicationEnabled = 0x36,
|
|
||||||
MediaPlayerSpeedControlEnabled = 0x37,
|
|
||||||
AlbumEntry1 = 0x38,
|
|
||||||
AlbumEntry2 = 0x39,
|
|
||||||
AlbumEntry3 = 0x3A,
|
|
||||||
AdditionalMediaData1 = 0x3B,
|
|
||||||
AdditionalMediaData2 = 0x3C,
|
|
||||||
AdditionalMediaData3 = 0x3D,
|
|
||||||
BootFooterButton = 0x3E,
|
|
||||||
OverrideWebAudioVolume = 0x3F,
|
|
||||||
OverrideMediaAudioVolume = 0x40,
|
|
||||||
BootMode = 0x41,
|
|
||||||
WebSessionEnabled = 0x42,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class ShimKind : u32 {
|
|
||||||
Shop = 1,
|
|
||||||
Login = 2,
|
|
||||||
Offline = 3,
|
|
||||||
Share = 4,
|
|
||||||
Web = 5,
|
|
||||||
Wifi = 6,
|
|
||||||
Lobby = 7,
|
|
||||||
};
|
|
||||||
|
|
||||||
enum class ShopWebTarget {
|
|
||||||
ApplicationInfo,
|
|
||||||
AddOnContentList,
|
|
||||||
SubscriptionList,
|
|
||||||
ConsumableItemList,
|
|
||||||
Home,
|
|
||||||
Settings,
|
|
||||||
};
|
|
||||||
|
|
||||||
namespace {
|
namespace {
|
||||||
|
|
||||||
constexpr std::size_t SHIM_KIND_COUNT = 0x8;
|
template <typename T>
|
||||||
|
void ParseRawValue(T& value, const std::vector<u8>& data) {
|
||||||
|
static_assert(std::is_trivially_copyable_v<T>,
|
||||||
|
"It's undefined behavior to use memcpy with non-trivially copyable objects");
|
||||||
|
std::memcpy(&value, data.data(), data.size());
|
||||||
|
}
|
||||||
|
|
||||||
struct WebArgHeader {
|
template <typename T>
|
||||||
u16 count;
|
T ParseRawValue(const std::vector<u8>& data) {
|
||||||
INSERT_PADDING_BYTES(2);
|
T value;
|
||||||
ShimKind kind;
|
ParseRawValue(value, data);
|
||||||
};
|
return value;
|
||||||
static_assert(sizeof(WebArgHeader) == 0x8, "WebArgHeader has incorrect size.");
|
}
|
||||||
|
|
||||||
struct WebArgTLV {
|
std::string ParseStringValue(const std::vector<u8>& data) {
|
||||||
WebArgTLVType type;
|
return Common::StringFromFixedZeroTerminatedBuffer(reinterpret_cast<const char*>(data.data()),
|
||||||
u16 size;
|
data.size());
|
||||||
u32 offset;
|
}
|
||||||
};
|
|
||||||
static_assert(sizeof(WebArgTLV) == 0x8, "WebArgTLV has incorrect size.");
|
|
||||||
|
|
||||||
struct WebCommonReturnValue {
|
std::string GetMainURL(const std::string& url) {
|
||||||
u32 result_code;
|
const auto index = url.find('?');
|
||||||
INSERT_PADDING_BYTES(0x4);
|
|
||||||
std::array<char, 0x1000> last_url;
|
|
||||||
u64 last_url_size;
|
|
||||||
};
|
|
||||||
static_assert(sizeof(WebCommonReturnValue) == 0x1010, "WebCommonReturnValue has incorrect size.");
|
|
||||||
|
|
||||||
struct WebWifiPageArg {
|
if (index == std::string::npos) {
|
||||||
INSERT_PADDING_BYTES(4);
|
return url;
|
||||||
std::array<char, 0x100> connection_test_url;
|
}
|
||||||
std::array<char, 0x400> initial_url;
|
|
||||||
std::array<u8, 0x10> nifm_network_uuid;
|
|
||||||
u32 nifm_requirement;
|
|
||||||
};
|
|
||||||
static_assert(sizeof(WebWifiPageArg) == 0x518, "WebWifiPageArg has incorrect size.");
|
|
||||||
|
|
||||||
struct WebWifiReturnValue {
|
return url.substr(0, index);
|
||||||
INSERT_PADDING_BYTES(4);
|
}
|
||||||
u32 result;
|
|
||||||
};
|
|
||||||
static_assert(sizeof(WebWifiReturnValue) == 0x8, "WebWifiReturnValue has incorrect size.");
|
|
||||||
|
|
||||||
enum class OfflineWebSource : u32 {
|
WebArgInputTLVMap ReadWebArgs(const std::vector<u8>& web_arg, WebArgHeader& web_arg_header) {
|
||||||
OfflineHtmlPage = 0x1,
|
std::memcpy(&web_arg_header, web_arg.data(), sizeof(WebArgHeader));
|
||||||
ApplicationLegalInformation = 0x2,
|
|
||||||
SystemDataPage = 0x3,
|
|
||||||
};
|
|
||||||
|
|
||||||
std::map<WebArgTLVType, std::vector<u8>> GetWebArguments(const std::vector<u8>& arg) {
|
if (web_arg.size() == sizeof(WebArgHeader)) {
|
||||||
if (arg.size() < sizeof(WebArgHeader))
|
|
||||||
return {};
|
return {};
|
||||||
|
|
||||||
WebArgHeader header{};
|
|
||||||
std::memcpy(&header, arg.data(), sizeof(WebArgHeader));
|
|
||||||
|
|
||||||
std::map<WebArgTLVType, std::vector<u8>> out;
|
|
||||||
u64 offset = sizeof(WebArgHeader);
|
|
||||||
for (std::size_t i = 0; i < header.count; ++i) {
|
|
||||||
if (arg.size() < (offset + sizeof(WebArgTLV)))
|
|
||||||
return out;
|
|
||||||
|
|
||||||
WebArgTLV tlv{};
|
|
||||||
std::memcpy(&tlv, arg.data() + offset, sizeof(WebArgTLV));
|
|
||||||
offset += sizeof(WebArgTLV);
|
|
||||||
|
|
||||||
offset += tlv.offset;
|
|
||||||
if (arg.size() < (offset + tlv.size))
|
|
||||||
return out;
|
|
||||||
|
|
||||||
std::vector<u8> data(tlv.size);
|
|
||||||
std::memcpy(data.data(), arg.data() + offset, tlv.size);
|
|
||||||
offset += tlv.size;
|
|
||||||
|
|
||||||
out.insert_or_assign(tlv.type, data);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return out;
|
WebArgInputTLVMap input_tlv_map;
|
||||||
|
|
||||||
|
u64 current_offset = sizeof(WebArgHeader);
|
||||||
|
|
||||||
|
for (std::size_t i = 0; i < web_arg_header.total_tlv_entries; ++i) {
|
||||||
|
if (web_arg.size() < current_offset + sizeof(WebArgInputTLV)) {
|
||||||
|
return input_tlv_map;
|
||||||
}
|
}
|
||||||
|
|
||||||
FileSys::VirtualFile GetApplicationRomFS(const Core::System& system, u64 title_id,
|
WebArgInputTLV input_tlv;
|
||||||
FileSys::ContentRecordType type) {
|
std::memcpy(&input_tlv, web_arg.data() + current_offset, sizeof(WebArgInputTLV));
|
||||||
const auto& installed{system.GetContentProvider()};
|
|
||||||
const auto res = installed.GetEntry(title_id, type);
|
|
||||||
|
|
||||||
if (res != nullptr) {
|
current_offset += sizeof(WebArgInputTLV);
|
||||||
return res->GetRomFS();
|
|
||||||
|
if (web_arg.size() < current_offset + input_tlv.arg_data_size) {
|
||||||
|
return input_tlv_map;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (type == FileSys::ContentRecordType::Data) {
|
std::vector<u8> data(input_tlv.arg_data_size);
|
||||||
|
std::memcpy(data.data(), web_arg.data() + current_offset, input_tlv.arg_data_size);
|
||||||
|
|
||||||
|
current_offset += input_tlv.arg_data_size;
|
||||||
|
|
||||||
|
input_tlv_map.insert_or_assign(input_tlv.input_tlv_type, std::move(data));
|
||||||
|
}
|
||||||
|
|
||||||
|
return input_tlv_map;
|
||||||
|
}
|
||||||
|
|
||||||
|
FileSys::VirtualFile GetOfflineRomFS(Core::System& system, u64 title_id,
|
||||||
|
FileSys::ContentRecordType nca_type) {
|
||||||
|
if (nca_type == FileSys::ContentRecordType::Data) {
|
||||||
|
const auto nca =
|
||||||
|
system.GetFileSystemController().GetSystemNANDContents()->GetEntry(title_id, nca_type);
|
||||||
|
|
||||||
|
if (nca == nullptr) {
|
||||||
|
LOG_ERROR(Service_AM,
|
||||||
|
"NCA of type={} with title_id={:016X} is not found in the System NAND!",
|
||||||
|
nca_type, title_id);
|
||||||
return FileSys::SystemArchive::SynthesizeSystemArchive(title_id);
|
return FileSys::SystemArchive::SynthesizeSystemArchive(title_id);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
return nca->GetRomFS();
|
||||||
|
} else {
|
||||||
|
const auto nca = system.GetContentProvider().GetEntry(title_id, nca_type);
|
||||||
|
|
||||||
|
if (nca == nullptr) {
|
||||||
|
LOG_ERROR(Service_AM,
|
||||||
|
"NCA of type={} with title_id={:016X} is not found in the ContentProvider!",
|
||||||
|
nca_type, title_id);
|
||||||
return nullptr;
|
return nullptr;
|
||||||
}
|
}
|
||||||
|
|
||||||
} // Anonymous namespace
|
const FileSys::PatchManager pm{title_id, system.GetFileSystemController(),
|
||||||
|
system.GetContentProvider()};
|
||||||
|
|
||||||
WebBrowser::WebBrowser(Core::System& system_, Core::Frontend::WebBrowserApplet& frontend_,
|
return pm.PatchRomFS(nca->GetRomFS(), nca->GetBaseIVFCOffset(), nca_type);
|
||||||
Core::Frontend::ECommerceApplet* frontend_e_commerce_)
|
}
|
||||||
: Applet{system_.Kernel()}, frontend(frontend_),
|
}
|
||||||
frontend_e_commerce(frontend_e_commerce_), system{system_} {}
|
|
||||||
|
void ExtractSharedFonts(Core::System& system) {
|
||||||
|
static constexpr std::array<const char*, 7> DECRYPTED_SHARED_FONTS{
|
||||||
|
"FontStandard.ttf",
|
||||||
|
"FontChineseSimplified.ttf",
|
||||||
|
"FontExtendedChineseSimplified.ttf",
|
||||||
|
"FontChineseTraditional.ttf",
|
||||||
|
"FontKorean.ttf",
|
||||||
|
"FontNintendoExtended.ttf",
|
||||||
|
"FontNintendoExtended2.ttf",
|
||||||
|
};
|
||||||
|
|
||||||
|
for (std::size_t i = 0; i < NS::SHARED_FONTS.size(); ++i) {
|
||||||
|
const auto fonts_dir = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/fonts", Common::FS::GetUserPath(Common::FS::UserPath::CacheDir)),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
const auto font_file_path =
|
||||||
|
Common::FS::SanitizePath(fmt::format("{}/{}", fonts_dir, DECRYPTED_SHARED_FONTS[i]),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
if (Common::FS::Exists(font_file_path)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto font = NS::SHARED_FONTS[i];
|
||||||
|
const auto font_title_id = static_cast<u64>(font.first);
|
||||||
|
|
||||||
|
const auto nca = system.GetFileSystemController().GetSystemNANDContents()->GetEntry(
|
||||||
|
font_title_id, FileSys::ContentRecordType::Data);
|
||||||
|
|
||||||
|
FileSys::VirtualFile romfs;
|
||||||
|
|
||||||
|
if (!nca) {
|
||||||
|
romfs = FileSys::SystemArchive::SynthesizeSystemArchive(font_title_id);
|
||||||
|
} else {
|
||||||
|
romfs = nca->GetRomFS();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!romfs) {
|
||||||
|
LOG_ERROR(Service_AM, "SharedFont RomFS with title_id={:016X} cannot be extracted!",
|
||||||
|
font_title_id);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto extracted_romfs = FileSys::ExtractRomFS(romfs);
|
||||||
|
|
||||||
|
if (!extracted_romfs) {
|
||||||
|
LOG_ERROR(Service_AM, "SharedFont RomFS with title_id={:016X} failed to extract!",
|
||||||
|
font_title_id);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto font_file = extracted_romfs->GetFile(font.second);
|
||||||
|
|
||||||
|
if (!font_file) {
|
||||||
|
LOG_ERROR(Service_AM, "SharedFont RomFS with title_id={:016X} has no font file \"{}\"!",
|
||||||
|
font_title_id, font.second);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
std::vector<u32> font_data_u32(font_file->GetSize() / sizeof(u32));
|
||||||
|
font_file->ReadBytes<u32>(font_data_u32.data(), font_file->GetSize());
|
||||||
|
|
||||||
|
std::transform(font_data_u32.begin(), font_data_u32.end(), font_data_u32.begin(),
|
||||||
|
Common::swap32);
|
||||||
|
|
||||||
|
std::vector<u8> decrypted_data(font_file->GetSize() - 8);
|
||||||
|
|
||||||
|
NS::DecryptSharedFontToTTF(font_data_u32, decrypted_data);
|
||||||
|
|
||||||
|
FileSys::VirtualFile decrypted_font = std::make_shared<FileSys::VectorVfsFile>(
|
||||||
|
std::move(decrypted_data), DECRYPTED_SHARED_FONTS[i]);
|
||||||
|
|
||||||
|
const auto temp_dir =
|
||||||
|
system.GetFilesystem()->CreateDirectory(fonts_dir, FileSys::Mode::ReadWrite);
|
||||||
|
|
||||||
|
const auto out_file = temp_dir->CreateFile(DECRYPTED_SHARED_FONTS[i]);
|
||||||
|
|
||||||
|
FileSys::VfsRawCopy(decrypted_font, out_file);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace
|
||||||
|
|
||||||
|
WebBrowser::WebBrowser(Core::System& system_, const Core::Frontend::WebBrowserApplet& frontend_)
|
||||||
|
: Applet{system_.Kernel()}, frontend(frontend_), system{system_} {}
|
||||||
|
|
||||||
WebBrowser::~WebBrowser() = default;
|
WebBrowser::~WebBrowser() = default;
|
||||||
|
|
||||||
void WebBrowser::Initialize() {
|
void WebBrowser::Initialize() {
|
||||||
Applet::Initialize();
|
Applet::Initialize();
|
||||||
|
|
||||||
complete = false;
|
LOG_INFO(Service_AM, "Initializing Web Browser Applet.");
|
||||||
temporary_dir.clear();
|
|
||||||
filename.clear();
|
LOG_DEBUG(Service_AM,
|
||||||
status = RESULT_SUCCESS;
|
"Initializing Applet with common_args: arg_version={}, lib_version={}, "
|
||||||
|
"play_startup_sound={}, size={}, system_tick={}, theme_color={}",
|
||||||
|
common_args.arguments_version, common_args.library_version,
|
||||||
|
common_args.play_startup_sound, common_args.size, common_args.system_tick,
|
||||||
|
common_args.theme_color);
|
||||||
|
|
||||||
|
web_applet_version = WebAppletVersion{common_args.library_version};
|
||||||
|
|
||||||
const auto web_arg_storage = broker.PopNormalDataToApplet();
|
const auto web_arg_storage = broker.PopNormalDataToApplet();
|
||||||
ASSERT(web_arg_storage != nullptr);
|
ASSERT(web_arg_storage != nullptr);
|
||||||
|
|
||||||
const auto& web_arg = web_arg_storage->GetData();
|
const auto& web_arg = web_arg_storage->GetData();
|
||||||
|
ASSERT_OR_EXECUTE(web_arg.size() >= sizeof(WebArgHeader), { return; });
|
||||||
|
|
||||||
ASSERT(web_arg.size() >= 0x8);
|
web_arg_input_tlv_map = ReadWebArgs(web_arg, web_arg_header);
|
||||||
std::memcpy(&kind, web_arg.data() + 0x4, sizeof(ShimKind));
|
|
||||||
|
|
||||||
args = GetWebArguments(web_arg);
|
LOG_DEBUG(Service_AM, "WebArgHeader: total_tlv_entries={}, shim_kind={}",
|
||||||
|
web_arg_header.total_tlv_entries, web_arg_header.shim_kind);
|
||||||
|
|
||||||
InitializeInternal();
|
ExtractSharedFonts(system);
|
||||||
|
|
||||||
|
switch (web_arg_header.shim_kind) {
|
||||||
|
case ShimKind::Shop:
|
||||||
|
InitializeShop();
|
||||||
|
break;
|
||||||
|
case ShimKind::Login:
|
||||||
|
InitializeLogin();
|
||||||
|
break;
|
||||||
|
case ShimKind::Offline:
|
||||||
|
InitializeOffline();
|
||||||
|
break;
|
||||||
|
case ShimKind::Share:
|
||||||
|
InitializeShare();
|
||||||
|
break;
|
||||||
|
case ShimKind::Web:
|
||||||
|
InitializeWeb();
|
||||||
|
break;
|
||||||
|
case ShimKind::Wifi:
|
||||||
|
InitializeWifi();
|
||||||
|
break;
|
||||||
|
case ShimKind::Lobby:
|
||||||
|
InitializeLobby();
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE_MSG("Invalid ShimKind={}", web_arg_header.shim_kind);
|
||||||
|
break;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
bool WebBrowser::TransactionComplete() const {
|
bool WebBrowser::TransactionComplete() const {
|
||||||
|
@ -244,315 +277,202 @@ ResultCode WebBrowser::GetStatus() const {
|
||||||
}
|
}
|
||||||
|
|
||||||
void WebBrowser::ExecuteInteractive() {
|
void WebBrowser::ExecuteInteractive() {
|
||||||
UNIMPLEMENTED_MSG("Unexpected interactive data recieved!");
|
UNIMPLEMENTED_MSG("WebSession is not implemented");
|
||||||
}
|
}
|
||||||
|
|
||||||
void WebBrowser::Execute() {
|
void WebBrowser::Execute() {
|
||||||
if (complete) {
|
switch (web_arg_header.shim_kind) {
|
||||||
return;
|
case ShimKind::Shop:
|
||||||
}
|
ExecuteShop();
|
||||||
|
|
||||||
if (status != RESULT_SUCCESS) {
|
|
||||||
complete = true;
|
|
||||||
|
|
||||||
// This is a workaround in order not to softlock yuzu when an error happens during the
|
|
||||||
// webapplet init. In order to avoid an svcBreak, the status is set to RESULT_SUCCESS
|
|
||||||
Finalize();
|
|
||||||
status = RESULT_SUCCESS;
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ExecuteInternal();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::UnpackRomFS() {
|
|
||||||
if (unpacked)
|
|
||||||
return;
|
|
||||||
|
|
||||||
ASSERT(offline_romfs != nullptr);
|
|
||||||
const auto dir =
|
|
||||||
FileSys::ExtractRomFS(offline_romfs, FileSys::RomFSExtractionType::SingleDiscard);
|
|
||||||
const auto& vfs{system.GetFilesystem()};
|
|
||||||
const auto temp_dir = vfs->CreateDirectory(temporary_dir, FileSys::Mode::ReadWrite);
|
|
||||||
FileSys::VfsRawCopyD(dir, temp_dir);
|
|
||||||
|
|
||||||
unpacked = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::Finalize() {
|
|
||||||
complete = true;
|
|
||||||
|
|
||||||
WebCommonReturnValue out{};
|
|
||||||
out.result_code = 0;
|
|
||||||
out.last_url_size = 0;
|
|
||||||
|
|
||||||
std::vector<u8> data(sizeof(WebCommonReturnValue));
|
|
||||||
std::memcpy(data.data(), &out, sizeof(WebCommonReturnValue));
|
|
||||||
|
|
||||||
broker.PushNormalDataFromApplet(std::make_shared<IStorage>(system, std::move(data)));
|
|
||||||
broker.SignalStateChanged();
|
|
||||||
|
|
||||||
if (!temporary_dir.empty() && Common::FS::IsDirectory(temporary_dir)) {
|
|
||||||
Common::FS::DeleteDirRecursively(temporary_dir);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::InitializeInternal() {
|
|
||||||
using WebAppletInitializer = void (WebBrowser::*)();
|
|
||||||
|
|
||||||
constexpr std::array<WebAppletInitializer, SHIM_KIND_COUNT> functions{
|
|
||||||
nullptr, &WebBrowser::InitializeShop,
|
|
||||||
nullptr, &WebBrowser::InitializeOffline,
|
|
||||||
nullptr, nullptr,
|
|
||||||
nullptr, nullptr,
|
|
||||||
};
|
|
||||||
|
|
||||||
const auto index = static_cast<u32>(kind);
|
|
||||||
|
|
||||||
if (index > functions.size() || functions[index] == nullptr) {
|
|
||||||
LOG_ERROR(Service_AM, "Invalid shim_kind={:08X}", index);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto function = functions[index];
|
|
||||||
(this->*function)();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::ExecuteInternal() {
|
|
||||||
using WebAppletExecutor = void (WebBrowser::*)();
|
|
||||||
|
|
||||||
constexpr std::array<WebAppletExecutor, SHIM_KIND_COUNT> functions{
|
|
||||||
nullptr, &WebBrowser::ExecuteShop,
|
|
||||||
nullptr, &WebBrowser::ExecuteOffline,
|
|
||||||
nullptr, nullptr,
|
|
||||||
nullptr, nullptr,
|
|
||||||
};
|
|
||||||
|
|
||||||
const auto index = static_cast<u32>(kind);
|
|
||||||
|
|
||||||
if (index > functions.size() || functions[index] == nullptr) {
|
|
||||||
LOG_ERROR(Service_AM, "Invalid shim_kind={:08X}", index);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto function = functions[index];
|
|
||||||
(this->*function)();
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::InitializeShop() {
|
|
||||||
if (frontend_e_commerce == nullptr) {
|
|
||||||
LOG_ERROR(Service_AM, "Missing ECommerce Applet frontend!");
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto user_id_data = args.find(WebArgTLVType::UserID);
|
|
||||||
|
|
||||||
user_id = std::nullopt;
|
|
||||||
if (user_id_data != args.end()) {
|
|
||||||
user_id = u128{};
|
|
||||||
std::memcpy(user_id->data(), user_id_data->second.data(), sizeof(u128));
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto url = args.find(WebArgTLVType::ShopArgumentsURL);
|
|
||||||
|
|
||||||
if (url == args.end()) {
|
|
||||||
LOG_ERROR(Service_AM, "Missing EShop Arguments URL for initialization!");
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<std::string> split_query;
|
|
||||||
Common::SplitString(Common::StringFromFixedZeroTerminatedBuffer(
|
|
||||||
reinterpret_cast<const char*>(url->second.data()), url->second.size()),
|
|
||||||
'?', split_query);
|
|
||||||
|
|
||||||
// 2 -> Main URL '?' Query Parameters
|
|
||||||
// Less is missing info, More is malformed
|
|
||||||
if (split_query.size() != 2) {
|
|
||||||
LOG_ERROR(Service_AM, "EShop Arguments has more than one question mark, malformed");
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
std::vector<std::string> queries;
|
|
||||||
Common::SplitString(split_query[1], '&', queries);
|
|
||||||
|
|
||||||
const auto split_single_query =
|
|
||||||
[](const std::string& in) -> std::pair<std::string, std::string> {
|
|
||||||
const auto index = in.find('=');
|
|
||||||
if (index == std::string::npos || index == in.size() - 1) {
|
|
||||||
return {in, ""};
|
|
||||||
}
|
|
||||||
|
|
||||||
return {in.substr(0, index), in.substr(index + 1)};
|
|
||||||
};
|
|
||||||
|
|
||||||
std::transform(queries.begin(), queries.end(),
|
|
||||||
std::inserter(shop_query, std::next(shop_query.begin())), split_single_query);
|
|
||||||
|
|
||||||
const auto scene = shop_query.find("scene");
|
|
||||||
|
|
||||||
if (scene == shop_query.end()) {
|
|
||||||
LOG_ERROR(Service_AM, "No scene parameter was passed via shop query!");
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
const std::map<std::string, ShopWebTarget, std::less<>> target_map{
|
|
||||||
{"product_detail", ShopWebTarget::ApplicationInfo},
|
|
||||||
{"aocs", ShopWebTarget::AddOnContentList},
|
|
||||||
{"subscriptions", ShopWebTarget::SubscriptionList},
|
|
||||||
{"consumption", ShopWebTarget::ConsumableItemList},
|
|
||||||
{"settings", ShopWebTarget::Settings},
|
|
||||||
{"top", ShopWebTarget::Home},
|
|
||||||
};
|
|
||||||
|
|
||||||
const auto target = target_map.find(scene->second);
|
|
||||||
if (target == target_map.end()) {
|
|
||||||
LOG_ERROR(Service_AM, "Scene for shop query is invalid! (scene={})", scene->second);
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
shop_web_target = target->second;
|
|
||||||
|
|
||||||
const auto title_id_data = shop_query.find("dst_app_id");
|
|
||||||
if (title_id_data != shop_query.end()) {
|
|
||||||
title_id = std::stoull(title_id_data->second, nullptr, 0x10);
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto mode_data = shop_query.find("mode");
|
|
||||||
if (mode_data != shop_query.end()) {
|
|
||||||
shop_full_display = mode_data->second == "full";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::InitializeOffline() {
|
|
||||||
if (args.find(WebArgTLVType::DocumentPath) == args.end() ||
|
|
||||||
args.find(WebArgTLVType::DocumentKind) == args.end() ||
|
|
||||||
args.find(WebArgTLVType::ApplicationID) == args.end()) {
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
LOG_ERROR(Service_AM, "Missing necessary parameters for initialization!");
|
|
||||||
}
|
|
||||||
|
|
||||||
const auto url_data = args[WebArgTLVType::DocumentPath];
|
|
||||||
filename = Common::StringFromFixedZeroTerminatedBuffer(
|
|
||||||
reinterpret_cast<const char*>(url_data.data()), url_data.size());
|
|
||||||
|
|
||||||
OfflineWebSource source;
|
|
||||||
ASSERT(args[WebArgTLVType::DocumentKind].size() >= 4);
|
|
||||||
std::memcpy(&source, args[WebArgTLVType::DocumentKind].data(), sizeof(OfflineWebSource));
|
|
||||||
|
|
||||||
constexpr std::array<const char*, 3> WEB_SOURCE_NAMES{
|
|
||||||
"manual",
|
|
||||||
"legal",
|
|
||||||
"system",
|
|
||||||
};
|
|
||||||
|
|
||||||
temporary_dir =
|
|
||||||
Common::FS::SanitizePath(Common::FS::GetUserPath(Common::FS::UserPath::CacheDir) +
|
|
||||||
"web_applet_" + WEB_SOURCE_NAMES[static_cast<u32>(source) - 1],
|
|
||||||
Common::FS::DirectorySeparator::PlatformDefault);
|
|
||||||
Common::FS::DeleteDirRecursively(temporary_dir);
|
|
||||||
|
|
||||||
u64 title_id = 0; // 0 corresponds to current process
|
|
||||||
ASSERT(args[WebArgTLVType::ApplicationID].size() >= 0x8);
|
|
||||||
std::memcpy(&title_id, args[WebArgTLVType::ApplicationID].data(), sizeof(u64));
|
|
||||||
FileSys::ContentRecordType type = FileSys::ContentRecordType::Data;
|
|
||||||
|
|
||||||
switch (source) {
|
|
||||||
case OfflineWebSource::OfflineHtmlPage:
|
|
||||||
// While there is an AppID TLV field, in official SW this is always ignored.
|
|
||||||
title_id = 0;
|
|
||||||
type = FileSys::ContentRecordType::HtmlDocument;
|
|
||||||
break;
|
break;
|
||||||
case OfflineWebSource::ApplicationLegalInformation:
|
case ShimKind::Login:
|
||||||
type = FileSys::ContentRecordType::LegalInformation;
|
ExecuteLogin();
|
||||||
break;
|
break;
|
||||||
case OfflineWebSource::SystemDataPage:
|
case ShimKind::Offline:
|
||||||
type = FileSys::ContentRecordType::Data;
|
ExecuteOffline();
|
||||||
break;
|
break;
|
||||||
}
|
case ShimKind::Share:
|
||||||
|
ExecuteShare();
|
||||||
if (title_id == 0) {
|
|
||||||
title_id = system.CurrentProcess()->GetTitleID();
|
|
||||||
}
|
|
||||||
|
|
||||||
offline_romfs = GetApplicationRomFS(system, title_id, type);
|
|
||||||
if (offline_romfs == nullptr) {
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
LOG_ERROR(Service_AM, "Failed to find offline data for request!");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string path_additional_directory;
|
|
||||||
if (source == OfflineWebSource::OfflineHtmlPage) {
|
|
||||||
path_additional_directory = std::string(DIR_SEP).append("html-document");
|
|
||||||
}
|
|
||||||
|
|
||||||
filename =
|
|
||||||
Common::FS::SanitizePath(temporary_dir + path_additional_directory + DIR_SEP + filename,
|
|
||||||
Common::FS::DirectorySeparator::PlatformDefault);
|
|
||||||
}
|
|
||||||
|
|
||||||
void WebBrowser::ExecuteShop() {
|
|
||||||
const auto callback = [this]() { Finalize(); };
|
|
||||||
|
|
||||||
const auto check_optional_parameter = [this](const auto& p) {
|
|
||||||
if (!p.has_value()) {
|
|
||||||
LOG_ERROR(Service_AM, "Missing one or more necessary parameters for execution!");
|
|
||||||
status = RESULT_UNKNOWN;
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
};
|
|
||||||
|
|
||||||
switch (shop_web_target) {
|
|
||||||
case ShopWebTarget::ApplicationInfo:
|
|
||||||
if (!check_optional_parameter(title_id))
|
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowApplicationInformation(callback, *title_id, user_id,
|
|
||||||
shop_full_display, shop_extra_parameter);
|
|
||||||
break;
|
break;
|
||||||
case ShopWebTarget::AddOnContentList:
|
case ShimKind::Web:
|
||||||
if (!check_optional_parameter(title_id))
|
ExecuteWeb();
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowAddOnContentList(callback, *title_id, user_id, shop_full_display);
|
|
||||||
break;
|
break;
|
||||||
case ShopWebTarget::ConsumableItemList:
|
case ShimKind::Wifi:
|
||||||
if (!check_optional_parameter(title_id))
|
ExecuteWifi();
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowConsumableItemList(callback, *title_id, user_id);
|
|
||||||
break;
|
break;
|
||||||
case ShopWebTarget::Home:
|
case ShimKind::Lobby:
|
||||||
if (!check_optional_parameter(user_id))
|
ExecuteLobby();
|
||||||
return;
|
|
||||||
if (!check_optional_parameter(shop_full_display))
|
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowShopHome(callback, *user_id, *shop_full_display);
|
|
||||||
break;
|
|
||||||
case ShopWebTarget::Settings:
|
|
||||||
if (!check_optional_parameter(user_id))
|
|
||||||
return;
|
|
||||||
if (!check_optional_parameter(shop_full_display))
|
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowSettings(callback, *user_id, *shop_full_display);
|
|
||||||
break;
|
|
||||||
case ShopWebTarget::SubscriptionList:
|
|
||||||
if (!check_optional_parameter(title_id))
|
|
||||||
return;
|
|
||||||
frontend_e_commerce->ShowSubscriptionList(callback, *title_id, user_id);
|
|
||||||
break;
|
break;
|
||||||
default:
|
default:
|
||||||
UNREACHABLE();
|
UNREACHABLE_MSG("Invalid ShimKind={}", web_arg_header.shim_kind);
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExtractOfflineRomFS() {
|
||||||
|
LOG_DEBUG(Service_AM, "Extracting RomFS to {}", offline_cache_dir);
|
||||||
|
|
||||||
|
const auto extracted_romfs_dir =
|
||||||
|
FileSys::ExtractRomFS(offline_romfs, FileSys::RomFSExtractionType::SingleDiscard);
|
||||||
|
|
||||||
|
const auto temp_dir =
|
||||||
|
system.GetFilesystem()->CreateDirectory(offline_cache_dir, FileSys::Mode::ReadWrite);
|
||||||
|
|
||||||
|
FileSys::VfsRawCopyD(extracted_romfs_dir, temp_dir);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::WebBrowserExit(WebExitReason exit_reason, std::string last_url) {
|
||||||
|
if ((web_arg_header.shim_kind == ShimKind::Share &&
|
||||||
|
web_applet_version >= WebAppletVersion::Version196608) ||
|
||||||
|
(web_arg_header.shim_kind == ShimKind::Web &&
|
||||||
|
web_applet_version >= WebAppletVersion::Version524288)) {
|
||||||
|
// TODO: Push Output TLVs instead of a WebCommonReturnValue
|
||||||
|
}
|
||||||
|
|
||||||
|
WebCommonReturnValue web_common_return_value;
|
||||||
|
|
||||||
|
web_common_return_value.exit_reason = exit_reason;
|
||||||
|
std::memcpy(&web_common_return_value.last_url, last_url.data(), last_url.size());
|
||||||
|
web_common_return_value.last_url_size = last_url.size();
|
||||||
|
|
||||||
|
LOG_DEBUG(Service_AM, "WebCommonReturnValue: exit_reason={}, last_url={}, last_url_size={}",
|
||||||
|
exit_reason, last_url, last_url.size());
|
||||||
|
|
||||||
|
complete = true;
|
||||||
|
std::vector<u8> out_data(sizeof(WebCommonReturnValue));
|
||||||
|
std::memcpy(out_data.data(), &web_common_return_value, out_data.size());
|
||||||
|
broker.PushNormalDataFromApplet(std::make_shared<IStorage>(system, std::move(out_data)));
|
||||||
|
broker.SignalStateChanged();
|
||||||
|
}
|
||||||
|
|
||||||
|
bool WebBrowser::InputTLVExistsInMap(WebArgInputTLVType input_tlv_type) const {
|
||||||
|
return web_arg_input_tlv_map.find(input_tlv_type) != web_arg_input_tlv_map.end();
|
||||||
|
}
|
||||||
|
|
||||||
|
std::optional<std::vector<u8>> WebBrowser::GetInputTLVData(WebArgInputTLVType input_tlv_type) {
|
||||||
|
const auto map_it = web_arg_input_tlv_map.find(input_tlv_type);
|
||||||
|
|
||||||
|
if (map_it == web_arg_input_tlv_map.end()) {
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
return map_it->second;
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeShop() {}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeLogin() {}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeOffline() {
|
||||||
|
const auto document_path =
|
||||||
|
ParseStringValue(GetInputTLVData(WebArgInputTLVType::DocumentPath).value());
|
||||||
|
|
||||||
|
const auto document_kind =
|
||||||
|
ParseRawValue<DocumentKind>(GetInputTLVData(WebArgInputTLVType::DocumentKind).value());
|
||||||
|
|
||||||
|
std::string additional_paths;
|
||||||
|
|
||||||
|
switch (document_kind) {
|
||||||
|
case DocumentKind::OfflineHtmlPage:
|
||||||
|
default:
|
||||||
|
title_id = system.CurrentProcess()->GetTitleID();
|
||||||
|
nca_type = FileSys::ContentRecordType::HtmlDocument;
|
||||||
|
additional_paths = "html-document";
|
||||||
|
break;
|
||||||
|
case DocumentKind::ApplicationLegalInformation:
|
||||||
|
title_id = ParseRawValue<u64>(GetInputTLVData(WebArgInputTLVType::ApplicationID).value());
|
||||||
|
nca_type = FileSys::ContentRecordType::LegalInformation;
|
||||||
|
break;
|
||||||
|
case DocumentKind::SystemDataPage:
|
||||||
|
title_id = ParseRawValue<u64>(GetInputTLVData(WebArgInputTLVType::SystemDataID).value());
|
||||||
|
nca_type = FileSys::ContentRecordType::Data;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
static constexpr std::array<const char*, 3> RESOURCE_TYPES{
|
||||||
|
"manual",
|
||||||
|
"legal_information",
|
||||||
|
"system_data",
|
||||||
|
};
|
||||||
|
|
||||||
|
offline_cache_dir = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/offline_web_applet_{}/{:016X}",
|
||||||
|
Common::FS::GetUserPath(Common::FS::UserPath::CacheDir),
|
||||||
|
RESOURCE_TYPES[static_cast<u32>(document_kind) - 1], title_id),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
offline_document = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/{}/{}", offline_cache_dir, additional_paths, document_path),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeShare() {}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeWeb() {
|
||||||
|
external_url = ParseStringValue(GetInputTLVData(WebArgInputTLVType::InitialURL).value());
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeWifi() {}
|
||||||
|
|
||||||
|
void WebBrowser::InitializeLobby() {}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteShop() {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, Shop Applet is not implemented");
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteLogin() {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, Login Applet is not implemented");
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
}
|
||||||
|
|
||||||
void WebBrowser::ExecuteOffline() {
|
void WebBrowser::ExecuteOffline() {
|
||||||
frontend.OpenPageLocal(
|
const auto main_url = Common::FS::SanitizePath(GetMainURL(offline_document),
|
||||||
filename, [this] { UnpackRomFS(); }, [this] { Finalize(); });
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
if (!Common::FS::Exists(main_url)) {
|
||||||
|
offline_romfs = GetOfflineRomFS(system, title_id, nca_type);
|
||||||
|
|
||||||
|
if (offline_romfs == nullptr) {
|
||||||
|
LOG_ERROR(Service_AM,
|
||||||
|
"RomFS with title_id={:016X} and nca_type={} cannot be extracted!", title_id,
|
||||||
|
nca_type);
|
||||||
|
WebBrowserExit(WebExitReason::WindowClosed);
|
||||||
|
return;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
LOG_INFO(Service_AM, "Opening offline document at {}", offline_document);
|
||||||
|
|
||||||
|
frontend.OpenLocalWebPage(
|
||||||
|
offline_document, [this] { ExtractOfflineRomFS(); },
|
||||||
|
[this](WebExitReason exit_reason, std::string last_url) {
|
||||||
|
WebBrowserExit(exit_reason, last_url);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteShare() {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, Share Applet is not implemented");
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteWeb() {
|
||||||
|
LOG_INFO(Service_AM, "Opening external URL at {}", external_url);
|
||||||
|
|
||||||
|
frontend.OpenExternalWebPage(external_url,
|
||||||
|
[this](WebExitReason exit_reason, std::string last_url) {
|
||||||
|
WebBrowserExit(exit_reason, last_url);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteWifi() {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, Wifi Applet is not implemented");
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
}
|
||||||
|
|
||||||
|
void WebBrowser::ExecuteLobby() {
|
||||||
|
LOG_WARNING(Service_AM, "(STUBBED) called, Lobby Applet is not implemented");
|
||||||
|
WebBrowserExit(WebExitReason::EndButtonPressed);
|
||||||
|
}
|
||||||
} // namespace Service::AM::Applets
|
} // namespace Service::AM::Applets
|
||||||
|
|
|
@ -1,28 +1,31 @@
|
||||||
// Copyright 2018 yuzu emulator team
|
// Copyright 2020 yuzu Emulator Project
|
||||||
// Licensed under GPLv2 or any later version
|
// Licensed under GPLv2 or any later version
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <map>
|
#include <optional>
|
||||||
|
|
||||||
|
#include "common/common_funcs.h"
|
||||||
|
#include "common/common_types.h"
|
||||||
#include "core/file_sys/vfs_types.h"
|
#include "core/file_sys/vfs_types.h"
|
||||||
#include "core/hle/service/am/am.h"
|
#include "core/hle/result.h"
|
||||||
#include "core/hle/service/am/applets/applets.h"
|
#include "core/hle/service/am/applets/applets.h"
|
||||||
|
#include "core/hle/service/am/applets/web_types.h"
|
||||||
|
|
||||||
namespace Core {
|
namespace Core {
|
||||||
class System;
|
class System;
|
||||||
}
|
}
|
||||||
|
|
||||||
namespace Service::AM::Applets {
|
namespace FileSys {
|
||||||
|
enum class ContentRecordType : u8;
|
||||||
|
}
|
||||||
|
|
||||||
enum class ShimKind : u32;
|
namespace Service::AM::Applets {
|
||||||
enum class ShopWebTarget;
|
|
||||||
enum class WebArgTLVType : u16;
|
|
||||||
|
|
||||||
class WebBrowser final : public Applet {
|
class WebBrowser final : public Applet {
|
||||||
public:
|
public:
|
||||||
WebBrowser(Core::System& system_, Core::Frontend::WebBrowserApplet& frontend_,
|
WebBrowser(Core::System& system_, const Core::Frontend::WebBrowserApplet& frontend_);
|
||||||
Core::Frontend::ECommerceApplet* frontend_e_commerce_ = nullptr);
|
|
||||||
|
|
||||||
~WebBrowser() override;
|
~WebBrowser() override;
|
||||||
|
|
||||||
|
@ -33,49 +36,50 @@ public:
|
||||||
void ExecuteInteractive() override;
|
void ExecuteInteractive() override;
|
||||||
void Execute() override;
|
void Execute() override;
|
||||||
|
|
||||||
// Callback to be fired when the frontend needs the manual RomFS unpacked to temporary
|
void ExtractOfflineRomFS();
|
||||||
// directory. This is a blocking call and may take a while as some manuals can be up to 100MB in
|
|
||||||
// size. Attempting to access files at filename before invocation is likely to not work.
|
|
||||||
void UnpackRomFS();
|
|
||||||
|
|
||||||
// Callback to be fired when the frontend is finished browsing. This will delete the temporary
|
void WebBrowserExit(WebExitReason exit_reason, std::string last_url = "");
|
||||||
// manual RomFS extracted files, so ensure this is only called at actual finalization.
|
|
||||||
void Finalize();
|
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void InitializeInternal();
|
bool InputTLVExistsInMap(WebArgInputTLVType input_tlv_type) const;
|
||||||
void ExecuteInternal();
|
|
||||||
|
|
||||||
// Specific initializers for the types of web applets
|
std::optional<std::vector<u8>> GetInputTLVData(WebArgInputTLVType input_tlv_type);
|
||||||
|
|
||||||
|
// Initializers for the various types of browser applets
|
||||||
void InitializeShop();
|
void InitializeShop();
|
||||||
|
void InitializeLogin();
|
||||||
void InitializeOffline();
|
void InitializeOffline();
|
||||||
|
void InitializeShare();
|
||||||
|
void InitializeWeb();
|
||||||
|
void InitializeWifi();
|
||||||
|
void InitializeLobby();
|
||||||
|
|
||||||
// Specific executors for the types of web applets
|
// Executors for the various types of browser applets
|
||||||
void ExecuteShop();
|
void ExecuteShop();
|
||||||
|
void ExecuteLogin();
|
||||||
void ExecuteOffline();
|
void ExecuteOffline();
|
||||||
|
void ExecuteShare();
|
||||||
|
void ExecuteWeb();
|
||||||
|
void ExecuteWifi();
|
||||||
|
void ExecuteLobby();
|
||||||
|
|
||||||
Core::Frontend::WebBrowserApplet& frontend;
|
const Core::Frontend::WebBrowserApplet& frontend;
|
||||||
|
|
||||||
// Extra frontends for specialized functions
|
bool complete{false};
|
||||||
Core::Frontend::ECommerceApplet* frontend_e_commerce;
|
ResultCode status{RESULT_SUCCESS};
|
||||||
|
|
||||||
bool complete = false;
|
WebAppletVersion web_applet_version;
|
||||||
bool unpacked = false;
|
WebExitReason web_exit_reason;
|
||||||
ResultCode status = RESULT_SUCCESS;
|
WebArgHeader web_arg_header;
|
||||||
|
WebArgInputTLVMap web_arg_input_tlv_map;
|
||||||
ShimKind kind;
|
|
||||||
std::map<WebArgTLVType, std::vector<u8>> args;
|
|
||||||
|
|
||||||
|
u64 title_id;
|
||||||
|
FileSys::ContentRecordType nca_type;
|
||||||
|
std::string offline_cache_dir;
|
||||||
|
std::string offline_document;
|
||||||
FileSys::VirtualFile offline_romfs;
|
FileSys::VirtualFile offline_romfs;
|
||||||
std::string temporary_dir;
|
|
||||||
std::string filename;
|
|
||||||
|
|
||||||
ShopWebTarget shop_web_target;
|
std::string external_url;
|
||||||
std::map<std::string, std::string, std::less<>> shop_query;
|
|
||||||
std::optional<u64> title_id = 0;
|
|
||||||
std::optional<u128> user_id;
|
|
||||||
std::optional<bool> shop_full_display;
|
|
||||||
std::string shop_extra_parameter;
|
|
||||||
|
|
||||||
Core::System& system;
|
Core::System& system;
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,178 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <array>
|
||||||
|
#include <unordered_map>
|
||||||
|
#include <vector>
|
||||||
|
|
||||||
|
#include "common/common_funcs.h"
|
||||||
|
#include "common/common_types.h"
|
||||||
|
#include "common/swap.h"
|
||||||
|
|
||||||
|
namespace Service::AM::Applets {
|
||||||
|
|
||||||
|
enum class WebAppletVersion : u32_le {
|
||||||
|
Version0 = 0x0, // Only used by WifiWebAuthApplet
|
||||||
|
Version131072 = 0x20000, // 1.0.0 - 2.3.0
|
||||||
|
Version196608 = 0x30000, // 3.0.0 - 4.1.0
|
||||||
|
Version327680 = 0x50000, // 5.0.0 - 5.1.0
|
||||||
|
Version393216 = 0x60000, // 6.0.0 - 7.0.1
|
||||||
|
Version524288 = 0x80000, // 8.0.0+
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class ShimKind : u32 {
|
||||||
|
Shop = 1,
|
||||||
|
Login = 2,
|
||||||
|
Offline = 3,
|
||||||
|
Share = 4,
|
||||||
|
Web = 5,
|
||||||
|
Wifi = 6,
|
||||||
|
Lobby = 7,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class WebExitReason : u32 {
|
||||||
|
EndButtonPressed = 0,
|
||||||
|
BackButtonPressed = 1,
|
||||||
|
ExitRequested = 2,
|
||||||
|
CallbackURL = 3,
|
||||||
|
WindowClosed = 4,
|
||||||
|
ErrorDialog = 7,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class WebArgInputTLVType : u16 {
|
||||||
|
InitialURL = 0x1,
|
||||||
|
CallbackURL = 0x3,
|
||||||
|
CallbackableURL = 0x4,
|
||||||
|
ApplicationID = 0x5,
|
||||||
|
DocumentPath = 0x6,
|
||||||
|
DocumentKind = 0x7,
|
||||||
|
SystemDataID = 0x8,
|
||||||
|
ShareStartPage = 0x9,
|
||||||
|
Whitelist = 0xA,
|
||||||
|
News = 0xB,
|
||||||
|
UserID = 0xE,
|
||||||
|
AlbumEntry0 = 0xF,
|
||||||
|
ScreenShotEnabled = 0x10,
|
||||||
|
EcClientCertEnabled = 0x11,
|
||||||
|
PlayReportEnabled = 0x13,
|
||||||
|
BootDisplayKind = 0x17,
|
||||||
|
BackgroundKind = 0x18,
|
||||||
|
FooterEnabled = 0x19,
|
||||||
|
PointerEnabled = 0x1A,
|
||||||
|
LeftStickMode = 0x1B,
|
||||||
|
KeyRepeatFrame1 = 0x1C,
|
||||||
|
KeyRepeatFrame2 = 0x1D,
|
||||||
|
BootAsMediaPlayerInverted = 0x1E,
|
||||||
|
DisplayURLKind = 0x1F,
|
||||||
|
BootAsMediaPlayer = 0x21,
|
||||||
|
ShopJumpEnabled = 0x22,
|
||||||
|
MediaAutoPlayEnabled = 0x23,
|
||||||
|
LobbyParameter = 0x24,
|
||||||
|
ApplicationAlbumEntry = 0x26,
|
||||||
|
JsExtensionEnabled = 0x27,
|
||||||
|
AdditionalCommentText = 0x28,
|
||||||
|
TouchEnabledOnContents = 0x29,
|
||||||
|
UserAgentAdditionalString = 0x2A,
|
||||||
|
AdditionalMediaData0 = 0x2B,
|
||||||
|
MediaPlayerAutoCloseEnabled = 0x2C,
|
||||||
|
PageCacheEnabled = 0x2D,
|
||||||
|
WebAudioEnabled = 0x2E,
|
||||||
|
YouTubeVideoWhitelist = 0x31,
|
||||||
|
FooterFixedKind = 0x32,
|
||||||
|
PageFadeEnabled = 0x33,
|
||||||
|
MediaCreatorApplicationRatingAge = 0x34,
|
||||||
|
BootLoadingIconEnabled = 0x35,
|
||||||
|
PageScrollIndicatorEnabled = 0x36,
|
||||||
|
MediaPlayerSpeedControlEnabled = 0x37,
|
||||||
|
AlbumEntry1 = 0x38,
|
||||||
|
AlbumEntry2 = 0x39,
|
||||||
|
AlbumEntry3 = 0x3A,
|
||||||
|
AdditionalMediaData1 = 0x3B,
|
||||||
|
AdditionalMediaData2 = 0x3C,
|
||||||
|
AdditionalMediaData3 = 0x3D,
|
||||||
|
BootFooterButton = 0x3E,
|
||||||
|
OverrideWebAudioVolume = 0x3F,
|
||||||
|
OverrideMediaAudioVolume = 0x40,
|
||||||
|
BootMode = 0x41,
|
||||||
|
WebSessionEnabled = 0x42,
|
||||||
|
MediaPlayerOfflineEnabled = 0x43,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class WebArgOutputTLVType : u16 {
|
||||||
|
ShareExitReason = 0x1,
|
||||||
|
LastURL = 0x2,
|
||||||
|
LastURLSize = 0x3,
|
||||||
|
SharePostResult = 0x4,
|
||||||
|
PostServiceName = 0x5,
|
||||||
|
PostServiceNameSize = 0x6,
|
||||||
|
PostID = 0x7,
|
||||||
|
PostIDSize = 0x8,
|
||||||
|
MediaPlayerAutoClosedByCompletion = 0x9,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class DocumentKind : u32 {
|
||||||
|
OfflineHtmlPage = 1,
|
||||||
|
ApplicationLegalInformation = 2,
|
||||||
|
SystemDataPage = 3,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class ShareStartPage : u32 {
|
||||||
|
Default,
|
||||||
|
Settings,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class BootDisplayKind : u32 {
|
||||||
|
Default,
|
||||||
|
White,
|
||||||
|
Black,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class BackgroundKind : u32 {
|
||||||
|
Default,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class LeftStickMode : u32 {
|
||||||
|
Pointer,
|
||||||
|
Cursor,
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class WebSessionBootMode : u32 {
|
||||||
|
AllForeground,
|
||||||
|
AllForegroundInitiallyHidden,
|
||||||
|
};
|
||||||
|
|
||||||
|
struct WebArgHeader {
|
||||||
|
u16 total_tlv_entries{};
|
||||||
|
INSERT_PADDING_BYTES(2);
|
||||||
|
ShimKind shim_kind{};
|
||||||
|
};
|
||||||
|
static_assert(sizeof(WebArgHeader) == 0x8, "WebArgHeader has incorrect size.");
|
||||||
|
|
||||||
|
struct WebArgInputTLV {
|
||||||
|
WebArgInputTLVType input_tlv_type{};
|
||||||
|
u16 arg_data_size{};
|
||||||
|
INSERT_PADDING_WORDS(1);
|
||||||
|
};
|
||||||
|
static_assert(sizeof(WebArgInputTLV) == 0x8, "WebArgInputTLV has incorrect size.");
|
||||||
|
|
||||||
|
struct WebArgOutputTLV {
|
||||||
|
WebArgOutputTLVType output_tlv_type{};
|
||||||
|
u16 arg_data_size{};
|
||||||
|
INSERT_PADDING_WORDS(1);
|
||||||
|
};
|
||||||
|
static_assert(sizeof(WebArgOutputTLV) == 0x8, "WebArgOutputTLV has incorrect size.");
|
||||||
|
|
||||||
|
struct WebCommonReturnValue {
|
||||||
|
WebExitReason exit_reason{};
|
||||||
|
INSERT_PADDING_WORDS(1);
|
||||||
|
std::array<char, 0x1000> last_url{};
|
||||||
|
u64 last_url_size{};
|
||||||
|
};
|
||||||
|
static_assert(sizeof(WebCommonReturnValue) == 0x1010, "WebCommonReturnValue has incorrect size.");
|
||||||
|
|
||||||
|
using WebArgInputTLVMap = std::unordered_map<WebArgInputTLVType, std::vector<u8>>;
|
||||||
|
|
||||||
|
} // namespace Service::AM::Applets
|
|
@ -1058,7 +1058,7 @@ void Controller_NPad::ClearAllControllers() {
|
||||||
}
|
}
|
||||||
|
|
||||||
u32 Controller_NPad::GetAndResetPressState() {
|
u32 Controller_NPad::GetAndResetPressState() {
|
||||||
return std::exchange(press_state, 0);
|
return press_state.exchange(0);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool Controller_NPad::IsControllerSupported(NPadControllerType controller) const {
|
bool Controller_NPad::IsControllerSupported(NPadControllerType controller) const {
|
||||||
|
|
|
@ -5,6 +5,7 @@
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <array>
|
#include <array>
|
||||||
|
#include <atomic>
|
||||||
#include "common/bit_field.h"
|
#include "common/bit_field.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
#include "core/frontend/input.h"
|
#include "core/frontend/input.h"
|
||||||
|
@ -415,7 +416,7 @@ private:
|
||||||
bool IsControllerSupported(NPadControllerType controller) const;
|
bool IsControllerSupported(NPadControllerType controller) const;
|
||||||
void RequestPadStateUpdate(u32 npad_id);
|
void RequestPadStateUpdate(u32 npad_id);
|
||||||
|
|
||||||
u32 press_state{};
|
std::atomic<u32> press_state{};
|
||||||
|
|
||||||
NpadStyleSet style{};
|
NpadStyleSet style{};
|
||||||
std::array<NPadEntry, 10> shared_memory_entries{};
|
std::array<NPadEntry, 10> shared_memory_entries{};
|
||||||
|
|
|
@ -673,7 +673,7 @@ public:
|
||||||
explicit NS_VM(Core::System& system_) : ServiceFramework{system_, "ns:vm"} {
|
explicit NS_VM(Core::System& system_) : ServiceFramework{system_, "ns:vm"} {
|
||||||
// clang-format off
|
// clang-format off
|
||||||
static const FunctionInfo functions[] = {
|
static const FunctionInfo functions[] = {
|
||||||
{1200, nullptr, "NeedsUpdateVulnerability"},
|
{1200, &NS_VM::NeedsUpdateVulnerability, "NeedsUpdateVulnerability"},
|
||||||
{1201, nullptr, "UpdateSafeSystemVersionForDebug"},
|
{1201, nullptr, "UpdateSafeSystemVersionForDebug"},
|
||||||
{1202, nullptr, "GetSafeSystemVersion"},
|
{1202, nullptr, "GetSafeSystemVersion"},
|
||||||
};
|
};
|
||||||
|
@ -681,6 +681,15 @@ public:
|
||||||
|
|
||||||
RegisterHandlers(functions);
|
RegisterHandlers(functions);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
void NeedsUpdateVulnerability(Kernel::HLERequestContext& ctx) {
|
||||||
|
LOG_WARNING(Service_NS, "(STUBBED) called");
|
||||||
|
|
||||||
|
IPC::ResponseBuilder rb{ctx, 3};
|
||||||
|
rb.Push(RESULT_SUCCESS);
|
||||||
|
rb.Push(false);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) {
|
void InstallInterfaces(SM::ServiceManager& service_manager, Core::System& system) {
|
||||||
|
|
|
@ -27,29 +27,11 @@
|
||||||
|
|
||||||
namespace Service::NS {
|
namespace Service::NS {
|
||||||
|
|
||||||
enum class FontArchives : u64 {
|
|
||||||
Extension = 0x0100000000000810,
|
|
||||||
Standard = 0x0100000000000811,
|
|
||||||
Korean = 0x0100000000000812,
|
|
||||||
ChineseTraditional = 0x0100000000000813,
|
|
||||||
ChineseSimple = 0x0100000000000814,
|
|
||||||
};
|
|
||||||
|
|
||||||
struct FontRegion {
|
struct FontRegion {
|
||||||
u32 offset;
|
u32 offset;
|
||||||
u32 size;
|
u32 size;
|
||||||
};
|
};
|
||||||
|
|
||||||
constexpr std::array<std::pair<FontArchives, const char*>, 7> SHARED_FONTS{
|
|
||||||
std::make_pair(FontArchives::Standard, "nintendo_udsg-r_std_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::ChineseSimple, "nintendo_udsg-r_org_zh-cn_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::ChineseSimple, "nintendo_udsg-r_ext_zh-cn_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::ChineseTraditional, "nintendo_udjxh-db_zh-tw_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::Korean, "nintendo_udsg-r_ko_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::Extension, "nintendo_ext_003.bfttf"),
|
|
||||||
std::make_pair(FontArchives::Extension, "nintendo_ext2_003.bfttf"),
|
|
||||||
};
|
|
||||||
|
|
||||||
// The below data is specific to shared font data dumped from Switch on f/w 2.2
|
// The below data is specific to shared font data dumped from Switch on f/w 2.2
|
||||||
// Virtual address and offsets/sizes likely will vary by dump
|
// Virtual address and offsets/sizes likely will vary by dump
|
||||||
[[maybe_unused]] constexpr VAddr SHARED_FONT_MEM_VADDR{0x00000009d3016000ULL};
|
[[maybe_unused]] constexpr VAddr SHARED_FONT_MEM_VADDR{0x00000009d3016000ULL};
|
||||||
|
@ -80,6 +62,18 @@ static void DecryptSharedFont(const std::vector<u32>& input, Kernel::PhysicalMem
|
||||||
offset += transformed_font.size() * sizeof(u32);
|
offset += transformed_font.size() * sizeof(u32);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void DecryptSharedFontToTTF(const std::vector<u32>& input, std::vector<u8>& output) {
|
||||||
|
ASSERT_MSG(input[0] == EXPECTED_MAGIC, "Failed to derive key, unexpected magic number");
|
||||||
|
|
||||||
|
const u32 KEY = input[0] ^ EXPECTED_RESULT; // Derive key using an inverse xor
|
||||||
|
std::vector<u32> transformed_font(input.size());
|
||||||
|
// TODO(ogniK): Figure out a better way to do this
|
||||||
|
std::transform(input.begin(), input.end(), transformed_font.begin(),
|
||||||
|
[&KEY](u32 font_data) { return Common::swap32(font_data ^ KEY); });
|
||||||
|
transformed_font[1] = Common::swap32(transformed_font[1]) ^ KEY; // "re-encrypt" the size
|
||||||
|
std::memcpy(output.data(), transformed_font.data() + 2, transformed_font.size() * sizeof(u32));
|
||||||
|
}
|
||||||
|
|
||||||
void EncryptSharedFont(const std::vector<u32>& input, std::vector<u8>& output,
|
void EncryptSharedFont(const std::vector<u32>& input, std::vector<u8>& output,
|
||||||
std::size_t& offset) {
|
std::size_t& offset) {
|
||||||
ASSERT_MSG(offset + (input.size() * sizeof(u32)) < SHARED_FONT_MEM_SIZE,
|
ASSERT_MSG(offset + (input.size() * sizeof(u32)) < SHARED_FONT_MEM_SIZE,
|
||||||
|
|
|
@ -16,6 +16,25 @@ class FileSystemController;
|
||||||
|
|
||||||
namespace NS {
|
namespace NS {
|
||||||
|
|
||||||
|
enum class FontArchives : u64 {
|
||||||
|
Extension = 0x0100000000000810,
|
||||||
|
Standard = 0x0100000000000811,
|
||||||
|
Korean = 0x0100000000000812,
|
||||||
|
ChineseTraditional = 0x0100000000000813,
|
||||||
|
ChineseSimple = 0x0100000000000814,
|
||||||
|
};
|
||||||
|
|
||||||
|
constexpr std::array<std::pair<FontArchives, const char*>, 7> SHARED_FONTS{
|
||||||
|
std::make_pair(FontArchives::Standard, "nintendo_udsg-r_std_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::ChineseSimple, "nintendo_udsg-r_org_zh-cn_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::ChineseSimple, "nintendo_udsg-r_ext_zh-cn_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::ChineseTraditional, "nintendo_udjxh-db_zh-tw_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::Korean, "nintendo_udsg-r_ko_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::Extension, "nintendo_ext_003.bfttf"),
|
||||||
|
std::make_pair(FontArchives::Extension, "nintendo_ext2_003.bfttf"),
|
||||||
|
};
|
||||||
|
|
||||||
|
void DecryptSharedFontToTTF(const std::vector<u32>& input, std::vector<u8>& output);
|
||||||
void EncryptSharedFont(const std::vector<u32>& input, std::vector<u8>& output, std::size_t& offset);
|
void EncryptSharedFont(const std::vector<u32>& input, std::vector<u8>& output, std::size_t& offset);
|
||||||
|
|
||||||
class PL_U final : public ServiceFramework<PL_U> {
|
class PL_U final : public ServiceFramework<PL_U> {
|
||||||
|
|
|
@ -141,6 +141,8 @@ add_executable(yuzu
|
||||||
util/limitable_input_dialog.h
|
util/limitable_input_dialog.h
|
||||||
util/sequence_dialog/sequence_dialog.cpp
|
util/sequence_dialog/sequence_dialog.cpp
|
||||||
util/sequence_dialog/sequence_dialog.h
|
util/sequence_dialog/sequence_dialog.h
|
||||||
|
util/url_request_interceptor.cpp
|
||||||
|
util/url_request_interceptor.h
|
||||||
util/util.cpp
|
util/util.cpp
|
||||||
util/util.h
|
util/util.h
|
||||||
compatdb.cpp
|
compatdb.cpp
|
||||||
|
|
|
@ -1,115 +1,414 @@
|
||||||
// Copyright 2018 yuzu Emulator Project
|
// Copyright 2020 yuzu Emulator Project
|
||||||
// Licensed under GPLv2 or any later version
|
// Licensed under GPLv2 or any later version
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#include <mutex>
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
#include <QKeyEvent>
|
#include <QKeyEvent>
|
||||||
|
|
||||||
#include "core/hle/lock.h"
|
#include <QWebEngineProfile>
|
||||||
|
#include <QWebEngineScript>
|
||||||
|
#include <QWebEngineScriptCollection>
|
||||||
|
#include <QWebEngineSettings>
|
||||||
|
#include <QWebEngineUrlScheme>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#include "common/file_util.h"
|
||||||
|
#include "core/core.h"
|
||||||
|
#include "core/frontend/input_interpreter.h"
|
||||||
|
#include "input_common/keyboard.h"
|
||||||
|
#include "input_common/main.h"
|
||||||
#include "yuzu/applets/web_browser.h"
|
#include "yuzu/applets/web_browser.h"
|
||||||
|
#include "yuzu/applets/web_browser_scripts.h"
|
||||||
#include "yuzu/main.h"
|
#include "yuzu/main.h"
|
||||||
|
#include "yuzu/util/url_request_interceptor.h"
|
||||||
|
|
||||||
#ifdef YUZU_USE_QT_WEB_ENGINE
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
constexpr char NX_SHIM_INJECT_SCRIPT[] = R"(
|
namespace {
|
||||||
window.nx = {};
|
|
||||||
window.nx.playReport = {};
|
|
||||||
window.nx.playReport.setCounterSetIdentifier = function () {
|
|
||||||
console.log("nx.playReport.setCounterSetIdentifier called - unimplemented");
|
|
||||||
};
|
|
||||||
|
|
||||||
window.nx.playReport.incrementCounter = function () {
|
constexpr int HIDButtonToKey(HIDButton button) {
|
||||||
console.log("nx.playReport.incrementCounter called - unimplemented");
|
switch (button) {
|
||||||
};
|
case HIDButton::DLeft:
|
||||||
|
case HIDButton::LStickLeft:
|
||||||
|
return Qt::Key_Left;
|
||||||
|
case HIDButton::DUp:
|
||||||
|
case HIDButton::LStickUp:
|
||||||
|
return Qt::Key_Up;
|
||||||
|
case HIDButton::DRight:
|
||||||
|
case HIDButton::LStickRight:
|
||||||
|
return Qt::Key_Right;
|
||||||
|
case HIDButton::DDown:
|
||||||
|
case HIDButton::LStickDown:
|
||||||
|
return Qt::Key_Down;
|
||||||
|
default:
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
window.nx.footer = {};
|
} // Anonymous namespace
|
||||||
window.nx.footer.unsetAssign = function () {
|
|
||||||
console.log("nx.footer.unsetAssign called - unimplemented");
|
|
||||||
};
|
|
||||||
|
|
||||||
var yuzu_key_callbacks = [];
|
QtNXWebEngineView::QtNXWebEngineView(QWidget* parent, Core::System& system,
|
||||||
window.nx.footer.setAssign = function(key, discard1, func, discard2) {
|
InputCommon::InputSubsystem* input_subsystem_)
|
||||||
switch (key) {
|
: QWebEngineView(parent), input_subsystem{input_subsystem_},
|
||||||
case 'A':
|
url_interceptor(std::make_unique<UrlRequestInterceptor>()),
|
||||||
yuzu_key_callbacks[0] = func;
|
input_interpreter(std::make_unique<InputInterpreter>(system)),
|
||||||
|
default_profile{QWebEngineProfile::defaultProfile()},
|
||||||
|
global_settings{QWebEngineSettings::globalSettings()} {
|
||||||
|
QWebEngineScript gamepad;
|
||||||
|
QWebEngineScript window_nx;
|
||||||
|
|
||||||
|
gamepad.setName(QStringLiteral("gamepad_script.js"));
|
||||||
|
window_nx.setName(QStringLiteral("window_nx_script.js"));
|
||||||
|
|
||||||
|
gamepad.setSourceCode(QString::fromStdString(GAMEPAD_SCRIPT));
|
||||||
|
window_nx.setSourceCode(QString::fromStdString(WINDOW_NX_SCRIPT));
|
||||||
|
|
||||||
|
gamepad.setInjectionPoint(QWebEngineScript::DocumentCreation);
|
||||||
|
window_nx.setInjectionPoint(QWebEngineScript::DocumentCreation);
|
||||||
|
|
||||||
|
gamepad.setWorldId(QWebEngineScript::MainWorld);
|
||||||
|
window_nx.setWorldId(QWebEngineScript::MainWorld);
|
||||||
|
|
||||||
|
gamepad.setRunsOnSubFrames(true);
|
||||||
|
window_nx.setRunsOnSubFrames(true);
|
||||||
|
|
||||||
|
default_profile->scripts()->insert(gamepad);
|
||||||
|
default_profile->scripts()->insert(window_nx);
|
||||||
|
|
||||||
|
default_profile->setRequestInterceptor(url_interceptor.get());
|
||||||
|
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::LocalContentCanAccessRemoteUrls, true);
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::FullScreenSupportEnabled, true);
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::AllowRunningInsecureContent, true);
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::FocusOnNavigationEnabled, true);
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::AllowWindowActivationFromJavaScript, true);
|
||||||
|
global_settings->setAttribute(QWebEngineSettings::ShowScrollBars, false);
|
||||||
|
|
||||||
|
global_settings->setFontFamily(QWebEngineSettings::StandardFont, QStringLiteral("Roboto"));
|
||||||
|
|
||||||
|
connect(
|
||||||
|
page(), &QWebEnginePage::windowCloseRequested, page(),
|
||||||
|
[this] {
|
||||||
|
if (page()->url() == url_interceptor->GetRequestedURL()) {
|
||||||
|
SetFinished(true);
|
||||||
|
SetExitReason(Service::AM::Applets::WebExitReason::WindowClosed);
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Qt::QueuedConnection);
|
||||||
|
}
|
||||||
|
|
||||||
|
QtNXWebEngineView::~QtNXWebEngineView() {
|
||||||
|
SetFinished(true);
|
||||||
|
StopInputThread();
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::LoadLocalWebPage(std::string_view main_url,
|
||||||
|
std::string_view additional_args) {
|
||||||
|
is_local = true;
|
||||||
|
|
||||||
|
LoadExtractedFonts();
|
||||||
|
SetUserAgent(UserAgent::WebApplet);
|
||||||
|
SetFinished(false);
|
||||||
|
SetExitReason(Service::AM::Applets::WebExitReason::EndButtonPressed);
|
||||||
|
SetLastURL("http://localhost/");
|
||||||
|
StartInputThread();
|
||||||
|
|
||||||
|
load(QUrl(QUrl::fromLocalFile(QString::fromStdString(std::string(main_url))).toString() +
|
||||||
|
QString::fromStdString(std::string(additional_args))));
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::LoadExternalWebPage(std::string_view main_url,
|
||||||
|
std::string_view additional_args) {
|
||||||
|
is_local = false;
|
||||||
|
|
||||||
|
SetUserAgent(UserAgent::WebApplet);
|
||||||
|
SetFinished(false);
|
||||||
|
SetExitReason(Service::AM::Applets::WebExitReason::EndButtonPressed);
|
||||||
|
SetLastURL("http://localhost/");
|
||||||
|
StartInputThread();
|
||||||
|
|
||||||
|
load(QUrl(QString::fromStdString(std::string(main_url)) +
|
||||||
|
QString::fromStdString(std::string(additional_args))));
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::SetUserAgent(UserAgent user_agent) {
|
||||||
|
const QString user_agent_str = [user_agent] {
|
||||||
|
switch (user_agent) {
|
||||||
|
case UserAgent::WebApplet:
|
||||||
|
default:
|
||||||
|
return QStringLiteral("WebApplet");
|
||||||
|
case UserAgent::ShopN:
|
||||||
|
return QStringLiteral("ShopN");
|
||||||
|
case UserAgent::LoginApplet:
|
||||||
|
return QStringLiteral("LoginApplet");
|
||||||
|
case UserAgent::ShareApplet:
|
||||||
|
return QStringLiteral("ShareApplet");
|
||||||
|
case UserAgent::LobbyApplet:
|
||||||
|
return QStringLiteral("LobbyApplet");
|
||||||
|
case UserAgent::WifiWebAuthApplet:
|
||||||
|
return QStringLiteral("WifiWebAuthApplet");
|
||||||
|
}
|
||||||
|
}();
|
||||||
|
|
||||||
|
QWebEngineProfile::defaultProfile()->setHttpUserAgent(
|
||||||
|
QStringLiteral("Mozilla/5.0 (Nintendo Switch; %1) AppleWebKit/606.4 "
|
||||||
|
"(KHTML, like Gecko) NF/6.0.1.15.4 NintendoBrowser/5.1.0.20389")
|
||||||
|
.arg(user_agent_str));
|
||||||
|
}
|
||||||
|
|
||||||
|
bool QtNXWebEngineView::IsFinished() const {
|
||||||
|
return finished;
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::SetFinished(bool finished_) {
|
||||||
|
finished = finished_;
|
||||||
|
}
|
||||||
|
|
||||||
|
Service::AM::Applets::WebExitReason QtNXWebEngineView::GetExitReason() const {
|
||||||
|
return exit_reason;
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::SetExitReason(Service::AM::Applets::WebExitReason exit_reason_) {
|
||||||
|
exit_reason = exit_reason_;
|
||||||
|
}
|
||||||
|
|
||||||
|
const std::string& QtNXWebEngineView::GetLastURL() const {
|
||||||
|
return last_url;
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::SetLastURL(std::string last_url_) {
|
||||||
|
last_url = std::move(last_url_);
|
||||||
|
}
|
||||||
|
|
||||||
|
QString QtNXWebEngineView::GetCurrentURL() const {
|
||||||
|
return url_interceptor->GetRequestedURL().toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::hide() {
|
||||||
|
SetFinished(true);
|
||||||
|
StopInputThread();
|
||||||
|
|
||||||
|
QWidget::hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::keyPressEvent(QKeyEvent* event) {
|
||||||
|
if (is_local) {
|
||||||
|
input_subsystem->GetKeyboard()->PressKey(event->key());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::keyReleaseEvent(QKeyEvent* event) {
|
||||||
|
if (is_local) {
|
||||||
|
input_subsystem->GetKeyboard()->ReleaseKey(event->key());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
template <HIDButton... T>
|
||||||
|
void QtNXWebEngineView::HandleWindowFooterButtonPressedOnce() {
|
||||||
|
const auto f = [this](HIDButton button) {
|
||||||
|
if (input_interpreter->IsButtonPressedOnce(button)) {
|
||||||
|
page()->runJavaScript(
|
||||||
|
QStringLiteral("yuzu_key_callbacks[%1] == null;").arg(static_cast<u8>(button)),
|
||||||
|
[&](const QVariant& variant) {
|
||||||
|
if (variant.toBool()) {
|
||||||
|
switch (button) {
|
||||||
|
case HIDButton::A:
|
||||||
|
SendMultipleKeyPressEvents<Qt::Key_A, Qt::Key_Space, Qt::Key_Return>();
|
||||||
break;
|
break;
|
||||||
case 'B':
|
case HIDButton::B:
|
||||||
yuzu_key_callbacks[1] = func;
|
SendKeyPressEvent(Qt::Key_B);
|
||||||
break;
|
break;
|
||||||
case 'X':
|
case HIDButton::X:
|
||||||
yuzu_key_callbacks[2] = func;
|
SendKeyPressEvent(Qt::Key_X);
|
||||||
break;
|
break;
|
||||||
case 'Y':
|
case HIDButton::Y:
|
||||||
yuzu_key_callbacks[3] = func;
|
SendKeyPressEvent(Qt::Key_Y);
|
||||||
break;
|
break;
|
||||||
case 'L':
|
default:
|
||||||
yuzu_key_callbacks[6] = func;
|
|
||||||
break;
|
|
||||||
case 'R':
|
|
||||||
yuzu_key_callbacks[7] = func;
|
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
page()->runJavaScript(
|
||||||
|
QStringLiteral("if (yuzu_key_callbacks[%1] != null) { yuzu_key_callbacks[%1](); }")
|
||||||
|
.arg(static_cast<u8>(button)));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
var applet_done = false;
|
(f(T), ...);
|
||||||
window.nx.endApplet = function() {
|
}
|
||||||
applet_done = true;
|
|
||||||
|
template <HIDButton... T>
|
||||||
|
void QtNXWebEngineView::HandleWindowKeyButtonPressedOnce() {
|
||||||
|
const auto f = [this](HIDButton button) {
|
||||||
|
if (input_interpreter->IsButtonPressedOnce(button)) {
|
||||||
|
SendKeyPressEvent(HIDButtonToKey(button));
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
window.onkeypress = function(e) { if (e.keyCode === 13) { applet_done = true; } };
|
(f(T), ...);
|
||||||
)";
|
|
||||||
|
|
||||||
QString GetNXShimInjectionScript() {
|
|
||||||
return QString::fromStdString(NX_SHIM_INJECT_SCRIPT);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
NXInputWebEngineView::NXInputWebEngineView(QWidget* parent) : QWebEngineView(parent) {}
|
template <HIDButton... T>
|
||||||
|
void QtNXWebEngineView::HandleWindowKeyButtonHold() {
|
||||||
|
const auto f = [this](HIDButton button) {
|
||||||
|
if (input_interpreter->IsButtonHeld(button)) {
|
||||||
|
SendKeyPressEvent(HIDButtonToKey(button));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
void NXInputWebEngineView::keyPressEvent(QKeyEvent* event) {
|
(f(T), ...);
|
||||||
parent()->event(event);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void NXInputWebEngineView::keyReleaseEvent(QKeyEvent* event) {
|
void QtNXWebEngineView::SendKeyPressEvent(int key) {
|
||||||
parent()->event(event);
|
if (key == 0) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QCoreApplication::postEvent(focusProxy(),
|
||||||
|
new QKeyEvent(QKeyEvent::KeyPress, key, Qt::NoModifier));
|
||||||
|
QCoreApplication::postEvent(focusProxy(),
|
||||||
|
new QKeyEvent(QKeyEvent::KeyRelease, key, Qt::NoModifier));
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::StartInputThread() {
|
||||||
|
if (input_thread_running) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
input_thread_running = true;
|
||||||
|
input_thread = std::thread(&QtNXWebEngineView::InputThread, this);
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::StopInputThread() {
|
||||||
|
if (is_local) {
|
||||||
|
QWidget::releaseKeyboard();
|
||||||
|
}
|
||||||
|
|
||||||
|
input_thread_running = false;
|
||||||
|
if (input_thread.joinable()) {
|
||||||
|
input_thread.join();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::InputThread() {
|
||||||
|
// Wait for 1 second before allowing any inputs to be processed.
|
||||||
|
std::this_thread::sleep_for(std::chrono::seconds(1));
|
||||||
|
|
||||||
|
if (is_local) {
|
||||||
|
QWidget::grabKeyboard();
|
||||||
|
}
|
||||||
|
|
||||||
|
while (input_thread_running) {
|
||||||
|
input_interpreter->PollInput();
|
||||||
|
|
||||||
|
HandleWindowFooterButtonPressedOnce<HIDButton::A, HIDButton::B, HIDButton::X, HIDButton::Y,
|
||||||
|
HIDButton::L, HIDButton::R>();
|
||||||
|
|
||||||
|
HandleWindowKeyButtonPressedOnce<HIDButton::DLeft, HIDButton::DUp, HIDButton::DRight,
|
||||||
|
HIDButton::DDown, HIDButton::LStickLeft,
|
||||||
|
HIDButton::LStickUp, HIDButton::LStickRight,
|
||||||
|
HIDButton::LStickDown>();
|
||||||
|
|
||||||
|
HandleWindowKeyButtonHold<HIDButton::DLeft, HIDButton::DUp, HIDButton::DRight,
|
||||||
|
HIDButton::DDown, HIDButton::LStickLeft, HIDButton::LStickUp,
|
||||||
|
HIDButton::LStickRight, HIDButton::LStickDown>();
|
||||||
|
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(50));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void QtNXWebEngineView::LoadExtractedFonts() {
|
||||||
|
QWebEngineScript nx_font_css;
|
||||||
|
QWebEngineScript load_nx_font;
|
||||||
|
|
||||||
|
const QString fonts_dir = QString::fromStdString(Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/fonts", Common::FS::GetUserPath(Common::FS::UserPath::CacheDir))));
|
||||||
|
|
||||||
|
nx_font_css.setName(QStringLiteral("nx_font_css.js"));
|
||||||
|
load_nx_font.setName(QStringLiteral("load_nx_font.js"));
|
||||||
|
|
||||||
|
nx_font_css.setSourceCode(
|
||||||
|
QString::fromStdString(NX_FONT_CSS)
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontStandard.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontChineseSimplified.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontExtendedChineseSimplified.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontChineseTraditional.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontKorean.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontNintendoExtended.ttf"))
|
||||||
|
.arg(fonts_dir + QStringLiteral("/FontNintendoExtended2.ttf")));
|
||||||
|
load_nx_font.setSourceCode(QString::fromStdString(LOAD_NX_FONT));
|
||||||
|
|
||||||
|
nx_font_css.setInjectionPoint(QWebEngineScript::DocumentReady);
|
||||||
|
load_nx_font.setInjectionPoint(QWebEngineScript::Deferred);
|
||||||
|
|
||||||
|
nx_font_css.setWorldId(QWebEngineScript::MainWorld);
|
||||||
|
load_nx_font.setWorldId(QWebEngineScript::MainWorld);
|
||||||
|
|
||||||
|
nx_font_css.setRunsOnSubFrames(true);
|
||||||
|
load_nx_font.setRunsOnSubFrames(true);
|
||||||
|
|
||||||
|
default_profile->scripts()->insert(nx_font_css);
|
||||||
|
default_profile->scripts()->insert(load_nx_font);
|
||||||
|
|
||||||
|
connect(
|
||||||
|
url_interceptor.get(), &UrlRequestInterceptor::FrameChanged, url_interceptor.get(),
|
||||||
|
[this] {
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(50));
|
||||||
|
page()->runJavaScript(QString::fromStdString(LOAD_NX_FONT));
|
||||||
|
},
|
||||||
|
Qt::QueuedConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
QtWebBrowser::QtWebBrowser(GMainWindow& main_window) {
|
QtWebBrowser::QtWebBrowser(GMainWindow& main_window) {
|
||||||
connect(this, &QtWebBrowser::MainWindowOpenPage, &main_window, &GMainWindow::WebBrowserOpenPage,
|
connect(this, &QtWebBrowser::MainWindowOpenWebPage, &main_window,
|
||||||
Qt::QueuedConnection);
|
&GMainWindow::WebBrowserOpenWebPage, Qt::QueuedConnection);
|
||||||
connect(&main_window, &GMainWindow::WebBrowserUnpackRomFS, this,
|
connect(&main_window, &GMainWindow::WebBrowserExtractOfflineRomFS, this,
|
||||||
&QtWebBrowser::MainWindowUnpackRomFS, Qt::QueuedConnection);
|
&QtWebBrowser::MainWindowExtractOfflineRomFS, Qt::QueuedConnection);
|
||||||
connect(&main_window, &GMainWindow::WebBrowserFinishedBrowsing, this,
|
connect(&main_window, &GMainWindow::WebBrowserClosed, this,
|
||||||
&QtWebBrowser::MainWindowFinishedBrowsing, Qt::QueuedConnection);
|
&QtWebBrowser::MainWindowWebBrowserClosed, Qt::QueuedConnection);
|
||||||
}
|
}
|
||||||
|
|
||||||
QtWebBrowser::~QtWebBrowser() = default;
|
QtWebBrowser::~QtWebBrowser() = default;
|
||||||
|
|
||||||
void QtWebBrowser::OpenPageLocal(std::string_view url, std::function<void()> unpack_romfs_callback_,
|
void QtWebBrowser::OpenLocalWebPage(
|
||||||
std::function<void()> finished_callback_) {
|
std::string_view local_url, std::function<void()> extract_romfs_callback_,
|
||||||
unpack_romfs_callback = std::move(unpack_romfs_callback_);
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback_) const {
|
||||||
finished_callback = std::move(finished_callback_);
|
extract_romfs_callback = std::move(extract_romfs_callback_);
|
||||||
|
callback = std::move(callback_);
|
||||||
|
|
||||||
|
const auto index = local_url.find('?');
|
||||||
|
|
||||||
const auto index = url.find('?');
|
|
||||||
if (index == std::string::npos) {
|
if (index == std::string::npos) {
|
||||||
emit MainWindowOpenPage(url, "");
|
emit MainWindowOpenWebPage(local_url, "", true);
|
||||||
} else {
|
} else {
|
||||||
const auto front = url.substr(0, index);
|
emit MainWindowOpenWebPage(local_url.substr(0, index), local_url.substr(index), true);
|
||||||
const auto back = url.substr(index);
|
|
||||||
emit MainWindowOpenPage(front, back);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QtWebBrowser::MainWindowUnpackRomFS() {
|
void QtWebBrowser::OpenExternalWebPage(
|
||||||
// Acquire the HLE mutex
|
std::string_view external_url,
|
||||||
std::lock_guard lock{HLE::g_hle_lock};
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback_) const {
|
||||||
unpack_romfs_callback();
|
callback = std::move(callback_);
|
||||||
|
|
||||||
|
const auto index = external_url.find('?');
|
||||||
|
|
||||||
|
if (index == std::string::npos) {
|
||||||
|
emit MainWindowOpenWebPage(external_url, "", false);
|
||||||
|
} else {
|
||||||
|
emit MainWindowOpenWebPage(external_url.substr(0, index), external_url.substr(index),
|
||||||
|
false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void QtWebBrowser::MainWindowFinishedBrowsing() {
|
void QtWebBrowser::MainWindowExtractOfflineRomFS() {
|
||||||
// Acquire the HLE mutex
|
extract_romfs_callback();
|
||||||
std::lock_guard lock{HLE::g_hle_lock};
|
}
|
||||||
finished_callback();
|
|
||||||
|
void QtWebBrowser::MainWindowWebBrowserClosed(Service::AM::Applets::WebExitReason exit_reason,
|
||||||
|
std::string last_url) {
|
||||||
|
callback(exit_reason, last_url);
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,10 +1,13 @@
|
||||||
// Copyright 2018 yuzu Emulator Project
|
// Copyright 2020 yuzu Emulator Project
|
||||||
// Licensed under GPLv2 or any later version
|
// Licensed under GPLv2 or any later version
|
||||||
// Refer to the license.txt file included.
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
#pragma once
|
#pragma once
|
||||||
|
|
||||||
#include <functional>
|
#include <atomic>
|
||||||
|
#include <memory>
|
||||||
|
#include <thread>
|
||||||
|
|
||||||
#include <QObject>
|
#include <QObject>
|
||||||
|
|
||||||
#ifdef YUZU_USE_QT_WEB_ENGINE
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
@ -13,19 +16,172 @@
|
||||||
|
|
||||||
#include "core/frontend/applets/web_browser.h"
|
#include "core/frontend/applets/web_browser.h"
|
||||||
|
|
||||||
|
enum class HIDButton : u8;
|
||||||
|
|
||||||
class GMainWindow;
|
class GMainWindow;
|
||||||
|
class InputInterpreter;
|
||||||
|
class UrlRequestInterceptor;
|
||||||
|
|
||||||
|
namespace Core {
|
||||||
|
class System;
|
||||||
|
}
|
||||||
|
|
||||||
|
namespace InputCommon {
|
||||||
|
class InputSubsystem;
|
||||||
|
}
|
||||||
|
|
||||||
#ifdef YUZU_USE_QT_WEB_ENGINE
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
QString GetNXShimInjectionScript();
|
enum class UserAgent {
|
||||||
|
WebApplet,
|
||||||
|
ShopN,
|
||||||
|
LoginApplet,
|
||||||
|
ShareApplet,
|
||||||
|
LobbyApplet,
|
||||||
|
WifiWebAuthApplet,
|
||||||
|
};
|
||||||
|
|
||||||
|
class QWebEngineProfile;
|
||||||
|
class QWebEngineSettings;
|
||||||
|
|
||||||
|
class QtNXWebEngineView : public QWebEngineView {
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
class NXInputWebEngineView : public QWebEngineView {
|
|
||||||
public:
|
public:
|
||||||
explicit NXInputWebEngineView(QWidget* parent = nullptr);
|
explicit QtNXWebEngineView(QWidget* parent, Core::System& system,
|
||||||
|
InputCommon::InputSubsystem* input_subsystem_);
|
||||||
|
~QtNXWebEngineView() override;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads a HTML document that exists locally. Cannot be used to load external websites.
|
||||||
|
*
|
||||||
|
* @param main_url The url to the file.
|
||||||
|
* @param additional_args Additional arguments appended to the main url.
|
||||||
|
*/
|
||||||
|
void LoadLocalWebPage(std::string_view main_url, std::string_view additional_args);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Loads an external website. Cannot be used to load local urls.
|
||||||
|
*
|
||||||
|
* @param main_url The url to the website.
|
||||||
|
* @param additional_args Additional arguments appended to the main url.
|
||||||
|
*/
|
||||||
|
void LoadExternalWebPage(std::string_view main_url, std::string_view additional_args);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the background color of the web page.
|
||||||
|
*
|
||||||
|
* @param color The color to set.
|
||||||
|
*/
|
||||||
|
void SetBackgroundColor(QColor color);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sets the user agent of the web browser.
|
||||||
|
*
|
||||||
|
* @param user_agent The user agent enum.
|
||||||
|
*/
|
||||||
|
void SetUserAgent(UserAgent user_agent);
|
||||||
|
|
||||||
|
[[nodiscard]] bool IsFinished() const;
|
||||||
|
void SetFinished(bool finished_);
|
||||||
|
|
||||||
|
[[nodiscard]] Service::AM::Applets::WebExitReason GetExitReason() const;
|
||||||
|
void SetExitReason(Service::AM::Applets::WebExitReason exit_reason_);
|
||||||
|
|
||||||
|
[[nodiscard]] const std::string& GetLastURL() const;
|
||||||
|
void SetLastURL(std::string last_url_);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This gets the current URL that has been requested by the webpage.
|
||||||
|
* This only applies to the main frame. Sub frames and other resources are ignored.
|
||||||
|
*
|
||||||
|
* @return Currently requested URL
|
||||||
|
*/
|
||||||
|
[[nodiscard]] QString GetCurrentURL() const;
|
||||||
|
|
||||||
|
public slots:
|
||||||
|
void hide();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void keyPressEvent(QKeyEvent* event) override;
|
void keyPressEvent(QKeyEvent* event) override;
|
||||||
void keyReleaseEvent(QKeyEvent* event) override;
|
void keyReleaseEvent(QKeyEvent* event) override;
|
||||||
|
|
||||||
|
private:
|
||||||
|
/**
|
||||||
|
* Handles button presses to execute functions assigned in yuzu_key_callbacks.
|
||||||
|
* yuzu_key_callbacks contains specialized functions for the buttons in the window footer
|
||||||
|
* that can be overriden by games to achieve desired functionality.
|
||||||
|
*
|
||||||
|
* @tparam HIDButton The list of buttons contained in yuzu_key_callbacks
|
||||||
|
*/
|
||||||
|
template <HIDButton... T>
|
||||||
|
void HandleWindowFooterButtonPressedOnce();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles button presses and converts them into keyboard input.
|
||||||
|
* This should only be used to convert D-Pad or Analog Stick input into arrow keys.
|
||||||
|
*
|
||||||
|
* @tparam HIDButton The list of buttons that can be converted into keyboard input.
|
||||||
|
*/
|
||||||
|
template <HIDButton... T>
|
||||||
|
void HandleWindowKeyButtonPressedOnce();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles button holds and converts them into keyboard input.
|
||||||
|
* This should only be used to convert D-Pad or Analog Stick input into arrow keys.
|
||||||
|
*
|
||||||
|
* @tparam HIDButton The list of buttons that can be converted into keyboard input.
|
||||||
|
*/
|
||||||
|
template <HIDButton... T>
|
||||||
|
void HandleWindowKeyButtonHold();
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends a key press event to QWebEngineView.
|
||||||
|
*
|
||||||
|
* @param key Qt key code.
|
||||||
|
*/
|
||||||
|
void SendKeyPressEvent(int key);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Sends multiple key press events to QWebEngineView.
|
||||||
|
*
|
||||||
|
* @tparam int Qt key code.
|
||||||
|
*/
|
||||||
|
template <int... T>
|
||||||
|
void SendMultipleKeyPressEvents() {
|
||||||
|
(SendKeyPressEvent(T), ...);
|
||||||
|
}
|
||||||
|
|
||||||
|
void StartInputThread();
|
||||||
|
void StopInputThread();
|
||||||
|
|
||||||
|
/// The thread where input is being polled and processed.
|
||||||
|
void InputThread();
|
||||||
|
|
||||||
|
/// Loads the extracted fonts using JavaScript.
|
||||||
|
void LoadExtractedFonts();
|
||||||
|
|
||||||
|
InputCommon::InputSubsystem* input_subsystem;
|
||||||
|
|
||||||
|
std::unique_ptr<UrlRequestInterceptor> url_interceptor;
|
||||||
|
|
||||||
|
std::unique_ptr<InputInterpreter> input_interpreter;
|
||||||
|
|
||||||
|
std::thread input_thread;
|
||||||
|
|
||||||
|
std::atomic<bool> input_thread_running{};
|
||||||
|
|
||||||
|
std::atomic<bool> finished{};
|
||||||
|
|
||||||
|
Service::AM::Applets::WebExitReason exit_reason{
|
||||||
|
Service::AM::Applets::WebExitReason::EndButtonPressed};
|
||||||
|
|
||||||
|
std::string last_url{"http://localhost/"};
|
||||||
|
|
||||||
|
bool is_local{};
|
||||||
|
|
||||||
|
QWebEngineProfile* default_profile;
|
||||||
|
QWebEngineSettings* global_settings;
|
||||||
};
|
};
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
@ -34,19 +190,28 @@ class QtWebBrowser final : public QObject, public Core::Frontend::WebBrowserAppl
|
||||||
Q_OBJECT
|
Q_OBJECT
|
||||||
|
|
||||||
public:
|
public:
|
||||||
explicit QtWebBrowser(GMainWindow& main_window);
|
explicit QtWebBrowser(GMainWindow& parent);
|
||||||
~QtWebBrowser() override;
|
~QtWebBrowser() override;
|
||||||
|
|
||||||
void OpenPageLocal(std::string_view url, std::function<void()> unpack_romfs_callback_,
|
void OpenLocalWebPage(std::string_view local_url, std::function<void()> extract_romfs_callback_,
|
||||||
std::function<void()> finished_callback_) override;
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)>
|
||||||
|
callback_) const override;
|
||||||
|
|
||||||
|
void OpenExternalWebPage(std::string_view external_url,
|
||||||
|
std::function<void(Service::AM::Applets::WebExitReason, std::string)>
|
||||||
|
callback_) const override;
|
||||||
|
|
||||||
signals:
|
signals:
|
||||||
void MainWindowOpenPage(std::string_view filename, std::string_view additional_args) const;
|
void MainWindowOpenWebPage(std::string_view main_url, std::string_view additional_args,
|
||||||
|
bool is_local) const;
|
||||||
|
|
||||||
private:
|
private:
|
||||||
void MainWindowUnpackRomFS();
|
void MainWindowExtractOfflineRomFS();
|
||||||
void MainWindowFinishedBrowsing();
|
|
||||||
|
|
||||||
std::function<void()> unpack_romfs_callback;
|
void MainWindowWebBrowserClosed(Service::AM::Applets::WebExitReason exit_reason,
|
||||||
std::function<void()> finished_callback;
|
std::string last_url);
|
||||||
|
|
||||||
|
mutable std::function<void()> extract_romfs_callback;
|
||||||
|
|
||||||
|
mutable std::function<void(Service::AM::Applets::WebExitReason, std::string)> callback;
|
||||||
};
|
};
|
||||||
|
|
|
@ -0,0 +1,193 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
constexpr char NX_FONT_CSS[] = R"(
|
||||||
|
(function() {
|
||||||
|
css = document.createElement('style');
|
||||||
|
css.type = 'text/css';
|
||||||
|
css.id = 'nx_font';
|
||||||
|
css.innerText = `
|
||||||
|
/* FontStandard */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'FontStandard';
|
||||||
|
src: url('%1') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontChineseSimplified */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'FontChineseSimplified';
|
||||||
|
src: url('%2') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontExtendedChineseSimplified */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'FontExtendedChineseSimplified';
|
||||||
|
src: url('%3') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontChineseTraditional */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'FontChineseTraditional';
|
||||||
|
src: url('%4') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontKorean */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'FontKorean';
|
||||||
|
src: url('%5') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontNintendoExtended */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'NintendoExt003';
|
||||||
|
src: url('%6') format('truetype');
|
||||||
|
}
|
||||||
|
|
||||||
|
/* FontNintendoExtended2 */
|
||||||
|
@font-face {
|
||||||
|
font-family: 'NintendoExt003';
|
||||||
|
src: url('%7') format('truetype');
|
||||||
|
}
|
||||||
|
`;
|
||||||
|
|
||||||
|
document.head.appendChild(css);
|
||||||
|
})();
|
||||||
|
)";
|
||||||
|
|
||||||
|
constexpr char LOAD_NX_FONT[] = R"(
|
||||||
|
(function() {
|
||||||
|
var elements = document.querySelectorAll("*");
|
||||||
|
|
||||||
|
for (var i = 0; i < elements.length; i++) {
|
||||||
|
var style = window.getComputedStyle(elements[i], null);
|
||||||
|
if (style.fontFamily.includes("Arial") || style.fontFamily.includes("Calibri") ||
|
||||||
|
style.fontFamily.includes("Century") || style.fontFamily.includes("Times New Roman")) {
|
||||||
|
elements[i].style.fontFamily = "FontStandard, FontChineseSimplified, FontExtendedChineseSimplified, FontChineseTraditional, FontKorean, NintendoExt003";
|
||||||
|
} else {
|
||||||
|
elements[i].style.fontFamily = style.fontFamily + ", FontStandard, FontChineseSimplified, FontExtendedChineseSimplified, FontChineseTraditional, FontKorean, NintendoExt003";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
})();
|
||||||
|
)";
|
||||||
|
|
||||||
|
constexpr char GAMEPAD_SCRIPT[] = R"(
|
||||||
|
window.addEventListener("gamepadconnected", function(e) {
|
||||||
|
console.log("Gamepad connected at index %d: %s. %d buttons, %d axes.",
|
||||||
|
e.gamepad.index, e.gamepad.id, e.gamepad.buttons.length, e.gamepad.axes.length);
|
||||||
|
});
|
||||||
|
|
||||||
|
window.addEventListener("gamepaddisconnected", function(e) {
|
||||||
|
console.log("Gamepad disconnected from index %d: %s", e.gamepad.index, e.gamepad.id);
|
||||||
|
});
|
||||||
|
)";
|
||||||
|
|
||||||
|
constexpr char WINDOW_NX_SCRIPT[] = R"(
|
||||||
|
var end_applet = false;
|
||||||
|
var yuzu_key_callbacks = [];
|
||||||
|
|
||||||
|
(function() {
|
||||||
|
class WindowNX {
|
||||||
|
constructor() {
|
||||||
|
yuzu_key_callbacks[1] = function() { window.history.back(); };
|
||||||
|
yuzu_key_callbacks[2] = function() { window.nx.endApplet(); };
|
||||||
|
}
|
||||||
|
|
||||||
|
addEventListener(type, listener, options) {
|
||||||
|
console.log("nx.addEventListener called, type=%s", type);
|
||||||
|
|
||||||
|
window.addEventListener(type, listener, options);
|
||||||
|
}
|
||||||
|
|
||||||
|
endApplet() {
|
||||||
|
console.log("nx.endApplet called");
|
||||||
|
|
||||||
|
end_applet = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
playSystemSe(system_se) {
|
||||||
|
console.log("nx.playSystemSe is not implemented, system_se=%s", system_se);
|
||||||
|
}
|
||||||
|
|
||||||
|
sendMessage(message) {
|
||||||
|
console.log("nx.sendMessage is not implemented, message=%s", message);
|
||||||
|
}
|
||||||
|
|
||||||
|
setCursorScrollSpeed(scroll_speed) {
|
||||||
|
console.log("nx.setCursorScrollSpeed is not implemented, scroll_speed=%d", scroll_speed);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class WindowNXFooter {
|
||||||
|
setAssign(key, label, func, option) {
|
||||||
|
console.log("nx.footer.setAssign called, key=%s", key);
|
||||||
|
|
||||||
|
switch (key) {
|
||||||
|
case "A":
|
||||||
|
yuzu_key_callbacks[0] = func;
|
||||||
|
break;
|
||||||
|
case "B":
|
||||||
|
yuzu_key_callbacks[1] = func;
|
||||||
|
break;
|
||||||
|
case "X":
|
||||||
|
yuzu_key_callbacks[2] = func;
|
||||||
|
break;
|
||||||
|
case "Y":
|
||||||
|
yuzu_key_callbacks[3] = func;
|
||||||
|
break;
|
||||||
|
case "L":
|
||||||
|
yuzu_key_callbacks[6] = func;
|
||||||
|
break;
|
||||||
|
case "R":
|
||||||
|
yuzu_key_callbacks[7] = func;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setFixed(kind) {
|
||||||
|
console.log("nx.footer.setFixed is not implemented, kind=%s", kind);
|
||||||
|
}
|
||||||
|
|
||||||
|
unsetAssign(key) {
|
||||||
|
console.log("nx.footer.unsetAssign called, key=%s", key);
|
||||||
|
|
||||||
|
switch (key) {
|
||||||
|
case "A":
|
||||||
|
yuzu_key_callbacks[0] = function() {};
|
||||||
|
break;
|
||||||
|
case "B":
|
||||||
|
yuzu_key_callbacks[1] = function() {};
|
||||||
|
break;
|
||||||
|
case "X":
|
||||||
|
yuzu_key_callbacks[2] = function() {};
|
||||||
|
break;
|
||||||
|
case "Y":
|
||||||
|
yuzu_key_callbacks[3] = function() {};
|
||||||
|
break;
|
||||||
|
case "L":
|
||||||
|
yuzu_key_callbacks[6] = function() {};
|
||||||
|
break;
|
||||||
|
case "R":
|
||||||
|
yuzu_key_callbacks[7] = function() {};
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class WindowNXPlayReport {
|
||||||
|
incrementCounter(counter_id) {
|
||||||
|
console.log("nx.playReport.incrementCounter is not implemented, counter_id=%d", counter_id);
|
||||||
|
}
|
||||||
|
|
||||||
|
setCounterSetIdentifier(counter_id) {
|
||||||
|
console.log("nx.playReport.setCounterSetIdentifier is not implemented, counter_id=%d", counter_id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
window.nx = new WindowNX();
|
||||||
|
window.nx.footer = new WindowNXFooter();
|
||||||
|
window.nx.playReport = new WindowNXPlayReport();
|
||||||
|
})();
|
||||||
|
)";
|
|
@ -569,6 +569,10 @@ void GRenderWindow::CaptureScreenshot(u32 res_scale, const QString& screenshot_p
|
||||||
layout);
|
layout);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool GRenderWindow::IsLoadingComplete() const {
|
||||||
|
return first_frame;
|
||||||
|
}
|
||||||
|
|
||||||
void GRenderWindow::OnMinimalClientAreaChangeRequest(std::pair<u32, u32> minimal_size) {
|
void GRenderWindow::OnMinimalClientAreaChangeRequest(std::pair<u32, u32> minimal_size) {
|
||||||
setMinimumSize(minimal_size.first, minimal_size.second);
|
setMinimumSize(minimal_size.first, minimal_size.second);
|
||||||
}
|
}
|
||||||
|
|
|
@ -162,6 +162,8 @@ public:
|
||||||
/// Destroy the previous run's child_widget which should also destroy the child_window
|
/// Destroy the previous run's child_widget which should also destroy the child_window
|
||||||
void ReleaseRenderTarget();
|
void ReleaseRenderTarget();
|
||||||
|
|
||||||
|
bool IsLoadingComplete() const;
|
||||||
|
|
||||||
void CaptureScreenshot(u32 res_scale, const QString& screenshot_path);
|
void CaptureScreenshot(u32 res_scale, const QString& screenshot_path);
|
||||||
|
|
||||||
std::pair<u32, u32> ScaleTouch(const QPointF& pos) const;
|
std::pair<u32, u32> ScaleTouch(const QPointF& pos) const;
|
||||||
|
|
|
@ -28,8 +28,6 @@
|
||||||
#include "core/hle/service/am/applet_ae.h"
|
#include "core/hle/service/am/applet_ae.h"
|
||||||
#include "core/hle/service/am/applet_oe.h"
|
#include "core/hle/service/am/applet_oe.h"
|
||||||
#include "core/hle/service/am/applets/applets.h"
|
#include "core/hle/service/am/applets/applets.h"
|
||||||
#include "core/hle/service/hid/controllers/npad.h"
|
|
||||||
#include "core/hle/service/hid/hid.h"
|
|
||||||
|
|
||||||
// These are wrappers to avoid the calls to CreateDirectory and CreateFile because of the Windows
|
// These are wrappers to avoid the calls to CreateDirectory and CreateFile because of the Windows
|
||||||
// defines.
|
// defines.
|
||||||
|
@ -125,14 +123,6 @@ static FileSys::VirtualFile VfsDirectoryCreateFileWrapper(const FileSys::Virtual
|
||||||
#include "yuzu/discord_impl.h"
|
#include "yuzu/discord_impl.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifdef YUZU_USE_QT_WEB_ENGINE
|
|
||||||
#include <QWebEngineProfile>
|
|
||||||
#include <QWebEngineScript>
|
|
||||||
#include <QWebEngineScriptCollection>
|
|
||||||
#include <QWebEngineSettings>
|
|
||||||
#include <QWebEngineView>
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef QT_STATICPLUGIN
|
#ifdef QT_STATICPLUGIN
|
||||||
Q_IMPORT_PLUGIN(QWindowsIntegrationPlugin);
|
Q_IMPORT_PLUGIN(QWindowsIntegrationPlugin);
|
||||||
#endif
|
#endif
|
||||||
|
@ -190,6 +180,30 @@ static void InitializeLogging() {
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void RemoveCachedContents() {
|
||||||
|
const auto offline_fonts = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/fonts", Common::FS::GetUserPath(Common::FS::UserPath::CacheDir)),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
const auto offline_manual = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/offline_web_applet_manual",
|
||||||
|
Common::FS::GetUserPath(Common::FS::UserPath::CacheDir)),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
const auto offline_legal_information = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/offline_web_applet_legal_information",
|
||||||
|
Common::FS::GetUserPath(Common::FS::UserPath::CacheDir)),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
const auto offline_system_data = Common::FS::SanitizePath(
|
||||||
|
fmt::format("{}/offline_web_applet_system_data",
|
||||||
|
Common::FS::GetUserPath(Common::FS::UserPath::CacheDir)),
|
||||||
|
Common::FS::DirectorySeparator::PlatformDefault);
|
||||||
|
|
||||||
|
Common::FS::DeleteDirRecursively(offline_fonts);
|
||||||
|
Common::FS::DeleteDirRecursively(offline_manual);
|
||||||
|
Common::FS::DeleteDirRecursively(offline_legal_information);
|
||||||
|
Common::FS::DeleteDirRecursively(offline_system_data);
|
||||||
|
}
|
||||||
|
|
||||||
GMainWindow::GMainWindow()
|
GMainWindow::GMainWindow()
|
||||||
: input_subsystem{std::make_shared<InputCommon::InputSubsystem>()},
|
: input_subsystem{std::make_shared<InputCommon::InputSubsystem>()},
|
||||||
config{std::make_unique<Config>()}, vfs{std::make_shared<FileSys::RealVfsFilesystem>()},
|
config{std::make_unique<Config>()}, vfs{std::make_shared<FileSys::RealVfsFilesystem>()},
|
||||||
|
@ -258,6 +272,9 @@ GMainWindow::GMainWindow()
|
||||||
FileSys::ContentProviderUnionSlot::FrontendManual, provider.get());
|
FileSys::ContentProviderUnionSlot::FrontendManual, provider.get());
|
||||||
Core::System::GetInstance().GetFileSystemController().CreateFactories(*vfs);
|
Core::System::GetInstance().GetFileSystemController().CreateFactories(*vfs);
|
||||||
|
|
||||||
|
// Remove cached contents generated during the previous session
|
||||||
|
RemoveCachedContents();
|
||||||
|
|
||||||
// Gen keys if necessary
|
// Gen keys if necessary
|
||||||
OnReinitializeKeys(ReinitializeKeyBehavior::NoWarning);
|
OnReinitializeKeys(ReinitializeKeyBehavior::NoWarning);
|
||||||
|
|
||||||
|
@ -349,151 +366,142 @@ void GMainWindow::SoftwareKeyboardInvokeCheckDialog(std::u16string error_message
|
||||||
emit SoftwareKeyboardFinishedCheckDialog();
|
emit SoftwareKeyboardFinishedCheckDialog();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void GMainWindow::WebBrowserOpenWebPage(std::string_view main_url, std::string_view additional_args,
|
||||||
|
bool is_local) {
|
||||||
#ifdef YUZU_USE_QT_WEB_ENGINE
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
void GMainWindow::WebBrowserOpenPage(std::string_view filename, std::string_view additional_args) {
|
if (disable_web_applet) {
|
||||||
NXInputWebEngineView web_browser_view(this);
|
emit WebBrowserClosed(Service::AM::Applets::WebExitReason::WindowClosed,
|
||||||
|
"http://localhost/");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
QtNXWebEngineView web_browser_view(this, Core::System::GetInstance(), input_subsystem.get());
|
||||||
|
|
||||||
|
ui.action_Pause->setEnabled(false);
|
||||||
|
ui.action_Restart->setEnabled(false);
|
||||||
|
ui.action_Stop->setEnabled(false);
|
||||||
|
|
||||||
// Scope to contain the QProgressDialog for initialization
|
|
||||||
{
|
{
|
||||||
QProgressDialog progress(this);
|
QProgressDialog loading_progress(this);
|
||||||
progress.setMinimumDuration(200);
|
loading_progress.setLabelText(tr("Loading Web Applet..."));
|
||||||
progress.setLabelText(tr("Loading Web Applet..."));
|
loading_progress.setRange(0, 3);
|
||||||
progress.setRange(0, 4);
|
loading_progress.setValue(0);
|
||||||
progress.setValue(0);
|
|
||||||
progress.show();
|
|
||||||
|
|
||||||
auto future = QtConcurrent::run([this] { emit WebBrowserUnpackRomFS(); });
|
if (is_local && !Common::FS::Exists(std::string(main_url))) {
|
||||||
|
loading_progress.show();
|
||||||
|
|
||||||
while (!future.isFinished())
|
auto future = QtConcurrent::run([this] { emit WebBrowserExtractOfflineRomFS(); });
|
||||||
QApplication::processEvents();
|
|
||||||
|
|
||||||
progress.setValue(1);
|
while (!future.isFinished()) {
|
||||||
|
QCoreApplication::processEvents();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Load the special shim script to handle input and exit.
|
loading_progress.setValue(1);
|
||||||
QWebEngineScript nx_shim;
|
|
||||||
nx_shim.setSourceCode(GetNXShimInjectionScript());
|
|
||||||
nx_shim.setWorldId(QWebEngineScript::MainWorld);
|
|
||||||
nx_shim.setName(QStringLiteral("nx_inject.js"));
|
|
||||||
nx_shim.setInjectionPoint(QWebEngineScript::DocumentCreation);
|
|
||||||
nx_shim.setRunsOnSubFrames(true);
|
|
||||||
web_browser_view.page()->profile()->scripts()->insert(nx_shim);
|
|
||||||
|
|
||||||
web_browser_view.load(
|
if (is_local) {
|
||||||
QUrl(QUrl::fromLocalFile(QString::fromStdString(std::string(filename))).toString() +
|
web_browser_view.LoadLocalWebPage(main_url, additional_args);
|
||||||
QString::fromStdString(std::string(additional_args))));
|
} else {
|
||||||
|
web_browser_view.LoadExternalWebPage(main_url, additional_args);
|
||||||
progress.setValue(2);
|
}
|
||||||
|
|
||||||
|
if (render_window->IsLoadingComplete()) {
|
||||||
render_window->hide();
|
render_window->hide();
|
||||||
web_browser_view.setFocus();
|
}
|
||||||
|
|
||||||
const auto& layout = render_window->GetFramebufferLayout();
|
const auto& layout = render_window->GetFramebufferLayout();
|
||||||
web_browser_view.resize(layout.screen.GetWidth(), layout.screen.GetHeight());
|
web_browser_view.resize(layout.screen.GetWidth(), layout.screen.GetHeight());
|
||||||
web_browser_view.move(layout.screen.left, layout.screen.top + menuBar()->height());
|
web_browser_view.move(layout.screen.left, layout.screen.top + menuBar()->height());
|
||||||
web_browser_view.setZoomFactor(static_cast<qreal>(layout.screen.GetWidth()) /
|
web_browser_view.setZoomFactor(static_cast<qreal>(layout.screen.GetWidth()) /
|
||||||
Layout::ScreenUndocked::Width);
|
static_cast<qreal>(Layout::ScreenUndocked::Width));
|
||||||
web_browser_view.settings()->setAttribute(
|
|
||||||
QWebEngineSettings::LocalContentCanAccessRemoteUrls, true);
|
|
||||||
|
|
||||||
|
web_browser_view.setFocus();
|
||||||
web_browser_view.show();
|
web_browser_view.show();
|
||||||
|
|
||||||
progress.setValue(3);
|
loading_progress.setValue(2);
|
||||||
|
|
||||||
QApplication::processEvents();
|
QCoreApplication::processEvents();
|
||||||
|
|
||||||
progress.setValue(4);
|
loading_progress.setValue(3);
|
||||||
}
|
}
|
||||||
|
|
||||||
bool finished = false;
|
bool exit_check = false;
|
||||||
QAction* exit_action = new QAction(tr("Exit Web Applet"), this);
|
|
||||||
connect(exit_action, &QAction::triggered, this, [&finished] { finished = true; });
|
// TODO (Morph): Remove this
|
||||||
|
QAction* exit_action = new QAction(tr("Disable Web Applet"), this);
|
||||||
|
connect(exit_action, &QAction::triggered, this, [this, &web_browser_view] {
|
||||||
|
const auto result = QMessageBox::warning(
|
||||||
|
this, tr("Disable Web Applet"),
|
||||||
|
tr("Disabling the web applet will cause it to not be shown again for the rest of the "
|
||||||
|
"emulated session. This can lead to undefined behavior and should only be used with "
|
||||||
|
"Super Mario 3D All-Stars. Are you sure you want to disable the web applet?"),
|
||||||
|
QMessageBox::Yes | QMessageBox::No);
|
||||||
|
if (result == QMessageBox::Yes) {
|
||||||
|
disable_web_applet = true;
|
||||||
|
web_browser_view.SetFinished(true);
|
||||||
|
}
|
||||||
|
});
|
||||||
ui.menubar->addAction(exit_action);
|
ui.menubar->addAction(exit_action);
|
||||||
|
|
||||||
auto& npad =
|
while (!web_browser_view.IsFinished()) {
|
||||||
Core::System::GetInstance()
|
QCoreApplication::processEvents();
|
||||||
.ServiceManager()
|
|
||||||
.GetService<Service::HID::Hid>("hid")
|
|
||||||
->GetAppletResource()
|
|
||||||
->GetController<Service::HID::Controller_NPad>(Service::HID::HidController::NPad);
|
|
||||||
|
|
||||||
const auto fire_js_keypress = [&web_browser_view](u32 key_code) {
|
if (!exit_check) {
|
||||||
web_browser_view.page()->runJavaScript(
|
web_browser_view.page()->runJavaScript(
|
||||||
QStringLiteral("document.dispatchEvent(new KeyboardEvent('keydown', {'key': %1}));")
|
QStringLiteral("end_applet;"), [&](const QVariant& variant) {
|
||||||
.arg(key_code));
|
exit_check = false;
|
||||||
};
|
if (variant.toBool()) {
|
||||||
|
web_browser_view.SetFinished(true);
|
||||||
QMessageBox::information(
|
web_browser_view.SetExitReason(
|
||||||
this, tr("Exit"),
|
Service::AM::Applets::WebExitReason::EndButtonPressed);
|
||||||
tr("To exit the web application, use the game provided controls to select exit, select the "
|
}
|
||||||
"'Exit Web Applet' option in the menu bar, or press the 'Enter' key."));
|
|
||||||
|
|
||||||
bool running_exit_check = false;
|
|
||||||
while (!finished) {
|
|
||||||
QApplication::processEvents();
|
|
||||||
|
|
||||||
if (!running_exit_check) {
|
|
||||||
web_browser_view.page()->runJavaScript(QStringLiteral("applet_done;"),
|
|
||||||
[&](const QVariant& res) {
|
|
||||||
running_exit_check = false;
|
|
||||||
if (res.toBool())
|
|
||||||
finished = true;
|
|
||||||
});
|
});
|
||||||
running_exit_check = true;
|
|
||||||
|
exit_check = true;
|
||||||
}
|
}
|
||||||
|
|
||||||
const auto input = npad.GetAndResetPressState();
|
if (web_browser_view.GetCurrentURL().contains(QStringLiteral("localhost"))) {
|
||||||
for (std::size_t i = 0; i < Settings::NativeButton::NumButtons; ++i) {
|
if (!web_browser_view.IsFinished()) {
|
||||||
if ((input & (1 << i)) != 0) {
|
web_browser_view.SetFinished(true);
|
||||||
LOG_DEBUG(Frontend, "firing input for button id={:02X}", i);
|
web_browser_view.SetExitReason(Service::AM::Applets::WebExitReason::CallbackURL);
|
||||||
web_browser_view.page()->runJavaScript(
|
|
||||||
QStringLiteral("yuzu_key_callbacks[%1]();").arg(i));
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if (input & 0x00888000) // RStick Down | LStick Down | DPad Down
|
web_browser_view.SetLastURL(web_browser_view.GetCurrentURL().toStdString());
|
||||||
fire_js_keypress(40); // Down Arrow Key
|
|
||||||
else if (input & 0x00444000) // RStick Right | LStick Right | DPad Right
|
|
||||||
fire_js_keypress(39); // Right Arrow Key
|
|
||||||
else if (input & 0x00222000) // RStick Up | LStick Up | DPad Up
|
|
||||||
fire_js_keypress(38); // Up Arrow Key
|
|
||||||
else if (input & 0x00111000) // RStick Left | LStick Left | DPad Left
|
|
||||||
fire_js_keypress(37); // Left Arrow Key
|
|
||||||
else if (input & 0x00000001) // A Button
|
|
||||||
fire_js_keypress(13); // Enter Key
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
std::this_thread::sleep_for(std::chrono::milliseconds(1));
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto exit_reason = web_browser_view.GetExitReason();
|
||||||
|
const auto last_url = web_browser_view.GetLastURL();
|
||||||
|
|
||||||
web_browser_view.hide();
|
web_browser_view.hide();
|
||||||
render_window->show();
|
|
||||||
render_window->setFocus();
|
render_window->setFocus();
|
||||||
|
|
||||||
|
if (render_window->IsLoadingComplete()) {
|
||||||
|
render_window->show();
|
||||||
|
}
|
||||||
|
|
||||||
|
ui.action_Pause->setEnabled(true);
|
||||||
|
ui.action_Restart->setEnabled(true);
|
||||||
|
ui.action_Stop->setEnabled(true);
|
||||||
|
|
||||||
ui.menubar->removeAction(exit_action);
|
ui.menubar->removeAction(exit_action);
|
||||||
|
|
||||||
// Needed to update render window focus/show and remove menubar action
|
QCoreApplication::processEvents();
|
||||||
QApplication::processEvents();
|
|
||||||
emit WebBrowserFinishedBrowsing();
|
emit WebBrowserClosed(exit_reason, last_url);
|
||||||
}
|
|
||||||
|
|
||||||
#else
|
#else
|
||||||
|
|
||||||
void GMainWindow::WebBrowserOpenPage(std::string_view filename, std::string_view additional_args) {
|
// Utilize the same fallback as the default web browser applet.
|
||||||
#ifndef __linux__
|
emit WebBrowserClosed(Service::AM::Applets::WebExitReason::WindowClosed, "http://localhost/");
|
||||||
QMessageBox::warning(
|
|
||||||
this, tr("Web Applet"),
|
|
||||||
tr("This version of yuzu was built without QtWebEngine support, meaning that yuzu cannot "
|
|
||||||
"properly display the game manual or web page requested."),
|
|
||||||
QMessageBox::Ok, QMessageBox::Ok);
|
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
LOG_INFO(Frontend,
|
|
||||||
"(STUBBED) called - Missing QtWebEngine dependency needed to open website page at "
|
|
||||||
"'{}' with arguments '{}'!",
|
|
||||||
filename, additional_args);
|
|
||||||
|
|
||||||
emit WebBrowserFinishedBrowsing();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
void GMainWindow::InitializeWidgets() {
|
void GMainWindow::InitializeWidgets() {
|
||||||
#ifdef YUZU_ENABLE_COMPATIBILITY_REPORTING
|
#ifdef YUZU_ENABLE_COMPATIBILITY_REPORTING
|
||||||
ui.action_Report_Compatibility->setVisible(true);
|
ui.action_Report_Compatibility->setVisible(true);
|
||||||
|
@ -993,7 +1001,6 @@ bool GMainWindow::LoadROM(const QString& filename, std::size_t program_index) {
|
||||||
|
|
||||||
system.SetAppletFrontendSet({
|
system.SetAppletFrontendSet({
|
||||||
std::make_unique<QtControllerSelector>(*this), // Controller Selector
|
std::make_unique<QtControllerSelector>(*this), // Controller Selector
|
||||||
nullptr, // E-Commerce
|
|
||||||
std::make_unique<QtErrorDisplay>(*this), // Error Display
|
std::make_unique<QtErrorDisplay>(*this), // Error Display
|
||||||
nullptr, // Parental Controls
|
nullptr, // Parental Controls
|
||||||
nullptr, // Photo Viewer
|
nullptr, // Photo Viewer
|
||||||
|
@ -2102,6 +2109,7 @@ void GMainWindow::OnStartGame() {
|
||||||
qRegisterMetaType<std::string>("std::string");
|
qRegisterMetaType<std::string>("std::string");
|
||||||
qRegisterMetaType<std::optional<std::u16string>>("std::optional<std::u16string>");
|
qRegisterMetaType<std::optional<std::u16string>>("std::optional<std::u16string>");
|
||||||
qRegisterMetaType<std::string_view>("std::string_view");
|
qRegisterMetaType<std::string_view>("std::string_view");
|
||||||
|
qRegisterMetaType<Service::AM::Applets::WebExitReason>("Service::AM::Applets::WebExitReason");
|
||||||
|
|
||||||
connect(emu_thread.get(), &EmuThread::ErrorThrown, this, &GMainWindow::OnCoreError);
|
connect(emu_thread.get(), &EmuThread::ErrorThrown, this, &GMainWindow::OnCoreError);
|
||||||
|
|
||||||
|
|
|
@ -55,6 +55,10 @@ namespace InputCommon {
|
||||||
class InputSubsystem;
|
class InputSubsystem;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
namespace Service::AM::Applets {
|
||||||
|
enum class WebExitReason : u32;
|
||||||
|
}
|
||||||
|
|
||||||
enum class EmulatedDirectoryTarget {
|
enum class EmulatedDirectoryTarget {
|
||||||
NAND,
|
NAND,
|
||||||
SDMC,
|
SDMC,
|
||||||
|
@ -126,8 +130,8 @@ signals:
|
||||||
void SoftwareKeyboardFinishedText(std::optional<std::u16string> text);
|
void SoftwareKeyboardFinishedText(std::optional<std::u16string> text);
|
||||||
void SoftwareKeyboardFinishedCheckDialog();
|
void SoftwareKeyboardFinishedCheckDialog();
|
||||||
|
|
||||||
void WebBrowserUnpackRomFS();
|
void WebBrowserExtractOfflineRomFS();
|
||||||
void WebBrowserFinishedBrowsing();
|
void WebBrowserClosed(Service::AM::Applets::WebExitReason exit_reason, std::string last_url);
|
||||||
|
|
||||||
public slots:
|
public slots:
|
||||||
void OnLoadComplete();
|
void OnLoadComplete();
|
||||||
|
@ -138,7 +142,8 @@ public slots:
|
||||||
void ProfileSelectorSelectProfile();
|
void ProfileSelectorSelectProfile();
|
||||||
void SoftwareKeyboardGetText(const Core::Frontend::SoftwareKeyboardParameters& parameters);
|
void SoftwareKeyboardGetText(const Core::Frontend::SoftwareKeyboardParameters& parameters);
|
||||||
void SoftwareKeyboardInvokeCheckDialog(std::u16string error_message);
|
void SoftwareKeyboardInvokeCheckDialog(std::u16string error_message);
|
||||||
void WebBrowserOpenPage(std::string_view filename, std::string_view arguments);
|
void WebBrowserOpenWebPage(std::string_view main_url, std::string_view additional_args,
|
||||||
|
bool is_local);
|
||||||
void OnAppFocusStateChanged(Qt::ApplicationState state);
|
void OnAppFocusStateChanged(Qt::ApplicationState state);
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
@ -321,6 +326,9 @@ private:
|
||||||
// Last game booted, used for multi-process apps
|
// Last game booted, used for multi-process apps
|
||||||
QString last_filename_booted;
|
QString last_filename_booted;
|
||||||
|
|
||||||
|
// Disables the web applet for the rest of the emulated session
|
||||||
|
bool disable_web_applet{};
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
void dropEvent(QDropEvent* event) override;
|
void dropEvent(QDropEvent* event) override;
|
||||||
void dragEnterEvent(QDragEnterEvent* event) override;
|
void dragEnterEvent(QDragEnterEvent* event) override;
|
||||||
|
|
|
@ -0,0 +1,32 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
|
#include "yuzu/util/url_request_interceptor.h"
|
||||||
|
|
||||||
|
UrlRequestInterceptor::UrlRequestInterceptor(QObject* p) : QWebEngineUrlRequestInterceptor(p) {}
|
||||||
|
|
||||||
|
UrlRequestInterceptor::~UrlRequestInterceptor() = default;
|
||||||
|
|
||||||
|
void UrlRequestInterceptor::interceptRequest(QWebEngineUrlRequestInfo& info) {
|
||||||
|
const auto resource_type = info.resourceType();
|
||||||
|
|
||||||
|
switch (resource_type) {
|
||||||
|
case QWebEngineUrlRequestInfo::ResourceTypeMainFrame:
|
||||||
|
requested_url = info.requestUrl();
|
||||||
|
emit FrameChanged();
|
||||||
|
break;
|
||||||
|
case QWebEngineUrlRequestInfo::ResourceTypeSubFrame:
|
||||||
|
case QWebEngineUrlRequestInfo::ResourceTypeXhr:
|
||||||
|
emit FrameChanged();
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
QUrl UrlRequestInterceptor::GetRequestedURL() const {
|
||||||
|
return requested_url;
|
||||||
|
}
|
||||||
|
|
||||||
|
#endif
|
|
@ -0,0 +1,30 @@
|
||||||
|
// Copyright 2020 yuzu Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#ifdef YUZU_USE_QT_WEB_ENGINE
|
||||||
|
|
||||||
|
#include <QObject>
|
||||||
|
#include <QWebEngineUrlRequestInterceptor>
|
||||||
|
|
||||||
|
class UrlRequestInterceptor : public QWebEngineUrlRequestInterceptor {
|
||||||
|
Q_OBJECT
|
||||||
|
|
||||||
|
public:
|
||||||
|
explicit UrlRequestInterceptor(QObject* p = nullptr);
|
||||||
|
~UrlRequestInterceptor() override;
|
||||||
|
|
||||||
|
void interceptRequest(QWebEngineUrlRequestInfo& info) override;
|
||||||
|
|
||||||
|
QUrl GetRequestedURL() const;
|
||||||
|
|
||||||
|
signals:
|
||||||
|
void FrameChanged();
|
||||||
|
|
||||||
|
private:
|
||||||
|
QUrl requested_url;
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
Reference in New Issue