Compare commits
No commits in common. "cf655625dfad90617a1f35f7145637c4eece45d9" and "0b39a44220985dd12a03a7e45f2d39db55c19135" have entirely different histories.
cf655625df
...
0b39a44220
@ -35,7 +35,7 @@
|
|||||||
<PackageVersion Include="Ryujinx.Graphics.Nvdec.Dependencies" Version="5.0.1-build13" />
|
<PackageVersion Include="Ryujinx.Graphics.Nvdec.Dependencies" Version="5.0.1-build13" />
|
||||||
<PackageVersion Include="Ryujinx.Graphics.Vulkan.Dependencies.MoltenVK" Version="1.2.3" />
|
<PackageVersion Include="Ryujinx.Graphics.Vulkan.Dependencies.MoltenVK" Version="1.2.3" />
|
||||||
<PackageVersion Include="Ryujinx.GtkSharp" Version="3.24.24.59-ryujinx" />
|
<PackageVersion Include="Ryujinx.GtkSharp" Version="3.24.24.59-ryujinx" />
|
||||||
<PackageVersion Include="Ryujinx.SDL2-CS" Version="2.30.0-build32" />
|
<PackageVersion Include="Ryujinx.SDL2-CS" Version="2.28.1-build28" />
|
||||||
<PackageVersion Include="securifybv.ShellLink" Version="0.1.0" />
|
<PackageVersion Include="securifybv.ShellLink" Version="0.1.0" />
|
||||||
<PackageVersion Include="shaderc.net" Version="0.1.0" />
|
<PackageVersion Include="shaderc.net" Version="0.1.0" />
|
||||||
<PackageVersion Include="SharpZipLib" Version="1.4.2" />
|
<PackageVersion Include="SharpZipLib" Version="1.4.2" />
|
||||||
|
@ -2,7 +2,6 @@
|
|||||||
using ARMeilleure.Memory;
|
using ARMeilleure.Memory;
|
||||||
using ARMeilleure.Translation;
|
using ARMeilleure.Translation;
|
||||||
using ARMeilleure.Translation.Cache;
|
using ARMeilleure.Translation.Cache;
|
||||||
using Ryujinx.Common;
|
|
||||||
using System;
|
using System;
|
||||||
using System.Runtime.CompilerServices;
|
using System.Runtime.CompilerServices;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
@ -113,16 +112,10 @@ namespace ARMeilleure.Signal
|
|||||||
|
|
||||||
ref SignalHandlerConfig config = ref GetConfigRef();
|
ref SignalHandlerConfig config = ref GetConfigRef();
|
||||||
|
|
||||||
if (OperatingSystem.IsLinux() || OperatingSystem.IsMacOS() || OperatingSystem.IsIOS() || PlatformInfo.IsBionic)
|
if (OperatingSystem.IsLinux() || OperatingSystem.IsMacOS() || OperatingSystem.IsIOS())
|
||||||
{
|
{
|
||||||
_signalHandlerPtr = Marshal.GetFunctionPointerForDelegate(GenerateUnixSignalHandler(_handlerConfig));
|
_signalHandlerPtr = Marshal.GetFunctionPointerForDelegate(GenerateUnixSignalHandler(_handlerConfig));
|
||||||
|
|
||||||
if (PlatformInfo.IsBionic)
|
|
||||||
{
|
|
||||||
config.StructAddressOffset = 16; // si_addr
|
|
||||||
config.StructWriteOffset = 8; // si_code
|
|
||||||
}
|
|
||||||
|
|
||||||
if (customSignalHandlerFactory != null)
|
if (customSignalHandlerFactory != null)
|
||||||
{
|
{
|
||||||
_signalHandlerPtr = customSignalHandlerFactory(UnixSignalHandlerRegistration.GetSegfaultExceptionHandler().sa_handler, _signalHandlerPtr);
|
_signalHandlerPtr = customSignalHandlerFactory(UnixSignalHandlerRegistration.GetSegfaultExceptionHandler().sa_handler, _signalHandlerPtr);
|
||||||
|
@ -1,6 +1,5 @@
|
|||||||
using System;
|
using System;
|
||||||
using System.Runtime.InteropServices;
|
using System.Runtime.InteropServices;
|
||||||
using System.Runtime.Versioning;
|
|
||||||
|
|
||||||
namespace ARMeilleure.Signal
|
namespace ARMeilleure.Signal
|
||||||
{
|
{
|
||||||
@ -21,25 +20,9 @@ namespace ARMeilleure.Signal
|
|||||||
public IntPtr sa_restorer;
|
public IntPtr sa_restorer;
|
||||||
}
|
}
|
||||||
|
|
||||||
[SupportedOSPlatform("android"), StructLayout(LayoutKind.Sequential, Pack = 8)]
|
|
||||||
public struct SigActionBionic
|
|
||||||
{
|
|
||||||
public int sa_flags;
|
|
||||||
public IntPtr sa_handler;
|
|
||||||
public SigSet sa_mask;
|
|
||||||
public IntPtr sa_restorer;
|
|
||||||
}
|
|
||||||
|
|
||||||
private const int SIGSEGV = 11;
|
private const int SIGSEGV = 11;
|
||||||
private const int SIGBUS = 10;
|
private const int SIGBUS = 10;
|
||||||
private const int SA_SIGINFO = 0x00000004;
|
private const int SA_SIGINFO = 0x00000004;
|
||||||
private const int SA_ONSTACK = 0x08000000;
|
|
||||||
|
|
||||||
[SupportedOSPlatform("android"), LibraryImport("libc", SetLastError = true)]
|
|
||||||
private static partial int sigaction(int signum, ref SigActionBionic sigAction, out SigActionBionic oldAction);
|
|
||||||
|
|
||||||
[SupportedOSPlatform("android"), LibraryImport("libc", SetLastError = true)]
|
|
||||||
private static partial int sigaction(int signum, IntPtr sigAction, out SigActionBionic oldAction);
|
|
||||||
|
|
||||||
[LibraryImport("libc", SetLastError = true)]
|
[LibraryImport("libc", SetLastError = true)]
|
||||||
private static partial int sigaction(int signum, ref SigAction sigAction, out SigAction oldAction);
|
private static partial int sigaction(int signum, ref SigAction sigAction, out SigAction oldAction);
|
||||||
@ -52,26 +35,7 @@ namespace ARMeilleure.Signal
|
|||||||
|
|
||||||
public static SigAction GetSegfaultExceptionHandler()
|
public static SigAction GetSegfaultExceptionHandler()
|
||||||
{
|
{
|
||||||
int result;
|
int result = sigaction(SIGSEGV, IntPtr.Zero, out SigAction old);
|
||||||
SigAction old;
|
|
||||||
|
|
||||||
if (Ryujinx.Common.PlatformInfo.IsBionic)
|
|
||||||
{
|
|
||||||
result = sigaction(SIGSEGV, IntPtr.Zero, out SigActionBionic tmp);
|
|
||||||
|
|
||||||
old = new SigAction
|
|
||||||
{
|
|
||||||
sa_handler = tmp.sa_handler,
|
|
||||||
sa_mask = tmp.sa_mask,
|
|
||||||
sa_flags = tmp.sa_flags,
|
|
||||||
sa_restorer = tmp.sa_restorer
|
|
||||||
};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
result = sigaction(SIGSEGV, IntPtr.Zero, out old);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
if (result != 0)
|
if (result != 0)
|
||||||
{
|
{
|
||||||
@ -83,36 +47,15 @@ namespace ARMeilleure.Signal
|
|||||||
|
|
||||||
public static SigAction RegisterExceptionHandler(IntPtr action)
|
public static SigAction RegisterExceptionHandler(IntPtr action)
|
||||||
{
|
{
|
||||||
int result;
|
SigAction sig = new()
|
||||||
SigAction old;
|
|
||||||
|
|
||||||
if (Ryujinx.Common.PlatformInfo.IsBionic)
|
|
||||||
{
|
|
||||||
SigActionBionic sig = new()
|
|
||||||
{
|
{
|
||||||
sa_handler = action,
|
sa_handler = action,
|
||||||
sa_flags = SA_SIGINFO | SA_ONSTACK,
|
sa_flags = SA_SIGINFO,
|
||||||
};
|
};
|
||||||
|
|
||||||
sigemptyset(ref sig.sa_mask);
|
sigemptyset(ref sig.sa_mask);
|
||||||
|
|
||||||
result = sigaction(SIGSEGV, ref sig, out SigActionBionic tmp);
|
int result = sigaction(SIGSEGV, ref sig, out SigAction old);
|
||||||
|
|
||||||
old = new SigAction
|
|
||||||
{
|
|
||||||
sa_handler = tmp.sa_handler,
|
|
||||||
sa_mask = tmp.sa_mask,
|
|
||||||
sa_flags = tmp.sa_flags,
|
|
||||||
sa_restorer = tmp.sa_restorer
|
|
||||||
};
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
SigAction sig = new() { sa_handler = action, sa_flags = SA_SIGINFO, };
|
|
||||||
|
|
||||||
sigemptyset(ref sig.sa_mask);
|
|
||||||
|
|
||||||
result = sigaction(SIGSEGV, ref sig, out old);
|
|
||||||
|
|
||||||
if (result != 0)
|
if (result != 0)
|
||||||
{
|
{
|
||||||
@ -128,36 +71,13 @@ namespace ARMeilleure.Signal
|
|||||||
throw new InvalidOperationException($"Could not register SIGBUS sigaction. Error: {result}");
|
throw new InvalidOperationException($"Could not register SIGBUS sigaction. Error: {result}");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
|
||||||
|
|
||||||
return old;
|
return old;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static bool RestoreExceptionHandler(SigAction oldAction)
|
public static bool RestoreExceptionHandler(SigAction oldAction)
|
||||||
{
|
{
|
||||||
if (Ryujinx.Common.PlatformInfo.IsBionic)
|
return sigaction(SIGSEGV, ref oldAction, out SigAction _) == 0 && (!OperatingSystem.IsMacOS() || OperatingSystem.IsIOS() || sigaction(SIGBUS, ref oldAction, out SigAction _) == 0);
|
||||||
{
|
|
||||||
SigActionBionic tmp = new SigActionBionic
|
|
||||||
{
|
|
||||||
sa_handler = oldAction.sa_handler,
|
|
||||||
sa_mask = oldAction.sa_mask,
|
|
||||||
sa_flags = oldAction.sa_flags,
|
|
||||||
sa_restorer = oldAction.sa_restorer
|
|
||||||
};
|
|
||||||
|
|
||||||
return sigaction(SIGSEGV, ref tmp, out SigActionBionic _) == 0;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
bool success = sigaction(SIGSEGV, ref oldAction, out SigAction _) == 0;
|
|
||||||
|
|
||||||
if (success && (OperatingSystem.IsMacOS() || OperatingSystem.IsIOS()))
|
|
||||||
{
|
|
||||||
success = sigaction(SIGBUS, ref oldAction, out SigAction _) == 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
return success;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1006,7 +1006,6 @@ namespace ARMeilleure.Translation.PTC
|
|||||||
osPlatform |= (OperatingSystem.IsMacOS() ? 1u : 0u) << 2;
|
osPlatform |= (OperatingSystem.IsMacOS() ? 1u : 0u) << 2;
|
||||||
osPlatform |= (OperatingSystem.IsWindows() ? 1u : 0u) << 3;
|
osPlatform |= (OperatingSystem.IsWindows() ? 1u : 0u) << 3;
|
||||||
osPlatform |= (OperatingSystem.IsIOS() ? 1u : 0u) << 4;
|
osPlatform |= (OperatingSystem.IsIOS() ? 1u : 0u) << 4;
|
||||||
osPlatform |= (Ryujinx.Common.PlatformInfo.IsBionic ? 1u : 0u) << 5;
|
|
||||||
#pragma warning restore IDE0055
|
#pragma warning restore IDE0055
|
||||||
|
|
||||||
return osPlatform;
|
return osPlatform;
|
||||||
|
15
src/MeloNX-Android/.gitignore
vendored
Normal file
@ -0,0 +1,15 @@
|
|||||||
|
*.iml
|
||||||
|
.gradle
|
||||||
|
/local.properties
|
||||||
|
/.idea/caches
|
||||||
|
/.idea/libraries
|
||||||
|
/.idea/modules.xml
|
||||||
|
/.idea/workspace.xml
|
||||||
|
/.idea/navEditor.xml
|
||||||
|
/.idea/assetWizardSettings.xml
|
||||||
|
.DS_Store
|
||||||
|
/build
|
||||||
|
/captures
|
||||||
|
.externalNativeBuild
|
||||||
|
.cxx
|
||||||
|
local.properties
|
1
src/MeloNX-Android/.idea/.name
generated
Normal file
@ -0,0 +1 @@
|
|||||||
|
MeloNX
|
6
src/MeloNX-Android/.idea/compiler.xml
generated
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="CompilerConfiguration">
|
||||||
|
<bytecodeTargetLevel target="21" />
|
||||||
|
</component>
|
||||||
|
</project>
|
19
src/MeloNX-Android/.idea/gradle.xml
generated
Normal file
@ -0,0 +1,19 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="GradleSettings">
|
||||||
|
<option name="linkedExternalProjectsSettings">
|
||||||
|
<GradleProjectSettings>
|
||||||
|
<option name="testRunner" value="CHOOSE_PER_TEST" />
|
||||||
|
<option name="externalProjectPath" value="$PROJECT_DIR$" />
|
||||||
|
<option name="gradleJvm" value="#GRADLE_LOCAL_JAVA_HOME" />
|
||||||
|
<option name="modules">
|
||||||
|
<set>
|
||||||
|
<option value="$PROJECT_DIR$" />
|
||||||
|
<option value="$PROJECT_DIR$/app" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
<option name="resolveExternalAnnotations" value="false" />
|
||||||
|
</GradleProjectSettings>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
6
src/MeloNX-Android/.idea/kotlinc.xml
generated
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="KotlinJpsPluginSettings">
|
||||||
|
<option name="version" value="2.0.0" />
|
||||||
|
</component>
|
||||||
|
</project>
|
10
src/MeloNX-Android/.idea/migrations.xml
generated
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ProjectMigrations">
|
||||||
|
<option name="MigrateToGradleLocalJavaHome">
|
||||||
|
<set>
|
||||||
|
<option value="$PROJECT_DIR$" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
10
src/MeloNX-Android/.idea/misc.xml
generated
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="ExternalStorageConfigurationManager" enabled="true" />
|
||||||
|
<component name="ProjectRootManager" version="2" languageLevel="JDK_21" default="true" project-jdk-name="jbr-21" project-jdk-type="JavaSDK">
|
||||||
|
<output url="file://$PROJECT_DIR$/build/classes" />
|
||||||
|
</component>
|
||||||
|
<component name="ProjectType">
|
||||||
|
<option name="id" value="Android" />
|
||||||
|
</component>
|
||||||
|
</project>
|
17
src/MeloNX-Android/.idea/runConfigurations.xml
generated
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="RunConfigurationProducerService">
|
||||||
|
<option name="ignoredProducers">
|
||||||
|
<set>
|
||||||
|
<option value="com.intellij.execution.junit.AbstractAllInDirectoryConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.AllInPackageConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.PatternConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.TestInClassConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.UniqueIdConfigurationProducer" />
|
||||||
|
<option value="com.intellij.execution.junit.testDiscovery.JUnitTestDiscoveryConfigurationProducer" />
|
||||||
|
<option value="org.jetbrains.kotlin.idea.junit.KotlinJUnitRunConfigurationProducer" />
|
||||||
|
<option value="org.jetbrains.kotlin.idea.junit.KotlinPatternConfigurationProducer" />
|
||||||
|
</set>
|
||||||
|
</option>
|
||||||
|
</component>
|
||||||
|
</project>
|
6
src/MeloNX-Android/.idea/vcs.xml
generated
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
|
<project version="4">
|
||||||
|
<component name="VcsDirectoryMappings">
|
||||||
|
<mapping directory="$PROJECT_DIR$/../.." vcs="Git" />
|
||||||
|
</component>
|
||||||
|
</project>
|
1
src/MeloNX-Android/app/.gitignore
vendored
Normal file
@ -0,0 +1 @@
|
|||||||
|
/build
|
61
src/MeloNX-Android/app/build.gradle.kts
Normal file
@ -0,0 +1,61 @@
|
|||||||
|
plugins {
|
||||||
|
alias(libs.plugins.android.application)
|
||||||
|
alias(libs.plugins.kotlin.android)
|
||||||
|
alias(libs.plugins.kotlin.compose)
|
||||||
|
}
|
||||||
|
|
||||||
|
android {
|
||||||
|
namespace = "com.xitrix.melonx"
|
||||||
|
compileSdk = 35
|
||||||
|
|
||||||
|
defaultConfig {
|
||||||
|
applicationId = "com.xitrix.melonx"
|
||||||
|
minSdk = 29
|
||||||
|
targetSdk = 35
|
||||||
|
versionCode = 1
|
||||||
|
versionName = "1.0"
|
||||||
|
|
||||||
|
testInstrumentationRunner = "androidx.test.runner.AndroidJUnitRunner"
|
||||||
|
}
|
||||||
|
|
||||||
|
buildTypes {
|
||||||
|
release {
|
||||||
|
isMinifyEnabled = false
|
||||||
|
proguardFiles(
|
||||||
|
getDefaultProguardFile("proguard-android-optimize.txt"),
|
||||||
|
"proguard-rules.pro"
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
compileOptions {
|
||||||
|
sourceCompatibility = JavaVersion.VERSION_11
|
||||||
|
targetCompatibility = JavaVersion.VERSION_11
|
||||||
|
}
|
||||||
|
kotlinOptions {
|
||||||
|
jvmTarget = "11"
|
||||||
|
}
|
||||||
|
buildFeatures {
|
||||||
|
compose = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
dependencies {
|
||||||
|
|
||||||
|
implementation(libs.androidx.core.ktx)
|
||||||
|
implementation(libs.androidx.lifecycle.runtime.ktx)
|
||||||
|
implementation(libs.androidx.activity.compose)
|
||||||
|
implementation(platform(libs.androidx.compose.bom))
|
||||||
|
implementation(libs.androidx.ui)
|
||||||
|
implementation(libs.androidx.ui.graphics)
|
||||||
|
implementation(libs.androidx.ui.tooling.preview)
|
||||||
|
implementation(libs.androidx.material3)
|
||||||
|
testImplementation(libs.junit)
|
||||||
|
androidTestImplementation(libs.androidx.junit)
|
||||||
|
androidTestImplementation(libs.androidx.espresso.core)
|
||||||
|
androidTestImplementation(platform(libs.androidx.compose.bom))
|
||||||
|
androidTestImplementation(libs.androidx.ui.test.junit4)
|
||||||
|
debugImplementation(libs.androidx.ui.tooling)
|
||||||
|
debugImplementation(libs.androidx.ui.test.manifest)
|
||||||
|
implementation(libs.androidx.navigation.compose)
|
||||||
|
|
||||||
|
}
|
21
src/MeloNX-Android/app/proguard-rules.pro
vendored
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
# Add project specific ProGuard rules here.
|
||||||
|
# You can control the set of applied configuration files using the
|
||||||
|
# proguardFiles setting in build.gradle.
|
||||||
|
#
|
||||||
|
# For more details, see
|
||||||
|
# http://developer.android.com/guide/developing/tools/proguard.html
|
||||||
|
|
||||||
|
# If your project uses WebView with JS, uncomment the following
|
||||||
|
# and specify the fully qualified class name to the JavaScript interface
|
||||||
|
# class:
|
||||||
|
#-keepclassmembers class fqcn.of.javascript.interface.for.webview {
|
||||||
|
# public *;
|
||||||
|
#}
|
||||||
|
|
||||||
|
# Uncomment this to preserve the line number information for
|
||||||
|
# debugging stack traces.
|
||||||
|
#-keepattributes SourceFile,LineNumberTable
|
||||||
|
|
||||||
|
# If you keep the line number information, uncomment this to
|
||||||
|
# hide the original source file name.
|
||||||
|
#-renamesourcefileattribute SourceFile
|
@ -0,0 +1,24 @@
|
|||||||
|
package com.xitrix.melonx
|
||||||
|
|
||||||
|
import androidx.test.platform.app.InstrumentationRegistry
|
||||||
|
import androidx.test.ext.junit.runners.AndroidJUnit4
|
||||||
|
|
||||||
|
import org.junit.Test
|
||||||
|
import org.junit.runner.RunWith
|
||||||
|
|
||||||
|
import org.junit.Assert.*
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Instrumented test, which will execute on an Android device.
|
||||||
|
*
|
||||||
|
* See [testing documentation](http://d.android.com/tools/testing).
|
||||||
|
*/
|
||||||
|
@RunWith(AndroidJUnit4::class)
|
||||||
|
class ExampleInstrumentedTest {
|
||||||
|
@Test
|
||||||
|
fun useAppContext() {
|
||||||
|
// Context of the app under test.
|
||||||
|
val appContext = InstrumentationRegistry.getInstrumentation().targetContext
|
||||||
|
assertEquals("com.xitrix.melonx", appContext.packageName)
|
||||||
|
}
|
||||||
|
}
|
28
src/MeloNX-Android/app/src/main/AndroidManifest.xml
Normal file
@ -0,0 +1,28 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<manifest xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:tools="http://schemas.android.com/tools">
|
||||||
|
|
||||||
|
<application
|
||||||
|
android:allowBackup="true"
|
||||||
|
android:dataExtractionRules="@xml/data_extraction_rules"
|
||||||
|
android:fullBackupContent="@xml/backup_rules"
|
||||||
|
android:icon="@mipmap/ic_launcher"
|
||||||
|
android:label="@string/app_name"
|
||||||
|
android:roundIcon="@mipmap/ic_launcher_round"
|
||||||
|
android:supportsRtl="true"
|
||||||
|
android:theme="@style/Theme.MeloNX"
|
||||||
|
tools:targetApi="31">
|
||||||
|
<activity
|
||||||
|
android:name=".MainActivity"
|
||||||
|
android:exported="true"
|
||||||
|
android:label="@string/app_name"
|
||||||
|
android:theme="@style/Theme.MeloNX">
|
||||||
|
<intent-filter>
|
||||||
|
<action android:name="android.intent.action.MAIN" />
|
||||||
|
|
||||||
|
<category android:name="android.intent.category.LAUNCHER" />
|
||||||
|
</intent-filter>
|
||||||
|
</activity>
|
||||||
|
</application>
|
||||||
|
|
||||||
|
</manifest>
|
@ -0,0 +1,34 @@
|
|||||||
|
package com.xitrix.melonx
|
||||||
|
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.material3.Scaffold
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.tooling.preview.Preview
|
||||||
|
import androidx.navigation.NavHostController
|
||||||
|
import com.xitrix.melonx.ui.theme.MeloNXTheme
|
||||||
|
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun HomeScreen(
|
||||||
|
scaffoldState: ScaffoldState,
|
||||||
|
navController: NavHostController
|
||||||
|
) {
|
||||||
|
Scaffold(
|
||||||
|
modifier = modifier
|
||||||
|
) { padding ->
|
||||||
|
Text(
|
||||||
|
text = "Hello $name!",
|
||||||
|
modifier = Modifier.padding(padding)
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Preview(showBackground = true)
|
||||||
|
@Composable
|
||||||
|
fun HomeScreenPreview() {
|
||||||
|
MeloNXTheme {
|
||||||
|
HomeScreen("Android")
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,124 @@
|
|||||||
|
package com.xitrix.melonx
|
||||||
|
|
||||||
|
import android.os.Bundle
|
||||||
|
import androidx.activity.ComponentActivity
|
||||||
|
import androidx.activity.compose.setContent
|
||||||
|
import androidx.activity.enableEdgeToEdge
|
||||||
|
import androidx.annotation.StringRes
|
||||||
|
import androidx.compose.foundation.layout.Box
|
||||||
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
|
import androidx.compose.foundation.layout.fillMaxSize
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.material.icons.Icons
|
||||||
|
import androidx.compose.material.icons.filled.*
|
||||||
|
import androidx.compose.material3.ExperimentalMaterial3Api
|
||||||
|
import androidx.compose.material3.Icon
|
||||||
|
import androidx.compose.material3.NavigationBar
|
||||||
|
import androidx.compose.material3.NavigationBarItem
|
||||||
|
import androidx.compose.material3.Scaffold
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.material3.TopAppBar
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.runtime.Immutable
|
||||||
|
import androidx.compose.runtime.LaunchedEffect
|
||||||
|
import androidx.compose.runtime.getValue
|
||||||
|
import androidx.compose.runtime.mutableStateOf
|
||||||
|
import androidx.compose.runtime.remember
|
||||||
|
import androidx.compose.runtime.setValue
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
import androidx.compose.ui.graphics.vector.ImageVector
|
||||||
|
import androidx.compose.ui.res.stringResource
|
||||||
|
import androidx.compose.ui.tooling.preview.Preview
|
||||||
|
import androidx.navigation.compose.NavHost
|
||||||
|
import com.xitrix.melonx.ui.theme.MeloNXTheme
|
||||||
|
|
||||||
|
@Immutable
|
||||||
|
data class ScaffoldViewState(
|
||||||
|
@StringRes val topAppBarTitle: Int? = null,
|
||||||
|
@StringRes val fabText: Int? = null,
|
||||||
|
// TODO : ...etc (top app bar actions, nav icon...)
|
||||||
|
)
|
||||||
|
|
||||||
|
class MainActivity : ComponentActivity() {
|
||||||
|
@OptIn(ExperimentalMaterial3Api::class)
|
||||||
|
override fun onCreate(savedInstanceState: Bundle?) {
|
||||||
|
super.onCreate(savedInstanceState)
|
||||||
|
enableEdgeToEdge()
|
||||||
|
setContent {
|
||||||
|
MeloNXTheme {
|
||||||
|
val items = listOf(
|
||||||
|
BottomNavItem("Home", "home", Icons.Default.Home),
|
||||||
|
BottomNavItem("Settings", "settings", Icons.Default.Settings)
|
||||||
|
)
|
||||||
|
var currentRoute by remember { mutableStateOf("home") }
|
||||||
|
var scaffoldViewState by remember {
|
||||||
|
mutableStateOf(ScaffoldViewState())
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Scaffold(
|
||||||
|
|
||||||
|
// modifier = Modifier.fillMaxSize(),
|
||||||
|
topBar = {
|
||||||
|
TopAppBar(
|
||||||
|
title = {
|
||||||
|
Text(text = "Хуета")
|
||||||
|
}
|
||||||
|
)
|
||||||
|
},
|
||||||
|
bottomBar = {
|
||||||
|
BottomNavigationBar(
|
||||||
|
items = items,
|
||||||
|
currentRoute = currentRoute,
|
||||||
|
onItemClick = { currentRoute = it }
|
||||||
|
)
|
||||||
|
}) { innerPadding ->
|
||||||
|
// NavHost {
|
||||||
|
//// composable("a") {
|
||||||
|
// LaunchedEffect(Unit) {
|
||||||
|
//// scaffoldViewState = // TODO : choose the top app bar and fab appearance for this page
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// AScreen()
|
||||||
|
//// }
|
||||||
|
// composable("b") {
|
||||||
|
// LaunchedEffect(Unit) {
|
||||||
|
//// scaffoldViewState = // TODO : choose the top app bar and fab appearance for this page
|
||||||
|
// }
|
||||||
|
//
|
||||||
|
// BScreen()
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
when (currentRoute) {
|
||||||
|
"home" -> HomeScreen("Android", Modifier.padding(innerPadding))
|
||||||
|
"settings" -> SettingsScreen(innerPadding)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun BottomNavigationBar(
|
||||||
|
items: List<BottomNavItem>,
|
||||||
|
currentRoute: String,
|
||||||
|
onItemClick: (String) -> Unit
|
||||||
|
) {
|
||||||
|
NavigationBar {
|
||||||
|
items.forEach { item ->
|
||||||
|
NavigationBarItem(
|
||||||
|
icon = { Icon(item.icon, contentDescription = item.title) },
|
||||||
|
label = { Text(item.title) },
|
||||||
|
selected = currentRoute == item.route,
|
||||||
|
onClick = { onItemClick(item.route) }
|
||||||
|
)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
data class BottomNavItem(
|
||||||
|
val title: String,
|
||||||
|
val route: String,
|
||||||
|
val icon: ImageVector
|
||||||
|
)
|
@ -0,0 +1,16 @@
|
|||||||
|
package com.xitrix.melonx
|
||||||
|
|
||||||
|
import androidx.compose.foundation.layout.Box
|
||||||
|
import androidx.compose.foundation.layout.PaddingValues
|
||||||
|
import androidx.compose.foundation.layout.padding
|
||||||
|
import androidx.compose.material3.Text
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.Modifier
|
||||||
|
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun SettingsScreen(paddingValues: PaddingValues) {
|
||||||
|
Box(modifier = Modifier.padding(paddingValues)) {
|
||||||
|
Text(text = "Screen One")
|
||||||
|
}
|
||||||
|
}
|
@ -0,0 +1,11 @@
|
|||||||
|
package com.xitrix.melonx.ui.theme
|
||||||
|
|
||||||
|
import androidx.compose.ui.graphics.Color
|
||||||
|
|
||||||
|
val Purple80 = Color(0xFFD0BCFF)
|
||||||
|
val PurpleGrey80 = Color(0xFFCCC2DC)
|
||||||
|
val Pink80 = Color(0xFFEFB8C8)
|
||||||
|
|
||||||
|
val Purple40 = Color(0xFF6650a4)
|
||||||
|
val PurpleGrey40 = Color(0xFF625b71)
|
||||||
|
val Pink40 = Color(0xFF7D5260)
|
@ -0,0 +1,58 @@
|
|||||||
|
package com.xitrix.melonx.ui.theme
|
||||||
|
|
||||||
|
import android.app.Activity
|
||||||
|
import android.os.Build
|
||||||
|
import androidx.compose.foundation.isSystemInDarkTheme
|
||||||
|
import androidx.compose.material3.MaterialTheme
|
||||||
|
import androidx.compose.material3.darkColorScheme
|
||||||
|
import androidx.compose.material3.dynamicDarkColorScheme
|
||||||
|
import androidx.compose.material3.dynamicLightColorScheme
|
||||||
|
import androidx.compose.material3.lightColorScheme
|
||||||
|
import androidx.compose.runtime.Composable
|
||||||
|
import androidx.compose.ui.platform.LocalContext
|
||||||
|
|
||||||
|
private val DarkColorScheme = darkColorScheme(
|
||||||
|
primary = Purple80,
|
||||||
|
secondary = PurpleGrey80,
|
||||||
|
tertiary = Pink80
|
||||||
|
)
|
||||||
|
|
||||||
|
private val LightColorScheme = lightColorScheme(
|
||||||
|
primary = Purple40,
|
||||||
|
secondary = PurpleGrey40,
|
||||||
|
tertiary = Pink40
|
||||||
|
|
||||||
|
/* Other default colors to override
|
||||||
|
background = Color(0xFFFFFBFE),
|
||||||
|
surface = Color(0xFFFFFBFE),
|
||||||
|
onPrimary = Color.White,
|
||||||
|
onSecondary = Color.White,
|
||||||
|
onTertiary = Color.White,
|
||||||
|
onBackground = Color(0xFF1C1B1F),
|
||||||
|
onSurface = Color(0xFF1C1B1F),
|
||||||
|
*/
|
||||||
|
)
|
||||||
|
|
||||||
|
@Composable
|
||||||
|
fun MeloNXTheme(
|
||||||
|
darkTheme: Boolean = isSystemInDarkTheme(),
|
||||||
|
// Dynamic color is available on Android 12+
|
||||||
|
dynamicColor: Boolean = true,
|
||||||
|
content: @Composable () -> Unit
|
||||||
|
) {
|
||||||
|
val colorScheme = when {
|
||||||
|
dynamicColor && Build.VERSION.SDK_INT >= Build.VERSION_CODES.S -> {
|
||||||
|
val context = LocalContext.current
|
||||||
|
if (darkTheme) dynamicDarkColorScheme(context) else dynamicLightColorScheme(context)
|
||||||
|
}
|
||||||
|
|
||||||
|
darkTheme -> DarkColorScheme
|
||||||
|
else -> LightColorScheme
|
||||||
|
}
|
||||||
|
|
||||||
|
MaterialTheme(
|
||||||
|
colorScheme = colorScheme,
|
||||||
|
typography = Typography,
|
||||||
|
content = content
|
||||||
|
)
|
||||||
|
}
|
@ -0,0 +1,34 @@
|
|||||||
|
package com.xitrix.melonx.ui.theme
|
||||||
|
|
||||||
|
import androidx.compose.material3.Typography
|
||||||
|
import androidx.compose.ui.text.TextStyle
|
||||||
|
import androidx.compose.ui.text.font.FontFamily
|
||||||
|
import androidx.compose.ui.text.font.FontWeight
|
||||||
|
import androidx.compose.ui.unit.sp
|
||||||
|
|
||||||
|
// Set of Material typography styles to start with
|
||||||
|
val Typography = Typography(
|
||||||
|
bodyLarge = TextStyle(
|
||||||
|
fontFamily = FontFamily.Default,
|
||||||
|
fontWeight = FontWeight.Normal,
|
||||||
|
fontSize = 16.sp,
|
||||||
|
lineHeight = 24.sp,
|
||||||
|
letterSpacing = 0.5.sp
|
||||||
|
)
|
||||||
|
/* Other default text styles to override
|
||||||
|
titleLarge = TextStyle(
|
||||||
|
fontFamily = FontFamily.Default,
|
||||||
|
fontWeight = FontWeight.Normal,
|
||||||
|
fontSize = 22.sp,
|
||||||
|
lineHeight = 28.sp,
|
||||||
|
letterSpacing = 0.sp
|
||||||
|
),
|
||||||
|
labelSmall = TextStyle(
|
||||||
|
fontFamily = FontFamily.Default,
|
||||||
|
fontWeight = FontWeight.Medium,
|
||||||
|
fontSize = 11.sp,
|
||||||
|
lineHeight = 16.sp,
|
||||||
|
letterSpacing = 0.5.sp
|
||||||
|
)
|
||||||
|
*/
|
||||||
|
)
|
@ -0,0 +1,170 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
android:width="108dp"
|
||||||
|
android:height="108dp"
|
||||||
|
android:viewportWidth="108"
|
||||||
|
android:viewportHeight="108">
|
||||||
|
<path
|
||||||
|
android:fillColor="#3DDC84"
|
||||||
|
android:pathData="M0,0h108v108h-108z" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M9,0L9,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,0L19,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M29,0L29,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M39,0L39,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M49,0L49,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M59,0L59,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M69,0L69,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M79,0L79,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M89,0L89,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M99,0L99,108"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,9L108,9"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,19L108,19"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,29L108,29"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,39L108,39"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,49L108,49"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,59L108,59"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,69L108,69"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,79L108,79"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,89L108,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M0,99L108,99"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,29L89,29"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,39L89,39"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,49L89,49"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,59L89,59"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,69L89,69"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M19,79L89,79"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M29,19L29,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M39,19L39,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M49,19L49,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M59,19L59,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M69,19L69,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
<path
|
||||||
|
android:fillColor="#00000000"
|
||||||
|
android:pathData="M79,19L79,89"
|
||||||
|
android:strokeWidth="0.8"
|
||||||
|
android:strokeColor="#33FFFFFF" />
|
||||||
|
</vector>
|
@ -0,0 +1,30 @@
|
|||||||
|
<vector xmlns:android="http://schemas.android.com/apk/res/android"
|
||||||
|
xmlns:aapt="http://schemas.android.com/aapt"
|
||||||
|
android:width="108dp"
|
||||||
|
android:height="108dp"
|
||||||
|
android:viewportWidth="108"
|
||||||
|
android:viewportHeight="108">
|
||||||
|
<path android:pathData="M31,63.928c0,0 6.4,-11 12.1,-13.1c7.2,-2.6 26,-1.4 26,-1.4l38.1,38.1L107,108.928l-32,-1L31,63.928z">
|
||||||
|
<aapt:attr name="android:fillColor">
|
||||||
|
<gradient
|
||||||
|
android:endX="85.84757"
|
||||||
|
android:endY="92.4963"
|
||||||
|
android:startX="42.9492"
|
||||||
|
android:startY="49.59793"
|
||||||
|
android:type="linear">
|
||||||
|
<item
|
||||||
|
android:color="#44000000"
|
||||||
|
android:offset="0.0" />
|
||||||
|
<item
|
||||||
|
android:color="#00000000"
|
||||||
|
android:offset="1.0" />
|
||||||
|
</gradient>
|
||||||
|
</aapt:attr>
|
||||||
|
</path>
|
||||||
|
<path
|
||||||
|
android:fillColor="#FFFFFF"
|
||||||
|
android:fillType="nonZero"
|
||||||
|
android:pathData="M65.3,45.828l3.8,-6.6c0.2,-0.4 0.1,-0.9 -0.3,-1.1c-0.4,-0.2 -0.9,-0.1 -1.1,0.3l-3.9,6.7c-6.3,-2.8 -13.4,-2.8 -19.7,0l-3.9,-6.7c-0.2,-0.4 -0.7,-0.5 -1.1,-0.3C38.8,38.328 38.7,38.828 38.9,39.228l3.8,6.6C36.2,49.428 31.7,56.028 31,63.928h46C76.3,56.028 71.8,49.428 65.3,45.828zM43.4,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2c-0.3,-0.7 -0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C45.3,56.528 44.5,57.328 43.4,57.328L43.4,57.328zM64.6,57.328c-0.8,0 -1.5,-0.5 -1.8,-1.2s-0.1,-1.5 0.4,-2.1c0.5,-0.5 1.4,-0.7 2.1,-0.4c0.7,0.3 1.2,1 1.2,1.8C66.5,56.528 65.6,57.328 64.6,57.328L64.6,57.328z"
|
||||||
|
android:strokeWidth="1"
|
||||||
|
android:strokeColor="#00000000" />
|
||||||
|
</vector>
|
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@drawable/ic_launcher_background" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
@ -0,0 +1,6 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<adaptive-icon xmlns:android="http://schemas.android.com/apk/res/android">
|
||||||
|
<background android:drawable="@drawable/ic_launcher_background" />
|
||||||
|
<foreground android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
<monochrome android:drawable="@drawable/ic_launcher_foreground" />
|
||||||
|
</adaptive-icon>
|
BIN
src/MeloNX-Android/app/src/main/res/mipmap-hdpi/ic_launcher.webp
Normal file
After Width: | Height: | Size: 1.4 KiB |
After Width: | Height: | Size: 2.8 KiB |
BIN
src/MeloNX-Android/app/src/main/res/mipmap-mdpi/ic_launcher.webp
Normal file
After Width: | Height: | Size: 982 B |
After Width: | Height: | Size: 1.7 KiB |
After Width: | Height: | Size: 1.9 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 2.8 KiB |
After Width: | Height: | Size: 5.8 KiB |
After Width: | Height: | Size: 3.8 KiB |
After Width: | Height: | Size: 7.6 KiB |
10
src/MeloNX-Android/app/src/main/res/values/colors.xml
Normal file
@ -0,0 +1,10 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<resources>
|
||||||
|
<color name="purple_200">#FFBB86FC</color>
|
||||||
|
<color name="purple_500">#FF6200EE</color>
|
||||||
|
<color name="purple_700">#FF3700B3</color>
|
||||||
|
<color name="teal_200">#FF03DAC5</color>
|
||||||
|
<color name="teal_700">#FF018786</color>
|
||||||
|
<color name="black">#FF000000</color>
|
||||||
|
<color name="white">#FFFFFFFF</color>
|
||||||
|
</resources>
|
3
src/MeloNX-Android/app/src/main/res/values/strings.xml
Normal file
@ -0,0 +1,3 @@
|
|||||||
|
<resources>
|
||||||
|
<string name="app_name">MeloNX</string>
|
||||||
|
</resources>
|
5
src/MeloNX-Android/app/src/main/res/values/themes.xml
Normal file
@ -0,0 +1,5 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?>
|
||||||
|
<resources>
|
||||||
|
|
||||||
|
<style name="Theme.MeloNX" parent="android:Theme.Material.Light.NoActionBar" />
|
||||||
|
</resources>
|
13
src/MeloNX-Android/app/src/main/res/xml/backup_rules.xml
Normal file
@ -0,0 +1,13 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?><!--
|
||||||
|
Sample backup rules file; uncomment and customize as necessary.
|
||||||
|
See https://developer.android.com/guide/topics/data/autobackup
|
||||||
|
for details.
|
||||||
|
Note: This file is ignored for devices older that API 31
|
||||||
|
See https://developer.android.com/about/versions/12/backup-restore
|
||||||
|
-->
|
||||||
|
<full-backup-content>
|
||||||
|
<!--
|
||||||
|
<include domain="sharedpref" path="."/>
|
||||||
|
<exclude domain="sharedpref" path="device.xml"/>
|
||||||
|
-->
|
||||||
|
</full-backup-content>
|
@ -0,0 +1,19 @@
|
|||||||
|
<?xml version="1.0" encoding="utf-8"?><!--
|
||||||
|
Sample data extraction rules file; uncomment and customize as necessary.
|
||||||
|
See https://developer.android.com/about/versions/12/backup-restore#xml-changes
|
||||||
|
for details.
|
||||||
|
-->
|
||||||
|
<data-extraction-rules>
|
||||||
|
<cloud-backup>
|
||||||
|
<!-- TODO: Use <include> and <exclude> to control what is backed up.
|
||||||
|
<include .../>
|
||||||
|
<exclude .../>
|
||||||
|
-->
|
||||||
|
</cloud-backup>
|
||||||
|
<!--
|
||||||
|
<device-transfer>
|
||||||
|
<include .../>
|
||||||
|
<exclude .../>
|
||||||
|
</device-transfer>
|
||||||
|
-->
|
||||||
|
</data-extraction-rules>
|
@ -0,0 +1,17 @@
|
|||||||
|
package com.xitrix.melonx
|
||||||
|
|
||||||
|
import org.junit.Test
|
||||||
|
|
||||||
|
import org.junit.Assert.*
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Example local unit test, which will execute on the development machine (host).
|
||||||
|
*
|
||||||
|
* See [testing documentation](http://d.android.com/tools/testing).
|
||||||
|
*/
|
||||||
|
class ExampleUnitTest {
|
||||||
|
@Test
|
||||||
|
fun addition_isCorrect() {
|
||||||
|
assertEquals(4, 2 + 2)
|
||||||
|
}
|
||||||
|
}
|
6
src/MeloNX-Android/build.gradle.kts
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
// Top-level build file where you can add configuration options common to all sub-projects/modules.
|
||||||
|
plugins {
|
||||||
|
alias(libs.plugins.android.application) apply false
|
||||||
|
alias(libs.plugins.kotlin.android) apply false
|
||||||
|
alias(libs.plugins.kotlin.compose) apply false
|
||||||
|
}
|
23
src/MeloNX-Android/gradle.properties
Normal file
@ -0,0 +1,23 @@
|
|||||||
|
# Project-wide Gradle settings.
|
||||||
|
# IDE (e.g. Android Studio) users:
|
||||||
|
# Gradle settings configured through the IDE *will override*
|
||||||
|
# any settings specified in this file.
|
||||||
|
# For more details on how to configure your build environment visit
|
||||||
|
# http://www.gradle.org/docs/current/userguide/build_environment.html
|
||||||
|
# Specifies the JVM arguments used for the daemon process.
|
||||||
|
# The setting is particularly useful for tweaking memory settings.
|
||||||
|
org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
|
||||||
|
# When configured, Gradle will run in incubating parallel mode.
|
||||||
|
# This option should only be used with decoupled projects. For more details, visit
|
||||||
|
# https://developer.android.com/r/tools/gradle-multi-project-decoupled-projects
|
||||||
|
# org.gradle.parallel=true
|
||||||
|
# AndroidX package structure to make it clearer which packages are bundled with the
|
||||||
|
# Android operating system, and which are packaged with your app's APK
|
||||||
|
# https://developer.android.com/topic/libraries/support-library/androidx-rn
|
||||||
|
android.useAndroidX=true
|
||||||
|
# Kotlin code style for this project: "official" or "obsolete":
|
||||||
|
kotlin.code.style=official
|
||||||
|
# Enables namespacing of each library's R class so that its R class includes only the
|
||||||
|
# resources declared in the library itself and none from the library's dependencies,
|
||||||
|
# thereby reducing the size of the R class for that library
|
||||||
|
android.nonTransitiveRClass=true
|
34
src/MeloNX-Android/gradle/libs.versions.toml
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
[versions]
|
||||||
|
agp = "8.7.2"
|
||||||
|
kotlin = "2.0.0"
|
||||||
|
coreKtx = "1.15.0"
|
||||||
|
junit = "4.13.2"
|
||||||
|
junitVersion = "1.2.1"
|
||||||
|
espressoCore = "3.6.1"
|
||||||
|
lifecycleRuntimeKtx = "2.8.7"
|
||||||
|
activityCompose = "1.10.1"
|
||||||
|
composeBom = "2025.02.00"
|
||||||
|
navigationCompose = "2.8.8"
|
||||||
|
|
||||||
|
[libraries]
|
||||||
|
androidx-core-ktx = { group = "androidx.core", name = "core-ktx", version.ref = "coreKtx" }
|
||||||
|
androidx-navigation-compose = { module = "androidx.navigation:navigation-compose", version.ref = "navigationCompose" }
|
||||||
|
junit = { group = "junit", name = "junit", version.ref = "junit" }
|
||||||
|
androidx-junit = { group = "androidx.test.ext", name = "junit", version.ref = "junitVersion" }
|
||||||
|
androidx-espresso-core = { group = "androidx.test.espresso", name = "espresso-core", version.ref = "espressoCore" }
|
||||||
|
androidx-lifecycle-runtime-ktx = { group = "androidx.lifecycle", name = "lifecycle-runtime-ktx", version.ref = "lifecycleRuntimeKtx" }
|
||||||
|
androidx-activity-compose = { group = "androidx.activity", name = "activity-compose", version.ref = "activityCompose" }
|
||||||
|
androidx-compose-bom = { group = "androidx.compose", name = "compose-bom", version.ref = "composeBom" }
|
||||||
|
androidx-ui = { group = "androidx.compose.ui", name = "ui" }
|
||||||
|
androidx-ui-graphics = { group = "androidx.compose.ui", name = "ui-graphics" }
|
||||||
|
androidx-ui-tooling = { group = "androidx.compose.ui", name = "ui-tooling" }
|
||||||
|
androidx-ui-tooling-preview = { group = "androidx.compose.ui", name = "ui-tooling-preview" }
|
||||||
|
androidx-ui-test-manifest = { group = "androidx.compose.ui", name = "ui-test-manifest" }
|
||||||
|
androidx-ui-test-junit4 = { group = "androidx.compose.ui", name = "ui-test-junit4" }
|
||||||
|
androidx-material3 = { group = "androidx.compose.material3", name = "material3" }
|
||||||
|
|
||||||
|
[plugins]
|
||||||
|
android-application = { id = "com.android.application", version.ref = "agp" }
|
||||||
|
kotlin-android = { id = "org.jetbrains.kotlin.android", version.ref = "kotlin" }
|
||||||
|
kotlin-compose = { id = "org.jetbrains.kotlin.plugin.compose", version.ref = "kotlin" }
|
||||||
|
|
BIN
src/MeloNX-Android/gradle/wrapper/gradle-wrapper.jar
vendored
Normal file
6
src/MeloNX-Android/gradle/wrapper/gradle-wrapper.properties
vendored
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
#Sun Mar 02 15:08:35 CET 2025
|
||||||
|
distributionBase=GRADLE_USER_HOME
|
||||||
|
distributionPath=wrapper/dists
|
||||||
|
distributionUrl=https\://services.gradle.org/distributions/gradle-8.9-bin.zip
|
||||||
|
zipStoreBase=GRADLE_USER_HOME
|
||||||
|
zipStorePath=wrapper/dists
|
185
src/MeloNX-Android/gradlew
vendored
Executable file
@ -0,0 +1,185 @@
|
|||||||
|
#!/usr/bin/env sh
|
||||||
|
|
||||||
|
#
|
||||||
|
# Copyright 2015 the original author or authors.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
# you may not use this file except in compliance with the License.
|
||||||
|
# You may obtain a copy of the License at
|
||||||
|
#
|
||||||
|
# https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
# See the License for the specific language governing permissions and
|
||||||
|
# limitations under the License.
|
||||||
|
#
|
||||||
|
|
||||||
|
##############################################################################
|
||||||
|
##
|
||||||
|
## Gradle start up script for UN*X
|
||||||
|
##
|
||||||
|
##############################################################################
|
||||||
|
|
||||||
|
# Attempt to set APP_HOME
|
||||||
|
# Resolve links: $0 may be a link
|
||||||
|
PRG="$0"
|
||||||
|
# Need this for relative symlinks.
|
||||||
|
while [ -h "$PRG" ] ; do
|
||||||
|
ls=`ls -ld "$PRG"`
|
||||||
|
link=`expr "$ls" : '.*-> \(.*\)$'`
|
||||||
|
if expr "$link" : '/.*' > /dev/null; then
|
||||||
|
PRG="$link"
|
||||||
|
else
|
||||||
|
PRG=`dirname "$PRG"`"/$link"
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
SAVED="`pwd`"
|
||||||
|
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" "-Xms64m"'
|
||||||
|
|
||||||
|
# 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.
|
||||||
|
if [ -n "$JAVA_HOME" ] ; then
|
||||||
|
if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
|
||||||
|
# IBM's JDK on AIX uses strange locations for the executables
|
||||||
|
JAVACMD="$JAVA_HOME/jre/sh/java"
|
||||||
|
else
|
||||||
|
JAVACMD="$JAVA_HOME/bin/java"
|
||||||
|
fi
|
||||||
|
if [ ! -x "$JAVACMD" ] ; then
|
||||||
|
die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
JAVACMD="java"
|
||||||
|
which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
|
||||||
|
Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
location of your Java installation."
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Increase the maximum file descriptors if we can.
|
||||||
|
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
|
||||||
|
MAX_FD="$MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
ulimit -n $MAX_FD
|
||||||
|
if [ $? -ne 0 ] ; then
|
||||||
|
warn "Could not set maximum file descriptor limit: $MAX_FD"
|
||||||
|
fi
|
||||||
|
else
|
||||||
|
warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Darwin, add options to specify how the application appears in the dock
|
||||||
|
if $darwin; then
|
||||||
|
GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
|
||||||
|
fi
|
||||||
|
|
||||||
|
# For Cygwin or MSYS, switch paths to Windows format before running java
|
||||||
|
if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
|
||||||
|
APP_HOME=`cygpath --path --mixed "$APP_HOME"`
|
||||||
|
CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
|
||||||
|
|
||||||
|
JAVACMD=`cygpath --unix "$JAVACMD"`
|
||||||
|
|
||||||
|
# We build the pattern for arguments to be converted via cygpath
|
||||||
|
ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
|
||||||
|
SEP=""
|
||||||
|
for dir in $ROOTDIRSRAW ; do
|
||||||
|
ROOTDIRS="$ROOTDIRS$SEP$dir"
|
||||||
|
SEP="|"
|
||||||
|
done
|
||||||
|
OURCYGPATTERN="(^($ROOTDIRS))"
|
||||||
|
# Add a user-defined pattern to the cygpath arguments
|
||||||
|
if [ "$GRADLE_CYGPATTERN" != "" ] ; then
|
||||||
|
OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
|
||||||
|
fi
|
||||||
|
# Now convert the arguments - kludge to limit ourselves to /bin/sh
|
||||||
|
i=0
|
||||||
|
for arg in "$@" ; do
|
||||||
|
CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
|
||||||
|
CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
|
||||||
|
|
||||||
|
if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
|
||||||
|
eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
|
||||||
|
else
|
||||||
|
eval `echo args$i`="\"$arg\""
|
||||||
|
fi
|
||||||
|
i=`expr $i + 1`
|
||||||
|
done
|
||||||
|
case $i in
|
||||||
|
0) set -- ;;
|
||||||
|
1) set -- "$args0" ;;
|
||||||
|
2) set -- "$args0" "$args1" ;;
|
||||||
|
3) set -- "$args0" "$args1" "$args2" ;;
|
||||||
|
4) set -- "$args0" "$args1" "$args2" "$args3" ;;
|
||||||
|
5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
|
||||||
|
6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
|
||||||
|
7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
|
||||||
|
8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
|
||||||
|
9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
|
||||||
|
esac
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Escape application args
|
||||||
|
save () {
|
||||||
|
for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
|
||||||
|
echo " "
|
||||||
|
}
|
||||||
|
APP_ARGS=`save "$@"`
|
||||||
|
|
||||||
|
# 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"
|
||||||
|
|
||||||
|
exec "$JAVACMD" "$@"
|
89
src/MeloNX-Android/gradlew.bat
vendored
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
@rem
|
||||||
|
@rem Copyright 2015 the original author or authors.
|
||||||
|
@rem
|
||||||
|
@rem Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
@rem you may not use this file except in compliance with the License.
|
||||||
|
@rem You may obtain a copy of the License at
|
||||||
|
@rem
|
||||||
|
@rem https://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
@rem
|
||||||
|
@rem Unless required by applicable law or agreed to in writing, software
|
||||||
|
@rem distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
@rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
@rem See the License for the specific language governing permissions and
|
||||||
|
@rem limitations under the License.
|
||||||
|
@rem
|
||||||
|
|
||||||
|
@if "%DEBUG%" == "" @echo off
|
||||||
|
@rem ##########################################################################
|
||||||
|
@rem
|
||||||
|
@rem Gradle startup script for Windows
|
||||||
|
@rem
|
||||||
|
@rem ##########################################################################
|
||||||
|
|
||||||
|
@rem Set local scope for the variables with windows NT shell
|
||||||
|
if "%OS%"=="Windows_NT" setlocal
|
||||||
|
|
||||||
|
set DIRNAME=%~dp0
|
||||||
|
if "%DIRNAME%" == "" set DIRNAME=.
|
||||||
|
set APP_BASE_NAME=%~n0
|
||||||
|
set APP_HOME=%DIRNAME%
|
||||||
|
|
||||||
|
@rem Resolve any "." and ".." in APP_HOME to make it shorter.
|
||||||
|
for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
|
||||||
|
|
||||||
|
@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" "-Xms64m"
|
||||||
|
|
||||||
|
@rem Find java.exe
|
||||||
|
if defined JAVA_HOME goto findJavaFromJavaHome
|
||||||
|
|
||||||
|
set JAVA_EXE=java.exe
|
||||||
|
%JAVA_EXE% -version >NUL 2>&1
|
||||||
|
if "%ERRORLEVEL%" == "0" goto execute
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:findJavaFromJavaHome
|
||||||
|
set JAVA_HOME=%JAVA_HOME:"=%
|
||||||
|
set JAVA_EXE=%JAVA_HOME%/bin/java.exe
|
||||||
|
|
||||||
|
if exist "%JAVA_EXE%" goto execute
|
||||||
|
|
||||||
|
echo.
|
||||||
|
echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
|
||||||
|
echo.
|
||||||
|
echo Please set the JAVA_HOME variable in your environment to match the
|
||||||
|
echo location of your Java installation.
|
||||||
|
|
||||||
|
goto fail
|
||||||
|
|
||||||
|
:execute
|
||||||
|
@rem Setup the command line
|
||||||
|
|
||||||
|
set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
|
||||||
|
|
||||||
|
|
||||||
|
@rem Execute Gradle
|
||||||
|
"%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
|
||||||
|
|
||||||
|
:end
|
||||||
|
@rem End local scope for the variables with windows NT shell
|
||||||
|
if "%ERRORLEVEL%"=="0" goto mainEnd
|
||||||
|
|
||||||
|
:fail
|
||||||
|
rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
|
||||||
|
rem the _cmd.exe /c_ return code!
|
||||||
|
if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
|
||||||
|
exit /b 1
|
||||||
|
|
||||||
|
:mainEnd
|
||||||
|
if "%OS%"=="Windows_NT" endlocal
|
||||||
|
|
||||||
|
:omega
|
@ -1,22 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.hardware.usb.UsbDevice;
|
|
||||||
|
|
||||||
interface HIDDevice
|
|
||||||
{
|
|
||||||
public int getId();
|
|
||||||
public int getVendorId();
|
|
||||||
public int getProductId();
|
|
||||||
public String getSerialNumber();
|
|
||||||
public int getVersion();
|
|
||||||
public String getManufacturerName();
|
|
||||||
public String getProductName();
|
|
||||||
public UsbDevice getDevice();
|
|
||||||
public boolean open();
|
|
||||||
public int sendFeatureReport(byte[] report);
|
|
||||||
public int sendOutputReport(byte[] report);
|
|
||||||
public boolean getFeatureReport(byte[] report);
|
|
||||||
public void setFrozen(boolean frozen);
|
|
||||||
public void close();
|
|
||||||
public void shutdown();
|
|
||||||
}
|
|
@ -1,650 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.bluetooth.BluetoothDevice;
|
|
||||||
import android.bluetooth.BluetoothGatt;
|
|
||||||
import android.bluetooth.BluetoothGattCallback;
|
|
||||||
import android.bluetooth.BluetoothGattCharacteristic;
|
|
||||||
import android.bluetooth.BluetoothGattDescriptor;
|
|
||||||
import android.bluetooth.BluetoothManager;
|
|
||||||
import android.bluetooth.BluetoothProfile;
|
|
||||||
import android.bluetooth.BluetoothGattService;
|
|
||||||
import android.hardware.usb.UsbDevice;
|
|
||||||
import android.os.Handler;
|
|
||||||
import android.os.Looper;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.os.*;
|
|
||||||
|
|
||||||
//import com.android.internal.util.HexDump;
|
|
||||||
|
|
||||||
import java.lang.Runnable;
|
|
||||||
import java.util.Arrays;
|
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.UUID;
|
|
||||||
|
|
||||||
class HIDDeviceBLESteamController extends BluetoothGattCallback implements HIDDevice {
|
|
||||||
|
|
||||||
private static final String TAG = "hidapi";
|
|
||||||
private HIDDeviceManager mManager;
|
|
||||||
private BluetoothDevice mDevice;
|
|
||||||
private int mDeviceId;
|
|
||||||
private BluetoothGatt mGatt;
|
|
||||||
private boolean mIsRegistered = false;
|
|
||||||
private boolean mIsConnected = false;
|
|
||||||
private boolean mIsChromebook = false;
|
|
||||||
private boolean mIsReconnecting = false;
|
|
||||||
private boolean mFrozen = false;
|
|
||||||
private LinkedList<GattOperation> mOperations;
|
|
||||||
GattOperation mCurrentOperation = null;
|
|
||||||
private Handler mHandler;
|
|
||||||
|
|
||||||
private static final int TRANSPORT_AUTO = 0;
|
|
||||||
private static final int TRANSPORT_BREDR = 1;
|
|
||||||
private static final int TRANSPORT_LE = 2;
|
|
||||||
|
|
||||||
private static final int CHROMEBOOK_CONNECTION_CHECK_INTERVAL = 10000;
|
|
||||||
|
|
||||||
static public final UUID steamControllerService = UUID.fromString("100F6C32-1735-4313-B402-38567131E5F3");
|
|
||||||
static public final UUID inputCharacteristic = UUID.fromString("100F6C33-1735-4313-B402-38567131E5F3");
|
|
||||||
static public final UUID reportCharacteristic = UUID.fromString("100F6C34-1735-4313-B402-38567131E5F3");
|
|
||||||
static private final byte[] enterValveMode = new byte[] { (byte)0xC0, (byte)0x87, 0x03, 0x08, 0x07, 0x00 };
|
|
||||||
|
|
||||||
static class GattOperation {
|
|
||||||
private enum Operation {
|
|
||||||
CHR_READ,
|
|
||||||
CHR_WRITE,
|
|
||||||
ENABLE_NOTIFICATION
|
|
||||||
}
|
|
||||||
|
|
||||||
Operation mOp;
|
|
||||||
UUID mUuid;
|
|
||||||
byte[] mValue;
|
|
||||||
BluetoothGatt mGatt;
|
|
||||||
boolean mResult = true;
|
|
||||||
|
|
||||||
private GattOperation(BluetoothGatt gatt, GattOperation.Operation operation, UUID uuid) {
|
|
||||||
mGatt = gatt;
|
|
||||||
mOp = operation;
|
|
||||||
mUuid = uuid;
|
|
||||||
}
|
|
||||||
|
|
||||||
private GattOperation(BluetoothGatt gatt, GattOperation.Operation operation, UUID uuid, byte[] value) {
|
|
||||||
mGatt = gatt;
|
|
||||||
mOp = operation;
|
|
||||||
mUuid = uuid;
|
|
||||||
mValue = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void run() {
|
|
||||||
// This is executed in main thread
|
|
||||||
BluetoothGattCharacteristic chr;
|
|
||||||
|
|
||||||
switch (mOp) {
|
|
||||||
case CHR_READ:
|
|
||||||
chr = getCharacteristic(mUuid);
|
|
||||||
//Log.v(TAG, "Reading characteristic " + chr.getUuid());
|
|
||||||
if (!mGatt.readCharacteristic(chr)) {
|
|
||||||
Log.e(TAG, "Unable to read characteristic " + mUuid.toString());
|
|
||||||
mResult = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
mResult = true;
|
|
||||||
break;
|
|
||||||
case CHR_WRITE:
|
|
||||||
chr = getCharacteristic(mUuid);
|
|
||||||
//Log.v(TAG, "Writing characteristic " + chr.getUuid() + " value=" + HexDump.toHexString(value));
|
|
||||||
chr.setValue(mValue);
|
|
||||||
if (!mGatt.writeCharacteristic(chr)) {
|
|
||||||
Log.e(TAG, "Unable to write characteristic " + mUuid.toString());
|
|
||||||
mResult = false;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
mResult = true;
|
|
||||||
break;
|
|
||||||
case ENABLE_NOTIFICATION:
|
|
||||||
chr = getCharacteristic(mUuid);
|
|
||||||
//Log.v(TAG, "Writing descriptor of " + chr.getUuid());
|
|
||||||
if (chr != null) {
|
|
||||||
BluetoothGattDescriptor cccd = chr.getDescriptor(UUID.fromString("00002902-0000-1000-8000-00805f9b34fb"));
|
|
||||||
if (cccd != null) {
|
|
||||||
int properties = chr.getProperties();
|
|
||||||
byte[] value;
|
|
||||||
if ((properties & BluetoothGattCharacteristic.PROPERTY_NOTIFY) == BluetoothGattCharacteristic.PROPERTY_NOTIFY) {
|
|
||||||
value = BluetoothGattDescriptor.ENABLE_NOTIFICATION_VALUE;
|
|
||||||
} else if ((properties & BluetoothGattCharacteristic.PROPERTY_INDICATE) == BluetoothGattCharacteristic.PROPERTY_INDICATE) {
|
|
||||||
value = BluetoothGattDescriptor.ENABLE_INDICATION_VALUE;
|
|
||||||
} else {
|
|
||||||
Log.e(TAG, "Unable to start notifications on input characteristic");
|
|
||||||
mResult = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mGatt.setCharacteristicNotification(chr, true);
|
|
||||||
cccd.setValue(value);
|
|
||||||
if (!mGatt.writeDescriptor(cccd)) {
|
|
||||||
Log.e(TAG, "Unable to write descriptor " + mUuid.toString());
|
|
||||||
mResult = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
mResult = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean finish() {
|
|
||||||
return mResult;
|
|
||||||
}
|
|
||||||
|
|
||||||
private BluetoothGattCharacteristic getCharacteristic(UUID uuid) {
|
|
||||||
BluetoothGattService valveService = mGatt.getService(steamControllerService);
|
|
||||||
if (valveService == null)
|
|
||||||
return null;
|
|
||||||
return valveService.getCharacteristic(uuid);
|
|
||||||
}
|
|
||||||
|
|
||||||
static public GattOperation readCharacteristic(BluetoothGatt gatt, UUID uuid) {
|
|
||||||
return new GattOperation(gatt, Operation.CHR_READ, uuid);
|
|
||||||
}
|
|
||||||
|
|
||||||
static public GattOperation writeCharacteristic(BluetoothGatt gatt, UUID uuid, byte[] value) {
|
|
||||||
return new GattOperation(gatt, Operation.CHR_WRITE, uuid, value);
|
|
||||||
}
|
|
||||||
|
|
||||||
static public GattOperation enableNotification(BluetoothGatt gatt, UUID uuid) {
|
|
||||||
return new GattOperation(gatt, Operation.ENABLE_NOTIFICATION, uuid);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public HIDDeviceBLESteamController(HIDDeviceManager manager, BluetoothDevice device) {
|
|
||||||
mManager = manager;
|
|
||||||
mDevice = device;
|
|
||||||
mDeviceId = mManager.getDeviceIDForIdentifier(getIdentifier());
|
|
||||||
mIsRegistered = false;
|
|
||||||
mIsChromebook = mManager.getContext().getPackageManager().hasSystemFeature("org.chromium.arc.device_management");
|
|
||||||
mOperations = new LinkedList<GattOperation>();
|
|
||||||
mHandler = new Handler(Looper.getMainLooper());
|
|
||||||
|
|
||||||
mGatt = connectGatt();
|
|
||||||
// final HIDDeviceBLESteamController finalThis = this;
|
|
||||||
// mHandler.postDelayed(new Runnable() {
|
|
||||||
// @Override
|
|
||||||
// public void run() {
|
|
||||||
// finalThis.checkConnectionForChromebookIssue();
|
|
||||||
// }
|
|
||||||
// }, CHROMEBOOK_CONNECTION_CHECK_INTERVAL);
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getIdentifier() {
|
|
||||||
return String.format("SteamController.%s", mDevice.getAddress());
|
|
||||||
}
|
|
||||||
|
|
||||||
public BluetoothGatt getGatt() {
|
|
||||||
return mGatt;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Because on Chromebooks we show up as a dual-mode device, it will attempt to connect TRANSPORT_AUTO, which will use TRANSPORT_BREDR instead
|
|
||||||
// of TRANSPORT_LE. Let's force ourselves to connect low energy.
|
|
||||||
private BluetoothGatt connectGatt(boolean managed) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 23 /* Android 6.0 (M) */) {
|
|
||||||
try {
|
|
||||||
return mDevice.connectGatt(mManager.getContext(), managed, this, TRANSPORT_LE);
|
|
||||||
} catch (Exception e) {
|
|
||||||
return mDevice.connectGatt(mManager.getContext(), managed, this);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
return mDevice.connectGatt(mManager.getContext(), managed, this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private BluetoothGatt connectGatt() {
|
|
||||||
return connectGatt(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
protected int getConnectionState() {
|
|
||||||
|
|
||||||
Context context = mManager.getContext();
|
|
||||||
if (context == null) {
|
|
||||||
// We are lacking any context to get our Bluetooth information. We'll just assume disconnected.
|
|
||||||
return BluetoothProfile.STATE_DISCONNECTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
BluetoothManager btManager = (BluetoothManager)context.getSystemService(Context.BLUETOOTH_SERVICE);
|
|
||||||
if (btManager == null) {
|
|
||||||
// This device doesn't support Bluetooth. We should never be here, because how did
|
|
||||||
// we instantiate a device to start with?
|
|
||||||
return BluetoothProfile.STATE_DISCONNECTED;
|
|
||||||
}
|
|
||||||
|
|
||||||
return btManager.getConnectionState(mDevice, BluetoothProfile.GATT);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void reconnect() {
|
|
||||||
|
|
||||||
if (getConnectionState() != BluetoothProfile.STATE_CONNECTED) {
|
|
||||||
mGatt.disconnect();
|
|
||||||
mGatt = connectGatt();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
protected void checkConnectionForChromebookIssue() {
|
|
||||||
if (!mIsChromebook) {
|
|
||||||
// We only do this on Chromebooks, because otherwise it's really annoying to just attempt
|
|
||||||
// over and over.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int connectionState = getConnectionState();
|
|
||||||
|
|
||||||
switch (connectionState) {
|
|
||||||
case BluetoothProfile.STATE_CONNECTED:
|
|
||||||
if (!mIsConnected) {
|
|
||||||
// We are in the Bad Chromebook Place. We can force a disconnect
|
|
||||||
// to try to recover.
|
|
||||||
Log.v(TAG, "Chromebook: We are in a very bad state; the controller shows as connected in the underlying Bluetooth layer, but we never received a callback. Forcing a reconnect.");
|
|
||||||
mIsReconnecting = true;
|
|
||||||
mGatt.disconnect();
|
|
||||||
mGatt = connectGatt(false);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
else if (!isRegistered()) {
|
|
||||||
if (mGatt.getServices().size() > 0) {
|
|
||||||
Log.v(TAG, "Chromebook: We are connected to a controller, but never got our registration. Trying to recover.");
|
|
||||||
probeService(this);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
Log.v(TAG, "Chromebook: We are connected to a controller, but never discovered services. Trying to recover.");
|
|
||||||
mIsReconnecting = true;
|
|
||||||
mGatt.disconnect();
|
|
||||||
mGatt = connectGatt(false);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
Log.v(TAG, "Chromebook: We are connected, and registered. Everything's good!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case BluetoothProfile.STATE_DISCONNECTED:
|
|
||||||
Log.v(TAG, "Chromebook: We have either been disconnected, or the Chromebook BtGatt.ContextMap bug has bitten us. Attempting a disconnect/reconnect, but we may not be able to recover.");
|
|
||||||
|
|
||||||
mIsReconnecting = true;
|
|
||||||
mGatt.disconnect();
|
|
||||||
mGatt = connectGatt(false);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case BluetoothProfile.STATE_CONNECTING:
|
|
||||||
Log.v(TAG, "Chromebook: We're still trying to connect. Waiting a bit longer.");
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
final HIDDeviceBLESteamController finalThis = this;
|
|
||||||
mHandler.postDelayed(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
finalThis.checkConnectionForChromebookIssue();
|
|
||||||
}
|
|
||||||
}, CHROMEBOOK_CONNECTION_CHECK_INTERVAL);
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isRegistered() {
|
|
||||||
return mIsRegistered;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void setRegistered() {
|
|
||||||
mIsRegistered = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean probeService(HIDDeviceBLESteamController controller) {
|
|
||||||
|
|
||||||
if (isRegistered()) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!mIsConnected) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.v(TAG, "probeService controller=" + controller);
|
|
||||||
|
|
||||||
for (BluetoothGattService service : mGatt.getServices()) {
|
|
||||||
if (service.getUuid().equals(steamControllerService)) {
|
|
||||||
Log.v(TAG, "Found Valve steam controller service " + service.getUuid());
|
|
||||||
|
|
||||||
for (BluetoothGattCharacteristic chr : service.getCharacteristics()) {
|
|
||||||
if (chr.getUuid().equals(inputCharacteristic)) {
|
|
||||||
Log.v(TAG, "Found input characteristic");
|
|
||||||
// Start notifications
|
|
||||||
BluetoothGattDescriptor cccd = chr.getDescriptor(UUID.fromString("00002902-0000-1000-8000-00805f9b34fb"));
|
|
||||||
if (cccd != null) {
|
|
||||||
enableNotification(chr.getUuid());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ((mGatt.getServices().size() == 0) && mIsChromebook && !mIsReconnecting) {
|
|
||||||
Log.e(TAG, "Chromebook: Discovered services were empty; this almost certainly means the BtGatt.ContextMap bug has bitten us.");
|
|
||||||
mIsConnected = false;
|
|
||||||
mIsReconnecting = true;
|
|
||||||
mGatt.disconnect();
|
|
||||||
mGatt = connectGatt(false);
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
private void finishCurrentGattOperation() {
|
|
||||||
GattOperation op = null;
|
|
||||||
synchronized (mOperations) {
|
|
||||||
if (mCurrentOperation != null) {
|
|
||||||
op = mCurrentOperation;
|
|
||||||
mCurrentOperation = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (op != null) {
|
|
||||||
boolean result = op.finish(); // TODO: Maybe in main thread as well?
|
|
||||||
|
|
||||||
// Our operation failed, let's add it back to the beginning of our queue.
|
|
||||||
if (!result) {
|
|
||||||
mOperations.addFirst(op);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
executeNextGattOperation();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void executeNextGattOperation() {
|
|
||||||
synchronized (mOperations) {
|
|
||||||
if (mCurrentOperation != null)
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (mOperations.isEmpty())
|
|
||||||
return;
|
|
||||||
|
|
||||||
mCurrentOperation = mOperations.removeFirst();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Run in main thread
|
|
||||||
mHandler.post(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
synchronized (mOperations) {
|
|
||||||
if (mCurrentOperation == null) {
|
|
||||||
Log.e(TAG, "Current operation null in executor?");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mCurrentOperation.run();
|
|
||||||
// now wait for the GATT callback and when it comes, finish this operation
|
|
||||||
}
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
private void queueGattOperation(GattOperation op) {
|
|
||||||
synchronized (mOperations) {
|
|
||||||
mOperations.add(op);
|
|
||||||
}
|
|
||||||
executeNextGattOperation();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void enableNotification(UUID chrUuid) {
|
|
||||||
GattOperation op = HIDDeviceBLESteamController.GattOperation.enableNotification(mGatt, chrUuid);
|
|
||||||
queueGattOperation(op);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void writeCharacteristic(UUID uuid, byte[] value) {
|
|
||||||
GattOperation op = HIDDeviceBLESteamController.GattOperation.writeCharacteristic(mGatt, uuid, value);
|
|
||||||
queueGattOperation(op);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void readCharacteristic(UUID uuid) {
|
|
||||||
GattOperation op = HIDDeviceBLESteamController.GattOperation.readCharacteristic(mGatt, uuid);
|
|
||||||
queueGattOperation(op);
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
////////////// BluetoothGattCallback overridden methods
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
public void onConnectionStateChange(BluetoothGatt g, int status, int newState) {
|
|
||||||
//Log.v(TAG, "onConnectionStateChange status=" + status + " newState=" + newState);
|
|
||||||
mIsReconnecting = false;
|
|
||||||
if (newState == 2) {
|
|
||||||
mIsConnected = true;
|
|
||||||
// Run directly, without GattOperation
|
|
||||||
if (!isRegistered()) {
|
|
||||||
mHandler.post(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
mGatt.discoverServices();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (newState == 0) {
|
|
||||||
mIsConnected = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Disconnection is handled in SteamLink using the ACTION_ACL_DISCONNECTED Intent.
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onServicesDiscovered(BluetoothGatt gatt, int status) {
|
|
||||||
//Log.v(TAG, "onServicesDiscovered status=" + status);
|
|
||||||
if (status == 0) {
|
|
||||||
if (gatt.getServices().size() == 0) {
|
|
||||||
Log.v(TAG, "onServicesDiscovered returned zero services; something has gone horribly wrong down in Android's Bluetooth stack.");
|
|
||||||
mIsReconnecting = true;
|
|
||||||
mIsConnected = false;
|
|
||||||
gatt.disconnect();
|
|
||||||
mGatt = connectGatt(false);
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
probeService(this);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onCharacteristicRead(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
|
|
||||||
//Log.v(TAG, "onCharacteristicRead status=" + status + " uuid=" + characteristic.getUuid());
|
|
||||||
|
|
||||||
if (characteristic.getUuid().equals(reportCharacteristic) && !mFrozen) {
|
|
||||||
mManager.HIDDeviceFeatureReport(getId(), characteristic.getValue());
|
|
||||||
}
|
|
||||||
|
|
||||||
finishCurrentGattOperation();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onCharacteristicWrite(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic, int status) {
|
|
||||||
//Log.v(TAG, "onCharacteristicWrite status=" + status + " uuid=" + characteristic.getUuid());
|
|
||||||
|
|
||||||
if (characteristic.getUuid().equals(reportCharacteristic)) {
|
|
||||||
// Only register controller with the native side once it has been fully configured
|
|
||||||
if (!isRegistered()) {
|
|
||||||
Log.v(TAG, "Registering Steam Controller with ID: " + getId());
|
|
||||||
mManager.HIDDeviceConnected(getId(), getIdentifier(), getVendorId(), getProductId(), getSerialNumber(), getVersion(), getManufacturerName(), getProductName(), 0, 0, 0, 0);
|
|
||||||
setRegistered();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
finishCurrentGattOperation();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onCharacteristicChanged(BluetoothGatt gatt, BluetoothGattCharacteristic characteristic) {
|
|
||||||
// Enable this for verbose logging of controller input reports
|
|
||||||
//Log.v(TAG, "onCharacteristicChanged uuid=" + characteristic.getUuid() + " data=" + HexDump.dumpHexString(characteristic.getValue()));
|
|
||||||
|
|
||||||
if (characteristic.getUuid().equals(inputCharacteristic) && !mFrozen) {
|
|
||||||
mManager.HIDDeviceInputReport(getId(), characteristic.getValue());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onDescriptorRead(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status) {
|
|
||||||
//Log.v(TAG, "onDescriptorRead status=" + status);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onDescriptorWrite(BluetoothGatt gatt, BluetoothGattDescriptor descriptor, int status) {
|
|
||||||
BluetoothGattCharacteristic chr = descriptor.getCharacteristic();
|
|
||||||
//Log.v(TAG, "onDescriptorWrite status=" + status + " uuid=" + chr.getUuid() + " descriptor=" + descriptor.getUuid());
|
|
||||||
|
|
||||||
if (chr.getUuid().equals(inputCharacteristic)) {
|
|
||||||
boolean hasWrittenInputDescriptor = true;
|
|
||||||
BluetoothGattCharacteristic reportChr = chr.getService().getCharacteristic(reportCharacteristic);
|
|
||||||
if (reportChr != null) {
|
|
||||||
Log.v(TAG, "Writing report characteristic to enter valve mode");
|
|
||||||
reportChr.setValue(enterValveMode);
|
|
||||||
gatt.writeCharacteristic(reportChr);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
finishCurrentGattOperation();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onReliableWriteCompleted(BluetoothGatt gatt, int status) {
|
|
||||||
//Log.v(TAG, "onReliableWriteCompleted status=" + status);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onReadRemoteRssi(BluetoothGatt gatt, int rssi, int status) {
|
|
||||||
//Log.v(TAG, "onReadRemoteRssi status=" + status);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void onMtuChanged(BluetoothGatt gatt, int mtu, int status) {
|
|
||||||
//Log.v(TAG, "onMtuChanged status=" + status);
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//////// Public API
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getId() {
|
|
||||||
return mDeviceId;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVendorId() {
|
|
||||||
// Valve Corporation
|
|
||||||
final int VALVE_USB_VID = 0x28DE;
|
|
||||||
return VALVE_USB_VID;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getProductId() {
|
|
||||||
// We don't have an easy way to query from the Bluetooth device, but we know what it is
|
|
||||||
final int D0G_BLE2_PID = 0x1106;
|
|
||||||
return D0G_BLE2_PID;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getSerialNumber() {
|
|
||||||
// This will be read later via feature report by Steam
|
|
||||||
return "12345";
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVersion() {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getManufacturerName() {
|
|
||||||
return "Valve Corporation";
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getProductName() {
|
|
||||||
return "Steam Controller";
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public UsbDevice getDevice() {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean open() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int sendFeatureReport(byte[] report) {
|
|
||||||
if (!isRegistered()) {
|
|
||||||
Log.e(TAG, "Attempted sendFeatureReport before Steam Controller is registered!");
|
|
||||||
if (mIsConnected) {
|
|
||||||
probeService(this);
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// We need to skip the first byte, as that doesn't go over the air
|
|
||||||
byte[] actual_report = Arrays.copyOfRange(report, 1, report.length - 1);
|
|
||||||
//Log.v(TAG, "sendFeatureReport " + HexDump.dumpHexString(actual_report));
|
|
||||||
writeCharacteristic(reportCharacteristic, actual_report);
|
|
||||||
return report.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int sendOutputReport(byte[] report) {
|
|
||||||
if (!isRegistered()) {
|
|
||||||
Log.e(TAG, "Attempted sendOutputReport before Steam Controller is registered!");
|
|
||||||
if (mIsConnected) {
|
|
||||||
probeService(this);
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
//Log.v(TAG, "sendFeatureReport " + HexDump.dumpHexString(report));
|
|
||||||
writeCharacteristic(reportCharacteristic, report);
|
|
||||||
return report.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean getFeatureReport(byte[] report) {
|
|
||||||
if (!isRegistered()) {
|
|
||||||
Log.e(TAG, "Attempted getFeatureReport before Steam Controller is registered!");
|
|
||||||
if (mIsConnected) {
|
|
||||||
probeService(this);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
//Log.v(TAG, "getFeatureReport");
|
|
||||||
readCharacteristic(reportCharacteristic);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void close() {
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFrozen(boolean frozen) {
|
|
||||||
mFrozen = frozen;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void shutdown() {
|
|
||||||
close();
|
|
||||||
|
|
||||||
BluetoothGatt g = mGatt;
|
|
||||||
if (g != null) {
|
|
||||||
g.disconnect();
|
|
||||||
g.close();
|
|
||||||
mGatt = null;
|
|
||||||
}
|
|
||||||
mManager = null;
|
|
||||||
mIsRegistered = false;
|
|
||||||
mIsConnected = false;
|
|
||||||
mOperations.clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
@ -1,698 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.app.Activity;
|
|
||||||
import android.app.AlertDialog;
|
|
||||||
import android.app.PendingIntent;
|
|
||||||
import android.bluetooth.BluetoothAdapter;
|
|
||||||
import android.bluetooth.BluetoothDevice;
|
|
||||||
import android.bluetooth.BluetoothManager;
|
|
||||||
import android.bluetooth.BluetoothProfile;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.content.BroadcastReceiver;
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.DialogInterface;
|
|
||||||
import android.content.Intent;
|
|
||||||
import android.content.IntentFilter;
|
|
||||||
import android.content.SharedPreferences;
|
|
||||||
import android.content.pm.PackageManager;
|
|
||||||
import android.hardware.usb.*;
|
|
||||||
import android.os.Handler;
|
|
||||||
import android.os.Looper;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.HashMap;
|
|
||||||
import java.util.Iterator;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
public class HIDDeviceManager {
|
|
||||||
private static final String TAG = "hidapi";
|
|
||||||
private static final String ACTION_USB_PERMISSION = "org.libsdl.app.USB_PERMISSION";
|
|
||||||
|
|
||||||
private static HIDDeviceManager sManager;
|
|
||||||
private static int sManagerRefCount = 0;
|
|
||||||
|
|
||||||
public static HIDDeviceManager acquire(Context context) {
|
|
||||||
if (sManagerRefCount == 0) {
|
|
||||||
sManager = new HIDDeviceManager(context);
|
|
||||||
}
|
|
||||||
++sManagerRefCount;
|
|
||||||
return sManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void release(HIDDeviceManager manager) {
|
|
||||||
if (manager == sManager) {
|
|
||||||
--sManagerRefCount;
|
|
||||||
if (sManagerRefCount == 0) {
|
|
||||||
sManager.close();
|
|
||||||
sManager = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private Context mContext;
|
|
||||||
private HashMap<Integer, HIDDevice> mDevicesById = new HashMap<Integer, HIDDevice>();
|
|
||||||
private HashMap<BluetoothDevice, HIDDeviceBLESteamController> mBluetoothDevices = new HashMap<BluetoothDevice, HIDDeviceBLESteamController>();
|
|
||||||
private int mNextDeviceId = 0;
|
|
||||||
private SharedPreferences mSharedPreferences = null;
|
|
||||||
private boolean mIsChromebook = false;
|
|
||||||
private UsbManager mUsbManager;
|
|
||||||
private Handler mHandler;
|
|
||||||
private BluetoothManager mBluetoothManager;
|
|
||||||
private List<BluetoothDevice> mLastBluetoothDevices;
|
|
||||||
|
|
||||||
private final BroadcastReceiver mUsbBroadcast = new BroadcastReceiver() {
|
|
||||||
@Override
|
|
||||||
public void onReceive(Context context, Intent intent) {
|
|
||||||
String action = intent.getAction();
|
|
||||||
if (action.equals(UsbManager.ACTION_USB_DEVICE_ATTACHED)) {
|
|
||||||
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
|
||||||
handleUsbDeviceAttached(usbDevice);
|
|
||||||
} else if (action.equals(UsbManager.ACTION_USB_DEVICE_DETACHED)) {
|
|
||||||
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
|
||||||
handleUsbDeviceDetached(usbDevice);
|
|
||||||
} else if (action.equals(HIDDeviceManager.ACTION_USB_PERMISSION)) {
|
|
||||||
UsbDevice usbDevice = intent.getParcelableExtra(UsbManager.EXTRA_DEVICE);
|
|
||||||
handleUsbDevicePermission(usbDevice, intent.getBooleanExtra(UsbManager.EXTRA_PERMISSION_GRANTED, false));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
private final BroadcastReceiver mBluetoothBroadcast = new BroadcastReceiver() {
|
|
||||||
@Override
|
|
||||||
public void onReceive(Context context, Intent intent) {
|
|
||||||
String action = intent.getAction();
|
|
||||||
// Bluetooth device was connected. If it was a Steam Controller, handle it
|
|
||||||
if (action.equals(BluetoothDevice.ACTION_ACL_CONNECTED)) {
|
|
||||||
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
|
|
||||||
Log.d(TAG, "Bluetooth device connected: " + device);
|
|
||||||
|
|
||||||
if (isSteamController(device)) {
|
|
||||||
connectBluetoothDevice(device);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Bluetooth device was disconnected, remove from controller manager (if any)
|
|
||||||
if (action.equals(BluetoothDevice.ACTION_ACL_DISCONNECTED)) {
|
|
||||||
BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE);
|
|
||||||
Log.d(TAG, "Bluetooth device disconnected: " + device);
|
|
||||||
|
|
||||||
disconnectBluetoothDevice(device);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
private HIDDeviceManager(final Context context) {
|
|
||||||
mContext = context;
|
|
||||||
|
|
||||||
HIDDeviceRegisterCallback();
|
|
||||||
|
|
||||||
mSharedPreferences = mContext.getSharedPreferences("hidapi", Context.MODE_PRIVATE);
|
|
||||||
mIsChromebook = mContext.getPackageManager().hasSystemFeature("org.chromium.arc.device_management");
|
|
||||||
|
|
||||||
// if (shouldClear) {
|
|
||||||
// SharedPreferences.Editor spedit = mSharedPreferences.edit();
|
|
||||||
// spedit.clear();
|
|
||||||
// spedit.commit();
|
|
||||||
// }
|
|
||||||
// else
|
|
||||||
{
|
|
||||||
mNextDeviceId = mSharedPreferences.getInt("next_device_id", 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Context getContext() {
|
|
||||||
return mContext;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int getDeviceIDForIdentifier(String identifier) {
|
|
||||||
SharedPreferences.Editor spedit = mSharedPreferences.edit();
|
|
||||||
|
|
||||||
int result = mSharedPreferences.getInt(identifier, 0);
|
|
||||||
if (result == 0) {
|
|
||||||
result = mNextDeviceId++;
|
|
||||||
spedit.putInt("next_device_id", mNextDeviceId);
|
|
||||||
}
|
|
||||||
|
|
||||||
spedit.putInt(identifier, result);
|
|
||||||
spedit.commit();
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initializeUSB() {
|
|
||||||
mUsbManager = (UsbManager)mContext.getSystemService(Context.USB_SERVICE);
|
|
||||||
if (mUsbManager == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
// Logging
|
|
||||||
for (UsbDevice device : mUsbManager.getDeviceList().values()) {
|
|
||||||
Log.i(TAG,"Path: " + device.getDeviceName());
|
|
||||||
Log.i(TAG,"Manufacturer: " + device.getManufacturerName());
|
|
||||||
Log.i(TAG,"Product: " + device.getProductName());
|
|
||||||
Log.i(TAG,"ID: " + device.getDeviceId());
|
|
||||||
Log.i(TAG,"Class: " + device.getDeviceClass());
|
|
||||||
Log.i(TAG,"Protocol: " + device.getDeviceProtocol());
|
|
||||||
Log.i(TAG,"Vendor ID " + device.getVendorId());
|
|
||||||
Log.i(TAG,"Product ID: " + device.getProductId());
|
|
||||||
Log.i(TAG,"Interface count: " + device.getInterfaceCount());
|
|
||||||
Log.i(TAG,"---------------------------------------");
|
|
||||||
|
|
||||||
// Get interface details
|
|
||||||
for (int index = 0; index < device.getInterfaceCount(); index++) {
|
|
||||||
UsbInterface mUsbInterface = device.getInterface(index);
|
|
||||||
Log.i(TAG," ***** *****");
|
|
||||||
Log.i(TAG," Interface index: " + index);
|
|
||||||
Log.i(TAG," Interface ID: " + mUsbInterface.getId());
|
|
||||||
Log.i(TAG," Interface class: " + mUsbInterface.getInterfaceClass());
|
|
||||||
Log.i(TAG," Interface subclass: " + mUsbInterface.getInterfaceSubclass());
|
|
||||||
Log.i(TAG," Interface protocol: " + mUsbInterface.getInterfaceProtocol());
|
|
||||||
Log.i(TAG," Endpoint count: " + mUsbInterface.getEndpointCount());
|
|
||||||
|
|
||||||
// Get endpoint details
|
|
||||||
for (int epi = 0; epi < mUsbInterface.getEndpointCount(); epi++)
|
|
||||||
{
|
|
||||||
UsbEndpoint mEndpoint = mUsbInterface.getEndpoint(epi);
|
|
||||||
Log.i(TAG," ++++ ++++ ++++");
|
|
||||||
Log.i(TAG," Endpoint index: " + epi);
|
|
||||||
Log.i(TAG," Attributes: " + mEndpoint.getAttributes());
|
|
||||||
Log.i(TAG," Direction: " + mEndpoint.getDirection());
|
|
||||||
Log.i(TAG," Number: " + mEndpoint.getEndpointNumber());
|
|
||||||
Log.i(TAG," Interval: " + mEndpoint.getInterval());
|
|
||||||
Log.i(TAG," Packet size: " + mEndpoint.getMaxPacketSize());
|
|
||||||
Log.i(TAG," Type: " + mEndpoint.getType());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Log.i(TAG," No more devices connected.");
|
|
||||||
*/
|
|
||||||
|
|
||||||
// Register for USB broadcasts and permission completions
|
|
||||||
IntentFilter filter = new IntentFilter();
|
|
||||||
filter.addAction(UsbManager.ACTION_USB_DEVICE_ATTACHED);
|
|
||||||
filter.addAction(UsbManager.ACTION_USB_DEVICE_DETACHED);
|
|
||||||
filter.addAction(HIDDeviceManager.ACTION_USB_PERMISSION);
|
|
||||||
mContext.registerReceiver(mUsbBroadcast, filter);
|
|
||||||
|
|
||||||
for (UsbDevice usbDevice : mUsbManager.getDeviceList().values()) {
|
|
||||||
handleUsbDeviceAttached(usbDevice);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
UsbManager getUSBManager() {
|
|
||||||
return mUsbManager;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void shutdownUSB() {
|
|
||||||
try {
|
|
||||||
mContext.unregisterReceiver(mUsbBroadcast);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// We may not have registered, that's okay
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isHIDDeviceInterface(UsbDevice usbDevice, UsbInterface usbInterface) {
|
|
||||||
if (usbInterface.getInterfaceClass() == UsbConstants.USB_CLASS_HID) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
if (isXbox360Controller(usbDevice, usbInterface) || isXboxOneController(usbDevice, usbInterface)) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isXbox360Controller(UsbDevice usbDevice, UsbInterface usbInterface) {
|
|
||||||
final int XB360_IFACE_SUBCLASS = 93;
|
|
||||||
final int XB360_IFACE_PROTOCOL = 1; // Wired
|
|
||||||
final int XB360W_IFACE_PROTOCOL = 129; // Wireless
|
|
||||||
final int[] SUPPORTED_VENDORS = {
|
|
||||||
0x0079, // GPD Win 2
|
|
||||||
0x044f, // Thrustmaster
|
|
||||||
0x045e, // Microsoft
|
|
||||||
0x046d, // Logitech
|
|
||||||
0x056e, // Elecom
|
|
||||||
0x06a3, // Saitek
|
|
||||||
0x0738, // Mad Catz
|
|
||||||
0x07ff, // Mad Catz
|
|
||||||
0x0e6f, // PDP
|
|
||||||
0x0f0d, // Hori
|
|
||||||
0x1038, // SteelSeries
|
|
||||||
0x11c9, // Nacon
|
|
||||||
0x12ab, // Unknown
|
|
||||||
0x1430, // RedOctane
|
|
||||||
0x146b, // BigBen
|
|
||||||
0x1532, // Razer Sabertooth
|
|
||||||
0x15e4, // Numark
|
|
||||||
0x162e, // Joytech
|
|
||||||
0x1689, // Razer Onza
|
|
||||||
0x1949, // Lab126, Inc.
|
|
||||||
0x1bad, // Harmonix
|
|
||||||
0x20d6, // PowerA
|
|
||||||
0x24c6, // PowerA
|
|
||||||
0x2c22, // Qanba
|
|
||||||
0x2dc8, // 8BitDo
|
|
||||||
0x9886, // ASTRO Gaming
|
|
||||||
};
|
|
||||||
|
|
||||||
if (usbInterface.getInterfaceClass() == UsbConstants.USB_CLASS_VENDOR_SPEC &&
|
|
||||||
usbInterface.getInterfaceSubclass() == XB360_IFACE_SUBCLASS &&
|
|
||||||
(usbInterface.getInterfaceProtocol() == XB360_IFACE_PROTOCOL ||
|
|
||||||
usbInterface.getInterfaceProtocol() == XB360W_IFACE_PROTOCOL)) {
|
|
||||||
int vendor_id = usbDevice.getVendorId();
|
|
||||||
for (int supportedVid : SUPPORTED_VENDORS) {
|
|
||||||
if (vendor_id == supportedVid) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private boolean isXboxOneController(UsbDevice usbDevice, UsbInterface usbInterface) {
|
|
||||||
final int XB1_IFACE_SUBCLASS = 71;
|
|
||||||
final int XB1_IFACE_PROTOCOL = 208;
|
|
||||||
final int[] SUPPORTED_VENDORS = {
|
|
||||||
0x03f0, // HP
|
|
||||||
0x044f, // Thrustmaster
|
|
||||||
0x045e, // Microsoft
|
|
||||||
0x0738, // Mad Catz
|
|
||||||
0x0b05, // ASUS
|
|
||||||
0x0e6f, // PDP
|
|
||||||
0x0f0d, // Hori
|
|
||||||
0x10f5, // Turtle Beach
|
|
||||||
0x1532, // Razer Wildcat
|
|
||||||
0x20d6, // PowerA
|
|
||||||
0x24c6, // PowerA
|
|
||||||
0x2dc8, // 8BitDo
|
|
||||||
0x2e24, // Hyperkin
|
|
||||||
0x3537, // GameSir
|
|
||||||
};
|
|
||||||
|
|
||||||
if (usbInterface.getId() == 0 &&
|
|
||||||
usbInterface.getInterfaceClass() == UsbConstants.USB_CLASS_VENDOR_SPEC &&
|
|
||||||
usbInterface.getInterfaceSubclass() == XB1_IFACE_SUBCLASS &&
|
|
||||||
usbInterface.getInterfaceProtocol() == XB1_IFACE_PROTOCOL) {
|
|
||||||
int vendor_id = usbDevice.getVendorId();
|
|
||||||
for (int supportedVid : SUPPORTED_VENDORS) {
|
|
||||||
if (vendor_id == supportedVid) {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleUsbDeviceAttached(UsbDevice usbDevice) {
|
|
||||||
connectHIDDeviceUSB(usbDevice);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleUsbDeviceDetached(UsbDevice usbDevice) {
|
|
||||||
List<Integer> devices = new ArrayList<Integer>();
|
|
||||||
for (HIDDevice device : mDevicesById.values()) {
|
|
||||||
if (usbDevice.equals(device.getDevice())) {
|
|
||||||
devices.add(device.getId());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (int id : devices) {
|
|
||||||
HIDDevice device = mDevicesById.get(id);
|
|
||||||
mDevicesById.remove(id);
|
|
||||||
device.shutdown();
|
|
||||||
HIDDeviceDisconnected(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void handleUsbDevicePermission(UsbDevice usbDevice, boolean permission_granted) {
|
|
||||||
for (HIDDevice device : mDevicesById.values()) {
|
|
||||||
if (usbDevice.equals(device.getDevice())) {
|
|
||||||
boolean opened = false;
|
|
||||||
if (permission_granted) {
|
|
||||||
opened = device.open();
|
|
||||||
}
|
|
||||||
HIDDeviceOpenResult(device.getId(), opened);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void connectHIDDeviceUSB(UsbDevice usbDevice) {
|
|
||||||
synchronized (this) {
|
|
||||||
int interface_mask = 0;
|
|
||||||
for (int interface_index = 0; interface_index < usbDevice.getInterfaceCount(); interface_index++) {
|
|
||||||
UsbInterface usbInterface = usbDevice.getInterface(interface_index);
|
|
||||||
if (isHIDDeviceInterface(usbDevice, usbInterface)) {
|
|
||||||
// Check to see if we've already added this interface
|
|
||||||
// This happens with the Xbox Series X controller which has a duplicate interface 0, which is inactive
|
|
||||||
int interface_id = usbInterface.getId();
|
|
||||||
if ((interface_mask & (1 << interface_id)) != 0) {
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
interface_mask |= (1 << interface_id);
|
|
||||||
|
|
||||||
HIDDeviceUSB device = new HIDDeviceUSB(this, usbDevice, interface_index);
|
|
||||||
int id = device.getId();
|
|
||||||
mDevicesById.put(id, device);
|
|
||||||
HIDDeviceConnected(id, device.getIdentifier(), device.getVendorId(), device.getProductId(), device.getSerialNumber(), device.getVersion(), device.getManufacturerName(), device.getProductName(), usbInterface.getId(), usbInterface.getInterfaceClass(), usbInterface.getInterfaceSubclass(), usbInterface.getInterfaceProtocol());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void initializeBluetooth() {
|
|
||||||
Log.d(TAG, "Initializing Bluetooth");
|
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT >= 31 /* Android 12 */ &&
|
|
||||||
mContext.getPackageManager().checkPermission(android.Manifest.permission.BLUETOOTH_CONNECT, mContext.getPackageName()) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
Log.d(TAG, "Couldn't initialize Bluetooth, missing android.permission.BLUETOOTH_CONNECT");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT <= 30 /* Android 11.0 (R) */ &&
|
|
||||||
mContext.getPackageManager().checkPermission(android.Manifest.permission.BLUETOOTH, mContext.getPackageName()) != PackageManager.PERMISSION_GRANTED) {
|
|
||||||
Log.d(TAG, "Couldn't initialize Bluetooth, missing android.permission.BLUETOOTH");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!mContext.getPackageManager().hasSystemFeature(PackageManager.FEATURE_BLUETOOTH_LE) || (Build.VERSION.SDK_INT < 18 /* Android 4.3 (JELLY_BEAN_MR2) */)) {
|
|
||||||
Log.d(TAG, "Couldn't initialize Bluetooth, this version of Android does not support Bluetooth LE");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find bonded bluetooth controllers and create SteamControllers for them
|
|
||||||
mBluetoothManager = (BluetoothManager)mContext.getSystemService(Context.BLUETOOTH_SERVICE);
|
|
||||||
if (mBluetoothManager == null) {
|
|
||||||
// This device doesn't support Bluetooth.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
BluetoothAdapter btAdapter = mBluetoothManager.getAdapter();
|
|
||||||
if (btAdapter == null) {
|
|
||||||
// This device has Bluetooth support in the codebase, but has no available adapters.
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Get our bonded devices.
|
|
||||||
for (BluetoothDevice device : btAdapter.getBondedDevices()) {
|
|
||||||
|
|
||||||
Log.d(TAG, "Bluetooth device available: " + device);
|
|
||||||
if (isSteamController(device)) {
|
|
||||||
connectBluetoothDevice(device);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
// NOTE: These don't work on Chromebooks, to my undying dismay.
|
|
||||||
IntentFilter filter = new IntentFilter();
|
|
||||||
filter.addAction(BluetoothDevice.ACTION_ACL_CONNECTED);
|
|
||||||
filter.addAction(BluetoothDevice.ACTION_ACL_DISCONNECTED);
|
|
||||||
mContext.registerReceiver(mBluetoothBroadcast, filter);
|
|
||||||
|
|
||||||
if (mIsChromebook) {
|
|
||||||
mHandler = new Handler(Looper.getMainLooper());
|
|
||||||
mLastBluetoothDevices = new ArrayList<BluetoothDevice>();
|
|
||||||
|
|
||||||
// final HIDDeviceManager finalThis = this;
|
|
||||||
// mHandler.postDelayed(new Runnable() {
|
|
||||||
// @Override
|
|
||||||
// public void run() {
|
|
||||||
// finalThis.chromebookConnectionHandler();
|
|
||||||
// }
|
|
||||||
// }, 5000);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private void shutdownBluetooth() {
|
|
||||||
try {
|
|
||||||
mContext.unregisterReceiver(mBluetoothBroadcast);
|
|
||||||
} catch (Exception e) {
|
|
||||||
// We may not have registered, that's okay
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Chromebooks do not pass along ACTION_ACL_CONNECTED / ACTION_ACL_DISCONNECTED properly.
|
|
||||||
// This function provides a sort of dummy version of that, watching for changes in the
|
|
||||||
// connected devices and attempting to add controllers as things change.
|
|
||||||
public void chromebookConnectionHandler() {
|
|
||||||
if (!mIsChromebook) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
ArrayList<BluetoothDevice> disconnected = new ArrayList<BluetoothDevice>();
|
|
||||||
ArrayList<BluetoothDevice> connected = new ArrayList<BluetoothDevice>();
|
|
||||||
|
|
||||||
List<BluetoothDevice> currentConnected = mBluetoothManager.getConnectedDevices(BluetoothProfile.GATT);
|
|
||||||
|
|
||||||
for (BluetoothDevice bluetoothDevice : currentConnected) {
|
|
||||||
if (!mLastBluetoothDevices.contains(bluetoothDevice)) {
|
|
||||||
connected.add(bluetoothDevice);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (BluetoothDevice bluetoothDevice : mLastBluetoothDevices) {
|
|
||||||
if (!currentConnected.contains(bluetoothDevice)) {
|
|
||||||
disconnected.add(bluetoothDevice);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mLastBluetoothDevices = currentConnected;
|
|
||||||
|
|
||||||
for (BluetoothDevice bluetoothDevice : disconnected) {
|
|
||||||
disconnectBluetoothDevice(bluetoothDevice);
|
|
||||||
}
|
|
||||||
for (BluetoothDevice bluetoothDevice : connected) {
|
|
||||||
connectBluetoothDevice(bluetoothDevice);
|
|
||||||
}
|
|
||||||
|
|
||||||
final HIDDeviceManager finalThis = this;
|
|
||||||
mHandler.postDelayed(new Runnable() {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
finalThis.chromebookConnectionHandler();
|
|
||||||
}
|
|
||||||
}, 10000);
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean connectBluetoothDevice(BluetoothDevice bluetoothDevice) {
|
|
||||||
Log.v(TAG, "connectBluetoothDevice device=" + bluetoothDevice);
|
|
||||||
synchronized (this) {
|
|
||||||
if (mBluetoothDevices.containsKey(bluetoothDevice)) {
|
|
||||||
Log.v(TAG, "Steam controller with address " + bluetoothDevice + " already exists, attempting reconnect");
|
|
||||||
|
|
||||||
HIDDeviceBLESteamController device = mBluetoothDevices.get(bluetoothDevice);
|
|
||||||
device.reconnect();
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
HIDDeviceBLESteamController device = new HIDDeviceBLESteamController(this, bluetoothDevice);
|
|
||||||
int id = device.getId();
|
|
||||||
mBluetoothDevices.put(bluetoothDevice, device);
|
|
||||||
mDevicesById.put(id, device);
|
|
||||||
|
|
||||||
// The Steam Controller will mark itself connected once initialization is complete
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void disconnectBluetoothDevice(BluetoothDevice bluetoothDevice) {
|
|
||||||
synchronized (this) {
|
|
||||||
HIDDeviceBLESteamController device = mBluetoothDevices.get(bluetoothDevice);
|
|
||||||
if (device == null)
|
|
||||||
return;
|
|
||||||
|
|
||||||
int id = device.getId();
|
|
||||||
mBluetoothDevices.remove(bluetoothDevice);
|
|
||||||
mDevicesById.remove(id);
|
|
||||||
device.shutdown();
|
|
||||||
HIDDeviceDisconnected(id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean isSteamController(BluetoothDevice bluetoothDevice) {
|
|
||||||
// Sanity check. If you pass in a null device, by definition it is never a Steam Controller.
|
|
||||||
if (bluetoothDevice == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// If the device has no local name, we really don't want to try an equality check against it.
|
|
||||||
if (bluetoothDevice.getName() == null) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return bluetoothDevice.getName().equals("SteamController") && ((bluetoothDevice.getType() & BluetoothDevice.DEVICE_TYPE_LE) != 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void close() {
|
|
||||||
shutdownUSB();
|
|
||||||
shutdownBluetooth();
|
|
||||||
synchronized (this) {
|
|
||||||
for (HIDDevice device : mDevicesById.values()) {
|
|
||||||
device.shutdown();
|
|
||||||
}
|
|
||||||
mDevicesById.clear();
|
|
||||||
mBluetoothDevices.clear();
|
|
||||||
HIDDeviceReleaseCallback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void setFrozen(boolean frozen) {
|
|
||||||
synchronized (this) {
|
|
||||||
for (HIDDevice device : mDevicesById.values()) {
|
|
||||||
device.setFrozen(frozen);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
private HIDDevice getDevice(int id) {
|
|
||||||
synchronized (this) {
|
|
||||||
HIDDevice result = mDevicesById.get(id);
|
|
||||||
if (result == null) {
|
|
||||||
Log.v(TAG, "No device for id: " + id);
|
|
||||||
Log.v(TAG, "Available devices: " + mDevicesById.keySet());
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
////////// JNI interface functions
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
public boolean initialize(boolean usb, boolean bluetooth) {
|
|
||||||
Log.v(TAG, "initialize(" + usb + ", " + bluetooth + ")");
|
|
||||||
|
|
||||||
if (usb) {
|
|
||||||
initializeUSB();
|
|
||||||
}
|
|
||||||
if (bluetooth) {
|
|
||||||
initializeBluetooth();
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean openDevice(int deviceID) {
|
|
||||||
Log.v(TAG, "openDevice deviceID=" + deviceID);
|
|
||||||
HIDDevice device = getDevice(deviceID);
|
|
||||||
if (device == null) {
|
|
||||||
HIDDeviceDisconnected(deviceID);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Look to see if this is a USB device and we have permission to access it
|
|
||||||
UsbDevice usbDevice = device.getDevice();
|
|
||||||
if (usbDevice != null && !mUsbManager.hasPermission(usbDevice)) {
|
|
||||||
HIDDeviceOpenPending(deviceID);
|
|
||||||
try {
|
|
||||||
final int FLAG_MUTABLE = 0x02000000; // PendingIntent.FLAG_MUTABLE, but don't require SDK 31
|
|
||||||
int flags;
|
|
||||||
if (Build.VERSION.SDK_INT >= 31 /* Android 12.0 (S) */) {
|
|
||||||
flags = FLAG_MUTABLE;
|
|
||||||
} else {
|
|
||||||
flags = 0;
|
|
||||||
}
|
|
||||||
if (Build.VERSION.SDK_INT >= 33 /* Android 14.0 (U) */) {
|
|
||||||
Intent intent = new Intent(HIDDeviceManager.ACTION_USB_PERMISSION);
|
|
||||||
intent.setPackage(mContext.getPackageName());
|
|
||||||
mUsbManager.requestPermission(usbDevice, PendingIntent.getBroadcast(mContext, 0, intent, flags));
|
|
||||||
} else {
|
|
||||||
mUsbManager.requestPermission(usbDevice, PendingIntent.getBroadcast(mContext, 0, new Intent(HIDDeviceManager.ACTION_USB_PERMISSION), flags));
|
|
||||||
}
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.v(TAG, "Couldn't request permission for USB device " + usbDevice);
|
|
||||||
HIDDeviceOpenResult(deviceID, false);
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
return device.open();
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(TAG, "Got exception: " + Log.getStackTraceString(e));
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int sendOutputReport(int deviceID, byte[] report) {
|
|
||||||
try {
|
|
||||||
//Log.v(TAG, "sendOutputReport deviceID=" + deviceID + " length=" + report.length);
|
|
||||||
HIDDevice device;
|
|
||||||
device = getDevice(deviceID);
|
|
||||||
if (device == null) {
|
|
||||||
HIDDeviceDisconnected(deviceID);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return device.sendOutputReport(report);
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(TAG, "Got exception: " + Log.getStackTraceString(e));
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int sendFeatureReport(int deviceID, byte[] report) {
|
|
||||||
try {
|
|
||||||
//Log.v(TAG, "sendFeatureReport deviceID=" + deviceID + " length=" + report.length);
|
|
||||||
HIDDevice device;
|
|
||||||
device = getDevice(deviceID);
|
|
||||||
if (device == null) {
|
|
||||||
HIDDeviceDisconnected(deviceID);
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return device.sendFeatureReport(report);
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(TAG, "Got exception: " + Log.getStackTraceString(e));
|
|
||||||
}
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean getFeatureReport(int deviceID, byte[] report) {
|
|
||||||
try {
|
|
||||||
//Log.v(TAG, "getFeatureReport deviceID=" + deviceID);
|
|
||||||
HIDDevice device;
|
|
||||||
device = getDevice(deviceID);
|
|
||||||
if (device == null) {
|
|
||||||
HIDDeviceDisconnected(deviceID);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return device.getFeatureReport(report);
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(TAG, "Got exception: " + Log.getStackTraceString(e));
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void closeDevice(int deviceID) {
|
|
||||||
try {
|
|
||||||
Log.v(TAG, "closeDevice deviceID=" + deviceID);
|
|
||||||
HIDDevice device;
|
|
||||||
device = getDevice(deviceID);
|
|
||||||
if (device == null) {
|
|
||||||
HIDDeviceDisconnected(deviceID);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
device.close();
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.e(TAG, "Got exception: " + Log.getStackTraceString(e));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
/////////////// Native methods
|
|
||||||
//////////////////////////////////////////////////////////////////////////////////////////////////////
|
|
||||||
|
|
||||||
private native void HIDDeviceRegisterCallback();
|
|
||||||
private native void HIDDeviceReleaseCallback();
|
|
||||||
|
|
||||||
native void HIDDeviceConnected(int deviceID, String identifier, int vendorId, int productId, String serial_number, int release_number, String manufacturer_string, String product_string, int interface_number, int interface_class, int interface_subclass, int interface_protocol);
|
|
||||||
native void HIDDeviceOpenPending(int deviceID);
|
|
||||||
native void HIDDeviceOpenResult(int deviceID, boolean opened);
|
|
||||||
native void HIDDeviceDisconnected(int deviceID);
|
|
||||||
|
|
||||||
native void HIDDeviceInputReport(int deviceID, byte[] report);
|
|
||||||
native void HIDDeviceFeatureReport(int deviceID, byte[] report);
|
|
||||||
}
|
|
@ -1,309 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.hardware.usb.*;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.Log;
|
|
||||||
import java.util.Arrays;
|
|
||||||
|
|
||||||
class HIDDeviceUSB implements HIDDevice {
|
|
||||||
|
|
||||||
private static final String TAG = "hidapi";
|
|
||||||
|
|
||||||
protected HIDDeviceManager mManager;
|
|
||||||
protected UsbDevice mDevice;
|
|
||||||
protected int mInterfaceIndex;
|
|
||||||
protected int mInterface;
|
|
||||||
protected int mDeviceId;
|
|
||||||
protected UsbDeviceConnection mConnection;
|
|
||||||
protected UsbEndpoint mInputEndpoint;
|
|
||||||
protected UsbEndpoint mOutputEndpoint;
|
|
||||||
protected InputThread mInputThread;
|
|
||||||
protected boolean mRunning;
|
|
||||||
protected boolean mFrozen;
|
|
||||||
|
|
||||||
public HIDDeviceUSB(HIDDeviceManager manager, UsbDevice usbDevice, int interface_index) {
|
|
||||||
mManager = manager;
|
|
||||||
mDevice = usbDevice;
|
|
||||||
mInterfaceIndex = interface_index;
|
|
||||||
mInterface = mDevice.getInterface(mInterfaceIndex).getId();
|
|
||||||
mDeviceId = manager.getDeviceIDForIdentifier(getIdentifier());
|
|
||||||
mRunning = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getIdentifier() {
|
|
||||||
return String.format("%s/%x/%x/%d", mDevice.getDeviceName(), mDevice.getVendorId(), mDevice.getProductId(), mInterfaceIndex);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getId() {
|
|
||||||
return mDeviceId;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVendorId() {
|
|
||||||
return mDevice.getVendorId();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getProductId() {
|
|
||||||
return mDevice.getProductId();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getSerialNumber() {
|
|
||||||
String result = null;
|
|
||||||
if (Build.VERSION.SDK_INT >= 21 /* Android 5.0 (LOLLIPOP) */) {
|
|
||||||
try {
|
|
||||||
result = mDevice.getSerialNumber();
|
|
||||||
}
|
|
||||||
catch (SecurityException exception) {
|
|
||||||
//Log.w(TAG, "App permissions mean we cannot get serial number for device " + getDeviceName() + " message: " + exception.getMessage());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (result == null) {
|
|
||||||
result = "";
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVersion() {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getManufacturerName() {
|
|
||||||
String result = null;
|
|
||||||
if (Build.VERSION.SDK_INT >= 21 /* Android 5.0 (LOLLIPOP) */) {
|
|
||||||
result = mDevice.getManufacturerName();
|
|
||||||
}
|
|
||||||
if (result == null) {
|
|
||||||
result = String.format("%x", getVendorId());
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public String getProductName() {
|
|
||||||
String result = null;
|
|
||||||
if (Build.VERSION.SDK_INT >= 21 /* Android 5.0 (LOLLIPOP) */) {
|
|
||||||
result = mDevice.getProductName();
|
|
||||||
}
|
|
||||||
if (result == null) {
|
|
||||||
result = String.format("%x", getProductId());
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public UsbDevice getDevice() {
|
|
||||||
return mDevice;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getDeviceName() {
|
|
||||||
return getManufacturerName() + " " + getProductName() + "(0x" + String.format("%x", getVendorId()) + "/0x" + String.format("%x", getProductId()) + ")";
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean open() {
|
|
||||||
mConnection = mManager.getUSBManager().openDevice(mDevice);
|
|
||||||
if (mConnection == null) {
|
|
||||||
Log.w(TAG, "Unable to open USB device " + getDeviceName());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Force claim our interface
|
|
||||||
UsbInterface iface = mDevice.getInterface(mInterfaceIndex);
|
|
||||||
if (!mConnection.claimInterface(iface, true)) {
|
|
||||||
Log.w(TAG, "Failed to claim interfaces on USB device " + getDeviceName());
|
|
||||||
close();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Find the endpoints
|
|
||||||
for (int j = 0; j < iface.getEndpointCount(); j++) {
|
|
||||||
UsbEndpoint endpt = iface.getEndpoint(j);
|
|
||||||
switch (endpt.getDirection()) {
|
|
||||||
case UsbConstants.USB_DIR_IN:
|
|
||||||
if (mInputEndpoint == null) {
|
|
||||||
mInputEndpoint = endpt;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
case UsbConstants.USB_DIR_OUT:
|
|
||||||
if (mOutputEndpoint == null) {
|
|
||||||
mOutputEndpoint = endpt;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the required endpoints were present
|
|
||||||
if (mInputEndpoint == null || mOutputEndpoint == null) {
|
|
||||||
Log.w(TAG, "Missing required endpoint on USB device " + getDeviceName());
|
|
||||||
close();
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Start listening for input
|
|
||||||
mRunning = true;
|
|
||||||
mInputThread = new InputThread();
|
|
||||||
mInputThread.start();
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int sendFeatureReport(byte[] report) {
|
|
||||||
int res = -1;
|
|
||||||
int offset = 0;
|
|
||||||
int length = report.length;
|
|
||||||
boolean skipped_report_id = false;
|
|
||||||
byte report_number = report[0];
|
|
||||||
|
|
||||||
if (report_number == 0x0) {
|
|
||||||
++offset;
|
|
||||||
--length;
|
|
||||||
skipped_report_id = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
res = mConnection.controlTransfer(
|
|
||||||
UsbConstants.USB_TYPE_CLASS | 0x01 /*RECIPIENT_INTERFACE*/ | UsbConstants.USB_DIR_OUT,
|
|
||||||
0x09/*HID set_report*/,
|
|
||||||
(3/*HID feature*/ << 8) | report_number,
|
|
||||||
mInterface,
|
|
||||||
report, offset, length,
|
|
||||||
1000/*timeout millis*/);
|
|
||||||
|
|
||||||
if (res < 0) {
|
|
||||||
Log.w(TAG, "sendFeatureReport() returned " + res + " on device " + getDeviceName());
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (skipped_report_id) {
|
|
||||||
++length;
|
|
||||||
}
|
|
||||||
return length;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int sendOutputReport(byte[] report) {
|
|
||||||
int r = mConnection.bulkTransfer(mOutputEndpoint, report, report.length, 1000);
|
|
||||||
if (r != report.length) {
|
|
||||||
Log.w(TAG, "sendOutputReport() returned " + r + " on device " + getDeviceName());
|
|
||||||
}
|
|
||||||
return r;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean getFeatureReport(byte[] report) {
|
|
||||||
int res = -1;
|
|
||||||
int offset = 0;
|
|
||||||
int length = report.length;
|
|
||||||
boolean skipped_report_id = false;
|
|
||||||
byte report_number = report[0];
|
|
||||||
|
|
||||||
if (report_number == 0x0) {
|
|
||||||
/* Offset the return buffer by 1, so that the report ID
|
|
||||||
will remain in byte 0. */
|
|
||||||
++offset;
|
|
||||||
--length;
|
|
||||||
skipped_report_id = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
res = mConnection.controlTransfer(
|
|
||||||
UsbConstants.USB_TYPE_CLASS | 0x01 /*RECIPIENT_INTERFACE*/ | UsbConstants.USB_DIR_IN,
|
|
||||||
0x01/*HID get_report*/,
|
|
||||||
(3/*HID feature*/ << 8) | report_number,
|
|
||||||
mInterface,
|
|
||||||
report, offset, length,
|
|
||||||
1000/*timeout millis*/);
|
|
||||||
|
|
||||||
if (res < 0) {
|
|
||||||
Log.w(TAG, "getFeatureReport() returned " + res + " on device " + getDeviceName());
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (skipped_report_id) {
|
|
||||||
++res;
|
|
||||||
++length;
|
|
||||||
}
|
|
||||||
|
|
||||||
byte[] data;
|
|
||||||
if (res == length) {
|
|
||||||
data = report;
|
|
||||||
} else {
|
|
||||||
data = Arrays.copyOfRange(report, 0, res);
|
|
||||||
}
|
|
||||||
mManager.HIDDeviceFeatureReport(mDeviceId, data);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void close() {
|
|
||||||
mRunning = false;
|
|
||||||
if (mInputThread != null) {
|
|
||||||
while (mInputThread.isAlive()) {
|
|
||||||
mInputThread.interrupt();
|
|
||||||
try {
|
|
||||||
mInputThread.join();
|
|
||||||
} catch (InterruptedException e) {
|
|
||||||
// Keep trying until we're done
|
|
||||||
}
|
|
||||||
}
|
|
||||||
mInputThread = null;
|
|
||||||
}
|
|
||||||
if (mConnection != null) {
|
|
||||||
UsbInterface iface = mDevice.getInterface(mInterfaceIndex);
|
|
||||||
mConnection.releaseInterface(iface);
|
|
||||||
mConnection.close();
|
|
||||||
mConnection = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void shutdown() {
|
|
||||||
close();
|
|
||||||
mManager = null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void setFrozen(boolean frozen) {
|
|
||||||
mFrozen = frozen;
|
|
||||||
}
|
|
||||||
|
|
||||||
protected class InputThread extends Thread {
|
|
||||||
@Override
|
|
||||||
public void run() {
|
|
||||||
int packetSize = mInputEndpoint.getMaxPacketSize();
|
|
||||||
byte[] packet = new byte[packetSize];
|
|
||||||
while (mRunning) {
|
|
||||||
int r;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
r = mConnection.bulkTransfer(mInputEndpoint, packet, packetSize, 1000);
|
|
||||||
}
|
|
||||||
catch (Exception e)
|
|
||||||
{
|
|
||||||
Log.v(TAG, "Exception in UsbDeviceConnection bulktransfer: " + e);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
if (r < 0) {
|
|
||||||
// Could be a timeout or an I/O error
|
|
||||||
}
|
|
||||||
if (r > 0) {
|
|
||||||
byte[] data;
|
|
||||||
if (r == packetSize) {
|
|
||||||
data = packet;
|
|
||||||
} else {
|
|
||||||
data = Arrays.copyOfRange(packet, 0, r);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!mFrozen) {
|
|
||||||
mManager.HIDDeviceInputReport(mDeviceId, data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,90 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
|
|
||||||
import java.lang.Class;
|
|
||||||
import java.lang.reflect.Method;
|
|
||||||
|
|
||||||
/**
|
|
||||||
SDL library initialization
|
|
||||||
*/
|
|
||||||
public class SDL {
|
|
||||||
|
|
||||||
// This function should be called first and sets up the native code
|
|
||||||
// so it can call into the Java classes
|
|
||||||
public static void setupJNI() {
|
|
||||||
SDLActivity.nativeSetupJNI();
|
|
||||||
SDLAudioManager.nativeSetupJNI();
|
|
||||||
SDLControllerManager.nativeSetupJNI();
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function should be called each time the activity is started
|
|
||||||
public static void initialize() {
|
|
||||||
setContext(null);
|
|
||||||
|
|
||||||
SDLActivity.initialize();
|
|
||||||
SDLAudioManager.initialize();
|
|
||||||
SDLControllerManager.initialize();
|
|
||||||
}
|
|
||||||
|
|
||||||
// This function stores the current activity (SDL or not)
|
|
||||||
public static void setContext(Context context) {
|
|
||||||
SDLAudioManager.setContext(context);
|
|
||||||
mContext = context;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Context getContext() {
|
|
||||||
return mContext;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void loadLibrary(String libraryName) throws UnsatisfiedLinkError, SecurityException, NullPointerException {
|
|
||||||
loadLibrary(libraryName, mContext);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void loadLibrary(String libraryName, Context context) throws UnsatisfiedLinkError, SecurityException, NullPointerException {
|
|
||||||
|
|
||||||
if (libraryName == null) {
|
|
||||||
throw new NullPointerException("No library name provided.");
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
// Let's see if we have ReLinker available in the project. This is necessary for
|
|
||||||
// some projects that have huge numbers of local libraries bundled, and thus may
|
|
||||||
// trip a bug in Android's native library loader which ReLinker works around. (If
|
|
||||||
// loadLibrary works properly, ReLinker will simply use the normal Android method
|
|
||||||
// internally.)
|
|
||||||
//
|
|
||||||
// To use ReLinker, just add it as a dependency. For more information, see
|
|
||||||
// https://github.com/KeepSafe/ReLinker for ReLinker's repository.
|
|
||||||
//
|
|
||||||
Class<?> relinkClass = context.getClassLoader().loadClass("com.getkeepsafe.relinker.ReLinker");
|
|
||||||
Class<?> relinkListenerClass = context.getClassLoader().loadClass("com.getkeepsafe.relinker.ReLinker$LoadListener");
|
|
||||||
Class<?> contextClass = context.getClassLoader().loadClass("android.content.Context");
|
|
||||||
Class<?> stringClass = context.getClassLoader().loadClass("java.lang.String");
|
|
||||||
|
|
||||||
// Get a 'force' instance of the ReLinker, so we can ensure libraries are reinstalled if
|
|
||||||
// they've changed during updates.
|
|
||||||
Method forceMethod = relinkClass.getDeclaredMethod("force");
|
|
||||||
Object relinkInstance = forceMethod.invoke(null);
|
|
||||||
Class<?> relinkInstanceClass = relinkInstance.getClass();
|
|
||||||
|
|
||||||
// Actually load the library!
|
|
||||||
Method loadMethod = relinkInstanceClass.getDeclaredMethod("loadLibrary", contextClass, stringClass, stringClass, relinkListenerClass);
|
|
||||||
loadMethod.invoke(relinkInstance, context, libraryName, null, null);
|
|
||||||
}
|
|
||||||
catch (final Throwable e) {
|
|
||||||
// Fall back
|
|
||||||
try {
|
|
||||||
System.loadLibrary(libraryName);
|
|
||||||
}
|
|
||||||
catch (final UnsatisfiedLinkError ule) {
|
|
||||||
throw ule;
|
|
||||||
}
|
|
||||||
catch (final SecurityException se) {
|
|
||||||
throw se;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static Context mContext;
|
|
||||||
}
|
|
@ -1,514 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.media.AudioDeviceCallback;
|
|
||||||
import android.media.AudioDeviceInfo;
|
|
||||||
import android.media.AudioFormat;
|
|
||||||
import android.media.AudioManager;
|
|
||||||
import android.media.AudioRecord;
|
|
||||||
import android.media.AudioTrack;
|
|
||||||
import android.media.MediaRecorder;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.Log;
|
|
||||||
|
|
||||||
import java.util.Arrays;
|
|
||||||
|
|
||||||
public class SDLAudioManager {
|
|
||||||
protected static final String TAG = "SDLAudio";
|
|
||||||
|
|
||||||
protected static AudioTrack mAudioTrack;
|
|
||||||
protected static AudioRecord mAudioRecord;
|
|
||||||
protected static Context mContext;
|
|
||||||
|
|
||||||
private static final int[] NO_DEVICES = {};
|
|
||||||
|
|
||||||
private static AudioDeviceCallback mAudioDeviceCallback;
|
|
||||||
|
|
||||||
public static void initialize() {
|
|
||||||
mAudioTrack = null;
|
|
||||||
mAudioRecord = null;
|
|
||||||
mAudioDeviceCallback = null;
|
|
||||||
|
|
||||||
if(Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */)
|
|
||||||
{
|
|
||||||
mAudioDeviceCallback = new AudioDeviceCallback() {
|
|
||||||
@Override
|
|
||||||
public void onAudioDevicesAdded(AudioDeviceInfo[] addedDevices) {
|
|
||||||
Arrays.stream(addedDevices).forEach(deviceInfo -> addAudioDevice(deviceInfo.isSink(), deviceInfo.getId()));
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onAudioDevicesRemoved(AudioDeviceInfo[] removedDevices) {
|
|
||||||
Arrays.stream(removedDevices).forEach(deviceInfo -> removeAudioDevice(deviceInfo.isSink(), deviceInfo.getId()));
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void setContext(Context context) {
|
|
||||||
mContext = context;
|
|
||||||
if (context != null) {
|
|
||||||
registerAudioDeviceCallback();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void release(Context context) {
|
|
||||||
unregisterAudioDeviceCallback(context);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Audio
|
|
||||||
|
|
||||||
protected static String getAudioFormatString(int audioFormat) {
|
|
||||||
switch (audioFormat) {
|
|
||||||
case AudioFormat.ENCODING_PCM_8BIT:
|
|
||||||
return "8-bit";
|
|
||||||
case AudioFormat.ENCODING_PCM_16BIT:
|
|
||||||
return "16-bit";
|
|
||||||
case AudioFormat.ENCODING_PCM_FLOAT:
|
|
||||||
return "float";
|
|
||||||
default:
|
|
||||||
return Integer.toString(audioFormat);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected static int[] open(boolean isCapture, int sampleRate, int audioFormat, int desiredChannels, int desiredFrames, int deviceId) {
|
|
||||||
int channelConfig;
|
|
||||||
int sampleSize;
|
|
||||||
int frameSize;
|
|
||||||
|
|
||||||
Log.v(TAG, "Opening " + (isCapture ? "capture" : "playback") + ", requested " + desiredFrames + " frames of " + desiredChannels + " channel " + getAudioFormatString(audioFormat) + " audio at " + sampleRate + " Hz");
|
|
||||||
|
|
||||||
/* On older devices let's use known good settings */
|
|
||||||
if (Build.VERSION.SDK_INT < 21 /* Android 5.0 (LOLLIPOP) */) {
|
|
||||||
if (desiredChannels > 2) {
|
|
||||||
desiredChannels = 2;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* AudioTrack has sample rate limitation of 48000 (fixed in 5.0.2) */
|
|
||||||
if (Build.VERSION.SDK_INT < 22 /* Android 5.1 (LOLLIPOP_MR1) */) {
|
|
||||||
if (sampleRate < 8000) {
|
|
||||||
sampleRate = 8000;
|
|
||||||
} else if (sampleRate > 48000) {
|
|
||||||
sampleRate = 48000;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (audioFormat == AudioFormat.ENCODING_PCM_FLOAT) {
|
|
||||||
int minSDKVersion = (isCapture ? 23 /* Android 6.0 (M) */ : 21 /* Android 5.0 (LOLLIPOP) */);
|
|
||||||
if (Build.VERSION.SDK_INT < minSDKVersion) {
|
|
||||||
audioFormat = AudioFormat.ENCODING_PCM_16BIT;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
switch (audioFormat)
|
|
||||||
{
|
|
||||||
case AudioFormat.ENCODING_PCM_8BIT:
|
|
||||||
sampleSize = 1;
|
|
||||||
break;
|
|
||||||
case AudioFormat.ENCODING_PCM_16BIT:
|
|
||||||
sampleSize = 2;
|
|
||||||
break;
|
|
||||||
case AudioFormat.ENCODING_PCM_FLOAT:
|
|
||||||
sampleSize = 4;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Log.v(TAG, "Requested format " + audioFormat + ", getting ENCODING_PCM_16BIT");
|
|
||||||
audioFormat = AudioFormat.ENCODING_PCM_16BIT;
|
|
||||||
sampleSize = 2;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (isCapture) {
|
|
||||||
switch (desiredChannels) {
|
|
||||||
case 1:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_IN_MONO;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_IN_STEREO;
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Log.v(TAG, "Requested " + desiredChannels + " channels, getting stereo");
|
|
||||||
desiredChannels = 2;
|
|
||||||
channelConfig = AudioFormat.CHANNEL_IN_STEREO;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
switch (desiredChannels) {
|
|
||||||
case 1:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_MONO;
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_STEREO;
|
|
||||||
break;
|
|
||||||
case 3:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_STEREO | AudioFormat.CHANNEL_OUT_FRONT_CENTER;
|
|
||||||
break;
|
|
||||||
case 4:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_QUAD;
|
|
||||||
break;
|
|
||||||
case 5:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_QUAD | AudioFormat.CHANNEL_OUT_FRONT_CENTER;
|
|
||||||
break;
|
|
||||||
case 6:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_5POINT1;
|
|
||||||
break;
|
|
||||||
case 7:
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_5POINT1 | AudioFormat.CHANNEL_OUT_BACK_CENTER;
|
|
||||||
break;
|
|
||||||
case 8:
|
|
||||||
if (Build.VERSION.SDK_INT >= 23 /* Android 6.0 (M) */) {
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_7POINT1_SURROUND;
|
|
||||||
} else {
|
|
||||||
Log.v(TAG, "Requested " + desiredChannels + " channels, getting 5.1 surround");
|
|
||||||
desiredChannels = 6;
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_5POINT1;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
Log.v(TAG, "Requested " + desiredChannels + " channels, getting stereo");
|
|
||||||
desiredChannels = 2;
|
|
||||||
channelConfig = AudioFormat.CHANNEL_OUT_STEREO;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*
|
|
||||||
Log.v(TAG, "Speaker configuration (and order of channels):");
|
|
||||||
|
|
||||||
if ((channelConfig & 0x00000004) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_FRONT_LEFT");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000008) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_FRONT_RIGHT");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000010) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_FRONT_CENTER");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000020) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_LOW_FREQUENCY");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000040) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_BACK_LEFT");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000080) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_BACK_RIGHT");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000100) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_FRONT_LEFT_OF_CENTER");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000200) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_FRONT_RIGHT_OF_CENTER");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000400) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_BACK_CENTER");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00000800) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_SIDE_LEFT");
|
|
||||||
}
|
|
||||||
if ((channelConfig & 0x00001000) != 0) {
|
|
||||||
Log.v(TAG, " CHANNEL_OUT_SIDE_RIGHT");
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
}
|
|
||||||
frameSize = (sampleSize * desiredChannels);
|
|
||||||
|
|
||||||
// Let the user pick a larger buffer if they really want -- but ye
|
|
||||||
// gods they probably shouldn't, the minimums are horrifyingly high
|
|
||||||
// latency already
|
|
||||||
int minBufferSize;
|
|
||||||
if (isCapture) {
|
|
||||||
minBufferSize = AudioRecord.getMinBufferSize(sampleRate, channelConfig, audioFormat);
|
|
||||||
} else {
|
|
||||||
minBufferSize = AudioTrack.getMinBufferSize(sampleRate, channelConfig, audioFormat);
|
|
||||||
}
|
|
||||||
desiredFrames = Math.max(desiredFrames, (minBufferSize + frameSize - 1) / frameSize);
|
|
||||||
|
|
||||||
int[] results = new int[4];
|
|
||||||
|
|
||||||
if (isCapture) {
|
|
||||||
if (mAudioRecord == null) {
|
|
||||||
mAudioRecord = new AudioRecord(MediaRecorder.AudioSource.DEFAULT, sampleRate,
|
|
||||||
channelConfig, audioFormat, desiredFrames * frameSize);
|
|
||||||
|
|
||||||
// see notes about AudioTrack state in audioOpen(), above. Probably also applies here.
|
|
||||||
if (mAudioRecord.getState() != AudioRecord.STATE_INITIALIZED) {
|
|
||||||
Log.e(TAG, "Failed during initialization of AudioRecord");
|
|
||||||
mAudioRecord.release();
|
|
||||||
mAudioRecord = null;
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */ && deviceId != 0) {
|
|
||||||
mAudioRecord.setPreferredDevice(getOutputAudioDeviceInfo(deviceId));
|
|
||||||
}
|
|
||||||
|
|
||||||
mAudioRecord.startRecording();
|
|
||||||
}
|
|
||||||
|
|
||||||
results[0] = mAudioRecord.getSampleRate();
|
|
||||||
results[1] = mAudioRecord.getAudioFormat();
|
|
||||||
results[2] = mAudioRecord.getChannelCount();
|
|
||||||
|
|
||||||
} else {
|
|
||||||
if (mAudioTrack == null) {
|
|
||||||
mAudioTrack = new AudioTrack(AudioManager.STREAM_MUSIC, sampleRate, channelConfig, audioFormat, desiredFrames * frameSize, AudioTrack.MODE_STREAM);
|
|
||||||
|
|
||||||
// Instantiating AudioTrack can "succeed" without an exception and the track may still be invalid
|
|
||||||
// Ref: https://android.googlesource.com/platform/frameworks/base/+/refs/heads/master/media/java/android/media/AudioTrack.java
|
|
||||||
// Ref: http://developer.android.com/reference/android/media/AudioTrack.html#getState()
|
|
||||||
if (mAudioTrack.getState() != AudioTrack.STATE_INITIALIZED) {
|
|
||||||
/* Try again, with safer values */
|
|
||||||
|
|
||||||
Log.e(TAG, "Failed during initialization of Audio Track");
|
|
||||||
mAudioTrack.release();
|
|
||||||
mAudioTrack = null;
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */ && deviceId != 0) {
|
|
||||||
mAudioTrack.setPreferredDevice(getInputAudioDeviceInfo(deviceId));
|
|
||||||
}
|
|
||||||
|
|
||||||
mAudioTrack.play();
|
|
||||||
}
|
|
||||||
|
|
||||||
results[0] = mAudioTrack.getSampleRate();
|
|
||||||
results[1] = mAudioTrack.getAudioFormat();
|
|
||||||
results[2] = mAudioTrack.getChannelCount();
|
|
||||||
}
|
|
||||||
results[3] = desiredFrames;
|
|
||||||
|
|
||||||
Log.v(TAG, "Opening " + (isCapture ? "capture" : "playback") + ", got " + results[3] + " frames of " + results[2] + " channel " + getAudioFormatString(results[1]) + " audio at " + results[0] + " Hz");
|
|
||||||
|
|
||||||
return results;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static AudioDeviceInfo getInputAudioDeviceInfo(int deviceId) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
return Arrays.stream(audioManager.getDevices(AudioManager.GET_DEVICES_INPUTS))
|
|
||||||
.filter(deviceInfo -> deviceInfo.getId() == deviceId)
|
|
||||||
.findFirst()
|
|
||||||
.orElse(null);
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static AudioDeviceInfo getOutputAudioDeviceInfo(int deviceId) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
return Arrays.stream(audioManager.getDevices(AudioManager.GET_DEVICES_OUTPUTS))
|
|
||||||
.filter(deviceInfo -> deviceInfo.getId() == deviceId)
|
|
||||||
.findFirst()
|
|
||||||
.orElse(null);
|
|
||||||
} else {
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void registerAudioDeviceCallback() {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
audioManager.registerAudioDeviceCallback(mAudioDeviceCallback, null);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private static void unregisterAudioDeviceCallback(Context context) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) context.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
audioManager.unregisterAudioDeviceCallback(mAudioDeviceCallback);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static int[] getAudioOutputDevices() {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
return Arrays.stream(audioManager.getDevices(AudioManager.GET_DEVICES_OUTPUTS)).mapToInt(AudioDeviceInfo::getId).toArray();
|
|
||||||
} else {
|
|
||||||
return NO_DEVICES;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static int[] getAudioInputDevices() {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
AudioManager audioManager = (AudioManager) mContext.getSystemService(Context.AUDIO_SERVICE);
|
|
||||||
return Arrays.stream(audioManager.getDevices(AudioManager.GET_DEVICES_INPUTS)).mapToInt(AudioDeviceInfo::getId).toArray();
|
|
||||||
} else {
|
|
||||||
return NO_DEVICES;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static int[] audioOpen(int sampleRate, int audioFormat, int desiredChannels, int desiredFrames, int deviceId) {
|
|
||||||
return open(false, sampleRate, audioFormat, desiredChannels, desiredFrames, deviceId);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void audioWriteFloatBuffer(float[] buffer) {
|
|
||||||
if (mAudioTrack == null) {
|
|
||||||
Log.e(TAG, "Attempted to make audio call with uninitialized audio!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (android.os.Build.VERSION.SDK_INT < 21 /* Android 5.0 (LOLLIPOP) */) {
|
|
||||||
Log.e(TAG, "Attempted to make an incompatible audio call with uninitialized audio! (floating-point output is supported since Android 5.0 Lollipop)");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < buffer.length;) {
|
|
||||||
int result = mAudioTrack.write(buffer, i, buffer.length - i, AudioTrack.WRITE_BLOCKING);
|
|
||||||
if (result > 0) {
|
|
||||||
i += result;
|
|
||||||
} else if (result == 0) {
|
|
||||||
try {
|
|
||||||
Thread.sleep(1);
|
|
||||||
} catch(InterruptedException e) {
|
|
||||||
// Nom nom
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.w(TAG, "SDL audio: error return from write(float)");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void audioWriteShortBuffer(short[] buffer) {
|
|
||||||
if (mAudioTrack == null) {
|
|
||||||
Log.e(TAG, "Attempted to make audio call with uninitialized audio!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < buffer.length;) {
|
|
||||||
int result = mAudioTrack.write(buffer, i, buffer.length - i);
|
|
||||||
if (result > 0) {
|
|
||||||
i += result;
|
|
||||||
} else if (result == 0) {
|
|
||||||
try {
|
|
||||||
Thread.sleep(1);
|
|
||||||
} catch(InterruptedException e) {
|
|
||||||
// Nom nom
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.w(TAG, "SDL audio: error return from write(short)");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void audioWriteByteBuffer(byte[] buffer) {
|
|
||||||
if (mAudioTrack == null) {
|
|
||||||
Log.e(TAG, "Attempted to make audio call with uninitialized audio!");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
for (int i = 0; i < buffer.length; ) {
|
|
||||||
int result = mAudioTrack.write(buffer, i, buffer.length - i);
|
|
||||||
if (result > 0) {
|
|
||||||
i += result;
|
|
||||||
} else if (result == 0) {
|
|
||||||
try {
|
|
||||||
Thread.sleep(1);
|
|
||||||
} catch(InterruptedException e) {
|
|
||||||
// Nom nom
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Log.w(TAG, "SDL audio: error return from write(byte)");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static int[] captureOpen(int sampleRate, int audioFormat, int desiredChannels, int desiredFrames, int deviceId) {
|
|
||||||
return open(true, sampleRate, audioFormat, desiredChannels, desiredFrames, deviceId);
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static int captureReadFloatBuffer(float[] buffer, boolean blocking) {
|
|
||||||
if (Build.VERSION.SDK_INT < 23 /* Android 6.0 (M) */) {
|
|
||||||
return 0;
|
|
||||||
} else {
|
|
||||||
return mAudioRecord.read(buffer, 0, buffer.length, blocking ? AudioRecord.READ_BLOCKING : AudioRecord.READ_NON_BLOCKING);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static int captureReadShortBuffer(short[] buffer, boolean blocking) {
|
|
||||||
if (Build.VERSION.SDK_INT < 23 /* Android 6.0 (M) */) {
|
|
||||||
return mAudioRecord.read(buffer, 0, buffer.length);
|
|
||||||
} else {
|
|
||||||
return mAudioRecord.read(buffer, 0, buffer.length, blocking ? AudioRecord.READ_BLOCKING : AudioRecord.READ_NON_BLOCKING);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static int captureReadByteBuffer(byte[] buffer, boolean blocking) {
|
|
||||||
if (Build.VERSION.SDK_INT < 23 /* Android 6.0 (M) */) {
|
|
||||||
return mAudioRecord.read(buffer, 0, buffer.length);
|
|
||||||
} else {
|
|
||||||
return mAudioRecord.read(buffer, 0, buffer.length, blocking ? AudioRecord.READ_BLOCKING : AudioRecord.READ_NON_BLOCKING);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static void audioClose() {
|
|
||||||
if (mAudioTrack != null) {
|
|
||||||
mAudioTrack.stop();
|
|
||||||
mAudioTrack.release();
|
|
||||||
mAudioTrack = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static void captureClose() {
|
|
||||||
if (mAudioRecord != null) {
|
|
||||||
mAudioRecord.stop();
|
|
||||||
mAudioRecord.release();
|
|
||||||
mAudioRecord = null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/** This method is called by SDL using JNI. */
|
|
||||||
public static void audioSetThreadPriority(boolean iscapture, int device_id) {
|
|
||||||
try {
|
|
||||||
|
|
||||||
/* Set thread name */
|
|
||||||
if (iscapture) {
|
|
||||||
Thread.currentThread().setName("SDLAudioC" + device_id);
|
|
||||||
} else {
|
|
||||||
Thread.currentThread().setName("SDLAudioP" + device_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Set thread priority */
|
|
||||||
android.os.Process.setThreadPriority(android.os.Process.THREAD_PRIORITY_AUDIO);
|
|
||||||
|
|
||||||
} catch (Exception e) {
|
|
||||||
Log.v(TAG, "modify thread properties failed " + e.toString());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static native int nativeSetupJNI();
|
|
||||||
|
|
||||||
public static native void removeAudioDevice(boolean isCapture, int deviceId);
|
|
||||||
|
|
||||||
public static native void addAudioDevice(boolean isCapture, int deviceId);
|
|
||||||
|
|
||||||
}
|
|
@ -1,856 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
import java.util.ArrayList;
|
|
||||||
import java.util.Collections;
|
|
||||||
import java.util.Comparator;
|
|
||||||
import java.util.List;
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.os.VibrationEffect;
|
|
||||||
import android.os.Vibrator;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.view.InputDevice;
|
|
||||||
import android.view.KeyEvent;
|
|
||||||
import android.view.MotionEvent;
|
|
||||||
import android.view.View;
|
|
||||||
|
|
||||||
|
|
||||||
public class SDLControllerManager
|
|
||||||
{
|
|
||||||
|
|
||||||
public static native int nativeSetupJNI();
|
|
||||||
|
|
||||||
public static native int nativeAddJoystick(int device_id, String name, String desc,
|
|
||||||
int vendor_id, int product_id,
|
|
||||||
boolean is_accelerometer, int button_mask,
|
|
||||||
int naxes, int axis_mask, int nhats, int nballs);
|
|
||||||
public static native int nativeRemoveJoystick(int device_id);
|
|
||||||
public static native int nativeAddHaptic(int device_id, String name);
|
|
||||||
public static native int nativeRemoveHaptic(int device_id);
|
|
||||||
public static native int onNativePadDown(int device_id, int keycode);
|
|
||||||
public static native int onNativePadUp(int device_id, int keycode);
|
|
||||||
public static native void onNativeJoy(int device_id, int axis,
|
|
||||||
float value);
|
|
||||||
public static native void onNativeHat(int device_id, int hat_id,
|
|
||||||
int x, int y);
|
|
||||||
|
|
||||||
protected static SDLJoystickHandler mJoystickHandler;
|
|
||||||
protected static SDLHapticHandler mHapticHandler;
|
|
||||||
|
|
||||||
private static final String TAG = "SDLControllerManager";
|
|
||||||
|
|
||||||
public static void initialize() {
|
|
||||||
if (mJoystickHandler == null) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 19 /* Android 4.4 (KITKAT) */) {
|
|
||||||
mJoystickHandler = new SDLJoystickHandler_API19();
|
|
||||||
} else {
|
|
||||||
mJoystickHandler = new SDLJoystickHandler_API16();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (mHapticHandler == null) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 26 /* Android 8.0 (O) */) {
|
|
||||||
mHapticHandler = new SDLHapticHandler_API26();
|
|
||||||
} else {
|
|
||||||
mHapticHandler = new SDLHapticHandler();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Joystick glue code, just a series of stubs that redirect to the SDLJoystickHandler instance
|
|
||||||
public static boolean handleJoystickMotionEvent(MotionEvent event) {
|
|
||||||
return mJoystickHandler.handleMotionEvent(event);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void pollInputDevices() {
|
|
||||||
mJoystickHandler.pollInputDevices();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void pollHapticDevices() {
|
|
||||||
mHapticHandler.pollHapticDevices();
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void hapticRun(int device_id, float intensity, int length) {
|
|
||||||
mHapticHandler.run(device_id, intensity, length);
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* This method is called by SDL using JNI.
|
|
||||||
*/
|
|
||||||
public static void hapticStop(int device_id)
|
|
||||||
{
|
|
||||||
mHapticHandler.stop(device_id);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Check if a given device is considered a possible SDL joystick
|
|
||||||
public static boolean isDeviceSDLJoystick(int deviceId) {
|
|
||||||
InputDevice device = InputDevice.getDevice(deviceId);
|
|
||||||
// We cannot use InputDevice.isVirtual before API 16, so let's accept
|
|
||||||
// only nonnegative device ids (VIRTUAL_KEYBOARD equals -1)
|
|
||||||
if ((device == null) || (deviceId < 0)) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
int sources = device.getSources();
|
|
||||||
|
|
||||||
/* This is called for every button press, so let's not spam the logs */
|
|
||||||
/*
|
|
||||||
if ((sources & InputDevice.SOURCE_CLASS_JOYSTICK) != 0) {
|
|
||||||
Log.v(TAG, "Input device " + device.getName() + " has class joystick.");
|
|
||||||
}
|
|
||||||
if ((sources & InputDevice.SOURCE_DPAD) == InputDevice.SOURCE_DPAD) {
|
|
||||||
Log.v(TAG, "Input device " + device.getName() + " is a dpad.");
|
|
||||||
}
|
|
||||||
if ((sources & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD) {
|
|
||||||
Log.v(TAG, "Input device " + device.getName() + " is a gamepad.");
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
return ((sources & InputDevice.SOURCE_CLASS_JOYSTICK) != 0 ||
|
|
||||||
((sources & InputDevice.SOURCE_DPAD) == InputDevice.SOURCE_DPAD) ||
|
|
||||||
((sources & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD)
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLJoystickHandler {
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles given MotionEvent.
|
|
||||||
* @param event the event to be handled.
|
|
||||||
* @return if given event was processed.
|
|
||||||
*/
|
|
||||||
public boolean handleMotionEvent(MotionEvent event) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Handles adding and removing of input devices.
|
|
||||||
*/
|
|
||||||
public void pollInputDevices() {
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Actual joystick functionality available for API >= 12 devices */
|
|
||||||
class SDLJoystickHandler_API16 extends SDLJoystickHandler {
|
|
||||||
|
|
||||||
static class SDLJoystick {
|
|
||||||
public int device_id;
|
|
||||||
public String name;
|
|
||||||
public String desc;
|
|
||||||
public ArrayList<InputDevice.MotionRange> axes;
|
|
||||||
public ArrayList<InputDevice.MotionRange> hats;
|
|
||||||
}
|
|
||||||
static class RangeComparator implements Comparator<InputDevice.MotionRange> {
|
|
||||||
@Override
|
|
||||||
public int compare(InputDevice.MotionRange arg0, InputDevice.MotionRange arg1) {
|
|
||||||
// Some controllers, like the Moga Pro 2, return AXIS_GAS (22) for right trigger and AXIS_BRAKE (23) for left trigger - swap them so they're sorted in the right order for SDL
|
|
||||||
int arg0Axis = arg0.getAxis();
|
|
||||||
int arg1Axis = arg1.getAxis();
|
|
||||||
if (arg0Axis == MotionEvent.AXIS_GAS) {
|
|
||||||
arg0Axis = MotionEvent.AXIS_BRAKE;
|
|
||||||
} else if (arg0Axis == MotionEvent.AXIS_BRAKE) {
|
|
||||||
arg0Axis = MotionEvent.AXIS_GAS;
|
|
||||||
}
|
|
||||||
if (arg1Axis == MotionEvent.AXIS_GAS) {
|
|
||||||
arg1Axis = MotionEvent.AXIS_BRAKE;
|
|
||||||
} else if (arg1Axis == MotionEvent.AXIS_BRAKE) {
|
|
||||||
arg1Axis = MotionEvent.AXIS_GAS;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Make sure the AXIS_Z is sorted between AXIS_RY and AXIS_RZ.
|
|
||||||
// This is because the usual pairing are:
|
|
||||||
// - AXIS_X + AXIS_Y (left stick).
|
|
||||||
// - AXIS_RX, AXIS_RY (sometimes the right stick, sometimes triggers).
|
|
||||||
// - AXIS_Z, AXIS_RZ (sometimes the right stick, sometimes triggers).
|
|
||||||
// This sorts the axes in the above order, which tends to be correct
|
|
||||||
// for Xbox-ish game pads that have the right stick on RX/RY and the
|
|
||||||
// triggers on Z/RZ.
|
|
||||||
//
|
|
||||||
// Gamepads that don't have AXIS_Z/AXIS_RZ but use
|
|
||||||
// AXIS_LTRIGGER/AXIS_RTRIGGER are unaffected by this.
|
|
||||||
//
|
|
||||||
// References:
|
|
||||||
// - https://developer.android.com/develop/ui/views/touch-and-input/game-controllers/controller-input
|
|
||||||
// - https://www.kernel.org/doc/html/latest/input/gamepad.html
|
|
||||||
if (arg0Axis == MotionEvent.AXIS_Z) {
|
|
||||||
arg0Axis = MotionEvent.AXIS_RZ - 1;
|
|
||||||
} else if (arg0Axis > MotionEvent.AXIS_Z && arg0Axis < MotionEvent.AXIS_RZ) {
|
|
||||||
--arg0Axis;
|
|
||||||
}
|
|
||||||
if (arg1Axis == MotionEvent.AXIS_Z) {
|
|
||||||
arg1Axis = MotionEvent.AXIS_RZ - 1;
|
|
||||||
} else if (arg1Axis > MotionEvent.AXIS_Z && arg1Axis < MotionEvent.AXIS_RZ) {
|
|
||||||
--arg1Axis;
|
|
||||||
}
|
|
||||||
|
|
||||||
return arg0Axis - arg1Axis;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private final ArrayList<SDLJoystick> mJoysticks;
|
|
||||||
|
|
||||||
public SDLJoystickHandler_API16() {
|
|
||||||
|
|
||||||
mJoysticks = new ArrayList<SDLJoystick>();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void pollInputDevices() {
|
|
||||||
int[] deviceIds = InputDevice.getDeviceIds();
|
|
||||||
|
|
||||||
for (int device_id : deviceIds) {
|
|
||||||
if (SDLControllerManager.isDeviceSDLJoystick(device_id)) {
|
|
||||||
SDLJoystick joystick = getJoystick(device_id);
|
|
||||||
if (joystick == null) {
|
|
||||||
InputDevice joystickDevice = InputDevice.getDevice(device_id);
|
|
||||||
joystick = new SDLJoystick();
|
|
||||||
joystick.device_id = device_id;
|
|
||||||
joystick.name = joystickDevice.getName();
|
|
||||||
joystick.desc = getJoystickDescriptor(joystickDevice);
|
|
||||||
joystick.axes = new ArrayList<InputDevice.MotionRange>();
|
|
||||||
joystick.hats = new ArrayList<InputDevice.MotionRange>();
|
|
||||||
|
|
||||||
List<InputDevice.MotionRange> ranges = joystickDevice.getMotionRanges();
|
|
||||||
Collections.sort(ranges, new RangeComparator());
|
|
||||||
for (InputDevice.MotionRange range : ranges) {
|
|
||||||
if ((range.getSource() & InputDevice.SOURCE_CLASS_JOYSTICK) != 0) {
|
|
||||||
if (range.getAxis() == MotionEvent.AXIS_HAT_X || range.getAxis() == MotionEvent.AXIS_HAT_Y) {
|
|
||||||
joystick.hats.add(range);
|
|
||||||
} else {
|
|
||||||
joystick.axes.add(range);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
mJoysticks.add(joystick);
|
|
||||||
SDLControllerManager.nativeAddJoystick(joystick.device_id, joystick.name, joystick.desc,
|
|
||||||
getVendorId(joystickDevice), getProductId(joystickDevice), false,
|
|
||||||
getButtonMask(joystickDevice), joystick.axes.size(), getAxisMask(joystick.axes), joystick.hats.size()/2, 0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check removed devices */
|
|
||||||
ArrayList<Integer> removedDevices = null;
|
|
||||||
for (SDLJoystick joystick : mJoysticks) {
|
|
||||||
int device_id = joystick.device_id;
|
|
||||||
int i;
|
|
||||||
for (i = 0; i < deviceIds.length; i++) {
|
|
||||||
if (device_id == deviceIds[i]) break;
|
|
||||||
}
|
|
||||||
if (i == deviceIds.length) {
|
|
||||||
if (removedDevices == null) {
|
|
||||||
removedDevices = new ArrayList<Integer>();
|
|
||||||
}
|
|
||||||
removedDevices.add(device_id);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (removedDevices != null) {
|
|
||||||
for (int device_id : removedDevices) {
|
|
||||||
SDLControllerManager.nativeRemoveJoystick(device_id);
|
|
||||||
for (int i = 0; i < mJoysticks.size(); i++) {
|
|
||||||
if (mJoysticks.get(i).device_id == device_id) {
|
|
||||||
mJoysticks.remove(i);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected SDLJoystick getJoystick(int device_id) {
|
|
||||||
for (SDLJoystick joystick : mJoysticks) {
|
|
||||||
if (joystick.device_id == device_id) {
|
|
||||||
return joystick;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean handleMotionEvent(MotionEvent event) {
|
|
||||||
int actionPointerIndex = event.getActionIndex();
|
|
||||||
int action = event.getActionMasked();
|
|
||||||
if (action == MotionEvent.ACTION_MOVE) {
|
|
||||||
SDLJoystick joystick = getJoystick(event.getDeviceId());
|
|
||||||
if (joystick != null) {
|
|
||||||
for (int i = 0; i < joystick.axes.size(); i++) {
|
|
||||||
InputDevice.MotionRange range = joystick.axes.get(i);
|
|
||||||
/* Normalize the value to -1...1 */
|
|
||||||
float value = (event.getAxisValue(range.getAxis(), actionPointerIndex) - range.getMin()) / range.getRange() * 2.0f - 1.0f;
|
|
||||||
SDLControllerManager.onNativeJoy(joystick.device_id, i, value);
|
|
||||||
}
|
|
||||||
for (int i = 0; i < joystick.hats.size() / 2; i++) {
|
|
||||||
int hatX = Math.round(event.getAxisValue(joystick.hats.get(2 * i).getAxis(), actionPointerIndex));
|
|
||||||
int hatY = Math.round(event.getAxisValue(joystick.hats.get(2 * i + 1).getAxis(), actionPointerIndex));
|
|
||||||
SDLControllerManager.onNativeHat(joystick.device_id, i, hatX, hatY);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public String getJoystickDescriptor(InputDevice joystickDevice) {
|
|
||||||
String desc = joystickDevice.getDescriptor();
|
|
||||||
|
|
||||||
if (desc != null && !desc.isEmpty()) {
|
|
||||||
return desc;
|
|
||||||
}
|
|
||||||
|
|
||||||
return joystickDevice.getName();
|
|
||||||
}
|
|
||||||
public int getProductId(InputDevice joystickDevice) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
public int getVendorId(InputDevice joystickDevice) {
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
public int getAxisMask(List<InputDevice.MotionRange> ranges) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
public int getButtonMask(InputDevice joystickDevice) {
|
|
||||||
return -1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLJoystickHandler_API19 extends SDLJoystickHandler_API16 {
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getProductId(InputDevice joystickDevice) {
|
|
||||||
return joystickDevice.getProductId();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getVendorId(InputDevice joystickDevice) {
|
|
||||||
return joystickDevice.getVendorId();
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getAxisMask(List<InputDevice.MotionRange> ranges) {
|
|
||||||
// For compatibility, keep computing the axis mask like before,
|
|
||||||
// only really distinguishing 2, 4 and 6 axes.
|
|
||||||
int axis_mask = 0;
|
|
||||||
if (ranges.size() >= 2) {
|
|
||||||
// ((1 << SDL_GAMEPAD_AXIS_LEFTX) | (1 << SDL_GAMEPAD_AXIS_LEFTY))
|
|
||||||
axis_mask |= 0x0003;
|
|
||||||
}
|
|
||||||
if (ranges.size() >= 4) {
|
|
||||||
// ((1 << SDL_GAMEPAD_AXIS_RIGHTX) | (1 << SDL_GAMEPAD_AXIS_RIGHTY))
|
|
||||||
axis_mask |= 0x000c;
|
|
||||||
}
|
|
||||||
if (ranges.size() >= 6) {
|
|
||||||
// ((1 << SDL_GAMEPAD_AXIS_LEFT_TRIGGER) | (1 << SDL_GAMEPAD_AXIS_RIGHT_TRIGGER))
|
|
||||||
axis_mask |= 0x0030;
|
|
||||||
}
|
|
||||||
// Also add an indicator bit for whether the sorting order has changed.
|
|
||||||
// This serves to disable outdated gamecontrollerdb.txt mappings.
|
|
||||||
boolean have_z = false;
|
|
||||||
boolean have_past_z_before_rz = false;
|
|
||||||
for (InputDevice.MotionRange range : ranges) {
|
|
||||||
int axis = range.getAxis();
|
|
||||||
if (axis == MotionEvent.AXIS_Z) {
|
|
||||||
have_z = true;
|
|
||||||
} else if (axis > MotionEvent.AXIS_Z && axis < MotionEvent.AXIS_RZ) {
|
|
||||||
have_past_z_before_rz = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (have_z && have_past_z_before_rz) {
|
|
||||||
// If both these exist, the compare() function changed sorting order.
|
|
||||||
// Set a bit to indicate this fact.
|
|
||||||
axis_mask |= 0x8000;
|
|
||||||
}
|
|
||||||
return axis_mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public int getButtonMask(InputDevice joystickDevice) {
|
|
||||||
int button_mask = 0;
|
|
||||||
int[] keys = new int[] {
|
|
||||||
KeyEvent.KEYCODE_BUTTON_A,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_B,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_X,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_Y,
|
|
||||||
KeyEvent.KEYCODE_BACK,
|
|
||||||
KeyEvent.KEYCODE_MENU,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_MODE,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_START,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_THUMBL,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_THUMBR,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_L1,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_R1,
|
|
||||||
KeyEvent.KEYCODE_DPAD_UP,
|
|
||||||
KeyEvent.KEYCODE_DPAD_DOWN,
|
|
||||||
KeyEvent.KEYCODE_DPAD_LEFT,
|
|
||||||
KeyEvent.KEYCODE_DPAD_RIGHT,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_SELECT,
|
|
||||||
KeyEvent.KEYCODE_DPAD_CENTER,
|
|
||||||
|
|
||||||
// These don't map into any SDL controller buttons directly
|
|
||||||
KeyEvent.KEYCODE_BUTTON_L2,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_R2,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_C,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_Z,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_1,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_2,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_3,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_4,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_5,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_6,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_7,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_8,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_9,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_10,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_11,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_12,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_13,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_14,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_15,
|
|
||||||
KeyEvent.KEYCODE_BUTTON_16,
|
|
||||||
};
|
|
||||||
int[] masks = new int[] {
|
|
||||||
(1 << 0), // A -> A
|
|
||||||
(1 << 1), // B -> B
|
|
||||||
(1 << 2), // X -> X
|
|
||||||
(1 << 3), // Y -> Y
|
|
||||||
(1 << 4), // BACK -> BACK
|
|
||||||
(1 << 6), // MENU -> START
|
|
||||||
(1 << 5), // MODE -> GUIDE
|
|
||||||
(1 << 6), // START -> START
|
|
||||||
(1 << 7), // THUMBL -> LEFTSTICK
|
|
||||||
(1 << 8), // THUMBR -> RIGHTSTICK
|
|
||||||
(1 << 9), // L1 -> LEFTSHOULDER
|
|
||||||
(1 << 10), // R1 -> RIGHTSHOULDER
|
|
||||||
(1 << 11), // DPAD_UP -> DPAD_UP
|
|
||||||
(1 << 12), // DPAD_DOWN -> DPAD_DOWN
|
|
||||||
(1 << 13), // DPAD_LEFT -> DPAD_LEFT
|
|
||||||
(1 << 14), // DPAD_RIGHT -> DPAD_RIGHT
|
|
||||||
(1 << 4), // SELECT -> BACK
|
|
||||||
(1 << 0), // DPAD_CENTER -> A
|
|
||||||
(1 << 15), // L2 -> ??
|
|
||||||
(1 << 16), // R2 -> ??
|
|
||||||
(1 << 17), // C -> ??
|
|
||||||
(1 << 18), // Z -> ??
|
|
||||||
(1 << 20), // 1 -> ??
|
|
||||||
(1 << 21), // 2 -> ??
|
|
||||||
(1 << 22), // 3 -> ??
|
|
||||||
(1 << 23), // 4 -> ??
|
|
||||||
(1 << 24), // 5 -> ??
|
|
||||||
(1 << 25), // 6 -> ??
|
|
||||||
(1 << 26), // 7 -> ??
|
|
||||||
(1 << 27), // 8 -> ??
|
|
||||||
(1 << 28), // 9 -> ??
|
|
||||||
(1 << 29), // 10 -> ??
|
|
||||||
(1 << 30), // 11 -> ??
|
|
||||||
(1 << 31), // 12 -> ??
|
|
||||||
// We're out of room...
|
|
||||||
0xFFFFFFFF, // 13 -> ??
|
|
||||||
0xFFFFFFFF, // 14 -> ??
|
|
||||||
0xFFFFFFFF, // 15 -> ??
|
|
||||||
0xFFFFFFFF, // 16 -> ??
|
|
||||||
};
|
|
||||||
boolean[] has_keys = joystickDevice.hasKeys(keys);
|
|
||||||
for (int i = 0; i < keys.length; ++i) {
|
|
||||||
if (has_keys[i]) {
|
|
||||||
button_mask |= masks[i];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return button_mask;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLHapticHandler_API26 extends SDLHapticHandler {
|
|
||||||
@Override
|
|
||||||
public void run(int device_id, float intensity, int length) {
|
|
||||||
SDLHaptic haptic = getHaptic(device_id);
|
|
||||||
if (haptic != null) {
|
|
||||||
Log.d("SDL", "Rtest: Vibe with intensity " + intensity + " for " + length);
|
|
||||||
if (intensity == 0.0f) {
|
|
||||||
stop(device_id);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
int vibeValue = Math.round(intensity * 255);
|
|
||||||
|
|
||||||
if (vibeValue > 255) {
|
|
||||||
vibeValue = 255;
|
|
||||||
}
|
|
||||||
if (vibeValue < 1) {
|
|
||||||
stop(device_id);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
haptic.vib.vibrate(VibrationEffect.createOneShot(length, vibeValue));
|
|
||||||
}
|
|
||||||
catch (Exception e) {
|
|
||||||
// Fall back to the generic method, which uses DEFAULT_AMPLITUDE, but works even if
|
|
||||||
// something went horribly wrong with the Android 8.0 APIs.
|
|
||||||
haptic.vib.vibrate(length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLHapticHandler {
|
|
||||||
|
|
||||||
static class SDLHaptic {
|
|
||||||
public int device_id;
|
|
||||||
public String name;
|
|
||||||
public Vibrator vib;
|
|
||||||
}
|
|
||||||
|
|
||||||
private final ArrayList<SDLHaptic> mHaptics;
|
|
||||||
|
|
||||||
public SDLHapticHandler() {
|
|
||||||
mHaptics = new ArrayList<SDLHaptic>();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void run(int device_id, float intensity, int length) {
|
|
||||||
SDLHaptic haptic = getHaptic(device_id);
|
|
||||||
if (haptic != null) {
|
|
||||||
haptic.vib.vibrate(length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void stop(int device_id) {
|
|
||||||
SDLHaptic haptic = getHaptic(device_id);
|
|
||||||
if (haptic != null) {
|
|
||||||
haptic.vib.cancel();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void pollHapticDevices() {
|
|
||||||
|
|
||||||
final int deviceId_VIBRATOR_SERVICE = 999999;
|
|
||||||
boolean hasVibratorService = false;
|
|
||||||
|
|
||||||
int[] deviceIds = InputDevice.getDeviceIds();
|
|
||||||
// It helps processing the device ids in reverse order
|
|
||||||
// For example, in the case of the XBox 360 wireless dongle,
|
|
||||||
// so the first controller seen by SDL matches what the receiver
|
|
||||||
// considers to be the first controller
|
|
||||||
|
|
||||||
for (int i = deviceIds.length - 1; i > -1; i--) {
|
|
||||||
SDLHaptic haptic = getHaptic(deviceIds[i]);
|
|
||||||
if (haptic == null) {
|
|
||||||
InputDevice device = InputDevice.getDevice(deviceIds[i]);
|
|
||||||
Vibrator vib = device.getVibrator();
|
|
||||||
if (vib != null) {
|
|
||||||
if (vib.hasVibrator()) {
|
|
||||||
haptic = new SDLHaptic();
|
|
||||||
haptic.device_id = deviceIds[i];
|
|
||||||
haptic.name = device.getName();
|
|
||||||
haptic.vib = vib;
|
|
||||||
mHaptics.add(haptic);
|
|
||||||
SDLControllerManager.nativeAddHaptic(haptic.device_id, haptic.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check VIBRATOR_SERVICE */
|
|
||||||
Vibrator vib = (Vibrator) SDL.getContext().getSystemService(Context.VIBRATOR_SERVICE);
|
|
||||||
if (vib != null) {
|
|
||||||
hasVibratorService = vib.hasVibrator();
|
|
||||||
|
|
||||||
if (hasVibratorService) {
|
|
||||||
SDLHaptic haptic = getHaptic(deviceId_VIBRATOR_SERVICE);
|
|
||||||
if (haptic == null) {
|
|
||||||
haptic = new SDLHaptic();
|
|
||||||
haptic.device_id = deviceId_VIBRATOR_SERVICE;
|
|
||||||
haptic.name = "VIBRATOR_SERVICE";
|
|
||||||
haptic.vib = vib;
|
|
||||||
mHaptics.add(haptic);
|
|
||||||
SDLControllerManager.nativeAddHaptic(haptic.device_id, haptic.name);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Check removed devices */
|
|
||||||
ArrayList<Integer> removedDevices = null;
|
|
||||||
for (SDLHaptic haptic : mHaptics) {
|
|
||||||
int device_id = haptic.device_id;
|
|
||||||
int i;
|
|
||||||
for (i = 0; i < deviceIds.length; i++) {
|
|
||||||
if (device_id == deviceIds[i]) break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (device_id != deviceId_VIBRATOR_SERVICE || !hasVibratorService) {
|
|
||||||
if (i == deviceIds.length) {
|
|
||||||
if (removedDevices == null) {
|
|
||||||
removedDevices = new ArrayList<Integer>();
|
|
||||||
}
|
|
||||||
removedDevices.add(device_id);
|
|
||||||
}
|
|
||||||
} // else: don't remove the vibrator if it is still present
|
|
||||||
}
|
|
||||||
|
|
||||||
if (removedDevices != null) {
|
|
||||||
for (int device_id : removedDevices) {
|
|
||||||
SDLControllerManager.nativeRemoveHaptic(device_id);
|
|
||||||
for (int i = 0; i < mHaptics.size(); i++) {
|
|
||||||
if (mHaptics.get(i).device_id == device_id) {
|
|
||||||
mHaptics.remove(i);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
protected SDLHaptic getHaptic(int device_id) {
|
|
||||||
for (SDLHaptic haptic : mHaptics) {
|
|
||||||
if (haptic.device_id == device_id) {
|
|
||||||
return haptic;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLGenericMotionListener_API12 implements View.OnGenericMotionListener {
|
|
||||||
// Generic Motion (mouse hover, joystick...) events go here
|
|
||||||
@Override
|
|
||||||
public boolean onGenericMotion(View v, MotionEvent event) {
|
|
||||||
float x, y;
|
|
||||||
int action;
|
|
||||||
|
|
||||||
switch ( event.getSource() ) {
|
|
||||||
case InputDevice.SOURCE_JOYSTICK:
|
|
||||||
return SDLControllerManager.handleJoystickMotionEvent(event);
|
|
||||||
|
|
||||||
case InputDevice.SOURCE_MOUSE:
|
|
||||||
action = event.getActionMasked();
|
|
||||||
switch (action) {
|
|
||||||
case MotionEvent.ACTION_SCROLL:
|
|
||||||
x = event.getAxisValue(MotionEvent.AXIS_HSCROLL, 0);
|
|
||||||
y = event.getAxisValue(MotionEvent.AXIS_VSCROLL, 0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_HOVER_MOVE:
|
|
||||||
x = event.getX(0);
|
|
||||||
y = event.getY(0);
|
|
||||||
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Event was not managed
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean supportsRelativeMouse() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean inRelativeMode() {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public boolean setRelativeMouseEnabled(boolean enabled) {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void reclaimRelativeMouseModeIfNeeded()
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public float getEventX(MotionEvent event) {
|
|
||||||
return event.getX(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
public float getEventY(MotionEvent event) {
|
|
||||||
return event.getY(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLGenericMotionListener_API24 extends SDLGenericMotionListener_API12 {
|
|
||||||
// Generic Motion (mouse hover, joystick...) events go here
|
|
||||||
|
|
||||||
private boolean mRelativeModeEnabled;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onGenericMotion(View v, MotionEvent event) {
|
|
||||||
|
|
||||||
// Handle relative mouse mode
|
|
||||||
if (mRelativeModeEnabled) {
|
|
||||||
if (event.getSource() == InputDevice.SOURCE_MOUSE) {
|
|
||||||
int action = event.getActionMasked();
|
|
||||||
if (action == MotionEvent.ACTION_HOVER_MOVE) {
|
|
||||||
float x = event.getAxisValue(MotionEvent.AXIS_RELATIVE_X);
|
|
||||||
float y = event.getAxisValue(MotionEvent.AXIS_RELATIVE_Y);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, true);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Event was not managed, call SDLGenericMotionListener_API12 method
|
|
||||||
return super.onGenericMotion(v, event);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean supportsRelativeMouse() {
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean inRelativeMode() {
|
|
||||||
return mRelativeModeEnabled;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean setRelativeMouseEnabled(boolean enabled) {
|
|
||||||
mRelativeModeEnabled = enabled;
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public float getEventX(MotionEvent event) {
|
|
||||||
if (mRelativeModeEnabled) {
|
|
||||||
return event.getAxisValue(MotionEvent.AXIS_RELATIVE_X);
|
|
||||||
} else {
|
|
||||||
return event.getX(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public float getEventY(MotionEvent event) {
|
|
||||||
if (mRelativeModeEnabled) {
|
|
||||||
return event.getAxisValue(MotionEvent.AXIS_RELATIVE_Y);
|
|
||||||
} else {
|
|
||||||
return event.getY(0);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
class SDLGenericMotionListener_API26 extends SDLGenericMotionListener_API24 {
|
|
||||||
// Generic Motion (mouse hover, joystick...) events go here
|
|
||||||
private boolean mRelativeModeEnabled;
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean onGenericMotion(View v, MotionEvent event) {
|
|
||||||
float x, y;
|
|
||||||
int action;
|
|
||||||
|
|
||||||
switch ( event.getSource() ) {
|
|
||||||
case InputDevice.SOURCE_JOYSTICK:
|
|
||||||
return SDLControllerManager.handleJoystickMotionEvent(event);
|
|
||||||
|
|
||||||
case InputDevice.SOURCE_MOUSE:
|
|
||||||
// DeX desktop mouse cursor is a separate non-standard input type.
|
|
||||||
case InputDevice.SOURCE_MOUSE | InputDevice.SOURCE_TOUCHSCREEN:
|
|
||||||
action = event.getActionMasked();
|
|
||||||
switch (action) {
|
|
||||||
case MotionEvent.ACTION_SCROLL:
|
|
||||||
x = event.getAxisValue(MotionEvent.AXIS_HSCROLL, 0);
|
|
||||||
y = event.getAxisValue(MotionEvent.AXIS_VSCROLL, 0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_HOVER_MOVE:
|
|
||||||
x = event.getX(0);
|
|
||||||
y = event.getY(0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case InputDevice.SOURCE_MOUSE_RELATIVE:
|
|
||||||
action = event.getActionMasked();
|
|
||||||
switch (action) {
|
|
||||||
case MotionEvent.ACTION_SCROLL:
|
|
||||||
x = event.getAxisValue(MotionEvent.AXIS_HSCROLL, 0);
|
|
||||||
y = event.getAxisValue(MotionEvent.AXIS_VSCROLL, 0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_HOVER_MOVE:
|
|
||||||
x = event.getX(0);
|
|
||||||
y = event.getY(0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, true);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Event was not managed
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean supportsRelativeMouse() {
|
|
||||||
return (!SDLActivity.isDeXMode() || Build.VERSION.SDK_INT >= 27 /* Android 8.1 (O_MR1) */);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean inRelativeMode() {
|
|
||||||
return mRelativeModeEnabled;
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public boolean setRelativeMouseEnabled(boolean enabled) {
|
|
||||||
if (!SDLActivity.isDeXMode() || Build.VERSION.SDK_INT >= 27 /* Android 8.1 (O_MR1) */) {
|
|
||||||
if (enabled) {
|
|
||||||
SDLActivity.getContentView().requestPointerCapture();
|
|
||||||
} else {
|
|
||||||
SDLActivity.getContentView().releasePointerCapture();
|
|
||||||
}
|
|
||||||
mRelativeModeEnabled = enabled;
|
|
||||||
return true;
|
|
||||||
} else {
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void reclaimRelativeMouseModeIfNeeded()
|
|
||||||
{
|
|
||||||
if (mRelativeModeEnabled && !SDLActivity.isDeXMode()) {
|
|
||||||
SDLActivity.getContentView().requestPointerCapture();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public float getEventX(MotionEvent event) {
|
|
||||||
// Relative mouse in capture mode will only have relative for X/Y
|
|
||||||
return event.getX(0);
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public float getEventY(MotionEvent event) {
|
|
||||||
// Relative mouse in capture mode will only have relative for X/Y
|
|
||||||
return event.getY(0);
|
|
||||||
}
|
|
||||||
}
|
|
@ -1,405 +0,0 @@
|
|||||||
package org.libsdl.app;
|
|
||||||
|
|
||||||
|
|
||||||
import android.content.Context;
|
|
||||||
import android.content.pm.ActivityInfo;
|
|
||||||
import android.hardware.Sensor;
|
|
||||||
import android.hardware.SensorEvent;
|
|
||||||
import android.hardware.SensorEventListener;
|
|
||||||
import android.hardware.SensorManager;
|
|
||||||
import android.os.Build;
|
|
||||||
import android.util.DisplayMetrics;
|
|
||||||
import android.util.Log;
|
|
||||||
import android.view.Display;
|
|
||||||
import android.view.InputDevice;
|
|
||||||
import android.view.KeyEvent;
|
|
||||||
import android.view.MotionEvent;
|
|
||||||
import android.view.Surface;
|
|
||||||
import android.view.SurfaceHolder;
|
|
||||||
import android.view.SurfaceView;
|
|
||||||
import android.view.View;
|
|
||||||
import android.view.WindowManager;
|
|
||||||
|
|
||||||
|
|
||||||
/**
|
|
||||||
SDLSurface. This is what we draw on, so we need to know when it's created
|
|
||||||
in order to do anything useful.
|
|
||||||
|
|
||||||
Because of this, that's where we set up the SDL thread
|
|
||||||
*/
|
|
||||||
public class SDLSurface extends SurfaceView implements SurfaceHolder.Callback,
|
|
||||||
View.OnKeyListener, View.OnTouchListener, SensorEventListener {
|
|
||||||
|
|
||||||
// Sensors
|
|
||||||
protected SensorManager mSensorManager;
|
|
||||||
protected Display mDisplay;
|
|
||||||
|
|
||||||
// Keep track of the surface size to normalize touch events
|
|
||||||
protected float mWidth, mHeight;
|
|
||||||
|
|
||||||
// Is SurfaceView ready for rendering
|
|
||||||
public boolean mIsSurfaceReady;
|
|
||||||
|
|
||||||
// Startup
|
|
||||||
public SDLSurface(Context context) {
|
|
||||||
super(context);
|
|
||||||
getHolder().addCallback(this);
|
|
||||||
|
|
||||||
setFocusable(true);
|
|
||||||
setFocusableInTouchMode(true);
|
|
||||||
requestFocus();
|
|
||||||
setOnKeyListener(this);
|
|
||||||
setOnTouchListener(this);
|
|
||||||
|
|
||||||
mDisplay = ((WindowManager)context.getSystemService(Context.WINDOW_SERVICE)).getDefaultDisplay();
|
|
||||||
mSensorManager = (SensorManager)context.getSystemService(Context.SENSOR_SERVICE);
|
|
||||||
|
|
||||||
setOnGenericMotionListener(SDLActivity.getMotionListener());
|
|
||||||
|
|
||||||
// Some arbitrary defaults to avoid a potential division by zero
|
|
||||||
mWidth = 1.0f;
|
|
||||||
mHeight = 1.0f;
|
|
||||||
|
|
||||||
mIsSurfaceReady = false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void handlePause() {
|
|
||||||
enableSensor(Sensor.TYPE_ACCELEROMETER, false);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void handleResume() {
|
|
||||||
setFocusable(true);
|
|
||||||
setFocusableInTouchMode(true);
|
|
||||||
requestFocus();
|
|
||||||
setOnKeyListener(this);
|
|
||||||
setOnTouchListener(this);
|
|
||||||
enableSensor(Sensor.TYPE_ACCELEROMETER, true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public Surface getNativeSurface() {
|
|
||||||
return getHolder().getSurface();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called when we have a valid drawing surface
|
|
||||||
@Override
|
|
||||||
public void surfaceCreated(SurfaceHolder holder) {
|
|
||||||
Log.v("SDL", "surfaceCreated()");
|
|
||||||
SDLActivity.onNativeSurfaceCreated();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called when we lose the surface
|
|
||||||
@Override
|
|
||||||
public void surfaceDestroyed(SurfaceHolder holder) {
|
|
||||||
Log.v("SDL", "surfaceDestroyed()");
|
|
||||||
|
|
||||||
// Transition to pause, if needed
|
|
||||||
SDLActivity.mNextNativeState = SDLActivity.NativeState.PAUSED;
|
|
||||||
SDLActivity.handleNativeState();
|
|
||||||
|
|
||||||
mIsSurfaceReady = false;
|
|
||||||
SDLActivity.onNativeSurfaceDestroyed();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Called when the surface is resized
|
|
||||||
@Override
|
|
||||||
public void surfaceChanged(SurfaceHolder holder,
|
|
||||||
int format, int width, int height) {
|
|
||||||
Log.v("SDL", "surfaceChanged()");
|
|
||||||
|
|
||||||
if (SDLActivity.mSingleton == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
mWidth = width;
|
|
||||||
mHeight = height;
|
|
||||||
int nDeviceWidth = width;
|
|
||||||
int nDeviceHeight = height;
|
|
||||||
try
|
|
||||||
{
|
|
||||||
if (Build.VERSION.SDK_INT >= 17 /* Android 4.2 (JELLY_BEAN_MR1) */) {
|
|
||||||
DisplayMetrics realMetrics = new DisplayMetrics();
|
|
||||||
mDisplay.getRealMetrics( realMetrics );
|
|
||||||
nDeviceWidth = realMetrics.widthPixels;
|
|
||||||
nDeviceHeight = realMetrics.heightPixels;
|
|
||||||
}
|
|
||||||
} catch(Exception ignored) {
|
|
||||||
}
|
|
||||||
|
|
||||||
synchronized(SDLActivity.getContext()) {
|
|
||||||
// In case we're waiting on a size change after going fullscreen, send a notification.
|
|
||||||
SDLActivity.getContext().notifyAll();
|
|
||||||
}
|
|
||||||
|
|
||||||
Log.v("SDL", "Window size: " + width + "x" + height);
|
|
||||||
Log.v("SDL", "Device size: " + nDeviceWidth + "x" + nDeviceHeight);
|
|
||||||
SDLActivity.nativeSetScreenResolution(width, height, nDeviceWidth, nDeviceHeight, mDisplay.getRefreshRate());
|
|
||||||
SDLActivity.onNativeResize();
|
|
||||||
|
|
||||||
// Prevent a screen distortion glitch,
|
|
||||||
// for instance when the device is in Landscape and a Portrait App is resumed.
|
|
||||||
boolean skip = false;
|
|
||||||
int requestedOrientation = SDLActivity.mSingleton.getRequestedOrientation();
|
|
||||||
|
|
||||||
if (requestedOrientation == ActivityInfo.SCREEN_ORIENTATION_PORTRAIT || requestedOrientation == ActivityInfo.SCREEN_ORIENTATION_SENSOR_PORTRAIT) {
|
|
||||||
if (mWidth > mHeight) {
|
|
||||||
skip = true;
|
|
||||||
}
|
|
||||||
} else if (requestedOrientation == ActivityInfo.SCREEN_ORIENTATION_LANDSCAPE || requestedOrientation == ActivityInfo.SCREEN_ORIENTATION_SENSOR_LANDSCAPE) {
|
|
||||||
if (mWidth < mHeight) {
|
|
||||||
skip = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Special Patch for Square Resolution: Black Berry Passport
|
|
||||||
if (skip) {
|
|
||||||
double min = Math.min(mWidth, mHeight);
|
|
||||||
double max = Math.max(mWidth, mHeight);
|
|
||||||
|
|
||||||
if (max / min < 1.20) {
|
|
||||||
Log.v("SDL", "Don't skip on such aspect-ratio. Could be a square resolution.");
|
|
||||||
skip = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Don't skip in MultiWindow.
|
|
||||||
if (skip) {
|
|
||||||
if (Build.VERSION.SDK_INT >= 24 /* Android 7.0 (N) */) {
|
|
||||||
if (SDLActivity.mSingleton.isInMultiWindowMode()) {
|
|
||||||
Log.v("SDL", "Don't skip in Multi-Window");
|
|
||||||
skip = false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (skip) {
|
|
||||||
Log.v("SDL", "Skip .. Surface is not ready.");
|
|
||||||
mIsSurfaceReady = false;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* If the surface has been previously destroyed by onNativeSurfaceDestroyed, recreate it here */
|
|
||||||
SDLActivity.onNativeSurfaceChanged();
|
|
||||||
|
|
||||||
/* Surface is ready */
|
|
||||||
mIsSurfaceReady = true;
|
|
||||||
|
|
||||||
SDLActivity.mNextNativeState = SDLActivity.NativeState.RESUMED;
|
|
||||||
SDLActivity.handleNativeState();
|
|
||||||
}
|
|
||||||
|
|
||||||
// Key events
|
|
||||||
@Override
|
|
||||||
public boolean onKey(View v, int keyCode, KeyEvent event) {
|
|
||||||
return SDLActivity.handleKeyEvent(v, keyCode, event, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
// Touch events
|
|
||||||
@Override
|
|
||||||
public boolean onTouch(View v, MotionEvent event) {
|
|
||||||
/* Ref: http://developer.android.com/training/gestures/multi.html */
|
|
||||||
int touchDevId = event.getDeviceId();
|
|
||||||
final int pointerCount = event.getPointerCount();
|
|
||||||
int action = event.getActionMasked();
|
|
||||||
int pointerFingerId;
|
|
||||||
int i = -1;
|
|
||||||
float x,y,p;
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Prevent id to be -1, since it's used in SDL internal for synthetic events
|
|
||||||
* Appears when using Android emulator, eg:
|
|
||||||
* adb shell input mouse tap 100 100
|
|
||||||
* adb shell input touchscreen tap 100 100
|
|
||||||
*/
|
|
||||||
if (touchDevId < 0) {
|
|
||||||
touchDevId -= 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
// 12290 = Samsung DeX mode desktop mouse
|
|
||||||
// 12290 = 0x3002 = 0x2002 | 0x1002 = SOURCE_MOUSE | SOURCE_TOUCHSCREEN
|
|
||||||
// 0x2 = SOURCE_CLASS_POINTER
|
|
||||||
if (event.getSource() == InputDevice.SOURCE_MOUSE || event.getSource() == (InputDevice.SOURCE_MOUSE | InputDevice.SOURCE_TOUCHSCREEN)) {
|
|
||||||
int mouseButton = 1;
|
|
||||||
try {
|
|
||||||
Object object = event.getClass().getMethod("getButtonState").invoke(event);
|
|
||||||
if (object != null) {
|
|
||||||
mouseButton = (Integer) object;
|
|
||||||
}
|
|
||||||
} catch(Exception ignored) {
|
|
||||||
}
|
|
||||||
|
|
||||||
// We need to check if we're in relative mouse mode and get the axis offset rather than the x/y values
|
|
||||||
// if we are. We'll leverage our existing mouse motion listener
|
|
||||||
SDLGenericMotionListener_API12 motionListener = SDLActivity.getMotionListener();
|
|
||||||
x = motionListener.getEventX(event);
|
|
||||||
y = motionListener.getEventY(event);
|
|
||||||
|
|
||||||
SDLActivity.onNativeMouse(mouseButton, action, x, y, motionListener.inRelativeMode());
|
|
||||||
} else {
|
|
||||||
switch(action) {
|
|
||||||
case MotionEvent.ACTION_MOVE:
|
|
||||||
for (i = 0; i < pointerCount; i++) {
|
|
||||||
pointerFingerId = event.getPointerId(i);
|
|
||||||
x = event.getX(i) / mWidth;
|
|
||||||
y = event.getY(i) / mHeight;
|
|
||||||
p = event.getPressure(i);
|
|
||||||
if (p > 1.0f) {
|
|
||||||
// may be larger than 1.0f on some devices
|
|
||||||
// see the documentation of getPressure(i)
|
|
||||||
p = 1.0f;
|
|
||||||
}
|
|
||||||
SDLActivity.onNativeTouch(touchDevId, pointerFingerId, action, x, y, p);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_UP:
|
|
||||||
case MotionEvent.ACTION_DOWN:
|
|
||||||
// Primary pointer up/down, the index is always zero
|
|
||||||
i = 0;
|
|
||||||
/* fallthrough */
|
|
||||||
case MotionEvent.ACTION_POINTER_UP:
|
|
||||||
case MotionEvent.ACTION_POINTER_DOWN:
|
|
||||||
// Non primary pointer up/down
|
|
||||||
if (i == -1) {
|
|
||||||
i = event.getActionIndex();
|
|
||||||
}
|
|
||||||
|
|
||||||
pointerFingerId = event.getPointerId(i);
|
|
||||||
x = event.getX(i) / mWidth;
|
|
||||||
y = event.getY(i) / mHeight;
|
|
||||||
p = event.getPressure(i);
|
|
||||||
if (p > 1.0f) {
|
|
||||||
// may be larger than 1.0f on some devices
|
|
||||||
// see the documentation of getPressure(i)
|
|
||||||
p = 1.0f;
|
|
||||||
}
|
|
||||||
SDLActivity.onNativeTouch(touchDevId, pointerFingerId, action, x, y, p);
|
|
||||||
break;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_CANCEL:
|
|
||||||
for (i = 0; i < pointerCount; i++) {
|
|
||||||
pointerFingerId = event.getPointerId(i);
|
|
||||||
x = event.getX(i) / mWidth;
|
|
||||||
y = event.getY(i) / mHeight;
|
|
||||||
p = event.getPressure(i);
|
|
||||||
if (p > 1.0f) {
|
|
||||||
// may be larger than 1.0f on some devices
|
|
||||||
// see the documentation of getPressure(i)
|
|
||||||
p = 1.0f;
|
|
||||||
}
|
|
||||||
SDLActivity.onNativeTouch(touchDevId, pointerFingerId, MotionEvent.ACTION_UP, x, y, p);
|
|
||||||
}
|
|
||||||
break;
|
|
||||||
|
|
||||||
default:
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Sensor events
|
|
||||||
public void enableSensor(int sensortype, boolean enabled) {
|
|
||||||
// TODO: This uses getDefaultSensor - what if we have >1 accels?
|
|
||||||
if (enabled) {
|
|
||||||
mSensorManager.registerListener(this,
|
|
||||||
mSensorManager.getDefaultSensor(sensortype),
|
|
||||||
SensorManager.SENSOR_DELAY_GAME, null);
|
|
||||||
} else {
|
|
||||||
mSensorManager.unregisterListener(this,
|
|
||||||
mSensorManager.getDefaultSensor(sensortype));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onAccuracyChanged(Sensor sensor, int accuracy) {
|
|
||||||
// TODO
|
|
||||||
}
|
|
||||||
|
|
||||||
@Override
|
|
||||||
public void onSensorChanged(SensorEvent event) {
|
|
||||||
if (event.sensor.getType() == Sensor.TYPE_ACCELEROMETER) {
|
|
||||||
|
|
||||||
// Since we may have an orientation set, we won't receive onConfigurationChanged events.
|
|
||||||
// We thus should check here.
|
|
||||||
int newOrientation;
|
|
||||||
|
|
||||||
float x, y;
|
|
||||||
switch (mDisplay.getRotation()) {
|
|
||||||
case Surface.ROTATION_90:
|
|
||||||
x = -event.values[1];
|
|
||||||
y = event.values[0];
|
|
||||||
newOrientation = SDLActivity.SDL_ORIENTATION_LANDSCAPE;
|
|
||||||
break;
|
|
||||||
case Surface.ROTATION_270:
|
|
||||||
x = event.values[1];
|
|
||||||
y = -event.values[0];
|
|
||||||
newOrientation = SDLActivity.SDL_ORIENTATION_LANDSCAPE_FLIPPED;
|
|
||||||
break;
|
|
||||||
case Surface.ROTATION_180:
|
|
||||||
x = -event.values[0];
|
|
||||||
y = -event.values[1];
|
|
||||||
newOrientation = SDLActivity.SDL_ORIENTATION_PORTRAIT_FLIPPED;
|
|
||||||
break;
|
|
||||||
case Surface.ROTATION_0:
|
|
||||||
default:
|
|
||||||
x = event.values[0];
|
|
||||||
y = event.values[1];
|
|
||||||
newOrientation = SDLActivity.SDL_ORIENTATION_PORTRAIT;
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (newOrientation != SDLActivity.mCurrentOrientation) {
|
|
||||||
SDLActivity.mCurrentOrientation = newOrientation;
|
|
||||||
SDLActivity.onNativeOrientationChanged(newOrientation);
|
|
||||||
}
|
|
||||||
|
|
||||||
SDLActivity.onNativeAccel(-x / SensorManager.GRAVITY_EARTH,
|
|
||||||
y / SensorManager.GRAVITY_EARTH,
|
|
||||||
event.values[2] / SensorManager.GRAVITY_EARTH);
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// Captured pointer events for API 26.
|
|
||||||
public boolean onCapturedPointerEvent(MotionEvent event)
|
|
||||||
{
|
|
||||||
int action = event.getActionMasked();
|
|
||||||
|
|
||||||
float x, y;
|
|
||||||
switch (action) {
|
|
||||||
case MotionEvent.ACTION_SCROLL:
|
|
||||||
x = event.getAxisValue(MotionEvent.AXIS_HSCROLL, 0);
|
|
||||||
y = event.getAxisValue(MotionEvent.AXIS_VSCROLL, 0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, false);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_HOVER_MOVE:
|
|
||||||
case MotionEvent.ACTION_MOVE:
|
|
||||||
x = event.getX(0);
|
|
||||||
y = event.getY(0);
|
|
||||||
SDLActivity.onNativeMouse(0, action, x, y, true);
|
|
||||||
return true;
|
|
||||||
|
|
||||||
case MotionEvent.ACTION_BUTTON_PRESS:
|
|
||||||
case MotionEvent.ACTION_BUTTON_RELEASE:
|
|
||||||
|
|
||||||
// Change our action value to what SDL's code expects.
|
|
||||||
if (action == MotionEvent.ACTION_BUTTON_PRESS) {
|
|
||||||
action = MotionEvent.ACTION_DOWN;
|
|
||||||
} else { /* MotionEvent.ACTION_BUTTON_RELEASE */
|
|
||||||
action = MotionEvent.ACTION_UP;
|
|
||||||
}
|
|
||||||
|
|
||||||
x = event.getX(0);
|
|
||||||
y = event.getY(0);
|
|
||||||
int button = event.getButtonState();
|
|
||||||
|
|
||||||
SDLActivity.onNativeMouse(button, action, x, y, true);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
Before Width: | Height: | Size: 6.2 KiB |
Before Width: | Height: | Size: 4.2 KiB |
Before Width: | Height: | Size: 1.7 KiB |
Before Width: | Height: | Size: 1.7 KiB |
Before Width: | Height: | Size: 3.3 KiB |
Before Width: | Height: | Size: 2.1 KiB |
Before Width: | Height: | Size: 1003 B |
Before Width: | Height: | Size: 1003 B |
Before Width: | Height: | Size: 8.8 KiB |
Before Width: | Height: | Size: 5.8 KiB |
Before Width: | Height: | Size: 2.3 KiB |
Before Width: | Height: | Size: 2.3 KiB |
Before Width: | Height: | Size: 17 KiB |
Before Width: | Height: | Size: 10 KiB |
Before Width: | Height: | Size: 4.3 KiB |
Before Width: | Height: | Size: 4.3 KiB |
Before Width: | Height: | Size: 25 KiB |
Before Width: | Height: | Size: 15 KiB |
Before Width: | Height: | Size: 6.1 KiB |
Before Width: | Height: | Size: 6.1 KiB |
@ -1 +0,0 @@
|
|||||||
melonx
|
|
Before Width: | Height: | Size: 2.6 KiB |
Before Width: | Height: | Size: 2.6 KiB |
Before Width: | Height: | Size: 4.8 KiB |
Before Width: | Height: | Size: 937 B |
Before Width: | Height: | Size: 1.5 KiB |
Before Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 4.7 KiB |
Before Width: | Height: | Size: 8.0 KiB |
Before Width: | Height: | Size: 1.5 KiB |
Before Width: | Height: | Size: 7.3 KiB |
Before Width: | Height: | Size: 7.3 KiB |
Before Width: | Height: | Size: 13 KiB |
Before Width: | Height: | Size: 2.6 KiB |
Before Width: | Height: | Size: 20 KiB |