mirror of https://github.com/buggins/dlangui.git
commit
1822edb88d
|
@ -2,9 +2,16 @@ PLATFORM_DIR=armeabi-v7a
|
|||
|
||||
echo "\nLOCAL_MODULE: $LOCAL_MODULE"
|
||||
|
||||
|
||||
OS_TYPE="unknown"
|
||||
case "$OSTYPE" in
|
||||
msys*|cygwin*) OS_TYPE="windows" ;;
|
||||
linux*) OS_TYPE="linux" ;;
|
||||
esac
|
||||
|
||||
LDC_PARAMS="-mtriple=armv7-none-linux-androideabi -relocation-model=pic "
|
||||
export LD=$NDK/toolchains/llvm/prebuilt/linux-$NDK_ARCH/bin/llvm-link
|
||||
export CC=$NDK/toolchains/llvm/prebuilt/linux-$NDK_ARCH/bin/clang
|
||||
export LD=$NDK/toolchains/llvm/prebuilt/$OS_TYPE-$NDK_ARCH/bin/llvm-link
|
||||
export CC=$NDK/toolchains/llvm/prebuilt/$OS_TYPE-$NDK_ARCH/bin/clang
|
||||
|
||||
SOURCES="$LOCAL_SRC_FILES $DLANGUI_SOURCES"
|
||||
SOURCE_PATHS="-I./jni $DLANGUI_SOURCE_PATHS $DLANGUI_IMPORT_PATHS"
|
||||
|
@ -24,7 +31,7 @@ LINK_OPTIONS="\
|
|||
-Wl,-soname,libnative-activity.so \
|
||||
-shared \
|
||||
--sysroot=$NDK/platforms/$ANDROID_TARGET/arch-arm \
|
||||
-gcc-toolchain $NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/linux-$NDK_ARCH \
|
||||
-gcc-toolchain $NDK/toolchains/arm-linux-androideabi-4.9/prebuilt/$OS_TYPE-$NDK_ARCH \
|
||||
-no-canonical-prefixes \
|
||||
-target armv7-none-linux-androideabi \
|
||||
-Wl,--fix-cortex-a8 \
|
||||
|
|
|
@ -23,11 +23,13 @@ android_build_config.mk
|
|||
Update paths to Android NDK, SDK, DlangUI source directory.
|
||||
|
||||
Default values:
|
||||
```
|
||||
export DLANGUI_DIR=$HOME/src/d/dlangui
|
||||
export NDK=$HOME/android-ndk-r11c
|
||||
export SDK=$HOME/android-sdk-linux
|
||||
export LDC=$HOME/ldc2-android-arm-0.17.0-alpha2-linux-x86_64
|
||||
export NDK_ARCH=x86_64
|
||||
```
|
||||
|
||||
|
||||
Use LDC cross compiler for armv7a build according instructions
|
||||
|
@ -35,6 +37,33 @@ Use LDC cross compiler for armv7a build according instructions
|
|||
https://wiki.dlang.org/Build_LDC_for_Android#Build_a_sample_OpenGL_Android_app_ported_to_D
|
||||
|
||||
|
||||
|
||||
Dependencies
|
||||
============
|
||||
|
||||
FreeType dependency is needed for dlangui, as well as D runtime for android
|
||||
|
||||
1\) Build D runtime with LDC
|
||||
(should this be done inside LDC install folder?)
|
||||
|
||||
For this and the next tasks ninja and cmake build systems is needed,
|
||||
refer to https://wiki.dlang.org/Build_D_for_Android for more info
|
||||
|
||||
|
||||
```
|
||||
set CC %ANDROID_HOME%\ndk-bundle\prebuilt\windows-x86_64\bin\clang.exe
|
||||
ldc-build-runtime --ninja --targetPreset=Android-arm --dFlags="-w;-mcpu=cortex-a8" --buildDir=droid32
|
||||
```
|
||||
|
||||
2\) Download and build FreeType
|
||||
|
||||
on Windows just run ./get_ft.ps1 && ./build_ft.bat
|
||||
|
||||
currently there is no scripts for Linux so you have to adapt those manually
|
||||
|
||||
then copy the binaries to your project android libs/ folder
|
||||
|
||||
|
||||
Building
|
||||
========
|
||||
|
||||
|
|
|
@ -0,0 +1,7 @@
|
|||
cd freetype-2.9.1
|
||||
mkdir build-arm
|
||||
cd build-arm
|
||||
set CC=%NDK%\prebuilt\windows-x86_64\bin\clang
|
||||
set NDK=%ANDROID_HOME%\ndk-bundle
|
||||
cmake .. -G"Ninja" -DCMAKE_TOOLCHAIN_FILE=%NDK%\build\cmake\android.toolchain.cmake -DCMAKE_SYSTEM_NAME="Android" -DANDROID_NDK=%NDK% -DANDROID_TOOLCHAIN=clang -DANDROID_PLATFORM=android-21 -DBUILD_SHARED_LIBS=ON
|
||||
cmake --build . --config Release
|
|
@ -0,0 +1,4 @@
|
|||
wget https://download.savannah.gnu.org/releases/freetype/ft291.zip -OutFile ./ft291.zip
|
||||
Remove-Item -Path "freetype-2.9.1" -Recurse -Force
|
||||
Expand-Archive -Path ft291.zip -DestinationPath .
|
||||
cmd compile_ft.bat
|
|
@ -4,6 +4,7 @@
|
|||
|
||||
DLANGUI_SOURCES="\
|
||||
$DLANGUI_DIR/src/dlangui/platforms/android/androidapp.d \
|
||||
$DLANGUI_DIR/src/dlangui/platforms/android/imm.d \
|
||||
$DLANGUI_DIR/src/dlangui/platforms/common/startup.d \
|
||||
$DLANGUI_DIR/src/dlangui/platforms/common/platform.d \
|
||||
$DLANGUI_DIR/src/dlangui/dialogs/filedlg.d \
|
||||
|
|
|
@ -0,0 +1,7 @@
|
|||
/local.properties
|
||||
/.idea
|
||||
/.gradle
|
||||
/build
|
||||
/*.iml
|
||||
/libs
|
||||
/app/*.iml
|
|
@ -3,9 +3,9 @@
|
|||
#=========================================================================
|
||||
# Modify this file to specify DLANGUI, Android NDK, SDK and LDC2 locations.
|
||||
|
||||
export DLANGUI_DIR=/media/toshiba/dev/ernomon
|
||||
export DLANGUI_DIR=$DLANGUI_DIR
|
||||
export NDK=$NDK
|
||||
export SDK=$ANDROID_HOME
|
||||
export LDC=$LDC
|
||||
export NDK_ARCH=x86_64
|
||||
export JAVA_HOME=/usr/lib/jvm/java-8-openjdk-amd64/
|
||||
export JAVA_HOME=$JAVA_HOME
|
||||
|
|
|
@ -1,138 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<module external.linked.project.id=":app" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$/.." external.system.id="GRADLE" type="JAVA_MODULE" version="4">
|
||||
<component name="FacetManager">
|
||||
<facet type="android-gradle" name="Android-Gradle">
|
||||
<configuration>
|
||||
<option name="GRADLE_PROJECT_PATH" value=":app" />
|
||||
</configuration>
|
||||
</facet>
|
||||
<facet type="android" name="Android">
|
||||
<configuration>
|
||||
<option name="SELECTED_BUILD_VARIANT" value="debug" />
|
||||
<option name="ASSEMBLE_TASK_NAME" value="assembleDebug" />
|
||||
<option name="COMPILE_JAVA_TASK_NAME" value="compileDebugSources" />
|
||||
<afterSyncTasks>
|
||||
<task>generateDebugSources</task>
|
||||
</afterSyncTasks>
|
||||
<option name="ALLOW_USER_CONFIGURATION" value="false" />
|
||||
<option name="MANIFEST_FILE_RELATIVE_PATH" value="/src/main/AndroidManifest.xml" />
|
||||
<option name="RES_FOLDER_RELATIVE_PATH" value="/src/main/res" />
|
||||
<option name="RES_FOLDERS_RELATIVE_PATH" value="file://$MODULE_DIR$/src/main/res" />
|
||||
<option name="ASSETS_FOLDER_RELATIVE_PATH" value="/src/main/assets" />
|
||||
</configuration>
|
||||
</facet>
|
||||
</component>
|
||||
<component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_7">
|
||||
<output url="file://$MODULE_DIR$/build/intermediates/classes/debug" />
|
||||
<output-test url="file://$MODULE_DIR$/build/intermediates/classes/test/debug" />
|
||||
<exclude-output />
|
||||
<content url="file://$MODULE_DIR$">
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/apt/debug" isTestSource="false" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/r/debug" isTestSource="false" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/aidl/debug" isTestSource="false" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/debug" isTestSource="false" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/rs/debug" isTestSource="false" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/debug" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/resValues/debug" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/apt/androidTest/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/r/androidTest/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/aidl/androidTest/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/buildConfig/androidTest/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/rs/androidTest/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/rs/androidTest/debug" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/res/resValues/androidTest/debug" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/build/generated/source/apt/test/debug" isTestSource="true" generated="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/res" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/resources" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/assets" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/aidl" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/java" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/rs" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/debug/shaders" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/res" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/resources" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/assets" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/aidl" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/java" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/rs" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/testDebug/shaders" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/res" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/resources" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/assets" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/aidl" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/java" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/rs" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTestDebug/shaders" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/res" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/resources" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/assets" type="java-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/aidl" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/java" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/rs" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/main/shaders" isTestSource="false" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/res" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/resources" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/assets" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/aidl" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/java" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/rs" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/androidTest/shaders" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/res" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/resources" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/assets" type="java-test-resource" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/aidl" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/java" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/rs" isTestSource="true" />
|
||||
<sourceFolder url="file://$MODULE_DIR$/src/test/shaders" isTestSource="true" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/assets" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/blame" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/check-manifest" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/classes" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/classes-jar" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/incremental" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/javaPrecompile" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/jniLibs" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/lint" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/manifests" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/prebuild" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/res" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/rs" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/shaders" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/splits-support" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/symbols" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/tmp" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/intermediates/transforms" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/outputs" />
|
||||
<excludeFolder url="file://$MODULE_DIR$/build/tmp" />
|
||||
</content>
|
||||
<orderEntry type="jdk" jdkName="Android API 26 Platform" jdkType="Android SDK" />
|
||||
<orderEntry type="sourceFolder" forTests="false" />
|
||||
<orderEntry type="library" name="com.android.support:support-media-compat-26.1.0" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="junit:junit:4.12@jar" level="project" />
|
||||
<orderEntry type="library" name="android.arch.lifecycle:common:1.0.0@jar" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.android.support.test.espresso:espresso-core-3.0.1" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-fragment-26.1.0" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="javax.inject:javax.inject:1@jar" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.squareup:javawriter:2.1.1@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-vector-drawable-26.1.0" level="project" />
|
||||
<orderEntry type="library" name="android.arch.lifecycle:runtime-1.0.0" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:appcompat-v7-26.1.0" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-annotations:26.1.0@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support.constraint:constraint-layout-solver:1.0.2@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-core-utils-26.1.0" level="project" />
|
||||
<orderEntry type="library" name="com.android.support.constraint:constraint-layout-1.0.2" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-core-ui-26.1.0" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.android.support.test:runner-1.0.1" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.android.support.test:rules-1.0.1" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.google.code.findbugs:jsr305:2.0.1@jar" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="com.android.support.test.espresso:espresso-idling-resource-3.0.1" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="org.hamcrest:hamcrest-core:1.3@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-compat-26.1.0" level="project" />
|
||||
<orderEntry type="library" name="android.arch.core:common:1.0.0@jar" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="org.hamcrest:hamcrest-library:1.3@jar" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="org.hamcrest:hamcrest-integration:1.3@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:support-v4-26.1.0" level="project" />
|
||||
<orderEntry type="library" scope="TEST" name="net.sf.kxml:kxml2:2.3.0@jar" level="project" />
|
||||
<orderEntry type="library" name="com.android.support:animated-vector-drawable-26.1.0" level="project" />
|
||||
</component>
|
||||
</module>
|
|
@ -38,7 +38,7 @@ dependencies {
|
|||
implementation fileTree(dir: 'libs', include: ['*.jar'])
|
||||
implementation 'com.android.support:appcompat-v7:26.1.0'
|
||||
implementation 'com.android.support.constraint:constraint-layout:1.0.2'
|
||||
testImplementation 'junit:junit:4.12'
|
||||
androidTestImplementation 'com.android.support.test:runner:1.0.1'
|
||||
androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.1'
|
||||
//testImplementation 'junit:junit:4.12'
|
||||
//androidTestImplementation 'com.android.support.test:runner:1.0.1'
|
||||
//androidTestImplementation 'com.android.support.test.espresso:espresso-core:3.0.1'
|
||||
}
|
||||
|
|
|
@ -7,7 +7,7 @@ buildscript {
|
|||
jcenter()
|
||||
}
|
||||
dependencies {
|
||||
classpath 'com.android.tools.build:gradle:3.0.0'
|
||||
classpath 'com.android.tools.build:gradle:3.3.1'
|
||||
|
||||
|
||||
// NOTE: Do not place your application dependencies here; they belong
|
||||
|
|
File diff suppressed because one or more lines are too long
Binary file not shown.
|
@ -35,8 +35,15 @@ die () {
|
|||
#=========================================================
|
||||
#$SDK/tools/android update project -p . -s --target $ANDROID_TARGET || die 3 "Android Project update is failed"
|
||||
|
||||
# This is not necessary, even in the docs gradle recommend to build with wrapper but I find it annoying to
|
||||
# download all that versions for all projects, so that's what this check is for
|
||||
GRADLE=gradlew
|
||||
if gradle --version >/dev/null 2>&1; then
|
||||
GRADLE=gradle
|
||||
fi
|
||||
|
||||
echo "Building APK..."
|
||||
#=========================================================
|
||||
./gradlew assembleDebug || die 4 "Android APK creation is failed"
|
||||
GRADLE assembleDebug || die 4 "Android APK creation is failed"
|
||||
|
||||
echo "Successful."
|
||||
|
|
|
@ -1,19 +0,0 @@
|
|||
<?xml version="1.0" encoding="UTF-8"?>
|
||||
<module external.linked.project.id="example" external.linked.project.path="$MODULE_DIR$" external.root.project.path="$MODULE_DIR$" external.system.id="GRADLE" type="JAVA_MODULE" version="4">
|
||||
<component name="FacetManager">
|
||||
<facet type="java-gradle" name="Java-Gradle">
|
||||
<configuration>
|
||||
<option name="BUILD_FOLDER_PATH" value="$MODULE_DIR$/build" />
|
||||
<option name="BUILDABLE" value="false" />
|
||||
</configuration>
|
||||
</facet>
|
||||
</component>
|
||||
<component name="NewModuleRootManager" LANGUAGE_LEVEL="JDK_1_7" inherit-compiler-output="true">
|
||||
<exclude-output />
|
||||
<content url="file://$MODULE_DIR$">
|
||||
<excludeFolder url="file://$MODULE_DIR$/.gradle" />
|
||||
</content>
|
||||
<orderEntry type="jdk" jdkName="JDK" jdkType="JavaSDK" />
|
||||
<orderEntry type="sourceFolder" forTests="false" />
|
||||
</component>
|
||||
</module>
|
Binary file not shown.
|
@ -1,6 +1,5 @@
|
|||
#Mon Nov 06 21:19:18 EET 2017
|
||||
distributionBase=GRADLE_USER_HOME
|
||||
distributionPath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-5.1.1-bin.zip
|
||||
zipStoreBase=GRADLE_USER_HOME
|
||||
zipStorePath=wrapper/dists
|
||||
distributionUrl=https\://services.gradle.org/distributions/gradle-4.1-all.zip
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
#!/usr/bin/env bash
|
||||
#!/usr/bin/env sh
|
||||
|
||||
##############################################################################
|
||||
##
|
||||
|
@ -6,42 +6,6 @@
|
|||
##
|
||||
##############################################################################
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS=""
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=`basename "$0"`
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD="maximum"
|
||||
|
||||
warn ( ) {
|
||||
echo "$*"
|
||||
}
|
||||
|
||||
die ( ) {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
}
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
case "`uname`" in
|
||||
CYGWIN* )
|
||||
cygwin=true
|
||||
;;
|
||||
Darwin* )
|
||||
darwin=true
|
||||
;;
|
||||
MINGW* )
|
||||
msys=true
|
||||
;;
|
||||
esac
|
||||
|
||||
# Attempt to set APP_HOME
|
||||
# Resolve links: $0 may be a link
|
||||
PRG="$0"
|
||||
|
@ -60,6 +24,46 @@ cd "`dirname \"$PRG\"`/" >/dev/null
|
|||
APP_HOME="`pwd -P`"
|
||||
cd "$SAVED" >/dev/null
|
||||
|
||||
APP_NAME="Gradle"
|
||||
APP_BASE_NAME=`basename "$0"`
|
||||
|
||||
# Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
DEFAULT_JVM_OPTS='"-Xmx64m"'
|
||||
|
||||
# Use the maximum available, or set MAX_FD != -1 to use that value.
|
||||
MAX_FD="maximum"
|
||||
|
||||
warn () {
|
||||
echo "$*"
|
||||
}
|
||||
|
||||
die () {
|
||||
echo
|
||||
echo "$*"
|
||||
echo
|
||||
exit 1
|
||||
}
|
||||
|
||||
# OS specific support (must be 'true' or 'false').
|
||||
cygwin=false
|
||||
msys=false
|
||||
darwin=false
|
||||
nonstop=false
|
||||
case "`uname`" in
|
||||
CYGWIN* )
|
||||
cygwin=true
|
||||
;;
|
||||
Darwin* )
|
||||
darwin=true
|
||||
;;
|
||||
MINGW* )
|
||||
msys=true
|
||||
;;
|
||||
NONSTOP* )
|
||||
nonstop=true
|
||||
;;
|
||||
esac
|
||||
|
||||
CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
|
||||
|
||||
# Determine the Java command to use to start the JVM.
|
||||
|
@ -85,7 +89,7 @@ location of your Java installation."
|
|||
fi
|
||||
|
||||
# Increase the maximum file descriptors if we can.
|
||||
if [ "$cygwin" = "false" -a "$darwin" = "false" ] ; then
|
||||
if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
|
||||
MAX_FD_LIMIT=`ulimit -H -n`
|
||||
if [ $? -eq 0 ] ; then
|
||||
if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
|
||||
|
@ -150,11 +154,19 @@ if $cygwin ; then
|
|||
esac
|
||||
fi
|
||||
|
||||
# Split up the JVM_OPTS And GRADLE_OPTS values into an array, following the shell quoting and substitution rules
|
||||
function splitJvmOpts() {
|
||||
JVM_OPTS=("$@")
|
||||
# Escape application args
|
||||
save () {
|
||||
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||
echo " "
|
||||
}
|
||||
eval splitJvmOpts $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS
|
||||
JVM_OPTS[${#JVM_OPTS[*]}]="-Dorg.gradle.appname=$APP_BASE_NAME"
|
||||
APP_ARGS=$(save "$@")
|
||||
|
||||
exec "$JAVACMD" "${JVM_OPTS[@]}" -classpath "$CLASSPATH" org.gradle.wrapper.GradleWrapperMain "$@"
|
||||
# Collect all arguments for the java command, following the shell quoting and substitution rules
|
||||
eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
|
||||
|
||||
# by default we should be in the correct project dir, but when run from Finder on Mac, the cwd is wrong
|
||||
if [ "$(uname)" = "Darwin" ] && [ "$HOME" = "$PWD" ]; then
|
||||
cd "$(dirname "$0")"
|
||||
fi
|
||||
|
||||
exec "$JAVACMD" "$@"
|
||||
|
|
|
@ -8,14 +8,14 @@
|
|||
@rem Set local scope for the variables with windows NT shell
|
||||
if "%OS%"=="Windows_NT" setlocal
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS=
|
||||
|
||||
set DIRNAME=%~dp0
|
||||
if "%DIRNAME%" == "" set DIRNAME=.
|
||||
set APP_BASE_NAME=%~n0
|
||||
set APP_HOME=%DIRNAME%
|
||||
|
||||
@rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
|
||||
set DEFAULT_JVM_OPTS="-Xmx64m"
|
||||
|
||||
@rem Find java.exe
|
||||
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||
|
||||
|
@ -46,10 +46,9 @@ echo location of your Java installation.
|
|||
goto fail
|
||||
|
||||
:init
|
||||
@rem Get command-line arguments, handling Windowz variants
|
||||
@rem Get command-line arguments, handling Windows variants
|
||||
|
||||
if not "%OS%" == "Windows_NT" goto win9xME_args
|
||||
if "%@eval[2+2]" == "4" goto 4NT_args
|
||||
|
||||
:win9xME_args
|
||||
@rem Slurp the command line arguments.
|
||||
|
@ -60,11 +59,6 @@ set _SKIP=2
|
|||
if "x%~1" == "x" goto execute
|
||||
|
||||
set CMD_LINE_ARGS=%*
|
||||
goto execute
|
||||
|
||||
:4NT_args
|
||||
@rem Get arguments from the 4NT Shell from JP Software
|
||||
set CMD_LINE_ARGS=%$
|
||||
|
||||
:execute
|
||||
@rem Setup the command line
|
||||
|
|
|
@ -1,12 +0,0 @@
|
|||
## This file is automatically generated by Android Studio.
|
||||
# Do not modify this file -- YOUR CHANGES WILL BE ERASED!
|
||||
#
|
||||
# This file must *NOT* be checked into Version Control Systems,
|
||||
# as it contains information specific to your local configuration.
|
||||
#
|
||||
# Location of the SDK. This is only used by Gradle.
|
||||
# For customization when using a Version Control System, please read the
|
||||
# header note.
|
||||
#Mon Nov 06 21:25:18 EET 2017
|
||||
ndk.dir=/media/erno/TOSHIBA_ISO/Android/Sdk/ndk-bundle
|
||||
sdk.dir=/media/erno/TOSHIBA_ISO/Android/Sdk
|
|
@ -16,6 +16,7 @@ import dlangui.platforms.common.platform;
|
|||
import android.input, android.looper : ALooper_pollAll;
|
||||
import android.native_window : ANativeWindow_setBuffersGeometry;
|
||||
import android.configuration;
|
||||
import android.keycodes;
|
||||
import android.log, android.android_native_app_glue;
|
||||
|
||||
/**
|
||||
|
@ -139,6 +140,8 @@ class AndroidWindow : Window {
|
|||
* Process the next input event.
|
||||
*/
|
||||
int handle_input(AInputEvent* event) {
|
||||
import imm = dlangui.platforms.android.imm;
|
||||
import std.conv : to;
|
||||
Log.i("handle input, event=", AInputEvent_getType(event));
|
||||
auto et = AInputEvent_getType(event);
|
||||
if (et == AINPUT_EVENT_TYPE_MOTION) {
|
||||
|
@ -173,7 +176,37 @@ class AndroidWindow : Window {
|
|||
return 1;
|
||||
} else if (et == AINPUT_EVENT_TYPE_KEY) {
|
||||
Log.d("AINPUT_EVENT_TYPE_KEY");
|
||||
return 0;
|
||||
KeyEvent evt;
|
||||
auto app = (cast(AndroidPlatform)platform)._appstate;
|
||||
int _keyFlags = AKeyEvent_getMetaState(event).toKeyFlag();
|
||||
int sysKeyCode = AKeyEvent_getKeyCode(event);
|
||||
int sysMeta = AKeyEvent_getMetaState(event);
|
||||
int keyCode = androidKeyMap.get(sysKeyCode, KeyCode.init);
|
||||
auto action = toKeyAction(AKeyEvent_getAction(event));
|
||||
int char_ = imm.GetUnicodeChar(app, action, sysKeyCode, sysMeta);
|
||||
dchar[] text;
|
||||
if (!isTextEditControl(sysKeyCode)) {
|
||||
if (AKeyEvent_getAction(event) == AKEY_EVENT_ACTION_MULTIPLE) {
|
||||
// it's a string from IME
|
||||
if (sysKeyCode == AKEYCODE_UNKNOWN) {
|
||||
text = cast(dchar[]) to!dstring(imm.GetUnicodeString(app, event));
|
||||
action = KeyAction.Text;
|
||||
}
|
||||
// else repeat character AKeyEvent_getRepeatCount() times
|
||||
}
|
||||
else if (AKeyEvent_getAction(event) == AKEY_EVENT_ACTION_DOWN && (char_ || isASCIIChar(sysKeyCode))) {
|
||||
text ~= cast(dchar)(char_ == 0 ? sysKeyCode : char_);
|
||||
action = KeyAction.Text;
|
||||
}
|
||||
}
|
||||
Log.d("ACTION: ", action, " syskeyCode: ", sysKeyCode, " sysMeta: ", sysMeta, "meta: ", _keyFlags, " char '", cast(dchar)char_, "' str:", cast(dstring)text);
|
||||
if (action == KeyAction.Text)
|
||||
evt = new KeyEvent(KeyAction.Text, 0, 0, cast(dstring)text);
|
||||
else
|
||||
evt = new KeyEvent(action, keyCode, _keyFlags);
|
||||
if (evt && dispatchKeyEvent(evt))
|
||||
update();
|
||||
return 1;
|
||||
}
|
||||
return 0;
|
||||
}
|
||||
|
@ -225,6 +258,12 @@ class AndroidPlatform : Platform {
|
|||
}
|
||||
|
||||
|
||||
void showSoftKeyboard(bool shouldShow) {
|
||||
import imm = dlangui.platforms.android.imm;
|
||||
imm.showSoftKeyboard(_appstate, shouldShow);
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* Initialize an EGL context for the current display.
|
||||
*/
|
||||
|
@ -706,3 +745,82 @@ extern (C) void android_main(android_app* state) {
|
|||
|
||||
}
|
||||
|
||||
|
||||
private KeyAction toKeyAction(int androidKeyAction) {
|
||||
switch(androidKeyAction) {
|
||||
case AKEY_EVENT_ACTION_DOWN: return KeyAction.KeyDown;
|
||||
case AKEY_EVENT_ACTION_UP: return KeyAction.KeyUp;
|
||||
case AKEY_EVENT_ACTION_MULTIPLE: return KeyAction.Repeat; // can also be text
|
||||
default:
|
||||
assert(0, "should never reach this");
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
private bool isASCIIChar(int ch) {
|
||||
return 31 < ch && ch < 127;
|
||||
}
|
||||
|
||||
// Text editor controls such as move caret or (de)select
|
||||
private bool isTextEditControl(int keyCode) {
|
||||
switch (keyCode){
|
||||
case AKEYCODE_DEL: // backspace ("DEL" or "<x" arrow on soft keyboard)
|
||||
case 112: // delete
|
||||
case 59: // lshift
|
||||
case 60: // rshift
|
||||
case 113: // lcontrol
|
||||
case 114: // rcontrol
|
||||
case 57: // lalt
|
||||
case 58: // ralt
|
||||
case 19: // up arrow
|
||||
case 20: // down arrow
|
||||
case 21: // left arrow
|
||||
case 22: // right arrow
|
||||
case 92: // page up?
|
||||
case 93: // page down?
|
||||
case 122: // home
|
||||
case 123: // end
|
||||
case 124: // insert
|
||||
return true;
|
||||
static foreach(fn; 131..143) // f1-f12
|
||||
case fn: return true;
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private int toKeyFlag(int keyMeta) {
|
||||
int state;
|
||||
if (keyMeta & AMETA_ALT_ON) state |= KeyFlag.Alt;
|
||||
if (keyMeta & AMETA_SHIFT_ON) state |= KeyFlag.Shift;
|
||||
return state;
|
||||
}
|
||||
|
||||
|
||||
|
||||
/// Android to dlangui key mapping
|
||||
private static immutable KeyCode[int] androidKeyMap;
|
||||
|
||||
static this() {
|
||||
import std.conv : text;
|
||||
androidKeyMap = [
|
||||
AKEYCODE_DEL: KeyCode.BACK, // Delete(key code 67) on Android seems to work as backspace(key 112)
|
||||
AKEYCODE_BACK: KeyCode.BACK,
|
||||
AKEYCODE_SPACE: KeyCode.SPACE,
|
||||
AKEYCODE_ENTER: KeyCode.RETURN,
|
||||
AKEYCODE_TAB: KeyCode.TAB,
|
||||
AKEYCODE_DPAD_LEFT: KeyCode.LEFT,
|
||||
AKEYCODE_DPAD_RIGHT: KeyCode.RIGHT,
|
||||
AKEYCODE_DPAD_UP: KeyCode.UP,
|
||||
AKEYCODE_DPAD_DOWN: KeyCode.DOWN,
|
||||
AKEYCODE_PAGE_UP: KeyCode.PAGEUP,
|
||||
AKEYCODE_PAGE_DOWN: KeyCode.PAGEDOWN,
|
||||
112: KeyCode.DEL,
|
||||
122: KeyCode.HOME,
|
||||
123: KeyCode.END,
|
||||
];
|
||||
static foreach(n; 0..10) // keys 0-9
|
||||
androidKeyMap[mixin("AKEYCODE_" ~ n.text)] = mixin("KeyCode.KEY_" ~ n.text);
|
||||
static foreach(char n; 'A'..'Z'+1) // A-Z
|
||||
androidKeyMap[mixin("AKEYCODE_" ~ n)] = mixin("KeyCode.KEY_" ~ n);
|
||||
}
|
|
@ -0,0 +1,221 @@
|
|||
module dlangui.platforms.android.imm;
|
||||
|
||||
version(Android):
|
||||
|
||||
import jni;
|
||||
import android.android_native_app_glue;
|
||||
import android.input;
|
||||
|
||||
import dlangui.core.logger;
|
||||
|
||||
|
||||
alias IMMResult = int;
|
||||
// values from InputMethodManager.java
|
||||
private enum : IMMResult
|
||||
{
|
||||
RESULT_UNCHANGED_SHOWN = 0,
|
||||
RESULT_UNCHANGED_HIDDEN = 1,
|
||||
RESULT_SHOWN = 2,
|
||||
RESULT_HIDDEN = 3,
|
||||
}
|
||||
|
||||
|
||||
alias IMMFlags = int;
|
||||
// values from InputMethodManager.java
|
||||
private enum : IMMFlags
|
||||
{
|
||||
SHOW_IMPLICIT = 0x0001,
|
||||
SHOW_FORCED = 0x0002,
|
||||
|
||||
HIDE_IMPLICIT_ONLY = 0x0001,
|
||||
HIDE_NOT_ALWAYS = 0x0002,
|
||||
}
|
||||
|
||||
|
||||
|
||||
/**
|
||||
* JNI wrapper used with native actitiy to show/hide software keyboard
|
||||
* It relies on java reflection and it might be slow.
|
||||
*/
|
||||
void showSoftKeyboard(android_app* app, bool shouldShow)
|
||||
{
|
||||
// The code is based on https://stackoverflow.com/questions/5864790/how-to-show-the-soft-keyboard-on-native-activity
|
||||
// Attaches the current thread to the JVM.
|
||||
jint result;
|
||||
IMMFlags flags;
|
||||
|
||||
auto javaVM = app.activity.vm;
|
||||
auto env = app.activity.env;
|
||||
|
||||
JavaVMAttachArgs attachArgs;
|
||||
attachArgs.version_ = JNI_VERSION_1_6;
|
||||
attachArgs.name = "NativeThread";
|
||||
attachArgs.group = null;
|
||||
|
||||
if ((*javaVM).AttachCurrentThread(javaVM, &env, &attachArgs) == JNI_ERR)
|
||||
{
|
||||
Log.e("showSoftKeyboard Unable to attach to JVM");
|
||||
return;
|
||||
}
|
||||
|
||||
// Retrieves NativeActivity.
|
||||
jobject nativeActivity = app.activity.clazz;
|
||||
jclass nativeActivityClass = (*env).GetObjectClass(env, nativeActivity);
|
||||
|
||||
// Retrieves Context.INPUT_METHOD_SERVICE.
|
||||
jclass contextClass = (*env).FindClass(env, "android/content/Context");
|
||||
jfieldID FieldINPUT_METHOD_SERVICE =
|
||||
(*env).GetStaticFieldID(env, contextClass,
|
||||
"INPUT_METHOD_SERVICE", "Ljava/lang/String;");
|
||||
jobject INPUT_METHOD_SERVICE =
|
||||
(*env).GetStaticObjectField(env, contextClass, FieldINPUT_METHOD_SERVICE);
|
||||
//jniCheck(INPUT_METHOD_SERVICE);
|
||||
|
||||
// Runs getSystemService(Context.INPUT_METHOD_SERVICE).
|
||||
jclass immClass = (*env).FindClass(
|
||||
env, "android/view/inputmethod/InputMethodManager");
|
||||
jmethodID MethodGetSystemService = (*env).GetMethodID(
|
||||
env, nativeActivityClass, "getSystemService",
|
||||
"(Ljava/lang/String;)Ljava/lang/Object;");
|
||||
jobject imm = (*env).CallObjectMethod(
|
||||
env, nativeActivity, MethodGetSystemService,
|
||||
INPUT_METHOD_SERVICE);
|
||||
|
||||
// Runs getWindow().getDecorView().
|
||||
jmethodID MethodGetWindow = (*env).GetMethodID(
|
||||
env, nativeActivityClass, "getWindow", "()Landroid/view/Window;");
|
||||
jobject window = (*env).CallObjectMethod(
|
||||
env, nativeActivity, MethodGetWindow);
|
||||
jclass windowClass = (*env).FindClass(
|
||||
env, "android/view/Window");
|
||||
jmethodID MethodGetDecorView = (*env).GetMethodID(
|
||||
env, windowClass, "getDecorView", "()Landroid/view/View;");
|
||||
jobject decorView = (*env).CallObjectMethod(
|
||||
env, window, MethodGetDecorView);
|
||||
|
||||
if (shouldShow) {
|
||||
// Runs imm.showSoftInput(...).
|
||||
jmethodID MethodShowSoftInput = (*env).GetMethodID(
|
||||
env, immClass, "showSoftInput", "(Landroid/view/View;I)Z");
|
||||
jboolean res = (*env).CallBooleanMethod(
|
||||
env, imm, MethodShowSoftInput, decorView, flags);
|
||||
} else {
|
||||
// Runs lWindow.getViewToken()
|
||||
jclass viewClass = (*env).FindClass(
|
||||
env, "android/view/View");
|
||||
jmethodID MethodGetWindowToken = (*env).GetMethodID(
|
||||
env, viewClass, "getWindowToken", "()Landroid/os/IBinder;");
|
||||
jobject binder = (*env).CallObjectMethod(
|
||||
env, decorView, MethodGetWindowToken);
|
||||
|
||||
// lInputMethodManager.hideSoftInput(...).
|
||||
jmethodID MethodHideSoftInput = (*env).GetMethodID(
|
||||
env, immClass, "hideSoftInputFromWindow",
|
||||
"(Landroid/os/IBinder;I)Z");
|
||||
jboolean res = (*env).CallBooleanMethod(
|
||||
env, imm, MethodHideSoftInput, binder, flags);
|
||||
}
|
||||
|
||||
// Finished with the JVM.
|
||||
(*javaVM).DetachCurrentThread(javaVM);
|
||||
}
|
||||
|
||||
|
||||
int GetUnicodeChar(android_app* app, int eventType, int keyCode, int metaState)
|
||||
{
|
||||
auto javaVM = app.activity.vm;
|
||||
auto env = app.activity.env;
|
||||
|
||||
JavaVMAttachArgs attachArgs;
|
||||
attachArgs.version_ = JNI_VERSION_1_6;
|
||||
attachArgs.name = "NativeThread";
|
||||
attachArgs.group = null;
|
||||
|
||||
if ((*javaVM).AttachCurrentThread(javaVM, &env, &attachArgs) == JNI_ERR)
|
||||
return 0;
|
||||
|
||||
jclass class_key_event = (*env).FindClass(env, "android/view/KeyEvent");
|
||||
int unicodeKey;
|
||||
|
||||
if(metaState == 0)
|
||||
{
|
||||
jmethodID method_get_unicode_char = (*env).GetMethodID(env, class_key_event, "getUnicodeChar", "()I");
|
||||
jmethodID eventConstructor = (*env).GetMethodID(env, class_key_event, "<init>", "(II)V");
|
||||
jobject eventObj = (*env).NewObject(env, class_key_event, eventConstructor, eventType, keyCode);
|
||||
|
||||
unicodeKey = (*env).CallIntMethod(env, eventObj, method_get_unicode_char);
|
||||
}
|
||||
else
|
||||
{
|
||||
jmethodID method_get_unicode_char = (*env).GetMethodID(env, class_key_event, "getUnicodeChar", "(I)I");
|
||||
jmethodID eventConstructor = (*env).GetMethodID(env, class_key_event, "<init>", "(II)V");
|
||||
jobject eventObj = (*env).NewObject(env, class_key_event, eventConstructor, eventType, keyCode);
|
||||
|
||||
unicodeKey = (*env).CallIntMethod(env, eventObj, method_get_unicode_char, metaState);
|
||||
}
|
||||
|
||||
(*javaVM).DetachCurrentThread(javaVM);
|
||||
|
||||
return unicodeKey;
|
||||
}
|
||||
|
||||
|
||||
// Issue: native app glue seems to mess up the input.
|
||||
// It is clearly seen in debugger that initally key event do have real input,
|
||||
// but second time it is called it is all messed up
|
||||
string GetUnicodeString(android_app* app, AInputEvent* event)
|
||||
{
|
||||
string str;
|
||||
auto javaVM = app.activity.vm;
|
||||
auto env = app.activity.env;
|
||||
|
||||
JavaVMAttachArgs attachArgs;
|
||||
attachArgs.version_ = JNI_VERSION_1_6;
|
||||
attachArgs.name = "NativeThread";
|
||||
attachArgs.group = null;
|
||||
|
||||
if ((*javaVM).AttachCurrentThread(javaVM, &env, &attachArgs) == JNI_ERR)
|
||||
{
|
||||
Log.e("showSoftKeyboard Unable to attach to JVM");
|
||||
return null;
|
||||
}
|
||||
|
||||
|
||||
jclass class_key_event = (*env).FindClass(env, "android/view/KeyEvent");
|
||||
|
||||
jmethodID eventConstructor = (*env).GetMethodID(env, class_key_event, "<init>", "(JJIIIIIIII)V");
|
||||
jobject eventObj = (*env).NewObject(env, class_key_event, eventConstructor,
|
||||
AKeyEvent_getDownTime(event),
|
||||
AKeyEvent_getEventTime(event),
|
||||
AKeyEvent_getAction(event),
|
||||
AKeyEvent_getKeyCode(event),
|
||||
AKeyEvent_getRepeatCount(event),
|
||||
AKeyEvent_getMetaState(event),
|
||||
AInputEvent_getDeviceId(event),
|
||||
AKeyEvent_getScanCode(event),
|
||||
AKeyEvent_getFlags(event),
|
||||
AInputEvent_getSource(event)
|
||||
);
|
||||
|
||||
// this won't work because characters is a member passed on construction and getCharacter() is just a getter
|
||||
jmethodID method_get_characters = (*env).GetMethodID(env, class_key_event, "getCharacters", "()Ljava/lang/String;");
|
||||
if (auto jstr = (*env).CallObjectMethod(env, eventObj, method_get_characters)) {
|
||||
str.length = (*env).GetStringUTFLength(env, jstr);
|
||||
(*env).GetStringUTFRegion(env, jstr, 0, str.length, cast(char*)str.ptr);
|
||||
}
|
||||
|
||||
{
|
||||
jmethodID method_get_unicode_char = (*env).GetMethodID(env, class_key_event, "getUnicodeChar", "()I");
|
||||
int unicodeKey = (*env).CallIntMethod(env, eventObj, method_get_unicode_char);
|
||||
if (str.length == 0) {
|
||||
import std.conv : to;
|
||||
dchar[] tmp;
|
||||
tmp ~= unicodeKey;
|
||||
str = to!string(tmp);
|
||||
}
|
||||
}
|
||||
|
||||
(*javaVM).DetachCurrentThread(javaVM);
|
||||
|
||||
return str;
|
||||
}
|
|
@ -345,8 +345,10 @@ class EditWidgetBase : ScrollWidgetBase, EditableContentListener, MenuItemAction
|
|||
/// sets focus to this widget or suitable focusable child, returns previously focused widget
|
||||
override Widget setFocus(FocusReason reason = FocusReason.Unspecified) {
|
||||
Widget res = super.setFocus(reason);
|
||||
if (focused)
|
||||
if (focused) {
|
||||
showSoftKeyboard();
|
||||
handleEditorStateChange();
|
||||
}
|
||||
return res;
|
||||
}
|
||||
|
||||
|
|
|
@ -1099,6 +1099,7 @@ public:
|
|||
// try to find focusable child
|
||||
return window.focusedWidget;
|
||||
}
|
||||
hideSoftKeyboard();
|
||||
return window.setFocus(this, reason);
|
||||
}
|
||||
/// searches children for first focusable item, returns null if not found
|
||||
|
@ -1116,6 +1117,24 @@ public:
|
|||
return null;
|
||||
}
|
||||
|
||||
///
|
||||
final void hideSoftKeyboard() {
|
||||
version(Android) {
|
||||
import dlangui.platforms.android.androidapp;
|
||||
if (auto androidPlatform = cast(AndroidPlatform)platform)
|
||||
androidPlatform.showSoftKeyboard(false);
|
||||
}
|
||||
}
|
||||
|
||||
/// Shows system virtual keyabord where applicable
|
||||
final void showSoftKeyboard() {
|
||||
version(Android) {
|
||||
import dlangui.platforms.android.androidapp;
|
||||
if (auto androidPlatform = cast(AndroidPlatform)platform)
|
||||
androidPlatform.showSoftKeyboard(true);
|
||||
}
|
||||
}
|
||||
|
||||
// =======================================================
|
||||
// Events
|
||||
|
||||
|
|
Loading…
Reference in New Issue