├── app
├── .gitignore
├── src
│ ├── main
│ │ ├── res
│ │ │ ├── values
│ │ │ │ ├── strings.xml
│ │ │ │ ├── colors.xml
│ │ │ │ └── themes.xml
│ │ │ ├── mipmap-hdpi
│ │ │ │ ├── ic_launcher.webp
│ │ │ │ └── ic_launcher_round.webp
│ │ │ ├── mipmap-mdpi
│ │ │ │ ├── ic_launcher.webp
│ │ │ │ └── ic_launcher_round.webp
│ │ │ ├── mipmap-xhdpi
│ │ │ │ ├── ic_launcher.webp
│ │ │ │ └── ic_launcher_round.webp
│ │ │ ├── mipmap-xxhdpi
│ │ │ │ ├── ic_launcher.webp
│ │ │ │ └── ic_launcher_round.webp
│ │ │ ├── mipmap-xxxhdpi
│ │ │ │ ├── ic_launcher.webp
│ │ │ │ └── ic_launcher_round.webp
│ │ │ ├── mipmap-anydpi-v26
│ │ │ │ ├── ic_launcher.xml
│ │ │ │ └── ic_launcher_round.xml
│ │ │ ├── xml
│ │ │ │ ├── backup_rules.xml
│ │ │ │ └── data_extraction_rules.xml
│ │ │ ├── values-night
│ │ │ │ └── themes.xml
│ │ │ ├── drawable-v24
│ │ │ │ └── ic_launcher_foreground.xml
│ │ │ ├── layout
│ │ │ │ └── activity_main.xml
│ │ │ └── drawable
│ │ │ │ └── ic_launcher_background.xml
│ │ ├── AndroidManifest.xml
│ │ └── java
│ │ │ └── com
│ │ │ └── sarmale
│ │ │ └── arduinobtexample_v3
│ │ │ ├── ConnectThread.java
│ │ │ ├── ConnectedThread.java
│ │ │ └── MainActivity.java
│ ├── test
│ │ └── java
│ │ │ └── com
│ │ │ └── sarmale
│ │ │ └── arduinobtexample_v3
│ │ │ └── ExampleUnitTest.java
│ └── androidTest
│ │ └── java
│ │ └── com
│ │ └── sarmale
│ │ └── arduinobtexample_v3
│ │ └── ExampleInstrumentedTest.java
├── proguard-rules.pro
└── build.gradle
├── .gitattributes
├── gradle
└── wrapper
│ ├── gradle-wrapper.jar
│ └── gradle-wrapper.properties
├── settings.gradle
├── .gitignore
├── gradle.properties
├── gradlew.bat
└── gradlew
/app/.gitignore:
--------------------------------------------------------------------------------
1 | /build
--------------------------------------------------------------------------------
/.gitattributes:
--------------------------------------------------------------------------------
1 | # Auto detect text files and perform LF normalization
2 | * text=auto
3 |
--------------------------------------------------------------------------------
/app/src/main/res/values/strings.xml:
--------------------------------------------------------------------------------
1 |
2 | ArduinoBTExample-v3
3 |
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.jar:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/gradle/wrapper/gradle-wrapper.jar
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-hdpi/ic_launcher_round.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-mdpi/ic_launcher_round.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xhdpi/ic_launcher_round.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xxhdpi/ic_launcher_round.webp
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp:
--------------------------------------------------------------------------------
https://raw.githubusercontent.com/The-Frugal-Engineer/ArduinoBTExamplev3/HEAD/app/src/main/res/mipmap-xxxhdpi/ic_launcher_round.webp
--------------------------------------------------------------------------------
/gradle/wrapper/gradle-wrapper.properties:
--------------------------------------------------------------------------------
1 | #Sun Nov 20 14:39:50 GMT 2022
2 | distributionBase=GRADLE_USER_HOME
3 | distributionUrl=https\://services.gradle.org/distributions/gradle-7.3.3-bin.zip
4 | distributionPath=wrapper/dists
5 | zipStorePath=wrapper/dists
6 | zipStoreBase=GRADLE_USER_HOME
7 |
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-anydpi-v26/ic_launcher.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/app/src/main/res/mipmap-anydpi-v26/ic_launcher_round.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
4 |
5 |
--------------------------------------------------------------------------------
/settings.gradle:
--------------------------------------------------------------------------------
1 | pluginManagement {
2 | repositories {
3 | gradlePluginPortal()
4 | google()
5 | mavenCentral()
6 | }
7 | }
8 | dependencyResolutionManagement {
9 | repositoriesMode.set(RepositoriesMode.FAIL_ON_PROJECT_REPOS)
10 | repositories {
11 | google()
12 | mavenCentral()
13 | }
14 | }
15 | rootProject.name = "ArduinoBTExample-v3"
16 | include ':app'
17 |
--------------------------------------------------------------------------------
/app/src/main/res/values/colors.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 | #FFBB86FC
4 | #FF6200EE
5 | #FF3700B3
6 | #FF03DAC5
7 | #FF018786
8 | #FF000000
9 | #FFFFFFFF
10 |
--------------------------------------------------------------------------------
/.gitignore:
--------------------------------------------------------------------------------
1 | # Gradle files
2 | .gradle/
3 | build/
4 |
5 | # Local configuration file (sdk path, etc)
6 | local.properties
7 |
8 | # Log/OS Files
9 | *.log
10 |
11 | # Android Studio generated files and folders
12 | captures/
13 | .externalNativeBuild/
14 | .cxx/
15 | *.apk
16 | output.json
17 |
18 | # IntelliJ
19 | *.iml
20 | .idea/
21 |
22 | # Keystore files
23 | *.jks
24 | *.keystore
25 |
26 | # Google Services (e.g. APIs or Firebase)
27 | google-services.json
28 |
29 | # Android Profiling
30 | *.hprof
31 |
--------------------------------------------------------------------------------
/app/src/test/java/com/sarmale/arduinobtexample_v3/ExampleUnitTest.java:
--------------------------------------------------------------------------------
1 | package com.sarmale.arduinobtexample_v3;
2 |
3 | import org.junit.Test;
4 |
5 | import static org.junit.Assert.*;
6 |
7 | /**
8 | * Example local unit test, which will execute on the development machine (host).
9 | *
10 | * @see Testing documentation
11 | */
12 | public class ExampleUnitTest {
13 | @Test
14 | public void addition_isCorrect() {
15 | assertEquals(4, 2 + 2);
16 | }
17 | }
--------------------------------------------------------------------------------
/app/src/main/res/xml/backup_rules.xml:
--------------------------------------------------------------------------------
1 |
8 |
9 |
13 |
--------------------------------------------------------------------------------
/app/src/main/res/xml/data_extraction_rules.xml:
--------------------------------------------------------------------------------
1 |
6 |
7 |
8 |
12 |
13 |
19 |
--------------------------------------------------------------------------------
/app/proguard-rules.pro:
--------------------------------------------------------------------------------
1 | # Add project specific ProGuard rules here.
2 | # You can control the set of applied configuration files using the
3 | # proguardFiles setting in build.gradle.
4 | #
5 | # For more details, see
6 | # http://developer.android.com/guide/developing/tools/proguard.html
7 |
8 | # If your project uses WebView with JS, uncomment the following
9 | # and specify the fully qualified class name to the JavaScript interface
10 | # class:
11 | #-keepclassmembers class fqcn.of.javascript.interface.for.webview {
12 | # public *;
13 | #}
14 |
15 | # Uncomment this to preserve the line number information for
16 | # debugging stack traces.
17 | #-keepattributes SourceFile,LineNumberTable
18 |
19 | # If you keep the line number information, uncomment this to
20 | # hide the original source file name.
21 | #-renamesourcefileattribute SourceFile
--------------------------------------------------------------------------------
/app/src/main/res/values/themes.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
16 |
--------------------------------------------------------------------------------
/app/src/main/res/values-night/themes.xml:
--------------------------------------------------------------------------------
1 |
2 |
3 |
16 |
--------------------------------------------------------------------------------
/app/src/androidTest/java/com/sarmale/arduinobtexample_v3/ExampleInstrumentedTest.java:
--------------------------------------------------------------------------------
1 | package com.sarmale.arduinobtexample_v3;
2 |
3 | import android.content.Context;
4 |
5 | import androidx.test.platform.app.InstrumentationRegistry;
6 | import androidx.test.ext.junit.runners.AndroidJUnit4;
7 |
8 | import org.junit.Test;
9 | import org.junit.runner.RunWith;
10 |
11 | import static org.junit.Assert.*;
12 |
13 | /**
14 | * Instrumented test, which will execute on an Android device.
15 | *
16 | * @see Testing documentation
17 | */
18 | @RunWith(AndroidJUnit4.class)
19 | public class ExampleInstrumentedTest {
20 | @Test
21 | public void useAppContext() {
22 | // Context of the app under test.
23 | Context appContext = InstrumentationRegistry.getInstrumentation().getTargetContext();
24 | assertEquals("com.sarmale.arduinobtexample_v3", appContext.getPackageName());
25 | }
26 | }
--------------------------------------------------------------------------------
/app/build.gradle:
--------------------------------------------------------------------------------
1 | plugins {
2 | id 'com.android.application'
3 | }
4 |
5 | android {
6 | compileSdk 32
7 |
8 | defaultConfig {
9 | applicationId "com.sarmale.arduinobtexample_v3"
10 | minSdk 21
11 | targetSdk 32
12 | versionCode 1
13 | versionName "1.0"
14 |
15 | testInstrumentationRunner "androidx.test.runner.AndroidJUnitRunner"
16 | }
17 |
18 | buildTypes {
19 | release {
20 | minifyEnabled false
21 | proguardFiles getDefaultProguardFile('proguard-android-optimize.txt'), 'proguard-rules.pro'
22 | }
23 | }
24 | compileOptions {
25 | sourceCompatibility JavaVersion.VERSION_1_8
26 | targetCompatibility JavaVersion.VERSION_1_8
27 | }
28 | }
29 |
30 | dependencies {
31 |
32 | implementation 'androidx.appcompat:appcompat:1.5.1'
33 | implementation 'com.google.android.material:material:1.7.0'
34 | implementation 'androidx.constraintlayout:constraintlayout:2.1.4'
35 | testImplementation 'junit:junit:4.13.2'
36 | androidTestImplementation 'androidx.test.ext:junit:1.1.4'
37 | androidTestImplementation 'androidx.test.espresso:espresso-core:3.5.0'
38 | implementation 'io.reactivex.rxjava2:rxandroid:2.0.1'
39 | implementation 'io.reactivex.rxjava2:rxjava:2.0.8'
40 | }
--------------------------------------------------------------------------------
/gradle.properties:
--------------------------------------------------------------------------------
1 | # Project-wide Gradle settings.
2 | # IDE (e.g. Android Studio) users:
3 | # Gradle settings configured through the IDE *will override*
4 | # any settings specified in this file.
5 | # For more details on how to configure your build environment visit
6 | # http://www.gradle.org/docs/current/userguide/build_environment.html
7 | # Specifies the JVM arguments used for the daemon process.
8 | # The setting is particularly useful for tweaking memory settings.
9 | org.gradle.jvmargs=-Xmx2048m -Dfile.encoding=UTF-8
10 | # When configured, Gradle will run in incubating parallel mode.
11 | # This option should only be used with decoupled projects. More details, visit
12 | # http://www.gradle.org/docs/current/userguide/multi_project_builds.html#sec:decoupled_projects
13 | # org.gradle.parallel=true
14 | # AndroidX package structure to make it clearer which packages are bundled with the
15 | # Android operating system, and which are packaged with your app"s APK
16 | # https://developer.android.com/topic/libraries/support-library/androidx-rn
17 | android.useAndroidX=true
18 | # Enables namespacing of each library's R class so that its R class includes only the
19 | # resources declared in the library itself and none from the library's dependencies,
20 | # thereby reducing the size of the R class for that library
21 | android.nonTransitiveRClass=true
--------------------------------------------------------------------------------
/app/src/main/res/drawable-v24/ic_launcher_foreground.xml:
--------------------------------------------------------------------------------
1 |
7 |
8 |
9 |
15 |
18 |
21 |
22 |
23 |
24 |
30 |
--------------------------------------------------------------------------------
/app/src/main/AndroidManifest.xml:
--------------------------------------------------------------------------------
1 |
2 |
5 |
6 |
7 |
9 |
11 |
12 |
16 |
17 |
18 |
20 |
21 |
22 |
24 |
25 |
26 |
36 |
39 |
40 |
41 |
42 |
43 |
44 |
45 |
46 |
47 |
--------------------------------------------------------------------------------
/app/src/main/java/com/sarmale/arduinobtexample_v3/ConnectThread.java:
--------------------------------------------------------------------------------
1 | package com.sarmale.arduinobtexample_v3;
2 |
3 | import android.annotation.SuppressLint;
4 | import android.bluetooth.BluetoothDevice;
5 | import android.bluetooth.BluetoothSocket;
6 | import android.os.Handler;
7 | import android.util.Log;
8 |
9 | import java.io.IOException;
10 | import java.util.UUID;
11 |
12 |
13 | //Class that will open the BT Socket to the Arduino BT Module
14 | //Given a BT device, the UUID and a Handler to set the results
15 | public class ConnectThread extends Thread {
16 | private final BluetoothSocket mmSocket;
17 | private static final String TAG = "FrugalLogs";
18 | public static Handler handler;
19 | private final static int ERROR_READ = 0;
20 |
21 | @SuppressLint("MissingPermission")
22 | public ConnectThread(BluetoothDevice device, UUID MY_UUID, Handler handler) {
23 | // Use a temporary object that is later assigned to mmSocket
24 | // because mmSocket is final.
25 | BluetoothSocket tmp = null;
26 | this.handler=handler;
27 |
28 | try {
29 | // Get a BluetoothSocket to connect with the given BluetoothDevice.
30 | // MY_UUID is the app's UUID string, also used in the server code.
31 | tmp = device.createRfcommSocketToServiceRecord(MY_UUID);
32 | } catch (IOException e) {
33 | Log.e(TAG, "Socket's create() method failed", e);
34 | }
35 | mmSocket = tmp;
36 | }
37 |
38 | @SuppressLint("MissingPermission")
39 | public void run() {
40 |
41 | try {
42 | // Connect to the remote device through the socket. This call blocks
43 | // until it succeeds or throws an exception.
44 | mmSocket.connect();
45 | } catch (IOException connectException) {
46 | // Unable to connect; close the socket and return.
47 | handler.obtainMessage(ERROR_READ, "Unable to connect to the BT device").sendToTarget();
48 | Log.e(TAG, "connectException: " + connectException);
49 | try {
50 | mmSocket.close();
51 | } catch (IOException closeException) {
52 | Log.e(TAG, "Could not close the client socket", closeException);
53 | }
54 | return;
55 | }
56 |
57 | // The connection attempt succeeded. Perform work associated with
58 | // the connection in a separate thread.
59 | //manageMyConnectedSocket(mmSocket);
60 | }
61 |
62 | // Closes the client socket and causes the thread to finish.
63 | public void cancel() {
64 | try {
65 | mmSocket.close();
66 | } catch (IOException e) {
67 | Log.e(TAG, "Could not close the client socket", e);
68 | }
69 | }
70 |
71 | public BluetoothSocket getMmSocket(){
72 | return mmSocket;
73 | }
74 | }
75 |
76 |
--------------------------------------------------------------------------------
/gradlew.bat:
--------------------------------------------------------------------------------
1 | @rem
2 | @rem Copyright 2015 the original author or authors.
3 | @rem
4 | @rem Licensed under the Apache License, Version 2.0 (the "License");
5 | @rem you may not use this file except in compliance with the License.
6 | @rem You may obtain a copy of the License at
7 | @rem
8 | @rem https://www.apache.org/licenses/LICENSE-2.0
9 | @rem
10 | @rem Unless required by applicable law or agreed to in writing, software
11 | @rem distributed under the License is distributed on an "AS IS" BASIS,
12 | @rem WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 | @rem See the License for the specific language governing permissions and
14 | @rem limitations under the License.
15 | @rem
16 |
17 | @if "%DEBUG%" == "" @echo off
18 | @rem ##########################################################################
19 | @rem
20 | @rem Gradle startup script for Windows
21 | @rem
22 | @rem ##########################################################################
23 |
24 | @rem Set local scope for the variables with windows NT shell
25 | if "%OS%"=="Windows_NT" setlocal
26 |
27 | set DIRNAME=%~dp0
28 | if "%DIRNAME%" == "" set DIRNAME=.
29 | set APP_BASE_NAME=%~n0
30 | set APP_HOME=%DIRNAME%
31 |
32 | @rem Resolve any "." and ".." in APP_HOME to make it shorter.
33 | for %%i in ("%APP_HOME%") do set APP_HOME=%%~fi
34 |
35 | @rem Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
36 | set DEFAULT_JVM_OPTS="-Xmx64m" "-Xms64m"
37 |
38 | @rem Find java.exe
39 | if defined JAVA_HOME goto findJavaFromJavaHome
40 |
41 | set JAVA_EXE=java.exe
42 | %JAVA_EXE% -version >NUL 2>&1
43 | if "%ERRORLEVEL%" == "0" goto execute
44 |
45 | echo.
46 | echo ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
47 | echo.
48 | echo Please set the JAVA_HOME variable in your environment to match the
49 | echo location of your Java installation.
50 |
51 | goto fail
52 |
53 | :findJavaFromJavaHome
54 | set JAVA_HOME=%JAVA_HOME:"=%
55 | set JAVA_EXE=%JAVA_HOME%/bin/java.exe
56 |
57 | if exist "%JAVA_EXE%" goto execute
58 |
59 | echo.
60 | echo ERROR: JAVA_HOME is set to an invalid directory: %JAVA_HOME%
61 | echo.
62 | echo Please set the JAVA_HOME variable in your environment to match the
63 | echo location of your Java installation.
64 |
65 | goto fail
66 |
67 | :execute
68 | @rem Setup the command line
69 |
70 | set CLASSPATH=%APP_HOME%\gradle\wrapper\gradle-wrapper.jar
71 |
72 |
73 | @rem Execute Gradle
74 | "%JAVA_EXE%" %DEFAULT_JVM_OPTS% %JAVA_OPTS% %GRADLE_OPTS% "-Dorg.gradle.appname=%APP_BASE_NAME%" -classpath "%CLASSPATH%" org.gradle.wrapper.GradleWrapperMain %*
75 |
76 | :end
77 | @rem End local scope for the variables with windows NT shell
78 | if "%ERRORLEVEL%"=="0" goto mainEnd
79 |
80 | :fail
81 | rem Set variable GRADLE_EXIT_CONSOLE if you need the _script_ return code instead of
82 | rem the _cmd.exe /c_ return code!
83 | if not "" == "%GRADLE_EXIT_CONSOLE%" exit 1
84 | exit /b 1
85 |
86 | :mainEnd
87 | if "%OS%"=="Windows_NT" endlocal
88 |
89 | :omega
90 |
--------------------------------------------------------------------------------
/app/src/main/java/com/sarmale/arduinobtexample_v3/ConnectedThread.java:
--------------------------------------------------------------------------------
1 | package com.sarmale.arduinobtexample_v3;
2 |
3 |
4 | import android.bluetooth.BluetoothSocket;
5 | import android.os.Handler;
6 | import android.util.Log;
7 |
8 | import java.io.IOException;
9 | import java.io.InputStream;
10 | import java.io.OutputStream;
11 |
12 | //Class that given an open BT Socket will
13 | //Open, manage and close the data Stream from the Arduino BT device
14 | public class ConnectedThread extends Thread {
15 |
16 | private static final String TAG = "FrugalLogs";
17 | private final BluetoothSocket mmSocket;
18 | private final InputStream mmInStream;
19 | private final OutputStream mmOutStream;
20 | private String valueRead;
21 |
22 | public ConnectedThread(BluetoothSocket socket) {
23 | mmSocket = socket;
24 | InputStream tmpIn = null;
25 | OutputStream tmpOut = null;
26 |
27 | // Get the input and output streams; using temp objects because
28 | // member streams are final.
29 | try {
30 | tmpIn = socket.getInputStream();
31 | } catch (IOException e) {
32 | Log.e(TAG, "Error occurred when creating input stream", e);
33 | }
34 | try {
35 | tmpOut = socket.getOutputStream();
36 | } catch (IOException e) {
37 | Log.e(TAG, "Error occurred when creating output stream", e);
38 | }
39 | //Input and Output streams members of the class
40 | //We wont use the Output stream of this project
41 | mmInStream = tmpIn;
42 | mmOutStream = tmpOut;
43 | }
44 |
45 | public String getValueRead(){
46 | return valueRead;
47 | }
48 |
49 | public void run() {
50 |
51 | byte[] buffer = new byte[1024];
52 | int bytes = 0; // bytes returned from read()
53 | int numberOfReadings = 0; //to control the number of readings from the Arduino
54 |
55 | // Keep listening to the InputStream until an exception occurs.
56 | //We just want to get 1 temperature readings from the Arduino
57 | while (numberOfReadings < 1) {
58 | try {
59 |
60 | buffer[bytes] = (byte) mmInStream.read();
61 | String readMessage;
62 | // If I detect a "\n" means I already read a full measurement
63 | if (buffer[bytes] == '\n') {
64 | readMessage = new String(buffer, 0, bytes);
65 | Log.e(TAG, readMessage);
66 | //Value to be read by the Observer streamed by the Obervable
67 | valueRead=readMessage;
68 | bytes = 0;
69 | numberOfReadings++;
70 | } else {
71 | bytes++;
72 | }
73 |
74 | } catch (IOException e) {
75 | Log.d(TAG, "Input stream was disconnected", e);
76 | break;
77 | }
78 | }
79 |
80 | }
81 |
82 | // Call this method from the main activity to shut down the connection.
83 | public void cancel() {
84 | try {
85 | mmSocket.close();
86 | } catch (IOException e) {
87 | Log.e(TAG, "Could not close the connect socket", e);
88 | }
89 | }
90 | }
--------------------------------------------------------------------------------
/app/src/main/res/layout/activity_main.xml:
--------------------------------------------------------------------------------
1 |
2 |
8 |
9 |
20 |
21 |
22 |
33 |
34 |
44 |
45 |
58 |
59 |
69 |
70 |
81 |
82 |
92 |
93 |
--------------------------------------------------------------------------------
/app/src/main/res/drawable/ic_launcher_background.xml:
--------------------------------------------------------------------------------
1 |
2 |
7 |
10 |
15 |
20 |
25 |
30 |
35 |
40 |
45 |
50 |
55 |
60 |
65 |
70 |
75 |
80 |
85 |
90 |
95 |
100 |
105 |
110 |
115 |
120 |
125 |
130 |
135 |
140 |
145 |
150 |
155 |
160 |
165 |
170 |
171 |
--------------------------------------------------------------------------------
/gradlew:
--------------------------------------------------------------------------------
1 | #!/usr/bin/env sh
2 |
3 | #
4 | # Copyright 2015 the original author or authors.
5 | #
6 | # Licensed under the Apache License, Version 2.0 (the "License");
7 | # you may not use this file except in compliance with the License.
8 | # You may obtain a copy of the License at
9 | #
10 | # https://www.apache.org/licenses/LICENSE-2.0
11 | #
12 | # Unless required by applicable law or agreed to in writing, software
13 | # distributed under the License is distributed on an "AS IS" BASIS,
14 | # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
15 | # See the License for the specific language governing permissions and
16 | # limitations under the License.
17 | #
18 |
19 | ##############################################################################
20 | ##
21 | ## Gradle start up script for UN*X
22 | ##
23 | ##############################################################################
24 |
25 | # Attempt to set APP_HOME
26 | # Resolve links: $0 may be a link
27 | PRG="$0"
28 | # Need this for relative symlinks.
29 | while [ -h "$PRG" ] ; do
30 | ls=`ls -ld "$PRG"`
31 | link=`expr "$ls" : '.*-> \(.*\)$'`
32 | if expr "$link" : '/.*' > /dev/null; then
33 | PRG="$link"
34 | else
35 | PRG=`dirname "$PRG"`"/$link"
36 | fi
37 | done
38 | SAVED="`pwd`"
39 | cd "`dirname \"$PRG\"`/" >/dev/null
40 | APP_HOME="`pwd -P`"
41 | cd "$SAVED" >/dev/null
42 |
43 | APP_NAME="Gradle"
44 | APP_BASE_NAME=`basename "$0"`
45 |
46 | # Add default JVM options here. You can also use JAVA_OPTS and GRADLE_OPTS to pass JVM options to this script.
47 | DEFAULT_JVM_OPTS='"-Xmx64m" "-Xms64m"'
48 |
49 | # Use the maximum available, or set MAX_FD != -1 to use that value.
50 | MAX_FD="maximum"
51 |
52 | warn () {
53 | echo "$*"
54 | }
55 |
56 | die () {
57 | echo
58 | echo "$*"
59 | echo
60 | exit 1
61 | }
62 |
63 | # OS specific support (must be 'true' or 'false').
64 | cygwin=false
65 | msys=false
66 | darwin=false
67 | nonstop=false
68 | case "`uname`" in
69 | CYGWIN* )
70 | cygwin=true
71 | ;;
72 | Darwin* )
73 | darwin=true
74 | ;;
75 | MINGW* )
76 | msys=true
77 | ;;
78 | NONSTOP* )
79 | nonstop=true
80 | ;;
81 | esac
82 |
83 | CLASSPATH=$APP_HOME/gradle/wrapper/gradle-wrapper.jar
84 |
85 |
86 | # Determine the Java command to use to start the JVM.
87 | if [ -n "$JAVA_HOME" ] ; then
88 | if [ -x "$JAVA_HOME/jre/sh/java" ] ; then
89 | # IBM's JDK on AIX uses strange locations for the executables
90 | JAVACMD="$JAVA_HOME/jre/sh/java"
91 | else
92 | JAVACMD="$JAVA_HOME/bin/java"
93 | fi
94 | if [ ! -x "$JAVACMD" ] ; then
95 | die "ERROR: JAVA_HOME is set to an invalid directory: $JAVA_HOME
96 |
97 | Please set the JAVA_HOME variable in your environment to match the
98 | location of your Java installation."
99 | fi
100 | else
101 | JAVACMD="java"
102 | which java >/dev/null 2>&1 || die "ERROR: JAVA_HOME is not set and no 'java' command could be found in your PATH.
103 |
104 | Please set the JAVA_HOME variable in your environment to match the
105 | location of your Java installation."
106 | fi
107 |
108 | # Increase the maximum file descriptors if we can.
109 | if [ "$cygwin" = "false" -a "$darwin" = "false" -a "$nonstop" = "false" ] ; then
110 | MAX_FD_LIMIT=`ulimit -H -n`
111 | if [ $? -eq 0 ] ; then
112 | if [ "$MAX_FD" = "maximum" -o "$MAX_FD" = "max" ] ; then
113 | MAX_FD="$MAX_FD_LIMIT"
114 | fi
115 | ulimit -n $MAX_FD
116 | if [ $? -ne 0 ] ; then
117 | warn "Could not set maximum file descriptor limit: $MAX_FD"
118 | fi
119 | else
120 | warn "Could not query maximum file descriptor limit: $MAX_FD_LIMIT"
121 | fi
122 | fi
123 |
124 | # For Darwin, add options to specify how the application appears in the dock
125 | if $darwin; then
126 | GRADLE_OPTS="$GRADLE_OPTS \"-Xdock:name=$APP_NAME\" \"-Xdock:icon=$APP_HOME/media/gradle.icns\""
127 | fi
128 |
129 | # For Cygwin or MSYS, switch paths to Windows format before running java
130 | if [ "$cygwin" = "true" -o "$msys" = "true" ] ; then
131 | APP_HOME=`cygpath --path --mixed "$APP_HOME"`
132 | CLASSPATH=`cygpath --path --mixed "$CLASSPATH"`
133 |
134 | JAVACMD=`cygpath --unix "$JAVACMD"`
135 |
136 | # We build the pattern for arguments to be converted via cygpath
137 | ROOTDIRSRAW=`find -L / -maxdepth 1 -mindepth 1 -type d 2>/dev/null`
138 | SEP=""
139 | for dir in $ROOTDIRSRAW ; do
140 | ROOTDIRS="$ROOTDIRS$SEP$dir"
141 | SEP="|"
142 | done
143 | OURCYGPATTERN="(^($ROOTDIRS))"
144 | # Add a user-defined pattern to the cygpath arguments
145 | if [ "$GRADLE_CYGPATTERN" != "" ] ; then
146 | OURCYGPATTERN="$OURCYGPATTERN|($GRADLE_CYGPATTERN)"
147 | fi
148 | # Now convert the arguments - kludge to limit ourselves to /bin/sh
149 | i=0
150 | for arg in "$@" ; do
151 | CHECK=`echo "$arg"|egrep -c "$OURCYGPATTERN" -`
152 | CHECK2=`echo "$arg"|egrep -c "^-"` ### Determine if an option
153 |
154 | if [ $CHECK -ne 0 ] && [ $CHECK2 -eq 0 ] ; then ### Added a condition
155 | eval `echo args$i`=`cygpath --path --ignore --mixed "$arg"`
156 | else
157 | eval `echo args$i`="\"$arg\""
158 | fi
159 | i=`expr $i + 1`
160 | done
161 | case $i in
162 | 0) set -- ;;
163 | 1) set -- "$args0" ;;
164 | 2) set -- "$args0" "$args1" ;;
165 | 3) set -- "$args0" "$args1" "$args2" ;;
166 | 4) set -- "$args0" "$args1" "$args2" "$args3" ;;
167 | 5) set -- "$args0" "$args1" "$args2" "$args3" "$args4" ;;
168 | 6) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" ;;
169 | 7) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" ;;
170 | 8) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" ;;
171 | 9) set -- "$args0" "$args1" "$args2" "$args3" "$args4" "$args5" "$args6" "$args7" "$args8" ;;
172 | esac
173 | fi
174 |
175 | # Escape application args
176 | save () {
177 | for i do printf %s\\n "$i" | sed "s/'/'\\\\''/g;1s/^/'/;\$s/\$/' \\\\/" ; done
178 | echo " "
179 | }
180 | APP_ARGS=`save "$@"`
181 |
182 | # Collect all arguments for the java command, following the shell quoting and substitution rules
183 | eval set -- $DEFAULT_JVM_OPTS $JAVA_OPTS $GRADLE_OPTS "\"-Dorg.gradle.appname=$APP_BASE_NAME\"" -classpath "\"$CLASSPATH\"" org.gradle.wrapper.GradleWrapperMain "$APP_ARGS"
184 |
185 | exec "$JAVACMD" "$@"
186 |
--------------------------------------------------------------------------------
/app/src/main/java/com/sarmale/arduinobtexample_v3/MainActivity.java:
--------------------------------------------------------------------------------
1 | package com.sarmale.arduinobtexample_v3;
2 |
3 |
4 | import android.Manifest;
5 | import android.annotation.SuppressLint;
6 | import android.bluetooth.BluetoothAdapter;
7 | import android.bluetooth.BluetoothDevice;
8 | import android.bluetooth.BluetoothManager;
9 | import android.bluetooth.BluetoothSocket;
10 | import android.content.Context;
11 | import android.content.Intent;
12 | import android.content.pm.PackageManager;
13 | import android.os.Build;
14 | import android.os.Bundle;
15 | import android.os.Handler;
16 | import android.os.Looper;
17 | import android.os.Message;
18 | import android.os.SystemClock;
19 | import android.util.Log;
20 | import android.view.View;
21 | import android.widget.Button;
22 | import android.widget.TextView;
23 |
24 | import androidx.annotation.RequiresApi;
25 | import androidx.appcompat.app.AppCompatActivity;
26 | import androidx.core.app.ActivityCompat;
27 |
28 | import java.io.IOException;
29 | import java.io.InputStream;
30 | import java.io.OutputStream;
31 | import java.util.Set;
32 | import java.util.UUID;
33 |
34 | import io.reactivex.Observable;
35 | import io.reactivex.android.schedulers.AndroidSchedulers;
36 | import io.reactivex.schedulers.Schedulers;
37 |
38 | public class MainActivity extends AppCompatActivity {
39 | // Global variables we will use in the
40 | private static final String TAG = "FrugalLogs";
41 | private static final int REQUEST_ENABLE_BT = 1;
42 | //We will use a Handler to get the BT Connection statys
43 | public static Handler handler;
44 | private final static int ERROR_READ = 0; // used in bluetooth handler to identify message update
45 | BluetoothDevice arduinoBTModule = null;
46 | UUID arduinoUUID = UUID.fromString("00001101-0000-1000-8000-00805F9B34FB"); //We declare a default UUID to create the global variable
47 | @RequiresApi(api = Build.VERSION_CODES.M)
48 | @Override
49 | protected void onCreate(Bundle savedInstanceState) {
50 | super.onCreate(savedInstanceState);
51 | setContentView(R.layout.activity_main);
52 | //Intances of BT Manager and BT Adapter needed to work with BT in Android.
53 | BluetoothManager bluetoothManager = getSystemService(BluetoothManager.class);
54 | BluetoothAdapter bluetoothAdapter = bluetoothManager.getAdapter();
55 | //Intances of the Android UI elements that will will use during the execution of the APP
56 | TextView btReadings = findViewById(R.id.btReadings);
57 | TextView btDevices = findViewById(R.id.btDevices);
58 | Button connectToDevice = (Button) findViewById(R.id.connectToDevice);
59 | Button seachDevices = (Button) findViewById(R.id.seachDevices);
60 | Button clearValues = (Button) findViewById(R.id.refresh);
61 | Log.d(TAG, "Begin Execution");
62 |
63 |
64 | //Using a handler to update the interface in case of an error connecting to the BT device
65 | //My idea is to show handler vs RxAndroid
66 | handler = new Handler(Looper.getMainLooper()) {
67 | @Override
68 | public void handleMessage(Message msg) {
69 | switch (msg.what) {
70 |
71 | case ERROR_READ:
72 | String arduinoMsg = msg.obj.toString(); // Read message from Arduino
73 | btReadings.setText(arduinoMsg);
74 | break;
75 | }
76 | }
77 | };
78 |
79 | // Set a listener event on a button to clear the texts
80 | clearValues.setOnClickListener(new View.OnClickListener() {
81 | @Override
82 | public void onClick(View view) {
83 | btDevices.setText("");
84 | btReadings.setText("");
85 | }
86 | });
87 |
88 | // Create an Observable from RxAndroid
89 | //The code will be executed when an Observer subscribes to the the Observable
90 | final Observable connectToBTObservable = Observable.create(emitter -> {
91 | Log.d(TAG, "Calling connectThread class");
92 | //Call the constructor of the ConnectThread class
93 | //Passing the Arguments: an Object that represents the BT device,
94 | // the UUID and then the handler to update the UI
95 | ConnectThread connectThread = new ConnectThread(arduinoBTModule, arduinoUUID, handler);
96 | connectThread.run();
97 | //Check if Socket connected
98 | if (connectThread.getMmSocket().isConnected()) {
99 | Log.d(TAG, "Calling ConnectedThread class");
100 | //The pass the Open socket as arguments to call the constructor of ConnectedThread
101 | ConnectedThread connectedThread = new ConnectedThread(connectThread.getMmSocket());
102 | connectedThread.run();
103 | if(connectedThread.getValueRead()!=null)
104 | {
105 | // If we have read a value from the Arduino
106 | // we call the onNext() function
107 | //This value will be observed by the observer
108 | emitter.onNext(connectedThread.getValueRead());
109 | }
110 | //We just want to stream 1 value, so we close the BT stream
111 | connectedThread.cancel();
112 | }
113 | // SystemClock.sleep(5000); // simulate delay
114 | //Then we close the socket connection
115 | connectThread.cancel();
116 | //We could Override the onComplete function
117 | emitter.onComplete();
118 |
119 | });
120 |
121 | connectToDevice.setOnClickListener(new View.OnClickListener() {
122 | @Override
123 | public void onClick(View view) {
124 | btReadings.setText("");
125 | if (arduinoBTModule != null) {
126 | //We subscribe to the observable until the onComplete() is called
127 | //We also define control the thread management with
128 | // subscribeOn: the thread in which you want to execute the action
129 | // observeOn: the thread in which you want to get the response
130 | connectToBTObservable.
131 | observeOn(AndroidSchedulers.mainThread()).
132 | subscribeOn(Schedulers.io()).
133 | subscribe(valueRead -> {
134 | //valueRead returned by the onNext() from the Observable
135 | btReadings.setText(valueRead);
136 | //We just scratched the surface with RxAndroid
137 | });
138 |
139 | }
140 | }
141 | });
142 |
143 | seachDevices.setOnClickListener(new View.OnClickListener() {
144 | //Display all the linked BT Devices
145 | @Override
146 | public void onClick(View view) {
147 | //Check if the phone supports BT
148 | if (bluetoothAdapter == null) {
149 | // Device doesn't support Bluetooth
150 | Log.d(TAG, "Device doesn't support Bluetooth");
151 | } else {
152 | Log.d(TAG, "Device support Bluetooth");
153 | //Check BT enabled. If disabled, we ask the user to enable BT
154 | if (!bluetoothAdapter.isEnabled()) {
155 | Log.d(TAG, "Bluetooth is disabled");
156 | Intent enableBtIntent = new Intent(BluetoothAdapter.ACTION_REQUEST_ENABLE);
157 | if (ActivityCompat.checkSelfPermission(getApplicationContext(), Manifest.permission.BLUETOOTH_CONNECT) != PackageManager.PERMISSION_GRANTED) {
158 | // TODO: Consider calling
159 | // ActivityCompat#requestPermissions
160 | // here to request the missing permissions, and then overriding
161 | // public void onRequestPermissionsResult(int requestCode, String[] permissions,
162 | // int[] grantResults)
163 | // to handle the case where the user grants the permission. See the documentation
164 | // for ActivityCompat#requestPermissions for more details.
165 | Log.d(TAG, "We don't BT Permissions");
166 | startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT);
167 | Log.d(TAG, "Bluetooth is enabled now");
168 | } else {
169 | Log.d(TAG, "We have BT Permissions");
170 | startActivityForResult(enableBtIntent, REQUEST_ENABLE_BT);
171 | Log.d(TAG, "Bluetooth is enabled now");
172 | }
173 |
174 | } else {
175 | Log.d(TAG, "Bluetooth is enabled");
176 | }
177 | String btDevicesString="";
178 | Set < BluetoothDevice > pairedDevices = bluetoothAdapter.getBondedDevices();
179 |
180 | if (pairedDevices.size() > 0) {
181 | // There are paired devices. Get the name and address of each paired device.
182 | for (BluetoothDevice device: pairedDevices) {
183 | String deviceName = device.getName();
184 | String deviceHardwareAddress = device.getAddress(); // MAC address
185 | Log.d(TAG, "deviceName:" + deviceName);
186 | Log.d(TAG, "deviceHardwareAddress:" + deviceHardwareAddress);
187 | //We append all devices to a String that we will display in the UI
188 | btDevicesString=btDevicesString+deviceName+" || "+deviceHardwareAddress+"\n";
189 | //If we find the HC 05 device (the Arduino BT module)
190 | //We assign the device value to the Global variable BluetoothDevice
191 | //We enable the button "Connect to HC 05 device"
192 | if (deviceName.equals("HC-05")) {
193 | Log.d(TAG, "HC-05 found");
194 | arduinoUUID = device.getUuids()[0].getUuid();
195 | arduinoBTModule = device;
196 | //HC -05 Found, enabling the button to read results
197 | connectToDevice.setEnabled(true);
198 | }
199 | btDevices.setText(btDevicesString);
200 | }
201 | }
202 | }
203 | Log.d(TAG, "Button Pressed");
204 | }
205 | });
206 | }
207 |
208 | }
--------------------------------------------------------------------------------