frontend/applets: frontend swkbd base
Original commits by @jroweboy: * Rebase out the other commit * changing branches * More work on stuff and things ecks DEE Changes by @zhaowenlan1779: * Removed #include of result.h
This commit is contained in:
parent
f9a89ff410
commit
caacefcc2e
7 changed files with 487 additions and 18 deletions
|
@ -68,6 +68,10 @@ add_library(core STATIC
|
||||||
file_sys/savedata_archive.h
|
file_sys/savedata_archive.h
|
||||||
file_sys/title_metadata.cpp
|
file_sys/title_metadata.cpp
|
||||||
file_sys/title_metadata.h
|
file_sys/title_metadata.h
|
||||||
|
frontend/applet/interface.cpp
|
||||||
|
frontend/applet/interface.h
|
||||||
|
frontend/applet/swkbd.cpp
|
||||||
|
frontend/applet/swkbd.h
|
||||||
frontend/camera/blank_camera.cpp
|
frontend/camera/blank_camera.cpp
|
||||||
frontend/camera/blank_camera.h
|
frontend/camera/blank_camera.h
|
||||||
frontend/camera/factory.cpp
|
frontend/camera/factory.cpp
|
||||||
|
|
20
src/core/frontend/applet/interface.cpp
Normal file
20
src/core/frontend/applet/interface.cpp
Normal file
|
@ -0,0 +1,20 @@
|
||||||
|
// Copyright 2018 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
|
#include "core/frontend/interface.h"
|
||||||
|
|
||||||
|
namespace Frontend {
|
||||||
|
|
||||||
|
std::unordered_map<AppletType, std::shared_ptr<AppletInterface>> registered_applets;
|
||||||
|
|
||||||
|
void RegisterFrontendApplet(std::shared_ptr<AppletInterface> applet, AppletType type) {
|
||||||
|
registered_applets[type] = applet;
|
||||||
|
}
|
||||||
|
|
||||||
|
void UnregisterFrontendApplet(AppletType type) {
|
||||||
|
registered_applets.erase(type);
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Frontend
|
66
src/core/frontend/applet/interface.h
Normal file
66
src/core/frontend/applet/interface.h
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
// Copyright 2018 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <atomic>
|
||||||
|
#include <memory>
|
||||||
|
|
||||||
|
namespace Frontend {
|
||||||
|
|
||||||
|
enum class AppletType {
|
||||||
|
SoftwareKeyboard,
|
||||||
|
};
|
||||||
|
|
||||||
|
class AppletConfig {};
|
||||||
|
class AppletData {};
|
||||||
|
|
||||||
|
class AppletInterface {
|
||||||
|
public:
|
||||||
|
virtual ~AppletInterface() = default;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* On applet start, the applet specific configuration will be passed in along with the
|
||||||
|
* framebuffer.
|
||||||
|
*/
|
||||||
|
// virtual void Setup(const Config* /*, framebuffer */) = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Called on a fixed schedule to have the applet update any state such as the framebuffer.
|
||||||
|
*/
|
||||||
|
virtual void Update() = 0;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Checked every update to see if the applet is still running. When the applet is done, the core
|
||||||
|
* will call ReceiveData
|
||||||
|
*/
|
||||||
|
virtual bool IsRunning() {
|
||||||
|
return running;
|
||||||
|
}
|
||||||
|
|
||||||
|
private:
|
||||||
|
// framebuffer;
|
||||||
|
std::atomic_bool running = false;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Frontends call this method to pass a frontend applet implementation to the core. If the core
|
||||||
|
* already has a applet registered, then this replaces the old applet
|
||||||
|
*
|
||||||
|
* @param applet - Frontend Applet implementation that the HLE applet code will launch
|
||||||
|
* @param type - Which type of applet
|
||||||
|
*/
|
||||||
|
void RegisterFrontendApplet(std::shared_ptr<AppletInterface> applet, AppletType type);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Frontends call this to prevent future requests
|
||||||
|
*/
|
||||||
|
void UnregisterFrontendApplet(AppletType type);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Returns the Frontend Applet for the provided type
|
||||||
|
*/
|
||||||
|
std::shared_ptr<AppletInterface> GetRegisteredApplet(AppletType type);
|
||||||
|
|
||||||
|
} // namespace Frontend
|
127
src/core/frontend/applet/swkbd.cpp
Normal file
127
src/core/frontend/applet/swkbd.cpp
Normal file
|
@ -0,0 +1,127 @@
|
||||||
|
// Copyright 2018 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#include "core/frontend/applet/swkbd.h"
|
||||||
|
|
||||||
|
namespace Frontend {
|
||||||
|
|
||||||
|
ValidationError SoftwareKeyboard::ValidateFilters(const std::string& input) {
|
||||||
|
if (config.filters.prevent_digit) {
|
||||||
|
if (std::any_of(input.begin(), input.end(), std::isdigit)) {
|
||||||
|
return ValidationError::DigitNotAllowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (config.filters.prevent_at) {
|
||||||
|
if (input.find('@') != std::string::npos) {
|
||||||
|
return ValidationError::AtSignNotAllowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (config.filters.prevent_percent) {
|
||||||
|
if (input.find('%') != std::string::npos) {
|
||||||
|
return ValidationError::PercentNotAllowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (config.filter.prevent_backslash) {
|
||||||
|
if (input.find('\\') != std::string::npos) {
|
||||||
|
return ValidationError::BackslashNotAllowed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (config.filters.prevent_profanity) {
|
||||||
|
// TODO: check the profanity filter
|
||||||
|
LOG_INFO(Frontend, "App requested swkbd profanity filter, but its not implemented.");
|
||||||
|
}
|
||||||
|
if (config.filters.enable_callback) {
|
||||||
|
// TODO: check the callback
|
||||||
|
LOG_INFO(Frontend, "App requested a swkbd callback, but its not implemented.");
|
||||||
|
}
|
||||||
|
return valid;
|
||||||
|
}
|
||||||
|
|
||||||
|
ValidationError SoftwareKeyboard::ValidateInput(const std::string& input) {
|
||||||
|
ValidationError error;
|
||||||
|
if ((error = ValidateFilters(input)) != ValidationError::None) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO(jroweboy): Is max_text_length inclusive or exclusive?
|
||||||
|
if (input.size() > config.max_text_length) {
|
||||||
|
return ValidationError::MaxLengthExceeded;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto is_blank = [&] { return std::all_of(input.begin(), input.end(), std::isspace); };
|
||||||
|
auto is_empty = [&] { return input.empty(); };
|
||||||
|
switch (config.valid_input) {
|
||||||
|
case AcceptedInput::FixedLength:
|
||||||
|
if (input.size() != config.max_text_length) {
|
||||||
|
return ValidationError::FixedLengthRequired;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case AcceptedInput::NotEmptyAndNotBlank:
|
||||||
|
if (is_blank()) {
|
||||||
|
return ValidationError::BlankInputNotAllowed;
|
||||||
|
}
|
||||||
|
if (is_empty()) {
|
||||||
|
return ValidationError::EmptyInputNotAllowed;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case AcceptedInput::NotBlank:
|
||||||
|
if (is_blank()) {
|
||||||
|
return ValidationError::BlankInputNotAllowed;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case AcceptedInput::NotEmpty:
|
||||||
|
if (is_empty()) {
|
||||||
|
return ValidationError::EmptyInputNotAllowed;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case AcceptedInput::Anything:
|
||||||
|
return ValidationError::None;
|
||||||
|
default:
|
||||||
|
// TODO(jroweboy): What does hardware do in this case?
|
||||||
|
NGLOG_CRITICAL(Frontend, "Application requested unknown validation method. Method: {}",
|
||||||
|
static_cast<u32>(config.valid_input));
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
|
||||||
|
return ValidationError::None;
|
||||||
|
} // namespace Frontend
|
||||||
|
|
||||||
|
ValidationError SoftwareKeyboard::ValidateButton(u8 button) {
|
||||||
|
switch (config.button_config) {
|
||||||
|
case ButtonConfig::None:
|
||||||
|
return ValidationError::None;
|
||||||
|
case ButtonConfig::Single:
|
||||||
|
if (button != 0) {
|
||||||
|
return ValidationError::ButtonOutOfRange;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ButtonConfig::Dual:
|
||||||
|
if (button > 1) {
|
||||||
|
return ValidationError::ButtonOutOfRange;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
case ButtonConfig::Triple:
|
||||||
|
if (button > 2) {
|
||||||
|
return ValidationError::ButtonOutOfRange;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
UNREACHABLE();
|
||||||
|
}
|
||||||
|
return ValidationError::None;
|
||||||
|
}
|
||||||
|
|
||||||
|
ValidationError Finalize(cosnt std::string& text, u8 button) {
|
||||||
|
ValidationError error;
|
||||||
|
if ((error = ValidateInput(text)) != ValidationError::NONE) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
if ((error = ValidateButton(button)) != ValidationError::NONE) {
|
||||||
|
return error;
|
||||||
|
}
|
||||||
|
data = {text, button};
|
||||||
|
running = false;
|
||||||
|
}
|
||||||
|
|
||||||
|
} // namespace Frontend
|
123
src/core/frontend/applet/swkbd.h
Normal file
123
src/core/frontend/applet/swkbd.h
Normal file
|
@ -0,0 +1,123 @@
|
||||||
|
// Copyright 2018 Citra Emulator Project
|
||||||
|
// Licensed under GPLv2 or any later version
|
||||||
|
// Refer to the license.txt file included.
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <unordered_map>
|
||||||
|
#include "core/frontend/applet/interface.h"
|
||||||
|
|
||||||
|
namespace Frontend {
|
||||||
|
|
||||||
|
enum class AcceptedInput {
|
||||||
|
Anything = 0, /// All inputs are accepted.
|
||||||
|
NotEmpty, /// Empty inputs are not accepted.
|
||||||
|
NotEmptyAndNotBlank, /// Empty or blank inputs (consisting solely of whitespace) are not
|
||||||
|
/// accepted.
|
||||||
|
NotBlank, /// Blank inputs (consisting solely of whitespace) are not accepted, but empty
|
||||||
|
/// inputs are.
|
||||||
|
FixedLength, /// The input must have a fixed length (specified by maxTextLength in
|
||||||
|
/// swkbdInit).
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class ButtonConfig {
|
||||||
|
Single = 0, /// Ok button
|
||||||
|
Dual, /// Cancel | Ok buttons
|
||||||
|
Triple, /// Cancel | I Forgot | Ok buttons
|
||||||
|
None, /// No button (returned by swkbdInputText in special cases)
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Default English button text mappings. Frontends may need to copy this to internationalize it.
|
||||||
|
static const char* BUTTON_OKAY = "Ok";
|
||||||
|
static const char* BUTTON_CANCEL = "Cancel";
|
||||||
|
static const char* BUTTON_FORGOT = "I Forgot";
|
||||||
|
static const std::unordered_map<ButtonConfig, std::vector<std::string>> DEFAULT_BUTTON_MAPPING = {
|
||||||
|
{ButtonConfig::Single, {BUTTON_OKAY}},
|
||||||
|
{ButtonConfig::Dual, {BUTTON_CANCEL, BUTTON_OKAY}},
|
||||||
|
{ButtonConfig::Triple, {BUTTON_CANCEL, BUTTON_FORGOT, BUTTON_OKAY}},
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Configuration thats relevent to frontend implementation of applets. Anything missing that we
|
||||||
|
/// later learn is needed can be added here and filled in by the backed HLE applet
|
||||||
|
struct KeyboardConfig {
|
||||||
|
ButtonConfig button_config;
|
||||||
|
AcceptedInput accept_mode; /// What kinds of input are accepted (blank/empty/fixed width)
|
||||||
|
bool multiline_mode; /// True if the keyboard accepts multiple lines of input
|
||||||
|
u16 max_text_length; /// Maximum number of letters allowed if its a text input
|
||||||
|
u16 max_digits; /// Maximum number of numbers allowed if its a number input
|
||||||
|
std::string hint_text; /// Displayed in the field as a hint before
|
||||||
|
bool has_custom_button_text; /// If true, use the button_text instead
|
||||||
|
std::vector<std::string> button_text; /// Contains the button text that the caller provides
|
||||||
|
struct Filters {
|
||||||
|
bool prevent_digit; /// Disallow the use of more than a certain number of digits (TODO how
|
||||||
|
/// many is a certain number)
|
||||||
|
bool prevent_at; /// Disallow the use of the @ sign.
|
||||||
|
bool prevent_percent; /// Disallow the use of the % sign.
|
||||||
|
bool prevent_backslash; /// Disallow the use of the \ sign.
|
||||||
|
bool prevent_profanity; /// Disallow profanity using Nintendo's profanity filter.
|
||||||
|
bool enable_callback; /// Use a callback in order to check the input.
|
||||||
|
} filters;
|
||||||
|
};
|
||||||
|
|
||||||
|
struct KeyboardData {
|
||||||
|
std::string text;
|
||||||
|
u8 button;
|
||||||
|
};
|
||||||
|
|
||||||
|
enum class ValidationError {
|
||||||
|
None,
|
||||||
|
// Button Selection
|
||||||
|
ButtonOutOfRange,
|
||||||
|
// Configured Filters
|
||||||
|
DigitNotAllowed,
|
||||||
|
AtSignNotAllowed,
|
||||||
|
PercentNotAllowed,
|
||||||
|
BackslashNotAllowed,
|
||||||
|
ProfanityNotAllowed,
|
||||||
|
CallbackFailed,
|
||||||
|
// Allowed Input Type
|
||||||
|
FixedLengthRequired,
|
||||||
|
BlankInputNotAllowed,
|
||||||
|
EmptyInputNotAllowed,
|
||||||
|
};
|
||||||
|
|
||||||
|
class SoftwareKeyboard : public AppletInterface {
|
||||||
|
public:
|
||||||
|
explict SoftwareKeyboard(KeyboardConfig config) : AppletInterface(), config(config) {}
|
||||||
|
|
||||||
|
protected:
|
||||||
|
/**
|
||||||
|
* Validates if the provided string breaks any of the filter rules. This is meant to be called
|
||||||
|
* whenever the user input changes to check to see if the new input is valid. Frontends can
|
||||||
|
* decide if they want to check the input continuously or once before submission
|
||||||
|
*/
|
||||||
|
ValidationError ValidateFilters(const std::string& input);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Validates the the provided string doesn't break any extra rules like "input must not be
|
||||||
|
* empty". This will be called by Finalize but can be called earlier if the frontend needs
|
||||||
|
*/
|
||||||
|
ValidationError ValidateInput(const std::string& input);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Verifies that the selected button is valid. This should be used as the last check before
|
||||||
|
* closing.
|
||||||
|
*/
|
||||||
|
ValidationError ValidateButton(u8 button);
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Runs all validation phases. If successful, stores the data so that the HLE applet in core can
|
||||||
|
* send this to the calling application
|
||||||
|
*/
|
||||||
|
ValidationError Finialize(const std::string&, u8 button);
|
||||||
|
|
||||||
|
private:
|
||||||
|
KeyboardData ReceiveData() override {
|
||||||
|
return data;
|
||||||
|
}
|
||||||
|
|
||||||
|
KeyboardConfig config;
|
||||||
|
KeyboardData data;
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace Frontend
|
|
@ -6,6 +6,7 @@
|
||||||
|
|
||||||
#include "common/common_funcs.h"
|
#include "common/common_funcs.h"
|
||||||
#include "common/common_types.h"
|
#include "common/common_types.h"
|
||||||
|
#include "core/frontend/applet/swkbd.h"
|
||||||
#include "core/hle/applets/applet.h"
|
#include "core/hle/applets/applet.h"
|
||||||
#include "core/hle/kernel/kernel.h"
|
#include "core/hle/kernel/kernel.h"
|
||||||
#include "core/hle/kernel/shared_memory.h"
|
#include "core/hle/kernel/shared_memory.h"
|
||||||
|
@ -15,33 +16,156 @@
|
||||||
namespace HLE {
|
namespace HLE {
|
||||||
namespace Applets {
|
namespace Applets {
|
||||||
|
|
||||||
|
/// Maximum number of buttons that can be in the keyboard.
|
||||||
|
constexpr int MAX_BUTTON = 3;
|
||||||
|
/// Maximum button text length, in UTF-16 code units.
|
||||||
|
constexpr int MAX_BUTTON_TEXT_LEN = 16;
|
||||||
|
/// Maximum hint text length, in UTF-16 code units.
|
||||||
|
constexpr int MAX_HINT_TEXT_LEN = 64;
|
||||||
|
/// Maximum filter callback error message length, in UTF-16 code units.
|
||||||
|
constexpr int MAX_CALLBACK_MSG_LEN = 256;
|
||||||
|
|
||||||
|
/// Keyboard types
|
||||||
|
enum class SoftwareKeyboardType : u32 {
|
||||||
|
NORMAL = 0, ///< Normal keyboard with several pages (QWERTY/accents/symbol/mobile)
|
||||||
|
QWERTY, ///< QWERTY keyboard only.
|
||||||
|
NUMPAD, ///< Number pad.
|
||||||
|
WESTERN, ///< On JPN systems, a text keyboard without Japanese input capabilities,
|
||||||
|
/// otherwise same as SWKBD_TYPE_NORMAL.
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Keyboard dialog buttons.
|
||||||
|
enum class SoftwareKeyboardButtonConfig : u32 {
|
||||||
|
SINGLE_BUTTON = 0, ///< Ok button
|
||||||
|
DUAL_BUTTON, ///< Cancel | Ok buttons
|
||||||
|
TRIPLE_BUTTON, ///< Cancel | I Forgot | Ok buttons
|
||||||
|
NO_BUTTON, ///< No button (returned by swkbdInputText in special cases)
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Accepted input types.
|
||||||
|
enum class SoftwareKeyboardValidInput : u32 {
|
||||||
|
ANYTHING = 0, ///< All inputs are accepted.
|
||||||
|
NOTEMPTY, ///< Empty inputs are not accepted.
|
||||||
|
NOTEMPTY_NOTBLANK, ///< Empty or blank inputs (consisting solely of whitespace) are not
|
||||||
|
/// accepted.
|
||||||
|
NOTBLANK, ///< Blank inputs (consisting solely of whitespace) are not accepted, but empty
|
||||||
|
/// inputs are.
|
||||||
|
FIXEDLEN, ///< The input must have a fixed length (specified by maxTextLength in
|
||||||
|
/// swkbdInit).
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Keyboard password modes.
|
||||||
|
enum class SoftwareKeyboardPasswordMode : u32 {
|
||||||
|
NONE = 0, ///< Characters are not concealed.
|
||||||
|
HIDE, ///< Characters are concealed immediately.
|
||||||
|
HIDE_DELAY, ///< Characters are concealed a second after they've been typed.
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Keyboard input filtering flags. Allows the caller to specify what input is explicitly not
|
||||||
|
/// allowed
|
||||||
|
namespace SoftwareKeyboardFilter {
|
||||||
|
enum Filter {
|
||||||
|
DIGITS = 1, ///< Disallow the use of more than a certain number of digits (0 or more)
|
||||||
|
AT = 1 << 1, ///< Disallow the use of the @ sign.
|
||||||
|
PERCENT = 1 << 2, ///< Disallow the use of the % sign.
|
||||||
|
BACKSLASH = 1 << 3, ///< Disallow the use of the \ sign.
|
||||||
|
PROFANITY = 1 << 4, ///< Disallow profanity using Nintendo's profanity filter.
|
||||||
|
CALLBACK = 1 << 5, ///< Use a callback in order to check the input.
|
||||||
|
};
|
||||||
|
} // namespace SoftwareKeyboardFilter
|
||||||
|
|
||||||
|
/// Keyboard features.
|
||||||
|
namespace SoftwareKeyboardFeature {
|
||||||
|
enum Feature {
|
||||||
|
PARENTAL = 1, ///< Parental PIN mode.
|
||||||
|
DARKEN_TOP_SCREEN = 1 << 1, ///< Darken the top screen when the keyboard is shown.
|
||||||
|
PREDICTIVE_INPUT =
|
||||||
|
1 << 2, ///< Enable predictive input (necessary for Kanji input in JPN systems).
|
||||||
|
MULTILINE = 1 << 3, ///< Enable multiline input.
|
||||||
|
FIXED_WIDTH = 1 << 4, ///< Enable fixed-width mode.
|
||||||
|
ALLOW_HOME = 1 << 5, ///< Allow the usage of the HOME button.
|
||||||
|
ALLOW_RESET = 1 << 6, ///< Allow the usage of a software-reset combination.
|
||||||
|
ALLOW_POWER = 1 << 7, ///< Allow the usage of the POWER button.
|
||||||
|
DEFAULT_QWERTY = 1 << 9, ///< Default to the QWERTY page when the keyboard is shown.
|
||||||
|
};
|
||||||
|
} // namespace SoftwareKeyboardFeature
|
||||||
|
|
||||||
|
/// Keyboard filter callback return values.
|
||||||
|
enum class SoftwareKeyboardCallbackResult : u32 {
|
||||||
|
OK = 0, ///< Specifies that the input is valid.
|
||||||
|
CLOSE, ///< Displays an error message, then closes the keyboard.
|
||||||
|
CONTINUE, ///< Displays an error message and continues displaying the keyboard.
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Keyboard return values.
|
||||||
|
enum class SoftwareKeyboardResult : s32 {
|
||||||
|
NONE = -1, ///< Dummy/unused.
|
||||||
|
INVALID_INPUT = -2, ///< Invalid parameters to swkbd.
|
||||||
|
OUTOFMEM = -3, ///< Out of memory.
|
||||||
|
|
||||||
|
D0_CLICK = 0, ///< The button was clicked in 1-button dialogs.
|
||||||
|
D1_CLICK0, ///< The left button was clicked in 2-button dialogs.
|
||||||
|
D1_CLICK1, ///< The right button was clicked in 2-button dialogs.
|
||||||
|
D2_CLICK0, ///< The left button was clicked in 3-button dialogs.
|
||||||
|
D2_CLICK1, ///< The middle button was clicked in 3-button dialogs.
|
||||||
|
D2_CLICK2, ///< The right button was clicked in 3-button dialogs.
|
||||||
|
|
||||||
|
HOMEPRESSED = 10, ///< The HOME button was pressed.
|
||||||
|
RESETPRESSED, ///< The soft-reset key combination was pressed.
|
||||||
|
POWERPRESSED, ///< The POWER button was pressed.
|
||||||
|
|
||||||
|
PARENTAL_OK = 20, ///< The parental PIN was verified successfully.
|
||||||
|
PARENTAL_FAIL, ///< The parental PIN was incorrect.
|
||||||
|
|
||||||
|
BANNED_INPUT = 30, ///< The filter callback returned SoftwareKeyboardCallback::CLOSE.
|
||||||
|
};
|
||||||
|
|
||||||
struct SoftwareKeyboardConfig {
|
struct SoftwareKeyboardConfig {
|
||||||
INSERT_PADDING_WORDS(0x8);
|
SoftwareKeyboardType type;
|
||||||
|
SoftwareKeyboardButtonConfig num_buttons_m1;
|
||||||
u16 max_text_length; ///< Maximum length of the input text
|
SoftwareKeyboardValidInput valid_input;
|
||||||
|
SoftwareKeyboardPasswordMode password_mode;
|
||||||
INSERT_PADDING_BYTES(0x6E);
|
s32 is_parental_screen;
|
||||||
|
s32 darken_top_screen;
|
||||||
char16_t display_text[65]; ///< Text to display when asking the user for input
|
u32 filter_flags;
|
||||||
|
u32 save_state_flags;
|
||||||
INSERT_PADDING_BYTES(0xE);
|
u16 max_text_length;
|
||||||
|
u16 dict_word_count;
|
||||||
u32 default_text_offset; ///< Offset of the default text in the output SharedMemory
|
u16 max_digits;
|
||||||
|
std::array<std::array<u16, MAX_BUTTON_TEXT_LEN + 1>, MAX_BUTTON> button_text;
|
||||||
INSERT_PADDING_WORDS(0x3);
|
std::array<u16, 2> numpad_keys;
|
||||||
|
std::array<u16, MAX_HINT_TEXT_LEN + 1>
|
||||||
|
hint_text; ///< Text to display when asking the user for input
|
||||||
|
bool predictive_input;
|
||||||
|
bool multiline;
|
||||||
|
bool fixed_width;
|
||||||
|
bool allow_home;
|
||||||
|
bool allow_reset;
|
||||||
|
bool allow_power;
|
||||||
|
bool unknown;
|
||||||
|
bool default_qwerty;
|
||||||
|
std::array<bool, 4> button_submits_text;
|
||||||
|
u16 language;
|
||||||
|
|
||||||
|
u32 initial_text_offset; ///< Offset of the default text in the output SharedMemory
|
||||||
|
u32 dict_offset;
|
||||||
|
u32 initial_status_offset;
|
||||||
|
u32 initial_learning_offset;
|
||||||
u32 shared_memory_size; ///< Size of the SharedMemory
|
u32 shared_memory_size; ///< Size of the SharedMemory
|
||||||
|
u32 version;
|
||||||
|
|
||||||
INSERT_PADDING_WORDS(0x1);
|
SoftwareKeyboardResult return_code;
|
||||||
|
|
||||||
u32 return_code; ///< Return code of the SoftwareKeyboard, usually 2, other values are unknown
|
u32 status_offset;
|
||||||
|
u32 learning_offset;
|
||||||
INSERT_PADDING_WORDS(0x2);
|
|
||||||
|
|
||||||
u32 text_offset; ///< Offset in the SharedMemory where the output text starts
|
u32 text_offset; ///< Offset in the SharedMemory where the output text starts
|
||||||
u16 text_length; ///< Length in characters of the output text
|
u16 text_length; ///< Length in characters of the output text
|
||||||
|
|
||||||
INSERT_PADDING_BYTES(0x2B6);
|
int callback_result;
|
||||||
|
std::array<u16, MAX_CALLBACK_MSG_LEN + 1> callback_msg;
|
||||||
|
bool skip_at_check;
|
||||||
|
INSERT_PADDING_BYTES(0xAB);
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -50,6 +174,8 @@ struct SoftwareKeyboardConfig {
|
||||||
*/
|
*/
|
||||||
static_assert(sizeof(SoftwareKeyboardConfig) == 0x400, "Software Keyboard Config size is wrong");
|
static_assert(sizeof(SoftwareKeyboardConfig) == 0x400, "Software Keyboard Config size is wrong");
|
||||||
|
|
||||||
|
class DefaultCitraKeyboard : Frontend::AppletInterface {};
|
||||||
|
|
||||||
class SoftwareKeyboard final : public Applet {
|
class SoftwareKeyboard final : public Applet {
|
||||||
public:
|
public:
|
||||||
SoftwareKeyboard(Service::APT::AppletId id, std::weak_ptr<Service::APT::AppletManager> manager)
|
SoftwareKeyboard(Service::APT::AppletId id, std::weak_ptr<Service::APT::AppletManager> manager)
|
||||||
|
|
|
@ -96,6 +96,9 @@ struct Values {
|
||||||
std::string motion_device;
|
std::string motion_device;
|
||||||
std::string touch_device;
|
std::string touch_device;
|
||||||
|
|
||||||
|
// Frontend Devices
|
||||||
|
std::string applet_swkbd;
|
||||||
|
|
||||||
// Core
|
// Core
|
||||||
bool use_cpu_jit;
|
bool use_cpu_jit;
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue