Skip to content

Commit 0649b35

Browse files
committed
upgrade tooling, move analysis to separate thread, update layout
1 parent 6c7c134 commit 0649b35

15 files changed

+226
-97
lines changed

.idea/compiler.xml

+6
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

.idea/gradle.xml

+10-4
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

.idea/jarRepositories.xml

+25
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

.idea/misc.xml

+1-1
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

app/build.gradle

+6-6
Original file line numberDiff line numberDiff line change
@@ -8,7 +8,7 @@ android {
88
}
99
}
1010
compileSdkVersion 29
11-
buildToolsVersion '28.0.3'
11+
buildToolsVersion '29.0.2'
1212
defaultConfig {
1313
applicationId "eu.berdosi.app.heartbeat"
1414
minSdkVersion 21
@@ -31,10 +31,10 @@ android {
3131

3232
dependencies {
3333
implementation fileTree(dir: 'libs', include: ['*.jar'])
34-
implementation 'androidx.appcompat:appcompat:1.1.0'
35-
implementation 'androidx.constraintlayout:constraintlayout:1.1.3'
36-
implementation 'com.google.android.material:material:1.0.0'
34+
implementation 'androidx.appcompat:appcompat:1.2.0'
35+
implementation 'androidx.constraintlayout:constraintlayout:2.0.2'
36+
implementation 'com.google.android.material:material:1.2.1'
3737
testImplementation 'junit:junit:4.13'
38-
androidTestImplementation 'androidx.test.ext:junit:1.1.1'
39-
androidTestImplementation 'androidx.test.espresso:espresso-core:3.2.0'
38+
androidTestImplementation 'androidx.test.ext:junit:1.1.2'
39+
androidTestImplementation 'androidx.test.espresso:espresso-core:3.3.0'
4040
}

app/src/main/AndroidManifest.xml

+2-2
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,6 @@
33
package="eu.berdosi.app.heartbeat">
44

55
<uses-permission android:name="android.permission.CAMERA" />
6-
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE" />
76

87
<application
98
android:allowBackup="true"
@@ -16,7 +15,8 @@
1615
<activity
1716
android:name=".MainActivity"
1817
android:label="@string/app_name"
19-
android:theme="@style/AppTheme.NoActionBar">
18+
android:theme="@style/AppTheme.NoActionBar"
19+
android:screenOrientation="portrait">
2020
<intent-filter>
2121
<action android:name="android.intent.action.MAIN" />
2222

app/src/main/java/eu/berdosi/app/heartbeat/MainActivity.java

+56-5
Original file line numberDiff line numberDiff line change
@@ -1,31 +1,67 @@
11
package eu.berdosi.app.heartbeat;
22

33
import android.Manifest;
4+
import android.annotation.SuppressLint;
5+
import android.app.Activity;
6+
import android.content.Intent;
47
import android.graphics.SurfaceTexture;
58
import android.os.Bundle;
69

10+
import androidx.annotation.NonNull;
711
import androidx.appcompat.app.AppCompatActivity;
812
import androidx.appcompat.widget.Toolbar;
913
import androidx.core.app.ActivityCompat;
1014

15+
import android.os.Handler;
16+
import android.os.Message;
1117
import android.view.TextureView;
1218
import android.view.Surface;
19+
import android.view.View;
20+
import android.widget.EditText;
21+
import android.widget.TextView;
1322

14-
public class MainActivity extends AppCompatActivity {
23+
import java.util.Date;
24+
25+
public class MainActivity extends Activity {
1526
private final CameraService cameraService = new CameraService(this);
1627

28+
private boolean justShared = false;
29+
30+
@SuppressLint("HandlerLeak")
31+
private final Handler mainHandler = new Handler() {
32+
@Override
33+
public void handleMessage(@NonNull Message msg) {
34+
super.handleMessage(msg);
35+
36+
if (msg.what == MESSAGE_UPDATE_REALTIME) {
37+
((TextView) findViewById(R.id.textView)).setText(msg.obj.toString());
38+
}
39+
40+
if (msg.what == MESSAGE_UPDATE_FINAL) {
41+
((EditText) findViewById(R.id.editText)).setText(msg.obj.toString());
42+
43+
findViewById(R.id.floatingActionButton).setClickable(true);
44+
}
45+
}
46+
};
47+
1748
private OutputAnalyzer analyzer;
1849

50+
public static final int MESSAGE_UPDATE_REALTIME = 1;
51+
public static final int MESSAGE_UPDATE_FINAL = 2;
52+
1953
@Override
2054
protected void onResume() {
2155
super.onResume();
2256

23-
analyzer = new OutputAnalyzer(this, findViewById(R.id.graphTextureView));
57+
analyzer = new OutputAnalyzer(this, findViewById(R.id.graphTextureView), mainHandler);
2458

2559
TextureView cameraTextureView = findViewById(R.id.textureView2);
2660

2761
SurfaceTexture previewSurfaceTexture = cameraTextureView.getSurfaceTexture();
28-
if (previewSurfaceTexture != null) {
62+
63+
// justShared is set if one clicks the share button.
64+
if ((previewSurfaceTexture != null) && !justShared) {
2965
// this first appears when we close the application and switch back - TextureView isn't quite ready at the first onResume.
3066
Surface previewSurface = new Surface(previewSurfaceTexture);
3167

@@ -39,19 +75,34 @@ protected void onPause() {
3975
super.onPause();
4076
cameraService.stop();
4177
if (analyzer != null ) analyzer.stop();
42-
analyzer = new OutputAnalyzer(this, findViewById(R.id.graphTextureView));
78+
analyzer = new OutputAnalyzer(this, findViewById(R.id.graphTextureView), mainHandler);
4379
}
4480

4581
@Override
4682
protected void onCreate(Bundle savedInstanceState) {
4783
super.onCreate(savedInstanceState);
4884
setContentView(R.layout.activity_main);
4985
Toolbar toolbar = findViewById(R.id.toolbar);
50-
setSupportActionBar(toolbar);
5186

5287
int MY_PERMISSIONS_REQUEST_READ_CONTACTS = 1;
5388
ActivityCompat.requestPermissions(this,
5489
new String[]{Manifest.permission.CAMERA},
5590
MY_PERMISSIONS_REQUEST_READ_CONTACTS);
91+
92+
}
93+
94+
public void onClickShareButton(View view) {
95+
final Intent intent = new Intent(Intent.ACTION_SEND);
96+
intent.setType("text/plain");
97+
intent.putExtra(Intent.EXTRA_SUBJECT, String.format(getString(R.string.output_header_template), new Date()));
98+
intent.putExtra(
99+
Intent.EXTRA_TEXT,
100+
String.format(
101+
getString(R.string.output_body_template),
102+
((TextView) findViewById(R.id.textView)).getText(),
103+
((EditText) findViewById(R.id.editText)).getText()));
104+
105+
justShared = true;
106+
startActivity(Intent.createChooser(intent, getString(R.string.send_output_to)));
56107
}
57108
}

app/src/main/java/eu/berdosi/app/heartbeat/MeasureStore.java

+10
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,15 @@ class MeasureStore {
77
private final CopyOnWriteArrayList<Measurement<Integer>> measurements = new CopyOnWriteArrayList<>();
88
private int minimum = 2147483647;
99
private int maximum = -2147483648;
10+
11+
/**
12+
* The latest N measurements are always averaged in order to smooth the values before it is
13+
* analyzed.
14+
*
15+
* This value may need to be experimented with - it is better on the class level than putting it
16+
* into local scope
17+
*/
18+
@SuppressWarnings("FieldCanBeLocal")
1019
private final int rollingAverageSize = 4;
1120

1221
void add(int measurement) {
@@ -36,6 +45,7 @@ CopyOnWriteArrayList<Measurement<Float>> getStdValues() {
3645
return stdValues;
3746
}
3847

48+
@SuppressWarnings("SameParameterValue") // this parameter can be set at OutputAnalyzer
3949
CopyOnWriteArrayList<Measurement<Integer>> getLastStdValues(int count) {
4050
if (count < measurements.size()) {
4151
return new CopyOnWriteArrayList<>(measurements.subList(measurements.size() - 1 - count, measurements.size() - 1));

app/src/main/java/eu/berdosi/app/heartbeat/OutputAnalyzer.java

+23-9
Original file line numberDiff line numberDiff line change
@@ -3,9 +3,9 @@
33
import android.app.Activity;
44
import android.graphics.Bitmap;
55
import android.os.CountDownTimer;
6+
import android.os.Handler;
7+
import android.os.Message;
68
import android.view.TextureView;
7-
import android.widget.EditText;
8-
import android.widget.TextView;
99

1010
import java.util.Locale;
1111
import java.util.concurrent.CopyOnWriteArrayList;
@@ -28,10 +28,12 @@ class OutputAnalyzer {
2828

2929
private CountDownTimer timer;
3030

31+
private final Handler mainHandler;
3132

32-
OutputAnalyzer(Activity activity, TextureView graphTextureView) {
33+
OutputAnalyzer(Activity activity, TextureView graphTextureView, Handler mainHandler) {
3334
this.activity = activity;
3435
this.chartDrawer = new ChartDrawer(graphTextureView);
36+
this.mainHandler = mainHandler;
3537
}
3638

3739
private boolean detectValley() {
@@ -40,15 +42,15 @@ private boolean detectValley() {
4042
if (subList.size() < valleyDetectionWindowSize) {
4143
return false;
4244
} else {
43-
Integer referenceValue = subList.get((int) Math.ceil(valleyDetectionWindowSize / 2)).measurement;
45+
Integer referenceValue = subList.get((int) Math.ceil(valleyDetectionWindowSize / 2f)).measurement;
4446

4547
for (Measurement<Integer> measurement : subList) {
4648
if (measurement.measurement < referenceValue) return false;
4749
}
4850

4951
// filter out consecutive measurements due to too high measurement rate
50-
return (!subList.get((int) Math.ceil(valleyDetectionWindowSize / 2)).measurement.equals(
51-
subList.get((int) Math.ceil(valleyDetectionWindowSize / 2) - 1).measurement));
52+
return (!subList.get((int) Math.ceil(valleyDetectionWindowSize / 2f)).measurement.equals(
53+
subList.get((int) Math.ceil(valleyDetectionWindowSize / 2f) - 1).measurement));
5254
}
5355
}
5456

@@ -62,6 +64,7 @@ void measurePulse(TextureView textureView, CameraService cameraService) {
6264
detectedValleys = 0;
6365

6466
timer = new CountDownTimer(measurementLength, measurementInterval) {
67+
6568
@Override
6669
public void onTick(long millisUntilFinished) {
6770
// skip the first measurements, which are broken by exposure metering
@@ -98,7 +101,8 @@ public void onTick(long millisUntilFinished) {
98101
: (60f * (detectedValleys - 1) / (Math.max(1, (valleys.get(valleys.size() - 1) - valleys.get(0)) / 1000f))),
99102
detectedValleys,
100103
1f * (measurementLength - millisUntilFinished - clipLength) / 1000f);
101-
((TextView) activity.findViewById(R.id.textView)).setText(currentValue);
104+
105+
sendMessage(MainActivity.MESSAGE_UPDATE_REALTIME, currentValue);
102106
}
103107

104108
// draw the chart on a separate thread.
@@ -120,7 +124,7 @@ public void onFinish() {
120124
detectedValleys - 1,
121125
1f * (valleys.get(valleys.size() - 1) - valleys.get(0)) / 1000f);
122126

123-
((TextView) activity.findViewById(R.id.textView)).setText(currentValue);
127+
sendMessage(MainActivity.MESSAGE_UPDATE_REALTIME, currentValue);
124128

125129
StringBuilder returnValueSb = new StringBuilder();
126130
returnValueSb.append(currentValue);
@@ -156,13 +160,16 @@ public void onFinish() {
156160
returnValueSb.append(activity.getString(R.string.row_separator));
157161
}
158162

163+
returnValueSb.append(activity.getString(R.string.output_detected_peaks_header));
164+
returnValueSb.append(activity.getString(R.string.row_separator));
165+
159166
// add detected valleys location
160167
for (long tick : valleys) {
161168
returnValueSb.append(tick);
162169
returnValueSb.append(activity.getString(R.string.row_separator));
163170
}
164171

165-
((EditText) activity.findViewById(R.id.editText)).setText(returnValueSb.toString());
172+
sendMessage(MainActivity.MESSAGE_UPDATE_FINAL, returnValueSb.toString());
166173

167174
cameraService.stop();
168175
}
@@ -176,4 +183,11 @@ void stop() {
176183
timer.cancel();
177184
}
178185
}
186+
187+
void sendMessage(int what, Object message) {
188+
Message msg = new Message();
189+
msg.what = what;
190+
msg.obj = message;
191+
mainHandler.sendMessage(msg);
192+
}
179193
}

0 commit comments

Comments
 (0)