From c5dc7b9cd02be7e242ef464d5f03ac625ed61966 Mon Sep 17 00:00:00 2001 From: Dhilan007 Date: Wed, 2 Jul 2014 14:19:10 +0800 Subject: [PATCH 1/6] fixed can't get valid value from trigger button. --- cocos/base/CCController-android.cpp | 3 +-- .../src/org/cocos2dx/lib/GameControllerHelper.java | 14 ++++++++++++++ .../proj.android/project.properties | 2 +- .../cocos2dx/game_controller_test/AppActivity.java | 2 +- 4 files changed, 17 insertions(+), 4 deletions(-) diff --git a/cocos/base/CCController-android.cpp b/cocos/base/CCController-android.cpp index e501c34c82..c268009de0 100644 --- a/cocos/base/CCController-android.cpp +++ b/cocos/base/CCController-android.cpp @@ -121,8 +121,7 @@ public: void sendEventButton(ControllerButtonInput* button, bool isPressed, float value, bool isAnalog) { button->setPressed(isPressed); - if (!isAnalog) - button->setValue(value); + button->setValue(value); button->setAnalog(isAnalog); EventController evt(EventController::ControllerEventType::BUTTON_STATUS_CHANGED, _controller, button); Director::getInstance()->getEventDispatcher()->dispatchEvent(&evt); diff --git a/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerHelper.java b/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerHelper.java index c70d8b48ba..acf0d7f299 100644 --- a/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerHelper.java +++ b/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerHelper.java @@ -1,5 +1,6 @@ package org.cocos2dx.lib; +import android.util.Log; import android.util.SparseIntArray; import android.view.InputDevice; import android.view.KeyEvent; @@ -17,6 +18,7 @@ public class GameControllerHelper { private static final int AXIS_RZ = 14; private static final int AXIS_LTRIGGER = 17; private static final int AXIS_RTRIGGER = 18; + public static final int AXIS_GAS = 22; private static final int AXIS_BRAKE = 23; private static final int AXIS_THROTTLE = 19; @@ -63,6 +65,7 @@ public class GameControllerHelper { private float mOldRightTrigger = 0.0f; private float mOldThrottle = 0.0f; private float mOldBrake = 0.0f; + private float mOldGas = 0.0f; public boolean dispatchGenericMotionEvent(MotionEvent event) { boolean handled = false; @@ -146,6 +149,17 @@ public class GameControllerHelper { mOldThrottle = newAXIS_THROTTLE; handled = true; } + + float newAXIS_GAS = event.getAxisValue(AXIS_GAS); + if (Float.compare(newAXIS_GAS , mOldGas) != 0) { + if (Float.compare(newAXIS_GAS, 0.0f) == 0) { + GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, false, 0.0f, true); + }else { + GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, true, newAXIS_GAS, true); + } + mOldGas = newAXIS_GAS; + handled = true; + } } } diff --git a/tests/game-controller-test/proj.android/project.properties b/tests/game-controller-test/proj.android/project.properties index 874fc735f4..5a9bc921d9 100644 --- a/tests/game-controller-test/proj.android/project.properties +++ b/tests/game-controller-test/proj.android/project.properties @@ -10,4 +10,4 @@ # Project target. target=android-10 -android.library.reference.1=../../../cocos/platform/android/ControllerAutoAdapter +android.library.reference.1=../../../cocos/platform/android/ControllerManualAdapter diff --git a/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java b/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java index 382fd95993..17096cc304 100644 --- a/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java +++ b/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java @@ -38,7 +38,7 @@ public class AppActivity extends GameControllerActivity { //Automatic adaptation for connect controller. //Supported Platform: Nibiru / Moga / Ouya TV - this.connectController(); + //this.connectController(); //Manually specify an adapter. //Requirements: using libControllerDriveAdapter project From adfc010555acd23edce4b4061b6f08405107763c Mon Sep 17 00:00:00 2001 From: Dhilan007 Date: Wed, 2 Jul 2014 15:09:24 +0800 Subject: [PATCH 2/6] fixed can't get valid value from trigger button[Moga] --- .../org/cocos2dx/lib/GameControllerMoga.java | 49 ++++++++++++++++--- 1 file changed, 43 insertions(+), 6 deletions(-) diff --git a/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerMoga.java b/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerMoga.java index c08bfb988f..0ebfc23a70 100644 --- a/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerMoga.java +++ b/cocos/platform/android/ControllerManualAdapter/src/org/cocos2dx/lib/GameControllerMoga.java @@ -20,6 +20,9 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel private float mOldLeftThumbstickY = 0.0f; private float mOldRightThumbstickX = 0.0f; private float mOldRightThumbstickY = 0.0f; + + private float mOldLeftTrigger = 0.0f; + private float mOldRightTrigger = 0.0f; private SparseIntArray mKeyMap = null; @@ -60,9 +63,13 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel GameControllerDelegate.BUTTON_RIGHT_THUMBSTICK); } - public void onKeyEvent(KeyEvent event) { - boolean isPressed = event.getAction() == KeyEvent.ACTION_DOWN; + public void onKeyEvent(KeyEvent event) { int keycode = event.getKeyCode(); + if (keycode == KeyEvent.KEYCODE_BUTTON_L2 + || keycode == KeyEvent.KEYCODE_BUTTON_R2) { + return; + } + boolean isPressed = event.getAction() == KeyEvent.ACTION_DOWN; boolean isAnalog = false; if (keycode == KeyEvent.KEYCODE_BUTTON_THUMBL @@ -82,10 +89,12 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel if (mControllerEventListener == null) { return; } + int controllerId = event.getControllerId(); + float newLeftThumbstickX = event.getAxisValue(MotionEvent.AXIS_X); if (newLeftThumbstickX != mOldLeftThumbstickX) { mControllerEventListener.onAxisEvent(mVendorName, - event.getControllerId(), + controllerId, GameControllerDelegate.THUMBSTICK_LEFT_X, newLeftThumbstickX, true); mOldLeftThumbstickX = newLeftThumbstickX; @@ -94,7 +103,7 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel float newLeftThumbstickY = event.getAxisValue(MotionEvent.AXIS_Y); if (newLeftThumbstickY != mOldLeftThumbstickY) { mControllerEventListener.onAxisEvent(mVendorName, - event.getControllerId(), + controllerId, GameControllerDelegate.THUMBSTICK_LEFT_Y, newLeftThumbstickY, true); mOldLeftThumbstickY = newLeftThumbstickY; @@ -103,7 +112,7 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel float newRightThumbstickX = event.getAxisValue(MotionEvent.AXIS_Z); if (newRightThumbstickX != mOldRightThumbstickX) { mControllerEventListener.onAxisEvent(mVendorName, - event.getControllerId(), + controllerId, GameControllerDelegate.THUMBSTICK_RIGHT_X, newRightThumbstickX, true); mOldRightThumbstickX = newRightThumbstickX; @@ -112,11 +121,39 @@ public class GameControllerMoga implements ControllerListener, GameControllerDel float newRightThumbstickY = event.getAxisValue(MotionEvent.AXIS_RZ); if (newRightThumbstickY != mOldRightThumbstickY) { mControllerEventListener.onAxisEvent(mVendorName, - event.getControllerId(), + controllerId, GameControllerDelegate.THUMBSTICK_RIGHT_Y, newRightThumbstickY, true); mOldRightThumbstickY = newRightThumbstickY; } + + float newLeftTrigger = event.getAxisValue(MotionEvent.AXIS_LTRIGGER); + if (newLeftTrigger != mOldLeftTrigger) { + boolean isPressed = true; + if (Float.compare(newLeftTrigger, 0.0f) == 0) { + isPressed = false; + } + mControllerEventListener.onButtonEvent(mVendorName, + controllerId, + GameControllerDelegate.BUTTON_LEFT_TRIGGER, + isPressed, + newLeftTrigger, true); + mOldLeftTrigger = newLeftTrigger; + } + + float newRightTrigger = event.getAxisValue(MotionEvent.AXIS_RTRIGGER); + if (newRightTrigger != mOldRightTrigger) { + boolean isPressed = true; + if (Float.compare(newRightTrigger, 0.0f) == 0) { + isPressed = false; + } + mControllerEventListener.onButtonEvent(mVendorName, + controllerId, + GameControllerDelegate.BUTTON_RIGHT_TRIGGER, + isPressed, + newRightTrigger, true); + mOldRightTrigger = newRightTrigger; + } } @Override From 35ff23fa6df6b0baef3bf5f400d469b7a2fdbb37 Mon Sep 17 00:00:00 2001 From: boyu0 Date: Wed, 2 Jul 2014 15:16:06 +0800 Subject: [PATCH 3/6] support for Xcode6 iphone 5s simulator --- .../proj.ios_mac/cocos2d_lua_bindings.xcodeproj/project.pbxproj | 2 ++ 1 file changed, 2 insertions(+) diff --git a/cocos/scripting/lua-bindings/proj.ios_mac/cocos2d_lua_bindings.xcodeproj/project.pbxproj b/cocos/scripting/lua-bindings/proj.ios_mac/cocos2d_lua_bindings.xcodeproj/project.pbxproj index 3aa43b7a1c..afa4c1df66 100644 --- a/cocos/scripting/lua-bindings/proj.ios_mac/cocos2d_lua_bindings.xcodeproj/project.pbxproj +++ b/cocos/scripting/lua-bindings/proj.ios_mac/cocos2d_lua_bindings.xcodeproj/project.pbxproj @@ -945,6 +945,7 @@ ); "OTHER_LDFLAGS[arch=arm64]" = "-llua"; "OTHER_LDFLAGS[sdk=iphonesimulator7.1]" = "-llua"; + "OTHER_LDFLAGS[sdk=iphonesimulator8.0]" = "-llua"; SDKROOT = iphoneos; TARGETED_DEVICE_FAMILY = "1,2"; USER_HEADER_SEARCH_PATHS = "$(inherited) $(SRCROOT)/../../../platform/ios"; @@ -970,6 +971,7 @@ ); "OTHER_LDFLAGS[arch=arm64]" = "-llua"; "OTHER_LDFLAGS[sdk=iphonesimulator7.1]" = "-llua"; + "OTHER_LDFLAGS[sdk=iphonesimulator8.0]" = "-llua"; SDKROOT = iphoneos; TARGETED_DEVICE_FAMILY = "1,2"; USER_HEADER_SEARCH_PATHS = "$(inherited) $(SRCROOT)/../../../platform/ios"; From 097f782c3826e49dfaf64b6e76b95908490253e5 Mon Sep 17 00:00:00 2001 From: Dhilan007 Date: Wed, 2 Jul 2014 15:17:29 +0800 Subject: [PATCH 4/6] remove ControllerAutoAdapter project. --- .../android/ControllerAutoAdapter/.classpath | 9 - .../android/ControllerAutoAdapter/.project | 33 - .../.settings/org.eclipse.jdt.core.prefs | 4 - .../ControllerAutoAdapter/AndroidManifest.xml | 9 - .../ControllerAutoAdapter/ant.properties | 17 - .../android/ControllerAutoAdapter/build.xml | 83 -- .../libs/android-async-http-1.4.4.jar | Bin 60962 -> 0 bytes .../android/ControllerAutoAdapter/lint.xml | 6 - .../proguard-project.txt | 20 - .../ControllerAutoAdapter/project.properties | 16 - .../ControllerAutoAdapter/res/.gitignore | 0 .../cocos2dx/lib/GameControllerActivity.java | 284 ------ .../cocos2dx/lib/GameControllerHelper.java | 856 ------------------ .../InputManagerCompat.java | 140 --- .../inputmanagercompat/InputManagerV16.java | 107 --- .../inputmanagercompat/InputManagerV9.java | 211 ----- 16 files changed, 1795 deletions(-) delete mode 100644 cocos/platform/android/ControllerAutoAdapter/.classpath delete mode 100644 cocos/platform/android/ControllerAutoAdapter/.project delete mode 100644 cocos/platform/android/ControllerAutoAdapter/.settings/org.eclipse.jdt.core.prefs delete mode 100644 cocos/platform/android/ControllerAutoAdapter/AndroidManifest.xml delete mode 100644 cocos/platform/android/ControllerAutoAdapter/ant.properties delete mode 100644 cocos/platform/android/ControllerAutoAdapter/build.xml delete mode 100644 cocos/platform/android/ControllerAutoAdapter/libs/android-async-http-1.4.4.jar delete mode 100644 cocos/platform/android/ControllerAutoAdapter/lint.xml delete mode 100644 cocos/platform/android/ControllerAutoAdapter/proguard-project.txt delete mode 100644 cocos/platform/android/ControllerAutoAdapter/project.properties delete mode 100644 cocos/platform/android/ControllerAutoAdapter/res/.gitignore delete mode 100644 cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerActivity.java delete mode 100644 cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerHelper.java delete mode 100644 cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerCompat.java delete mode 100644 cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV16.java delete mode 100644 cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV9.java diff --git a/cocos/platform/android/ControllerAutoAdapter/.classpath b/cocos/platform/android/ControllerAutoAdapter/.classpath deleted file mode 100644 index 51769745b2..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/.classpath +++ /dev/null @@ -1,9 +0,0 @@ - - - - - - - - - diff --git a/cocos/platform/android/ControllerAutoAdapter/.project b/cocos/platform/android/ControllerAutoAdapter/.project deleted file mode 100644 index c49a037939..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/.project +++ /dev/null @@ -1,33 +0,0 @@ - - - libControllerAutoAdapter - - - - - - com.android.ide.eclipse.adt.ResourceManagerBuilder - - - - - com.android.ide.eclipse.adt.PreCompilerBuilder - - - - - org.eclipse.jdt.core.javabuilder - - - - - com.android.ide.eclipse.adt.ApkBuilder - - - - - - com.android.ide.eclipse.adt.AndroidNature - org.eclipse.jdt.core.javanature - - diff --git a/cocos/platform/android/ControllerAutoAdapter/.settings/org.eclipse.jdt.core.prefs b/cocos/platform/android/ControllerAutoAdapter/.settings/org.eclipse.jdt.core.prefs deleted file mode 100644 index b080d2ddc8..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/.settings/org.eclipse.jdt.core.prefs +++ /dev/null @@ -1,4 +0,0 @@ -eclipse.preferences.version=1 -org.eclipse.jdt.core.compiler.codegen.targetPlatform=1.6 -org.eclipse.jdt.core.compiler.compliance=1.6 -org.eclipse.jdt.core.compiler.source=1.6 diff --git a/cocos/platform/android/ControllerAutoAdapter/AndroidManifest.xml b/cocos/platform/android/ControllerAutoAdapter/AndroidManifest.xml deleted file mode 100644 index d1f4a837d8..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/AndroidManifest.xml +++ /dev/null @@ -1,9 +0,0 @@ - - - - - - diff --git a/cocos/platform/android/ControllerAutoAdapter/ant.properties b/cocos/platform/android/ControllerAutoAdapter/ant.properties deleted file mode 100644 index b0971e891e..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/ant.properties +++ /dev/null @@ -1,17 +0,0 @@ -# This file is used to override default values used by the Ant build system. -# -# This file must be checked into Version Control Systems, as it is -# integral to the build system of your project. - -# This file is only used by the Ant script. - -# You can use this to override default values such as -# 'source.dir' for the location of your java source folder and -# 'out.dir' for the location of your output folder. - -# You can also use it define how the release builds are signed by declaring -# the following properties: -# 'key.store' for the location of your keystore and -# 'key.alias' for the name of the key to use. -# The password will be asked during the build when you use the 'release' target. - diff --git a/cocos/platform/android/ControllerAutoAdapter/build.xml b/cocos/platform/android/ControllerAutoAdapter/build.xml deleted file mode 100644 index 413effd90c..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/build.xml +++ /dev/null @@ -1,83 +0,0 @@ - - - - - - - - - - - - - - - - - - - - - - - diff --git a/cocos/platform/android/ControllerAutoAdapter/libs/android-async-http-1.4.4.jar b/cocos/platform/android/ControllerAutoAdapter/libs/android-async-http-1.4.4.jar deleted file mode 100644 index 75af7015db5f770102182bfc35c9e6fe167d83a7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 60962 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 - - - - - \ No newline at end of file diff --git a/cocos/platform/android/ControllerAutoAdapter/proguard-project.txt b/cocos/platform/android/ControllerAutoAdapter/proguard-project.txt deleted file mode 100644 index f2fe1559a2..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/proguard-project.txt +++ /dev/null @@ -1,20 +0,0 @@ -# To enable ProGuard in your project, edit project.properties -# to define the proguard.config property as described in that file. -# -# Add project specific ProGuard rules here. -# By default, the flags in this file are appended to flags specified -# in ${sdk.dir}/tools/proguard/proguard-android.txt -# You can edit the include path and order by changing the ProGuard -# include property in project.properties. -# -# For more details, see -# http://developer.android.com/guide/developing/tools/proguard.html - -# Add any project specific keep options here: - -# 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 *; -#} diff --git a/cocos/platform/android/ControllerAutoAdapter/project.properties b/cocos/platform/android/ControllerAutoAdapter/project.properties deleted file mode 100644 index 9990d6c114..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/project.properties +++ /dev/null @@ -1,16 +0,0 @@ -# This file is automatically generated by Android Tools. -# Do not modify this file -- YOUR CHANGES WILL BE ERASED! -# -# This file must be checked in Version Control Systems. -# -# To customize properties used by the Ant build system edit -# "ant.properties", and override values to adapt the script to your -# project structure. -# -# To enable ProGuard to shrink and obfuscate your code, uncomment this (available properties: sdk.dir, user.home): -#proguard.config=${sdk.dir}/tools/proguard/proguard-android.txt:proguard-project.txt - -android.library=true -# Project target. -target=android-16 -android.library.reference.1=../java diff --git a/cocos/platform/android/ControllerAutoAdapter/res/.gitignore b/cocos/platform/android/ControllerAutoAdapter/res/.gitignore deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerActivity.java b/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerActivity.java deleted file mode 100644 index c38c34665c..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerActivity.java +++ /dev/null @@ -1,284 +0,0 @@ -/**************************************************************************** -Copyright (c) 2010-2013 cocos2d-x.org - -http://www.cocos2d-x.org - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is -furnished to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in -all copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. - ****************************************************************************/ -package org.cocos2dx.lib; - -import org.cocos2dx.lib.GameControllerDelegate.ControllerEventListener; -import org.cocos2dx.lib.inputmanagercompat.InputManagerCompat; -import org.cocos2dx.lib.inputmanagercompat.InputManagerCompat.InputDeviceListener; -import org.cocos2dx.lib.Cocos2dxActivity; - -import android.os.Bundle; -import android.view.InputDevice; -import android.view.KeyEvent; -import android.view.MotionEvent; -import android.util.Log; - -public abstract class GameControllerActivity extends Cocos2dxActivity implements InputDeviceListener { - // =========================================================== - // Constants - // =========================================================== - - private final static String TAG = GameControllerActivity.class.getSimpleName(); - - public static final int DRIVERTYPE_NIBIRU = 0; - public static final int DRIVERTYPE_MOGA = 1; - public static final int DRIVERTYPE_OUYA = 2; - public static final int DRIVERTYPE_STANDARD = 3; - public static final int DRIVERTYPE_UNKNOWN = 4; - - // =========================================================== - // Fields - // =========================================================== - private static GameControllerActivity sGameControllerActivity; - private InputManagerCompat mInputManager = null; - - protected GameControllerHelper mControllerHelper = null; - - protected GameControllerDelegate mControllerNibiru = null; - protected GameControllerDelegate mControllerMoga = null; - protected GameControllerDelegate mControllerOuya = null; - - public void connectController(){ - mControllerHelper.connectController(); - } - - public void setGameControllerInstance(GameControllerDelegate controllerDelegate, int driveType) { - if (driveType == DRIVERTYPE_NIBIRU) { - mControllerNibiru = controllerDelegate; - }else if (driveType == DRIVERTYPE_MOGA) { - mControllerMoga = controllerDelegate; - } - else if (driveType == DRIVERTYPE_OUYA) { - mControllerOuya = controllerDelegate; - } - controllerDelegate.setControllerEventListener(mControllerEventListener); - controllerDelegate.onCreate(sGameControllerActivity); - } - - public GameControllerDelegate getGameControllerDelegate(int driveType){ - if (driveType == DRIVERTYPE_NIBIRU) { - return mControllerNibiru; - }else if (driveType == DRIVERTYPE_MOGA) { - return mControllerMoga; - } - else if (driveType == DRIVERTYPE_OUYA) { - return mControllerOuya; - } - - return null; - } - - ControllerEventListener mControllerEventListener = new ControllerEventListener() { - - @Override - public void onButtonEvent(String vendorName, int controller, int button, - boolean isPressed, float value, boolean isAnalog) { - GameControllerAdapter.onButtonEvent(vendorName, controller, button, isPressed, value, isAnalog); - } - - @Override - public void onAxisEvent(String vendorName, int controller, int axisID, - float value, boolean isAnalog) { - GameControllerAdapter.onAxisEvent(vendorName, controller, axisID, value, isAnalog); - } - - @Override - public void onConnected(String vendorName, int controller) { - GameControllerAdapter.onConnected(vendorName, controller); - } - - @Override - public void onDisconnected(String vendorName, int controller) { - GameControllerAdapter.onDisconnected(vendorName, controller); - } - }; - - // =========================================================== - // Constructors - // =========================================================== - - @Override - protected void onCreate(final Bundle savedInstanceState) { - super.onCreate(savedInstanceState); - - sGameControllerActivity = this; - mInputManager = InputManagerCompat.Factory.getInputManager(this); - mInputManager.registerInputDeviceListener(this, null); - - if (mControllerNibiru != null) { - mControllerNibiru.onCreate(this); - } - if (mControllerMoga != null) { - mControllerMoga.onCreate(this); - } - if (mControllerOuya != null) { - mControllerOuya.onCreate(this); - } - if (mControllerHelper == null) { - mControllerHelper = new GameControllerHelper(this); - } - } - - // =========================================================== - // Getter & Setter - // =========================================================== - - // =========================================================== - // Methods for/from SuperClass/Interfaces - // =========================================================== - - @Override - public boolean dispatchKeyEvent(KeyEvent event) { - boolean handled = false; - if (mControllerNibiru != null) { - handled |= mControllerNibiru.dispatchKeyEvent(event); - } - if (mControllerMoga != null) { - handled |= mControllerMoga.dispatchKeyEvent(event); - } - if (mControllerOuya != null) { - handled |= mControllerOuya.dispatchKeyEvent(event); - } - - handled |= mControllerHelper.dispatchKeyEvent(event); - Log.d(TAG, "dispatchKeyEvent:" + handled); - - return handled || super.dispatchKeyEvent(event); - } - - @Override - public boolean dispatchGenericMotionEvent(MotionEvent event) { - boolean handled = false; - if (mControllerNibiru != null) { - handled |= mControllerNibiru.dispatchGenericMotionEvent(event); - } - if (mControllerMoga != null) { - handled |= mControllerMoga.dispatchGenericMotionEvent(event); - } - if (mControllerOuya != null) { - handled |= mControllerOuya.dispatchGenericMotionEvent(event); - } - - handled |= mControllerHelper.dispatchGenericMotionEvent(event); - - return handled || super.dispatchGenericMotionEvent(event); - } - - @Override - public void onInputDeviceAdded(int deviceId) { - - Log.d(TAG,"onInputDeviceAdded:" + deviceId); - - InputDevice device = InputDevice.getDevice(deviceId); - int deviceSource = device.getSources(); - - if ( ((deviceSource & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD) - || ((deviceSource & InputDevice.SOURCE_JOYSTICK) == InputDevice.SOURCE_JOYSTICK) ) - { - GameControllerAdapter.onConnected("Standard", deviceId); - } - } - /* - * This is an unusual case. Input devices don't typically change, but they - * certainly can --- for example a device may have different modes. We use - * this to make sure that the ship has an up-to-date InputDevice. - * - * @see - * com.example.inputmanagercompat.InputManagerCompat.InputDeviceListener - * #onInputDeviceChanged(int) - */ - @Override - public void onInputDeviceChanged(int deviceId) { - Log.d(TAG,"onInputDeviceChanged:" + deviceId); - } - - /* - * Remove any ship associated with the ID. - * - * @see - * com.example.inputmanagercompat.InputManagerCompat.InputDeviceListener - * #onInputDeviceRemoved(int) - */ - @Override - public void onInputDeviceRemoved(int deviceId) { - Log.d(TAG,"onInputDeviceRemoved:" + deviceId); - - InputDevice device = InputDevice.getDevice(deviceId); - int deviceSource = device.getSources(); - - if ( ((deviceSource & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD) - || ((deviceSource & InputDevice.SOURCE_JOYSTICK) == InputDevice.SOURCE_JOYSTICK) ) - { - GameControllerAdapter.onDisconnected("Standard", deviceId); - } - } - - @Override - protected void onResume() { - super.onResume(); - - if (mControllerNibiru != null) { - mControllerNibiru.onResume(); - } - if (mControllerMoga != null) { - mControllerMoga.onResume(); - } - if (mControllerOuya != null) { - mControllerOuya.onResume(); - } - } - - @Override - protected void onPause() { - if (mControllerNibiru != null) { - mControllerNibiru.onPause(); - } - if (mControllerMoga != null) { - mControllerMoga.onPause(); - } - if (mControllerOuya != null) { - mControllerOuya.onPause(); - } - - super.onPause(); - } - - @Override - protected void onDestroy() { - if (mControllerNibiru != null) { - mControllerNibiru.onDestroy(); - } - if (mControllerMoga != null) { - mControllerMoga.onDestroy(); - } - if (mControllerOuya != null) { - mControllerOuya.onDestroy(); - } - - mControllerHelper.destrory(); - - super.onDestroy(); - } - -} \ No newline at end of file diff --git a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerHelper.java b/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerHelper.java deleted file mode 100644 index 52480b7524..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerHelper.java +++ /dev/null @@ -1,856 +0,0 @@ -package org.cocos2dx.lib; - -import java.io.File; -import java.io.FileOutputStream; -import java.lang.reflect.Method; -import java.util.ArrayList; -import java.util.Iterator; -import java.util.List; -import java.util.Set; - -import org.json.JSONArray; -import org.json.JSONObject; -import org.apache.http.Header; - -import com.loopj.android.http.AsyncHttpClient; -import com.loopj.android.http.FileAsyncHttpResponseHandler; -import com.loopj.android.http.JsonHttpResponseHandler; - -import dalvik.system.DexClassLoader; -import android.bluetooth.BluetoothAdapter; -import android.bluetooth.BluetoothDevice; -import android.content.BroadcastReceiver; -import android.content.Context; -import android.content.Intent; -import android.content.IntentFilter; -import android.content.pm.ApplicationInfo; -import android.content.pm.PackageManager; -import android.content.pm.PackageManager.NameNotFoundException; -import android.net.Uri; -import android.os.Environment; -import android.util.Log; -import android.util.SparseIntArray; -import android.view.InputDevice; -import android.view.KeyEvent; -import android.view.MotionEvent; -import android.widget.Toast; - -public class GameControllerHelper { - - private final static String TAG = GameControllerHelper.class.getSimpleName(); - - public static final String StandardControllerName = "Standard"; - public static final String[] DRIVERS_NAME = {"nibiru","moga","ouya",StandardControllerName}; - - public static final String SPP_UUID = "00001101-0000-1000-8000-00805F9B34FB"; - - SparseIntArray ControllerKeyMap; - - private static final String COCOS_CONTROLLER_URL = "http://115.28.134.83:9000/download/cocoscontroller/"; - private static final String COCOS_CONTROLLER_CONFIG = "controller.json"; - - private static final String NIBIRU_DEP_PACKAGE = "com.nibiru"; - private static final String MOGA__DEP_PACKAGE = "com.bda.pivot.mogapgp"; - - private static GameControllerActivity sGameControllerActivity; - private static GameControllerHelper sControllerHelper; - - private String mLibAdapterFileName; - private int mLibAdapterFileSize; - - private List mNibiruSupportedDrives; - private String mNibiruDepFileName; - private int mNibiruDepFileSize; - - private List mMogaSupportedDrives; - private String mMogaDepFileName; - private int mMogaDepFileSize; - - private List mOuyaSupportedDrives; - - private AsyncHttpClient mDownDepsHttpClient = null; - - private BluetoothAdapter mBluetoothAdapter = null; - private ArrayList mBluetoothDevices = null; - private SparseIntArray mDevicesDriver; - private int mClearDevices = 0; - - private String mConfigFilePath; - private String mLocalSavePath = null; - - private boolean mLazyInit = true; - private boolean mLazyConfigInit = true; - private static ControllerListener mControllerListener = null; - - public static interface ControllerListener{ - void onDownloadConfigStarted(); - void onDownloadConfigFinished(boolean isSuccess); - - void onControllerDiscoveryStarted(); - // - void onControllerDiscoveryFinish(ArrayList devices); - - void onDownloadDepsStarted(); - void onDownloadDepsProgress(int bytesWritten, int totalSize); - void onDownloadDepsFinished(boolean isSuccess); - - void onInstallDriver(String filePath); - void onConnectController(); - } - - public void setControllerListener(ControllerListener listener){ - mControllerListener = listener; - } - - private static final int AXIS_X = 0; - private static final int AXIS_Y = 1; - private static final int AXIS_Z = 11; - private static final int AXIS_RZ = 14; - public static final int AXIS_LTRIGGER = 17; - public static final int AXIS_RTRIGGER = 18; - public static final int AXIS_BRAKE = 23; - public static final int AXIS_THROTTLE = 19; - - public GameControllerHelper(GameControllerActivity activity){ - sGameControllerActivity = activity; - sControllerHelper = this; - - ControllerKeyMap = new SparseIntArray(25); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_A, GameControllerDelegate.BUTTON_A); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_B, GameControllerDelegate.BUTTON_B); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_C, GameControllerDelegate.BUTTON_C); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_X, GameControllerDelegate.BUTTON_X); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_Y, GameControllerDelegate.BUTTON_Y); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_Z, GameControllerDelegate.BUTTON_Z); - - ControllerKeyMap.put(KeyEvent.KEYCODE_DPAD_UP, GameControllerDelegate.BUTTON_DPAD_UP); - ControllerKeyMap.put(KeyEvent.KEYCODE_DPAD_DOWN, GameControllerDelegate.BUTTON_DPAD_DOWN); - ControllerKeyMap.put(KeyEvent.KEYCODE_DPAD_LEFT, GameControllerDelegate.BUTTON_DPAD_LEFT); - ControllerKeyMap.put(KeyEvent.KEYCODE_DPAD_RIGHT, GameControllerDelegate.BUTTON_DPAD_RIGHT); - ControllerKeyMap.put(KeyEvent.KEYCODE_DPAD_CENTER, GameControllerDelegate.BUTTON_DPAD_CENTER); - - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_THUMBL, GameControllerDelegate.BUTTON_LEFT_THUMBSTICK); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_THUMBR, GameControllerDelegate.BUTTON_RIGHT_THUMBSTICK); - - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_L1, GameControllerDelegate.BUTTON_LEFT_SHOULDER); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_R1, GameControllerDelegate.BUTTON_RIGHT_SHOULDER); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_L2, GameControllerDelegate.BUTTON_LEFT_TRIGGER); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_R2, GameControllerDelegate.BUTTON_RIGHT_TRIGGER); - - ControllerKeyMap.put(AXIS_X, GameControllerDelegate.THUMBSTICK_LEFT_X); - ControllerKeyMap.put(AXIS_Y, GameControllerDelegate.THUMBSTICK_LEFT_Y); - ControllerKeyMap.put(AXIS_Z, GameControllerDelegate.THUMBSTICK_RIGHT_X); - ControllerKeyMap.put(AXIS_RZ, GameControllerDelegate.THUMBSTICK_RIGHT_Y); - - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_START, GameControllerDelegate.BUTTON_START); - ControllerKeyMap.put(KeyEvent.KEYCODE_BUTTON_SELECT, GameControllerDelegate.BUTTON_SELECT); - //KEYCODE_BUTTON_MODE - mDownDepsHttpClient = new AsyncHttpClient(); - mDownDepsHttpClient.setTimeout(360 * 1000); - } - - public void connectController(){ - if (mLazyInit) { - mLazyInit = false; - mNibiruSupportedDrives = new ArrayList(30); - mMogaSupportedDrives = new ArrayList(5); - mOuyaSupportedDrives = new ArrayList(5); - - mLocalSavePath = Environment.getExternalStorageDirectory() + File.separator + "CocosGameController" + File.separator; - Log.i(TAG, "mLocalSavePath:" + mLocalSavePath); - mConfigFilePath = sGameControllerActivity.getFilesDir().getAbsolutePath() + File.separator + COCOS_CONTROLLER_CONFIG; - Log.i(TAG, "mConfigFilePath:" + mConfigFilePath); - - } - - if (mControllerListener != null) { - mControllerListener.onDownloadConfigStarted(); - } - if (mLazyConfigInit) { - //if (mDownDepsHttpClient != null) { - // mDownDepsHttpClient.cancelRequests(sGameControllerActivity, true); - //} - requestControllerConfig(); - } - else { - scanBluetoothDrive(); - } - } - - public Set getBondedDevices(){ - if (mBluetoothAdapter == null) { - mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); - if (mBluetoothAdapter == null) { - return null; - } - } - - return mBluetoothAdapter.getBondedDevices(); - } - - public void destrory(){ - if (mDownDepsHttpClient != null) { - mDownDepsHttpClient.cancelRequests(sGameControllerActivity, true); - } - } - - private boolean scanBluetoothDrive(){ - if (mControllerListener != null) { - mControllerListener.onDownloadConfigFinished(true); - } - - if (mBluetoothAdapter == null) { - mBluetoothAdapter = BluetoothAdapter.getDefaultAdapter(); - if (mBluetoothAdapter == null) { - return false; - } - - if (mBluetoothDevices == null) { - mBluetoothDevices = new ArrayList(5); - mDevicesDriver = new SparseIntArray(); - } - - IntentFilter filter = new IntentFilter(BluetoothDevice.ACTION_FOUND); - //filter.addAction(BluetoothDevice.ACTION_BOND_STATE_CHANGED); - //filter.addAction(BluetoothAdapter.ACTION_SCAN_MODE_CHANGED); - //filter.addAction(BluetoothAdapter.ACTION_STATE_CHANGED); - filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_FINISHED); - filter.addAction(BluetoothAdapter.ACTION_DISCOVERY_STARTED); - sGameControllerActivity.registerReceiver(mBluetoothReceiver, filter); - - IntentFilter appFilter = new IntentFilter(); - appFilter.addAction("android.intent.action.PACKAGE_ADDED"); - appFilter.addDataScheme("package"); - sGameControllerActivity.registerReceiver(mAppReceiver, appFilter); - } - - if (!mBluetoothAdapter.isEnabled()) { - mBluetoothAdapter.enable(); - } - - if (mBluetoothAdapter.isDiscovering()) { - mBluetoothAdapter.cancelDiscovery(); - } - - mBluetoothAdapter.startDiscovery(); - - return true; - } - - public int checkDriverType(BluetoothDevice device){ - String deviceName = device.getName(); - - if (mNibiruSupportedDrives.contains(deviceName)) { - return GameControllerActivity.DRIVERTYPE_NIBIRU; - } - else if (mMogaSupportedDrives.contains(deviceName)) { - return GameControllerActivity.DRIVERTYPE_MOGA; - } - else if (mOuyaSupportedDrives.contains(deviceName)) { - return GameControllerActivity.DRIVERTYPE_OUYA; - } - else { - - } - - return GameControllerActivity.DRIVERTYPE_UNKNOWN; - } - - public static void installApplication(String filePath){ - if (sGameControllerActivity != null) { - Intent intent = new Intent(); - intent.setAction(Intent.ACTION_VIEW); - intent.setDataAndType(Uri.parse("file://" + filePath), "application/vnd.android.package-archive"); - intent.setFlags(Intent.FLAG_ACTIVITY_NEW_TASK); - sGameControllerActivity.startActivity(intent); - } - } - - public static boolean checkApplication(String packName){ - try { - ApplicationInfo applicationInfo = sGameControllerActivity.getPackageManager().getApplicationInfo(packName, PackageManager.GET_UNINSTALLED_PACKAGES); - Log.d(TAG, applicationInfo.toString()); - return true; - } catch (NameNotFoundException e) { - return false; - } - } - - private BroadcastReceiver mBluetoothReceiver = new BroadcastReceiver() { - - @Override - public void onReceive(Context context, Intent intent) { - String action = intent.getAction(); - - if (BluetoothDevice.ACTION_FOUND.equals(action)) { - BluetoothDevice device = intent.getParcelableExtra(BluetoothDevice.EXTRA_DEVICE); - if (mBluetoothDevices.contains(device)) { - return; - } - Log.d(TAG, "Remote device discovered :" + device.getName()); - //We can't ensure non-controller can be filtered out.Some game controller marked with computer class. - /*String deviceName = device.getName(); - - if(device.getBluetoothClass().getMajorDeviceClass() == BluetoothClass.Device.Major.COMPUTER - || device.getBluetoothClass().getMajorDeviceClass() == BluetoothClass.Device.Major.PHONE) - { - Log.w(TAG, "Remote device discovered :" + deviceName + " is computer or phone." + device.getBluetoothClass().getMajorDeviceClass()); - return; - }*/ - - mBluetoothDevices.add(device); - int type = checkDriverType(device); - if (type != GameControllerActivity.DRIVERTYPE_UNKNOWN) { - mTargetDriverType = type; - mClearDevices += 1; - } - mDevicesDriver.append(mBluetoothDevices.size() - 1, type); - } - else if (BluetoothAdapter.ACTION_DISCOVERY_FINISHED.equals(action)) { - Log.d(TAG, "The local Bluetooth adapter has finished the device discovery process."); - - if (mControllerListener != null) { - mControllerListener.onControllerDiscoveryFinish(mBluetoothDevices); - } - else { - if (mBluetoothDevices.size() == 0) { - if (checkApplication(NIBIRU_DEP_PACKAGE)) { - downControllerDeps(GameControllerActivity.DRIVERTYPE_NIBIRU); - } - Log.w(TAG, "Not found any supported bluetooth game controller!"); - }else { - if (mClearDevices == 1 ) { - downControllerDeps(mTargetDriverType); - } - else { - Log.i(TAG, "Not clear target!"); - if (checkApplication(NIBIRU_DEP_PACKAGE)) { - downControllerDeps(GameControllerActivity.DRIVERTYPE_NIBIRU); - } - //todo:show sel - } - } - } - } - else if (BluetoothAdapter.ACTION_DISCOVERY_STARTED.equals(action)) { - Log.d(TAG, "The local Bluetooth adapter has started the remote device discovery process."); - if(mControllerListener != null){ - mControllerListener.onControllerDiscoveryStarted(); - } - - mBluetoothDevices.clear(); - mDevicesDriver.clear(); - mClearDevices = 0; - mTargetDriverType = GameControllerActivity.DRIVERTYPE_UNKNOWN; - - //check moga controller - Iterator it = mBluetoothAdapter.getBondedDevices().iterator(); - while (it.hasNext()) { - BluetoothDevice device = it.next(); - //if(device.getBluetoothClass().getMajorDeviceClass() != BluetoothClass.Device.Major.COMPUTER - // && device.getBluetoothClass().getMajorDeviceClass() != BluetoothClass.Device.Major.PHONE) - { - if (!mBluetoothDevices.contains(device)) { - mBluetoothDevices.add(device); - int type = checkDriverType(device); - Log.d(TAG, "BluetoothDevice objects that are bonded (paired) to the local adapter :" + device.getName()); - if (type != GameControllerActivity.DRIVERTYPE_UNKNOWN) { - mClearDevices += 1; - mTargetDriverType = type; - } - mDevicesDriver.append(mBluetoothDevices.size() - 1, type); - } - } - } - } - } - }; - - private static int depsCount = 0; - private static int mTargetDriverType = GameControllerActivity.DRIVERTYPE_UNKNOWN; - - private static int mDownloadTotalSize; - private static int mDownloadLibSize; - private static int mLibDownloadCompletedSize; - private static int mDepDownloadCompletedSize; - - public void downControllerDeps(int driverType){ - if (driverType != GameControllerActivity.DRIVERTYPE_NIBIRU - && driverType != GameControllerActivity.DRIVERTYPE_MOGA - && driverType != GameControllerActivity.DRIVERTYPE_OUYA) { - return; - } - - if (mControllerListener != null) { - mControllerListener.onDownloadDepsStarted(); - } - - mDownloadTotalSize = 0; - mLibDownloadCompletedSize = 0; - mDepDownloadCompletedSize = 0; - - mTargetDriverType = driverType; - depsCount = 1; - - GameControllerUtils.ensureDirectoryExist(mLocalSavePath); - - String remoteDir = COCOS_CONTROLLER_URL + DRIVERS_NAME[driverType] + File.separator; - - if (driverType == GameControllerActivity.DRIVERTYPE_NIBIRU) { - if (!checkApplication(NIBIRU_DEP_PACKAGE)) { - depsCount += 1; - mDownloadTotalSize += mNibiruDepFileSize; - mDownDepsHttpClient.get(remoteDir + mNibiruDepFileName, new MyDepsAsyncHandler( - new File(mLocalSavePath + mNibiruDepFileName), MyDepsAsyncHandler.FILETYPE_DEP_APK)); - } - } - else if (driverType == GameControllerActivity.DRIVERTYPE_MOGA) { - if (!checkApplication(MOGA__DEP_PACKAGE)) { - mDownloadTotalSize += mMogaDepFileSize; - depsCount += 1; - mDownDepsHttpClient.get(remoteDir + mMogaDepFileName, new MyDepsAsyncHandler( - new File(mLocalSavePath + mMogaDepFileName), MyDepsAsyncHandler.FILETYPE_DEP_APK)); - } - } - else if(driverType == GameControllerActivity.DRIVERTYPE_OUYA){ - - } - - File libFile = new File(mLocalSavePath + mLibAdapterFileName); - if (libFile.exists() && libFile.length() == mLibAdapterFileSize) { - depsCount -= 1; - if (depsCount == 0) { - onDepsReady(); - } - }else { - mDownloadLibSize = mLibAdapterFileSize; - mDownloadTotalSize += mDownloadLibSize; - mDownDepsHttpClient.get(COCOS_CONTROLLER_URL + mLibAdapterFileName, new MyDepsAsyncHandler( - new File(mLocalSavePath + mLibAdapterFileName), MyDepsAsyncHandler.FILETYPE_JAR)); - } - } - - static class MyDepsAsyncHandler extends FileAsyncHttpResponseHandler{ - - public static final int FILETYPE_DEP_APK = 0; - public static final int FILETYPE_JAR = 1; - - private int mFileType = FILETYPE_DEP_APK; - - public MyDepsAsyncHandler(File file, int fileType) { - super(file); - mFileType = fileType; - } - - @Override - public void onFailure(int statusCode, Header[] headers, - Throwable e, File file) { - if (mFileType == FILETYPE_JAR) { - if (file.exists() && file.length() == mDownloadLibSize) { - depsCount -= 1; - if (depsCount == 0) { - if (mControllerListener != null) { - mControllerListener.onDownloadDepsFinished(true); - } - sControllerHelper.onDepsReady(); - } - } - } - else if (mFileType == FILETYPE_DEP_APK) { - if (mControllerListener != null) { - mControllerListener.onDownloadDepsFinished(false); - } - } - - Log.e(TAG,"Failed to download:" + file.getName()); - } - - @Override - public void onProgress(int bytesWritten, int totalSize) { - if (mFileType == FILETYPE_JAR) { - mLibDownloadCompletedSize = bytesWritten; - } else { - mDepDownloadCompletedSize = bytesWritten; - } - if (mControllerListener != null) { - - mControllerListener.onDownloadDepsProgress(mLibDownloadCompletedSize + mDepDownloadCompletedSize, mDownloadTotalSize); - } - Log.d(TAG, "totalSize:" + totalSize + ", bytesWritten:" + bytesWritten); - } - - - public void onSuccess(File file) { - Log.d(TAG, "11Down file success:" + file.getName()); - - depsCount -= 1; - if (depsCount == 0) { - if (mControllerListener != null) { - mControllerListener.onDownloadDepsFinished(true); - } - sControllerHelper.onDepsReady(); - } - super.onSuccess(file); - } - - /*@Override - public void onSuccess(int arg0, Header[] arg1, File file) { - Log.d(TAG, "22Down file success:" + file.getName()); - - depsCount -= 1; - if (depsCount == 0) { - if (mControllerListener != null) { - mControllerListener.onDownloadDepsFinished(true); - } - sControllerHelper.onDepsReady(); - } - }*/ - } - - private void onDepsReady(){ - Log.d(TAG, "onDepsReady:" + mTargetDriverType); - - if (mTargetDriverType == GameControllerActivity.DRIVERTYPE_NIBIRU) { - if (checkApplication(NIBIRU_DEP_PACKAGE)) { - createControllerInstance(mLocalSavePath + mLibAdapterFileName, mTargetDriverType); - } - else { - if (mControllerListener != null) { - mControllerListener.onInstallDriver(mLocalSavePath + mMogaDepFileName); - } - installApplication(mLocalSavePath + mNibiruDepFileName); - } - } - else if (mTargetDriverType == GameControllerActivity.DRIVERTYPE_MOGA) { - if (checkApplication(MOGA__DEP_PACKAGE)) { - createControllerInstance(mLocalSavePath + mLibAdapterFileName, mTargetDriverType); - } - else { - if (mControllerListener != null) { - mControllerListener.onInstallDriver(mLocalSavePath + mMogaDepFileName); - } - installApplication(mLocalSavePath + mMogaDepFileName); - } - } - else if (mTargetDriverType == GameControllerActivity.DRIVERTYPE_OUYA) { - createControllerInstance(mLocalSavePath + mLibAdapterFileName, mTargetDriverType); - } - } - - private static final String CONFIGKEY_DRIVES = "drives"; - private static final String CONFIGKEY_LIBADAPTER_FILENAME = "adapter-file"; - private static final String CONFIGKEY_LIBADAPTER_FILESIZE = "adapter-filesize"; - private static final String CONFIGKEY_DEP_FILENAME = "dep-apk"; - private static final String CONFIGKEY_DEP_FILESIZE = "dep-size"; - - private boolean parseConfig(String jsonString){ - mMogaSupportedDrives.clear(); - mNibiruSupportedDrives.clear(); - mOuyaSupportedDrives.clear(); - - try { - int tint = 9879; - JSONObject jsonObject = new JSONObject(); - jsonObject.put("intvalue",tint); - Log.w(TAG, "intJson:" + jsonObject); - - JSONObject configObject = new JSONObject(jsonString); - mLibAdapterFileName = configObject.getString(CONFIGKEY_LIBADAPTER_FILENAME); - mLibAdapterFileSize = configObject.getInt(CONFIGKEY_LIBADAPTER_FILESIZE); - - JSONObject nibiruObject = configObject.getJSONObject("nibiru"); - JSONArray drives = nibiruObject.getJSONArray(CONFIGKEY_DRIVES); - int count = drives.length(); - for (int i = 0; i < count; i++) { - mNibiruSupportedDrives.add(drives.getString(i)); - } - mNibiruDepFileName = nibiruObject.getString(CONFIGKEY_DEP_FILENAME); - mNibiruDepFileSize = nibiruObject.getInt(CONFIGKEY_DEP_FILESIZE); - - JSONObject mogaObject = configObject.getJSONObject("moga"); - drives = mogaObject.getJSONArray(CONFIGKEY_DRIVES); - count = drives.length(); - for (int i = 0; i < count; i++) { - mMogaSupportedDrives.add(drives.getString(i)); - } - mMogaDepFileName = mogaObject.getString(CONFIGKEY_DEP_FILENAME); - mMogaDepFileSize = mogaObject.getInt(CONFIGKEY_DEP_FILESIZE); - - JSONObject ouyaObject = configObject.getJSONObject("ouya"); - drives = ouyaObject.getJSONArray(CONFIGKEY_DRIVES); - count = drives.length(); - for (int i = 0; i < count; i++) { - mOuyaSupportedDrives.add(drives.getString(i)); - } - - mLazyConfigInit = false; - return true; - } catch (Exception e1) { - e1.printStackTrace(); - return false; - } - } - - private void requestControllerConfig() { - - final JsonHttpResponseHandler configResponseHandler = new JsonHttpResponseHandler() { - - @Override - public void onFailure(int statusCode, Header[] headers, - String responseBody, Throwable e) { - - Log.e(TAG, "Failed to download game controller config!"); - - String configJSON = GameControllerUtils.readJsonFile(mConfigFilePath); - if (configJSON != null) { - if (parseConfig(configJSON)) { - scanBluetoothDrive(); - return; - } - } - if (mControllerListener != null) { - mControllerListener.onDownloadConfigFinished(false); - } - /*new AlertDialog.Builder(sCocos2dxActivity) - .setTitle("Loading controller config failed!") - .setMessage( - "Please make sure internet connection works ok!") - .setPositiveButton("Retry", - new DialogInterface.OnClickListener() { - public void onClick(DialogInterface dialog, - int which) { - dialog.dismiss(); - requestControllerConfig(); - } - }).setCancelable(false).show();*/ - } - - @Override - public void onSuccess(int statusCode, Header[] headers, - String responseBody) { - - String jsonString = responseBody.trim(); - - try { - if (parseConfig(jsonString)) { - scanBluetoothDrive(); - showToast("Get controller config succeed!"); - - File configFile = new File(mConfigFilePath); - FileOutputStream outputStream = new FileOutputStream(configFile); - byte[] contentString = jsonString.getBytes(); - outputStream.write(contentString, 0, contentString.length); - outputStream.flush(); - outputStream.close(); - return; - } - else { - String jsonStr = GameControllerUtils.readJsonFile(mConfigFilePath); - if (jsonStr != null) { - if (parseConfig(jsonStr)) { - scanBluetoothDrive(); - showToast("Get controller config succeed!"); - return; - } - } - } - } catch (Exception e1) { - e1.printStackTrace(); - } - - if (mControllerListener != null) { - mControllerListener.onDownloadConfigFinished(false); - } - } - }; - - mDownDepsHttpClient.get(COCOS_CONTROLLER_URL + COCOS_CONTROLLER_CONFIG, configResponseHandler); - } - - private static void showToast(String message){ - Toast.makeText(sGameControllerActivity, message, Toast.LENGTH_SHORT).show(); - } - - private static void createControllerInstance(String libFilePath,int driveType) { - //if (sGameControllerActivity.getGameControllerDelegate(driveType) != null) { - // return; - //} - - File libFile = new File(libFilePath); - if (! libFile.exists()) { - Log.w(TAG, libFile.toString() + "not exist!"); - return; - } - - DexClassLoader classLoader = null; - - try { - File dexOutputDir = sGameControllerActivity.getDir("dex", Context.MODE_PRIVATE); - classLoader = new DexClassLoader(libFile.getCanonicalPath(), dexOutputDir.getCanonicalPath(), - null, sGameControllerActivity.getClassLoader()); - } catch (Exception e1) { - e1.printStackTrace(); - } - - try { - Class controllerDelegate = null; - if (driveType == GameControllerActivity.DRIVERTYPE_MOGA) { - controllerDelegate = classLoader.loadClass("org.cocos2dx.lib.GameControllerMoga"); - } else if (driveType == GameControllerActivity.DRIVERTYPE_NIBIRU) { - controllerDelegate = classLoader.loadClass("org.cocos2dx.lib.GameControllerNibiru"); - } else if (driveType == GameControllerActivity.DRIVERTYPE_OUYA) { - controllerDelegate = classLoader.loadClass("org.cocos2dx.lib.GameControllerOuya"); - } - - GameControllerDelegate instance = (GameControllerDelegate)controllerDelegate.newInstance(); - - if (mControllerListener != null) { - mControllerListener.onConnectController(); - } - sGameControllerActivity.setGameControllerInstance(instance, driveType); - if (driveType == GameControllerActivity.DRIVERTYPE_NIBIRU) { - Method method = controllerDelegate.getDeclaredMethod("onResume"); - method.invoke(instance); - } - } - catch (Exception e) { - e.printStackTrace(); - } - } - - private BroadcastReceiver mAppReceiver = new BroadcastReceiver() { - - @Override - public void onReceive(Context context, Intent intent) { - String packageName = intent.getDataString(); - Log.d(TAG, "mAppReceiver:" + intent); - if (packageName.contains(NIBIRU_DEP_PACKAGE)) { - createControllerInstance(mLocalSavePath + mLibAdapterFileName, GameControllerActivity.DRIVERTYPE_NIBIRU); - } - else if (packageName.contains(MOGA__DEP_PACKAGE)) { - createControllerInstance(mLocalSavePath + mLibAdapterFileName, GameControllerActivity.DRIVERTYPE_MOGA); - } - } - }; - - private float mOldLeftThumbstickX = 0.0f; - private float mOldLeftThumbstickY = 0.0f; - private float mOldRightThumbstickX = 0.0f; - private float mOldRightThumbstickY = 0.0f; - - private float mOldLeftTrigger = 0.0f; - private float mOldRightTrigger = 0.0f; - private float mOldThrottle = 0.0f; - private float mOldBrake = 0.0f; - - public boolean dispatchGenericMotionEvent(MotionEvent event) { - boolean handled = false; - - int eventSource = event.getSource(); - - if ( ((eventSource & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD) - || ((eventSource & InputDevice.SOURCE_JOYSTICK) == InputDevice.SOURCE_JOYSTICK) ) - { - if (event.getAction() == MotionEvent.ACTION_MOVE) { - int devicedId = event.getDeviceId(); - - float newAXIS_LX = event.getAxisValue(AXIS_X); - if (Float.compare(newAXIS_LX , mOldLeftThumbstickX) != 0) { - GameControllerAdapter.onAxisEvent(StandardControllerName, devicedId, GameControllerDelegate.THUMBSTICK_LEFT_X, newAXIS_LX, true); - mOldLeftThumbstickX = newAXIS_LX; - handled = true; - } - - float newAXIS_LY = event.getAxisValue(AXIS_Y); - if (Float.compare(newAXIS_LY , mOldLeftThumbstickY) != 0) { - GameControllerAdapter.onAxisEvent(StandardControllerName, devicedId, GameControllerDelegate.THUMBSTICK_LEFT_Y, newAXIS_LY, true); - mOldLeftThumbstickY = newAXIS_LY; - handled = true; - } - - float newAXIS_RX = event.getAxisValue(AXIS_Z); - if (Float.compare(newAXIS_RX , mOldRightThumbstickX) != 0) { - GameControllerAdapter.onAxisEvent(StandardControllerName, devicedId, GameControllerDelegate.THUMBSTICK_RIGHT_X, newAXIS_RX, true); - mOldRightThumbstickX = newAXIS_RX; - handled = true; - } - - float newAXIS_RY = event.getAxisValue(AXIS_RZ); - if (Float.compare(newAXIS_RY , mOldRightThumbstickY) != 0) { - GameControllerAdapter.onAxisEvent(StandardControllerName, devicedId, GameControllerDelegate.THUMBSTICK_RIGHT_Y, newAXIS_RY, true); - mOldRightThumbstickY = newAXIS_RY; - handled = true; - } - - float newAXIS_LTRIGGER = event.getAxisValue(AXIS_LTRIGGER); - if (Float.compare(newAXIS_LTRIGGER , mOldLeftTrigger) != 0) { - if (Float.compare(newAXIS_LTRIGGER, 0.0f) == 0) { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_LEFT_TRIGGER, false, 0.0f, true); - }else { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_LEFT_TRIGGER, true, newAXIS_LTRIGGER, true); - } - mOldLeftTrigger = newAXIS_LTRIGGER; - handled = true; - } - - float newAXIS_RTRIGGER = event.getAxisValue(AXIS_RTRIGGER); - if (Float.compare(newAXIS_RTRIGGER , mOldRightTrigger) != 0) { - if (Float.compare(newAXIS_RTRIGGER, 0.0f) == 0) { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, false, 0.0f, true); - }else { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, true, newAXIS_RTRIGGER, true); - } - mOldRightTrigger = newAXIS_RTRIGGER; - handled = true; - } - - float newAXIS_BRAKE = event.getAxisValue(AXIS_BRAKE); - if (Float.compare(newAXIS_BRAKE , mOldBrake) != 0) { - if (Float.compare(newAXIS_BRAKE, 0.0f) == 0) { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_LEFT_TRIGGER, false, 0.0f, true); - }else { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_LEFT_TRIGGER, true, newAXIS_BRAKE, true); - } - mOldBrake = newAXIS_BRAKE; - handled = true; - } - - float newAXIS_THROTTLE = event.getAxisValue(AXIS_THROTTLE); - if (Float.compare(newAXIS_THROTTLE , mOldThrottle) != 0) { - if (Float.compare(newAXIS_THROTTLE, 0.0f) == 0) { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, false, 0.0f, true); - }else { - GameControllerAdapter.onButtonEvent(StandardControllerName, devicedId, GameControllerDelegate.BUTTON_RIGHT_TRIGGER, true, newAXIS_THROTTLE, true); - } - mOldThrottle = newAXIS_THROTTLE; - handled = true; - } - } - } - - return handled; - } - - public boolean dispatchKeyEvent(KeyEvent event) { - boolean handled = false; - - int eventSource = event.getSource(); - int controllerKey = ControllerKeyMap.get(event.getKeyCode()); - - if (controllerKey != 0 && (((eventSource & InputDevice.SOURCE_GAMEPAD) == InputDevice.SOURCE_GAMEPAD) - || ((eventSource & InputDevice.SOURCE_JOYSTICK) == InputDevice.SOURCE_JOYSTICK) - || ((eventSource & InputDevice.SOURCE_DPAD) == InputDevice.SOURCE_DPAD))) - { - int action = event.getAction(); - if (action == KeyEvent.ACTION_DOWN) { - handled = true; - GameControllerAdapter.onButtonEvent(StandardControllerName,event.getDeviceId(), controllerKey,true, 1.0f, false); - }else if (action == KeyEvent.ACTION_UP) { - handled = true; - GameControllerAdapter.onButtonEvent(StandardControllerName,event.getDeviceId(), controllerKey,false, 0.0f, false); - } - } - - return handled; - } -} diff --git a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerCompat.java b/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerCompat.java deleted file mode 100644 index 65b919bdae..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerCompat.java +++ /dev/null @@ -1,140 +0,0 @@ -/* - * Copyright (C) 2013 The Android Open Source Project - * - * 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 - * - * http://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. - */ - -package org.cocos2dx.lib.inputmanagercompat; - -import android.content.Context; -import android.os.Build; -import android.os.Handler; -import android.view.InputDevice; -import android.view.MotionEvent; - -public interface InputManagerCompat { - /** - * Gets information about the input device with the specified id. - * - * @param id The device id - * @return The input device or null if not found - */ - public InputDevice getInputDevice(int id); - - /** - * Gets the ids of all input devices in the system. - * - * @return The input device ids. - */ - public int[] getInputDeviceIds(); - - /** - * Registers an input device listener to receive notifications about when - * input devices are added, removed or changed. - * - * @param listener The listener to register. - * @param handler The handler on which the listener should be invoked, or - * null if the listener should be invoked on the calling thread's - * looper. - */ - public void registerInputDeviceListener(InputManagerCompat.InputDeviceListener listener, - Handler handler); - - /** - * Unregisters an input device listener. - * - * @param listener The listener to unregister. - */ - public void unregisterInputDeviceListener(InputManagerCompat.InputDeviceListener listener); - - /* - * The following three calls are to simulate V16 behavior on pre-Jellybean - * devices. If you don't call them, your callback will never be called - * pre-API 16. - */ - - /** - * Pass the motion events to the InputManagerCompat. This is used to - * optimize for polling for controllers. If you do not pass these events in, - * polling will cause regular object creation. - * - * @param event the motion event from the app - */ - public void onGenericMotionEvent(MotionEvent event); - - /** - * Tell the V9 input manager that it should stop polling for disconnected - * devices. You can call this during onPause in your activity, although you - * might want to call it whenever your game is not active (or whenever you - * don't care about being notified of new input devices) - */ - public void onPause(); - - /** - * Tell the V9 input manager that it should start polling for disconnected - * devices. You can call this during onResume in your activity, although you - * might want to call it less often (only when the gameplay is actually - * active) - */ - public void onResume(); - - public interface InputDeviceListener { - /** - * Called whenever the input manager detects that a device has been - * added. This will only be called in the V9 version when a motion event - * is detected. - * - * @param deviceId The id of the input device that was added. - */ - void onInputDeviceAdded(int deviceId); - - /** - * Called whenever the properties of an input device have changed since - * they were last queried. This will not be called for the V9 version of - * the API. - * - * @param deviceId The id of the input device that changed. - */ - void onInputDeviceChanged(int deviceId); - - /** - * Called whenever the input manager detects that a device has been - * removed. For the V9 version, this can take some time depending on the - * poll rate. - * - * @param deviceId The id of the input device that was removed. - */ - void onInputDeviceRemoved(int deviceId); - } - - /** - * Use this to construct a compatible InputManager. - */ - public static class Factory { - - /** - * Constructs and returns a compatible InputManger - * - * @param context the Context that will be used to get the system - * service from - * @return a compatible implementation of InputManager - */ - public static InputManagerCompat getInputManager(Context context) { - if (Build.VERSION.SDK_INT >= Build.VERSION_CODES.JELLY_BEAN) { - return new InputManagerV16(context); - } else { - return new InputManagerV9(); - } - } - } -} diff --git a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV16.java b/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV16.java deleted file mode 100644 index f0d265adb0..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV16.java +++ /dev/null @@ -1,107 +0,0 @@ -/* - * Copyright (C) 2013 The Android Open Source Project - * - * 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 - * - * http://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. - */ - -package org.cocos2dx.lib.inputmanagercompat; - -import android.annotation.TargetApi; -import android.content.Context; -import android.hardware.input.InputManager; -import android.os.Build; -import android.os.Handler; -import android.view.InputDevice; -import android.view.MotionEvent; - -import java.util.HashMap; -import java.util.Map; - -@TargetApi(Build.VERSION_CODES.JELLY_BEAN) -public class InputManagerV16 implements InputManagerCompat { - - private final InputManager mInputManager; - private final Map mListeners; - - public InputManagerV16(Context context) { - mInputManager = (InputManager) context.getSystemService(Context.INPUT_SERVICE); - mListeners = new HashMap(); - } - - @Override - public InputDevice getInputDevice(int id) { - return mInputManager.getInputDevice(id); - } - - @Override - public int[] getInputDeviceIds() { - return mInputManager.getInputDeviceIds(); - } - - static class V16InputDeviceListener implements InputManager.InputDeviceListener { - final InputManagerCompat.InputDeviceListener mIDL; - - public V16InputDeviceListener(InputDeviceListener idl) { - mIDL = idl; - } - - @Override - public void onInputDeviceAdded(int deviceId) { - mIDL.onInputDeviceAdded(deviceId); - } - - @Override - public void onInputDeviceChanged(int deviceId) { - mIDL.onInputDeviceChanged(deviceId); - } - - @Override - public void onInputDeviceRemoved(int deviceId) { - mIDL.onInputDeviceRemoved(deviceId); - } - - } - - @Override - public void registerInputDeviceListener(InputDeviceListener listener, Handler handler) { - V16InputDeviceListener v16Listener = new V16InputDeviceListener(listener); - mInputManager.registerInputDeviceListener(v16Listener, handler); - mListeners.put(listener, v16Listener); - } - - @Override - public void unregisterInputDeviceListener(InputDeviceListener listener) { - V16InputDeviceListener curListener = mListeners.remove(listener); - if (null != curListener) - { - mInputManager.unregisterInputDeviceListener(curListener); - } - - } - - @Override - public void onGenericMotionEvent(MotionEvent event) { - // unused in V16 - } - - @Override - public void onPause() { - // unused in V16 - } - - @Override - public void onResume() { - // unused in V16 - } - -} diff --git a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV9.java b/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV9.java deleted file mode 100644 index abb47a8719..0000000000 --- a/cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV9.java +++ /dev/null @@ -1,211 +0,0 @@ -/* - * Copyright (C) 2013 The Android Open Source Project - * - * 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 - * - * http://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. - */ - -package org.cocos2dx.lib.inputmanagercompat; - -import android.os.Handler; -import android.os.Message; -import android.os.SystemClock; -import android.util.Log; -import android.util.SparseArray; -import android.view.InputDevice; -import android.view.MotionEvent; - -import java.lang.ref.WeakReference; -import java.util.ArrayDeque; -import java.util.HashMap; -import java.util.Map; -import java.util.Queue; - -public class InputManagerV9 implements InputManagerCompat { - private static final String LOG_TAG = "InputManagerV9"; - private static final int MESSAGE_TEST_FOR_DISCONNECT = 101; - private static final long CHECK_ELAPSED_TIME = 3000L; - - private static final int ON_DEVICE_ADDED = 0; - private static final int ON_DEVICE_CHANGED = 1; - private static final int ON_DEVICE_REMOVED = 2; - - private final SparseArray mDevices; - private final Map mListeners; - private final Handler mDefaultHandler; - - private static class PollingMessageHandler extends Handler { - private final WeakReference mInputManager; - - PollingMessageHandler(InputManagerV9 im) { - mInputManager = new WeakReference(im); - } - - @Override - public void handleMessage(Message msg) { - super.handleMessage(msg); - switch (msg.what) { - case MESSAGE_TEST_FOR_DISCONNECT: - InputManagerV9 imv = mInputManager.get(); - if (null != imv) { - long time = SystemClock.elapsedRealtime(); - int size = imv.mDevices.size(); - for (int i = 0; i < size; i++) { - long[] lastContact = imv.mDevices.valueAt(i); - if (null != lastContact) { - if (time - lastContact[0] > CHECK_ELAPSED_TIME) { - // check to see if the device has been - // disconnected - int id = imv.mDevices.keyAt(i); - if (null == InputDevice.getDevice(id)) { - // disconnected! - imv.notifyListeners(ON_DEVICE_REMOVED, id); - imv.mDevices.remove(id); - } else { - lastContact[0] = time; - } - } - } - } - sendEmptyMessageDelayed(MESSAGE_TEST_FOR_DISCONNECT, - CHECK_ELAPSED_TIME); - } - break; - } - } - - } - - public InputManagerV9() { - mDevices = new SparseArray(); - mListeners = new HashMap(); - mDefaultHandler = new PollingMessageHandler(this); - // as a side-effect, populates our collection of watched - // input devices - getInputDeviceIds(); - } - - @Override - public InputDevice getInputDevice(int id) { - return InputDevice.getDevice(id); - } - - @Override - public int[] getInputDeviceIds() { - // add any hitherto unknown devices to our - // collection of watched input devices - int[] activeDevices = InputDevice.getDeviceIds(); - long time = SystemClock.elapsedRealtime(); - for ( int id : activeDevices ) { - long[] lastContact = mDevices.get(id); - if ( null == lastContact ) { - // we have a new device - mDevices.put(id, new long[] { time }); - } - } - return activeDevices; - } - - @Override - public void registerInputDeviceListener(InputDeviceListener listener, Handler handler) { - mListeners.remove(listener); - if (handler == null) { - handler = mDefaultHandler; - } - mListeners.put(listener, handler); - } - - @Override - public void unregisterInputDeviceListener(InputDeviceListener listener) { - mListeners.remove(listener); - } - - private void notifyListeners(int why, int deviceId) { - // the state of some device has changed - if (!mListeners.isEmpty()) { - // yes... this will cause an object to get created... hopefully - // it won't happen very often - for (InputDeviceListener listener : mListeners.keySet()) { - Handler handler = mListeners.get(listener); - DeviceEvent odc = DeviceEvent.getDeviceEvent(why, deviceId, listener); - handler.post(odc); - } - } - } - - private static class DeviceEvent implements Runnable { - private int mMessageType; - private int mId; - private InputDeviceListener mListener; - private static Queue sEventQueue = new ArrayDeque(); - - private DeviceEvent() { - } - - static DeviceEvent getDeviceEvent(int messageType, int id, - InputDeviceListener listener) { - DeviceEvent curChanged = sEventQueue.poll(); - if (null == curChanged) { - curChanged = new DeviceEvent(); - } - curChanged.mMessageType = messageType; - curChanged.mId = id; - curChanged.mListener = listener; - return curChanged; - } - - @Override - public void run() { - switch (mMessageType) { - case ON_DEVICE_ADDED: - mListener.onInputDeviceAdded(mId); - break; - case ON_DEVICE_CHANGED: - mListener.onInputDeviceChanged(mId); - break; - case ON_DEVICE_REMOVED: - mListener.onInputDeviceRemoved(mId); - break; - default: - Log.e(LOG_TAG, "Unknown Message Type"); - break; - } - // dump this runnable back in the queue - sEventQueue.offer(this); - } - } - - @Override - public void onGenericMotionEvent(MotionEvent event) { - // detect new devices - int id = event.getDeviceId(); - long[] timeArray = mDevices.get(id); - if (null == timeArray) { - notifyListeners(ON_DEVICE_ADDED, id); - timeArray = new long[1]; - mDevices.put(id, timeArray); - } - long time = SystemClock.elapsedRealtime(); - timeArray[0] = time; - } - - @Override - public void onPause() { - mDefaultHandler.removeMessages(MESSAGE_TEST_FOR_DISCONNECT); - } - - @Override - public void onResume() { - mDefaultHandler.sendEmptyMessage(MESSAGE_TEST_FOR_DISCONNECT); - } - -} From e1bb1158bc4fc627d1f1f4d1eabc44481ce83df4 Mon Sep 17 00:00:00 2001 From: Dhilan007 Date: Wed, 2 Jul 2014 15:25:18 +0800 Subject: [PATCH 5/6] update test case. --- .../src/org/cocos2dx/game_controller_test/AppActivity.java | 5 ----- 1 file changed, 5 deletions(-) diff --git a/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java b/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java index 17096cc304..c4a4b68f7d 100644 --- a/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java +++ b/tests/game-controller-test/proj.android/src/org/cocos2dx/game_controller_test/AppActivity.java @@ -36,12 +36,7 @@ public class AppActivity extends GameControllerActivity { //The standard controller,without doing anything special. e.g: Amazon Fire TV - //Automatic adaptation for connect controller. - //Supported Platform: Nibiru / Moga / Ouya TV - //this.connectController(); - //Manually specify an adapter. - //Requirements: using libControllerDriveAdapter project //this.connectController(DRIVERTYPE_NIBIRU); //this.connectController(DRIVERTYPE_MOGA); //this.connectController(DRIVERTYPE_OUYA); From 0b0b9ccdf3a8bb7a7a1645cf407bca8fab8a2ba6 Mon Sep 17 00:00:00 2001 From: CocosRobot Date: Wed, 2 Jul 2014 08:21:52 +0000 Subject: [PATCH 6/6] [AUTO][ci skip]: updating cocos2dx_files.json --- templates/cocos2dx_files.json | 14 -------------- 1 file changed, 14 deletions(-) diff --git a/templates/cocos2dx_files.json b/templates/cocos2dx_files.json index f130130767..8133421da1 100644 --- a/templates/cocos2dx_files.json +++ b/templates/cocos2dx_files.json @@ -747,20 +747,6 @@ "cocos/platform/android/CCGLView.h", "cocos/platform/android/CCPlatformDefine.h", "cocos/platform/android/CCStdC.h", - "cocos/platform/android/ControllerAutoAdapter/.classpath", - "cocos/platform/android/ControllerAutoAdapter/.project", - "cocos/platform/android/ControllerAutoAdapter/.settings/org.eclipse.jdt.core.prefs", - "cocos/platform/android/ControllerAutoAdapter/AndroidManifest.xml", - "cocos/platform/android/ControllerAutoAdapter/ant.properties", - "cocos/platform/android/ControllerAutoAdapter/build.xml", - "cocos/platform/android/ControllerAutoAdapter/lint.xml", - "cocos/platform/android/ControllerAutoAdapter/proguard-project.txt", - "cocos/platform/android/ControllerAutoAdapter/project.properties", - "cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerActivity.java", - "cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/GameControllerHelper.java", - "cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerCompat.java", - "cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV16.java", - "cocos/platform/android/ControllerAutoAdapter/src/org/cocos2dx/lib/inputmanagercompat/InputManagerV9.java", "cocos/platform/android/ControllerManualAdapter/.classpath", "cocos/platform/android/ControllerManualAdapter/.project", "cocos/platform/android/ControllerManualAdapter/.settings/org.eclipse.jdt.core.prefs",