Android: minor update

master
Maksim 2022-07-03 21:32:55 +03:00
parent f9d97b5d05
commit 17ebe562a3
7 changed files with 247 additions and 211 deletions

View File

@ -3,7 +3,7 @@ apply plugin: 'kotlin-android'
android {
compileSdkVersion 32
buildToolsVersion '32.0.0'
buildToolsVersion '33.0.0'
ndkVersion '23.2.8568313'
defaultConfig {
applicationId 'com.multicraft.game'
@ -136,7 +136,7 @@ dependencies {
/* Third-party libraries */
implementation 'androidx.appcompat:appcompat:1.4.2'
implementation 'androidx.appcompat:appcompat-resources:1.4.2'
implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.4.1'
implementation 'androidx.lifecycle:lifecycle-runtime-ktx:2.5.0'
implementation 'androidx.work:work-runtime-ktx:2.7.1'
implementation 'com.google.android.material:material:1.6.1'
}

View File

@ -1,198 +0,0 @@
/*
MultiCraft
Copyright (C) 2014-2021 MoNTE48, Maksim Gamarnik <MoNTE48@mail.ua>
Copyright (C) 2014-2021 ubulem, Bektur Mambetov <berkut87@gmail.com>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3.0 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package com.multicraft.game;
import static android.content.res.Configuration.HARDKEYBOARDHIDDEN_NO;
import static android.text.InputType.TYPE_CLASS_TEXT;
import static android.text.InputType.TYPE_TEXT_FLAG_MULTI_LINE;
import static android.text.InputType.TYPE_TEXT_VARIATION_PASSWORD;
import static com.multicraft.game.helpers.Utilities.finishApp;
import static com.multicraft.game.helpers.Utilities.makeFullScreen;
import android.app.NativeActivity;
import android.content.Intent;
import android.content.res.Configuration;
import android.net.Uri;
import android.os.Bundle;
import android.view.KeyEvent;
import android.view.WindowManager.LayoutParams;
import android.view.inputmethod.EditorInfo;
import android.view.inputmethod.InputMethodManager;
import android.widget.Button;
import android.widget.EditText;
import androidx.appcompat.app.AlertDialog;
import com.multicraft.game.databinding.InputTextBinding;
public class GameActivity extends NativeActivity {
public static boolean isMultiPlayer;
static {
try {
System.loadLibrary("MultiCraft");
} catch (UnsatisfiedLinkError e) {
System.exit(0);
}
}
private int messageReturnCode = -1;
private String messageReturnValue = "";
private boolean hasKeyboard;
public static native void pauseGame();
public static native void keyboardEvent(boolean keyboard);
@Override
public void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
getWindow().addFlags(LayoutParams.FLAG_KEEP_SCREEN_ON);
hasKeyboard = getResources().getConfiguration().hardKeyboardHidden == HARDKEYBOARDHIDDEN_NO;
keyboardEvent(hasKeyboard);
}
@Override
public void onWindowFocusChanged(boolean hasFocus) {
super.onWindowFocusChanged(hasFocus);
if (hasFocus) makeFullScreen(getWindow());
}
@Override
public void onBackPressed() {
// Ignore the back press so MultiCraft can handle it
}
@Override
protected void onPause() {
super.onPause();
pauseGame();
}
@Override
protected void onResume() {
super.onResume();
makeFullScreen(getWindow());
}
@Override
public void onConfigurationChanged(Configuration newConfig) {
super.onConfigurationChanged(newConfig);
boolean statusKeyboard = getResources().getConfiguration().hardKeyboardHidden == HARDKEYBOARDHIDDEN_NO;
if (hasKeyboard != statusKeyboard) {
hasKeyboard = statusKeyboard;
keyboardEvent(hasKeyboard);
}
}
public void showDialog(String s, String hint, String current, int editType) {
runOnUiThread(() -> showDialogUI(hint, current, editType));
}
private void showDialogUI(String hint, String current, int editType) {
final AlertDialog.Builder builder = new AlertDialog.Builder(this);
if (editType == 1) builder.setPositiveButton(R.string.done, null);
InputTextBinding binding = InputTextBinding.inflate(getLayoutInflater());
String hintText = (!hint.isEmpty()) ? hint : getResources().getString(
(editType == 3) ? R.string.input_password : R.string.input_text);
binding.inputLayout.setHint(hintText);
builder.setView(binding.getRoot());
AlertDialog alertDialog = builder.create();
EditText editText = binding.editText;
editText.requestFocus();
editText.setText(current);
if (editType != 1) editText.setImeOptions(EditorInfo.IME_FLAG_NO_FULLSCREEN);
final InputMethodManager imm = (InputMethodManager) getSystemService(INPUT_METHOD_SERVICE);
int inputType = TYPE_CLASS_TEXT;
if (editType == 1) {
inputType = inputType | TYPE_TEXT_FLAG_MULTI_LINE;
editText.setMaxLines(8);
} else if (editType == 3)
inputType = inputType | TYPE_TEXT_VARIATION_PASSWORD;
editText.setInputType(inputType);
editText.setSelection(editText.getText().length());
editText.setOnEditorActionListener((view, KeyCode, event) -> {
if (KeyCode == KeyEvent.KEYCODE_ENTER || KeyCode == KeyEvent.KEYCODE_ENDCALL) {
imm.hideSoftInputFromWindow(editText.getWindowToken(), 0);
messageReturnCode = 0;
messageReturnValue = editText.getText().toString();
alertDialog.dismiss();
return true;
}
return false;
});
// should be above `show()`
alertDialog.getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_VISIBLE);
alertDialog.show();
Button button = alertDialog.getButton(AlertDialog.BUTTON_POSITIVE);
if (button != null) {
button.setOnClickListener(view -> {
imm.hideSoftInputFromWindow(editText.getWindowToken(), 0);
messageReturnCode = 0;
messageReturnValue = editText.getText().toString();
alertDialog.dismiss();
});
}
alertDialog.setOnCancelListener(dialog -> {
getWindow().setSoftInputMode(LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN);
messageReturnValue = current;
messageReturnCode = -1;
});
}
public int getDialogState() {
return messageReturnCode;
}
public String getDialogValue() {
messageReturnCode = -1;
return messageReturnValue;
}
public float getDensity() {
return getResources().getDisplayMetrics().density;
}
public void notifyServerConnect(boolean multiplayer) {
isMultiPlayer = multiplayer;
}
public void notifyExitGame() {
}
public void openURI(String uri) {
try {
Intent browserIntent = new Intent(Intent.ACTION_VIEW, Uri.parse(uri));
startActivity(browserIntent);
} catch (Exception ignored) {
}
}
public void finishGame(String exc) {
finishApp(true, this);
}
public void handleError(String exc) {
}
public void upgrade(String item) {
}
}

View File

@ -0,0 +1,219 @@
/*
MultiCraft
Copyright (C) 2014-2022 MoNTE48, Maksim Gamarnik <MoNTE48@mail.ua>
Copyright (C) 2014-2022 ubulem, Bektur Mambetov <berkut87@gmail.com>
This program is free software; you can redistribute it and/or modify
it under the terms of the GNU Lesser General Public License as published by
the Free Software Foundation; either version 3.0 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU Lesser General Public License for more details.
You should have received a copy of the GNU Lesser General Public License along
with this program; if not, write to the Free Software Foundation, Inc.,
51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
*/
package com.multicraft.game
import android.app.NativeActivity
import android.content.Intent
import android.content.res.Configuration
import android.content.res.Configuration.HARDKEYBOARDHIDDEN_NO
import android.net.Uri
import android.os.Bundle
import android.text.InputType
import android.view.*
import android.view.inputmethod.EditorInfo
import android.view.inputmethod.InputMethodManager
import android.widget.TextView
import androidx.appcompat.app.AlertDialog
import com.multicraft.game.databinding.InputTextBinding
import com.multicraft.game.helpers.Utilities.finishApp
import com.multicraft.game.helpers.Utilities.makeFullScreen
import kotlin.system.exitProcess
class GameActivity : NativeActivity() {
companion object {
var isMultiPlayer = false
var isInputActive = false
@JvmStatic
external fun pauseGame()
@JvmStatic
external fun keyboardEvent(keyboard: Boolean)
init {
try {
System.loadLibrary("MultiCraft")
} catch (e: UnsatisfiedLinkError) {
exitProcess(0)
}
}
}
private var messageReturnCode = -1
private var messageReturnValue = ""
private var hasKeyboard = false
public override fun onCreate(savedInstanceState: Bundle?) {
super.onCreate(savedInstanceState)
window.addFlags(WindowManager.LayoutParams.FLAG_KEEP_SCREEN_ON)
hasKeyboard = resources.configuration.hardKeyboardHidden == HARDKEYBOARDHIDDEN_NO
keyboardEvent(hasKeyboard)
}
override fun onWindowFocusChanged(hasFocus: Boolean) {
super.onWindowFocusChanged(hasFocus)
if (hasFocus) makeFullScreen(window)
}
override fun onBackPressed() {
// Ignore the back press so MultiCraft can handle it
}
override fun onPause() {
super.onPause()
pauseGame()
}
override fun onResume() {
super.onResume()
makeFullScreen(window)
}
override fun onConfigurationChanged(newConfig: Configuration) {
super.onConfigurationChanged(newConfig)
val statusKeyboard =
resources.configuration.hardKeyboardHidden == HARDKEYBOARDHIDDEN_NO
if (hasKeyboard != statusKeyboard) {
hasKeyboard = statusKeyboard
keyboardEvent(hasKeyboard)
}
}
@Suppress("unused")
fun showDialog(
@Suppress("UNUSED_PARAMETER") s: String?,
hint: String?, current: String?, editType: Int
) {
runOnUiThread { showDialogUI(hint, current, editType) }
}
private fun showDialogUI(hint: String?, current: String?, editType: Int) {
isInputActive = true
val builder = AlertDialog.Builder(this)
if (editType == 1) builder.setPositiveButton(R.string.done, null)
val binding = InputTextBinding.inflate(layoutInflater)
val hintText = hint?.ifEmpty {
resources.getString(if (editType == 3) R.string.input_password else R.string.input_text)
}
binding.inputLayout.hint = hintText
builder.setView(binding.root)
val alertDialog = builder.create()
val editText = binding.editText
editText.requestFocus()
editText.setText(current.toString())
if (editType != 1) editText.imeOptions = EditorInfo.IME_FLAG_NO_FULLSCREEN
val imm = getSystemService(INPUT_METHOD_SERVICE) as InputMethodManager
var inputType = InputType.TYPE_CLASS_TEXT
when (editType) {
1 -> {
inputType = inputType or InputType.TYPE_TEXT_FLAG_MULTI_LINE
editText.maxLines = 8
}
3 -> inputType = inputType or InputType.TYPE_TEXT_VARIATION_PASSWORD
}
editText.inputType = inputType
editText.setSelection(editText.text?.length ?: 0)
// for Android OS
editText.setOnEditorActionListener { _: TextView?, KeyCode: Int, _: KeyEvent? ->
if (KeyCode == KeyEvent.KEYCODE_ENTER || KeyCode == KeyEvent.KEYCODE_ENDCALL) {
imm.hideSoftInputFromWindow(editText.windowToken, 0)
messageReturnCode = 0
messageReturnValue = editText.text.toString()
alertDialog.dismiss()
isInputActive = false
return@setOnEditorActionListener true
}
return@setOnEditorActionListener false
}
// for Chrome OS
editText.setOnKeyListener { _: View?, KeyCode: Int, _: KeyEvent? ->
if (KeyCode == KeyEvent.KEYCODE_ENTER || KeyCode == KeyEvent.KEYCODE_ENDCALL) {
imm.hideSoftInputFromWindow(editText.windowToken, 0)
messageReturnCode = 0
messageReturnValue = editText.text.toString()
alertDialog.dismiss()
isInputActive = false
return@setOnKeyListener true
}
return@setOnKeyListener false
}
// should be above `show()`
alertDialog.window!!.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_VISIBLE)
alertDialog.show()
val button = alertDialog.getButton(AlertDialog.BUTTON_POSITIVE)
button?.setOnClickListener {
imm.hideSoftInputFromWindow(editText.windowToken, 0)
messageReturnCode = 0
messageReturnValue = editText.text.toString()
alertDialog.dismiss()
isInputActive = false
}
alertDialog.setOnCancelListener {
window.setSoftInputMode(WindowManager.LayoutParams.SOFT_INPUT_STATE_ALWAYS_HIDDEN)
messageReturnValue = current.toString()
messageReturnCode = -1
isInputActive = false
}
}
@Suppress("unused")
fun getDialogState() = messageReturnCode
@Suppress("unused")
fun getDialogValue(): String {
messageReturnCode = -1
return messageReturnValue
}
@Suppress("unused")
fun getDensity() = resources.displayMetrics.density
@Suppress("unused")
fun notifyServerConnect(multiplayer: Boolean) {
isMultiPlayer = multiplayer
}
@Suppress("unused")
fun notifyExitGame() {
}
@Suppress("unused")
fun openURI(uri: String?) {
try {
val browserIntent = Intent(Intent.ACTION_VIEW, Uri.parse(uri))
startActivity(browserIntent)
} catch (ignored: Exception) {
}
}
@Suppress("unused")
fun finishGame(exc: String?) {
finishApp(true, this)
}
@Suppress("unused")
fun handleError(exc: String?) {
}
@Suppress("unused")
fun upgrade(item: String) {
}
}

View File

@ -81,12 +81,12 @@ class MainActivity : AppCompatActivity() {
}
}
@Deprecated("Deprecated in Java")
override fun onActivityResult(requestCode: Int, resultCode: Int, data: Intent?) {
@Suppress("DEPRECATION")
super.onActivityResult(requestCode, resultCode, data)
if (requestCode == REQUEST_CONNECTION)
checkAppVersion()
}
override fun onBackPressed() {

View File

@ -18,7 +18,7 @@ buildscript {
classpath 'com.android.tools.build:gradle:7.2.1'
//noinspection GradleDependency
classpath 'de.undercouch:gradle-download-task:4.1.2'
classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:1.6.21'
classpath 'org.jetbrains.kotlin:kotlin-gradle-plugin:1.7.10'
// NOTE: Do not place your application dependencies here; they belong
// in the individual module build.gradle files
}

View File

@ -3,7 +3,7 @@ apply plugin: 'de.undercouch.download'
android {
compileSdkVersion 32
buildToolsVersion '32.0.0'
buildToolsVersion '33.0.0'
ndkVersion '23.2.8568313'
defaultConfig {
minSdkVersion 21

View File

@ -44,8 +44,6 @@ extern "C" void external_pause_game();
static std::atomic<bool> ran = {false};
jmethodID notifyExit;
void android_main(android_app *app)
{
if (ran.exchange(true)) {
@ -105,6 +103,12 @@ jclass findClass(const std::string &classname)
return nullptr;
jclass nativeactivity = jnienv->FindClass("android/app/NativeActivity");
if (jnienv->ExceptionCheck()) {
jnienv->ExceptionClear();
return nullptr;
}
jmethodID getClassLoader = jnienv->GetMethodID(
nativeactivity, "getClassLoader", "()Ljava/lang/ClassLoader;");
jobject cls = jnienv->CallObjectMethod(activityObj, getClassLoader);
@ -112,7 +116,14 @@ jclass findClass(const std::string &classname)
jmethodID findClass = jnienv->GetMethodID(classLoader, "loadClass",
"(Ljava/lang/String;)Ljava/lang/Class;");
jstring strClassName = jnienv->NewStringUTF(classname.c_str());
return (jclass) jnienv->CallObjectMethod(cls, findClass, strClassName);
jclass result = (jclass) jnienv->CallObjectMethod(cls, findClass, strClassName);
if (jnienv->ExceptionCheck()) {
jnienv->ExceptionClear();
return nullptr;
}
return result;
}
void initAndroid()
@ -136,10 +147,6 @@ void initAndroid()
"porting::initAndroid unable to find java native activity class" <<
std::endl;
notifyExit = jnienv->GetMethodID(nativeActivity, "notifyExitGame", "()V");
FATAL_ERROR_IF(notifyExit == nullptr,
"porting::initAndroid unable to find java notifyExit method");
#ifdef GPROF
// in the start-up code
__android_log_print(ANDROID_LOG_ERROR, PROJECT_NAME_C,
@ -317,9 +324,17 @@ void notifyServerConnect(bool is_multiplayer)
void notifyExitGame()
{
if (jnienv == nullptr || activityObj == nullptr || notifyExit == nullptr)
if (jnienv == nullptr || activityObj == nullptr)
return;
jclass _nativeActivity = findClass("com/multicraft/game/GameActivity");
FATAL_ERROR_IF(_nativeActivity == nullptr,
"porting::notifyExitGame unable to find java native activity class");
jmethodID notifyExit = jnienv->GetMethodID(_nativeActivity, "notifyExitGame", "()V");
FATAL_ERROR_IF(notifyExit == nullptr,
"porting::notifyExitGame unable to find java notifyExit method");
jnienv->CallVoidMethod(activityObj, notifyExit);
if (jnienv->ExceptionOccurred())