2015-05-26 04:30:20 +00:00
|
|
|
// Copyright 2015 Citra Emulator Project
|
|
|
|
// Licensed under GPLv2 or any later version
|
|
|
|
// Refer to the license.txt file included.
|
|
|
|
|
2018-06-20 12:01:50 +00:00
|
|
|
#include <algorithm>
|
2015-07-12 21:10:37 +00:00
|
|
|
#include <cstring>
|
|
|
|
#include <string>
|
2015-05-26 04:30:20 +00:00
|
|
|
#include "common/assert.h"
|
|
|
|
#include "common/logging/log.h"
|
2015-05-27 20:21:06 +00:00
|
|
|
#include "common/string_util.h"
|
2016-09-21 06:52:38 +00:00
|
|
|
#include "core/hle/applets/swkbd.h"
|
2015-07-12 21:10:37 +00:00
|
|
|
#include "core/hle/kernel/kernel.h"
|
|
|
|
#include "core/hle/kernel/shared_memory.h"
|
|
|
|
#include "core/hle/result.h"
|
2017-12-16 19:35:37 +00:00
|
|
|
#include "core/hle/service/gsp/gsp.h"
|
2016-09-18 00:38:01 +00:00
|
|
|
#include "core/hle/service/hid/hid.h"
|
2015-07-12 21:10:37 +00:00
|
|
|
#include "core/memory.h"
|
2015-05-26 04:30:20 +00:00
|
|
|
|
|
|
|
////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
|
|
namespace HLE {
|
|
|
|
namespace Applets {
|
|
|
|
|
|
|
|
ResultCode SoftwareKeyboard::ReceiveParameter(Service::APT::MessageParameter const& parameter) {
|
2018-01-25 13:39:54 +00:00
|
|
|
if (parameter.signal != Service::APT::SignalType::Request) {
|
2018-06-29 11:18:07 +00:00
|
|
|
LOG_ERROR(Service_APT, "unsupported signal {}", static_cast<u32>(parameter.signal));
|
2015-05-26 04:30:20 +00:00
|
|
|
UNIMPLEMENTED();
|
|
|
|
// TODO(Subv): Find the right error code
|
|
|
|
return ResultCode(-1);
|
|
|
|
}
|
|
|
|
|
2016-09-18 00:38:01 +00:00
|
|
|
// The LibAppJustStarted message contains a buffer with the size of the framebuffer shared
|
|
|
|
// memory.
|
2016-05-05 17:36:07 +00:00
|
|
|
// Create the SharedMemory that will hold the framebuffer data
|
|
|
|
Service::APT::CaptureBufferInfo capture_info;
|
2016-04-16 10:18:49 +00:00
|
|
|
ASSERT(sizeof(capture_info) == parameter.buffer.size());
|
2016-05-05 17:36:07 +00:00
|
|
|
|
2016-04-16 10:18:49 +00:00
|
|
|
memcpy(&capture_info, parameter.buffer.data(), sizeof(capture_info));
|
2016-05-05 17:36:07 +00:00
|
|
|
|
|
|
|
using Kernel::MemoryPermission;
|
2016-05-08 22:10:53 +00:00
|
|
|
// Allocate a heap block of the required size for this applet.
|
|
|
|
heap_memory = std::make_shared<std::vector<u8>>(capture_info.size);
|
|
|
|
// Create a SharedMemory that directly points to this heap block.
|
2016-09-18 00:38:01 +00:00
|
|
|
framebuffer_memory = Kernel::SharedMemory::CreateForApplet(
|
2017-08-19 16:37:21 +00:00
|
|
|
heap_memory, 0, capture_info.size, MemoryPermission::ReadWrite, MemoryPermission::ReadWrite,
|
|
|
|
"SoftwareKeyboard Memory");
|
2016-05-05 17:36:07 +00:00
|
|
|
|
|
|
|
// Send the response message with the newly created SharedMemory
|
2015-05-26 04:30:20 +00:00
|
|
|
Service::APT::MessageParameter result;
|
2018-01-25 13:39:54 +00:00
|
|
|
result.signal = Service::APT::SignalType::Response;
|
2016-04-16 10:18:49 +00:00
|
|
|
result.buffer.clear();
|
2018-01-25 13:39:54 +00:00
|
|
|
result.destination_id = Service::APT::AppletId::Application;
|
|
|
|
result.sender_id = id;
|
2015-05-26 04:30:20 +00:00
|
|
|
result.object = framebuffer_memory;
|
|
|
|
|
2018-01-25 13:39:54 +00:00
|
|
|
SendParameter(result);
|
2015-05-26 04:30:20 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
2015-05-27 20:21:06 +00:00
|
|
|
ResultCode SoftwareKeyboard::StartImpl(Service::APT::AppletStartupParameter const& parameter) {
|
2016-09-18 00:38:01 +00:00
|
|
|
ASSERT_MSG(parameter.buffer.size() == sizeof(config),
|
|
|
|
"The size of the parameter (SoftwareKeyboardConfig) is wrong");
|
2015-05-27 20:21:06 +00:00
|
|
|
|
2016-04-16 10:18:49 +00:00
|
|
|
memcpy(&config, parameter.buffer.data(), parameter.buffer.size());
|
2016-09-18 00:38:01 +00:00
|
|
|
text_memory =
|
|
|
|
boost::static_pointer_cast<Kernel::SharedMemory, Kernel::Object>(parameter.object);
|
2015-05-26 04:30:20 +00:00
|
|
|
|
|
|
|
// TODO(Subv): Verify if this is the correct behavior
|
|
|
|
memset(text_memory->GetPointer(), 0, text_memory->size);
|
|
|
|
|
2015-05-26 16:00:26 +00:00
|
|
|
DrawScreenKeyboard();
|
|
|
|
|
2018-06-20 04:01:54 +00:00
|
|
|
using namespace Frontend;
|
2018-06-20 12:01:50 +00:00
|
|
|
frontend_applet = GetRegisteredSoftwareKeyboard();
|
2018-06-20 04:01:54 +00:00
|
|
|
if (frontend_applet) {
|
|
|
|
KeyboardConfig frontend_config = ToFrontendConfig(config);
|
|
|
|
frontend_applet->Setup(&frontend_config);
|
|
|
|
}
|
|
|
|
|
2016-12-07 22:11:39 +00:00
|
|
|
is_running = true;
|
2015-05-26 16:00:26 +00:00
|
|
|
return RESULT_SUCCESS;
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoftwareKeyboard::Update() {
|
2018-06-20 04:01:54 +00:00
|
|
|
using namespace Frontend;
|
2018-06-20 12:01:50 +00:00
|
|
|
KeyboardData data(*frontend_applet->ReceiveData());
|
2018-06-20 04:01:54 +00:00
|
|
|
std::u16string text = Common::UTF8ToUTF16(data.text);
|
2015-05-26 16:00:26 +00:00
|
|
|
memcpy(text_memory->GetPointer(), text.c_str(), text.length() * sizeof(char16_t));
|
2018-06-20 04:01:54 +00:00
|
|
|
switch (config.num_buttons_m1) {
|
2018-06-20 12:01:50 +00:00
|
|
|
case SoftwareKeyboardButtonConfig::SingleButton:
|
|
|
|
config.return_code = SoftwareKeyboardResult::D0Click;
|
2018-06-20 04:01:54 +00:00
|
|
|
break;
|
2018-06-20 12:01:50 +00:00
|
|
|
case SoftwareKeyboardButtonConfig::DualButton:
|
2018-06-20 04:01:54 +00:00
|
|
|
if (data.button == 0)
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::D1Click0;
|
2018-06-20 04:01:54 +00:00
|
|
|
else
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::D1Click1;
|
2018-06-20 04:01:54 +00:00
|
|
|
break;
|
2018-06-20 12:01:50 +00:00
|
|
|
case SoftwareKeyboardButtonConfig::TripleButton:
|
2018-06-20 04:01:54 +00:00
|
|
|
if (data.button == 0)
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::D2Click0;
|
2018-06-20 04:01:54 +00:00
|
|
|
else if (data.button == 1)
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::D2Click1;
|
2018-06-20 04:01:54 +00:00
|
|
|
else
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::D2Click2;
|
2018-06-20 04:01:54 +00:00
|
|
|
break;
|
2018-06-20 12:01:50 +00:00
|
|
|
case SoftwareKeyboardButtonConfig::NoButton:
|
2018-06-20 04:01:54 +00:00
|
|
|
// TODO: find out what is actually returned
|
2018-06-20 12:01:50 +00:00
|
|
|
config.return_code = SoftwareKeyboardResult::None;
|
2018-06-20 04:01:54 +00:00
|
|
|
break;
|
|
|
|
default:
|
2018-06-20 12:01:50 +00:00
|
|
|
LOG_CRITICAL(Applet_SWKBD, "Unknown button config {}",
|
|
|
|
static_cast<int>(config.num_buttons_m1));
|
2018-06-20 04:01:54 +00:00
|
|
|
UNREACHABLE();
|
|
|
|
}
|
2015-05-26 16:00:26 +00:00
|
|
|
|
2018-06-20 12:01:50 +00:00
|
|
|
config.text_length = static_cast<u16>(text.size());
|
2015-05-26 04:30:20 +00:00
|
|
|
config.text_offset = 0;
|
|
|
|
|
2015-05-27 20:21:06 +00:00
|
|
|
// TODO(Subv): We're finalizing the applet immediately after it's started,
|
2015-05-26 16:00:26 +00:00
|
|
|
// but we should defer this call until after all the input has been collected.
|
|
|
|
Finalize();
|
|
|
|
}
|
|
|
|
|
|
|
|
void SoftwareKeyboard::DrawScreenKeyboard() {
|
2018-06-20 12:01:50 +00:00
|
|
|
// TODO(Subv): Draw the HLE keyboard, for now just do nothing
|
2015-05-26 16:00:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void SoftwareKeyboard::Finalize() {
|
|
|
|
// Let the application know that we're closing
|
2015-05-26 04:30:20 +00:00
|
|
|
Service::APT::MessageParameter message;
|
2016-04-16 10:18:49 +00:00
|
|
|
message.buffer.resize(sizeof(SoftwareKeyboardConfig));
|
|
|
|
std::memcpy(message.buffer.data(), &config, message.buffer.size());
|
2018-01-25 13:39:54 +00:00
|
|
|
message.signal = Service::APT::SignalType::WakeupByExit;
|
|
|
|
message.destination_id = Service::APT::AppletId::Application;
|
|
|
|
message.sender_id = id;
|
|
|
|
SendParameter(message);
|
2015-05-26 04:30:20 +00:00
|
|
|
|
2016-12-07 22:11:39 +00:00
|
|
|
is_running = false;
|
2015-05-26 04:30:20 +00:00
|
|
|
}
|
2018-06-20 04:01:54 +00:00
|
|
|
|
2018-06-20 12:01:50 +00:00
|
|
|
Frontend::KeyboardConfig SoftwareKeyboard::ToFrontendConfig(
|
|
|
|
const SoftwareKeyboardConfig& config) const {
|
2018-06-20 04:01:54 +00:00
|
|
|
using namespace Frontend;
|
|
|
|
KeyboardConfig frontend_config;
|
|
|
|
frontend_config.button_config = static_cast<ButtonConfig>(config.num_buttons_m1);
|
|
|
|
frontend_config.accept_mode = static_cast<AcceptedInput>(config.valid_input);
|
|
|
|
frontend_config.multiline_mode = config.multiline;
|
|
|
|
frontend_config.max_text_length = config.max_text_length;
|
|
|
|
frontend_config.max_digits = config.max_digits;
|
2018-06-20 12:01:50 +00:00
|
|
|
std::u16string buffer(config.hint_text.size(), 0);
|
|
|
|
std::memcpy(buffer.data(), config.hint_text.data(), config.hint_text.size() * sizeof(u16));
|
|
|
|
frontend_config.hint_text = Common::UTF16ToUTF8(buffer);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.has_custom_button_text =
|
2018-06-20 12:01:50 +00:00
|
|
|
!std::all_of(config.button_text.begin(), config.button_text.end(),
|
|
|
|
[](std::array<u16, HLE::Applets::MAX_BUTTON_TEXT_LEN + 1> x) {
|
|
|
|
return std::all_of(x.begin(), x.end(), [](u16 x) { return x == 0; });
|
|
|
|
});
|
2018-06-20 04:01:54 +00:00
|
|
|
if (frontend_config.has_custom_button_text) {
|
|
|
|
for (const auto& text : config.button_text) {
|
2018-06-20 12:01:50 +00:00
|
|
|
buffer.resize(text.size());
|
|
|
|
std::memcpy(buffer.data(), text.data(), text.size() * sizeof(u16));
|
|
|
|
frontend_config.button_text.push_back(Common::UTF16ToUTF8(buffer));
|
2018-06-20 04:01:54 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
frontend_config.filters.prevent_digit =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::Digits);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.filters.prevent_at =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::At);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.filters.prevent_percent =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::Percent);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.filters.prevent_backslash =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::Backslash);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.filters.prevent_profanity =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::Profanity);
|
2018-06-20 04:01:54 +00:00
|
|
|
frontend_config.filters.enable_callback =
|
2018-06-20 12:01:50 +00:00
|
|
|
static_cast<bool>(config.filter_flags & SoftwareKeyboardFilter::Callback);
|
2018-06-20 04:01:54 +00:00
|
|
|
return frontend_config;
|
|
|
|
}
|
2018-03-09 17:54:43 +00:00
|
|
|
} // namespace Applets
|
|
|
|
} // namespace HLE
|