Mobile: improve the chat experience (#146)
* Some chat input dialog fixes. - If getAndroidChatOpen() is true then input dialog is created by chat. - Hide touchscreengui when chat input dialog is open. * Check input dialog owner in config registration, just in case * Make sure there is no menu active before showing touchscreengui * Reset input dialog owner when reading value
This commit is contained in:
parent
2c244aa28c
commit
52228db808
@ -259,6 +259,9 @@ class GameActivity : SDLActivity() {
|
||||
}
|
||||
}
|
||||
|
||||
@Suppress("unused")
|
||||
fun isDialogActive() = isInputActive
|
||||
|
||||
@Suppress("unused")
|
||||
fun getDialogState() = messageReturnCode
|
||||
|
||||
|
@ -819,7 +819,7 @@ protected:
|
||||
}
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
void handleAndroidChatInput();
|
||||
void handleTouchChatInput();
|
||||
#endif
|
||||
|
||||
private:
|
||||
@ -1873,6 +1873,14 @@ void Game::processUserInput(f32 dtime)
|
||||
g_touchscreengui->hide();
|
||||
#endif
|
||||
}
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
if (porting::isInputDialogActive() && porting::getInputDialogOwner() == "chat") {
|
||||
input->clear();
|
||||
g_touchscreengui->hide();
|
||||
}
|
||||
#endif
|
||||
|
||||
#ifdef HAVE_TOUCHSCREENGUI
|
||||
else if (g_touchscreengui) {
|
||||
/* on touchscreengui step may generate own input events which ain't
|
||||
@ -1889,13 +1897,11 @@ void Game::processUserInput(f32 dtime)
|
||||
input->step(dtime);
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
if (!porting::hasRealKeyboard()) {
|
||||
auto formspec = m_game_ui->getFormspecGUI();
|
||||
if (formspec)
|
||||
formspec->getAndroidUIInput();
|
||||
else
|
||||
handleAndroidChatInput();
|
||||
}
|
||||
handleTouchChatInput();
|
||||
|
||||
auto formspec = m_game_ui->getFormspecGUI();
|
||||
if (formspec)
|
||||
formspec->getTouchUIInput();
|
||||
#endif
|
||||
|
||||
bool doubletap_jump = m_cache_doubletap_jump;
|
||||
@ -1923,9 +1929,6 @@ void Game::processKeyInput()
|
||||
} else if (wasKeyDown(KeyType::INVENTORY)) {
|
||||
openInventory();
|
||||
} else if (input->cancelPressed()) {
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
gui_chat_console->setAndroidChatOpen(false);
|
||||
#endif
|
||||
if (!gui_chat_console->isOpenInhibited()) {
|
||||
showPauseMenu();
|
||||
}
|
||||
@ -2137,21 +2140,21 @@ void Game::openConsole(float scale, const wchar_t *line)
|
||||
{
|
||||
assert(scale > 0.0f && scale <= 1.0f);
|
||||
|
||||
if (gui_chat_console->getAndroidChatOpen())
|
||||
if (gui_chat_console->isOpenInhibited())
|
||||
return;
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
if (porting::isInputDialogActive())
|
||||
return;
|
||||
|
||||
if (!porting::hasRealKeyboard()) {
|
||||
porting::showInputDialog("", "", 2);
|
||||
gui_chat_console->setAndroidChatOpen(true);
|
||||
porting::showInputDialog("", "", 2, "chat");
|
||||
}
|
||||
|
||||
if (!RenderingEngine::isTablet())
|
||||
return;
|
||||
#endif
|
||||
|
||||
if (gui_chat_console->isOpenInhibited())
|
||||
return;
|
||||
gui_chat_console->openConsole(scale);
|
||||
if (line) {
|
||||
gui_chat_console->setCloseOnEnter(true);
|
||||
@ -2160,16 +2163,21 @@ void Game::openConsole(float scale, const wchar_t *line)
|
||||
}
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
void Game::handleAndroidChatInput()
|
||||
void Game::handleTouchChatInput()
|
||||
{
|
||||
if (gui_chat_console->getAndroidChatOpen() &&
|
||||
if (porting::getInputDialogOwner() == "chat" &&
|
||||
porting::getInputDialogState() == 0) {
|
||||
std::string text = porting::getInputDialogValue();
|
||||
client->typeChatMessage(utf8_to_wide(text));
|
||||
gui_chat_console->setAndroidChatOpen(false);
|
||||
if (!text.empty() && gui_chat_console->isOpen()) {
|
||||
gui_chat_console->closeConsole();
|
||||
}
|
||||
#ifdef HAVE_TOUCHSCREENGUI
|
||||
if (!gui_chat_console->isOpen() && !isMenuActive()) {
|
||||
if (g_touchscreengui && g_touchscreengui->isActive())
|
||||
g_touchscreengui->show();
|
||||
}
|
||||
#endif
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
@ -1589,10 +1589,9 @@ bool GUIChatConsole::preprocessEvent(SEvent event)
|
||||
event.TouchInput.Y >= prompt_y &&
|
||||
event.TouchInput.Y <= m_height) {
|
||||
if (event.TouchInput.Event == ETIE_PRESSED_DOWN &&
|
||||
!m_android_chat_open) {
|
||||
!porting::isInputDialogActive()) {
|
||||
ChatPrompt& prompt = m_chat_backend->getPrompt();
|
||||
porting::showInputDialog("", "", 2);
|
||||
m_android_chat_open = true;
|
||||
porting::showInputDialog("", "", 2, "chat");
|
||||
}
|
||||
}
|
||||
#endif
|
||||
|
@ -170,9 +170,6 @@ public:
|
||||
|
||||
bool preprocessEvent(SEvent event);
|
||||
|
||||
bool getAndroidChatOpen() { return m_android_chat_open; }
|
||||
void setAndroidChatOpen(bool value) { m_android_chat_open = value; }
|
||||
|
||||
void onLinesModified();
|
||||
void onPromptModified();
|
||||
|
||||
@ -253,6 +250,4 @@ private:
|
||||
u32 m_scrollbar_width = 0;
|
||||
GUIScrollBar *m_vscrollbar = nullptr;
|
||||
s32 m_bottom_scroll_pos = 0;
|
||||
|
||||
bool m_android_chat_open = false;
|
||||
};
|
||||
|
@ -220,7 +220,7 @@ void GUIConfirmRegistration::drawMenu()
|
||||
|
||||
gui::IGUIElement::draw();
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
getAndroidUIInput();
|
||||
getTouchUIInput();
|
||||
#endif
|
||||
}
|
||||
|
||||
@ -308,11 +308,14 @@ bool GUIConfirmRegistration::OnEvent(const SEvent &event)
|
||||
}
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
bool GUIConfirmRegistration::getAndroidUIInput()
|
||||
bool GUIConfirmRegistration::getTouchUIInput()
|
||||
{
|
||||
if (m_jni_field_name.empty() || m_jni_field_name != "password")
|
||||
return false;
|
||||
|
||||
if (porting::getInputDialogOwner() != "modalmenu")
|
||||
return false;
|
||||
|
||||
// still waiting
|
||||
if (porting::getInputDialogState() == -1)
|
||||
return true;
|
||||
|
@ -52,7 +52,7 @@ public:
|
||||
|
||||
bool OnEvent(const SEvent &event);
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
bool getAndroidUIInput();
|
||||
bool getTouchUIInput();
|
||||
#endif
|
||||
|
||||
private:
|
||||
|
@ -348,7 +348,7 @@ void GUIEngine::run()
|
||||
m_script->step();
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
m_menu->getAndroidUIInput();
|
||||
m_menu->getTouchUIInput();
|
||||
#endif
|
||||
}
|
||||
}
|
||||
|
@ -3602,11 +3602,14 @@ void GUIFormSpecMenu::legacySortElements(core::list<IGUIElement *>::Iterator fro
|
||||
}
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
bool GUIFormSpecMenu::getAndroidUIInput()
|
||||
bool GUIFormSpecMenu::getTouchUIInput()
|
||||
{
|
||||
if (m_jni_field_name.empty())
|
||||
return false;
|
||||
|
||||
if (porting::getInputDialogOwner() != "modalmenu")
|
||||
return false;
|
||||
|
||||
// still waiting
|
||||
if (porting::getInputDialogState() == -1)
|
||||
return true;
|
||||
|
@ -266,7 +266,7 @@ public:
|
||||
std::vector<std::string>* getDropDownValues(const std::string &name);
|
||||
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
bool getAndroidUIInput();
|
||||
bool getTouchUIInput();
|
||||
#endif
|
||||
|
||||
protected:
|
||||
|
@ -346,6 +346,9 @@ bool GUIModalMenu::preprocessEvent(const SEvent &event)
|
||||
if (field_name.empty() || porting::hasRealKeyboard())
|
||||
return retval;
|
||||
|
||||
if (porting::isInputDialogActive())
|
||||
return retval;
|
||||
|
||||
m_jni_field_name = field_name;
|
||||
|
||||
// single line text input
|
||||
@ -360,7 +363,8 @@ bool GUIModalMenu::preprocessEvent(const SEvent &event)
|
||||
type = 3;
|
||||
|
||||
porting::showInputDialog(wide_to_utf8(getLabelByID(hovered->getID())),
|
||||
wide_to_utf8(((gui::IGUIEditBox *)hovered)->getText()), type);
|
||||
wide_to_utf8(((gui::IGUIEditBox *)hovered)->getText()), type,
|
||||
"modalmenu");
|
||||
return retval;
|
||||
}
|
||||
}
|
||||
|
@ -58,7 +58,7 @@ public:
|
||||
virtual bool OnEvent(const SEvent &event) { return false; };
|
||||
virtual bool pausesGame() { return false; } // Used for pause menu
|
||||
#if defined(__ANDROID__) || defined(__IOS__)
|
||||
virtual bool getAndroidUIInput() { return false; }
|
||||
virtual bool getTouchUIInput() { return false; }
|
||||
#endif
|
||||
|
||||
protected:
|
||||
|
@ -95,6 +95,7 @@ namespace porting {
|
||||
JNIEnv *jnienv;
|
||||
jclass activityClass;
|
||||
jobject activityObj;
|
||||
std::string input_dialog_owner;
|
||||
|
||||
jclass findClass(const std::string &classname)
|
||||
{
|
||||
@ -209,8 +210,10 @@ void initializePaths()
|
||||
activityObj, mt_getAbsPath, "getCacheDir");
|
||||
}
|
||||
|
||||
void showInputDialog(const std::string &hint, const std::string ¤t, int editType)
|
||||
void showInputDialog(const std::string &hint, const std::string ¤t, int editType, std::string owner)
|
||||
{
|
||||
input_dialog_owner = owner;
|
||||
|
||||
jmethodID showdialog = jnienv->GetMethodID(activityClass, "showDialog",
|
||||
"(Ljava/lang/String;Ljava/lang/String;I)V");
|
||||
|
||||
@ -236,6 +239,22 @@ void openURIAndroid(const std::string &url)
|
||||
jnienv->CallVoidMethod(activityObj, url_open, jurl);
|
||||
}
|
||||
|
||||
std::string getInputDialogOwner()
|
||||
{
|
||||
return input_dialog_owner;
|
||||
}
|
||||
|
||||
bool isInputDialogActive()
|
||||
{
|
||||
jmethodID dialog_active = jnienv->GetMethodID(activityClass,
|
||||
"isDialogActive", "()Z");
|
||||
|
||||
FATAL_ERROR_IF(dialog_active == nullptr,
|
||||
"porting::isInputDialogActive unable to find Java dialog state method");
|
||||
|
||||
return jnienv->CallBooleanMethod(activityObj, dialog_active);
|
||||
}
|
||||
|
||||
int getInputDialogState()
|
||||
{
|
||||
jmethodID dialogstate = jnienv->GetMethodID(activityClass,
|
||||
@ -249,6 +268,8 @@ int getInputDialogState()
|
||||
|
||||
std::string getInputDialogValue()
|
||||
{
|
||||
input_dialog_owner = "";
|
||||
|
||||
jmethodID dialogvalue = jnienv->GetMethodID(activityClass,
|
||||
"getDialogValue", "()Ljava/lang/String;");
|
||||
|
||||
|
@ -34,6 +34,8 @@ namespace porting {
|
||||
// java <-> c++ interaction interface
|
||||
extern JNIEnv *jnienv;
|
||||
|
||||
extern std::string input_dialog_owner;
|
||||
|
||||
// do initialization required on android only
|
||||
void initAndroid();
|
||||
|
||||
@ -51,7 +53,11 @@ void initializePaths();
|
||||
* @param editType type of texfield
|
||||
* (1 == multiline text input; 2 == single line text input; 3 == password field)
|
||||
*/
|
||||
void showInputDialog(const std::string &hint, const std::string ¤t, int editType);
|
||||
void showInputDialog(const std::string &hint, const std::string ¤t, int editType, std::string owner = "");
|
||||
|
||||
std::string getInputDialogOwner();
|
||||
|
||||
bool isInputDialogActive();
|
||||
|
||||
void openURIAndroid(const std::string &url);
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user