From 768d43d48cf83a05a572ae15eff50f38188adb6e Mon Sep 17 00:00:00 2001 From: elFuchso Date: Mon, 8 Jan 2024 20:22:24 +0000 Subject: [PATCH 1/4] Update apps_tab.dart --- open_earable/lib/apps_tab.dart | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/open_earable/lib/apps_tab.dart b/open_earable/lib/apps_tab.dart index 17e5a27..1e5c32b 100644 --- a/open_earable/lib/apps_tab.dart +++ b/open_earable/lib/apps_tab.dart @@ -45,6 +45,16 @@ class AppsTab extends StatelessWidget { MaterialPageRoute( builder: (context) => Recorder(_openEarable))); }), + AppInfo( + iconData: Icons.music_note, + title: "Tightness Meter", + description: "Track your headbanging.", + onTap: () { + Navigator.push( + context, + MaterialPageRoute( + builder: (context) => TightnessMeter(_openEarable))); + }), // ... similarly for other apps ]; } From 26e5a292eb7675fce5218b80e220e827f8e2eae1 Mon Sep 17 00:00:00 2001 From: elFuchso Date: Mon, 8 Jan 2024 20:23:03 +0000 Subject: [PATCH 2/4] Update apps_tab.dart --- open_earable/lib/apps_tab.dart | 1 + 1 file changed, 1 insertion(+) diff --git a/open_earable/lib/apps_tab.dart b/open_earable/lib/apps_tab.dart index 1e5c32b..9131c9c 100644 --- a/open_earable/lib/apps_tab.dart +++ b/open_earable/lib/apps_tab.dart @@ -1,6 +1,7 @@ import 'package:flutter/material.dart'; import 'package:open_earable/apps/posture_tracker/model/earable_attitude_tracker.dart'; import 'package:open_earable/apps/posture_tracker/view/posture_tracker_view.dart'; +import 'package:open_earable/apps/tightness.dart'; import 'package:open_earable/apps/recorder.dart'; import 'package:open_earable_flutter/src/open_earable_flutter.dart'; From d8ae6e4209e894910e8a29474ac924236519639a Mon Sep 17 00:00:00 2001 From: elFuchso Date: Mon, 8 Jan 2024 20:23:34 +0000 Subject: [PATCH 3/4] Add tightness meter --- open_earable/lib/apps/tightness.dart | 455 +++++++++++++++++++++++++++ 1 file changed, 455 insertions(+) create mode 100644 open_earable/lib/apps/tightness.dart diff --git a/open_earable/lib/apps/tightness.dart b/open_earable/lib/apps/tightness.dart new file mode 100644 index 0000000..45b8073 --- /dev/null +++ b/open_earable/lib/apps/tightness.dart @@ -0,0 +1,455 @@ +import 'package:flutter/material.dart'; +import 'dart:async'; +import 'dart:math' as math; +import 'package:open_earable_flutter/src/open_earable_flutter.dart'; + +class TightnessMeter extends StatefulWidget { + final OpenEarable _openEarable; + TightnessMeter(this._openEarable); + @override + _TightnessMeterState createState() => _TightnessMeterState(_openEarable); +} + + + +class _TightnessMeterState extends State { + final OpenEarable _openEarable; + StreamSubscription? _imuSubscription; + _TightnessMeterState(this._openEarable); + bool _monitoring = false; + int lastTime = 0; + double x = 0; + double y = 0; + double z = 0; + double magnitude = 0; + double difficulty = 0; + int bpm = 80; + int score = 0; + int streak = 0; + int tightness = 0; + double nodThreshold = 4; // Time frame in milliseconds to consider for a nod + final List bpmList = [80, 100, 120, 170, 200]; + // Variables to keep track of nodding + DateTime lastNodTime = DateTime.now(); + + @override + void initState() { + super.initState(); + if (_openEarable.bleManager.connected) { + print('init'); + // _setupListeners(); + } + } + + @override + void dispose() { + super.dispose(); + _imuSubscription?.cancel(); + } + + _setupListeners() { + _imuSubscription = + _openEarable.sensorManager.subscribeToSensorData(0).listen((data) { + if (!_monitoring) { + return; + } + int timestamp = data["timestamp"]; + setState(() { + lastTime = timestamp; + x = data["ACC"]["X"]; + y = data["ACC"]["Y"]; + z = data["ACC"]["Z"]; + }); + _processAccelerometerData(x, y, z); + }); + } + + void _processAccelerometerData(double x, double y, double z) { + // Calculate the overall acceleration magnitude + //print(x.toString() + y.toString() + z.toString()); + magnitude = _calculateMagnitude(x, y, z); + //print(magnitude); + + // Check if the magnitude exceeds the nodding threshold + if (magnitude > nodThreshold) { + DateTime now = DateTime.now(); + //print("Nod detected! 00000000000000000000000000000000"); + // Check if the last nod was within the time frame + if (now.difference(lastNodTime).inMilliseconds > + _bpmToMilliseconds(bpm) * 0.7) { + // Detected a nod + _isNodTight(lastNodTime, now); + lastNodTime = now; + } + } + } + + // Calculate the magnitude of the acceleration vector + double _calculateMagnitude(double x, double y, double z) { + return math.sqrt(x * x); + } + + // Check if the nod is tight + void _isNodTight(DateTime last, DateTime secondToLast) { + int difference = last.difference(secondToLast).inMilliseconds.abs(); + int expected = _bpmToMilliseconds(bpm); + if (_isWithinMargin(difference, expected, 25.0-difficulty)) { + setState(() { + streak += 1; + }); + _updateScore(); + } else { + setState(() { + streak = 0; + tightness = 0; + + }); + } + } + + void _updateScore() { + setState(() { + score = score + (((10 * streak) + difficulty) / tightness.abs() ).round(); + }); + } + + bool _isWithinMargin( + int givenInterval, int expectedInterval, double marginPercentage) { + double margin = expectedInterval * marginPercentage / 100; + // Calculate the acceptable range + double lowerBound = expectedInterval - margin; + double upperBound = expectedInterval + margin; + setState(() { + tightness = math.min(_bpmToMilliseconds(bpm), (givenInterval - expectedInterval)); + }); + return givenInterval >= lowerBound && givenInterval <= upperBound; + } + + int _bpmToMilliseconds(int bpm) { + if (bpm <= 0) { + throw ArgumentError("BPM must be greater than 0."); + } + return (60000 / bpm).round(); + } + + void startStopMonitoring() async { + if (_monitoring) { + setState(() { + _monitoring = false; + }); + _openEarable.audioPlayer.setState(AudioPlayerState.stop); + } else { + _setupListeners(); + setState(() { + _monitoring = true; + streak = 0; + }); + //start playing music + _setWAV(bpm.toString()); + } + } + + void _setWAV(String bpm) { + String fileName = bpm + ".wav"; + print("Setting source to wav file with file name '" + fileName + "'"); + _openEarable.audioPlayer.wavFile(fileName); + } + + + @override + Widget build(BuildContext context) { + return Scaffold( + backgroundColor: Theme.of(context).colorScheme.background, + appBar: AppBar( + title: Text('Tightness Meter'), + ), + body: Center( + child: Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.only( + topRight: Radius.circular(40.0), + bottomRight: Radius.circular(40.0), + topLeft: Radius.circular(40.0), + bottomLeft: Radius.circular(40.0)), + ), + width: 500, + child: Column( + mainAxisAlignment: MainAxisAlignment.center, + children: [ + Card( + margin: EdgeInsets.all(20), + color: Colors.black, + child: Column( + children: [ + Padding( + padding: const EdgeInsets.symmetric(horizontal: 10), + child: Container( + padding: EdgeInsets.all(16), + decoration: BoxDecoration( + borderRadius: BorderRadius.only( + topRight: Radius.circular(20.0), + bottomRight: Radius.circular(20.0), + topLeft: Radius.circular(20.0), + bottomLeft: Radius.circular(20.0)), + ), + child: Row( + children: [ + Text( + 'Score:', + style: TextStyle(fontSize: 20), + ), + Spacer(), + Padding( + padding: EdgeInsets.all(5), + child: Text( + _monitoring ? score.toString() : '0', + style: TextStyle(fontSize: 20), + ), + ), + ], + ), + ), + ), + Padding( + padding: const EdgeInsets.symmetric(horizontal: 10), + child: Container( + decoration: BoxDecoration( + color: (streak > 0) ? Colors.green : Colors.red, + borderRadius: BorderRadius.only( + topRight: Radius.circular(20.0), + bottomRight: Radius.circular(20.0), + topLeft: Radius.circular(20.0), + bottomLeft: Radius.circular(20.0)), + ), + padding: EdgeInsets.all(16), + child: Row( + children: [ + Text( + 'Streak:', + style: TextStyle(fontSize: 20), + ), + Spacer(), + Padding( + padding: EdgeInsets.all(5), + child: Text( + _monitoring ? streak.toString() : '0', + style: TextStyle(fontSize: 20), + ), + ), + ], + ), + ), + ), + Padding( + padding: const EdgeInsets.symmetric(horizontal: 10), + child: Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.only( + topRight: Radius.circular(20.0), + bottomRight: Radius.circular(20.0), + topLeft: Radius.circular(20.0), + bottomLeft: Radius.circular(20.0)), + ), + padding: EdgeInsets.only(top: 16, right: 16, left: 16), + child: Row( + children: [ + Text( + 'Tightness:', + style: TextStyle(fontSize: 20), + ), + Spacer(), + Padding( + padding: EdgeInsets.all(5), + child: Text( + _monitoring ? tightness.toString() : '0', + style: TextStyle(fontSize: 20), + ), + ), + ], + ), + ), + ), + Padding( + padding: const EdgeInsets.symmetric(horizontal: 10), + child: Container( + decoration: BoxDecoration( + borderRadius: BorderRadius.only( + topRight: Radius.circular(20.0), + bottomRight: Radius.circular(20.0), + topLeft: Radius.circular(20.0), + bottomLeft: Radius.circular(20.0)), + ), + child: Slider( + thumbColor: Colors.purple, + activeColor: Colors.grey, + secondaryActiveColor: Colors.purpleAccent, + inactiveColor: Colors.grey, + value: tightness.toDouble(), + min: -_bpmToMilliseconds(bpm).toDouble(), + max: _bpmToMilliseconds(bpm).toDouble(), + divisions: 2000, + label: tightness.toString(), + onChanged: (double value) { + setState(() { + }); + }), + ), + ), + Padding( + padding: const EdgeInsets.only(bottom:20.0), + child: Row(children: [ + Spacer(), + Text('Early'), + Spacer(flex: 5), + Text('Tight'), + Spacer(flex: 5), + Text('Late'), + Spacer() + ], + ), + ), + ], + ), + ), + Card(margin: EdgeInsets.all(20), + color: Colors.black, + child: Column( + children: [ + Padding( + padding: EdgeInsets.all(20), + child: ElevatedButton( + onPressed: startStopMonitoring, + style: ElevatedButton.styleFrom( + minimumSize: Size(1000, 80), + backgroundColor: _monitoring + ? Color(0xfff27777) + : Theme.of(context).colorScheme.secondary, + foregroundColor: Colors.black, + ), + child: Text( + _monitoring ? 'Stop' : 'Start', + style: TextStyle(fontSize: 30), + ), + ), + ), + ], + ), + ), + Card( + margin: EdgeInsets.all(20), + color: Colors.black, + child: Column( + children: [ + Padding( + padding: const EdgeInsets.all(8.0), + child: Row( + children: [ + Spacer(), + Text('BPM', + style: TextStyle(fontSize: 30)), + Spacer(flex: 5), + DropdownButton( + style: TextStyle( + fontSize: 30, + ), + value: bpm, + icon: const Icon(Icons.arrow_drop_down), + onChanged: _monitoring ? null :(int? newValue) { + setState(() { + bpm = newValue!; + }); + }, + items: bpmList.map>((int value) { + return DropdownMenuItem( + value: value, + child: Text(value.toString()), + ); + }).toList(), + ), + Spacer(), + ], + ), + ), + Padding( + padding: EdgeInsets.only(top: 20), + child: Text('Sensitivity', + style: TextStyle(fontSize: 20) + ), + ), + Padding( + padding: EdgeInsets.all(16), + child: Column( + children: [ + Slider( + thumbColor: Colors.purple, + activeColor: Colors.purpleAccent, + secondaryActiveColor: Colors.purpleAccent, + inactiveColor: Colors.grey, + value: nodThreshold, + min: 1, + max: 20, + divisions: 10, + label: nodThreshold.round().toString(), + onChanged: (double value) { + setState(() { + nodThreshold = value; + }); + }), + Row(children: [ + Spacer(), + Text('Cool Nodding'), + Spacer(flex: 10), + Text('Headbanging'), + Spacer() + ], + ), + ], + ), + ), + Padding( + padding: EdgeInsets.only(top: 10), + child: Text('Difficulty', + style: TextStyle(fontSize: 20) + ), + ), + Padding( + padding: EdgeInsets.all(16), + child: Column( + children: [ + Slider( + thumbColor: Colors.purple, + activeColor: Colors.purpleAccent, + secondaryActiveColor: Colors.purpleAccent, + inactiveColor: Colors.grey, + value: difficulty, + min: 0, + max: 25, + divisions: 10, + label: difficulty.round().toString(), + onChanged: (double value) { + setState(() { + difficulty = value; + }); + }), + Row( + children: [ + Spacer(), + Text('Beginner'), + Spacer(flex: 10), + Text('Impossible'), + Spacer() + ], + ), + ], + ), + ), + ], + ), + ), + ], + ), + ), + ), + ); + } +} From 9aa7e37ea5005b8e65eaacc1764cbf4eaba0702f Mon Sep 17 00:00:00 2001 From: Oliver Bagge Date: Sun, 21 Jan 2024 14:07:12 +0100 Subject: [PATCH 4/4] add scroll view to fix UI overflowing problems --- open_earable/lib/apps/tightness.dart | 142 +++++++++++++-------------- 1 file changed, 71 insertions(+), 71 deletions(-) diff --git a/open_earable/lib/apps/tightness.dart b/open_earable/lib/apps/tightness.dart index 45b8073..e2b0ae8 100644 --- a/open_earable/lib/apps/tightness.dart +++ b/open_earable/lib/apps/tightness.dart @@ -10,8 +10,6 @@ class TightnessMeter extends StatefulWidget { _TightnessMeterState createState() => _TightnessMeterState(_openEarable); } - - class _TightnessMeterState extends State { final OpenEarable _openEarable; StreamSubscription? _imuSubscription; @@ -93,7 +91,7 @@ class _TightnessMeterState extends State { void _isNodTight(DateTime last, DateTime secondToLast) { int difference = last.difference(secondToLast).inMilliseconds.abs(); int expected = _bpmToMilliseconds(bpm); - if (_isWithinMargin(difference, expected, 25.0-difficulty)) { + if (_isWithinMargin(difference, expected, 25.0 - difficulty)) { setState(() { streak += 1; }); @@ -102,14 +100,13 @@ class _TightnessMeterState extends State { setState(() { streak = 0; tightness = 0; - }); } } void _updateScore() { setState(() { - score = score + (((10 * streak) + difficulty) / tightness.abs() ).round(); + score = score + (((10 * streak) + difficulty) / tightness.abs()).round(); }); } @@ -120,7 +117,8 @@ class _TightnessMeterState extends State { double lowerBound = expectedInterval - margin; double upperBound = expectedInterval + margin; setState(() { - tightness = math.min(_bpmToMilliseconds(bpm), (givenInterval - expectedInterval)); + tightness = + math.min(_bpmToMilliseconds(bpm), (givenInterval - expectedInterval)); }); return givenInterval >= lowerBound && givenInterval <= upperBound; } @@ -155,7 +153,6 @@ class _TightnessMeterState extends State { _openEarable.audioPlayer.wavFile(fileName); } - @override Widget build(BuildContext context) { return Scaffold( @@ -163,7 +160,8 @@ class _TightnessMeterState extends State { appBar: AppBar( title: Text('Tightness Meter'), ), - body: Center( + body: SingleChildScrollView( + child: Center( child: Container( decoration: BoxDecoration( borderRadius: BorderRadius.only( @@ -280,59 +278,60 @@ class _TightnessMeterState extends State { bottomLeft: Radius.circular(20.0)), ), child: Slider( - thumbColor: Colors.purple, - activeColor: Colors.grey, - secondaryActiveColor: Colors.purpleAccent, - inactiveColor: Colors.grey, - value: tightness.toDouble(), - min: -_bpmToMilliseconds(bpm).toDouble(), - max: _bpmToMilliseconds(bpm).toDouble(), - divisions: 2000, - label: tightness.toString(), - onChanged: (double value) { - setState(() { - }); - }), + thumbColor: Colors.purple, + activeColor: Colors.grey, + secondaryActiveColor: Colors.purpleAccent, + inactiveColor: Colors.grey, + value: tightness.toDouble(), + min: -_bpmToMilliseconds(bpm).toDouble(), + max: _bpmToMilliseconds(bpm).toDouble(), + divisions: 2000, + label: tightness.toString(), + onChanged: (double value) { + setState(() {}); + }), ), ), Padding( - padding: const EdgeInsets.only(bottom:20.0), - child: Row(children: [ - Spacer(), - Text('Early'), - Spacer(flex: 5), - Text('Tight'), - Spacer(flex: 5), - Text('Late'), - Spacer() - ], + padding: const EdgeInsets.only(bottom: 20.0), + child: Row( + children: [ + Spacer(), + Text('Early'), + Spacer(flex: 5), + Text('Tight'), + Spacer(flex: 5), + Text('Late'), + Spacer() + ], ), ), ], ), ), - Card(margin: EdgeInsets.all(20), + Card( + margin: EdgeInsets.all(20), color: Colors.black, child: Column( - children: [ - Padding( - padding: EdgeInsets.all(20), - child: ElevatedButton( - onPressed: startStopMonitoring, - style: ElevatedButton.styleFrom( - minimumSize: Size(1000, 80), - backgroundColor: _monitoring - ? Color(0xfff27777) - : Theme.of(context).colorScheme.secondary, - foregroundColor: Colors.black, - ), - child: Text( - _monitoring ? 'Stop' : 'Start', - style: TextStyle(fontSize: 30), - ), + children: [ + Padding( + padding: EdgeInsets.all(20), + child: ElevatedButton( + onPressed: startStopMonitoring, + style: ElevatedButton.styleFrom( + minimumSize: Size(1000, 80), + backgroundColor: _monitoring + ? Color(0xfff27777) + : Theme.of(context).colorScheme.secondary, + foregroundColor: Colors.black, + ), + child: Text( + _monitoring ? 'Stop' : 'Start', + style: TextStyle(fontSize: 30), ), ), - ], + ), + ], ), ), Card( @@ -345,8 +344,7 @@ class _TightnessMeterState extends State { child: Row( children: [ Spacer(), - Text('BPM', - style: TextStyle(fontSize: 30)), + Text('BPM', style: TextStyle(fontSize: 30)), Spacer(flex: 5), DropdownButton( style: TextStyle( @@ -354,12 +352,15 @@ class _TightnessMeterState extends State { ), value: bpm, icon: const Icon(Icons.arrow_drop_down), - onChanged: _monitoring ? null :(int? newValue) { - setState(() { - bpm = newValue!; - }); - }, - items: bpmList.map>((int value) { + onChanged: _monitoring + ? null + : (int? newValue) { + setState(() { + bpm = newValue!; + }); + }, + items: + bpmList.map>((int value) { return DropdownMenuItem( value: value, child: Text(value.toString()), @@ -372,9 +373,8 @@ class _TightnessMeterState extends State { ), Padding( padding: EdgeInsets.only(top: 20), - child: Text('Sensitivity', - style: TextStyle(fontSize: 20) - ), + child: + Text('Sensitivity', style: TextStyle(fontSize: 20)), ), Padding( padding: EdgeInsets.all(16), @@ -395,22 +395,21 @@ class _TightnessMeterState extends State { nodThreshold = value; }); }), - Row(children: [ - Spacer(), - Text('Cool Nodding'), - Spacer(flex: 10), - Text('Headbanging'), - Spacer() - ], + Row( + children: [ + Spacer(), + Text('Cool Nodding'), + Spacer(flex: 10), + Text('Headbanging'), + Spacer() + ], ), ], ), ), Padding( padding: EdgeInsets.only(top: 10), - child: Text('Difficulty', - style: TextStyle(fontSize: 20) - ), + child: Text('Difficulty', style: TextStyle(fontSize: 20)), ), Padding( padding: EdgeInsets.all(16), @@ -446,10 +445,11 @@ class _TightnessMeterState extends State { ], ), ), + SizedBox(height: 40), ], ), ), - ), + )), ); } }