// Copyright 2018 yuzu emulator team // Licensed under GPLv2 or any later version // Refer to the license.txt file included. #include #include "common/assert.h" #include "common/string_util.h" #include "core/core.h" #include "core/frontend/applets/software_keyboard.h" #include "core/hle/service/am/am.h" #include "core/hle/service/am/applets/software_keyboard.h" namespace Service::AM::Applets { constexpr std::size_t SWKBD_OUTPUT_BUFFER_SIZE = 0x7D8; constexpr std::size_t SWKBD_OUTPUT_INTERACTIVE_BUFFER_SIZE = 0x7D4; constexpr std::size_t DEFAULT_MAX_LENGTH = 500; constexpr bool INTERACTIVE_STATUS_OK = false; static Core::Frontend::SoftwareKeyboardParameters ConvertToFrontendParameters( KeyboardConfig config, std::u16string initial_text) { Core::Frontend::SoftwareKeyboardParameters params{}; params.submit_text = Common::UTF16StringFromFixedZeroTerminatedBuffer( config.submit_text.data(), config.submit_text.size()); params.header_text = Common::UTF16StringFromFixedZeroTerminatedBuffer( config.header_text.data(), config.header_text.size()); params.sub_text = Common::UTF16StringFromFixedZeroTerminatedBuffer(config.sub_text.data(), config.sub_text.size()); params.guide_text = Common::UTF16StringFromFixedZeroTerminatedBuffer(config.guide_text.data(), config.guide_text.size()); params.initial_text = initial_text; params.max_length = config.length_limit == 0 ? DEFAULT_MAX_LENGTH : config.length_limit; params.password = static_cast(config.is_password); params.cursor_at_beginning = static_cast(config.initial_cursor_position); params.value = static_cast(config.keyset_disable_bitmask); return params; } SoftwareKeyboard::SoftwareKeyboard() = default; SoftwareKeyboard::~SoftwareKeyboard() = default; void SoftwareKeyboard::Initialize(std::queue> storage_) { complete = false; initial_text.clear(); final_data.clear(); Applet::Initialize(std::move(storage_)); ASSERT(storage_stack.size() >= 2); const auto& keyboard_config = storage_stack.front()->GetData(); storage_stack.pop(); ASSERT(keyboard_config.size() >= sizeof(KeyboardConfig)); std::memcpy(&config, keyboard_config.data(), sizeof(KeyboardConfig)); const auto& work_buffer = storage_stack.front()->GetData(); storage_stack.pop(); if (config.initial_string_size == 0) return; std::vector string(config.initial_string_size); std::memcpy(string.data(), work_buffer.data() + 4, string.size() * 2); initial_text = Common::UTF16StringFromFixedZeroTerminatedBuffer(string.data(), string.size()); } bool SoftwareKeyboard::TransactionComplete() const { return complete; } ResultCode SoftwareKeyboard::GetStatus() const { return status; } void SoftwareKeyboard::ReceiveInteractiveData(std::shared_ptr storage) { if (complete) return; const auto data = storage->GetData(); const auto status = static_cast(data[0]); if (status == INTERACTIVE_STATUS_OK) { complete = true; } else { const auto& frontend{Core::System::GetInstance().GetSoftwareKeyboard()}; std::array string; std::memcpy(string.data(), data.data() + 4, string.size() * 2); frontend.SendTextCheckDialog( Common::UTF16StringFromFixedZeroTerminatedBuffer(string.data(), string.size()), state); } } void SoftwareKeyboard::Execute(AppletStorageProxyFunction out_data, AppletStorageProxyFunction out_interactive_data, AppletStateProxyFunction state) { if (complete) return; const auto& frontend{Core::System::GetInstance().GetSoftwareKeyboard()}; const auto parameters = ConvertToFrontendParameters(config, initial_text); this->out_data = out_data; this->out_interactive_data = out_interactive_data; this->state = state; frontend.RequestText([this](std::optional text) { WriteText(text); }, parameters); } void SoftwareKeyboard::WriteText(std::optional text) { std::vector output(SWKBD_OUTPUT_BUFFER_SIZE); if (text.has_value()) { status = RESULT_SUCCESS; if (config.text_check) { const auto size = static_cast(text->size() * 2 + 4); std::memcpy(output.data(), &size, sizeof(u32)); } else { output[0] = 1; } const auto size = static_cast(text->size()); std::memcpy(output.data() + 4, &size, sizeof(u32)); std::memcpy(output.data() + 8, text->data(), std::min(text->size() * 2, SWKBD_OUTPUT_BUFFER_SIZE - 8)); } else { status = ResultCode(-1); complete = true; out_data(IStorage{output}); return; } complete = !config.text_check; out_data(IStorage{output}); out_interactive_data(IStorage{output}); state(); } } // namespace Service::AM::Applets